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-09-20 16:27:29 -07:00
|
|
|
import {AotCompiler, AotCompilerHost, AotCompilerOptions, EmitterVisitorContext, GeneratedFile, MessageBundle, NgAnalyzedFile, NgAnalyzedModules, ParseSourceSpan, Serializer, TypeScriptEmitter, Xliff, Xliff2, Xmb, core, createAotCompiler, getParseErrors, isSyntaxError} from '@angular/compiler';
|
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-09-11 15:18:19 -07:00
|
|
|
import {TypeCheckHost, translateDiagnostics} from '../diagnostics/translate_diagnostics';
|
2017-09-12 09:40:28 -07:00
|
|
|
import {ModuleMetadata, createBundleIndexHost} from '../metadata/index';
|
2017-06-09 14:50:57 -07:00
|
|
|
|
2017-10-20 09:46:41 -07:00
|
|
|
import {CompilerHost, CompilerOptions, CustomTransformers, DEFAULT_ERROR_CODE, Diagnostic, EmitFlags, LazyRoute, LibrarySummary, Program, SOURCE, TsEmitArguments, TsEmitCallback} from './api';
|
2017-09-21 18:05:07 -07:00
|
|
|
import {CodeGenerator, TsCompilerAotCompilerTypeCheckHostAdapter, getOriginalReferences} from './compiler_host';
|
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-10-26 09:45:01 -07:00
|
|
|
import {GENERATED_FILES, StructureIsReused, createMessageDiagnostic, isInRootDir, ngToTsDiagnostic, tsStructureIsReused} from './util';
|
2017-10-17 16:51:04 -07:00
|
|
|
|
2017-09-29 15:02:11 -07:00
|
|
|
|
2017-10-20 09:46:41 -07:00
|
|
|
|
2017-09-29 15:02:11 -07:00
|
|
|
/**
|
|
|
|
* Maximum number of files that are emitable via calling ts.Program.emit
|
|
|
|
* passing individual targetSourceFiles.
|
|
|
|
*/
|
|
|
|
const MAX_FILE_COUNT_FOR_SINGLE_FILE_EMIT = 20;
|
2017-06-09 14:50:57 -07:00
|
|
|
|
|
|
|
const emptyModules: NgAnalyzedModules = {
|
|
|
|
ngModules: [],
|
|
|
|
ngModuleByPipeOrDirective: new Map(),
|
|
|
|
files: []
|
|
|
|
};
|
|
|
|
|
2017-10-12 16:09:49 -07:00
|
|
|
const defaultEmitCallback: TsEmitCallback =
|
|
|
|
({program, targetSourceFile, writeFile, cancellationToken, emitOnlyDtsFiles,
|
|
|
|
customTransformers}) =>
|
|
|
|
program.emit(
|
|
|
|
targetSourceFile, writeFile, cancellationToken, emitOnlyDtsFiles, customTransformers);
|
2017-08-16 15:35:19 -07:00
|
|
|
|
2017-06-09 14:50:57 -07:00
|
|
|
class AngularCompilerProgram implements Program {
|
2017-07-13 14:25:17 -07:00
|
|
|
private metadataCache: LowerMetadataCache;
|
2017-09-29 15:02:11 -07:00
|
|
|
private oldProgramLibrarySummaries: Map<string, LibrarySummary>|undefined;
|
|
|
|
private oldProgramEmittedGeneratedFiles: Map<string, GeneratedFile>|undefined;
|
2017-10-03 09:53:58 -07:00
|
|
|
private oldProgramEmittedSourceFiles: Map<string, ts.SourceFile>|undefined;
|
2017-09-19 11:43:34 -07:00
|
|
|
// Note: This will be cleared out as soon as we create the _tsProgram
|
|
|
|
private oldTsProgram: ts.Program|undefined;
|
2017-09-21 18:05:07 -07:00
|
|
|
private emittedLibrarySummaries: LibrarySummary[]|undefined;
|
2017-09-29 15:02:11 -07:00
|
|
|
private emittedGeneratedFiles: GeneratedFile[]|undefined;
|
2017-10-03 09:53:58 -07:00
|
|
|
private emittedSourceFiles: ts.SourceFile[]|undefined;
|
2017-09-19 11:41:47 -07:00
|
|
|
|
2017-06-09 14:50:57 -07:00
|
|
|
// Lazily initialized fields
|
2017-09-12 09:40:28 -07:00
|
|
|
private _compiler: AotCompiler;
|
2017-10-20 09:46:41 -07:00
|
|
|
private _hostAdapter: TsCompilerAotCompilerTypeCheckHostAdapter;
|
2017-09-12 09:40:28 -07:00
|
|
|
private _tsProgram: ts.Program;
|
2017-06-09 14:50:57 -07:00
|
|
|
private _analyzedModules: NgAnalyzedModules|undefined;
|
2017-09-12 09:40:28 -07:00
|
|
|
private _structuralDiagnostics: Diagnostic[]|undefined;
|
2017-06-09 14:50:57 -07:00
|
|
|
private _programWithStubs: ts.Program|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-10-12 16:09:49 -07:00
|
|
|
private oldProgram?: Program) {
|
2017-09-12 09:40:28 -07:00
|
|
|
const [major, minor] = ts.version.split('.');
|
|
|
|
if (Number(major) < 2 || (Number(major) === 2 && Number(minor) < 4)) {
|
|
|
|
throw new Error('The Angular Compiler requires TypeScript >= 2.4.');
|
|
|
|
}
|
2017-10-12 16:09:49 -07:00
|
|
|
this.oldTsProgram = oldProgram ? oldProgram.getTsProgram() : undefined;
|
2017-09-19 11:43:34 -07:00
|
|
|
if (oldProgram) {
|
2017-10-12 16:09:49 -07:00
|
|
|
this.oldProgramLibrarySummaries = oldProgram.getLibrarySummaries();
|
|
|
|
this.oldProgramEmittedGeneratedFiles = oldProgram.getEmittedGeneratedFiles();
|
|
|
|
this.oldProgramEmittedSourceFiles = oldProgram.getEmittedSourceFiles();
|
2017-09-19 11:43:34 -07:00
|
|
|
}
|
2017-09-12 09:40:28 -07:00
|
|
|
|
2017-09-12 15:53:17 -07:00
|
|
|
if (options.flatModuleOutFile) {
|
2017-08-09 13:45:45 -07:00
|
|
|
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 !);
|
2017-09-12 09:40:28 -07:00
|
|
|
this.host = bundleHost;
|
2017-08-09 13:45:45 -07:00
|
|
|
}
|
|
|
|
}
|
2017-07-13 14:25:17 -07:00
|
|
|
this.metadataCache = new LowerMetadataCache({quotedNames: true}, !!options.strictMetadataEmit);
|
2017-06-09 14:50:57 -07:00
|
|
|
}
|
|
|
|
|
2017-09-29 15:02:11 -07:00
|
|
|
getLibrarySummaries(): Map<string, LibrarySummary> {
|
|
|
|
const result = new Map<string, LibrarySummary>();
|
|
|
|
if (this.oldProgramLibrarySummaries) {
|
|
|
|
this.oldProgramLibrarySummaries.forEach((summary, fileName) => result.set(fileName, summary));
|
|
|
|
}
|
2017-09-21 18:05:07 -07:00
|
|
|
if (this.emittedLibrarySummaries) {
|
2017-09-29 15:02:11 -07:00
|
|
|
this.emittedLibrarySummaries.forEach(
|
|
|
|
(summary, fileName) => result.set(summary.fileName, summary));
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
getEmittedGeneratedFiles(): Map<string, GeneratedFile> {
|
|
|
|
const result = new Map<string, GeneratedFile>();
|
|
|
|
if (this.oldProgramEmittedGeneratedFiles) {
|
|
|
|
this.oldProgramEmittedGeneratedFiles.forEach(
|
|
|
|
(genFile, fileName) => result.set(fileName, genFile));
|
|
|
|
}
|
|
|
|
if (this.emittedGeneratedFiles) {
|
2017-10-12 16:09:49 -07:00
|
|
|
this.emittedGeneratedFiles.forEach((genFile) => result.set(genFile.genFileUrl, genFile));
|
2017-09-19 11:43:34 -07:00
|
|
|
}
|
2017-09-21 18:05:07 -07:00
|
|
|
return result;
|
2017-09-19 11:43:34 -07:00
|
|
|
}
|
|
|
|
|
2017-10-03 09:53:58 -07:00
|
|
|
getEmittedSourceFiles(): Map<string, ts.SourceFile> {
|
|
|
|
const result = new Map<string, ts.SourceFile>();
|
|
|
|
if (this.oldProgramEmittedSourceFiles) {
|
|
|
|
this.oldProgramEmittedSourceFiles.forEach((sf, fileName) => result.set(fileName, sf));
|
|
|
|
}
|
|
|
|
if (this.emittedSourceFiles) {
|
2017-10-12 16:09:49 -07:00
|
|
|
this.emittedSourceFiles.forEach((sf) => result.set(sf.fileName, sf));
|
2017-10-03 09:53:58 -07:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
getTsProgram(): ts.Program { return this.tsProgram; }
|
2017-06-09 14:50:57 -07:00
|
|
|
|
|
|
|
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[] {
|
2017-10-26 09:45:01 -07:00
|
|
|
const sourceFiles = sourceFile ? [sourceFile] : this.tsProgram.getSourceFiles();
|
2017-10-05 13:48:40 -07:00
|
|
|
let diags: ts.Diagnostic[] = [];
|
2017-10-26 09:45:01 -07:00
|
|
|
sourceFiles.forEach(sf => {
|
2017-10-05 13:48:40 -07:00
|
|
|
if (!GENERATED_FILES.test(sf.fileName)) {
|
|
|
|
diags.push(...this.tsProgram.getSemanticDiagnostics(sf, cancellationToken));
|
|
|
|
}
|
2017-10-12 16:09:49 -07:00
|
|
|
});
|
2017-10-05 13:48:40 -07:00
|
|
|
return diags;
|
2017-06-09 14:50:57 -07:00
|
|
|
}
|
|
|
|
|
2017-10-12 16:09:49 -07:00
|
|
|
getNgSemanticDiagnostics(fileName?: string, cancellationToken?: ts.CancellationToken):
|
2017-06-09 14:50:57 -07:00
|
|
|
Diagnostic[] {
|
2017-10-05 13:48:40 -07:00
|
|
|
let diags: ts.Diagnostic[] = [];
|
2017-10-12 16:09:49 -07:00
|
|
|
this.tsProgram.getSourceFiles().forEach(sf => {
|
|
|
|
if (GENERATED_FILES.test(sf.fileName) && !sf.isDeclarationFile) {
|
|
|
|
diags.push(...this.tsProgram.getSemanticDiagnostics(sf, cancellationToken));
|
2017-10-05 13:48:40 -07:00
|
|
|
}
|
2017-10-12 16:09:49 -07:00
|
|
|
});
|
2017-10-20 09:46:41 -07:00
|
|
|
const {ng} = translateDiagnostics(this.hostAdapter, diags);
|
2017-10-05 13:48:40 -07:00
|
|
|
return ng;
|
2017-06-09 14:50:57 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
loadNgStructureAsync(): Promise<void> {
|
2017-09-12 09:40:28 -07:00
|
|
|
if (this._analyzedModules) {
|
|
|
|
throw new Error('Angular structure already loaded');
|
|
|
|
}
|
2017-10-26 15:24:54 -07:00
|
|
|
return Promise.resolve()
|
|
|
|
.then(() => {
|
|
|
|
const {tmpProgram, sourceFiles, rootNames} = this._createProgramWithBasicStubs();
|
|
|
|
return this.compiler.loadFilesAsync(sourceFiles).then(analyzedModules => {
|
|
|
|
if (this._analyzedModules) {
|
|
|
|
throw new Error('Angular structure loaded both synchronously and asynchronsly');
|
|
|
|
}
|
|
|
|
this._updateProgramWithTypeCheckStubs(tmpProgram, analyzedModules, rootNames);
|
|
|
|
});
|
|
|
|
})
|
|
|
|
.catch(e => this._createProgramOnError(e));
|
2017-06-09 14:50:57 -07:00
|
|
|
}
|
|
|
|
|
2017-10-20 09:46:41 -07:00
|
|
|
listLazyRoutes(route?: string): LazyRoute[] {
|
|
|
|
// Note: Don't analyzedModules if a route is given
|
|
|
|
// to be fast enough.
|
|
|
|
return this.compiler.listLazyRoutes(route, route ? undefined : this.analyzedModules);
|
|
|
|
}
|
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
emit(
|
2017-10-12 16:09:49 -07:00
|
|
|
{emitFlags = EmitFlags.Default, cancellationToken, customTransformers,
|
2017-09-12 09:40:28 -07:00
|
|
|
emitCallback = defaultEmitCallback}: {
|
|
|
|
emitFlags?: EmitFlags,
|
|
|
|
cancellationToken?: ts.CancellationToken,
|
|
|
|
customTransformers?: CustomTransformers,
|
2017-10-12 16:09:49 -07:00
|
|
|
emitCallback?: TsEmitCallback
|
2017-09-12 09:40:28 -07:00
|
|
|
} = {}): ts.EmitResult {
|
2017-09-29 15:02:11 -07:00
|
|
|
const emitStart = Date.now();
|
2017-09-12 15:53:17 -07:00
|
|
|
if (emitFlags & EmitFlags.I18nBundle) {
|
|
|
|
const locale = this.options.i18nOutLocale || null;
|
|
|
|
const file = this.options.i18nOutFile || null;
|
|
|
|
const format = this.options.i18nOutFormat || null;
|
|
|
|
const bundle = this.compiler.emitMessageBundle(this.analyzedModules, locale);
|
|
|
|
i18nExtract(format, file, this.host, this.options, bundle);
|
|
|
|
}
|
2017-09-19 11:41:47 -07:00
|
|
|
if ((emitFlags & (EmitFlags.JS | EmitFlags.DTS | EmitFlags.Metadata | EmitFlags.Codegen)) ===
|
2017-09-12 09:40:28 -07:00
|
|
|
0) {
|
2017-10-12 16:09:49 -07:00
|
|
|
return {emitSkipped: true, diagnostics: [], emittedFiles: []};
|
2017-09-12 09:40:28 -07:00
|
|
|
}
|
2017-10-12 16:09:49 -07:00
|
|
|
let {genFiles, genDiags} = this.generateFilesForEmit(emitFlags);
|
|
|
|
if (genDiags.length) {
|
|
|
|
return {
|
|
|
|
diagnostics: genDiags,
|
|
|
|
emitSkipped: true,
|
|
|
|
emittedFiles: [],
|
|
|
|
};
|
|
|
|
}
|
|
|
|
this.emittedGeneratedFiles = genFiles;
|
2017-09-21 18:05:07 -07:00
|
|
|
const outSrcMapping: Array<{sourceFile: ts.SourceFile, outFileName: string}> = [];
|
2017-10-12 16:09:49 -07:00
|
|
|
const genFileByFileName = new Map<string, GeneratedFile>();
|
|
|
|
genFiles.forEach(genFile => genFileByFileName.set(genFile.genFileUrl, genFile));
|
|
|
|
this.emittedLibrarySummaries = [];
|
|
|
|
const emittedSourceFiles = [] as ts.SourceFile[];
|
2017-09-21 18:05:07 -07:00
|
|
|
const writeTsFile: ts.WriteFileCallback =
|
|
|
|
(outFileName, outData, writeByteOrderMark, onError?, sourceFiles?) => {
|
|
|
|
const sourceFile = sourceFiles && sourceFiles.length == 1 ? sourceFiles[0] : null;
|
|
|
|
let genFile: GeneratedFile|undefined;
|
|
|
|
if (sourceFile) {
|
|
|
|
outSrcMapping.push({outFileName: outFileName, sourceFile});
|
2017-10-12 16:09:49 -07:00
|
|
|
genFile = genFileByFileName.get(sourceFile.fileName);
|
|
|
|
if (!sourceFile.isDeclarationFile && !GENERATED_FILES.test(sourceFile.fileName)) {
|
|
|
|
// Note: sourceFile is the transformed sourcefile, not the original one!
|
|
|
|
emittedSourceFiles.push(this.tsProgram.getSourceFile(sourceFile.fileName));
|
2017-10-03 09:53:58 -07:00
|
|
|
}
|
2017-09-21 18:05:07 -07:00
|
|
|
}
|
2017-10-12 16:09:49 -07:00
|
|
|
this.writeFile(outFileName, outData, writeByteOrderMark, onError, genFile, sourceFiles);
|
2017-09-21 18:05:07 -07:00
|
|
|
};
|
2017-10-12 16:09:49 -07:00
|
|
|
const tsCustomTansformers = this.calculateTransforms(genFileByFileName, customTransformers);
|
2017-09-29 15:02:11 -07:00
|
|
|
const emitOnlyDtsFiles = (emitFlags & (EmitFlags.DTS | EmitFlags.JS)) == EmitFlags.DTS;
|
2017-09-20 09:54:47 -07:00
|
|
|
// Restore the original references before we emit so TypeScript doesn't emit
|
|
|
|
// a reference to the .d.ts file.
|
|
|
|
const augmentedReferences = new Map<ts.SourceFile, ts.FileReference[]>();
|
2017-10-12 16:09:49 -07:00
|
|
|
for (const sourceFile of this.tsProgram.getSourceFiles()) {
|
2017-09-20 09:54:47 -07:00
|
|
|
const originalReferences = getOriginalReferences(sourceFile);
|
|
|
|
if (originalReferences) {
|
|
|
|
augmentedReferences.set(sourceFile, sourceFile.referencedFiles);
|
|
|
|
sourceFile.referencedFiles = originalReferences;
|
|
|
|
}
|
|
|
|
}
|
2017-10-12 16:09:49 -07:00
|
|
|
const genTsFiles: GeneratedFile[] = [];
|
|
|
|
const genJsonFiles: GeneratedFile[] = [];
|
|
|
|
genFiles.forEach(gf => {
|
|
|
|
if (gf.stmts) {
|
|
|
|
genTsFiles.push(gf);
|
|
|
|
}
|
|
|
|
if (gf.source) {
|
|
|
|
genJsonFiles.push(gf);
|
|
|
|
}
|
|
|
|
});
|
2017-09-20 09:54:47 -07:00
|
|
|
let emitResult: ts.EmitResult;
|
2017-10-12 16:09:49 -07:00
|
|
|
let emittedUserTsCount: number;
|
2017-09-20 09:54:47 -07:00
|
|
|
try {
|
2017-10-12 16:09:49 -07:00
|
|
|
const sourceFilesToEmit = this.getSourceFilesForEmit();
|
|
|
|
if (sourceFilesToEmit &&
|
|
|
|
(sourceFilesToEmit.length + genTsFiles.length) < MAX_FILE_COUNT_FOR_SINGLE_FILE_EMIT) {
|
|
|
|
const fileNamesToEmit =
|
|
|
|
[...sourceFilesToEmit.map(sf => sf.fileName), ...genTsFiles.map(gf => gf.genFileUrl)];
|
|
|
|
emitResult = mergeEmitResults(
|
|
|
|
fileNamesToEmit.map((fileName) => emitResult = emitCallback({
|
|
|
|
program: this.tsProgram,
|
|
|
|
host: this.host,
|
|
|
|
options: this.options,
|
|
|
|
writeFile: writeTsFile, emitOnlyDtsFiles,
|
|
|
|
customTransformers: tsCustomTansformers,
|
|
|
|
targetSourceFile: this.tsProgram.getSourceFile(fileName),
|
|
|
|
})));
|
|
|
|
emittedUserTsCount = sourceFilesToEmit.length;
|
|
|
|
} else {
|
|
|
|
emitResult = emitCallback({
|
|
|
|
program: this.tsProgram,
|
|
|
|
host: this.host,
|
|
|
|
options: this.options,
|
|
|
|
writeFile: writeTsFile, emitOnlyDtsFiles,
|
|
|
|
customTransformers: tsCustomTansformers
|
|
|
|
});
|
|
|
|
emittedUserTsCount = this.tsProgram.getSourceFiles().length - genTsFiles.length;
|
|
|
|
}
|
2017-09-20 09:54:47 -07:00
|
|
|
} finally {
|
|
|
|
// Restore the references back to the augmented value to ensure that the
|
|
|
|
// checks that TypeScript makes for project structure reuse will succeed.
|
|
|
|
for (const [sourceFile, references] of Array.from(augmentedReferences)) {
|
|
|
|
sourceFile.referencedFiles = references;
|
|
|
|
}
|
|
|
|
}
|
2017-10-12 16:09:49 -07:00
|
|
|
this.emittedSourceFiles = emittedSourceFiles;
|
2017-10-26 09:45:01 -07:00
|
|
|
// translate the diagnostics in the emitResult as well.
|
|
|
|
const translatedEmitDiags = translateDiagnostics(this.hostAdapter, emitResult.diagnostics);
|
|
|
|
emitResult.diagnostics = translatedEmitDiags.ts.concat(
|
|
|
|
this.structuralDiagnostics.concat(translatedEmitDiags.ng).map(ngToTsDiagnostic));
|
2017-09-19 11:41:47 -07:00
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
if (!outSrcMapping.length) {
|
|
|
|
// if no files were emitted by TypeScript, also don't emit .json files
|
2017-09-29 15:02:11 -07:00
|
|
|
emitResult.diagnostics.push(createMessageDiagnostic(`Emitted no files.`));
|
2017-09-12 09:40:28 -07:00
|
|
|
return emitResult;
|
|
|
|
}
|
|
|
|
|
2017-09-28 11:42:58 -07:00
|
|
|
let sampleSrcFileName: string|undefined;
|
|
|
|
let sampleOutFileName: string|undefined;
|
|
|
|
if (outSrcMapping.length) {
|
|
|
|
sampleSrcFileName = outSrcMapping[0].sourceFile.fileName;
|
|
|
|
sampleOutFileName = outSrcMapping[0].outFileName;
|
|
|
|
}
|
|
|
|
const srcToOutPath =
|
|
|
|
createSrcToOutPathMapper(this.options.outDir, sampleSrcFileName, sampleOutFileName);
|
2017-09-19 11:41:47 -07:00
|
|
|
if (emitFlags & EmitFlags.Codegen) {
|
2017-10-12 16:09:49 -07:00
|
|
|
genJsonFiles.forEach(gf => {
|
|
|
|
const outFileName = srcToOutPath(gf.genFileUrl);
|
|
|
|
this.writeFile(outFileName, gf.source !, false, undefined, gf);
|
|
|
|
});
|
2017-09-12 09:40:28 -07:00
|
|
|
}
|
2017-10-12 16:09:49 -07:00
|
|
|
let metadataJsonCount = 0;
|
2017-09-12 09:40:28 -07:00
|
|
|
if (emitFlags & EmitFlags.Metadata) {
|
2017-10-12 16:09:49 -07:00
|
|
|
this.tsProgram.getSourceFiles().forEach(sf => {
|
|
|
|
if (!sf.isDeclarationFile && !GENERATED_FILES.test(sf.fileName)) {
|
|
|
|
metadataJsonCount++;
|
|
|
|
const metadata = this.metadataCache.getMetadata(sf);
|
|
|
|
const metadataText = JSON.stringify([metadata]);
|
|
|
|
const outFileName = srcToOutPath(sf.fileName.replace(/\.ts$/, '.metadata.json'));
|
|
|
|
this.writeFile(outFileName, metadataText, false, undefined, undefined, [sf]);
|
|
|
|
}
|
|
|
|
});
|
2017-09-12 15:53:17 -07:00
|
|
|
}
|
2017-09-29 15:02:11 -07:00
|
|
|
const emitEnd = Date.now();
|
|
|
|
if (this.options.diagnostics) {
|
|
|
|
emitResult.diagnostics.push(createMessageDiagnostic([
|
|
|
|
`Emitted in ${emitEnd - emitStart}ms`,
|
2017-10-12 16:09:49 -07:00
|
|
|
`- ${emittedUserTsCount} user ts files`,
|
|
|
|
`- ${genTsFiles.length} generated ts files`,
|
|
|
|
`- ${genJsonFiles.length + metadataJsonCount} generated json files`,
|
2017-09-29 15:02:11 -07:00
|
|
|
].join('\n')));
|
|
|
|
}
|
2017-09-12 09:40:28 -07:00
|
|
|
return emitResult;
|
2017-06-09 14:50:57 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Private members
|
2017-09-12 09:40:28 -07:00
|
|
|
private get compiler(): AotCompiler {
|
|
|
|
if (!this._compiler) {
|
2017-10-20 09:46:41 -07:00
|
|
|
this._createCompiler();
|
2017-09-12 09:40:28 -07:00
|
|
|
}
|
|
|
|
return this._compiler !;
|
2017-06-09 14:50:57 -07:00
|
|
|
}
|
|
|
|
|
2017-10-20 09:46:41 -07:00
|
|
|
private get hostAdapter(): TsCompilerAotCompilerTypeCheckHostAdapter {
|
|
|
|
if (!this._hostAdapter) {
|
|
|
|
this._createCompiler();
|
|
|
|
}
|
|
|
|
return this._hostAdapter !;
|
|
|
|
}
|
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
private get analyzedModules(): NgAnalyzedModules {
|
|
|
|
if (!this._analyzedModules) {
|
|
|
|
this.initSync();
|
|
|
|
}
|
|
|
|
return this._analyzedModules !;
|
2017-06-09 14:50:57 -07:00
|
|
|
}
|
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
private get structuralDiagnostics(): Diagnostic[] {
|
|
|
|
if (!this._structuralDiagnostics) {
|
|
|
|
this.initSync();
|
|
|
|
}
|
|
|
|
return this._structuralDiagnostics !;
|
2017-06-09 14:50:57 -07:00
|
|
|
}
|
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
private get tsProgram(): ts.Program {
|
|
|
|
if (!this._tsProgram) {
|
|
|
|
this.initSync();
|
|
|
|
}
|
|
|
|
return this._tsProgram !;
|
2017-06-09 14:50:57 -07:00
|
|
|
}
|
|
|
|
|
2017-10-12 16:09:49 -07:00
|
|
|
private calculateTransforms(
|
|
|
|
genFiles: Map<string, GeneratedFile>,
|
|
|
|
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-10-24 11:26:04 -07:00
|
|
|
beforeTs.push(getExpressionLoweringTransformFactory(this.metadataCache, this.tsProgram));
|
2017-07-13 14:25:17 -07:00
|
|
|
}
|
2017-10-12 16:09:49 -07:00
|
|
|
beforeTs.push(getAngularEmitterTransformFactory(genFiles));
|
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-09-12 09:40:28 -07:00
|
|
|
private initSync() {
|
|
|
|
if (this._analyzedModules) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
try {
|
2017-10-26 15:24:54 -07:00
|
|
|
const {tmpProgram, sourceFiles, rootNames} = this._createProgramWithBasicStubs();
|
|
|
|
const analyzedModules = this.compiler.loadFilesSync(sourceFiles);
|
|
|
|
this._updateProgramWithTypeCheckStubs(tmpProgram, analyzedModules, rootNames);
|
2017-09-12 09:40:28 -07:00
|
|
|
} catch (e) {
|
2017-10-26 15:24:54 -07:00
|
|
|
this._createProgramOnError(e);
|
2017-09-12 09:40:28 -07:00
|
|
|
}
|
2017-10-20 09:46:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
private _createCompiler() {
|
|
|
|
const codegen: CodeGenerator = {
|
|
|
|
generateFile: (genFileName, baseFileName) =>
|
|
|
|
this._compiler.emitBasicStub(genFileName, baseFileName),
|
|
|
|
findGeneratedFileNames: (fileName) => this._compiler.findGeneratedFileNames(fileName),
|
|
|
|
};
|
|
|
|
|
|
|
|
this._hostAdapter = new TsCompilerAotCompilerTypeCheckHostAdapter(
|
|
|
|
this.rootNames, this.options, this.host, this.metadataCache, codegen,
|
|
|
|
this.oldProgramLibrarySummaries);
|
|
|
|
const aotOptions = getAotCompilerOptions(this.options);
|
|
|
|
this._structuralDiagnostics = [];
|
2017-10-24 12:52:14 -07:00
|
|
|
const errorCollector =
|
|
|
|
(this.options.collectAllErrors || this.options.fullTemplateTypeCheck) ? (err: any) => {
|
|
|
|
this._structuralDiagnostics !.push({
|
|
|
|
messageText: err.toString(),
|
|
|
|
category: ts.DiagnosticCategory.Error,
|
|
|
|
source: SOURCE,
|
|
|
|
code: DEFAULT_ERROR_CODE
|
|
|
|
});
|
|
|
|
} : undefined;
|
2017-10-20 09:46:41 -07:00
|
|
|
this._compiler = createAotCompiler(this._hostAdapter, aotOptions, errorCollector).compiler;
|
2017-09-12 09:40:28 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
private _createProgramWithBasicStubs(): {
|
|
|
|
tmpProgram: ts.Program,
|
2017-09-20 16:27:29 -07:00
|
|
|
rootNames: string[],
|
2017-09-21 18:05:07 -07:00
|
|
|
sourceFiles: string[],
|
2017-09-12 09:40:28 -07:00
|
|
|
} {
|
|
|
|
if (this._analyzedModules) {
|
|
|
|
throw new Error(`Internal Error: already initalized!`);
|
|
|
|
}
|
2017-09-19 11:43:34 -07:00
|
|
|
// Note: This is important to not produce a memory leak!
|
|
|
|
const oldTsProgram = this.oldTsProgram;
|
|
|
|
this.oldTsProgram = undefined;
|
2017-09-21 18:05:07 -07:00
|
|
|
|
|
|
|
const codegen: CodeGenerator = {
|
|
|
|
generateFile: (genFileName, baseFileName) =>
|
2017-10-20 09:46:41 -07:00
|
|
|
this.compiler.emitBasicStub(genFileName, baseFileName),
|
|
|
|
findGeneratedFileNames: (fileName) => this.compiler.findGeneratedFileNames(fileName),
|
2017-09-12 09:40:28 -07:00
|
|
|
};
|
2017-09-21 18:05:07 -07:00
|
|
|
|
2017-09-12 09:40:28 -07:00
|
|
|
|
2017-10-20 09:46:41 -07:00
|
|
|
let rootNames = this.rootNames;
|
|
|
|
if (this.options.generateCodeForLibraries !== false) {
|
2017-10-26 15:24:54 -07:00
|
|
|
// if we should generateCodeForLibraries, never include
|
2017-10-20 09:46:41 -07:00
|
|
|
// generated files in the program as otherwise we will
|
|
|
|
// ovewrite them and typescript will report the error
|
|
|
|
// TS5055: Cannot write file ... because it would overwrite input file.
|
|
|
|
rootNames = this.rootNames.filter(fn => !GENERATED_FILES.test(fn));
|
|
|
|
}
|
2017-09-20 16:27:29 -07:00
|
|
|
if (this.options.noResolve) {
|
2017-10-12 16:09:49 -07:00
|
|
|
this.rootNames.forEach(rootName => {
|
2017-10-20 09:46:41 -07:00
|
|
|
if (this.hostAdapter.shouldGenerateFilesFor(rootName)) {
|
|
|
|
rootNames.push(...this.compiler.findGeneratedFileNames(rootName));
|
2017-09-21 18:05:07 -07:00
|
|
|
}
|
2017-10-12 16:09:49 -07:00
|
|
|
});
|
2017-09-20 16:27:29 -07:00
|
|
|
}
|
|
|
|
|
2017-10-20 09:46:41 -07:00
|
|
|
const tmpProgram = ts.createProgram(rootNames, this.options, this.hostAdapter, oldTsProgram);
|
2017-09-21 18:05:07 -07:00
|
|
|
const sourceFiles: string[] = [];
|
2017-10-12 16:09:49 -07:00
|
|
|
tmpProgram.getSourceFiles().forEach(sf => {
|
2017-10-20 09:46:41 -07:00
|
|
|
if (this.hostAdapter.isSourceFile(sf.fileName)) {
|
2017-09-21 18:05:07 -07:00
|
|
|
sourceFiles.push(sf.fileName);
|
|
|
|
}
|
2017-10-12 16:09:49 -07:00
|
|
|
});
|
2017-10-20 09:46:41 -07:00
|
|
|
return {tmpProgram, sourceFiles, rootNames};
|
2017-09-12 09:40:28 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
private _updateProgramWithTypeCheckStubs(
|
2017-10-26 15:24:54 -07:00
|
|
|
tmpProgram: ts.Program, analyzedModules: NgAnalyzedModules, rootNames: string[]) {
|
|
|
|
this._analyzedModules = analyzedModules;
|
|
|
|
tmpProgram.getSourceFiles().forEach(sf => {
|
|
|
|
if (sf.fileName.endsWith('.ngfactory.ts')) {
|
|
|
|
const {generate, baseFileName} = this.hostAdapter.shouldGenerateFile(sf.fileName);
|
|
|
|
if (generate) {
|
|
|
|
// Note: ! is ok as hostAdapter.shouldGenerateFile will always return a basefileName
|
|
|
|
// for .ngfactory.ts files.
|
|
|
|
const genFile = this.compiler.emitTypeCheckStub(sf.fileName, baseFileName !);
|
|
|
|
if (genFile) {
|
|
|
|
this.hostAdapter.updateGeneratedFile(genFile);
|
2017-09-21 18:05:07 -07:00
|
|
|
}
|
|
|
|
}
|
2017-10-26 15:24:54 -07:00
|
|
|
}
|
|
|
|
});
|
2017-10-20 09:46:41 -07:00
|
|
|
this._tsProgram = ts.createProgram(rootNames, this.options, this.hostAdapter, tmpProgram);
|
2017-09-12 09:40:28 -07:00
|
|
|
// Note: the new ts program should be completely reusable by TypeScript as:
|
|
|
|
// - we cache all the files in the hostAdapter
|
|
|
|
// - new new stubs use the exactly same imports/exports as the old once (we assert that in
|
|
|
|
// hostAdapter.updateGeneratedFile).
|
|
|
|
if (tsStructureIsReused(tmpProgram) !== StructureIsReused.Completely) {
|
|
|
|
throw new Error(`Internal Error: The structure of the program changed during codegen.`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-26 15:24:54 -07:00
|
|
|
private _createProgramOnError(e: any) {
|
|
|
|
// Still fill the analyzedModules and the tsProgram
|
|
|
|
// so that we don't cause other errors for users who e.g. want to emit the ngProgram.
|
|
|
|
this._analyzedModules = emptyModules;
|
|
|
|
this.oldTsProgram = undefined;
|
|
|
|
this._hostAdapter.isSourceFile = () => false;
|
|
|
|
this._tsProgram = ts.createProgram(this.rootNames, this.options, this.hostAdapter);
|
2017-06-09 14:50:57 -07:00
|
|
|
if (isSyntaxError(e)) {
|
|
|
|
const parserErrors = getParseErrors(e);
|
|
|
|
if (parserErrors && parserErrors.length) {
|
2017-10-23 18:29:06 -07:00
|
|
|
this._structuralDiagnostics = [
|
|
|
|
...(this._structuralDiagnostics || []),
|
|
|
|
...parserErrors.map<Diagnostic>(e => ({
|
|
|
|
messageText: e.contextualMessage(),
|
|
|
|
category: ts.DiagnosticCategory.Error,
|
|
|
|
span: e.span,
|
|
|
|
source: SOURCE,
|
|
|
|
code: DEFAULT_ERROR_CODE
|
|
|
|
}))
|
|
|
|
];
|
2017-06-09 14:50:57 -07:00
|
|
|
} else {
|
2017-10-23 18:29:06 -07:00
|
|
|
this._structuralDiagnostics = [
|
|
|
|
...(this._structuralDiagnostics || []), {
|
|
|
|
messageText: e.message,
|
|
|
|
category: ts.DiagnosticCategory.Error,
|
|
|
|
source: SOURCE,
|
|
|
|
code: DEFAULT_ERROR_CODE
|
|
|
|
}
|
|
|
|
];
|
2017-06-09 14:50:57 -07:00
|
|
|
}
|
2017-10-26 15:24:54 -07:00
|
|
|
return;
|
2017-06-09 14:50:57 -07:00
|
|
|
}
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
|
2017-10-12 16:09:49 -07:00
|
|
|
// Note: this returns a ts.Diagnostic so that we
|
|
|
|
// can return errors in a ts.EmitResult
|
|
|
|
private generateFilesForEmit(emitFlags: EmitFlags):
|
|
|
|
{genFiles: GeneratedFile[], genDiags: ts.Diagnostic[]} {
|
|
|
|
try {
|
|
|
|
if (!(emitFlags & EmitFlags.Codegen)) {
|
|
|
|
return {genFiles: [], genDiags: []};
|
|
|
|
}
|
2017-10-17 16:51:04 -07:00
|
|
|
// TODO(tbosch): allow generating files that are not in the rootDir
|
|
|
|
// See https://github.com/angular/angular/issues/19337
|
|
|
|
let genFiles = this.compiler.emitAllImpls(this.analyzedModules)
|
|
|
|
.filter(genFile => isInRootDir(genFile.genFileUrl, this.options));
|
2017-10-12 16:09:49 -07:00
|
|
|
if (this.oldProgramEmittedGeneratedFiles) {
|
|
|
|
const oldProgramEmittedGeneratedFiles = this.oldProgramEmittedGeneratedFiles;
|
|
|
|
genFiles = genFiles.filter(genFile => {
|
|
|
|
const oldGenFile = oldProgramEmittedGeneratedFiles.get(genFile.genFileUrl);
|
|
|
|
return !oldGenFile || !genFile.isEquivalent(oldGenFile);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return {genFiles, genDiags: []};
|
|
|
|
} catch (e) {
|
|
|
|
// TODO(tbosch): check whether we can actually have syntax errors here,
|
|
|
|
// as we already parsed the metadata and templates before to create the type check block.
|
|
|
|
if (isSyntaxError(e)) {
|
|
|
|
const genDiags: ts.Diagnostic[] = [{
|
|
|
|
file: undefined,
|
|
|
|
start: undefined,
|
|
|
|
length: undefined,
|
|
|
|
messageText: e.message,
|
|
|
|
category: ts.DiagnosticCategory.Error,
|
|
|
|
source: SOURCE,
|
|
|
|
code: DEFAULT_ERROR_CODE
|
|
|
|
}];
|
|
|
|
return {genFiles: [], genDiags};
|
|
|
|
}
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns undefined if all files should be emitted.
|
|
|
|
*/
|
|
|
|
private getSourceFilesForEmit(): ts.SourceFile[]|undefined {
|
|
|
|
// TODO(tbosch): if one of the files contains a `const enum`
|
|
|
|
// always emit all files -> return undefined!
|
|
|
|
let sourceFilesToEmit: ts.SourceFile[]|undefined;
|
|
|
|
if (this.oldProgramEmittedSourceFiles) {
|
|
|
|
sourceFilesToEmit = this.tsProgram.getSourceFiles().filter(sf => {
|
|
|
|
const oldFile = this.oldProgramEmittedSourceFiles !.get(sf.fileName);
|
|
|
|
return !sf.isDeclarationFile && !GENERATED_FILES.test(sf.fileName) && sf !== oldFile;
|
|
|
|
});
|
2017-10-03 09:53:58 -07:00
|
|
|
}
|
2017-10-12 16:09:49 -07:00
|
|
|
return sourceFilesToEmit;
|
2017-10-03 09:53:58 -07:00
|
|
|
}
|
|
|
|
|
2017-09-21 18:05:07 -07:00
|
|
|
private writeFile(
|
|
|
|
outFileName: string, outData: string, writeByteOrderMark: boolean,
|
2017-10-12 16:09:49 -07:00
|
|
|
onError?: (message: string) => void, genFile?: GeneratedFile, sourceFiles?: ts.SourceFile[]) {
|
2017-09-21 18:05:07 -07:00
|
|
|
// collect emittedLibrarySummaries
|
|
|
|
let baseFile: ts.SourceFile|undefined;
|
|
|
|
if (genFile) {
|
2017-10-12 16:09:49 -07:00
|
|
|
baseFile = this.tsProgram.getSourceFile(genFile.srcFileUrl);
|
2017-09-21 18:05:07 -07:00
|
|
|
if (baseFile) {
|
|
|
|
if (!this.emittedLibrarySummaries) {
|
|
|
|
this.emittedLibrarySummaries = [];
|
|
|
|
}
|
2017-10-12 16:09:49 -07:00
|
|
|
if (genFile.genFileUrl.endsWith('.ngsummary.json') && baseFile.fileName.endsWith('.d.ts')) {
|
2017-09-21 18:05:07 -07:00
|
|
|
this.emittedLibrarySummaries.push({
|
|
|
|
fileName: baseFile.fileName,
|
|
|
|
text: baseFile.text,
|
|
|
|
sourceFile: baseFile,
|
|
|
|
});
|
2017-10-12 16:09:49 -07:00
|
|
|
this.emittedLibrarySummaries.push({fileName: genFile.genFileUrl, text: outData});
|
2017-09-28 11:44:05 -07:00
|
|
|
if (!this.options.declaration) {
|
|
|
|
// If we don't emit declarations, still record an empty .ngfactory.d.ts file,
|
|
|
|
// as we might need it lateron for resolving module names from summaries.
|
2017-10-12 16:09:49 -07:00
|
|
|
const ngFactoryDts =
|
|
|
|
genFile.genFileUrl.substring(0, genFile.genFileUrl.length - 15) + '.ngfactory.d.ts';
|
2017-09-28 11:44:05 -07:00
|
|
|
this.emittedLibrarySummaries.push({fileName: ngFactoryDts, text: ''});
|
|
|
|
}
|
2017-09-21 18:05:07 -07:00
|
|
|
} else if (outFileName.endsWith('.d.ts') && baseFile.fileName.endsWith('.d.ts')) {
|
2017-10-12 16:09:49 -07:00
|
|
|
const dtsSourceFilePath = genFile.genFileUrl.replace(/\.ts$/, '.d.ts');
|
2017-09-21 18:05:07 -07:00
|
|
|
// Note: Don't use sourceFiles here as the created .d.ts has a path in the outDir,
|
|
|
|
// but we need one that is next to the .ts file
|
|
|
|
this.emittedLibrarySummaries.push({fileName: dtsSourceFilePath, text: outData});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Filter out generated files for which we didn't generate code.
|
2017-10-12 16:09:49 -07:00
|
|
|
// This can happen as the stub caclulation is not completely exact.
|
2017-09-21 18:05:07 -07:00
|
|
|
// Note: sourceFile refers to the .ngfactory.ts / .ngsummary.ts file
|
2017-10-12 16:09:49 -07:00
|
|
|
const isGenerated = GENERATED_FILES.test(outFileName);
|
2017-09-21 18:05:07 -07:00
|
|
|
if (isGenerated) {
|
|
|
|
if (!genFile || !genFile.stmts || genFile.stmts.length === 0) {
|
|
|
|
if (this.options.allowEmptyCodegenFiles) {
|
|
|
|
outData = '';
|
|
|
|
} else {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (baseFile) {
|
|
|
|
sourceFiles = sourceFiles ? [...sourceFiles, baseFile] : [baseFile];
|
|
|
|
}
|
|
|
|
this.host.writeFile(outFileName, outData, writeByteOrderMark, onError, sourceFiles);
|
|
|
|
}
|
2017-08-14 11:04:18 -07:00
|
|
|
}
|
|
|
|
|
2017-10-12 16:09:49 -07:00
|
|
|
export function createProgram(
|
|
|
|
{rootNames, options, host, oldProgram}:
|
|
|
|
{rootNames: string[], options: CompilerOptions, host: CompilerHost, oldProgram?: Program}):
|
|
|
|
Program {
|
2017-06-09 14:50:57 -07:00
|
|
|
return new AngularCompilerProgram(rootNames, options, host, oldProgram);
|
2017-10-12 16:09:49 -07:00
|
|
|
}
|
2017-06-09 14:50:57 -07:00
|
|
|
|
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) {
|
2017-08-31 23:11:29 +02:00
|
|
|
if (!options.i18nInLocale) {
|
2017-08-02 11:20:07 -07:00
|
|
|
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,
|
2017-09-29 14:55:44 -07:00
|
|
|
enableSummariesForJit: options.enableSummariesForJit,
|
2017-07-28 15:58:28 +02:00
|
|
|
preserveWhitespaces: options.preserveWhitespaces,
|
2017-09-11 15:18:19 -07:00
|
|
|
fullTemplateTypeCheck: options.fullTemplateTypeCheck,
|
2017-09-21 18:05:07 -07:00
|
|
|
allowEmptyCodegenFiles: options.allowEmptyCodegenFiles,
|
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 [];
|
|
|
|
}
|
|
|
|
|
2017-10-24 14:44:25 -07:00
|
|
|
function normalizeSeparators(path: string): string {
|
|
|
|
return path.replace(/\\/g, '/');
|
|
|
|
}
|
|
|
|
|
2017-09-28 11:42:58 -07:00
|
|
|
/**
|
|
|
|
* Returns a function that can adjust a path from source path to out path,
|
|
|
|
* based on an existing mapping from source to out path.
|
|
|
|
*
|
|
|
|
* TODO(tbosch): talk to the TypeScript team to expose their logic for calculating the `rootDir`
|
|
|
|
* if none was specified.
|
|
|
|
*
|
2017-10-06 15:12:32 -07:00
|
|
|
* Note: This function works on normalized paths from typescript.
|
|
|
|
*
|
2017-09-28 11:42:58 -07:00
|
|
|
* @param outDir
|
|
|
|
* @param outSrcMappings
|
|
|
|
*/
|
|
|
|
export function createSrcToOutPathMapper(
|
|
|
|
outDir: string | undefined, sampleSrcFileName: string | undefined,
|
2017-10-06 15:12:32 -07:00
|
|
|
sampleOutFileName: string | undefined, host: {
|
|
|
|
dirname: typeof path.dirname,
|
|
|
|
resolve: typeof path.resolve,
|
|
|
|
relative: typeof path.relative
|
|
|
|
} = path): (srcFileName: string) => string {
|
2017-09-28 11:42:58 -07:00
|
|
|
let srcToOutPath: (srcFileName: string) => string;
|
|
|
|
if (outDir) {
|
2017-10-24 14:44:25 -07:00
|
|
|
let path: {} = {}; // Ensure we error if we use `path` instead of `host`.
|
2017-09-28 11:42:58 -07:00
|
|
|
if (sampleSrcFileName == null || sampleOutFileName == null) {
|
|
|
|
throw new Error(`Can't calculate the rootDir without a sample srcFileName / outFileName. `);
|
|
|
|
}
|
2017-10-24 14:44:25 -07:00
|
|
|
const srcFileDir = normalizeSeparators(host.dirname(sampleSrcFileName));
|
|
|
|
const outFileDir = normalizeSeparators(host.dirname(sampleOutFileName));
|
2017-09-28 11:42:58 -07:00
|
|
|
if (srcFileDir === outFileDir) {
|
|
|
|
return (srcFileName) => srcFileName;
|
|
|
|
}
|
2017-10-17 16:51:04 -07:00
|
|
|
// calculate the common suffix, stopping
|
|
|
|
// at `outDir`.
|
2017-10-06 15:12:32 -07:00
|
|
|
const srcDirParts = srcFileDir.split('/');
|
2017-10-24 14:44:25 -07:00
|
|
|
const outDirParts = normalizeSeparators(host.relative(outDir, outFileDir)).split('/');
|
2017-09-28 11:42:58 -07:00
|
|
|
let i = 0;
|
|
|
|
while (i < Math.min(srcDirParts.length, outDirParts.length) &&
|
|
|
|
srcDirParts[srcDirParts.length - 1 - i] === outDirParts[outDirParts.length - 1 - i])
|
|
|
|
i++;
|
2017-10-06 15:12:32 -07:00
|
|
|
const rootDir = srcDirParts.slice(0, srcDirParts.length - i).join('/');
|
|
|
|
srcToOutPath = (srcFileName) => host.resolve(outDir, host.relative(rootDir, srcFileName));
|
2017-09-28 11:42:58 -07:00
|
|
|
} else {
|
|
|
|
srcToOutPath = (srcFileName) => srcFileName;
|
2017-06-09 14:50:57 -07:00
|
|
|
}
|
2017-09-28 11:42:58 -07:00
|
|
|
return srcToOutPath;
|
2017-07-28 15:58:28 +02:00
|
|
|
}
|
2017-09-12 15:53:17 -07:00
|
|
|
|
|
|
|
export function i18nExtract(
|
|
|
|
formatName: string | null, outFile: string | null, host: ts.CompilerHost,
|
|
|
|
options: CompilerOptions, bundle: MessageBundle): string[] {
|
2017-10-24 11:26:04 -07:00
|
|
|
formatName = formatName || 'xlf';
|
2017-09-12 15:53:17 -07:00
|
|
|
// Checks the format and returns the extension
|
|
|
|
const ext = i18nGetExtension(formatName);
|
|
|
|
const content = i18nSerialize(bundle, formatName, options);
|
|
|
|
const dstFile = outFile || `messages.${ext}`;
|
|
|
|
const dstPath = path.resolve(options.outDir || options.basePath, dstFile);
|
|
|
|
host.writeFile(dstPath, content, false);
|
|
|
|
return [dstPath];
|
|
|
|
}
|
|
|
|
|
|
|
|
export function i18nSerialize(
|
|
|
|
bundle: MessageBundle, formatName: string, options: CompilerOptions): string {
|
|
|
|
const format = formatName.toLowerCase();
|
|
|
|
let serializer: Serializer;
|
|
|
|
|
|
|
|
switch (format) {
|
|
|
|
case 'xmb':
|
|
|
|
serializer = new Xmb();
|
|
|
|
break;
|
|
|
|
case 'xliff2':
|
|
|
|
case 'xlf2':
|
|
|
|
serializer = new Xliff2();
|
|
|
|
break;
|
|
|
|
case 'xlf':
|
|
|
|
case 'xliff':
|
|
|
|
default:
|
|
|
|
serializer = new Xliff();
|
|
|
|
}
|
|
|
|
return bundle.write(
|
|
|
|
serializer, (sourcePath: string) =>
|
|
|
|
options.basePath ? path.relative(options.basePath, sourcePath) : sourcePath);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function i18nGetExtension(formatName: string): string {
|
2017-10-24 11:26:04 -07:00
|
|
|
const format = formatName.toLowerCase();
|
2017-09-12 15:53:17 -07:00
|
|
|
|
|
|
|
switch (format) {
|
|
|
|
case 'xmb':
|
|
|
|
return 'xmb';
|
|
|
|
case 'xlf':
|
|
|
|
case 'xlif':
|
|
|
|
case 'xliff':
|
|
|
|
case 'xlf2':
|
|
|
|
case 'xliff2':
|
|
|
|
return 'xlf';
|
|
|
|
}
|
|
|
|
|
|
|
|
throw new Error(`Unsupported format "${formatName}"`);
|
|
|
|
}
|
2017-10-12 16:09:49 -07:00
|
|
|
|
|
|
|
function mergeEmitResults(emitResults: ts.EmitResult[]): ts.EmitResult {
|
|
|
|
const diagnostics: ts.Diagnostic[] = [];
|
|
|
|
let emitSkipped = false;
|
|
|
|
const emittedFiles: string[] = [];
|
|
|
|
for (const er of emitResults) {
|
|
|
|
diagnostics.push(...er.diagnostics);
|
|
|
|
emitSkipped = emitSkipped || er.emitSkipped;
|
|
|
|
emittedFiles.push(...er.emittedFiles);
|
|
|
|
}
|
|
|
|
return {diagnostics, emitSkipped, emittedFiles};
|
|
|
|
}
|