
After this, neither @angular/compiler nor @angular/comnpiler-cli depend on @angular/core. This add a duplication of some interfaces and enums which is stored in @angular/compiler/src/core.ts BREAKING CHANGE: - `@angular/platform-server` now additionally depends on `@angular/platform-browser-dynamic` as a peer dependency. PR Close #18683
104 lines
3.8 KiB
TypeScript
104 lines
3.8 KiB
TypeScript
/**
|
|
* @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
|
|
*/
|
|
|
|
import {identifierName} from '../compile_metadata';
|
|
import {CompileReflector} from '../compile_reflector';
|
|
|
|
import {EmitterVisitorContext} from './abstract_emitter';
|
|
import {AbstractJsEmitterVisitor} from './abstract_js_emitter';
|
|
import * as o from './output_ast';
|
|
|
|
function evalExpression(
|
|
sourceUrl: string, ctx: EmitterVisitorContext, vars: {[key: string]: any},
|
|
createSourceMap: boolean): any {
|
|
let fnBody = `${ctx.toSource()}\n//# sourceURL=${sourceUrl}`;
|
|
const fnArgNames: string[] = [];
|
|
const fnArgValues: any[] = [];
|
|
for (const argName in vars) {
|
|
fnArgNames.push(argName);
|
|
fnArgValues.push(vars[argName]);
|
|
}
|
|
if (createSourceMap) {
|
|
// using `new Function(...)` generates a header, 1 line of no arguments, 2 lines otherwise
|
|
// E.g. ```
|
|
// function anonymous(a,b,c
|
|
// /**/) { ... }```
|
|
// We don't want to hard code this fact, so we auto detect it via an empty function first.
|
|
const emptyFn = new Function(...fnArgNames.concat('return null;')).toString();
|
|
const headerLines = emptyFn.slice(0, emptyFn.indexOf('return null;')).split('\n').length - 1;
|
|
fnBody += `\n${ctx.toSourceMapGenerator(sourceUrl, sourceUrl, headerLines).toJsComment()}`;
|
|
}
|
|
return new Function(...fnArgNames.concat(fnBody))(...fnArgValues);
|
|
}
|
|
|
|
export function jitStatements(
|
|
sourceUrl: string, statements: o.Statement[], reflector: CompileReflector,
|
|
createSourceMaps: boolean): {[key: string]: any} {
|
|
const converter = new JitEmitterVisitor(reflector);
|
|
const ctx = EmitterVisitorContext.createRoot();
|
|
converter.visitAllStatements(statements, ctx);
|
|
converter.createReturnStmt(ctx);
|
|
return evalExpression(sourceUrl, ctx, converter.getArgs(), createSourceMaps);
|
|
}
|
|
|
|
export class JitEmitterVisitor extends AbstractJsEmitterVisitor {
|
|
private _evalArgNames: string[] = [];
|
|
private _evalArgValues: any[] = [];
|
|
private _evalExportedVars: string[] = [];
|
|
|
|
constructor(private reflector: CompileReflector) { super(); }
|
|
|
|
createReturnStmt(ctx: EmitterVisitorContext) {
|
|
const stmt = new o.ReturnStatement(new o.LiteralMapExpr(this._evalExportedVars.map(
|
|
resultVar => new o.LiteralMapEntry(resultVar, o.variable(resultVar), false))));
|
|
stmt.visitStatement(this, ctx);
|
|
}
|
|
|
|
getArgs(): {[key: string]: any} {
|
|
const result: {[key: string]: any} = {};
|
|
for (let i = 0; i < this._evalArgNames.length; i++) {
|
|
result[this._evalArgNames[i]] = this._evalArgValues[i];
|
|
}
|
|
return result;
|
|
}
|
|
|
|
visitExternalExpr(ast: o.ExternalExpr, ctx: EmitterVisitorContext): any {
|
|
const value = this.reflector.resolveExternalReference(ast.value);
|
|
let id = this._evalArgValues.indexOf(value);
|
|
if (id === -1) {
|
|
id = this._evalArgValues.length;
|
|
this._evalArgValues.push(value);
|
|
const name = identifierName({reference: value}) || 'val';
|
|
this._evalArgNames.push(`jit_${name}_${id}`);
|
|
}
|
|
ctx.print(ast, this._evalArgNames[id]);
|
|
return null;
|
|
}
|
|
|
|
visitDeclareVarStmt(stmt: o.DeclareVarStmt, ctx: EmitterVisitorContext): any {
|
|
if (stmt.hasModifier(o.StmtModifier.Exported)) {
|
|
this._evalExportedVars.push(stmt.name);
|
|
}
|
|
return super.visitDeclareVarStmt(stmt, ctx);
|
|
}
|
|
|
|
visitDeclareFunctionStmt(stmt: o.DeclareFunctionStmt, ctx: EmitterVisitorContext): any {
|
|
if (stmt.hasModifier(o.StmtModifier.Exported)) {
|
|
this._evalExportedVars.push(stmt.name);
|
|
}
|
|
return super.visitDeclareFunctionStmt(stmt, ctx);
|
|
}
|
|
|
|
visitDeclareClassStmt(stmt: o.ClassStmt, ctx: EmitterVisitorContext): any {
|
|
if (stmt.hasModifier(o.StmtModifier.Exported)) {
|
|
this._evalExportedVars.push(stmt.name);
|
|
}
|
|
return super.visitDeclareClassStmt(stmt, ctx);
|
|
}
|
|
}
|