1 | /**
|
---|
2 | * @license
|
---|
3 | * Copyright Google LLC All Rights Reserved.
|
---|
4 | *
|
---|
5 | * Use of this source code is governed by an MIT-style license that can be
|
---|
6 | * found in the LICENSE file at https://angular.io/license
|
---|
7 | */
|
---|
8 | import { identifierName } from '../parse_util';
|
---|
9 | import { EmitterVisitorContext } from './abstract_emitter';
|
---|
10 | import { AbstractJsEmitterVisitor } from './abstract_js_emitter';
|
---|
11 | import * as o from './output_ast';
|
---|
12 | import { newTrustedFunctionForJIT } from './output_jit_trusted_types';
|
---|
13 | /**
|
---|
14 | * A helper class to manage the evaluation of JIT generated code.
|
---|
15 | */
|
---|
16 | export class JitEvaluator {
|
---|
17 | /**
|
---|
18 | *
|
---|
19 | * @param sourceUrl The URL of the generated code.
|
---|
20 | * @param statements An array of Angular statement AST nodes to be evaluated.
|
---|
21 | * @param reflector A helper used when converting the statements to executable code.
|
---|
22 | * @param createSourceMaps If true then create a source-map for the generated code and include it
|
---|
23 | * inline as a source-map comment.
|
---|
24 | * @returns A map of all the variables in the generated code.
|
---|
25 | */
|
---|
26 | evaluateStatements(sourceUrl, statements, reflector, createSourceMaps) {
|
---|
27 | const converter = new JitEmitterVisitor(reflector);
|
---|
28 | const ctx = EmitterVisitorContext.createRoot();
|
---|
29 | // Ensure generated code is in strict mode
|
---|
30 | if (statements.length > 0 && !isUseStrictStatement(statements[0])) {
|
---|
31 | statements = [
|
---|
32 | o.literal('use strict').toStmt(),
|
---|
33 | ...statements,
|
---|
34 | ];
|
---|
35 | }
|
---|
36 | converter.visitAllStatements(statements, ctx);
|
---|
37 | converter.createReturnStmt(ctx);
|
---|
38 | return this.evaluateCode(sourceUrl, ctx, converter.getArgs(), createSourceMaps);
|
---|
39 | }
|
---|
40 | /**
|
---|
41 | * Evaluate a piece of JIT generated code.
|
---|
42 | * @param sourceUrl The URL of this generated code.
|
---|
43 | * @param ctx A context object that contains an AST of the code to be evaluated.
|
---|
44 | * @param vars A map containing the names and values of variables that the evaluated code might
|
---|
45 | * reference.
|
---|
46 | * @param createSourceMap If true then create a source-map for the generated code and include it
|
---|
47 | * inline as a source-map comment.
|
---|
48 | * @returns The result of evaluating the code.
|
---|
49 | */
|
---|
50 | evaluateCode(sourceUrl, ctx, vars, createSourceMap) {
|
---|
51 | let fnBody = `"use strict";${ctx.toSource()}\n//# sourceURL=${sourceUrl}`;
|
---|
52 | const fnArgNames = [];
|
---|
53 | const fnArgValues = [];
|
---|
54 | for (const argName in vars) {
|
---|
55 | fnArgValues.push(vars[argName]);
|
---|
56 | fnArgNames.push(argName);
|
---|
57 | }
|
---|
58 | if (createSourceMap) {
|
---|
59 | // using `new Function(...)` generates a header, 1 line of no arguments, 2 lines otherwise
|
---|
60 | // E.g. ```
|
---|
61 | // function anonymous(a,b,c
|
---|
62 | // /**/) { ... }```
|
---|
63 | // We don't want to hard code this fact, so we auto detect it via an empty function first.
|
---|
64 | const emptyFn = newTrustedFunctionForJIT(...fnArgNames.concat('return null;')).toString();
|
---|
65 | const headerLines = emptyFn.slice(0, emptyFn.indexOf('return null;')).split('\n').length - 1;
|
---|
66 | fnBody += `\n${ctx.toSourceMapGenerator(sourceUrl, headerLines).toJsComment()}`;
|
---|
67 | }
|
---|
68 | const fn = newTrustedFunctionForJIT(...fnArgNames.concat(fnBody));
|
---|
69 | return this.executeFunction(fn, fnArgValues);
|
---|
70 | }
|
---|
71 | /**
|
---|
72 | * Execute a JIT generated function by calling it.
|
---|
73 | *
|
---|
74 | * This method can be overridden in tests to capture the functions that are generated
|
---|
75 | * by this `JitEvaluator` class.
|
---|
76 | *
|
---|
77 | * @param fn A function to execute.
|
---|
78 | * @param args The arguments to pass to the function being executed.
|
---|
79 | * @returns The return value of the executed function.
|
---|
80 | */
|
---|
81 | executeFunction(fn, args) {
|
---|
82 | return fn(...args);
|
---|
83 | }
|
---|
84 | }
|
---|
85 | /**
|
---|
86 | * An Angular AST visitor that converts AST nodes into executable JavaScript code.
|
---|
87 | */
|
---|
88 | export class JitEmitterVisitor extends AbstractJsEmitterVisitor {
|
---|
89 | constructor(reflector) {
|
---|
90 | super();
|
---|
91 | this.reflector = reflector;
|
---|
92 | this._evalArgNames = [];
|
---|
93 | this._evalArgValues = [];
|
---|
94 | this._evalExportedVars = [];
|
---|
95 | }
|
---|
96 | createReturnStmt(ctx) {
|
---|
97 | const stmt = new o.ReturnStatement(new o.LiteralMapExpr(this._evalExportedVars.map(resultVar => new o.LiteralMapEntry(resultVar, o.variable(resultVar), false))));
|
---|
98 | stmt.visitStatement(this, ctx);
|
---|
99 | }
|
---|
100 | getArgs() {
|
---|
101 | const result = {};
|
---|
102 | for (let i = 0; i < this._evalArgNames.length; i++) {
|
---|
103 | result[this._evalArgNames[i]] = this._evalArgValues[i];
|
---|
104 | }
|
---|
105 | return result;
|
---|
106 | }
|
---|
107 | visitExternalExpr(ast, ctx) {
|
---|
108 | this._emitReferenceToExternal(ast, this.reflector.resolveExternalReference(ast.value), ctx);
|
---|
109 | return null;
|
---|
110 | }
|
---|
111 | visitWrappedNodeExpr(ast, ctx) {
|
---|
112 | this._emitReferenceToExternal(ast, ast.node, ctx);
|
---|
113 | return null;
|
---|
114 | }
|
---|
115 | visitDeclareVarStmt(stmt, ctx) {
|
---|
116 | if (stmt.hasModifier(o.StmtModifier.Exported)) {
|
---|
117 | this._evalExportedVars.push(stmt.name);
|
---|
118 | }
|
---|
119 | return super.visitDeclareVarStmt(stmt, ctx);
|
---|
120 | }
|
---|
121 | visitDeclareFunctionStmt(stmt, ctx) {
|
---|
122 | if (stmt.hasModifier(o.StmtModifier.Exported)) {
|
---|
123 | this._evalExportedVars.push(stmt.name);
|
---|
124 | }
|
---|
125 | return super.visitDeclareFunctionStmt(stmt, ctx);
|
---|
126 | }
|
---|
127 | visitDeclareClassStmt(stmt, ctx) {
|
---|
128 | if (stmt.hasModifier(o.StmtModifier.Exported)) {
|
---|
129 | this._evalExportedVars.push(stmt.name);
|
---|
130 | }
|
---|
131 | return super.visitDeclareClassStmt(stmt, ctx);
|
---|
132 | }
|
---|
133 | _emitReferenceToExternal(ast, value, ctx) {
|
---|
134 | let id = this._evalArgValues.indexOf(value);
|
---|
135 | if (id === -1) {
|
---|
136 | id = this._evalArgValues.length;
|
---|
137 | this._evalArgValues.push(value);
|
---|
138 | const name = identifierName({ reference: value }) || 'val';
|
---|
139 | this._evalArgNames.push(`jit_${name}_${id}`);
|
---|
140 | }
|
---|
141 | ctx.print(ast, this._evalArgNames[id]);
|
---|
142 | }
|
---|
143 | }
|
---|
144 | function isUseStrictStatement(statement) {
|
---|
145 | return statement.isEquivalent(o.literal('use strict').toStmt());
|
---|
146 | }
|
---|
147 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"output_jit.js","sourceRoot":"","sources":["../../../../../../../packages/compiler/src/output/output_jit.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {CompileReflector} from '../compile_reflector';\nimport {identifierName} from '../parse_util';\n\nimport {EmitterVisitorContext} from './abstract_emitter';\nimport {AbstractJsEmitterVisitor} from './abstract_js_emitter';\nimport * as o from './output_ast';\nimport {newTrustedFunctionForJIT} from './output_jit_trusted_types';\n\n/**\n * A helper class to manage the evaluation of JIT generated code.\n */\nexport class JitEvaluator {\n  /**\n   *\n   * @param sourceUrl The URL of the generated code.\n   * @param statements An array of Angular statement AST nodes to be evaluated.\n   * @param reflector A helper used when converting the statements to executable code.\n   * @param createSourceMaps If true then create a source-map for the generated code and include it\n   * inline as a source-map comment.\n   * @returns A map of all the variables in the generated code.\n   */\n  evaluateStatements(\n      sourceUrl: string, statements: o.Statement[], reflector: CompileReflector,\n      createSourceMaps: boolean): {[key: string]: any} {\n    const converter = new JitEmitterVisitor(reflector);\n    const ctx = EmitterVisitorContext.createRoot();\n    // Ensure generated code is in strict mode\n    if (statements.length > 0 && !isUseStrictStatement(statements[0])) {\n      statements = [\n        o.literal('use strict').toStmt(),\n        ...statements,\n      ];\n    }\n    converter.visitAllStatements(statements, ctx);\n    converter.createReturnStmt(ctx);\n    return this.evaluateCode(sourceUrl, ctx, converter.getArgs(), createSourceMaps);\n  }\n\n  /**\n   * Evaluate a piece of JIT generated code.\n   * @param sourceUrl The URL of this generated code.\n   * @param ctx A context object that contains an AST of the code to be evaluated.\n   * @param vars A map containing the names and values of variables that the evaluated code might\n   * reference.\n   * @param createSourceMap If true then create a source-map for the generated code and include it\n   * inline as a source-map comment.\n   * @returns The result of evaluating the code.\n   */\n  evaluateCode(\n      sourceUrl: string, ctx: EmitterVisitorContext, vars: {[key: string]: any},\n      createSourceMap: boolean): any {\n    let fnBody = `\"use strict\";${ctx.toSource()}\\n//# sourceURL=${sourceUrl}`;\n    const fnArgNames: string[] = [];\n    const fnArgValues: any[] = [];\n    for (const argName in vars) {\n      fnArgValues.push(vars[argName]);\n      fnArgNames.push(argName);\n    }\n    if (createSourceMap) {\n      // using `new Function(...)` generates a header, 1 line of no arguments, 2 lines otherwise\n      // E.g. ```\n      // function anonymous(a,b,c\n      // /**/) { ... }```\n      // We don't want to hard code this fact, so we auto detect it via an empty function first.\n      const emptyFn = newTrustedFunctionForJIT(...fnArgNames.concat('return null;')).toString();\n      const headerLines = emptyFn.slice(0, emptyFn.indexOf('return null;')).split('\\n').length - 1;\n      fnBody += `\\n${ctx.toSourceMapGenerator(sourceUrl, headerLines).toJsComment()}`;\n    }\n    const fn = newTrustedFunctionForJIT(...fnArgNames.concat(fnBody));\n    return this.executeFunction(fn, fnArgValues);\n  }\n\n  /**\n   * Execute a JIT generated function by calling it.\n   *\n   * This method can be overridden in tests to capture the functions that are generated\n   * by this `JitEvaluator` class.\n   *\n   * @param fn A function to execute.\n   * @param args The arguments to pass to the function being executed.\n   * @returns The return value of the executed function.\n   */\n  executeFunction(fn: Function, args: any[]) {\n    return fn(...args);\n  }\n}\n\n/**\n * An Angular AST visitor that converts AST nodes into executable JavaScript code.\n */\nexport class JitEmitterVisitor extends AbstractJsEmitterVisitor {\n  private _evalArgNames: string[] = [];\n  private _evalArgValues: any[] = [];\n  private _evalExportedVars: string[] = [];\n\n  constructor(private reflector: CompileReflector) {\n    super();\n  }\n\n  createReturnStmt(ctx: EmitterVisitorContext) {\n    const stmt = new o.ReturnStatement(new o.LiteralMapExpr(this._evalExportedVars.map(\n        resultVar => new o.LiteralMapEntry(resultVar, o.variable(resultVar), false))));\n    stmt.visitStatement(this, ctx);\n  }\n\n  getArgs(): {[key: string]: any} {\n    const result: {[key: string]: any} = {};\n    for (let i = 0; i < this._evalArgNames.length; i++) {\n      result[this._evalArgNames[i]] = this._evalArgValues[i];\n    }\n    return result;\n  }\n\n  override visitExternalExpr(ast: o.ExternalExpr, ctx: EmitterVisitorContext): any {\n    this._emitReferenceToExternal(ast, this.reflector.resolveExternalReference(ast.value), ctx);\n    return null;\n  }\n\n  override visitWrappedNodeExpr(ast: o.WrappedNodeExpr<any>, ctx: EmitterVisitorContext): any {\n    this._emitReferenceToExternal(ast, ast.node, ctx);\n    return null;\n  }\n\n  override visitDeclareVarStmt(stmt: o.DeclareVarStmt, ctx: EmitterVisitorContext): any {\n    if (stmt.hasModifier(o.StmtModifier.Exported)) {\n      this._evalExportedVars.push(stmt.name);\n    }\n    return super.visitDeclareVarStmt(stmt, ctx);\n  }\n\n  override visitDeclareFunctionStmt(stmt: o.DeclareFunctionStmt, ctx: EmitterVisitorContext): any {\n    if (stmt.hasModifier(o.StmtModifier.Exported)) {\n      this._evalExportedVars.push(stmt.name);\n    }\n    return super.visitDeclareFunctionStmt(stmt, ctx);\n  }\n\n  override visitDeclareClassStmt(stmt: o.ClassStmt, ctx: EmitterVisitorContext): any {\n    if (stmt.hasModifier(o.StmtModifier.Exported)) {\n      this._evalExportedVars.push(stmt.name);\n    }\n    return super.visitDeclareClassStmt(stmt, ctx);\n  }\n\n  private _emitReferenceToExternal(ast: o.Expression, value: any, ctx: EmitterVisitorContext):\n      void {\n    let id = this._evalArgValues.indexOf(value);\n    if (id === -1) {\n      id = this._evalArgValues.length;\n      this._evalArgValues.push(value);\n      const name = identifierName({reference: value}) || 'val';\n      this._evalArgNames.push(`jit_${name}_${id}`);\n    }\n    ctx.print(ast, this._evalArgNames[id]);\n  }\n}\n\n\nfunction isUseStrictStatement(statement: o.Statement): boolean {\n  return statement.isEquivalent(o.literal('use strict').toStmt());\n}\n"]} |
---|