Skip to content

Commit

Permalink
feat(jsii): emit warnings when using reserved words
Browse files Browse the repository at this point in the history
Emits warnings when an exported API is named after a reserved word in
any of the supported target languages (best-effort). This warns users
that their code may have toruble compiling for certain target languages
and invites them to use a different name.

Additionally, a `--fail-on-warnings` / `--Werr` option was added to the
CLI that allows treating warnings as errors, allowing users to create
and maintain a codebase that does not rely on naming things in ways that
will cause conflicts in target languages.

Fixes #701
  • Loading branch information
RomainMuller committed Aug 13, 2019
1 parent 09dfcd1 commit 4073f10
Show file tree
Hide file tree
Showing 8 changed files with 255 additions and 22 deletions.
10 changes: 8 additions & 2 deletions packages/jsii/bin/jsii.ts
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,11 @@ import { VERSION } from '../lib/version';
default: true,
desc: 'Automatically add missing entries in the peerDependencies section of package.json'
})
.options('fail-on-warnings', {
alias: 'Werr',
type: 'boolean',
desc: 'Treat warnings as errors'
})
.help()
.version(VERSION)
.argv;
Expand All @@ -35,15 +40,16 @@ import { VERSION } from '../lib/version';
const compiler = new Compiler({
projectInfo,
watch: argv.watch,
projectReferences: argv['project-references']
projectReferences: argv['project-references'],
failOnWarnings: argv['fail-on-warnings']
});

