refactor(compiler-cli): skip class decorators in tooling constructor parameters transform (#37545)

We recently added a transformer to NGC that is responsible for downleveling Angular
decorators and constructor parameter types. The primary goal was to mitigate a
TypeScript limitation/issue that surfaces in Angular projects due to the heavy
reliance on type metadata being captured for DI. Additionally this is a pre-requisite
of making `tsickle` optional in the Angular bazel toolchain.

See: 401ef71ae5 for more context on this.

Another (less important) goal was to make sure that the CLI can re-use
this transformer for its JIT mode compilation. The CLI (as outlined in
the commit mentioned above), already has a transformer for downleveling
constructor parameters. We want to avoid this duplication and exported
the transform through the tooling-private compiler entry-point.

Early experiments in using this transformer over the current one, highlighted
that in JIT, class decorators cannot be downleveled. Angular relies on those
to be invoked immediately for JIT (so that factories etc. are generated upon loading)

The transformer we exposed, always downlevels such class decorators
though, so that would break CLI's JIT mode. We can address the CLI's
needs by adding another flag to skip class decorators. This will allow
us to continue with the goal of de-duplication.

PR Close #37545
This commit is contained in:
Paul Gschwendtner 2020-06-12 00:22:00 +02:00 committed by Misko Hevery
parent cc49a91de7
commit 66e6b932d8
4 changed files with 122 additions and 13 deletions

View File

@ -35,15 +35,15 @@ export const GLOBAL_DEFS_FOR_TERSER_WITH_AOT = {
/** /**
* Transform for downleveling Angular decorators and Angular-decorated class constructor * Transform for downleveling Angular decorators and Angular-decorated class constructor
* parameters for dependency injection. This transform can be used by the CLI for JIT-mode * parameters for dependency injection. This transform can be used by the CLI for JIT-mode
* compilation where decorators should be preserved, but downleveled so that apps are not * compilation where constructor parameters and associated Angular decorators should be
* exposed to the ES2015 temporal dead zone limitation in TypeScript's metadata. * downleveled so that apps are not exposed to the ES2015 temporal dead zone limitation
* See https://github.com/angular/angular-cli/pull/14473 for more details. * in TypeScript. See https://github.com/angular/angular-cli/pull/14473 for more details.
*/ */
export function decoratorDownlevelTransformerFactory(program: ts.Program): export function constructorParametersDownlevelTransform(program: ts.Program):
ts.TransformerFactory<ts.SourceFile> { ts.TransformerFactory<ts.SourceFile> {
const typeChecker = program.getTypeChecker(); const typeChecker = program.getTypeChecker();
const reflectionHost = new TypeScriptReflectionHost(typeChecker); const reflectionHost = new TypeScriptReflectionHost(typeChecker);
return getDownlevelDecoratorsTransform( return getDownlevelDecoratorsTransform(
typeChecker, reflectionHost, [], /* isCore */ false, typeChecker, reflectionHost, [], /* isCore */ false,
/* enableClosureCompiler */ false); /* enableClosureCompiler */ false, /* skipClassDecorators */ true);
} }

View File

@ -327,10 +327,17 @@ interface ParameterDecorationInfo {
* @param diagnostics List which will be populated with diagnostics if any. * @param diagnostics List which will be populated with diagnostics if any.
* @param isCore Whether the current TypeScript program is for the `@angular/core` package. * @param isCore Whether the current TypeScript program is for the `@angular/core` package.
* @param isClosureCompilerEnabled Whether closure annotations need to be added where needed. * @param isClosureCompilerEnabled Whether closure annotations need to be added where needed.
* @param skipClassDecorators Whether class decorators should be skipped from downleveling.
* This is useful for JIT mode where class decorators should be preserved as they could rely
* on immediate execution. e.g. downleveling `@Injectable` means that the injectable factory
* is not created, and injecting the token will not work. If this decorator would not be
* downleveled, the `Injectable` decorator will execute immediately on file load, and
* Angular will generate the corresponding injectable factory.
*/ */
export function getDownlevelDecoratorsTransform( export function getDownlevelDecoratorsTransform(
typeChecker: ts.TypeChecker, host: ReflectionHost, diagnostics: ts.Diagnostic[], typeChecker: ts.TypeChecker, host: ReflectionHost, diagnostics: ts.Diagnostic[],
isCore: boolean, isClosureCompilerEnabled: boolean): ts.TransformerFactory<ts.SourceFile> { isCore: boolean, isClosureCompilerEnabled: boolean,
skipClassDecorators: boolean): ts.TransformerFactory<ts.SourceFile> {
return (context: ts.TransformationContext) => { return (context: ts.TransformationContext) => {
let referencedParameterTypes = new Set<ts.Declaration>(); let referencedParameterTypes = new Set<ts.Declaration>();
@ -517,13 +524,22 @@ export function getDownlevelDecoratorsTransform(
} }
const decorators = host.getDecoratorsOfDeclaration(classDecl) || []; const decorators = host.getDecoratorsOfDeclaration(classDecl) || [];
let hasAngularDecorator = false;
const decoratorsToLower = []; const decoratorsToLower = [];
const decoratorsToKeep: ts.Decorator[] = []; const decoratorsToKeep: ts.Decorator[] = [];
for (const decorator of decorators) { for (const decorator of decorators) {
// We only deal with concrete nodes in TypeScript sources, so we don't // We only deal with concrete nodes in TypeScript sources, so we don't
// need to handle synthetically created decorators. // need to handle synthetically created decorators.
const decoratorNode = decorator.node! as ts.Decorator; const decoratorNode = decorator.node! as ts.Decorator;
if (isAngularDecorator(decorator, isCore)) { const isNgDecorator = isAngularDecorator(decorator, isCore);
// Keep track if we come across an Angular class decorator. This is used
// for to determine whether constructor parameters should be captured or not.
if (isNgDecorator) {
hasAngularDecorator = true;
}
if (isNgDecorator && !skipClassDecorators) {
decoratorsToLower.push(extractMetadataFromSingleDecorator(decoratorNode, diagnostics)); decoratorsToLower.push(extractMetadataFromSingleDecorator(decoratorNode, diagnostics));
} else { } else {
decoratorsToKeep.push(decoratorNode); decoratorsToKeep.push(decoratorNode);
@ -536,9 +552,9 @@ export function getDownlevelDecoratorsTransform(
newMembers.push(createDecoratorClassProperty(decoratorsToLower)); newMembers.push(createDecoratorClassProperty(decoratorsToLower));
} }
if (classParameters) { if (classParameters) {
if ((decoratorsToLower.length) || classParameters.some(p => !!p.decorators.length)) { if (hasAngularDecorator || classParameters.some(p => !!p.decorators.length)) {
// emit ctorParameters if the class was decoratored at all, or if any of its ctors // Capture constructor parameters if the class has Angular decorator applied,
// were classParameters // or if any of the parameters has decorators applied directly.
newMembers.push(createCtorParametersClassProperty( newMembers.push(createCtorParametersClassProperty(
diagnostics, entityNameToExpression, classParameters, isClosureCompilerEnabled)); diagnostics, entityNameToExpression, classParameters, isClosureCompilerEnabled));
} }

View File

@ -535,8 +535,8 @@ class AngularCompilerProgram implements Program {
// ignore diagnostics that have been collected by the transformer. These are // ignore diagnostics that have been collected by the transformer. These are
// non-significant failures that shouldn't prevent apps from compiling. // non-significant failures that shouldn't prevent apps from compiling.
beforeTs.push(getDownlevelDecoratorsTransform( beforeTs.push(getDownlevelDecoratorsTransform(
typeChecker, reflectionHost, [], this.isCompilingAngularCore, typeChecker, reflectionHost, [], this.isCompilingAngularCore, annotateForClosureCompiler,
annotateForClosureCompiler)); /* skipClassDecorators */ false));
} }
if (metadataTransforms.length > 0) { if (metadataTransforms.length > 0) {

View File

@ -21,6 +21,7 @@ describe('downlevel decorator transform', () => {
let context: MockAotContext; let context: MockAotContext;
let diagnostics: ts.Diagnostic[]; let diagnostics: ts.Diagnostic[];
let isClosureEnabled: boolean; let isClosureEnabled: boolean;
let skipClassDecorators: boolean;
beforeEach(() => { beforeEach(() => {
diagnostics = []; diagnostics = [];
@ -32,6 +33,7 @@ describe('downlevel decorator transform', () => {
}); });
host = new MockCompilerHost(context); host = new MockCompilerHost(context);
isClosureEnabled = false; isClosureEnabled = false;
skipClassDecorators = false;
}); });
function transform( function transform(
@ -58,7 +60,7 @@ describe('downlevel decorator transform', () => {
...preTransformers, ...preTransformers,
getDownlevelDecoratorsTransform( getDownlevelDecoratorsTransform(
program.getTypeChecker(), reflectionHost, diagnostics, program.getTypeChecker(), reflectionHost, diagnostics,
/* isCore */ false, isClosureEnabled) /* isCore */ false, isClosureEnabled, skipClassDecorators)
] ]
}; };
let output: string|null = null; let output: string|null = null;
@ -606,6 +608,97 @@ describe('downlevel decorator transform', () => {
expect(output).not.toContain('MyDir.ctorParameters'); expect(output).not.toContain('MyDir.ctorParameters');
expect(output).not.toContain('tslib'); expect(output).not.toContain('tslib');
}); });
describe('class decorators skipped', () => {
beforeEach(() => skipClassDecorators = true);
it('should not downlevel Angular class decorators', () => {
const {output} = transform(`
import {Injectable} from '@angular/core';
@Injectable()
export class MyService {}
`);
expect(diagnostics.length).toBe(0);
expect(output).not.toContain('MyService.decorators');
expect(output).toContain(dedent`
MyService = tslib_1.__decorate([
core_1.Injectable()
], MyService);
`);
});
it('should downlevel constructor parameters', () => {
const {output} = transform(`
import {Injectable} from '@angular/core';
@Injectable()
export class InjectClass {}
@Injectable()
export class MyService {
constructor(dep: InjectClass) {}
}
`);
expect(diagnostics.length).toBe(0);
expect(output).not.toContain('MyService.decorators');
expect(output).toContain('MyService.ctorParameters');
expect(output).toContain(dedent`
MyService.ctorParameters = () => [
{ type: InjectClass }
];
MyService = tslib_1.__decorate([
core_1.Injectable()
], MyService);
`);
});
it('should downlevel constructor parameter decorators', () => {
const {output} = transform(`
import {Injectable, Inject} from '@angular/core';
@Injectable()
export class InjectClass {}
@Injectable()
export class MyService {
constructor(@Inject('test') dep: InjectClass) {}
}
`);
expect(diagnostics.length).toBe(0);
expect(output).not.toContain('MyService.decorators');
expect(output).toContain('MyService.ctorParameters');
expect(output).toContain(dedent`
MyService.ctorParameters = () => [
{ type: InjectClass, decorators: [{ type: core_1.Inject, args: ['test',] }] }
];
MyService = tslib_1.__decorate([
core_1.Injectable()
], MyService);
`);
});
it('should downlevel class member Angular decorators', () => {
const {output} = transform(`
import {Injectable, Input} from '@angular/core';
export class MyService {
@Input() disabled: boolean;
}
`);
expect(diagnostics.length).toBe(0);
expect(output).not.toContain('tslib');
expect(output).toContain(dedent`
MyService.propDecorators = {
disabled: [{ type: core_1.Input }]
};
`);
});
});
}); });
/** Template string function that can be used to dedent a given string literal. */ /** Template string function that can be used to dedent a given string literal. */