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
|
|
|
|
*/
|
|
|
|
|
2016-08-25 00:50:16 -07:00
|
|
|
import {SchemaMetadata} from '@angular/core';
|
2016-08-10 15:55:18 -07:00
|
|
|
|
2016-11-14 17:37:47 -08:00
|
|
|
import {AnimationCompiler} from '../animation/animation_compiler';
|
|
|
|
import {AnimationParser} from '../animation/animation_parser';
|
2016-11-29 15:36:33 -08:00
|
|
|
import {CompileDirectiveMetadata, CompileIdentifierMetadata, CompileNgModuleMetadata, CompilePipeMetadata, CompileProviderMetadata, CompileTypeSummary, createHostComponentMeta, identifierModuleUrl, identifierName} from '../compile_metadata';
|
2016-11-14 17:37:47 -08:00
|
|
|
import {DirectiveNormalizer} from '../directive_normalizer';
|
|
|
|
import {DirectiveWrapperCompileResult, DirectiveWrapperCompiler} from '../directive_wrapper_compiler';
|
|
|
|
import {ListWrapper} from '../facade/collection';
|
2016-11-23 09:42:19 -08:00
|
|
|
import {Identifiers, createIdentifier, createIdentifierToken} from '../identifiers';
|
2016-11-14 17:37:47 -08:00
|
|
|
import {CompileMetadataResolver} from '../metadata_resolver';
|
|
|
|
import {NgModuleCompiler} from '../ng_module_compiler';
|
|
|
|
import {OutputEmitter} from '../output/abstract_emitter';
|
|
|
|
import * as o from '../output/output_ast';
|
|
|
|
import {CompiledStylesheet, StyleCompiler} from '../style_compiler';
|
|
|
|
import {TemplateParser} from '../template_parser/template_parser';
|
|
|
|
import {ComponentFactoryDependency, DirectiveWrapperDependency, ViewClassDependency, ViewCompileResult, ViewCompiler} from '../view_compiler/view_compiler';
|
|
|
|
|
2016-11-15 13:57:25 -08:00
|
|
|
import {AotCompilerOptions} from './compiler_options';
|
2016-11-29 15:36:33 -08:00
|
|
|
import {GeneratedFile} from './generated_file';
|
2016-11-15 13:57:25 -08:00
|
|
|
import {StaticReflector} from './static_reflector';
|
2016-11-14 17:37:47 -08:00
|
|
|
import {StaticSymbol} from './static_symbol';
|
2016-11-29 15:36:33 -08:00
|
|
|
import {AotSummaryResolver} from './summary_resolver';
|
|
|
|
import {filterFileByPatterns} from './utils';
|
2016-01-06 14:13:44 -08:00
|
|
|
|
2016-11-14 17:37:47 -08:00
|
|
|
export class AotCompiler {
|
2016-09-23 16:37:04 -04:00
|
|
|
private _animationCompiler = new AnimationCompiler();
|
|
|
|
|
2016-06-08 16:38:52 -07:00
|
|
|
constructor(
|
2016-11-10 14:07:30 -08:00
|
|
|
private _metadataResolver: CompileMetadataResolver, private _templateParser: TemplateParser,
|
2016-06-08 16:38:52 -07:00
|
|
|
private _styleCompiler: StyleCompiler, private _viewCompiler: ViewCompiler,
|
2016-10-13 16:34:37 -07:00
|
|
|
private _dirWrapperCompiler: DirectiveWrapperCompiler,
|
2016-08-12 14:45:36 -07:00
|
|
|
private _ngModuleCompiler: NgModuleCompiler, private _outputEmitter: OutputEmitter,
|
2016-11-29 15:36:33 -08:00
|
|
|
private _summaryResolver: AotSummaryResolver, private _localeId: string,
|
|
|
|
private _translationFormat: string, private _animationParser: AnimationParser,
|
|
|
|
private _staticReflector: StaticReflector, private _options: AotCompilerOptions) {}
|
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
|
|
|
|
2016-11-29 15:36:33 -08:00
|
|
|
compileAll(rootFiles: string[]): Promise<GeneratedFile[]> {
|
2016-11-17 20:11:55 -08:00
|
|
|
const programSymbols = extractProgramSymbols(this._staticReflector, rootFiles, this._options);
|
2016-11-16 10:22:11 -08:00
|
|
|
const {ngModuleByPipeOrDirective, files, ngModules} =
|
2016-11-17 20:11:55 -08:00
|
|
|
analyzeAndValidateNgModules(programSymbols, this._options, this._metadataResolver);
|
2016-11-29 08:08:22 -08:00
|
|
|
return Promise
|
|
|
|
.all(ngModules.map(
|
|
|
|
ngModule => this._metadataResolver.loadNgModuleDirectiveAndPipeMetadata(
|
|
|
|
ngModule.type.reference, false)))
|
|
|
|
.then(() => {
|
|
|
|
const sourceModules = files.map(
|
|
|
|
file => this._compileSrcFile(
|
2016-11-29 15:36:33 -08:00
|
|
|
file.srcUrl, ngModuleByPipeOrDirective, file.directives, file.pipes,
|
|
|
|
file.ngModules));
|
2016-11-29 08:08:22 -08:00
|
|
|
return ListWrapper.flatten(sourceModules);
|
|
|
|
});
|
2016-10-24 13:28:23 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
private _compileSrcFile(
|
2016-10-25 16:28:22 -07:00
|
|
|
srcFileUrl: string, ngModuleByPipeOrDirective: Map<StaticSymbol, CompileNgModuleMetadata>,
|
2016-11-29 15:36:33 -08:00
|
|
|
directives: StaticSymbol[], pipes: StaticSymbol[],
|
|
|
|
ngModules: StaticSymbol[]): GeneratedFile[] {
|
2016-10-24 13:28:23 -07:00
|
|
|
const fileSuffix = _splitTypescriptSuffix(srcFileUrl)[1];
|
2016-09-15 15:38:17 -07:00
|
|
|
const statements: o.Statement[] = [];
|
|
|
|
const exportedVars: string[] = [];
|
2016-11-29 15:36:33 -08:00
|
|
|
const generatedFiles: GeneratedFile[] = [];
|
|
|
|
|
|
|
|
// write summary files
|
|
|
|
const summaries: CompileTypeSummary[] = [
|
2016-12-02 10:08:46 -08:00
|
|
|
...ngModules.map(ref => this._metadataResolver.getNgModuleSummary(ref)),
|
2016-11-29 15:36:33 -08:00
|
|
|
...directives.map(ref => this._metadataResolver.getDirectiveSummary(ref)),
|
2016-12-02 10:08:46 -08:00
|
|
|
...pipes.map(ref => this._metadataResolver.getPipeSummary(ref))
|
2016-11-29 15:36:33 -08:00
|
|
|
];
|
|
|
|
generatedFiles.push(this._summaryResolver.serializeSummaries(srcFileUrl, summaries));
|
2016-06-28 09:54:42 -07:00
|
|
|
|
2016-07-18 03:50:31 -07:00
|
|
|
// compile all ng modules
|
2016-06-28 09:54:42 -07:00
|
|
|
exportedVars.push(
|
2016-07-18 03:50:31 -07:00
|
|
|
...ngModules.map((ngModuleType) => this._compileModule(ngModuleType, statements)));
|
2016-06-28 09:54:42 -07:00
|
|
|
|
2016-10-13 16:34:37 -07:00
|
|
|
// compile directive wrappers
|
|
|
|
exportedVars.push(...directives.map(
|
|
|
|
(directiveType) => this._compileDirectiveWrapper(directiveType, statements)));
|
|
|
|
|
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) {
|
|
|
|
return Promise.resolve(null);
|
|
|
|
}
|
|
|
|
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
|
|
|
}
|
|
|
|
|
|
|
|
_assertComponent(compMeta);
|
|
|
|
|
|
|
|
// compile styles
|
|
|
|
const stylesCompileResults = this._styleCompiler.compileComponent(compMeta);
|
|
|
|
stylesCompileResults.externalStylesheets.forEach((compiledStyleSheet) => {
|
2016-11-29 15:36:33 -08:00
|
|
|
generatedFiles.push(this._codgenStyles(srcFileUrl, compiledStyleSheet, fileSuffix));
|
2016-11-10 14:07:30 -08:00
|
|
|
});
|
|
|
|
|
|
|
|
// compile components
|
|
|
|
exportedVars.push(
|
|
|
|
this._compileComponentFactory(compMeta, ngModule, fileSuffix, statements),
|
|
|
|
this._compileComponent(
|
|
|
|
compMeta, ngModule, ngModule.transitiveModule.directives,
|
|
|
|
stylesCompileResults.componentStylesheet, fileSuffix, statements));
|
|
|
|
});
|
|
|
|
if (statements.length > 0) {
|
|
|
|
const srcModule = this._codegenSourceModule(
|
|
|
|
srcFileUrl, _ngfactoryModuleUrl(srcFileUrl), statements, exportedVars);
|
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-07-18 03:50:31 -07:00
|
|
|
private _compileModule(ngModuleType: StaticSymbol, targetStatements: o.Statement[]): string {
|
2016-09-15 15:38:17 -07:00
|
|
|
const ngModule = this._metadataResolver.getNgModuleMetadata(ngModuleType);
|
2016-09-15 15:43:00 -07:00
|
|
|
const providers: CompileProviderMetadata[] = [];
|
|
|
|
|
|
|
|
if (this._localeId) {
|
2016-11-30 10:52:51 -08:00
|
|
|
providers.push({
|
2016-11-23 09:42:19 -08:00
|
|
|
token: createIdentifierToken(Identifiers.LOCALE_ID),
|
2016-09-15 15:43:00 -07:00
|
|
|
useValue: this._localeId,
|
2016-11-30 10:52:51 -08:00
|
|
|
});
|
2016-09-15 15:43:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (this._translationFormat) {
|
2016-11-30 10:52:51 -08:00
|
|
|
providers.push({
|
2016-11-23 09:42:19 -08:00
|
|
|
token: createIdentifierToken(Identifiers.TRANSLATIONS_FORMAT),
|
2016-08-12 14:45:36 -07:00
|
|
|
useValue: this._translationFormat
|
2016-11-30 10:52:51 -08:00
|
|
|
});
|
2016-09-15 15:43:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
const appCompileResult = this._ngModuleCompiler.compile(ngModule, providers);
|
|
|
|
|
2016-06-28 09:54:42 -07:00
|
|
|
appCompileResult.dependencies.forEach((dep) => {
|
2016-11-23 09:42:19 -08:00
|
|
|
dep.placeholder.reference = this._staticReflector.getStaticSymbol(
|
|
|
|
_ngfactoryModuleUrl(identifierModuleUrl(dep.comp)), _componentFactoryName(dep.comp));
|
2016-06-28 09:54:42 -07:00
|
|
|
});
|
2016-09-15 15:43:00 -07:00
|
|
|
|
2016-06-28 09:54:42 -07:00
|
|
|
targetStatements.push(...appCompileResult.statements);
|
2016-07-18 03:50:31 -07:00
|
|
|
return appCompileResult.ngModuleFactoryVar;
|
2016-06-28 09:54:42 -07:00
|
|
|
}
|
|
|
|
|
2016-10-13 16:34:37 -07:00
|
|
|
private _compileDirectiveWrapper(directiveType: StaticSymbol, targetStatements: o.Statement[]):
|
|
|
|
string {
|
|
|
|
const dirMeta = this._metadataResolver.getDirectiveMetadata(directiveType);
|
|
|
|
const dirCompileResult = this._dirWrapperCompiler.compile(dirMeta);
|
|
|
|
|
|
|
|
targetStatements.push(...dirCompileResult.statements);
|
|
|
|
return dirCompileResult.dirWrapperClassVar;
|
|
|
|
}
|
|
|
|
|
2016-06-28 09:54:42 -07:00
|
|
|
private _compileComponentFactory(
|
2016-11-10 14:07:30 -08:00
|
|
|
compMeta: CompileDirectiveMetadata, ngModule: CompileNgModuleMetadata, fileSuffix: string,
|
2016-06-28 09:54:42 -07:00
|
|
|
targetStatements: o.Statement[]): string {
|
2016-11-23 09:42:19 -08:00
|
|
|
const hostMeta = createHostComponentMeta(
|
|
|
|
this._staticReflector.getStaticSymbol(
|
|
|
|
identifierModuleUrl(compMeta.type), `${identifierName(compMeta.type)}_Host`),
|
|
|
|
compMeta);
|
2016-11-10 14:07:30 -08:00
|
|
|
const hostViewFactoryVar = this._compileComponent(
|
|
|
|
hostMeta, ngModule, [compMeta.type], null, fileSuffix, targetStatements);
|
2016-09-15 15:38:17 -07:00
|
|
|
const compFactoryVar = _componentFactoryName(compMeta.type);
|
2016-06-28 09:54:42 -07:00
|
|
|
targetStatements.push(
|
|
|
|
o.variable(compFactoryVar)
|
2016-11-23 09:42:19 -08:00
|
|
|
.set(o.importExpr(
|
|
|
|
createIdentifier(Identifiers.ComponentFactory), [o.importType(compMeta.type)])
|
2016-06-28 09:54:42 -07:00
|
|
|
.instantiate(
|
|
|
|
[
|
2016-09-15 15:38:17 -07:00
|
|
|
o.literal(compMeta.selector),
|
|
|
|
o.variable(hostViewFactoryVar),
|
|
|
|
o.importExpr(compMeta.type),
|
2016-06-28 09:54:42 -07:00
|
|
|
],
|
|
|
|
o.importType(
|
2016-11-23 09:42:19 -08:00
|
|
|
createIdentifier(Identifiers.ComponentFactory),
|
2016-08-24 17:39:49 -07:00
|
|
|
[o.importType(compMeta.type)], [o.TypeModifier.Const])))
|
2016-06-28 09:54:42 -07:00
|
|
|
.toDeclStmt(null, [o.StmtModifier.Final]));
|
|
|
|
return compFactoryVar;
|
2016-01-06 14:13:44 -08:00
|
|
|
}
|
|
|
|
|
2016-06-08 16:38:52 -07:00
|
|
|
private _compileComponent(
|
2016-11-10 14:07:30 -08:00
|
|
|
compMeta: CompileDirectiveMetadata, ngModule: CompileNgModuleMetadata,
|
|
|
|
directiveIdentifiers: CompileIdentifierMetadata[], componentStyles: CompiledStylesheet,
|
2016-07-25 03:02:57 -07:00
|
|
|
fileSuffix: string, targetStatements: o.Statement[]): string {
|
2016-09-23 16:37:04 -04:00
|
|
|
const parsedAnimations = this._animationParser.parseComponent(compMeta);
|
2016-11-10 14:07:30 -08:00
|
|
|
const directives =
|
2016-11-10 16:27:53 -08:00
|
|
|
directiveIdentifiers.map(dir => this._metadataResolver.getDirectiveSummary(dir.reference));
|
2016-11-10 14:07:30 -08:00
|
|
|
const pipes = ngModule.transitiveModule.pipes.map(
|
2016-11-10 16:27:53 -08:00
|
|
|
pipe => this._metadataResolver.getPipeSummary(pipe.reference));
|
2016-11-10 14:07:30 -08:00
|
|
|
|
2016-09-15 15:38:17 -07:00
|
|
|
const parsedTemplate = this._templateParser.parse(
|
2016-11-10 14:07:30 -08:00
|
|
|
compMeta, compMeta.template.template, directives, pipes, ngModule.schemas,
|
2016-11-23 09:42:19 -08:00
|
|
|
identifierName(compMeta.type));
|
2016-09-15 15:38:17 -07:00
|
|
|
const stylesExpr = componentStyles ? o.variable(componentStyles.stylesVar) : o.literalArr([]);
|
2016-09-23 16:37:04 -04:00
|
|
|
const compiledAnimations =
|
2016-11-23 09:42:19 -08:00
|
|
|
this._animationCompiler.compile(identifierName(compMeta.type), parsedAnimations);
|
2016-09-23 16:37:04 -04:00
|
|
|
const viewResult = this._viewCompiler.compileComponent(
|
|
|
|
compMeta, parsedTemplate, stylesExpr, pipes, compiledAnimations);
|
2016-06-24 08:46:43 -07:00
|
|
|
if (componentStyles) {
|
2016-11-23 09:42:19 -08:00
|
|
|
targetStatements.push(
|
|
|
|
..._resolveStyleStatements(this._staticReflector, componentStyles, fileSuffix));
|
2016-06-24 08:46:43 -07:00
|
|
|
}
|
2016-11-02 08:11:10 -07:00
|
|
|
compiledAnimations.forEach(entry => targetStatements.push(...entry.statements));
|
2016-11-23 09:42:19 -08:00
|
|
|
targetStatements.push(..._resolveViewStatements(this._staticReflector, viewResult));
|
2016-11-02 08:36:23 -07:00
|
|
|
return viewResult.viewClassVar;
|
2016-01-06 14:13:44 -08:00
|
|
|
}
|
|
|
|
|
2016-10-24 13:28:23 -07:00
|
|
|
private _codgenStyles(
|
2016-11-29 15:36:33 -08:00
|
|
|
fileUrl: string, stylesCompileResult: CompiledStylesheet, fileSuffix: string): GeneratedFile {
|
2016-11-23 09:42:19 -08:00
|
|
|
_resolveStyleStatements(this._staticReflector, stylesCompileResult, fileSuffix);
|
2016-06-08 16:38:52 -07:00
|
|
|
return this._codegenSourceModule(
|
2016-10-24 13:28:23 -07:00
|
|
|
fileUrl, _stylesModuleUrl(
|
|
|
|
stylesCompileResult.meta.moduleUrl, stylesCompileResult.isShimmed, fileSuffix),
|
2016-06-24 08:46:43 -07:00
|
|
|
stylesCompileResult.statements, [stylesCompileResult.stylesVar]);
|
2016-05-02 09:38:46 -07:00
|
|
|
}
|
2016-01-06 14:13:44 -08:00
|
|
|
|
2016-06-08 16:38:52 -07:00
|
|
|
private _codegenSourceModule(
|
2016-11-29 15:36:33 -08:00
|
|
|
srcFileUrl: string, genFileUrl: string, statements: o.Statement[],
|
|
|
|
exportedVars: string[]): GeneratedFile {
|
|
|
|
return new GeneratedFile(
|
|
|
|
srcFileUrl, genFileUrl,
|
|
|
|
this._outputEmitter.emitStatements(genFileUrl, statements, exportedVars));
|
2016-01-06 14:13:44 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-23 09:42:19 -08:00
|
|
|
function _resolveViewStatements(
|
|
|
|
reflector: StaticReflector, compileResult: ViewCompileResult): o.Statement[] {
|
2016-06-22 14:06:23 -07:00
|
|
|
compileResult.dependencies.forEach((dep) => {
|
2016-11-02 08:36:23 -07:00
|
|
|
if (dep instanceof ViewClassDependency) {
|
|
|
|
const vfd = <ViewClassDependency>dep;
|
2016-11-23 09:42:19 -08:00
|
|
|
vfd.placeholder.reference =
|
|
|
|
reflector.getStaticSymbol(_ngfactoryModuleUrl(identifierModuleUrl(vfd.comp)), dep.name);
|
2016-06-22 14:06:23 -07:00
|
|
|
} else if (dep instanceof ComponentFactoryDependency) {
|
2016-09-19 15:36:25 -07:00
|
|
|
const cfd = <ComponentFactoryDependency>dep;
|
2016-11-23 09:42:19 -08:00
|
|
|
cfd.placeholder.reference = reflector.getStaticSymbol(
|
|
|
|
_ngfactoryModuleUrl(identifierModuleUrl(cfd.comp)), _componentFactoryName(cfd.comp));
|
2016-10-13 16:34:37 -07:00
|
|
|
} else if (dep instanceof DirectiveWrapperDependency) {
|
|
|
|
const dwd = <DirectiveWrapperDependency>dep;
|
2016-11-23 09:42:19 -08:00
|
|
|
dwd.placeholder.reference =
|
|
|
|
reflector.getStaticSymbol(_ngfactoryModuleUrl(identifierModuleUrl(dwd.dir)), dwd.name);
|
2016-06-22 14:06:23 -07:00
|
|
|
}
|
|
|
|
});
|
2016-01-06 14:13:44 -08:00
|
|
|
return compileResult.statements;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-06-08 16:38:52 -07:00
|
|
|
function _resolveStyleStatements(
|
2016-11-23 09:42:19 -08:00
|
|
|
reflector: StaticReflector, compileResult: CompiledStylesheet,
|
|
|
|
fileSuffix: string): o.Statement[] {
|
2016-01-06 14:13:44 -08:00
|
|
|
compileResult.dependencies.forEach((dep) => {
|
2016-11-23 09:42:19 -08:00
|
|
|
dep.valuePlaceholder.reference = reflector.getStaticSymbol(
|
|
|
|
_stylesModuleUrl(dep.moduleUrl, dep.isShimmed, fileSuffix), dep.name);
|
2016-01-06 14:13:44 -08:00
|
|
|
});
|
|
|
|
return compileResult.statements;
|
|
|
|
}
|
|
|
|
|
2016-10-13 16:34:37 -07:00
|
|
|
function _ngfactoryModuleUrl(dirUrl: string): string {
|
|
|
|
const urlWithSuffix = _splitTypescriptSuffix(dirUrl);
|
2016-05-02 09:38:46 -07:00
|
|
|
return `${urlWithSuffix[0]}.ngfactory${urlWithSuffix[1]}`;
|
2016-01-06 14:13:44 -08:00
|
|
|
}
|
|
|
|
|
2016-06-22 14:06:23 -07:00
|
|
|
function _componentFactoryName(comp: CompileIdentifierMetadata): string {
|
2016-11-23 09:42:19 -08:00
|
|
|
return `${identifierName(comp)}NgFactory`;
|
2016-06-22 14:06:23 -07: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
|
|
|
}
|
|
|
|
|
|
|
|
function _assertComponent(meta: CompileDirectiveMetadata) {
|
|
|
|
if (!meta.isComponent) {
|
2016-11-23 09:42:19 -08:00
|
|
|
throw new Error(
|
|
|
|
`Could not compile '${identifierName(meta.type)}' because it is not a component.`);
|
2016-01-06 14:13:44 -08:00
|
|
|
}
|
|
|
|
}
|
2016-05-02 09:38:46 -07:00
|
|
|
|
2016-08-18 10:11:06 -07:00
|
|
|
function _splitTypescriptSuffix(path: string): string[] {
|
2016-09-15 15:38:17 -07:00
|
|
|
if (path.endsWith('.d.ts')) {
|
|
|
|
return [path.slice(0, -5), '.ts'];
|
2016-08-18 10:11:06 -07:00
|
|
|
}
|
2016-09-15 15:38:17 -07:00
|
|
|
|
|
|
|
const lastDot = path.lastIndexOf('.');
|
|
|
|
|
2016-05-02 09:38:46 -07:00
|
|
|
if (lastDot !== -1) {
|
|
|
|
return [path.substring(0, lastDot), path.substring(lastDot)];
|
|
|
|
}
|
2016-09-15 15:38:17 -07:00
|
|
|
|
|
|
|
return [path, ''];
|
2016-05-25 12:46:22 -07:00
|
|
|
}
|
2016-10-25 16:28:22 -07:00
|
|
|
|
2016-11-15 13:57:25 -08:00
|
|
|
export interface NgAnalyzedModules {
|
|
|
|
ngModules: CompileNgModuleMetadata[];
|
|
|
|
ngModuleByPipeOrDirective: Map<StaticSymbol, CompileNgModuleMetadata>;
|
2016-11-29 15:36:33 -08:00
|
|
|
files: Array<{
|
|
|
|
srcUrl: string,
|
|
|
|
directives: StaticSymbol[],
|
|
|
|
pipes: StaticSymbol[],
|
|
|
|
ngModules: StaticSymbol[]
|
|
|
|
}>;
|
2016-11-15 13:57:25 -08:00
|
|
|
symbolsMissingModule?: StaticSymbol[];
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns all the source files and a mapping from modules to directives
|
|
|
|
export function analyzeNgModules(
|
|
|
|
programStaticSymbols: StaticSymbol[],
|
2016-11-17 20:11:55 -08:00
|
|
|
options: {includeFilePattern?: RegExp, excludeFilePattern?: RegExp},
|
2016-11-15 13:57:25 -08:00
|
|
|
metadataResolver: CompileMetadataResolver): NgAnalyzedModules {
|
|
|
|
const {ngModules, symbolsMissingModule} =
|
|
|
|
_createNgModules(programStaticSymbols, options, metadataResolver);
|
|
|
|
return _analyzeNgModules(ngModules, symbolsMissingModule);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function analyzeAndValidateNgModules(
|
2016-11-17 20:11:55 -08:00
|
|
|
programStaticSymbols: StaticSymbol[],
|
|
|
|
options: {includeFilePattern?: RegExp, excludeFilePattern?: RegExp},
|
2016-11-15 13:57:25 -08:00
|
|
|
metadataResolver: CompileMetadataResolver): NgAnalyzedModules {
|
|
|
|
const result = analyzeNgModules(programStaticSymbols, options, metadataResolver);
|
|
|
|
if (result.symbolsMissingModule && result.symbolsMissingModule.length) {
|
|
|
|
const messages = result.symbolsMissingModule.map(
|
|
|
|
s => `Cannot determine the module for class ${s.name} in ${s.filePath}!`);
|
|
|
|
throw new Error(messages.join('\n'));
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
function _analyzeNgModules(
|
|
|
|
ngModuleMetas: CompileNgModuleMetadata[],
|
|
|
|
symbolsMissingModule: StaticSymbol[]): NgAnalyzedModules {
|
|
|
|
const moduleMetasByRef = new Map<any, CompileNgModuleMetadata>();
|
|
|
|
ngModuleMetas.forEach((ngModule) => moduleMetasByRef.set(ngModule.type.reference, ngModule));
|
|
|
|
const ngModuleByPipeOrDirective = new Map<StaticSymbol, CompileNgModuleMetadata>();
|
|
|
|
const ngModulesByFile = new Map<string, StaticSymbol[]>();
|
|
|
|
const ngDirectivesByFile = new Map<string, StaticSymbol[]>();
|
2016-11-29 15:36:33 -08:00
|
|
|
const ngPipesByFile = new Map<string, StaticSymbol[]>();
|
2016-11-15 13:57:25 -08:00
|
|
|
const filePaths = new Set<string>();
|
|
|
|
|
|
|
|
// Looping over all modules to construct:
|
|
|
|
// - a map from file to modules `ngModulesByFile`,
|
|
|
|
// - a map from file to directives `ngDirectivesByFile`,
|
2016-11-29 15:36:33 -08:00
|
|
|
// - a map from file to pipes `ngPipesByFile`,
|
2016-11-15 13:57:25 -08:00
|
|
|
// - a map from directive/pipe to module `ngModuleByPipeOrDirective`.
|
|
|
|
ngModuleMetas.forEach((ngModuleMeta) => {
|
|
|
|
const srcFileUrl = ngModuleMeta.type.reference.filePath;
|
|
|
|
filePaths.add(srcFileUrl);
|
|
|
|
ngModulesByFile.set(
|
|
|
|
srcFileUrl, (ngModulesByFile.get(srcFileUrl) || []).concat(ngModuleMeta.type.reference));
|
|
|
|
|
|
|
|
ngModuleMeta.declaredDirectives.forEach((dirIdentifier) => {
|
|
|
|
const fileUrl = dirIdentifier.reference.filePath;
|
|
|
|
filePaths.add(fileUrl);
|
|
|
|
ngDirectivesByFile.set(
|
|
|
|
fileUrl, (ngDirectivesByFile.get(fileUrl) || []).concat(dirIdentifier.reference));
|
|
|
|
ngModuleByPipeOrDirective.set(dirIdentifier.reference, ngModuleMeta);
|
|
|
|
});
|
|
|
|
ngModuleMeta.declaredPipes.forEach((pipeIdentifier) => {
|
|
|
|
const fileUrl = pipeIdentifier.reference.filePath;
|
|
|
|
filePaths.add(fileUrl);
|
2016-11-29 15:36:33 -08:00
|
|
|
ngPipesByFile.set(
|
|
|
|
fileUrl, (ngPipesByFile.get(fileUrl) || []).concat(pipeIdentifier.reference));
|
2016-11-15 13:57:25 -08:00
|
|
|
ngModuleByPipeOrDirective.set(pipeIdentifier.reference, ngModuleMeta);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2016-11-29 15:36:33 -08:00
|
|
|
const files:
|
|
|
|
{srcUrl: string,
|
|
|
|
directives: StaticSymbol[],
|
|
|
|
pipes: StaticSymbol[],
|
|
|
|
ngModules: StaticSymbol[]}[] = [];
|
2016-11-15 13:57:25 -08:00
|
|
|
|
|
|
|
filePaths.forEach((srcUrl) => {
|
|
|
|
const directives = ngDirectivesByFile.get(srcUrl) || [];
|
2016-11-29 15:36:33 -08:00
|
|
|
const pipes = ngPipesByFile.get(srcUrl) || [];
|
2016-11-15 13:57:25 -08:00
|
|
|
const ngModules = ngModulesByFile.get(srcUrl) || [];
|
2016-11-29 15:36:33 -08:00
|
|
|
files.push({srcUrl, directives, pipes, ngModules});
|
2016-11-15 13:57:25 -08:00
|
|
|
});
|
|
|
|
|
|
|
|
return {
|
|
|
|
// map directive/pipe to module
|
|
|
|
ngModuleByPipeOrDirective,
|
|
|
|
// list modules and directives for every source file
|
|
|
|
files,
|
|
|
|
ngModules: ngModuleMetas, symbolsMissingModule
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
export function extractProgramSymbols(
|
|
|
|
staticReflector: StaticReflector, files: string[],
|
|
|
|
options: {includeFilePattern?: RegExp, excludeFilePattern?: RegExp} = {}): StaticSymbol[] {
|
|
|
|
const staticSymbols: StaticSymbol[] = [];
|
2016-11-29 15:36:33 -08:00
|
|
|
files.filter(fileName => filterFileByPatterns(fileName, options)).forEach(sourceFile => {
|
2016-11-17 20:11:55 -08:00
|
|
|
const moduleMetadata = staticReflector.getModuleMetadata(sourceFile);
|
|
|
|
if (!moduleMetadata) {
|
2016-11-22 13:29:53 -08:00
|
|
|
console.error(`WARNING: no metadata found for ${sourceFile}`);
|
2016-11-17 20:11:55 -08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const metadata = moduleMetadata['metadata'];
|
|
|
|
|
|
|
|
if (!metadata) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const symbol of Object.keys(metadata)) {
|
|
|
|
if (metadata[symbol] && metadata[symbol].__symbolic == 'error') {
|
|
|
|
// Ignore symbols that are only included to record error information.
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
staticSymbols.push(staticReflector.getStaticSymbol(sourceFile, symbol));
|
|
|
|
}
|
|
|
|
});
|
2016-11-15 13:57:25 -08:00
|
|
|
|
|
|
|
return staticSymbols;
|
|
|
|
}
|
|
|
|
|
2016-11-10 14:07:30 -08:00
|
|
|
// Load the NgModules and check
|
|
|
|
// that all directives / pipes that are present in the program
|
|
|
|
// are also declared by a module.
|
2016-11-16 10:22:11 -08:00
|
|
|
function _createNgModules(
|
2016-11-15 13:57:25 -08:00
|
|
|
programStaticSymbols: StaticSymbol[],
|
2016-11-17 20:11:55 -08:00
|
|
|
options: {includeFilePattern?: RegExp, excludeFilePattern?: RegExp},
|
2016-11-16 10:22:11 -08:00
|
|
|
metadataResolver: CompileMetadataResolver):
|
|
|
|
{ngModules: CompileNgModuleMetadata[], symbolsMissingModule: StaticSymbol[]} {
|
2016-11-10 14:07:30 -08:00
|
|
|
const ngModules = new Map<any, CompileNgModuleMetadata>();
|
|
|
|
const programPipesAndDirectives: StaticSymbol[] = [];
|
|
|
|
const ngModulePipesAndDirective = new Set<StaticSymbol>();
|
|
|
|
|
|
|
|
const addNgModule = (staticSymbol: any) => {
|
2016-11-29 15:36:33 -08:00
|
|
|
if (ngModules.has(staticSymbol) || !filterFileByPatterns(staticSymbol.filePath, options)) {
|
2016-11-10 14:07:30 -08:00
|
|
|
return false;
|
|
|
|
}
|
2016-11-29 08:08:22 -08:00
|
|
|
const ngModule = metadataResolver.getNgModuleMetadata(staticSymbol, false);
|
2016-10-25 16:28:22 -07:00
|
|
|
if (ngModule) {
|
2016-11-10 14:07:30 -08:00
|
|
|
ngModules.set(ngModule.type.reference, ngModule);
|
|
|
|
ngModule.declaredDirectives.forEach((dir) => ngModulePipesAndDirective.add(dir.reference));
|
|
|
|
ngModule.declaredPipes.forEach((pipe) => ngModulePipesAndDirective.add(pipe.reference));
|
2016-11-17 20:11:55 -08:00
|
|
|
// For every input module add the list of transitively included modules
|
2016-11-29 08:08:22 -08:00
|
|
|
ngModule.transitiveModule.modules.forEach(modMeta => addNgModule(modMeta.reference));
|
2016-11-10 14:07:30 -08:00
|
|
|
}
|
|
|
|
return !!ngModule;
|
|
|
|
};
|
|
|
|
programStaticSymbols.forEach((staticSymbol) => {
|
|
|
|
if (!addNgModule(staticSymbol) &&
|
|
|
|
(metadataResolver.isDirective(staticSymbol) || metadataResolver.isPipe(staticSymbol))) {
|
|
|
|
programPipesAndDirectives.push(staticSymbol);
|
2016-10-25 16:28:22 -07:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2016-11-10 14:07:30 -08:00
|
|
|
// Throw an error if any of the program pipe or directives is not declared by a module
|
|
|
|
const symbolsMissingModule =
|
|
|
|
programPipesAndDirectives.filter(s => !ngModulePipesAndDirective.has(s));
|
|
|
|
|
2016-11-16 10:22:11 -08:00
|
|
|
return {ngModules: Array.from(ngModules.values()), symbolsMissingModule};
|
2016-10-25 16:28:22 -07:00
|
|
|
}
|