return { projectRoot, emitResult: await compiler.emit() };
})().then(({ projectRoot, emitResult }) => {
for (const diagnostic of emitResult.diagnostics) {
utils.logDiagnostic(diagnostic, projectRoot);
}
if (emitResult.hasErrors) {
if (emitResult.emitSkipped) {
process.exit(1);
}
}).catch(e => {
Expand Down
29 changes: 26 additions & 3 deletions packages/jsii/lib/assembler.ts
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ import { getReferencedDocParams, parseSymbolDocumentation } from './docs';
import { Diagnostic, EmitResult, Emitter } from './emitter';
import literate = require('./literate');
import { ProjectInfo } from './project-info';
import { isReservedName } from './reserved-words';
import { Validator } from './validator';
import { SHORT_VERSION, VERSION } from './version';

Expand Down Expand Up @@ -91,7 +92,7 @@ export class Assembler implements Emitter {
// Clearing ``this._types`` to allow contents to be garbage-collected.
delete this._types;
try {
return { diagnostics: this._diagnostics, hasErrors: true };
return { diagnostics: this._diagnostics, emitSkipped: true };
} finally {
// Clearing ``this._diagnostics`` to allow contents to be garbage-collected.
delete this._diagnostics;
Expand Down Expand Up @@ -124,7 +125,7 @@ export class Assembler implements Emitter {

const validator = new Validator(this.projectInfo, assembly);
const validationResult = await validator.emit();
if (!validationResult.hasErrors) {
if (!validationResult.emitSkipped) {
const assemblyPath = path.join(this.projectInfo.projectRoot, '.jsii');
LOG.trace(`Emitting assembly: ${colors.blue(assemblyPath)}`);
await fs.writeJson(assemblyPath, _fingerprint(assembly), { encoding: 'utf8', spaces: 2 });
Expand All @@ -133,7 +134,7 @@ export class Assembler implements Emitter {
try {
return {
diagnostics: [...this._diagnostics, ...validationResult.diagnostics],
hasErrors: validationResult.hasErrors
emitSkipped: validationResult.emitSkipped
};
} finally {
// Clearing ``this._types`` to allow contents to be garbage-collected.
Expand Down Expand Up @@ -439,6 +440,8 @@ export class Assembler implements Emitter {
return undefined;
}

this._warnAboutReservedWords(type.symbol);

const fqn = `${[this.projectInfo.name, ...ctx.namespace].join('.')}.${type.symbol.name}`;

const jsiiType: spec.ClassType = {
Expand Down Expand Up @@ -747,6 +750,8 @@ export class Assembler implements Emitter {
return undefined;
}

this._warnAboutReservedWords(type.symbol);

const decl = symbol.valueDeclaration;
const flags = ts.getCombinedModifierFlags(decl);
// tslint:disable-next-line:no-bitwise
Expand Down Expand Up @@ -822,6 +827,8 @@ export class Assembler implements Emitter {
return undefined;
}

this._warnAboutReservedWords(type.symbol);

const fqn = `${[this.projectInfo.name, ...ctx.namespace].join('.')}.${type.symbol.name}`;

const jsiiType: spec.InterfaceType = {
Expand Down Expand Up @@ -954,6 +961,8 @@ export class Assembler implements Emitter {
this._diagnostic(declaration, ts.DiagnosticCategory.Error, `Prohibited member name: ${symbol.name}`);
return;
}
this._warnAboutReservedWords(symbol);

const parameters = await Promise.all(signature.getParameters().map(p => this._toParameter(p, ctx)));

const returnType = signature.getReturnType();
Expand Down Expand Up @@ -1006,6 +1015,16 @@ export class Assembler implements Emitter {
type.methods.push(method);
}

private _warnAboutReservedWords(symbol: ts.Symbol) {
const reservingLanguages = isReservedName(symbol.name);
if (reservingLanguages) {
this._diagnostic(symbol.valueDeclaration,
ts.DiagnosticCategory.Warning,
`'${symbol.name}' is a reserved word in ${reservingLanguages.join(', ')}. Using this name may cause problems `
+ 'when generating language bindings. Consider using a different name.');
}
}

private async _visitProperty(symbol: ts.Symbol, type: spec.ClassType | spec.InterfaceType, ctx: EmitContext) {
if (type.properties && type.properties.find(p => p.name === symbol.name)) {
/*
Expand All @@ -1024,6 +1043,8 @@ export class Assembler implements Emitter {
return;
}

this._warnAboutReservedWords(symbol);

const signature = symbol.valueDeclaration as (ts.PropertySignature
| ts.PropertyDeclaration
| ts.AccessorDeclaration
Expand Down Expand Up @@ -1069,6 +1090,8 @@ export class Assembler implements Emitter {
}
const paramDeclaration = paramSymbol.valueDeclaration as ts.ParameterDeclaration;

this._warnAboutReservedWords(paramSymbol);

const parameter: spec.Parameter = {
...await this._optionalValue(this._typeChecker.getTypeAtLocation(paramSymbol.valueDeclaration), paramSymbol.valueDeclaration),
name: paramSymbol.name,
Expand Down
17 changes: 11 additions & 6 deletions packages/jsii/lib/compiler.ts
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,8 @@ export interface CompilerOptions {
watch?: boolean;
/** Whether to detect and generate TypeScript project references */
projectReferences?: boolean;
/** Whether to fail when a warning is emitted */
failOnWarnings?: boolean;
}

export interface TypescriptConfig {
Expand Down Expand Up @@ -147,7 +149,7 @@ export class Compiler implements Emitter {

private async _consumeProgram(program: ts.Program, stdlib: string): Promise<EmitResult> {
const emit = program.emit();
let hasErrors = emitHasErrors(emit);
let hasErrors = emitHasErrors(emit, this.options.failOnWarnings);
const diagnostics = [...emit.diagnostics];

if (hasErrors) {
Expand All @@ -160,17 +162,17 @@ export class Compiler implements Emitter {
try {
const assembler = new Assembler(this.options.projectInfo, program, stdlib);
const assmEmit = await assembler.emit();
if (assmEmit.hasErrors) {
if (assmEmit.emitSkipped) {
LOG.error('Type model errors prevented the JSII assembly from being created');
}

hasErrors = hasErrors || assmEmit.hasErrors;
hasErrors = hasErrors || emitHasErrors(assmEmit, this.options.failOnWarnings);
diagnostics.push(...assmEmit.diagnostics);
} catch (e) {
LOG.error(`Error during type model analysis: ${e}`);
}

return { hasErrors, diagnostics };
return { emitSkipped: hasErrors, diagnostics, emittedFiles: emit.emittedFiles };
}

/**
Expand Down Expand Up @@ -371,6 +373,9 @@ function parseConfigHostFromCompilerHost(host: ts.CompilerHost): ts.ParseConfigH
};
}

function emitHasErrors(result: ts.EmitResult) {
return result.diagnostics.some(d => d.category === ts.DiagnosticCategory.Error) || result.emitSkipped;
function emitHasErrors(result: ts.EmitResult, includeWarnings?: boolean) {
return result.diagnostics.some(d =>
d.category === ts.DiagnosticCategory.Error
|| (includeWarnings && d.category === ts.DiagnosticCategory.Warning)
) || result.emitSkipped;
}
5 changes: 1 addition & 4 deletions packages/jsii/lib/emitter.ts
Original file line number Diff line number Diff line change
Expand Up @@ -15,10 +15,7 @@ export interface Emitter {
/**
* The result of attempting to emit stuff.
*/
export interface EmitResult {
/** Whether the emit was skipped as a result of errors (found in ``diagnostics``) */
hasErrors: boolean;

export interface EmitResult extends ts.EmitResult {
/** Diagnostic information created when trying to emit stuff */
diagnostics: ReadonlyArray<ts.Diagnostic | Diagnostic>;
}
Expand Down
185 changes: 185 additions & 0 deletions packages/jsii/lib/reserved-words.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,185 @@
export function isReservedName(name: string): string[] | undefined {
const reserved = new Array<string>();
if (CSHARP_RESERVED.has(name)) {
reserved.push('C#');
}
if (JAVA_RESERVED.has(name)) {
reserved.push('Java');
}
if (PYTHON_RESERVED.has(name)) {
reserved.push('Python');
}
return reserved.length > 0 ? reserved : undefined;
}

const CSHARP_RESERVED = new Set([
'abstract',
'as',
'base',
'bool',
'break',
'byte',
'case',
'catch',
'char',
'checked',
'class',
'const',
'continue',
'decimal',
'default',
'delegate',
'do',
'double',
'else',
'enum',
'event',
'explicit',
'extern',
'false',
'finally',
'fixed',
'float',
'for',
'foreach',
'goto',
'if',
'implicit',
'in',
'int',
'interface',
'internal',
'is',
'lock',
'long',
'namespace',
'new',
'null',
'object',
'operator',
'out',
'override',
'params',
'private',
'protected',
'public',
'readonly',
'ref',
'return',
'sbyte',
'sealed',
'short',
'sizeof',
'stackalloc',
'static',
'string',
'struct',
'switch',
'this',
'throw',
'true',
'try',
'typeof',
'uint',
'ulong',
'unchecked',
'unsafe',
'ushort',
'using',
'virtual',
'void',
'volatile',
'while',
]);

const JAVA_RESERVED = new Set([
'abstract',
'continue',
'for',
'new',
'switch',
'assert',
'default',
'goto',
'package',
'synchronized',
'boolean',
'do',
'if',
'private',
'this',
'break',
'double',
'implements',
'protected',
'throw',
'byte',
'else',
'import',
'public',
'throws',
'case',
'enum',
'instanceof',
'return',
'transient',
'catch',
'extends',
'int',
'short',
'try',
'char',
'final',
'interface',
'static',
'void',
'class',
'finally',
'long',
'strictfp',
'volatile',
'const',
'float',
'native',
'super',
'while',
'true',
'false',
'null',
]);

const PYTHON_RESERVED = new Set([
'False',
'class',
'finally',
'is',
'return',
'None',
'continue',
'for',
'lambda',
'try',
'True',
'def',
'from',
'nonlocal',
'while',
'and',
'del',
'global',
'not',
'with',
'as',
'elif',
'if',
'or',
'yield',
'assert',
'else',
'import',
'pass',
'break',
'except',
'in',
'raise',
]);
2 changes: 1 addition & 1 deletion packages/jsii/lib/validator.ts
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ export class Validator implements Emitter {
try {
return {
diagnostics: this._diagnostics,
hasErrors: this._diagnostics.find(diag => diag.category === ts.DiagnosticCategory.Error) != null
emitSkipped: this._diagnostics.some(diag => diag.category === ts.DiagnosticCategory.Error)
};
} finally {
// Clearing ``this._diagnostics`` to allow contents to be garbage-collected.
Expand Down
Loading

0 comments on commit 4073f10

Please sign in to comment.