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
|
|
|
|
*/
|
|
|
|
|
2017-05-16 16:30:37 -07:00
|
|
|
import {CompileDirectiveMetadata, CompileDirectiveSummary, CompileIdentifierMetadata, CompileNgModuleMetadata, CompileNgModuleSummary, CompilePipeMetadata, CompileProviderMetadata, CompileStylesheetMetadata, CompileSummaryKind, CompileTypeMetadata, CompileTypeSummary, componentFactoryName, createHostComponentMeta, flatten, identifierName, sourceUrl, templateSourceUrl} from '../compile_metadata';
|
2017-02-17 08:56:49 -08:00
|
|
|
import {CompilerConfig} from '../config';
|
2017-05-18 13:46:51 -07:00
|
|
|
import {Identifiers, createTokenForExternalReference} 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';
|
2016-12-15 09:12:40 -08:00
|
|
|
import {SummaryResolver} from '../summary_resolver';
|
2016-11-14 17:37:47 -08:00
|
|
|
import {TemplateParser} from '../template_parser/template_parser';
|
2017-05-16 16:30:37 -07:00
|
|
|
import {OutputContext, syntaxError} from '../util';
|
|
|
|
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';
|
2016-11-29 15:36:33 -08:00
|
|
|
import {GeneratedFile} from './generated_file';
|
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-04-26 09:24:42 -07:00
|
|
|
import {ngfactoryFilePath, splitTypescriptSuffix, summaryFileName, summaryForJitFileName, summaryForJitName} from './util';
|
2016-01-06 14:13:44 -08:00
|
|
|
|
2016-11-14 17:37:47 -08:00
|
|
|
export class AotCompiler {
|
2016-06-08 16:38:52 -07:00
|
|
|
constructor(
|
2017-02-17 08:56:49 -08:00
|
|
|
private _config: CompilerConfig, private _host: AotCompilerHost,
|
2017-05-18 13:46:51 -07:00
|
|
|
private _reflector: StaticReflector, private _metadataResolver: CompileMetadataResolver,
|
|
|
|
private _templateParser: TemplateParser, private _styleCompiler: StyleCompiler,
|
|
|
|
private _viewCompiler: ViewCompiler, private _ngModuleCompiler: NgModuleCompiler,
|
|
|
|
private _outputEmitter: OutputEmitter,
|
2017-03-24 09:59:58 -07:00
|
|
|
private _summaryResolver: SummaryResolver<StaticSymbol>, private _localeId: string|null,
|
2017-06-09 14:00:03 -07:00
|
|
|
private _translationFormat: string|null, private _enableSummariesForJit: boolean|null,
|
|
|
|
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 {
|
2016-12-15 09:12:40 -08:00
|
|
|
const programSymbols = extractProgramSymbols(this._symbolResolver, rootFiles, this._host);
|
2017-05-23 13:40:50 -07:00
|
|
|
const analyzeResult =
|
|
|
|
analyzeAndValidateNgModules(programSymbols, this._host, this._metadataResolver);
|
|
|
|
analyzeResult.ngModules.forEach(
|
|
|
|
ngModule => this._metadataResolver.loadNgModuleDirectiveAndPipeMetadata(
|
|
|
|
ngModule.type.reference, true));
|
|
|
|
return analyzeResult;
|
|
|
|
}
|
|
|
|
|
|
|
|
analyzeModulesAsync(rootFiles: string[]): Promise<NgAnalyzedModules> {
|
|
|
|
const programSymbols = extractProgramSymbols(this._symbolResolver, rootFiles, this._host);
|
|
|
|
const analyzeResult =
|
2016-12-15 09:12:40 -08:00
|
|
|
analyzeAndValidateNgModules(programSymbols, this._host, 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-05-23 13:40:50 -07:00
|
|
|
emitAllStubs(analyzeResult: NgAnalyzedModules): GeneratedFile[] {
|
|
|
|
const {files} = analyzeResult;
|
|
|
|
const sourceModules =
|
|
|
|
files.map(file => this._compileStubFile(file.srcUrl, file.directives, file.ngModules));
|
|
|
|
return flatten(sourceModules);
|
|
|
|
}
|
|
|
|
|
|
|
|
emitAllImpls(analyzeResult: NgAnalyzedModules): GeneratedFile[] {
|
|
|
|
const {ngModuleByPipeOrDirective, files} = analyzeResult;
|
2017-05-17 15:39:08 -07:00
|
|
|
const sourceModules = files.map(
|
2017-05-23 13:40:50 -07:00
|
|
|
file => this._compileImplFile(
|
2017-05-17 15:39:08 -07:00
|
|
|
file.srcUrl, ngModuleByPipeOrDirective, file.directives, file.pipes, file.ngModules,
|
|
|
|
file.injectables));
|
|
|
|
return flatten(sourceModules);
|
|
|
|
}
|
|
|
|
|
2017-05-23 13:40:50 -07:00
|
|
|
private _compileStubFile(
|
|
|
|
srcFileUrl: string, directives: StaticSymbol[], ngModules: StaticSymbol[]): GeneratedFile[] {
|
|
|
|
const fileSuffix = splitTypescriptSuffix(srcFileUrl, true)[1];
|
|
|
|
const generatedFiles: GeneratedFile[] = [];
|
|
|
|
|
|
|
|
const jitSummaryStmts: o.Statement[] = [];
|
|
|
|
const ngFactoryStms: o.Statement[] = [];
|
|
|
|
|
|
|
|
const ngFactoryOutputCtx = this._createOutputContext(ngfactoryFilePath(srcFileUrl, true));
|
|
|
|
const jitSummaryOutputCtx = this._createOutputContext(summaryForJitFileName(srcFileUrl, true));
|
|
|
|
|
|
|
|
// create exports that user code can reference
|
|
|
|
ngModules.forEach((ngModuleReference) => {
|
|
|
|
this._ngModuleCompiler.createStub(ngFactoryOutputCtx, ngModuleReference);
|
|
|
|
createForJitStub(jitSummaryOutputCtx, ngModuleReference);
|
|
|
|
});
|
|
|
|
// Note: we are creating stub ngfactory/ngsummary for all source files,
|
|
|
|
// as the real calculation requires almost the same logic as producing the real content for
|
|
|
|
// them.
|
|
|
|
// Our pipeline will filter out empty ones at the end.
|
|
|
|
generatedFiles.push(this._codegenSourceModule(srcFileUrl, ngFactoryOutputCtx));
|
|
|
|
generatedFiles.push(this._codegenSourceModule(srcFileUrl, jitSummaryOutputCtx));
|
|
|
|
|
|
|
|
// create stubs for external stylesheets (always empty, as users should not import anything from
|
|
|
|
// the generated code)
|
|
|
|
directives.forEach((dirType) => {
|
|
|
|
const compMeta = this._metadataResolver.getDirectiveMetadata(<any>dirType);
|
|
|
|
if (!compMeta.isComponent) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Note: compMeta is a component and therefore template is non null.
|
|
|
|
compMeta.template !.externalStylesheets.forEach((stylesheetMeta) => {
|
|
|
|
generatedFiles.push(this._codegenSourceModule(
|
|
|
|
stylesheetMeta.moduleUrl !,
|
|
|
|
this._createOutputContext(_stylesModuleUrl(
|
|
|
|
stylesheetMeta.moduleUrl !, this._styleCompiler.needsStyleShim(compMeta),
|
|
|
|
fileSuffix))));
|
|
|
|
});
|
|
|
|
});
|
|
|
|
return generatedFiles;
|
|
|
|
}
|
|
|
|
|
|
|
|
private _compileImplFile(
|
2016-10-25 16:28:22 -07:00
|
|
|
srcFileUrl: string, ngModuleByPipeOrDirective: Map<StaticSymbol, CompileNgModuleMetadata>,
|
2016-12-15 09:12:40 -08:00
|
|
|
directives: StaticSymbol[], pipes: StaticSymbol[], ngModules: StaticSymbol[],
|
|
|
|
injectables: StaticSymbol[]): GeneratedFile[] {
|
2017-04-26 09:24:42 -07:00
|
|
|
const fileSuffix = splitTypescriptSuffix(srcFileUrl, true)[1];
|
2016-11-29 15:36:33 -08:00
|
|
|
const generatedFiles: GeneratedFile[] = [];
|
|
|
|
|
2017-05-16 16:30:37 -07:00
|
|
|
const outputCtx = this._createOutputContext(ngfactoryFilePath(srcFileUrl, true));
|
|
|
|
|
|
|
|
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-05-16 16:30:37 -07:00
|
|
|
ngModules.forEach((ngModuleType) => this._compileModule(outputCtx, ngModuleType));
|
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-05-23 13:40:50 -07:00
|
|
|
generatedFiles.push(
|
|
|
|
this._codegenStyles(stylesheetMeta.moduleUrl !, compMeta, stylesheetMeta, 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-05-16 16:30:37 -07:00
|
|
|
if (outputCtx.statements.length > 0) {
|
|
|
|
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(
|
|
|
|
srcFileUrl: string, directives: StaticSymbol[], pipes: StaticSymbol[],
|
2017-05-16 16:30:37 -07:00
|
|
|
ngModules: StaticSymbol[], injectables: StaticSymbol[],
|
|
|
|
ngFactoryCtx: OutputContext): GeneratedFile[] {
|
2016-12-15 09:12:40 -08:00
|
|
|
const symbolSummaries = this._symbolResolver.getSymbolsOf(srcFileUrl)
|
|
|
|
.map(symbol => this._symbolResolver.resolveSymbol(symbol));
|
2017-04-26 09:24:42 -07:00
|
|
|
const typeData: {
|
|
|
|
summary: CompileTypeSummary,
|
|
|
|
metadata: CompileNgModuleMetadata | CompileDirectiveMetadata | CompilePipeMetadata |
|
|
|
|
CompileTypeMetadata
|
|
|
|
}[] =
|
|
|
|
[
|
|
|
|
...ngModules.map(ref => ({
|
|
|
|
summary: this._metadataResolver.getNgModuleSummary(ref) !,
|
|
|
|
metadata: this._metadataResolver.getNgModuleMetadata(ref) !
|
|
|
|
})),
|
|
|
|
...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) !
|
|
|
|
})),
|
|
|
|
...injectables.map(ref => ({
|
|
|
|
summary: this._metadataResolver.getInjectableSummary(ref) !,
|
|
|
|
metadata: this._metadataResolver.getInjectableSummary(ref) !.type
|
|
|
|
}))
|
|
|
|
];
|
2017-05-16 16:30:37 -07:00
|
|
|
const forJitOutputCtx = this._createOutputContext(summaryForJitFileName(srcFileUrl, true));
|
|
|
|
const {json, exportAs} = serializeSummaries(
|
|
|
|
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-06-09 14:00:03 -07:00
|
|
|
const summaryJson = new GeneratedFile(srcFileUrl, summaryFileName(srcFileUrl), json);
|
|
|
|
if (this._enableSummariesForJit) {
|
|
|
|
return [summaryJson, this._codegenSourceModule(srcFileUrl, forJitOutputCtx)];
|
|
|
|
};
|
|
|
|
|
|
|
|
return [summaryJson];
|
2016-12-15 09:12:40 -08:00
|
|
|
}
|
|
|
|
|
2017-05-16 16:30:37 -07:00
|
|
|
private _compileModule(outputCtx: OutputContext, ngModuleType: StaticSymbol): void {
|
2017-03-24 09:59:58 -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({
|
2017-05-18 13:46:51 -07:00
|
|
|
token: createTokenForExternalReference(this._reflector, 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({
|
2017-05-18 13:46:51 -07:00
|
|
|
token: createTokenForExternalReference(this._reflector, 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
|
|
|
}
|
|
|
|
|
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 {
|
2016-12-27 09:36:47 -08:00
|
|
|
const hostType = this._metadataResolver.getHostComponentType(compMeta.type.reference);
|
2016-11-23 09:42:19 -08:00
|
|
|
const hostMeta = createHostComponentMeta(
|
2016-12-27 09:36:47 -08:00
|
|
|
hostType, compMeta, this._metadataResolver.getHostComponentViewClass(hostType));
|
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
|
|
|
}
|
|
|
|
|
2016-06-08 16:38:52 -07:00
|
|
|
private _compileComponent(
|
2017-05-16 16:30:37 -07:00
|
|
|
outputCtx: OutputContext, compMeta: CompileDirectiveMetadata,
|
|
|
|
ngModule: CompileNgModuleMetadata, directiveIdentifiers: CompileIdentifierMetadata[],
|
|
|
|
componentStyles: CompiledStylesheet|null, fileSuffix: string): ViewCompileResult {
|
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
|
|
|
|
2017-02-09 14:59:57 -08:00
|
|
|
const {template: parsedTemplate, pipes: usedPipes} = this._templateParser.parse(
|
2017-03-24 09:59:58 -07:00
|
|
|
compMeta, compMeta.template !.template !, directives, pipes, ngModule.schemas,
|
|
|
|
templateSourceUrl(ngModule.type, compMeta, compMeta.template !));
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
private _createOutputContext(genFilePath: string): OutputContext {
|
|
|
|
const importExpr = (symbol: StaticSymbol, typeParams: o.Type[] | null = null) => {
|
|
|
|
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) || symbol;
|
2017-06-21 15:05:11 -07:00
|
|
|
const importModule = this._symbolResolver.fileNameToModuleName(filePath, genFilePath);
|
2017-06-22 10:36:42 -07:00
|
|
|
|
|
|
|
// 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.
|
2017-06-21 15:05:11 -07:00
|
|
|
const selfReference = this._symbolResolver.fileNameToModuleName(genFilePath, genFilePath);
|
|
|
|
const moduleName = importModule === selfReference ? null : importModule;
|
|
|
|
|
2017-05-16 16:30:37 -07:00
|
|
|
// 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));
|
|
|
|
};
|
|
|
|
|
|
|
|
return {statements: [], genFilePath, importExpr};
|
2016-01-06 14:13:44 -08:00
|
|
|
}
|
|
|
|
|
2017-05-16 16:30:37 -07:00
|
|
|
private _codegenStyles(
|
|
|
|
srcFileUrl: string, compMeta: CompileDirectiveMetadata,
|
|
|
|
stylesheetMetadata: CompileStylesheetMetadata, fileSuffix: string): GeneratedFile {
|
|
|
|
const outputCtx = this._createOutputContext(_stylesModuleUrl(
|
|
|
|
stylesheetMetadata.moduleUrl !, this._styleCompiler.needsStyleShim(compMeta), fileSuffix));
|
|
|
|
const compiledStylesheet =
|
|
|
|
this._styleCompiler.compileStyles(outputCtx, compMeta, stylesheetMetadata);
|
|
|
|
_resolveStyleStatements(
|
|
|
|
this._symbolResolver, compiledStylesheet, this._styleCompiler.needsStyleShim(compMeta),
|
|
|
|
fileSuffix);
|
|
|
|
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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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>;
|
2016-11-29 15:36:33 -08:00
|
|
|
files: Array<{
|
|
|
|
srcUrl: string,
|
|
|
|
directives: StaticSymbol[],
|
|
|
|
pipes: StaticSymbol[],
|
2016-12-15 09:12:40 -08:00
|
|
|
ngModules: StaticSymbol[],
|
|
|
|
injectables: StaticSymbol[]
|
2016-11-29 15:36:33 -08:00
|
|
|
}>;
|
2016-11-15 13:57:25 -08:00
|
|
|
symbolsMissingModule?: StaticSymbol[];
|
|
|
|
}
|
|
|
|
|
2016-12-15 09:12:40 -08:00
|
|
|
export interface NgAnalyzeModulesHost { isSourceFile(filePath: string): boolean; }
|
|
|
|
|
2016-11-15 13:57:25 -08:00
|
|
|
// Returns all the source files and a mapping from modules to directives
|
|
|
|
export function analyzeNgModules(
|
2016-12-15 09:12:40 -08:00
|
|
|
programStaticSymbols: StaticSymbol[], host: NgAnalyzeModulesHost,
|
2016-11-15 13:57:25 -08:00
|
|
|
metadataResolver: CompileMetadataResolver): NgAnalyzedModules {
|
|
|
|
const {ngModules, symbolsMissingModule} =
|
2016-12-15 09:12:40 -08:00
|
|
|
_createNgModules(programStaticSymbols, host, metadataResolver);
|
|
|
|
return _analyzeNgModules(programStaticSymbols, ngModules, symbolsMissingModule, metadataResolver);
|
2016-11-15 13:57:25 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
export function analyzeAndValidateNgModules(
|
2016-12-15 09:12:40 -08:00
|
|
|
programStaticSymbols: StaticSymbol[], host: NgAnalyzeModulesHost,
|
2016-11-15 13:57:25 -08:00
|
|
|
metadataResolver: CompileMetadataResolver): NgAnalyzedModules {
|
2016-12-15 09:12:40 -08:00
|
|
|
const result = analyzeNgModules(programStaticSymbols, host, metadataResolver);
|
2016-11-15 13:57:25 -08:00
|
|
|
if (result.symbolsMissingModule && result.symbolsMissingModule.length) {
|
|
|
|
const messages = result.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
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
function _analyzeNgModules(
|
2016-12-15 09:12:40 -08:00
|
|
|
programSymbols: StaticSymbol[], ngModuleMetas: CompileNgModuleMetadata[],
|
|
|
|
symbolsMissingModule: StaticSymbol[],
|
|
|
|
metadataResolver: CompileMetadataResolver): NgAnalyzedModules {
|
2016-11-15 13:57:25 -08:00
|
|
|
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-12-15 09:12:40 -08:00
|
|
|
const ngInjectablesByFile = new Map<string, StaticSymbol[]>();
|
2016-11-15 13:57:25 -08:00
|
|
|
const filePaths = new Set<string>();
|
|
|
|
|
2016-12-15 09:12:40 -08:00
|
|
|
// Make sure we produce an analyzed file for each input file
|
|
|
|
programSymbols.forEach((symbol) => {
|
|
|
|
const filePath = symbol.filePath;
|
|
|
|
filePaths.add(filePath);
|
|
|
|
if (metadataResolver.isInjectable(symbol)) {
|
|
|
|
ngInjectablesByFile.set(filePath, (ngInjectablesByFile.get(filePath) || []).concat(symbol));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2016-11-15 13:57:25 -08:00
|
|
|
// 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-12-15 09:12:40 -08:00
|
|
|
const files: {
|
|
|
|
srcUrl: string,
|
|
|
|
directives: StaticSymbol[],
|
|
|
|
pipes: StaticSymbol[],
|
|
|
|
ngModules: StaticSymbol[],
|
|
|
|
injectables: 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-12-15 09:12:40 -08:00
|
|
|
const injectables = ngInjectablesByFile.get(srcUrl) || [];
|
|
|
|
files.push({srcUrl, directives, pipes, ngModules, injectables});
|
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(
|
2016-12-15 09:12:40 -08:00
|
|
|
staticSymbolResolver: StaticSymbolResolver, files: string[],
|
|
|
|
host: NgAnalyzeModulesHost): StaticSymbol[] {
|
2016-11-15 13:57:25 -08:00
|
|
|
const staticSymbols: StaticSymbol[] = [];
|
2016-12-15 09:12:40 -08:00
|
|
|
files.filter(fileName => host.isSourceFile(fileName)).forEach(sourceFile => {
|
|
|
|
staticSymbolResolver.getSymbolsOf(sourceFile).forEach((symbol) => {
|
|
|
|
const resolvedSymbol = staticSymbolResolver.resolveSymbol(symbol);
|
|
|
|
const symbolMeta = resolvedSymbol.metadata;
|
|
|
|
if (symbolMeta) {
|
|
|
|
if (symbolMeta.__symbolic != 'error') {
|
|
|
|
// Ignore symbols that are only included to record error information.
|
|
|
|
staticSymbols.push(resolvedSymbol.symbol);
|
|
|
|
}
|
2016-11-17 20:11:55 -08:00
|
|
|
}
|
2016-12-15 09:12:40 -08:00
|
|
|
});
|
2016-11-17 20:11:55 -08:00
|
|
|
});
|
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-12-15 09:12:40 -08:00
|
|
|
programStaticSymbols: StaticSymbol[], host: NgAnalyzeModulesHost,
|
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-12-15 09:12:40 -08:00
|
|
|
if (ngModules.has(staticSymbol) || !host.isSourceFile(staticSymbol.filePath)) {
|
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
|
|
|
}
|