2017-06-09 14:50:57 -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-08-16 09:00:03 -07:00
|
|
|
import {AotCompiler, AotCompilerHost, AotCompilerOptions, GeneratedFile, NgAnalyzedModules, core, createAotCompiler, getParseErrors, isSyntaxError, toTypeScript} from '@angular/compiler';
|
2017-08-09 13:45:45 -07:00
|
|
|
import {createBundleIndexHost} from '@angular/tsc-wrapped';
|
2017-08-02 11:20:07 -07:00
|
|
|
import * as fs from 'fs';
|
2017-06-09 14:50:57 -07:00
|
|
|
import * as path from 'path';
|
|
|
|
import * as ts from 'typescript';
|
|
|
|
|
2017-08-14 11:04:18 -07:00
|
|
|
import {BaseAotCompilerHost} from '../compiler_host';
|
2017-06-09 14:50:57 -07:00
|
|
|
import {TypeChecker} from '../diagnostics/check_types';
|
|
|
|
|
2017-08-18 14:03:59 -07:00
|
|
|
import {CompilerHost, CompilerOptions, CustomTransformers, DEFAULT_ERROR_CODE, Diagnostic, EmitFlags, Program, SOURCE, TsEmitArguments, TsEmitCallback} from './api';
|
2017-07-13 14:25:17 -07:00
|
|
|
import {LowerMetadataCache, getExpressionLoweringTransformFactory} from './lower_expressions';
|
2017-06-09 14:50:57 -07:00
|
|
|
import {getAngularEmitterTransformFactory} from './node_emitter_transform';
|
|
|
|
|
2017-08-23 13:57:37 -07:00
|
|
|
const GENERATED_FILES = /(.*?)\.(ngfactory|shim\.ngstyle|ngstyle|ngsummary)\.(js|d\.ts|ts)$/;
|
2017-06-09 14:50:57 -07:00
|
|
|
|
|
|
|
const emptyModules: NgAnalyzedModules = {
|
|
|
|
ngModules: [],
|
|
|
|
ngModuleByPipeOrDirective: new Map(),
|
|
|
|
files: []
|
|
|
|
};
|
|
|
|
|
2017-08-16 15:35:19 -07:00
|
|
|
const defaultEmitCallback: TsEmitCallback =
|
|
|
|
({program, targetSourceFile, writeFile, cancellationToken, emitOnlyDtsFiles,
|
|
|
|
customTransformers}) =>
|
|
|
|
program.emit(
|
|
|
|
targetSourceFile, writeFile, cancellationToken, emitOnlyDtsFiles, customTransformers);
|
|
|
|
|
|
|
|
|
2017-06-09 14:50:57 -07:00
|
|
|
class AngularCompilerProgram implements Program {
|
|
|
|
private tsProgram: ts.Program;
|
|
|
|
private aotCompilerHost: AotCompilerHost;
|
|
|
|
private compiler: AotCompiler;
|
|
|
|
private srcNames: string[];
|
2017-07-13 14:25:17 -07:00
|
|
|
private metadataCache: LowerMetadataCache;
|
2017-06-09 14:50:57 -07:00
|
|
|
// Lazily initialized fields
|
|
|
|
private _analyzedModules: NgAnalyzedModules|undefined;
|
|
|
|
private _structuralDiagnostics: Diagnostic[] = [];
|
|
|
|
private _stubs: GeneratedFile[]|undefined;
|
|
|
|
private _stubFiles: string[]|undefined;
|
|
|
|
private _programWithStubsHost: ts.CompilerHost|undefined;
|
|
|
|
private _programWithStubs: ts.Program|undefined;
|
|
|
|
private _generatedFiles: GeneratedFile[]|undefined;
|
|
|
|
private _generatedFileDiagnostics: Diagnostic[]|undefined;
|
|
|
|
private _typeChecker: TypeChecker|undefined;
|
|
|
|
private _semanticDiagnostics: Diagnostic[]|undefined;
|
2017-08-09 13:45:45 -07:00
|
|
|
private _optionsDiagnostics: Diagnostic[] = [];
|
2017-06-09 14:50:57 -07:00
|
|
|
|
|
|
|
constructor(
|
|
|
|
private rootNames: string[], private options: CompilerOptions, private host: CompilerHost,
|
2017-08-30 14:26:19 -07:00
|
|
|
oldProgram?: Program) {
|
2017-08-09 13:45:45 -07:00
|
|
|
if (options.flatModuleOutFile && !options.skipMetadataEmit) {
|
|
|
|
const {host: bundleHost, indexName, errors} = createBundleIndexHost(options, rootNames, host);
|
|
|
|
if (errors) {
|
|
|
|
// TODO(tbosch): once we move MetadataBundler from tsc_wrapped into compiler_cli,
|
|
|
|
// directly create ng.Diagnostic instead of using ts.Diagnostic here.
|
2017-08-18 14:03:59 -07:00
|
|
|
this._optionsDiagnostics.push(...errors.map(e => ({
|
|
|
|
category: e.category,
|
|
|
|
messageText: e.messageText as string,
|
|
|
|
source: SOURCE,
|
|
|
|
code: DEFAULT_ERROR_CODE
|
|
|
|
})));
|
2017-08-09 13:45:45 -07:00
|
|
|
} else {
|
|
|
|
rootNames.push(indexName !);
|
|
|
|
this.host = host = bundleHost;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const oldTsProgram = oldProgram ? oldProgram.getTsProgram() : undefined;
|
|
|
|
this.tsProgram = ts.createProgram(rootNames, options, host, oldTsProgram);
|
2017-08-02 11:20:07 -07:00
|
|
|
this.srcNames =
|
|
|
|
this.tsProgram.getSourceFiles()
|
|
|
|
.map(sf => sf.fileName)
|
|
|
|
.filter(f => !f.match(/\.ngfactory\.[\w.]+$|\.ngstyle\.[\w.]+$|\.ngsummary\.[\w.]+$/));
|
2017-07-13 14:25:17 -07:00
|
|
|
this.metadataCache = new LowerMetadataCache({quotedNames: true}, !!options.strictMetadataEmit);
|
2017-08-14 11:04:18 -07:00
|
|
|
this.aotCompilerHost =
|
|
|
|
new AotCompilerHostImpl(this.tsProgram, options, host, this.metadataCache);
|
2017-08-02 11:20:07 -07:00
|
|
|
|
|
|
|
const aotOptions = getAotCompilerOptions(options);
|
|
|
|
this.compiler = createAotCompiler(this.aotCompilerHost, aotOptions).compiler;
|
2017-06-09 14:50:57 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Program implementation
|
|
|
|
getTsProgram(): ts.Program { return this.programWithStubs; }
|
|
|
|
|
|
|
|
getTsOptionDiagnostics(cancellationToken?: ts.CancellationToken) {
|
|
|
|
return this.tsProgram.getOptionsDiagnostics(cancellationToken);
|
|
|
|
}
|
|
|
|
|
|
|
|
getNgOptionDiagnostics(cancellationToken?: ts.CancellationToken): Diagnostic[] {
|
2017-08-09 13:45:45 -07:00
|
|
|
return [...this._optionsDiagnostics, ...getNgOptionDiagnostics(this.options)];
|
2017-06-09 14:50:57 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
getTsSyntacticDiagnostics(sourceFile?: ts.SourceFile, cancellationToken?: ts.CancellationToken):
|
|
|
|
ts.Diagnostic[] {
|
|
|
|
return this.tsProgram.getSyntacticDiagnostics(sourceFile, cancellationToken);
|
|
|
|
}
|
|
|
|
|
|
|
|
getNgStructuralDiagnostics(cancellationToken?: ts.CancellationToken): Diagnostic[] {
|
|
|
|
return this.structuralDiagnostics;
|
|
|
|
}
|
|
|
|
|
|
|
|
getTsSemanticDiagnostics(sourceFile?: ts.SourceFile, cancellationToken?: ts.CancellationToken):
|
|
|
|
ts.Diagnostic[] {
|
|
|
|
return this.programWithStubs.getSemanticDiagnostics(sourceFile, cancellationToken);
|
|
|
|
}
|
|
|
|
|
|
|
|
getNgSemanticDiagnostics(fileName?: string, cancellationToken?: ts.CancellationToken):
|
|
|
|
Diagnostic[] {
|
|
|
|
const compilerDiagnostics = this.generatedFileDiagnostics;
|
|
|
|
|
|
|
|
// If we have diagnostics during the parser phase the type check phase is not meaningful so skip
|
|
|
|
// it.
|
|
|
|
if (compilerDiagnostics && compilerDiagnostics.length) return compilerDiagnostics;
|
|
|
|
|
|
|
|
return this.typeChecker.getDiagnostics(fileName, cancellationToken);
|
|
|
|
}
|
|
|
|
|
|
|
|
loadNgStructureAsync(): Promise<void> {
|
|
|
|
return this.compiler.analyzeModulesAsync(this.rootNames)
|
|
|
|
.catch(this.catchAnalysisError.bind(this))
|
|
|
|
.then(analyzedModules => {
|
|
|
|
if (this._analyzedModules) {
|
|
|
|
throw new Error('Angular structure loaded both synchronously and asynchronsly');
|
|
|
|
}
|
|
|
|
this._analyzedModules = analyzedModules;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-08-16 15:35:19 -07:00
|
|
|
emit({emitFlags = EmitFlags.Default, cancellationToken, customTransformers,
|
|
|
|
emitCallback = defaultEmitCallback}: {
|
|
|
|
emitFlags?: EmitFlags,
|
|
|
|
cancellationToken?: ts.CancellationToken,
|
|
|
|
customTransformers?: CustomTransformers,
|
|
|
|
emitCallback?: TsEmitCallback
|
|
|
|
}): ts.EmitResult {
|
2017-08-23 13:57:37 -07:00
|
|
|
return emitCallback({
|
2017-08-16 15:35:19 -07:00
|
|
|
program: this.programWithStubs,
|
|
|
|
host: this.host,
|
|
|
|
options: this.options,
|
|
|
|
targetSourceFile: undefined,
|
2017-08-23 13:57:37 -07:00
|
|
|
writeFile:
|
|
|
|
createWriteFileCallback(emitFlags, this.host, this.metadataCache, this.generatedFiles),
|
2017-08-16 15:35:19 -07:00
|
|
|
cancellationToken,
|
|
|
|
emitOnlyDtsFiles: (emitFlags & (EmitFlags.DTS | EmitFlags.JS)) == EmitFlags.DTS,
|
|
|
|
customTransformers: this.calculateTransforms(customTransformers)
|
|
|
|
});
|
2017-06-09 14:50:57 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Private members
|
|
|
|
private get analyzedModules(): NgAnalyzedModules {
|
|
|
|
return this._analyzedModules || (this._analyzedModules = this.analyzeModules());
|
|
|
|
}
|
|
|
|
|
|
|
|
private get structuralDiagnostics(): Diagnostic[] {
|
2017-07-27 16:13:16 -07:00
|
|
|
return this.analyzedModules && this._structuralDiagnostics;
|
2017-06-09 14:50:57 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
private get stubs(): GeneratedFile[] {
|
|
|
|
return this._stubs || (this._stubs = this.generateStubs());
|
|
|
|
}
|
|
|
|
|
|
|
|
private get stubFiles(): string[] {
|
|
|
|
return this._stubFiles ||
|
|
|
|
(this._stubFiles = this.stubs.reduce((files: string[], generatedFile) => {
|
|
|
|
if (generatedFile.source || (generatedFile.stmts && generatedFile.stmts.length)) {
|
|
|
|
return [...files, generatedFile.genFileUrl];
|
|
|
|
}
|
|
|
|
return files;
|
|
|
|
}, []));
|
|
|
|
}
|
|
|
|
|
|
|
|
private get programWithStubsHost(): ts.CompilerHost {
|
|
|
|
return this._programWithStubsHost || (this._programWithStubsHost = createProgramWithStubsHost(
|
|
|
|
this.stubs, this.tsProgram, this.host));
|
|
|
|
}
|
|
|
|
|
|
|
|
private get programWithStubs(): ts.Program {
|
|
|
|
return this._programWithStubs || (this._programWithStubs = this.createProgramWithStubs());
|
|
|
|
}
|
|
|
|
|
|
|
|
private get generatedFiles(): GeneratedFile[] {
|
2017-07-27 16:13:16 -07:00
|
|
|
return this._generatedFiles || (this._generatedFiles = this.generateFiles());
|
2017-06-09 14:50:57 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
private get typeChecker(): TypeChecker {
|
|
|
|
return (this._typeChecker && !this._typeChecker.partialResults) ?
|
|
|
|
this._typeChecker :
|
|
|
|
(this._typeChecker = this.createTypeChecker());
|
|
|
|
}
|
|
|
|
|
|
|
|
private get generatedFileDiagnostics(): Diagnostic[]|undefined {
|
|
|
|
return this.generatedFiles && this._generatedFileDiagnostics !;
|
|
|
|
}
|
|
|
|
|
2017-08-16 15:35:19 -07:00
|
|
|
private calculateTransforms(customTransformers?: CustomTransformers): ts.CustomTransformers {
|
2017-08-02 11:20:07 -07:00
|
|
|
const beforeTs: ts.TransformerFactory<ts.SourceFile>[] = [];
|
2017-07-13 14:25:17 -07:00
|
|
|
if (!this.options.disableExpressionLowering) {
|
2017-08-02 11:20:07 -07:00
|
|
|
beforeTs.push(getExpressionLoweringTransformFactory(this.metadataCache));
|
2017-07-13 14:25:17 -07:00
|
|
|
}
|
|
|
|
if (!this.options.skipTemplateCodegen) {
|
2017-08-02 11:20:07 -07:00
|
|
|
beforeTs.push(getAngularEmitterTransformFactory(this.generatedFiles));
|
2017-07-13 14:25:17 -07:00
|
|
|
}
|
2017-08-16 15:35:19 -07:00
|
|
|
if (customTransformers && customTransformers.beforeTs) {
|
|
|
|
beforeTs.push(...customTransformers.beforeTs);
|
|
|
|
}
|
|
|
|
const afterTs = customTransformers ? customTransformers.afterTs : undefined;
|
|
|
|
return {before: beforeTs, after: afterTs};
|
2017-07-13 14:25:17 -07:00
|
|
|
}
|
|
|
|
|
2017-06-09 14:50:57 -07:00
|
|
|
private catchAnalysisError(e: any): NgAnalyzedModules {
|
|
|
|
if (isSyntaxError(e)) {
|
|
|
|
const parserErrors = getParseErrors(e);
|
|
|
|
if (parserErrors && parserErrors.length) {
|
|
|
|
this._structuralDiagnostics =
|
|
|
|
parserErrors.map<Diagnostic>(e => ({
|
2017-08-18 14:03:59 -07:00
|
|
|
messageText: e.contextualMessage(),
|
2017-08-09 13:45:45 -07:00
|
|
|
category: ts.DiagnosticCategory.Error,
|
2017-08-18 14:03:59 -07:00
|
|
|
span: e.span,
|
|
|
|
source: SOURCE,
|
|
|
|
code: DEFAULT_ERROR_CODE
|
2017-06-09 14:50:57 -07:00
|
|
|
}));
|
|
|
|
} else {
|
2017-08-18 14:03:59 -07:00
|
|
|
this._structuralDiagnostics = [{
|
|
|
|
messageText: e.message,
|
|
|
|
category: ts.DiagnosticCategory.Error,
|
|
|
|
source: SOURCE,
|
|
|
|
code: DEFAULT_ERROR_CODE
|
|
|
|
}];
|
2017-06-09 14:50:57 -07:00
|
|
|
}
|
|
|
|
this._analyzedModules = emptyModules;
|
|
|
|
return emptyModules;
|
|
|
|
}
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
|
|
|
|
private analyzeModules() {
|
|
|
|
try {
|
|
|
|
return this.compiler.analyzeModulesSync(this.srcNames);
|
|
|
|
} catch (e) {
|
|
|
|
return this.catchAnalysisError(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private generateStubs() {
|
2017-08-22 16:17:44 -07:00
|
|
|
return this.options.skipTemplateCodegen ? [] : this.compiler.emitAllStubs(this.analyzedModules);
|
2017-06-09 14:50:57 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
private generateFiles() {
|
|
|
|
try {
|
|
|
|
// Always generate the files if requested to ensure we capture any diagnostic errors but only
|
|
|
|
// keep the results if we are not skipping template code generation.
|
|
|
|
const result = this.compiler.emitAllImpls(this.analyzedModules);
|
|
|
|
return this.options.skipTemplateCodegen ? [] : result;
|
|
|
|
} catch (e) {
|
|
|
|
if (isSyntaxError(e)) {
|
2017-08-18 14:03:59 -07:00
|
|
|
this._generatedFileDiagnostics = [{
|
|
|
|
messageText: e.message,
|
|
|
|
category: ts.DiagnosticCategory.Error,
|
|
|
|
source: SOURCE,
|
|
|
|
code: DEFAULT_ERROR_CODE
|
|
|
|
}];
|
2017-06-09 14:50:57 -07:00
|
|
|
return [];
|
|
|
|
}
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private createTypeChecker(): TypeChecker {
|
|
|
|
return new TypeChecker(
|
|
|
|
this.tsProgram, this.options, this.host, this.aotCompilerHost, this.options,
|
|
|
|
this.analyzedModules, this.generatedFiles);
|
|
|
|
}
|
|
|
|
|
|
|
|
private createProgramWithStubs(): ts.Program {
|
|
|
|
// If we are skipping code generation just use the original program.
|
|
|
|
// Otherwise, create a new program that includes the stub files.
|
|
|
|
return this.options.skipTemplateCodegen ?
|
|
|
|
this.tsProgram :
|
|
|
|
ts.createProgram(
|
|
|
|
[...this.rootNames, ...this.stubFiles], this.options, this.programWithStubsHost);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-14 11:04:18 -07:00
|
|
|
class AotCompilerHostImpl extends BaseAotCompilerHost<CompilerHost> {
|
|
|
|
moduleNameToFileName(m: string, containingFile: string): string|null {
|
|
|
|
return this.context.moduleNameToFileName(m, containingFile);
|
|
|
|
}
|
|
|
|
|
|
|
|
fileNameToModuleName(importedFile: string, containingFile: string): string|null {
|
|
|
|
return this.context.fileNameToModuleName(importedFile, containingFile);
|
|
|
|
}
|
2017-08-15 14:41:48 -07:00
|
|
|
|
2017-08-15 17:06:09 -07:00
|
|
|
resourceNameToFileName(resourceName: string, containingFile: string): string|null {
|
|
|
|
return this.context.resourceNameToFileName(resourceName, containingFile);
|
|
|
|
}
|
|
|
|
|
2017-08-15 14:41:48 -07:00
|
|
|
toSummaryFileName(fileName: string, referringSrcFileName: string): string {
|
|
|
|
return this.context.toSummaryFileName(fileName, referringSrcFileName);
|
|
|
|
}
|
|
|
|
|
|
|
|
fromSummaryFileName(fileName: string, referringLibFileName: string): string {
|
|
|
|
return this.context.fromSummaryFileName(fileName, referringLibFileName);
|
|
|
|
}
|
2017-08-14 11:04:18 -07:00
|
|
|
}
|
|
|
|
|
2017-06-09 14:50:57 -07:00
|
|
|
export function createProgram(
|
|
|
|
{rootNames, options, host, oldProgram}:
|
|
|
|
{rootNames: string[], options: CompilerOptions, host: CompilerHost, oldProgram?: Program}):
|
|
|
|
Program {
|
|
|
|
return new AngularCompilerProgram(rootNames, options, host, oldProgram);
|
|
|
|
}
|
|
|
|
|
2017-08-02 11:20:07 -07:00
|
|
|
// Compute the AotCompiler options
|
|
|
|
function getAotCompilerOptions(options: CompilerOptions): AotCompilerOptions {
|
2017-08-16 09:00:03 -07:00
|
|
|
let missingTranslation = core.MissingTranslationStrategy.Warning;
|
2017-08-02 11:20:07 -07:00
|
|
|
|
|
|
|
switch (options.i18nInMissingTranslations) {
|
|
|
|
case 'ignore':
|
2017-08-16 09:00:03 -07:00
|
|
|
missingTranslation = core.MissingTranslationStrategy.Ignore;
|
2017-08-02 11:20:07 -07:00
|
|
|
break;
|
|
|
|
case 'error':
|
2017-08-16 09:00:03 -07:00
|
|
|
missingTranslation = core.MissingTranslationStrategy.Error;
|
2017-08-02 11:20:07 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
let translations: string = '';
|
|
|
|
|
|
|
|
if (options.i18nInFile) {
|
|
|
|
if (!options.locale) {
|
|
|
|
throw new Error(`The translation file (${options.i18nInFile}) locale must be provided.`);
|
|
|
|
}
|
|
|
|
translations = fs.readFileSync(options.i18nInFile, 'utf8');
|
|
|
|
} else {
|
|
|
|
// No translations are provided, ignore any errors
|
|
|
|
// We still go through i18n to remove i18n attributes
|
2017-08-16 09:00:03 -07:00
|
|
|
missingTranslation = core.MissingTranslationStrategy.Ignore;
|
2017-08-02 11:20:07 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
locale: options.i18nInLocale,
|
|
|
|
i18nFormat: options.i18nInFormat || options.i18nOutFormat, translations, missingTranslation,
|
|
|
|
enableLegacyTemplate: options.enableLegacyTemplate,
|
|
|
|
enableSummariesForJit: true,
|
2017-07-28 15:58:28 +02:00
|
|
|
preserveWhitespaces: options.preserveWhitespaces,
|
2017-08-02 11:20:07 -07:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2017-06-09 14:50:57 -07:00
|
|
|
function writeMetadata(
|
2017-08-16 15:35:19 -07:00
|
|
|
host: ts.CompilerHost, emitFilePath: string, sourceFile: ts.SourceFile,
|
2017-08-23 13:57:37 -07:00
|
|
|
metadataCache: LowerMetadataCache, onError?: (message: string) => void) {
|
2017-06-09 14:50:57 -07:00
|
|
|
if (/\.js$/.test(emitFilePath)) {
|
|
|
|
const path = emitFilePath.replace(/\.js$/, '.metadata.json');
|
|
|
|
|
|
|
|
// Beginning with 2.1, TypeScript transforms the source tree before emitting it.
|
|
|
|
// We need the original, unmodified, tree which might be several levels back
|
|
|
|
// depending on the number of transforms performed. All SourceFile's prior to 2.1
|
|
|
|
// will appear to be the original source since they didn't include an original field.
|
|
|
|
let collectableFile = sourceFile;
|
|
|
|
while ((collectableFile as any).original) {
|
|
|
|
collectableFile = (collectableFile as any).original;
|
|
|
|
}
|
|
|
|
|
2017-07-13 14:25:17 -07:00
|
|
|
const metadata = metadataCache.getMetadata(collectableFile);
|
2017-06-09 14:50:57 -07:00
|
|
|
if (metadata) {
|
|
|
|
const metadataText = JSON.stringify([metadata]);
|
2017-08-23 13:57:37 -07:00
|
|
|
host.writeFile(path, metadataText, false, onError, [sourceFile]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function writeNgSummaryJson(
|
|
|
|
host: ts.CompilerHost, emitFilePath: string, sourceFile: ts.SourceFile,
|
|
|
|
generatedFilesByName: Map<string, GeneratedFile>, onError?: (message: string) => void) {
|
|
|
|
// Note: some files have an empty .ngfactory.js/.d.ts file but still need
|
|
|
|
// .ngsummary.json files (e.g. directives / pipes).
|
|
|
|
// We write the ngSummary when we try to emit the .ngfactory.js files
|
|
|
|
// and not the regular .js files as the latter are not emitted when
|
|
|
|
// we generate code for a npm library which ships .js / .d.ts / .metadata.json files.
|
|
|
|
if (/\.ngfactory.js$/.test(emitFilePath)) {
|
|
|
|
const emitPath = emitFilePath.replace(/\.ngfactory\.js$/, '.ngsummary.json');
|
|
|
|
const genFilePath = sourceFile.fileName.replace(/\.ngfactory\.ts$/, '.ngsummary.json');
|
|
|
|
const genFile = generatedFilesByName.get(genFilePath);
|
|
|
|
if (genFile) {
|
|
|
|
host.writeFile(emitPath, genFile.source !, false, onError, [sourceFile]);
|
2017-06-09 14:50:57 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function createWriteFileCallback(
|
2017-07-13 14:25:17 -07:00
|
|
|
emitFlags: EmitFlags, host: ts.CompilerHost, metadataCache: LowerMetadataCache,
|
2017-08-23 13:57:37 -07:00
|
|
|
generatedFiles: GeneratedFile[]) {
|
|
|
|
const generatedFilesByName = new Map<string, GeneratedFile>();
|
|
|
|
generatedFiles.forEach(f => generatedFilesByName.set(f.genFileUrl, f));
|
2017-08-02 11:20:07 -07:00
|
|
|
return (fileName: string, data: string, writeByteOrderMark: boolean,
|
|
|
|
onError?: (message: string) => void, sourceFiles?: ts.SourceFile[]) => {
|
2017-08-23 13:57:37 -07:00
|
|
|
const sourceFile = sourceFiles && sourceFiles.length == 1 ? sourceFiles[0] : null;
|
|
|
|
if (sourceFile) {
|
|
|
|
const isGenerated = GENERATED_FILES.test(fileName);
|
|
|
|
if (isGenerated) {
|
|
|
|
writeNgSummaryJson(host, fileName, sourceFile, generatedFilesByName, onError);
|
|
|
|
}
|
|
|
|
if (!isGenerated && (emitFlags & EmitFlags.Metadata)) {
|
|
|
|
writeMetadata(host, fileName, sourceFile, metadataCache, onError);
|
|
|
|
}
|
|
|
|
if (isGenerated) {
|
|
|
|
const genFile = generatedFilesByName.get(sourceFile.fileName);
|
|
|
|
if (!genFile || !genFile.stmts || !genFile.stmts.length) {
|
|
|
|
// Don't emit empty generated files
|
|
|
|
return;
|
|
|
|
}
|
2017-08-02 11:20:07 -07:00
|
|
|
}
|
|
|
|
}
|
2017-08-23 13:57:37 -07:00
|
|
|
host.writeFile(fileName, data, writeByteOrderMark, onError, sourceFiles);
|
2017-08-02 11:20:07 -07:00
|
|
|
};
|
2017-06-09 14:50:57 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
function getNgOptionDiagnostics(options: CompilerOptions): Diagnostic[] {
|
|
|
|
if (options.annotationsAs) {
|
|
|
|
switch (options.annotationsAs) {
|
|
|
|
case 'decorators':
|
|
|
|
case 'static fields':
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return [{
|
2017-08-18 14:03:59 -07:00
|
|
|
messageText:
|
2017-06-09 14:50:57 -07:00
|
|
|
'Angular compiler options "annotationsAs" only supports "static fields" and "decorators"',
|
2017-08-18 14:03:59 -07:00
|
|
|
category: ts.DiagnosticCategory.Error,
|
|
|
|
source: SOURCE,
|
|
|
|
code: DEFAULT_ERROR_CODE
|
2017-06-09 14:50:57 -07:00
|
|
|
}];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
|
|
|
function createProgramWithStubsHost(
|
|
|
|
generatedFiles: GeneratedFile[], originalProgram: ts.Program,
|
|
|
|
originalHost: ts.CompilerHost): ts.CompilerHost {
|
|
|
|
interface FileData {
|
2017-07-27 16:13:16 -07:00
|
|
|
g: GeneratedFile;
|
2017-06-09 14:50:57 -07:00
|
|
|
s?: ts.SourceFile;
|
|
|
|
}
|
|
|
|
return new class implements ts.CompilerHost {
|
|
|
|
private generatedFiles: Map<string, FileData>;
|
|
|
|
writeFile: ts.WriteFileCallback;
|
|
|
|
getCancellationToken: () => ts.CancellationToken;
|
|
|
|
getDefaultLibLocation: () => string;
|
|
|
|
trace: (s: string) => void;
|
|
|
|
getDirectories: (path: string) => string[];
|
|
|
|
directoryExists: (directoryName: string) => boolean;
|
|
|
|
constructor() {
|
|
|
|
this.generatedFiles =
|
|
|
|
new Map(generatedFiles.filter(g => g.source || (g.stmts && g.stmts.length))
|
|
|
|
.map<[string, FileData]>(g => [g.genFileUrl, {g}]));
|
|
|
|
this.writeFile = originalHost.writeFile;
|
|
|
|
if (originalHost.getDirectories) {
|
|
|
|
this.getDirectories = path => originalHost.getDirectories !(path);
|
|
|
|
}
|
|
|
|
if (originalHost.directoryExists) {
|
|
|
|
this.directoryExists = directoryName => originalHost.directoryExists !(directoryName);
|
|
|
|
}
|
|
|
|
if (originalHost.getCancellationToken) {
|
|
|
|
this.getCancellationToken = () => originalHost.getCancellationToken !();
|
|
|
|
}
|
|
|
|
if (originalHost.getDefaultLibLocation) {
|
|
|
|
this.getDefaultLibLocation = () => originalHost.getDefaultLibLocation !();
|
|
|
|
}
|
|
|
|
if (originalHost.trace) {
|
|
|
|
this.trace = s => originalHost.trace !(s);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
getSourceFile(
|
|
|
|
fileName: string, languageVersion: ts.ScriptTarget,
|
|
|
|
onError?: ((message: string) => void)|undefined): ts.SourceFile {
|
|
|
|
const data = this.generatedFiles.get(fileName);
|
|
|
|
if (data) {
|
|
|
|
return data.s || (data.s = ts.createSourceFile(
|
|
|
|
fileName, data.g.source || toTypeScript(data.g), languageVersion));
|
|
|
|
}
|
|
|
|
return originalProgram.getSourceFile(fileName) ||
|
|
|
|
originalHost.getSourceFile(fileName, languageVersion, onError);
|
|
|
|
}
|
|
|
|
readFile(fileName: string): string {
|
|
|
|
const data = this.generatedFiles.get(fileName);
|
|
|
|
if (data) {
|
|
|
|
return data.g.source || toTypeScript(data.g);
|
|
|
|
}
|
|
|
|
return originalHost.readFile(fileName);
|
|
|
|
}
|
|
|
|
getDefaultLibFileName = (options: ts.CompilerOptions) =>
|
|
|
|
originalHost.getDefaultLibFileName(options);
|
|
|
|
getCurrentDirectory = () => originalHost.getCurrentDirectory();
|
|
|
|
getCanonicalFileName = (fileName: string) => originalHost.getCanonicalFileName(fileName);
|
|
|
|
useCaseSensitiveFileNames = () => originalHost.useCaseSensitiveFileNames();
|
|
|
|
getNewLine = () => originalHost.getNewLine();
|
2017-07-07 17:29:39 -06:00
|
|
|
realPath = (p: string) => p;
|
2017-06-09 14:50:57 -07:00
|
|
|
fileExists = (fileName: string) =>
|
|
|
|
this.generatedFiles.has(fileName) || originalHost.fileExists(fileName);
|
|
|
|
};
|
2017-07-28 15:58:28 +02:00
|
|
|
}
|