2016-06-23 09:47:54 -07:00
|
|
|
/**
|
|
|
|
* @license
|
|
|
|
* Copyright Google Inc. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
|
|
* found in the LICENSE file at https://angular.io/license
|
|
|
|
*/
|
|
|
|
|
2016-04-28 21:57:16 -07:00
|
|
|
/**
|
|
|
|
* Transform template html and css into executable code.
|
|
|
|
* Intended to be used in a build step.
|
|
|
|
*/
|
2016-04-28 17:50:03 -07:00
|
|
|
import * as compiler from '@angular/compiler';
|
2016-07-18 03:50:31 -07:00
|
|
|
import {ComponentMetadata, NgModuleMetadata, ViewEncapsulation} from '@angular/core';
|
2016-06-08 16:38:52 -07:00
|
|
|
import {AngularCompilerOptions} from '@angular/tsc-wrapped';
|
|
|
|
import * as path from 'path';
|
|
|
|
import * as ts from 'typescript';
|
2016-04-28 21:57:16 -07:00
|
|
|
|
2016-07-18 03:50:31 -07:00
|
|
|
import {CompileMetadataResolver, DirectiveNormalizer, DomElementSchemaRegistry, HtmlParser, Lexer, NgModuleCompiler, Parser, StyleCompiler, TemplateParser, TypeScriptEmitter, ViewCompiler} from './compiler_private';
|
2016-07-28 19:39:10 +02:00
|
|
|
import {Console} from './core_private';
|
2016-08-09 14:58:19 -07:00
|
|
|
import {GENERATED_FILES, ReflectorHost, ReflectorHostContext} from './reflector_host';
|
2016-05-03 17:31:40 -07:00
|
|
|
import {StaticAndDynamicReflectionCapabilities} from './static_reflection_capabilities';
|
2016-06-28 09:54:42 -07:00
|
|
|
import {StaticReflector, StaticSymbol} from './static_reflector';
|
2016-04-28 21:57:16 -07:00
|
|
|
|
|
|
|
const PREAMBLE = `/**
|
|
|
|
* This file is generated by the Angular 2 template compiler.
|
|
|
|
* Do not edit.
|
|
|
|
*/
|
2016-05-27 09:16:46 -07:00
|
|
|
/* tslint:disable */
|
|
|
|
|
2016-04-28 21:57:16 -07:00
|
|
|
`;
|
|
|
|
|
|
|
|
export class CodeGenerator {
|
2016-06-08 16:38:52 -07:00
|
|
|
constructor(
|
|
|
|
private options: AngularCompilerOptions, private program: ts.Program,
|
|
|
|
public host: ts.CompilerHost, private staticReflector: StaticReflector,
|
2016-07-09 10:12:39 -07:00
|
|
|
private compiler: compiler.OfflineCompiler, private reflectorHost: ReflectorHost) {}
|
2016-04-28 21:57:16 -07:00
|
|
|
|
2016-06-28 09:54:42 -07:00
|
|
|
private readFileMetadata(absSourcePath: string): FileMetadata {
|
2016-05-31 11:00:39 -07:00
|
|
|
const moduleMetadata = this.staticReflector.getModuleMetadata(absSourcePath);
|
2016-07-18 03:50:31 -07:00
|
|
|
const result: FileMetadata = {components: [], ngModules: [], fileUrl: absSourcePath};
|
2016-05-31 11:00:39 -07:00
|
|
|
if (!moduleMetadata) {
|
2016-04-28 21:57:16 -07:00
|
|
|
console.log(`WARNING: no metadata found for ${absSourcePath}`);
|
|
|
|
return result;
|
|
|
|
}
|
2016-05-31 11:00:39 -07:00
|
|
|
const metadata = moduleMetadata['metadata'];
|
|
|
|
const symbols = metadata && Object.keys(metadata);
|
2016-04-28 21:57:16 -07:00
|
|
|
if (!symbols || !symbols.length) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
for (const symbol of symbols) {
|
2016-05-31 11:00:39 -07:00
|
|
|
if (metadata[symbol] && metadata[symbol].__symbolic == 'error') {
|
|
|
|
// Ignore symbols that are only included to record error information.
|
|
|
|
continue;
|
|
|
|
}
|
2016-04-28 21:57:16 -07:00
|
|
|
const staticType = this.reflectorHost.findDeclaration(absSourcePath, symbol, absSourcePath);
|
2016-06-28 09:54:42 -07:00
|
|
|
const annotations = this.staticReflector.annotations(staticType);
|
|
|
|
annotations.forEach((annotation) => {
|
2016-07-18 03:50:31 -07:00
|
|
|
if (annotation instanceof NgModuleMetadata) {
|
|
|
|
result.ngModules.push(staticType);
|
2016-06-28 09:54:42 -07:00
|
|
|
} else if (annotation instanceof ComponentMetadata) {
|
|
|
|
result.components.push(staticType);
|
|
|
|
}
|
|
|
|
});
|
2016-04-28 21:57:16 -07:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2016-05-03 11:50:55 -06:00
|
|
|
// Write codegen in a directory structure matching the sources.
|
2016-08-09 14:58:19 -07:00
|
|
|
private calculateEmitPath(filePath: string): string {
|
2016-05-24 10:53:48 -07:00
|
|
|
let root = this.options.basePath;
|
2016-05-03 11:50:55 -06:00
|
|
|
for (let eachRootDir of this.options.rootDirs || []) {
|
2016-05-24 10:53:48 -07:00
|
|
|
if (this.options.trace) {
|
2016-05-03 18:49:59 -07:00
|
|
|
console.log(`Check if ${filePath} is under rootDirs element ${eachRootDir}`);
|
2016-05-03 11:50:55 -06:00
|
|
|
}
|
|
|
|
if (path.relative(eachRootDir, filePath).indexOf('.') !== 0) {
|
|
|
|
root = eachRootDir;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-03 21:34:03 -07:00
|
|
|
// transplant the codegen path to be inside the `genDir`
|
|
|
|
var relativePath: string = path.relative(root, filePath);
|
|
|
|
while (relativePath.startsWith('..' + path.sep)) {
|
|
|
|
// Strip out any `..` path such as: `../node_modules/@foo` as we want to put everything
|
|
|
|
// into `genDir`.
|
|
|
|
relativePath = relativePath.substr(3);
|
|
|
|
}
|
|
|
|
return path.join(this.options.genDir, relativePath);
|
2016-05-03 11:50:55 -06:00
|
|
|
}
|
|
|
|
|
2016-05-02 09:38:46 -07:00
|
|
|
codegen(): Promise<any> {
|
2016-07-21 13:56:58 -07:00
|
|
|
const filePaths =
|
2016-06-28 09:54:42 -07:00
|
|
|
this.program.getSourceFiles().map(sf => sf.fileName).filter(f => !GENERATED_FILES.test(f));
|
2016-07-21 13:56:58 -07:00
|
|
|
const fileMetas = filePaths.map((filePath) => this.readFileMetadata(filePath));
|
|
|
|
const ngModules = fileMetas.reduce((ngModules, fileMeta) => {
|
2016-07-18 03:50:31 -07:00
|
|
|
ngModules.push(...fileMeta.ngModules);
|
|
|
|
return ngModules;
|
2016-06-28 09:54:42 -07:00
|
|
|
}, <StaticSymbol[]>[]);
|
2016-07-21 13:56:58 -07:00
|
|
|
const analyzedNgModules = this.compiler.analyzeModules(ngModules);
|
2016-06-28 09:54:42 -07:00
|
|
|
return Promise
|
|
|
|
.all(fileMetas.map(
|
|
|
|
(fileMeta) => this.compiler
|
|
|
|
.compile(
|
2016-07-18 03:50:31 -07:00
|
|
|
fileMeta.fileUrl, analyzedNgModules, fileMeta.components,
|
|
|
|
fileMeta.ngModules)
|
2016-06-28 09:54:42 -07:00
|
|
|
.then((generatedModules) => {
|
|
|
|
generatedModules.forEach((generatedModule) => {
|
|
|
|
const sourceFile = this.program.getSourceFile(fileMeta.fileUrl);
|
|
|
|
const emitPath =
|
|
|
|
this.calculateEmitPath(generatedModule.moduleUrl);
|
|
|
|
this.host.writeFile(
|
|
|
|
emitPath, PREAMBLE + generatedModule.source, false, () => {},
|
|
|
|
[sourceFile]);
|
|
|
|
});
|
|
|
|
})))
|
|
|
|
.catch((e) => { console.error(e.stack); });
|
2016-04-28 21:57:16 -07:00
|
|
|
}
|
|
|
|
|
2016-06-08 16:38:52 -07:00
|
|
|
static create(
|
|
|
|
options: AngularCompilerOptions, program: ts.Program, compilerHost: ts.CompilerHost,
|
2016-08-11 15:04:00 -07:00
|
|
|
reflectorHostContext?: ReflectorHostContext, xhr?: compiler.XHR): CodeGenerator {
|
|
|
|
xhr = xhr || {
|
2016-06-28 20:07:00 -07:00
|
|
|
get: (s: string) => {
|
|
|
|
if (!compilerHost.fileExists(s)) {
|
|
|
|
// TODO: We should really have a test for error cases like this!
|
|
|
|
throw new Error(`Compilation failed. Resource file not found: ${s}`);
|
|
|
|
}
|
|
|
|
return Promise.resolve(compilerHost.readFile(s));
|
|
|
|
}
|
|
|
|
};
|
2016-04-28 21:57:16 -07:00
|
|
|
const urlResolver: compiler.UrlResolver = compiler.createOfflineCompileUrlResolver();
|
2016-06-09 14:51:53 -07:00
|
|
|
const reflectorHost = new ReflectorHost(program, compilerHost, options, reflectorHostContext);
|
2016-04-28 21:57:16 -07:00
|
|
|
const staticReflector = new StaticReflector(reflectorHost);
|
2016-05-03 17:31:40 -07:00
|
|
|
StaticAndDynamicReflectionCapabilities.install(staticReflector);
|
2016-08-11 21:00:35 -07:00
|
|
|
const htmlParser = new compiler.i18n.HtmlParser(new HtmlParser());
|
2016-06-13 10:06:40 -07:00
|
|
|
const config = new compiler.CompilerConfig({
|
2016-06-15 14:56:56 -07:00
|
|
|
genDebugInfo: options.debug === true,
|
2016-06-13 10:06:40 -07:00
|
|
|
defaultEncapsulation: ViewEncapsulation.Emulated,
|
|
|
|
logBindingUpdate: false,
|
2016-07-08 13:40:54 -07:00
|
|
|
useJit: false
|
2016-06-13 10:06:40 -07:00
|
|
|
});
|
2016-04-03 08:34:44 +09:00
|
|
|
const normalizer = new DirectiveNormalizer(xhr, urlResolver, htmlParser, config);
|
2016-06-24 14:31:35 -07:00
|
|
|
const expressionParser = new Parser(new Lexer());
|
2016-07-28 19:39:10 +02:00
|
|
|
const elementSchemaRegistry = new DomElementSchemaRegistry();
|
|
|
|
const console = new Console();
|
|
|
|
const tmplParser =
|
|
|
|
new TemplateParser(expressionParser, elementSchemaRegistry, htmlParser, console, []);
|
2016-04-28 21:57:16 -07:00
|
|
|
const resolver = new CompileMetadataResolver(
|
2016-07-18 03:50:31 -07:00
|
|
|
new compiler.NgModuleResolver(staticReflector),
|
2016-04-28 21:57:16 -07:00
|
|
|
new compiler.DirectiveResolver(staticReflector), new compiler.PipeResolver(staticReflector),
|
2016-07-28 06:31:26 -07:00
|
|
|
config, console, elementSchemaRegistry, staticReflector);
|
2016-06-28 09:54:42 -07:00
|
|
|
const offlineCompiler = new compiler.OfflineCompiler(
|
|
|
|
resolver, normalizer, tmplParser, new StyleCompiler(urlResolver), new ViewCompiler(config),
|
2016-07-18 03:50:31 -07:00
|
|
|
new NgModuleCompiler(), new TypeScriptEmitter(reflectorHost));
|
2016-04-28 21:57:16 -07:00
|
|
|
|
2016-06-08 16:38:52 -07:00
|
|
|
return new CodeGenerator(
|
2016-07-09 10:12:39 -07:00
|
|
|
options, program, compilerHost, staticReflector, offlineCompiler, reflectorHost);
|
2016-04-28 21:57:16 -07:00
|
|
|
}
|
|
|
|
}
|
2016-06-28 09:54:42 -07:00
|
|
|
|
|
|
|
interface FileMetadata {
|
|
|
|
fileUrl: string;
|
|
|
|
components: StaticSymbol[];
|
2016-07-18 03:50:31 -07:00
|
|
|
ngModules: StaticSymbol[];
|
|
|
|
}
|