2016-06-23 09:47:54 -07:00
|
|
|
/**
|
|
|
|
* @license
|
|
|
|
* Copyright Google Inc. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
|
|
* found in the LICENSE file at https://angular.io/license
|
|
|
|
*/
|
|
|
|
|
2018-02-16 08:45:21 -08:00
|
|
|
import {CompileDirectiveMetadata, CompileDirectiveSummary, CompileIdentifierMetadata, CompileInjectableMetadata, CompileNgModuleMetadata, CompileNgModuleSummary, CompilePipeMetadata, CompilePipeSummary, CompileProviderMetadata, CompileShallowModuleMetadata, CompileStylesheetMetadata, CompileSummaryKind, CompileTypeMetadata, CompileTypeSummary, componentFactoryName, flatten, identifierName, templateSourceUrl, tokenReference} from '../compile_metadata';
|
2017-02-17 08:56:49 -08:00
|
|
|
import {CompilerConfig} from '../config';
|
2017-11-20 10:21:17 -08:00
|
|
|
import {ConstantPool} from '../constant_pool';
|
2017-09-12 09:40:28 -07:00
|
|
|
import {ViewEncapsulation} from '../core';
|
2017-09-12 15:53:17 -07:00
|
|
|
import {MessageBundle} from '../i18n/message_bundle';
|
2017-05-18 13:46:51 -07:00
|
|
|
import {Identifiers, createTokenForExternalReference} from '../identifiers';
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
import {InjectableCompiler} from '../injectable_compiler';
|
2016-11-14 17:37:47 -08:00
|
|
|
import {CompileMetadataResolver} from '../metadata_resolver';
|
2017-09-12 15:53:17 -07:00
|
|
|
import {HtmlParser} from '../ml_parser/html_parser';
|
2018-02-15 16:43:16 -08:00
|
|
|
import {DEFAULT_INTERPOLATION_CONFIG, InterpolationConfig} from '../ml_parser/interpolation_config';
|
2016-11-14 17:37:47 -08:00
|
|
|
import {NgModuleCompiler} from '../ng_module_compiler';
|
|
|
|
import {OutputEmitter} from '../output/abstract_emitter';
|
|
|
|
import * as o from '../output/output_ast';
|
2017-09-12 15:53:17 -07:00
|
|
|
import {ParseError} from '../parse_util';
|
2018-02-16 08:45:21 -08:00
|
|
|
import {compileNgModule as compileIvyModule} from '../render3/r3_module_compiler';
|
2018-02-05 17:31:12 -08:00
|
|
|
import {compilePipe as compileIvyPipe} from '../render3/r3_pipe_compiler';
|
2018-02-28 14:56:41 -08:00
|
|
|
import {OutputMode} from '../render3/r3_types';
|
2018-01-11 15:37:56 -08:00
|
|
|
import {compileComponent as compileIvyComponent, compileDirective as compileIvyDirective} from '../render3/r3_view_compiler';
|
2016-11-14 17:37:47 -08:00
|
|
|
import {CompiledStylesheet, StyleCompiler} from '../style_compiler';
|
2016-12-15 09:12:40 -08:00
|
|
|
import {SummaryResolver} from '../summary_resolver';
|
2018-02-15 16:43:16 -08:00
|
|
|
import {BindingParser} from '../template_parser/binding_parser';
|
2017-09-11 15:18:19 -07:00
|
|
|
import {TemplateAst} from '../template_parser/template_ast';
|
2016-11-14 17:37:47 -08:00
|
|
|
import {TemplateParser} from '../template_parser/template_parser';
|
2017-11-20 10:21:17 -08:00
|
|
|
import {OutputContext, ValueVisitor, error, syntaxError, visitValue} from '../util';
|
2017-09-11 15:18:19 -07:00
|
|
|
import {TypeCheckCompiler} from '../view_compiler/type_check_compiler';
|
2017-05-16 16:30:37 -07:00
|
|
|
import {ViewCompileResult, ViewCompiler} from '../view_compiler/view_compiler';
|
2016-11-14 17:37:47 -08:00
|
|
|
|
2016-12-15 09:12:40 -08:00
|
|
|
import {AotCompilerHost} from './compiler_host';
|
2017-09-20 09:54:47 -07:00
|
|
|
import {AotCompilerOptions} from './compiler_options';
|
2016-11-29 15:36:33 -08:00
|
|
|
import {GeneratedFile} from './generated_file';
|
2017-10-20 09:46:41 -07:00
|
|
|
import {LazyRoute, listLazyRoutes, parseLazyRoute} from './lazy_routes';
|
2017-11-20 10:21:17 -08:00
|
|
|
import {PartialModule} from './partial_module';
|
2017-05-18 13:46:51 -07:00
|
|
|
import {StaticReflector} from './static_reflector';
|
2016-11-14 17:37:47 -08:00
|
|
|
import {StaticSymbol} from './static_symbol';
|
2017-04-26 09:24:42 -07:00
|
|
|
import {ResolvedStaticSymbol, StaticSymbolResolver} from './static_symbol_resolver';
|
2017-05-23 13:40:50 -07:00
|
|
|
import {createForJitStub, serializeSummaries} from './summary_serializer';
|
2017-11-29 15:27:16 +08:00
|
|
|
import {ngfactoryFilePath, normalizeGenFileSuffix, splitTypescriptSuffix, summaryFileName, summaryForJitFileName, summaryForJitName} from './util';
|
2016-01-06 14:13:44 -08:00
|
|
|
|
2017-09-20 16:31:02 -07:00
|
|
|
enum StubEmitFlags {
|
2017-09-12 09:40:28 -07:00
|
|
|
Basic = 1 << 0,
|
|
|
|
TypeCheck = 1 << 1,
|
|
|
|
All = TypeCheck | Basic
|
|
|
|
}
|
|
|
|
|
2016-11-14 17:37:47 -08:00
|
|
|
export class AotCompiler {
|
2017-09-11 15:18:19 -07:00
|
|
|
private _templateAstCache =
|
|
|
|
new Map<StaticSymbol, {template: TemplateAst[], pipes: CompilePipeSummary[]}>();
|
2017-09-21 18:05:07 -07:00
|
|
|
private _analyzedFiles = new Map<string, NgAnalyzedFile>();
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
private _analyzedFilesForInjectables = new Map<string, NgAnalyzedFileWithInjectables>();
|
2017-09-11 15:18:19 -07:00
|
|
|
|
2016-06-08 16:38:52 -07:00
|
|
|
constructor(
|
2017-09-21 18:05:07 -07:00
|
|
|
private _config: CompilerConfig, private _options: AotCompilerOptions,
|
2018-02-16 08:45:21 -08:00
|
|
|
private _host: AotCompilerHost, readonly reflector: StaticReflector,
|
2017-09-20 09:54:47 -07:00
|
|
|
private _metadataResolver: CompileMetadataResolver, private _templateParser: TemplateParser,
|
|
|
|
private _styleCompiler: StyleCompiler, private _viewCompiler: ViewCompiler,
|
|
|
|
private _typeCheckCompiler: TypeCheckCompiler, private _ngModuleCompiler: NgModuleCompiler,
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
private _injectableCompiler: InjectableCompiler, private _outputEmitter: OutputEmitter,
|
2017-09-20 09:54:47 -07:00
|
|
|
private _summaryResolver: SummaryResolver<StaticSymbol>,
|
|
|
|
private _symbolResolver: StaticSymbolResolver) {}
|
2016-07-18 03:50:31 -07:00
|
|
|
|
2016-11-10 14:07:30 -08:00
|
|
|
clearCache() { this._metadataResolver.clearCache(); }
|
2016-06-28 09:54:42 -07:00
|
|
|
|
2017-05-23 13:40:50 -07:00
|
|
|
analyzeModulesSync(rootFiles: string[]): NgAnalyzedModules {
|
2017-08-18 14:03:59 -07:00
|
|
|
const analyzeResult = analyzeAndValidateNgModules(
|
2017-09-12 09:40:28 -07:00
|
|
|
rootFiles, this._host, this._symbolResolver, this._metadataResolver);
|
2017-05-23 13:40:50 -07:00
|
|
|
analyzeResult.ngModules.forEach(
|
|
|
|
ngModule => this._metadataResolver.loadNgModuleDirectiveAndPipeMetadata(
|
|
|
|
ngModule.type.reference, true));
|
|
|
|
return analyzeResult;
|
|
|
|
}
|
|
|
|
|
|
|
|
analyzeModulesAsync(rootFiles: string[]): Promise<NgAnalyzedModules> {
|
2017-08-18 14:03:59 -07:00
|
|
|
const analyzeResult = analyzeAndValidateNgModules(
|
2017-09-12 09:40:28 -07:00
|
|
|
rootFiles, this._host, this._symbolResolver, this._metadataResolver);
|
2016-11-29 08:08:22 -08:00
|
|
|
return Promise
|
2017-05-23 13:40:50 -07:00
|
|
|
.all(analyzeResult.ngModules.map(
|
2016-11-29 08:08:22 -08:00
|
|
|
ngModule => this._metadataResolver.loadNgModuleDirectiveAndPipeMetadata(
|
|
|
|
ngModule.type.reference, false)))
|
2017-05-23 13:40:50 -07:00
|
|
|
.then(() => analyzeResult);
|
2016-10-24 13:28:23 -07:00
|
|
|
}
|
|
|
|
|
2017-09-21 18:05:07 -07:00
|
|
|
private _analyzeFile(fileName: string): NgAnalyzedFile {
|
|
|
|
let analyzedFile = this._analyzedFiles.get(fileName);
|
|
|
|
if (!analyzedFile) {
|
|
|
|
analyzedFile =
|
|
|
|
analyzeFile(this._host, this._symbolResolver, this._metadataResolver, fileName);
|
|
|
|
this._analyzedFiles.set(fileName, analyzedFile);
|
|
|
|
}
|
|
|
|
return analyzedFile;
|
|
|
|
}
|
|
|
|
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
private _analyzeFileForInjectables(fileName: string): NgAnalyzedFileWithInjectables {
|
|
|
|
let analyzedFile = this._analyzedFilesForInjectables.get(fileName);
|
|
|
|
if (!analyzedFile) {
|
|
|
|
analyzedFile = analyzeFileForInjectables(
|
|
|
|
this._host, this._symbolResolver, this._metadataResolver, fileName);
|
|
|
|
this._analyzedFilesForInjectables.set(fileName, analyzedFile);
|
|
|
|
}
|
|
|
|
return analyzedFile;
|
|
|
|
}
|
|
|
|
|
2017-09-21 18:05:07 -07:00
|
|
|
findGeneratedFileNames(fileName: string): string[] {
|
|
|
|
const genFileNames: string[] = [];
|
|
|
|
const file = this._analyzeFile(fileName);
|
|
|
|
// Make sure we create a .ngfactory if we have a injectable/directive/pipe/NgModule
|
|
|
|
// or a reference to a non source file.
|
|
|
|
// Note: This is overestimating the required .ngfactory files as the real calculation is harder.
|
|
|
|
// Only do this for StubEmitFlags.Basic, as adding a type check block
|
|
|
|
// does not change this file (as we generate type check blocks based on NgModules).
|
|
|
|
if (this._options.allowEmptyCodegenFiles || file.directives.length || file.pipes.length ||
|
|
|
|
file.injectables.length || file.ngModules.length || file.exportsNonSourceFiles) {
|
|
|
|
genFileNames.push(ngfactoryFilePath(file.fileName, true));
|
2017-09-29 14:55:44 -07:00
|
|
|
if (this._options.enableSummariesForJit) {
|
|
|
|
genFileNames.push(summaryForJitFileName(file.fileName, true));
|
|
|
|
}
|
2017-09-21 18:05:07 -07:00
|
|
|
}
|
2017-11-29 15:27:16 +08:00
|
|
|
const fileSuffix = normalizeGenFileSuffix(splitTypescriptSuffix(file.fileName, true)[1]);
|
2017-09-21 18:05:07 -07:00
|
|
|
file.directives.forEach((dirSymbol) => {
|
|
|
|
const compMeta =
|
|
|
|
this._metadataResolver.getNonNormalizedDirectiveMetadata(dirSymbol) !.metadata;
|
|
|
|
if (!compMeta.isComponent) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Note: compMeta is a component and therefore template is non null.
|
|
|
|
compMeta.template !.styleUrls.forEach((styleUrl) => {
|
|
|
|
const normalizedUrl = this._host.resourceNameToFileName(styleUrl, file.fileName);
|
|
|
|
if (!normalizedUrl) {
|
2017-10-17 16:10:15 -07:00
|
|
|
throw syntaxError(`Couldn't resolve resource ${styleUrl} relative to ${file.fileName}`);
|
2017-09-21 18:05:07 -07:00
|
|
|
}
|
|
|
|
const needsShim = (compMeta.template !.encapsulation ||
|
|
|
|
this._config.defaultEncapsulation) === ViewEncapsulation.Emulated;
|
|
|
|
genFileNames.push(_stylesModuleUrl(normalizedUrl, needsShim, fileSuffix));
|
|
|
|
if (this._options.allowEmptyCodegenFiles) {
|
|
|
|
genFileNames.push(_stylesModuleUrl(normalizedUrl, !needsShim, fileSuffix));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
return genFileNames;
|
2017-05-23 13:40:50 -07:00
|
|
|
}
|
|
|
|
|
2017-09-21 18:05:07 -07:00
|
|
|
emitBasicStub(genFileName: string, originalFileName?: string): GeneratedFile {
|
|
|
|
const outputCtx = this._createOutputContext(genFileName);
|
|
|
|
if (genFileName.endsWith('.ngfactory.ts')) {
|
|
|
|
if (!originalFileName) {
|
|
|
|
throw new Error(
|
|
|
|
`Assertion error: require the original file for .ngfactory.ts stubs. File: ${genFileName}`);
|
|
|
|
}
|
|
|
|
const originalFile = this._analyzeFile(originalFileName);
|
|
|
|
this._createNgFactoryStub(outputCtx, originalFile, StubEmitFlags.Basic);
|
|
|
|
} else if (genFileName.endsWith('.ngsummary.ts')) {
|
|
|
|
if (this._options.enableSummariesForJit) {
|
|
|
|
if (!originalFileName) {
|
|
|
|
throw new Error(
|
|
|
|
`Assertion error: require the original file for .ngsummary.ts stubs. File: ${genFileName}`);
|
|
|
|
}
|
|
|
|
const originalFile = this._analyzeFile(originalFileName);
|
|
|
|
_createEmptyStub(outputCtx);
|
|
|
|
originalFile.ngModules.forEach(ngModule => {
|
|
|
|
// create exports that user code can reference
|
|
|
|
createForJitStub(outputCtx, ngModule.type.reference);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
} else if (genFileName.endsWith('.ngstyle.ts')) {
|
|
|
|
_createEmptyStub(outputCtx);
|
|
|
|
}
|
|
|
|
// Note: for the stubs, we don't need a property srcFileUrl,
|
2018-02-28 14:56:41 -08:00
|
|
|
// as later on in emitAllImpls we will create the proper GeneratedFiles with the
|
2017-09-21 18:05:07 -07:00
|
|
|
// correct srcFileUrl.
|
|
|
|
// This is good as e.g. for .ngstyle.ts files we can't derive
|
|
|
|
// the url of components based on the genFileUrl.
|
|
|
|
return this._codegenSourceModule('unknown', outputCtx);
|
2017-09-12 09:40:28 -07:00
|
|
|
}
|
|
|
|
|
2017-09-21 18:05:07 -07:00
|
|
|
emitTypeCheckStub(genFileName: string, originalFileName: string): GeneratedFile|null {
|
|
|
|
const originalFile = this._analyzeFile(originalFileName);
|
|
|
|
const outputCtx = this._createOutputContext(genFileName);
|
|
|
|
if (genFileName.endsWith('.ngfactory.ts')) {
|
|
|
|
this._createNgFactoryStub(outputCtx, originalFile, StubEmitFlags.TypeCheck);
|
|
|
|
}
|
|
|
|
return outputCtx.statements.length > 0 ?
|
|
|
|
this._codegenSourceModule(originalFile.fileName, outputCtx) :
|
|
|
|
null;
|
2017-09-12 09:40:28 -07:00
|
|
|
}
|
|
|
|
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
loadFilesAsync(fileNames: string[], tsFiles: string[]): Promise<
|
|
|
|
{analyzedModules: NgAnalyzedModules, analyzedInjectables: NgAnalyzedFileWithInjectables[]}> {
|
2017-09-21 18:05:07 -07:00
|
|
|
const files = fileNames.map(fileName => this._analyzeFile(fileName));
|
2017-09-12 09:40:28 -07:00
|
|
|
const loadingPromises: Promise<NgAnalyzedModules>[] = [];
|
|
|
|
files.forEach(
|
|
|
|
file => file.ngModules.forEach(
|
|
|
|
ngModule =>
|
|
|
|
loadingPromises.push(this._metadataResolver.loadNgModuleDirectiveAndPipeMetadata(
|
|
|
|
ngModule.type.reference, false))));
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
const analyzedInjectables = tsFiles.map(tsFile => this._analyzeFileForInjectables(tsFile));
|
|
|
|
return Promise.all(loadingPromises).then(_ => ({
|
|
|
|
analyzedModules: mergeAndValidateNgFiles(files),
|
|
|
|
analyzedInjectables: analyzedInjectables,
|
|
|
|
}));
|
2017-09-12 09:40:28 -07:00
|
|
|
}
|
|
|
|
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
loadFilesSync(fileNames: string[], tsFiles: string[]):
|
|
|
|
{analyzedModules: NgAnalyzedModules, analyzedInjectables: NgAnalyzedFileWithInjectables[]} {
|
2017-09-21 18:05:07 -07:00
|
|
|
const files = fileNames.map(fileName => this._analyzeFile(fileName));
|
2017-09-12 09:40:28 -07:00
|
|
|
files.forEach(
|
|
|
|
file => file.ngModules.forEach(
|
|
|
|
ngModule => this._metadataResolver.loadNgModuleDirectiveAndPipeMetadata(
|
|
|
|
ngModule.type.reference, true)));
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
const analyzedInjectables = tsFiles.map(tsFile => this._analyzeFileForInjectables(tsFile));
|
|
|
|
return {
|
|
|
|
analyzedModules: mergeAndValidateNgFiles(files),
|
|
|
|
analyzedInjectables: analyzedInjectables,
|
|
|
|
};
|
2017-09-12 09:40:28 -07:00
|
|
|
}
|
|
|
|
|
2017-09-21 18:05:07 -07:00
|
|
|
private _createNgFactoryStub(
|
|
|
|
outputCtx: OutputContext, file: NgAnalyzedFile, emitFlags: StubEmitFlags) {
|
2017-10-26 09:45:01 -07:00
|
|
|
let componentId = 0;
|
2017-09-12 09:40:28 -07:00
|
|
|
file.ngModules.forEach((ngModuleMeta, ngModuleIndex) => {
|
|
|
|
// Note: the code below needs to executed for StubEmitFlags.Basic and StubEmitFlags.TypeCheck,
|
2018-02-28 14:56:41 -08:00
|
|
|
// so we don't change the .ngfactory file too much when adding the type-check block.
|
2017-09-12 09:40:28 -07:00
|
|
|
|
|
|
|
// create exports that user code can reference
|
|
|
|
this._ngModuleCompiler.createStub(outputCtx, ngModuleMeta.type.reference);
|
|
|
|
|
|
|
|
// add references to the symbols from the metadata.
|
|
|
|
// These can be used by the type check block for components,
|
|
|
|
// and they also cause TypeScript to include these files into the program too,
|
|
|
|
// which will make them part of the analyzedFiles.
|
|
|
|
const externalReferences: StaticSymbol[] = [
|
2017-11-15 13:20:51 -08:00
|
|
|
// Add references that are available from all the modules and imports.
|
2017-10-05 14:36:15 -07:00
|
|
|
...ngModuleMeta.transitiveModule.directives.map(d => d.reference),
|
|
|
|
...ngModuleMeta.transitiveModule.pipes.map(d => d.reference),
|
2017-09-12 09:40:28 -07:00
|
|
|
...ngModuleMeta.importedModules.map(m => m.type.reference),
|
|
|
|
...ngModuleMeta.exportedModules.map(m => m.type.reference),
|
2017-11-15 13:20:51 -08:00
|
|
|
|
|
|
|
// Add references that might be inserted by the template compiler.
|
|
|
|
...this._externalIdentifierReferences([Identifiers.TemplateRef, Identifiers.ElementRef]),
|
2017-09-12 09:40:28 -07:00
|
|
|
];
|
2017-11-15 13:20:51 -08:00
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
const externalReferenceVars = new Map<any, string>();
|
|
|
|
externalReferences.forEach((ref, typeIndex) => {
|
2017-12-11 08:50:46 -08:00
|
|
|
externalReferenceVars.set(ref, `_decl${ngModuleIndex}_${typeIndex}`);
|
2017-09-12 09:40:28 -07:00
|
|
|
});
|
|
|
|
externalReferenceVars.forEach((varName, reference) => {
|
|
|
|
outputCtx.statements.push(
|
|
|
|
o.variable(varName)
|
|
|
|
.set(o.NULL_EXPR.cast(o.DYNAMIC_TYPE))
|
2017-12-11 08:50:46 -08:00
|
|
|
.toDeclStmt(o.expressionType(outputCtx.importExpr(
|
|
|
|
reference, /* typeParams */ null, /* useSummaries */ false))));
|
2017-09-12 09:40:28 -07:00
|
|
|
});
|
|
|
|
|
|
|
|
if (emitFlags & StubEmitFlags.TypeCheck) {
|
2018-02-28 14:56:41 -08:00
|
|
|
// add the type-check block for all components of the NgModule
|
2017-09-12 09:40:28 -07:00
|
|
|
ngModuleMeta.declaredDirectives.forEach((dirId) => {
|
|
|
|
const compMeta = this._metadataResolver.getDirectiveMetadata(dirId.reference);
|
|
|
|
if (!compMeta.isComponent) {
|
|
|
|
return;
|
|
|
|
}
|
2017-10-26 09:45:01 -07:00
|
|
|
componentId++;
|
2017-09-12 09:40:28 -07:00
|
|
|
this._createTypeCheckBlock(
|
2017-10-26 09:45:01 -07:00
|
|
|
outputCtx, `${compMeta.type.reference.name}_Host_${componentId}`, ngModuleMeta,
|
|
|
|
this._metadataResolver.getHostComponentMetadata(compMeta), [compMeta.type],
|
2017-09-12 09:40:28 -07:00
|
|
|
externalReferenceVars);
|
2017-10-26 09:45:01 -07:00
|
|
|
this._createTypeCheckBlock(
|
|
|
|
outputCtx, `${compMeta.type.reference.name}_${componentId}`, ngModuleMeta, compMeta,
|
|
|
|
ngModuleMeta.transitiveModule.directives, externalReferenceVars);
|
2017-09-12 09:40:28 -07:00
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2017-09-21 18:05:07 -07:00
|
|
|
if (outputCtx.statements.length === 0) {
|
2017-09-12 09:40:28 -07:00
|
|
|
_createEmptyStub(outputCtx);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-15 13:20:51 -08:00
|
|
|
private _externalIdentifierReferences(references: o.ExternalReference[]): StaticSymbol[] {
|
|
|
|
const result: StaticSymbol[] = [];
|
|
|
|
for (let reference of references) {
|
2018-02-16 08:45:21 -08:00
|
|
|
const token = createTokenForExternalReference(this.reflector, reference);
|
2017-11-15 13:20:51 -08:00
|
|
|
if (token.identifier) {
|
|
|
|
result.push(token.identifier.reference);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
private _createTypeCheckBlock(
|
2017-10-26 09:45:01 -07:00
|
|
|
ctx: OutputContext, componentId: string, moduleMeta: CompileNgModuleMetadata,
|
|
|
|
compMeta: CompileDirectiveMetadata, directives: CompileIdentifierMetadata[],
|
|
|
|
externalReferenceVars: Map<any, string>) {
|
2017-09-12 09:40:28 -07:00
|
|
|
const {template: parsedTemplate, pipes: usedPipes} =
|
|
|
|
this._parseTemplate(compMeta, moduleMeta, directives);
|
|
|
|
ctx.statements.push(...this._typeCheckCompiler.compileComponent(
|
2017-11-29 16:29:05 -08:00
|
|
|
componentId, compMeta, parsedTemplate, usedPipes, externalReferenceVars, ctx));
|
2017-05-17 15:39:08 -07:00
|
|
|
}
|
|
|
|
|
2017-09-12 15:53:17 -07:00
|
|
|
emitMessageBundle(analyzeResult: NgAnalyzedModules, locale: string|null): MessageBundle {
|
|
|
|
const errors: ParseError[] = [];
|
|
|
|
const htmlParser = new HtmlParser();
|
|
|
|
|
|
|
|
// TODO(vicb): implicit tags & attributes
|
|
|
|
const messageBundle = new MessageBundle(htmlParser, [], {}, locale);
|
|
|
|
|
|
|
|
analyzeResult.files.forEach(file => {
|
|
|
|
const compMetas: CompileDirectiveMetadata[] = [];
|
|
|
|
file.directives.forEach(directiveType => {
|
|
|
|
const dirMeta = this._metadataResolver.getDirectiveMetadata(directiveType);
|
|
|
|
if (dirMeta && dirMeta.isComponent) {
|
|
|
|
compMetas.push(dirMeta);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
compMetas.forEach(compMeta => {
|
|
|
|
const html = compMeta.template !.template !;
|
|
|
|
const interpolationConfig =
|
|
|
|
InterpolationConfig.fromArray(compMeta.template !.interpolation);
|
2017-09-12 09:40:28 -07:00
|
|
|
errors.push(
|
|
|
|
...messageBundle.updateFromTemplate(html, file.fileName, interpolationConfig) !);
|
2017-09-12 15:53:17 -07:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
if (errors.length) {
|
|
|
|
throw new Error(errors.map(e => e.toString()).join('\n'));
|
|
|
|
}
|
|
|
|
|
|
|
|
return messageBundle;
|
|
|
|
}
|
|
|
|
|
2018-02-16 08:45:21 -08:00
|
|
|
emitAllPartialModules(
|
|
|
|
{ngModuleByPipeOrDirective, files}: NgAnalyzedModules,
|
|
|
|
r3Files: NgAnalyzedFileWithInjectables[]): PartialModule[] {
|
|
|
|
const contextMap = new Map<string, OutputContext>();
|
|
|
|
|
|
|
|
const getContext = (fileName: string): OutputContext => {
|
|
|
|
if (!contextMap.has(fileName)) {
|
|
|
|
contextMap.set(fileName, this._createOutputContext(fileName));
|
|
|
|
}
|
|
|
|
return contextMap.get(fileName) !;
|
|
|
|
};
|
|
|
|
|
|
|
|
files.forEach(
|
|
|
|
file => this._compilePartialModule(
|
|
|
|
file.fileName, ngModuleByPipeOrDirective, file.directives, file.pipes, file.ngModules,
|
|
|
|
file.injectables, getContext(file.fileName)));
|
|
|
|
r3Files.forEach(
|
|
|
|
file => this._compileShallowModules(
|
|
|
|
file.fileName, file.shallowModules, getContext(file.fileName)));
|
|
|
|
|
|
|
|
return Array.from(contextMap.values())
|
|
|
|
.map(context => ({
|
|
|
|
fileName: context.genFilePath,
|
|
|
|
statements: [...context.constantPool.statements, ...context.statements],
|
|
|
|
}));
|
2017-11-20 10:21:17 -08:00
|
|
|
}
|
|
|
|
|
2018-02-16 08:45:21 -08:00
|
|
|
private _compileShallowModules(
|
|
|
|
fileName: string, shallowModules: CompileShallowModuleMetadata[],
|
|
|
|
context: OutputContext): void {
|
|
|
|
shallowModules.forEach(module => compileIvyModule(context, module, this._injectableCompiler));
|
|
|
|
}
|
|
|
|
|
|
|
|
private _compilePartialModule(
|
2017-11-20 10:21:17 -08:00
|
|
|
fileName: string, ngModuleByPipeOrDirective: Map<StaticSymbol, CompileNgModuleMetadata>,
|
|
|
|
directives: StaticSymbol[], pipes: StaticSymbol[], ngModules: CompileNgModuleMetadata[],
|
2018-02-16 08:45:21 -08:00
|
|
|
injectables: CompileInjectableMetadata[], context: OutputContext): void {
|
2017-11-20 10:21:17 -08:00
|
|
|
const classes: o.ClassStmt[] = [];
|
2018-02-15 16:43:16 -08:00
|
|
|
const errors: ParseError[] = [];
|
2017-11-20 10:21:17 -08:00
|
|
|
|
2018-02-15 16:43:16 -08:00
|
|
|
const hostBindingParser = new BindingParser(
|
|
|
|
this._templateParser.expressionParser, DEFAULT_INTERPOLATION_CONFIG, null !, [], errors);
|
|
|
|
|
2018-01-11 15:37:56 -08:00
|
|
|
// Process all components and directives
|
2017-11-20 10:21:17 -08:00
|
|
|
directives.forEach(directiveType => {
|
|
|
|
const directiveMetadata = this._metadataResolver.getDirectiveMetadata(directiveType);
|
|
|
|
if (directiveMetadata.isComponent) {
|
|
|
|
const module = ngModuleByPipeOrDirective.get(directiveType) !;
|
|
|
|
module ||
|
|
|
|
error(
|
|
|
|
`Cannot determine the module for component '${identifierName(directiveMetadata.type)}'`);
|
|
|
|
|
2018-02-05 17:31:12 -08:00
|
|
|
const {template: parsedTemplate, pipes: parsedPipes} =
|
2017-11-20 10:21:17 -08:00
|
|
|
this._parseTemplate(directiveMetadata, module, module.transitiveModule.directives);
|
2018-02-05 17:31:12 -08:00
|
|
|
compileIvyComponent(
|
2018-02-16 08:45:21 -08:00
|
|
|
context, directiveMetadata, parsedPipes, parsedTemplate, this.reflector,
|
2018-02-28 14:56:41 -08:00
|
|
|
hostBindingParser, OutputMode.PartialClass);
|
2018-01-11 15:37:56 -08:00
|
|
|
} else {
|
2018-02-28 14:56:41 -08:00
|
|
|
compileIvyDirective(
|
2018-02-16 08:45:21 -08:00
|
|
|
context, directiveMetadata, this.reflector, hostBindingParser, OutputMode.PartialClass);
|
2017-11-20 10:21:17 -08:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2018-02-05 17:31:12 -08:00
|
|
|
pipes.forEach(pipeType => {
|
|
|
|
const pipeMetadata = this._metadataResolver.getPipeMetadata(pipeType);
|
|
|
|
if (pipeMetadata) {
|
2018-02-16 08:45:21 -08:00
|
|
|
compileIvyPipe(context, pipeMetadata, this.reflector, OutputMode.PartialClass);
|
2018-02-05 17:31:12 -08:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
injectables.forEach(injectable => this._injectableCompiler.compile(injectable, context));
|
|
|
|
}
|
|
|
|
|
|
|
|
emitAllPartialModules2(files: NgAnalyzedFileWithInjectables[]): PartialModule[] {
|
|
|
|
// Using reduce like this is a select many pattern (where map is a select pattern)
|
|
|
|
return files.reduce<PartialModule[]>((r, file) => {
|
|
|
|
r.push(...this._emitPartialModule2(file.fileName, file.injectables));
|
|
|
|
return r;
|
|
|
|
}, []);
|
|
|
|
}
|
|
|
|
|
|
|
|
private _emitPartialModule2(fileName: string, injectables: CompileInjectableMetadata[]):
|
|
|
|
PartialModule[] {
|
|
|
|
const context = this._createOutputContext(fileName);
|
|
|
|
|
|
|
|
injectables.forEach(injectable => this._injectableCompiler.compile(injectable, context));
|
|
|
|
|
|
|
|
if (context.statements && context.statements.length > 0) {
|
2017-11-20 10:21:17 -08:00
|
|
|
return [{fileName, statements: [...context.constantPool.statements, ...context.statements]}];
|
|
|
|
}
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
emitAllImpls(analyzeResult: NgAnalyzedModules): GeneratedFile[] {
|
|
|
|
const {ngModuleByPipeOrDirective, files} = analyzeResult;
|
|
|
|
const sourceModules = files.map(
|
|
|
|
file => this._compileImplFile(
|
|
|
|
file.fileName, ngModuleByPipeOrDirective, file.directives, file.pipes, file.ngModules,
|
|
|
|
file.injectables));
|
|
|
|
return flatten(sourceModules);
|
2017-05-23 13:40:50 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
private _compileImplFile(
|
2016-10-25 16:28:22 -07:00
|
|
|
srcFileUrl: string, ngModuleByPipeOrDirective: Map<StaticSymbol, CompileNgModuleMetadata>,
|
2017-09-12 09:40:28 -07:00
|
|
|
directives: StaticSymbol[], pipes: StaticSymbol[], ngModules: CompileNgModuleMetadata[],
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
injectables: CompileInjectableMetadata[]): GeneratedFile[] {
|
2017-11-29 15:27:16 +08:00
|
|
|
const fileSuffix = normalizeGenFileSuffix(splitTypescriptSuffix(srcFileUrl, true)[1]);
|
2016-11-29 15:36:33 -08:00
|
|
|
const generatedFiles: GeneratedFile[] = [];
|
|
|
|
|
2017-09-21 18:05:07 -07:00
|
|
|
const outputCtx = this._createOutputContext(ngfactoryFilePath(srcFileUrl, true));
|
2017-05-16 16:30:37 -07:00
|
|
|
|
|
|
|
generatedFiles.push(
|
|
|
|
...this._createSummary(srcFileUrl, directives, pipes, ngModules, injectables, outputCtx));
|
2016-06-28 09:54:42 -07:00
|
|
|
|
2016-07-18 03:50:31 -07:00
|
|
|
// compile all ng modules
|
2017-09-12 09:40:28 -07:00
|
|
|
ngModules.forEach((ngModuleMeta) => this._compileModule(outputCtx, ngModuleMeta));
|
2016-06-28 09:54:42 -07:00
|
|
|
|
|
|
|
// compile components
|
2016-11-10 14:07:30 -08:00
|
|
|
directives.forEach((dirType) => {
|
|
|
|
const compMeta = this._metadataResolver.getDirectiveMetadata(<any>dirType);
|
|
|
|
if (!compMeta.isComponent) {
|
2017-05-23 13:40:50 -07:00
|
|
|
return;
|
2016-11-10 14:07:30 -08:00
|
|
|
}
|
|
|
|
const ngModule = ngModuleByPipeOrDirective.get(dirType);
|
|
|
|
if (!ngModule) {
|
|
|
|
throw new Error(
|
2016-11-23 09:42:19 -08:00
|
|
|
`Internal Error: cannot determine the module for component ${identifierName(compMeta.type)}!`);
|
2016-11-10 14:07:30 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
// compile styles
|
2017-05-16 16:30:37 -07:00
|
|
|
const componentStylesheet = this._styleCompiler.compileComponent(outputCtx, compMeta);
|
|
|
|
// Note: compMeta is a component and therefore template is non null.
|
|
|
|
compMeta.template !.externalStylesheets.forEach((stylesheetMeta) => {
|
2017-09-21 18:05:07 -07:00
|
|
|
// Note: fill non shim and shim style files as they might
|
|
|
|
// be shared by component with and without ViewEncapsulation.
|
|
|
|
const shim = this._styleCompiler.needsStyleShim(compMeta);
|
2017-05-23 13:40:50 -07:00
|
|
|
generatedFiles.push(
|
2017-09-21 18:05:07 -07:00
|
|
|
this._codegenStyles(srcFileUrl, compMeta, stylesheetMeta, shim, fileSuffix));
|
|
|
|
if (this._options.allowEmptyCodegenFiles) {
|
|
|
|
generatedFiles.push(
|
|
|
|
this._codegenStyles(srcFileUrl, compMeta, stylesheetMeta, !shim, fileSuffix));
|
|
|
|
}
|
2016-11-10 14:07:30 -08:00
|
|
|
});
|
|
|
|
|
|
|
|
// compile components
|
2017-02-16 13:55:55 -08:00
|
|
|
const compViewVars = this._compileComponent(
|
2017-05-16 16:30:37 -07:00
|
|
|
outputCtx, compMeta, ngModule, ngModule.transitiveModule.directives, componentStylesheet,
|
|
|
|
fileSuffix);
|
|
|
|
this._compileComponentFactory(outputCtx, compMeta, ngModule, fileSuffix);
|
2016-11-10 14:07:30 -08:00
|
|
|
});
|
2017-09-21 18:05:07 -07:00
|
|
|
if (outputCtx.statements.length > 0 || this._options.allowEmptyCodegenFiles) {
|
2017-05-16 16:30:37 -07:00
|
|
|
const srcModule = this._codegenSourceModule(srcFileUrl, outputCtx);
|
2016-11-29 15:36:33 -08:00
|
|
|
generatedFiles.unshift(srcModule);
|
2016-11-10 14:07:30 -08:00
|
|
|
}
|
2016-11-29 15:36:33 -08:00
|
|
|
return generatedFiles;
|
2016-06-28 09:54:42 -07:00
|
|
|
}
|
|
|
|
|
2016-12-15 09:12:40 -08:00
|
|
|
private _createSummary(
|
2017-08-15 14:41:48 -07:00
|
|
|
srcFileName: string, directives: StaticSymbol[], pipes: StaticSymbol[],
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
ngModules: CompileNgModuleMetadata[], injectables: CompileInjectableMetadata[],
|
2017-05-16 16:30:37 -07:00
|
|
|
ngFactoryCtx: OutputContext): GeneratedFile[] {
|
2017-08-15 14:41:48 -07:00
|
|
|
const symbolSummaries = this._symbolResolver.getSymbolsOf(srcFileName)
|
2016-12-15 09:12:40 -08:00
|
|
|
.map(symbol => this._symbolResolver.resolveSymbol(symbol));
|
2017-04-26 09:24:42 -07:00
|
|
|
const typeData: {
|
|
|
|
summary: CompileTypeSummary,
|
|
|
|
metadata: CompileNgModuleMetadata | CompileDirectiveMetadata | CompilePipeMetadata |
|
|
|
|
CompileTypeMetadata
|
|
|
|
}[] =
|
|
|
|
[
|
2017-09-12 09:40:28 -07:00
|
|
|
...ngModules.map(
|
|
|
|
meta => ({
|
|
|
|
summary: this._metadataResolver.getNgModuleSummary(meta.type.reference) !,
|
|
|
|
metadata: this._metadataResolver.getNgModuleMetadata(meta.type.reference) !
|
|
|
|
})),
|
2017-04-26 09:24:42 -07:00
|
|
|
...directives.map(ref => ({
|
|
|
|
summary: this._metadataResolver.getDirectiveSummary(ref) !,
|
|
|
|
metadata: this._metadataResolver.getDirectiveMetadata(ref) !
|
|
|
|
})),
|
|
|
|
...pipes.map(ref => ({
|
|
|
|
summary: this._metadataResolver.getPipeSummary(ref) !,
|
|
|
|
metadata: this._metadataResolver.getPipeMetadata(ref) !
|
|
|
|
})),
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
...injectables.map(
|
|
|
|
ref => ({
|
|
|
|
summary: this._metadataResolver.getInjectableSummary(ref.symbol) !,
|
|
|
|
metadata: this._metadataResolver.getInjectableSummary(ref.symbol) !.type
|
|
|
|
}))
|
2017-04-26 09:24:42 -07:00
|
|
|
];
|
2017-09-29 14:55:44 -07:00
|
|
|
const forJitOutputCtx = this._options.enableSummariesForJit ?
|
|
|
|
this._createOutputContext(summaryForJitFileName(srcFileName, true)) :
|
|
|
|
null;
|
2017-05-16 16:30:37 -07:00
|
|
|
const {json, exportAs} = serializeSummaries(
|
2017-08-15 14:41:48 -07:00
|
|
|
srcFileName, forJitOutputCtx, this._summaryResolver, this._symbolResolver, symbolSummaries,
|
|
|
|
typeData);
|
2016-12-27 09:36:47 -08:00
|
|
|
exportAs.forEach((entry) => {
|
2017-05-16 16:30:37 -07:00
|
|
|
ngFactoryCtx.statements.push(
|
|
|
|
o.variable(entry.exportAs).set(ngFactoryCtx.importExpr(entry.symbol)).toDeclStmt(null, [
|
|
|
|
o.StmtModifier.Exported
|
|
|
|
]));
|
2016-12-27 09:36:47 -08:00
|
|
|
});
|
2017-08-15 14:41:48 -07:00
|
|
|
const summaryJson = new GeneratedFile(srcFileName, summaryFileName(srcFileName), json);
|
2017-09-29 14:55:44 -07:00
|
|
|
const result = [summaryJson];
|
|
|
|
if (forJitOutputCtx) {
|
|
|
|
result.push(this._codegenSourceModule(srcFileName, forJitOutputCtx));
|
2017-09-22 19:51:03 +02:00
|
|
|
}
|
2017-09-29 14:55:44 -07:00
|
|
|
return result;
|
2016-12-15 09:12:40 -08:00
|
|
|
}
|
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
private _compileModule(outputCtx: OutputContext, ngModule: CompileNgModuleMetadata): void {
|
2016-09-15 15:43:00 -07:00
|
|
|
const providers: CompileProviderMetadata[] = [];
|
|
|
|
|
2017-09-21 18:05:07 -07:00
|
|
|
if (this._options.locale) {
|
|
|
|
const normalizedLocale = this._options.locale.replace(/_/g, '-');
|
2016-11-30 10:52:51 -08:00
|
|
|
providers.push({
|
2018-02-16 08:45:21 -08:00
|
|
|
token: createTokenForExternalReference(this.reflector, Identifiers.LOCALE_ID),
|
2017-08-30 17:33:26 -07:00
|
|
|
useValue: normalizedLocale,
|
2016-11-30 10:52:51 -08:00
|
|
|
});
|
2016-09-15 15:43:00 -07:00
|
|
|
}
|
|
|
|
|
2017-09-21 18:05:07 -07:00
|
|
|
if (this._options.i18nFormat) {
|
2016-11-30 10:52:51 -08:00
|
|
|
providers.push({
|
2018-02-16 08:45:21 -08:00
|
|
|
token: createTokenForExternalReference(this.reflector, Identifiers.TRANSLATIONS_FORMAT),
|
2017-09-21 18:05:07 -07:00
|
|
|
useValue: this._options.i18nFormat
|
2016-11-30 10:52:51 -08:00
|
|
|
});
|
2016-09-15 15:43:00 -07:00
|
|
|
}
|
|
|
|
|
2017-05-16 16:30:37 -07:00
|
|
|
this._ngModuleCompiler.compile(outputCtx, ngModule, providers);
|
2016-06-28 09:54:42 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
private _compileComponentFactory(
|
2017-05-16 16:30:37 -07:00
|
|
|
outputCtx: OutputContext, compMeta: CompileDirectiveMetadata,
|
|
|
|
ngModule: CompileNgModuleMetadata, fileSuffix: string): void {
|
2017-09-12 09:40:28 -07:00
|
|
|
const hostMeta = this._metadataResolver.getHostComponentMetadata(compMeta);
|
2017-02-16 13:55:55 -08:00
|
|
|
const hostViewFactoryVar =
|
2017-05-16 16:30:37 -07:00
|
|
|
this._compileComponent(outputCtx, hostMeta, ngModule, [compMeta.type], null, fileSuffix)
|
2017-02-16 13:55:55 -08:00
|
|
|
.viewClassVar;
|
2016-12-27 09:36:47 -08:00
|
|
|
const compFactoryVar = componentFactoryName(compMeta.type.reference);
|
2017-03-14 14:54:36 -07:00
|
|
|
const inputsExprs: o.LiteralMapEntry[] = [];
|
|
|
|
for (let propName in compMeta.inputs) {
|
|
|
|
const templateName = compMeta.inputs[propName];
|
|
|
|
// Don't quote so that the key gets minified...
|
|
|
|
inputsExprs.push(new o.LiteralMapEntry(propName, o.literal(templateName), false));
|
|
|
|
}
|
|
|
|
const outputsExprs: o.LiteralMapEntry[] = [];
|
|
|
|
for (let propName in compMeta.outputs) {
|
|
|
|
const templateName = compMeta.outputs[propName];
|
|
|
|
// Don't quote so that the key gets minified...
|
|
|
|
outputsExprs.push(new o.LiteralMapEntry(propName, o.literal(templateName), false));
|
|
|
|
}
|
|
|
|
|
2017-05-16 16:30:37 -07:00
|
|
|
outputCtx.statements.push(
|
2017-02-27 23:08:19 -08:00
|
|
|
o.variable(compFactoryVar)
|
2017-05-16 16:30:37 -07:00
|
|
|
.set(o.importExpr(Identifiers.createComponentFactory).callFn([
|
|
|
|
o.literal(compMeta.selector), outputCtx.importExpr(compMeta.type.reference),
|
2017-03-14 14:54:36 -07:00
|
|
|
o.variable(hostViewFactoryVar), new o.LiteralMapExpr(inputsExprs),
|
|
|
|
new o.LiteralMapExpr(outputsExprs),
|
|
|
|
o.literalArr(
|
2017-03-24 09:59:58 -07:00
|
|
|
compMeta.template !.ngContentSelectors.map(selector => o.literal(selector)))
|
2017-02-27 23:08:19 -08:00
|
|
|
]))
|
|
|
|
.toDeclStmt(
|
|
|
|
o.importType(
|
2017-05-16 16:30:37 -07:00
|
|
|
Identifiers.ComponentFactory,
|
|
|
|
[o.expressionType(outputCtx.importExpr(compMeta.type.reference)) !],
|
2017-02-27 23:08:19 -08:00
|
|
|
[o.TypeModifier.Const]),
|
2017-05-16 16:30:37 -07:00
|
|
|
[o.StmtModifier.Final, o.StmtModifier.Exported]));
|
2016-01-06 14:13:44 -08:00
|
|
|
}
|
|
|
|
|
2017-09-11 15:18:19 -07:00
|
|
|
private _compileComponent(
|
|
|
|
outputCtx: OutputContext, compMeta: CompileDirectiveMetadata,
|
|
|
|
ngModule: CompileNgModuleMetadata, directiveIdentifiers: CompileIdentifierMetadata[],
|
|
|
|
componentStyles: CompiledStylesheet|null, fileSuffix: string): ViewCompileResult {
|
|
|
|
const {template: parsedTemplate, pipes: usedPipes} =
|
|
|
|
this._parseTemplate(compMeta, ngModule, directiveIdentifiers);
|
2016-09-15 15:38:17 -07:00
|
|
|
const stylesExpr = componentStyles ? o.variable(componentStyles.stylesVar) : o.literalArr([]);
|
2017-05-16 16:30:37 -07:00
|
|
|
const viewResult = this._viewCompiler.compileComponent(
|
|
|
|
outputCtx, compMeta, parsedTemplate, stylesExpr, usedPipes);
|
2016-06-24 08:46:43 -07:00
|
|
|
if (componentStyles) {
|
2017-05-16 16:30:37 -07:00
|
|
|
_resolveStyleStatements(
|
|
|
|
this._symbolResolver, componentStyles, this._styleCompiler.needsStyleShim(compMeta),
|
|
|
|
fileSuffix);
|
2016-06-24 08:46:43 -07:00
|
|
|
}
|
2017-05-16 16:30:37 -07:00
|
|
|
return viewResult;
|
|
|
|
}
|
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
private _parseTemplate(
|
|
|
|
compMeta: CompileDirectiveMetadata, ngModule: CompileNgModuleMetadata,
|
|
|
|
directiveIdentifiers: CompileIdentifierMetadata[]):
|
|
|
|
{template: TemplateAst[], pipes: CompilePipeSummary[]} {
|
|
|
|
if (this._templateAstCache.has(compMeta.type.reference)) {
|
|
|
|
return this._templateAstCache.get(compMeta.type.reference) !;
|
|
|
|
}
|
|
|
|
const preserveWhitespaces = compMeta !.template !.preserveWhitespaces;
|
|
|
|
const directives =
|
|
|
|
directiveIdentifiers.map(dir => this._metadataResolver.getDirectiveSummary(dir.reference));
|
|
|
|
const pipes = ngModule.transitiveModule.pipes.map(
|
|
|
|
pipe => this._metadataResolver.getPipeSummary(pipe.reference));
|
|
|
|
const result = this._templateParser.parse(
|
|
|
|
compMeta, compMeta.template !.htmlAst !, directives, pipes, ngModule.schemas,
|
|
|
|
templateSourceUrl(ngModule.type, compMeta, compMeta.template !), preserveWhitespaces);
|
|
|
|
this._templateAstCache.set(compMeta.type.reference, result);
|
|
|
|
return result;
|
2017-09-11 15:18:19 -07:00
|
|
|
}
|
|
|
|
|
2017-05-16 16:30:37 -07:00
|
|
|
private _createOutputContext(genFilePath: string): OutputContext {
|
2017-12-11 08:50:46 -08:00
|
|
|
const importExpr =
|
|
|
|
(symbol: StaticSymbol, typeParams: o.Type[] | null = null,
|
|
|
|
useSummaries: boolean = true) => {
|
|
|
|
if (!(symbol instanceof StaticSymbol)) {
|
|
|
|
throw new Error(`Internal error: unknown identifier ${JSON.stringify(symbol)}`);
|
|
|
|
}
|
|
|
|
const arity = this._symbolResolver.getTypeArity(symbol) || 0;
|
|
|
|
const {filePath, name, members} =
|
|
|
|
this._symbolResolver.getImportAs(symbol, useSummaries) || symbol;
|
|
|
|
const importModule = this._fileNameToModuleName(filePath, genFilePath);
|
|
|
|
|
|
|
|
// It should be good enough to compare filePath to genFilePath and if they are equal
|
|
|
|
// there is a self reference. However, ngfactory files generate to .ts but their
|
|
|
|
// symbols have .d.ts so a simple compare is insufficient. They should be canonical
|
|
|
|
// and is tracked by #17705.
|
|
|
|
const selfReference = this._fileNameToModuleName(genFilePath, genFilePath);
|
|
|
|
const moduleName = importModule === selfReference ? null : importModule;
|
|
|
|
|
|
|
|
// If we are in a type expression that refers to a generic type then supply
|
|
|
|
// the required type parameters. If there were not enough type parameters
|
|
|
|
// supplied, supply any as the type. Outside a type expression the reference
|
|
|
|
// should not supply type parameters and be treated as a simple value reference
|
|
|
|
// to the constructor function itself.
|
|
|
|
const suppliedTypeParams = typeParams || [];
|
|
|
|
const missingTypeParamsCount = arity - suppliedTypeParams.length;
|
|
|
|
const allTypeParams =
|
|
|
|
suppliedTypeParams.concat(new Array(missingTypeParamsCount).fill(o.DYNAMIC_TYPE));
|
|
|
|
return members.reduce(
|
|
|
|
(expr, memberName) => expr.prop(memberName),
|
|
|
|
<o.Expression>o.importExpr(
|
|
|
|
new o.ExternalReference(moduleName, name, null), allTypeParams));
|
|
|
|
};
|
2017-05-16 16:30:37 -07:00
|
|
|
|
2017-11-20 10:21:17 -08:00
|
|
|
return {statements: [], genFilePath, importExpr, constantPool: new ConstantPool()};
|
2016-01-06 14:13:44 -08:00
|
|
|
}
|
|
|
|
|
2017-10-20 09:46:41 -07:00
|
|
|
private _fileNameToModuleName(importedFilePath: string, containingFilePath: string): string {
|
|
|
|
return this._summaryResolver.getKnownModuleName(importedFilePath) ||
|
|
|
|
this._symbolResolver.getKnownModuleName(importedFilePath) ||
|
|
|
|
this._host.fileNameToModuleName(importedFilePath, containingFilePath);
|
|
|
|
}
|
|
|
|
|
2017-05-16 16:30:37 -07:00
|
|
|
private _codegenStyles(
|
|
|
|
srcFileUrl: string, compMeta: CompileDirectiveMetadata,
|
2017-09-21 18:05:07 -07:00
|
|
|
stylesheetMetadata: CompileStylesheetMetadata, isShimmed: boolean,
|
|
|
|
fileSuffix: string): GeneratedFile {
|
|
|
|
const outputCtx = this._createOutputContext(
|
|
|
|
_stylesModuleUrl(stylesheetMetadata.moduleUrl !, isShimmed, fileSuffix));
|
2017-05-16 16:30:37 -07:00
|
|
|
const compiledStylesheet =
|
2017-09-21 18:05:07 -07:00
|
|
|
this._styleCompiler.compileStyles(outputCtx, compMeta, stylesheetMetadata, isShimmed);
|
|
|
|
_resolveStyleStatements(this._symbolResolver, compiledStylesheet, isShimmed, fileSuffix);
|
2017-05-16 16:30:37 -07:00
|
|
|
return this._codegenSourceModule(srcFileUrl, outputCtx);
|
2016-05-02 09:38:46 -07:00
|
|
|
}
|
2016-01-06 14:13:44 -08:00
|
|
|
|
2017-05-16 16:30:37 -07:00
|
|
|
private _codegenSourceModule(srcFileUrl: string, ctx: OutputContext): GeneratedFile {
|
2017-05-17 11:21:08 -07:00
|
|
|
return new GeneratedFile(srcFileUrl, ctx.genFilePath, ctx.statements);
|
2016-01-06 14:13:44 -08:00
|
|
|
}
|
2017-10-20 09:46:41 -07:00
|
|
|
|
|
|
|
listLazyRoutes(entryRoute?: string, analyzedModules?: NgAnalyzedModules): LazyRoute[] {
|
|
|
|
const self = this;
|
|
|
|
if (entryRoute) {
|
2018-02-16 08:45:21 -08:00
|
|
|
const symbol = parseLazyRoute(entryRoute, this.reflector).referencedModule;
|
2017-10-20 09:46:41 -07:00
|
|
|
return visitLazyRoute(symbol);
|
|
|
|
} else if (analyzedModules) {
|
|
|
|
const allLazyRoutes: LazyRoute[] = [];
|
|
|
|
for (const ngModule of analyzedModules.ngModules) {
|
2018-02-16 08:45:21 -08:00
|
|
|
const lazyRoutes = listLazyRoutes(ngModule, this.reflector);
|
2017-10-20 09:46:41 -07:00
|
|
|
for (const lazyRoute of lazyRoutes) {
|
|
|
|
allLazyRoutes.push(lazyRoute);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return allLazyRoutes;
|
|
|
|
} else {
|
|
|
|
throw new Error(`Either route or analyzedModules has to be specified!`);
|
|
|
|
}
|
|
|
|
|
|
|
|
function visitLazyRoute(
|
|
|
|
symbol: StaticSymbol, seenRoutes = new Set<StaticSymbol>(),
|
|
|
|
allLazyRoutes: LazyRoute[] = []): LazyRoute[] {
|
|
|
|
// Support pointing to default exports, but stop recursing there,
|
|
|
|
// as the StaticReflector does not yet support default exports.
|
|
|
|
if (seenRoutes.has(symbol) || !symbol.name) {
|
|
|
|
return allLazyRoutes;
|
|
|
|
}
|
|
|
|
seenRoutes.add(symbol);
|
|
|
|
const lazyRoutes = listLazyRoutes(
|
2018-02-16 08:45:21 -08:00
|
|
|
self._metadataResolver.getNgModuleMetadata(symbol, true) !, self.reflector);
|
2017-10-20 09:46:41 -07:00
|
|
|
for (const lazyRoute of lazyRoutes) {
|
|
|
|
allLazyRoutes.push(lazyRoute);
|
|
|
|
visitLazyRoute(lazyRoute.referencedModule, seenRoutes, allLazyRoutes);
|
|
|
|
}
|
|
|
|
return allLazyRoutes;
|
|
|
|
}
|
|
|
|
}
|
2016-01-06 14:13:44 -08:00
|
|
|
}
|
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
function _createEmptyStub(outputCtx: OutputContext) {
|
2017-08-22 16:17:44 -07:00
|
|
|
// Note: We need to produce at least one import statement so that
|
|
|
|
// TypeScript knows that the file is an es6 module. Otherwise our generated
|
|
|
|
// exports / imports won't be emitted properly by TypeScript.
|
|
|
|
outputCtx.statements.push(o.importExpr(Identifiers.ComponentFactory).toStmt());
|
2017-06-09 14:50:57 -07:00
|
|
|
}
|
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
|
2016-06-08 16:38:52 -07:00
|
|
|
function _resolveStyleStatements(
|
2017-05-16 16:30:37 -07:00
|
|
|
symbolResolver: StaticSymbolResolver, compileResult: CompiledStylesheet, needsShim: boolean,
|
|
|
|
fileSuffix: string): void {
|
2016-01-06 14:13:44 -08:00
|
|
|
compileResult.dependencies.forEach((dep) => {
|
2017-05-16 16:30:37 -07:00
|
|
|
dep.setValue(symbolResolver.getStaticSymbol(
|
|
|
|
_stylesModuleUrl(dep.moduleUrl, needsShim, fileSuffix), dep.name));
|
2016-01-06 14:13:44 -08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2016-05-02 09:38:46 -07:00
|
|
|
function _stylesModuleUrl(stylesheetUrl: string, shim: boolean, suffix: string): string {
|
2016-12-13 17:34:46 -08:00
|
|
|
return `${stylesheetUrl}${shim ? '.shim' : ''}.ngstyle${suffix}`;
|
2016-01-06 14:13:44 -08:00
|
|
|
}
|
|
|
|
|
2016-11-15 13:57:25 -08:00
|
|
|
export interface NgAnalyzedModules {
|
|
|
|
ngModules: CompileNgModuleMetadata[];
|
|
|
|
ngModuleByPipeOrDirective: Map<StaticSymbol, CompileNgModuleMetadata>;
|
2017-09-12 09:40:28 -07:00
|
|
|
files: NgAnalyzedFile[];
|
2016-11-15 13:57:25 -08:00
|
|
|
symbolsMissingModule?: StaticSymbol[];
|
|
|
|
}
|
|
|
|
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
export interface NgAnalyzedFileWithInjectables {
|
|
|
|
fileName: string;
|
|
|
|
injectables: CompileInjectableMetadata[];
|
2018-02-16 08:45:21 -08:00
|
|
|
shallowModules: CompileShallowModuleMetadata[];
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
}
|
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
export interface NgAnalyzedFile {
|
|
|
|
fileName: string;
|
|
|
|
directives: StaticSymbol[];
|
|
|
|
pipes: StaticSymbol[];
|
|
|
|
ngModules: CompileNgModuleMetadata[];
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
injectables: CompileInjectableMetadata[];
|
2017-09-12 09:40:28 -07:00
|
|
|
exportsNonSourceFiles: boolean;
|
|
|
|
}
|
|
|
|
|
2016-12-15 09:12:40 -08:00
|
|
|
export interface NgAnalyzeModulesHost { isSourceFile(filePath: string): boolean; }
|
|
|
|
|
2016-11-15 13:57:25 -08:00
|
|
|
export function analyzeNgModules(
|
2017-09-12 09:40:28 -07:00
|
|
|
fileNames: string[], host: NgAnalyzeModulesHost, staticSymbolResolver: StaticSymbolResolver,
|
2016-11-15 13:57:25 -08:00
|
|
|
metadataResolver: CompileMetadataResolver): NgAnalyzedModules {
|
2017-09-12 09:40:28 -07:00
|
|
|
const files = _analyzeFilesIncludingNonProgramFiles(
|
|
|
|
fileNames, host, staticSymbolResolver, metadataResolver);
|
|
|
|
return mergeAnalyzedFiles(files);
|
2016-11-15 13:57:25 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
export function analyzeAndValidateNgModules(
|
2017-09-12 09:40:28 -07:00
|
|
|
fileNames: string[], host: NgAnalyzeModulesHost, staticSymbolResolver: StaticSymbolResolver,
|
2016-11-15 13:57:25 -08:00
|
|
|
metadataResolver: CompileMetadataResolver): NgAnalyzedModules {
|
2017-09-12 09:40:28 -07:00
|
|
|
return validateAnalyzedModules(
|
|
|
|
analyzeNgModules(fileNames, host, staticSymbolResolver, metadataResolver));
|
|
|
|
}
|
|
|
|
|
|
|
|
function validateAnalyzedModules(analyzedModules: NgAnalyzedModules): NgAnalyzedModules {
|
|
|
|
if (analyzedModules.symbolsMissingModule && analyzedModules.symbolsMissingModule.length) {
|
|
|
|
const messages = analyzedModules.symbolsMissingModule.map(
|
2017-02-06 15:26:30 -08:00
|
|
|
s =>
|
|
|
|
`Cannot determine the module for class ${s.name} in ${s.filePath}! Add ${s.name} to the NgModule to fix it.`);
|
|
|
|
throw syntaxError(messages.join('\n'));
|
2016-11-15 13:57:25 -08:00
|
|
|
}
|
2017-09-12 09:40:28 -07:00
|
|
|
return analyzedModules;
|
2016-11-15 13:57:25 -08:00
|
|
|
}
|
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
// Analyzes all of the program files,
|
|
|
|
// including files that are not part of the program
|
|
|
|
// but are referenced by an NgModule.
|
|
|
|
function _analyzeFilesIncludingNonProgramFiles(
|
|
|
|
fileNames: string[], host: NgAnalyzeModulesHost, staticSymbolResolver: StaticSymbolResolver,
|
|
|
|
metadataResolver: CompileMetadataResolver): NgAnalyzedFile[] {
|
|
|
|
const seenFiles = new Set<string>();
|
|
|
|
const files: NgAnalyzedFile[] = [];
|
|
|
|
|
|
|
|
const visitFile = (fileName: string) => {
|
|
|
|
if (seenFiles.has(fileName) || !host.isSourceFile(fileName)) {
|
|
|
|
return false;
|
2016-12-15 09:12:40 -08:00
|
|
|
}
|
2017-09-12 09:40:28 -07:00
|
|
|
seenFiles.add(fileName);
|
|
|
|
const analyzedFile = analyzeFile(host, staticSymbolResolver, metadataResolver, fileName);
|
|
|
|
files.push(analyzedFile);
|
|
|
|
analyzedFile.ngModules.forEach(ngModule => {
|
|
|
|
ngModule.transitiveModule.modules.forEach(modMeta => visitFile(modMeta.reference.filePath));
|
2016-11-15 13:57:25 -08:00
|
|
|
});
|
|
|
|
};
|
2017-09-12 09:40:28 -07:00
|
|
|
fileNames.forEach((fileName) => visitFile(fileName));
|
|
|
|
return files;
|
2016-11-15 13:57:25 -08:00
|
|
|
}
|
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
export function analyzeFile(
|
|
|
|
host: NgAnalyzeModulesHost, staticSymbolResolver: StaticSymbolResolver,
|
|
|
|
metadataResolver: CompileMetadataResolver, fileName: string): NgAnalyzedFile {
|
|
|
|
const directives: StaticSymbol[] = [];
|
|
|
|
const pipes: StaticSymbol[] = [];
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
const injectables: CompileInjectableMetadata[] = [];
|
2017-09-12 09:40:28 -07:00
|
|
|
const ngModules: CompileNgModuleMetadata[] = [];
|
|
|
|
const hasDecorators = staticSymbolResolver.hasDecorators(fileName);
|
|
|
|
let exportsNonSourceFiles = false;
|
|
|
|
// Don't analyze .d.ts files that have no decorators as a shortcut
|
|
|
|
// to speed up the analysis. This prevents us from
|
|
|
|
// resolving the references in these files.
|
|
|
|
// Note: exportsNonSourceFiles is only needed when compiling with summaries,
|
|
|
|
// which is not the case when .d.ts files are treated as input files.
|
|
|
|
if (!fileName.endsWith('.d.ts') || hasDecorators) {
|
|
|
|
staticSymbolResolver.getSymbolsOf(fileName).forEach((symbol) => {
|
2016-12-15 09:12:40 -08:00
|
|
|
const resolvedSymbol = staticSymbolResolver.resolveSymbol(symbol);
|
|
|
|
const symbolMeta = resolvedSymbol.metadata;
|
2017-09-12 09:40:28 -07:00
|
|
|
if (!symbolMeta || symbolMeta.__symbolic === 'error') {
|
|
|
|
return;
|
|
|
|
}
|
2017-09-20 16:31:02 -07:00
|
|
|
let isNgSymbol = false;
|
2017-09-12 09:40:28 -07:00
|
|
|
if (symbolMeta.__symbolic === 'class') {
|
|
|
|
if (metadataResolver.isDirective(symbol)) {
|
2017-09-20 16:31:02 -07:00
|
|
|
isNgSymbol = true;
|
2017-09-12 09:40:28 -07:00
|
|
|
directives.push(symbol);
|
|
|
|
} else if (metadataResolver.isPipe(symbol)) {
|
2017-09-20 16:31:02 -07:00
|
|
|
isNgSymbol = true;
|
2017-09-12 09:40:28 -07:00
|
|
|
pipes.push(symbol);
|
2017-11-09 16:52:19 -08:00
|
|
|
} else if (metadataResolver.isNgModule(symbol)) {
|
2017-09-12 09:40:28 -07:00
|
|
|
const ngModule = metadataResolver.getNgModuleMetadata(symbol, false);
|
|
|
|
if (ngModule) {
|
2017-09-20 16:31:02 -07:00
|
|
|
isNgSymbol = true;
|
2017-09-12 09:40:28 -07:00
|
|
|
ngModules.push(ngModule);
|
|
|
|
}
|
2017-11-09 16:52:19 -08:00
|
|
|
} else if (metadataResolver.isInjectable(symbol)) {
|
|
|
|
isNgSymbol = true;
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
const injectable = metadataResolver.getInjectableMetadata(symbol, null, false);
|
|
|
|
if (injectable) {
|
|
|
|
injectables.push(injectable);
|
|
|
|
}
|
2016-12-15 09:12:40 -08:00
|
|
|
}
|
2016-11-17 20:11:55 -08:00
|
|
|
}
|
2017-09-20 16:31:02 -07:00
|
|
|
if (!isNgSymbol) {
|
|
|
|
exportsNonSourceFiles =
|
|
|
|
exportsNonSourceFiles || isValueExportingNonSourceFile(host, symbolMeta);
|
|
|
|
}
|
2016-12-15 09:12:40 -08:00
|
|
|
});
|
2017-09-12 09:40:28 -07:00
|
|
|
}
|
|
|
|
return {
|
|
|
|
fileName, directives, pipes, ngModules, injectables, exportsNonSourceFiles,
|
|
|
|
};
|
2016-11-15 13:57:25 -08:00
|
|
|
}
|
|
|
|
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
export function analyzeFileForInjectables(
|
|
|
|
host: NgAnalyzeModulesHost, staticSymbolResolver: StaticSymbolResolver,
|
|
|
|
metadataResolver: CompileMetadataResolver, fileName: string): NgAnalyzedFileWithInjectables {
|
|
|
|
const injectables: CompileInjectableMetadata[] = [];
|
2018-02-16 08:45:21 -08:00
|
|
|
const shallowModules: CompileShallowModuleMetadata[] = [];
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
if (staticSymbolResolver.hasDecorators(fileName)) {
|
|
|
|
staticSymbolResolver.getSymbolsOf(fileName).forEach((symbol) => {
|
|
|
|
const resolvedSymbol = staticSymbolResolver.resolveSymbol(symbol);
|
|
|
|
const symbolMeta = resolvedSymbol.metadata;
|
|
|
|
if (!symbolMeta || symbolMeta.__symbolic === 'error') {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
let isNgSymbol = false;
|
|
|
|
if (symbolMeta.__symbolic === 'class') {
|
|
|
|
if (metadataResolver.isInjectable(symbol)) {
|
|
|
|
isNgSymbol = true;
|
|
|
|
const injectable = metadataResolver.getInjectableMetadata(symbol, null, false);
|
|
|
|
if (injectable) {
|
|
|
|
injectables.push(injectable);
|
|
|
|
}
|
2018-02-16 08:45:21 -08:00
|
|
|
} else if (metadataResolver.isNgModule(symbol)) {
|
|
|
|
isNgSymbol = true;
|
|
|
|
const module = metadataResolver.getShallowModuleMetadata(symbol);
|
|
|
|
if (module) {
|
|
|
|
shallowModules.push(module);
|
|
|
|
}
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2018-02-16 08:45:21 -08:00
|
|
|
return {fileName, injectables, shallowModules};
|
feat: change @Injectable() to support tree-shakeable tokens (#22005)
This commit bundles 3 important changes, with the goal of enabling tree-shaking
of services which are never injected. Ordinarily, this tree-shaking is prevented
by the existence of a hard dependency on the service by the module in which it
is declared.
Firstly, @Injectable() is modified to accept a 'scope' parameter, which points
to an @NgModule(). This reverses the dependency edge, permitting the module to
not depend on the service which it "provides".
Secondly, the runtime is modified to understand the new relationship created
above. When a module receives a request to inject a token, and cannot find that
token in its list of providers, it will then look at the token for a special
ngInjectableDef field which indicates which module the token is scoped to. If
that module happens to be in the injector, it will behave as if the token
itself was in the injector to begin with.
Thirdly, the compiler is modified to read the @Injectable() metadata and to
generate the special ngInjectableDef field as part of TS compilation, using the
PartialModules system.
Additionally, this commit adds several unit and integration tests of various
flavors to test this change.
PR Close #22005
2018-02-02 10:33:48 -08:00
|
|
|
}
|
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
function isValueExportingNonSourceFile(host: NgAnalyzeModulesHost, metadata: any): boolean {
|
|
|
|
let exportsNonSourceFiles = false;
|
|
|
|
|
|
|
|
class Visitor implements ValueVisitor {
|
|
|
|
visitArray(arr: any[], context: any): any { arr.forEach(v => visitValue(v, this, context)); }
|
|
|
|
visitStringMap(map: {[key: string]: any}, context: any): any {
|
|
|
|
Object.keys(map).forEach((key) => visitValue(map[key], this, context));
|
2016-11-10 14:07:30 -08:00
|
|
|
}
|
2017-09-12 09:40:28 -07:00
|
|
|
visitPrimitive(value: any, context: any): any {}
|
|
|
|
visitOther(value: any, context: any): any {
|
|
|
|
if (value instanceof StaticSymbol && !host.isSourceFile(value.filePath)) {
|
|
|
|
exportsNonSourceFiles = true;
|
|
|
|
}
|
2016-10-25 16:28:22 -07:00
|
|
|
}
|
2017-09-12 09:40:28 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
visitValue(metadata, new Visitor(), null);
|
|
|
|
return exportsNonSourceFiles;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function mergeAnalyzedFiles(analyzedFiles: NgAnalyzedFile[]): NgAnalyzedModules {
|
|
|
|
const allNgModules: CompileNgModuleMetadata[] = [];
|
|
|
|
const ngModuleByPipeOrDirective = new Map<StaticSymbol, CompileNgModuleMetadata>();
|
|
|
|
const allPipesAndDirectives = new Set<StaticSymbol>();
|
|
|
|
|
|
|
|
analyzedFiles.forEach(af => {
|
|
|
|
af.ngModules.forEach(ngModule => {
|
|
|
|
allNgModules.push(ngModule);
|
|
|
|
ngModule.declaredDirectives.forEach(
|
|
|
|
d => ngModuleByPipeOrDirective.set(d.reference, ngModule));
|
|
|
|
ngModule.declaredPipes.forEach(p => ngModuleByPipeOrDirective.set(p.reference, ngModule));
|
|
|
|
});
|
|
|
|
af.directives.forEach(d => allPipesAndDirectives.add(d));
|
|
|
|
af.pipes.forEach(p => allPipesAndDirectives.add(p));
|
2016-10-25 16:28:22 -07:00
|
|
|
});
|
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
const symbolsMissingModule: StaticSymbol[] = [];
|
|
|
|
allPipesAndDirectives.forEach(ref => {
|
|
|
|
if (!ngModuleByPipeOrDirective.has(ref)) {
|
|
|
|
symbolsMissingModule.push(ref);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return {
|
|
|
|
ngModules: allNgModules,
|
|
|
|
ngModuleByPipeOrDirective,
|
|
|
|
symbolsMissingModule,
|
|
|
|
files: analyzedFiles
|
|
|
|
};
|
|
|
|
}
|
2016-11-10 14:07:30 -08:00
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
function mergeAndValidateNgFiles(files: NgAnalyzedFile[]): NgAnalyzedModules {
|
|
|
|
return validateAnalyzedModules(mergeAnalyzedFiles(files));
|
2016-10-25 16:28:22 -07:00
|
|
|
}
|