source: trip-planner-front/node_modules/@angular/compiler/esm2015/src/constant_pool.js@ 76712b2

Last change on this file since 76712b2 was 6a3a178, checked in by Ema <ema_spirova@…>, 3 years ago

initial commit

  • Property mode set to 100644
File size: 41.3 KB
Line 
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 */
8import * as o from './output/output_ast';
9const CONSTANT_PREFIX = '_c';
10/**
11 * `ConstantPool` tries to reuse literal factories when two or more literals are identical.
12 * We determine whether literals are identical by creating a key out of their AST using the
13 * `KeyVisitor`. This constant is used to replace dynamic expressions which can't be safely
14 * converted into a key. E.g. given an expression `{foo: bar()}`, since we don't know what
15 * the result of `bar` will be, we create a key that looks like `{foo: <unknown>}`. Note
16 * that we use a variable, rather than something like `null` in order to avoid collisions.
17 */
18const UNKNOWN_VALUE_KEY = o.variable('<unknown>');
19/**
20 * Context to use when producing a key.
21 *
22 * This ensures we see the constant not the reference variable when producing
23 * a key.
24 */
25const KEY_CONTEXT = {};
26/**
27 * Generally all primitive values are excluded from the `ConstantPool`, but there is an exclusion
28 * for strings that reach a certain length threshold. This constant defines the length threshold for
29 * strings.
30 */
31const POOL_INCLUSION_LENGTH_THRESHOLD_FOR_STRINGS = 50;
32/**
33 * A node that is a place-holder that allows the node to be replaced when the actual
34 * node is known.
35 *
36 * This allows the constant pool to change an expression from a direct reference to
37 * a constant to a shared constant. It returns a fix-up node that is later allowed to
38 * change the referenced expression.
39 */
40class FixupExpression extends o.Expression {
41 constructor(resolved) {
42 super(resolved.type);
43 this.resolved = resolved;
44 this.original = resolved;
45 }
46 visitExpression(visitor, context) {
47 if (context === KEY_CONTEXT) {
48 // When producing a key we want to traverse the constant not the
49 // variable used to refer to it.
50 return this.original.visitExpression(visitor, context);
51 }
52 else {
53 return this.resolved.visitExpression(visitor, context);
54 }
55 }
56 isEquivalent(e) {
57 return e instanceof FixupExpression && this.resolved.isEquivalent(e.resolved);
58 }
59 isConstant() {
60 return true;
61 }
62 fixup(expression) {
63 this.resolved = expression;
64 this.shared = true;
65 }
66}
67/**
68 * A constant pool allows a code emitter to share constant in an output context.
69 *
70 * The constant pool also supports sharing access to ivy definitions references.
71 */
72export class ConstantPool {
73 constructor(isClosureCompilerEnabled = false) {
74 this.isClosureCompilerEnabled = isClosureCompilerEnabled;
75 this.statements = [];
76 this.literals = new Map();
77 this.literalFactories = new Map();
78 this.injectorDefinitions = new Map();
79 this.directiveDefinitions = new Map();
80 this.componentDefinitions = new Map();
81 this.pipeDefinitions = new Map();
82 this.nextNameIndex = 0;
83 }
84 getConstLiteral(literal, forceShared) {
85 if ((literal instanceof o.LiteralExpr && !isLongStringLiteral(literal)) ||
86 literal instanceof FixupExpression) {
87 // Do no put simple literals into the constant pool or try to produce a constant for a
88 // reference to a constant.
89 return literal;
90 }
91 const key = this.keyOf(literal);
92 let fixup = this.literals.get(key);
93 let newValue = false;
94 if (!fixup) {
95 fixup = new FixupExpression(literal);
96 this.literals.set(key, fixup);
97 newValue = true;
98 }
99 if ((!newValue && !fixup.shared) || (newValue && forceShared)) {
100 // Replace the expression with a variable
101 const name = this.freshName();
102 let definition;
103 let usage;
104 if (this.isClosureCompilerEnabled && isLongStringLiteral(literal)) {
105 // For string literals, Closure will **always** inline the string at
106 // **all** usages, duplicating it each time. For large strings, this
107 // unnecessarily bloats bundle size. To work around this restriction, we
108 // wrap the string in a function, and call that function for each usage.
109 // This tricks Closure into using inline logic for functions instead of
110 // string literals. Function calls are only inlined if the body is small
111 // enough to be worth it. By doing this, very large strings will be
112 // shared across multiple usages, rather than duplicating the string at
113 // each usage site.
114 //
115 // const myStr = function() { return "very very very long string"; };
116 // const usage1 = myStr();
117 // const usage2 = myStr();
118 definition = o.variable(name).set(new o.FunctionExpr([], // Params.
119 [
120 // Statements.
121 new o.ReturnStatement(literal),
122 ]));
123 usage = o.variable(name).callFn([]);
124 }
125 else {
126 // Just declare and use the variable directly, without a function call
127 // indirection. This saves a few bytes and avoids an unncessary call.
128 definition = o.variable(name).set(literal);
129 usage = o.variable(name);
130 }
131 this.statements.push(definition.toDeclStmt(o.INFERRED_TYPE, [o.StmtModifier.Final]));
132 fixup.fixup(usage);
133 }
134 return fixup;
135 }
136 getDefinition(type, kind, ctx, forceShared = false) {
137 const definitions = this.definitionsOf(kind);
138 let fixup = definitions.get(type);
139 let newValue = false;
140 if (!fixup) {
141 const property = this.propertyNameOf(kind);
142 fixup = new FixupExpression(ctx.importExpr(type).prop(property));
143 definitions.set(type, fixup);
144 newValue = true;
145 }
146 if ((!newValue && !fixup.shared) || (newValue && forceShared)) {
147 const name = this.freshName();
148 this.statements.push(o.variable(name).set(fixup.resolved).toDeclStmt(o.INFERRED_TYPE, [o.StmtModifier.Final]));
149 fixup.fixup(o.variable(name));
150 }
151 return fixup;
152 }
153 getLiteralFactory(literal) {
154 // Create a pure function that builds an array of a mix of constant and variable expressions
155 if (literal instanceof o.LiteralArrayExpr) {
156 const argumentsForKey = literal.entries.map(e => e.isConstant() ? e : UNKNOWN_VALUE_KEY);
157 const key = this.keyOf(o.literalArr(argumentsForKey));
158 return this._getLiteralFactory(key, literal.entries, entries => o.literalArr(entries));
159 }
160 else {
161 const expressionForKey = o.literalMap(literal.entries.map(e => ({
162 key: e.key,
163 value: e.value.isConstant() ? e.value : UNKNOWN_VALUE_KEY,
164 quoted: e.quoted
165 })));
166 const key = this.keyOf(expressionForKey);
167 return this._getLiteralFactory(key, literal.entries.map(e => e.value), entries => o.literalMap(entries.map((value, index) => ({
168 key: literal.entries[index].key,
169 value,
170 quoted: literal.entries[index].quoted
171 }))));
172 }
173 }
174 _getLiteralFactory(key, values, resultMap) {
175 let literalFactory = this.literalFactories.get(key);
176 const literalFactoryArguments = values.filter((e => !e.isConstant()));
177 if (!literalFactory) {
178 const resultExpressions = values.map((e, index) => e.isConstant() ? this.getConstLiteral(e, true) : o.variable(`a${index}`));
179 const parameters = resultExpressions.filter(isVariable).map(e => new o.FnParam(e.name, o.DYNAMIC_TYPE));
180 const pureFunctionDeclaration = o.fn(parameters, [new o.ReturnStatement(resultMap(resultExpressions))], o.INFERRED_TYPE);
181 const name = this.freshName();
182 this.statements.push(o.variable(name).set(pureFunctionDeclaration).toDeclStmt(o.INFERRED_TYPE, [
183 o.StmtModifier.Final
184 ]));
185 literalFactory = o.variable(name);
186 this.literalFactories.set(key, literalFactory);
187 }
188 return { literalFactory, literalFactoryArguments };
189 }
190 /**
191 * Produce a unique name.
192 *
193 * The name might be unique among different prefixes if any of the prefixes end in
194 * a digit so the prefix should be a constant string (not based on user input) and
195 * must not end in a digit.
196 */
197 uniqueName(prefix) {
198 return `${prefix}${this.nextNameIndex++}`;
199 }
200 definitionsOf(kind) {
201 switch (kind) {
202 case 2 /* Component */:
203 return this.componentDefinitions;
204 case 1 /* Directive */:
205 return this.directiveDefinitions;
206 case 0 /* Injector */:
207 return this.injectorDefinitions;
208 case 3 /* Pipe */:
209 return this.pipeDefinitions;
210 }
211 }
212 propertyNameOf(kind) {
213 switch (kind) {
214 case 2 /* Component */:
215 return 'ɵcmp';
216 case 1 /* Directive */:
217 return 'ɵdir';
218 case 0 /* Injector */:
219 return 'ɵinj';
220 case 3 /* Pipe */:
221 return 'ɵpipe';
222 }
223 }
224 freshName() {
225 return this.uniqueName(CONSTANT_PREFIX);
226 }
227 keyOf(expression) {
228 return expression.visitExpression(new KeyVisitor(), KEY_CONTEXT);
229 }
230}
231/**
232 * Visitor used to determine if 2 expressions are equivalent and can be shared in the
233 * `ConstantPool`.
234 *
235 * When the id (string) generated by the visitor is equal, expressions are considered equivalent.
236 */
237class KeyVisitor {
238 constructor() {
239 this.visitWrappedNodeExpr = invalid;
240 this.visitWriteVarExpr = invalid;
241 this.visitWriteKeyExpr = invalid;
242 this.visitWritePropExpr = invalid;
243 this.visitInvokeMethodExpr = invalid;
244 this.visitInvokeFunctionExpr = invalid;
245 this.visitTaggedTemplateExpr = invalid;
246 this.visitInstantiateExpr = invalid;
247 this.visitConditionalExpr = invalid;
248 this.visitNotExpr = invalid;
249 this.visitAssertNotNullExpr = invalid;
250 this.visitCastExpr = invalid;
251 this.visitFunctionExpr = invalid;
252 this.visitUnaryOperatorExpr = invalid;
253 this.visitBinaryOperatorExpr = invalid;
254 this.visitReadPropExpr = invalid;
255 this.visitReadKeyExpr = invalid;
256 this.visitCommaExpr = invalid;
257 this.visitLocalizedString = invalid;
258 }
259 visitLiteralExpr(ast) {
260 return `${typeof ast.value === 'string' ? '"' + ast.value + '"' : ast.value}`;
261 }
262 visitLiteralArrayExpr(ast, context) {
263 return `[${ast.entries.map(entry => entry.visitExpression(this, context)).join(',')}]`;
264 }
265 visitLiteralMapExpr(ast, context) {
266 const mapKey = (entry) => {
267 const quote = entry.quoted ? '"' : '';
268 return `${quote}${entry.key}${quote}`;
269 };
270 const mapEntry = (entry) => `${mapKey(entry)}:${entry.value.visitExpression(this, context)}`;
271 return `{${ast.entries.map(mapEntry).join(',')}`;
272 }
273 visitExternalExpr(ast) {
274 return ast.value.moduleName ? `EX:${ast.value.moduleName}:${ast.value.name}` :
275 `EX:${ast.value.runtime.name}`;
276 }
277 visitReadVarExpr(node) {
278 return `VAR:${node.name}`;
279 }
280 visitTypeofExpr(node, context) {
281 return `TYPEOF:${node.expr.visitExpression(this, context)}`;
282 }
283}
284function invalid(arg) {
285 throw new Error(`Invalid state: Visitor ${this.constructor.name} doesn't handle ${arg.constructor.name}`);
286}
287function isVariable(e) {
288 return e instanceof o.ReadVarExpr;
289}
290function isLongStringLiteral(expr) {
291 return expr instanceof o.LiteralExpr && typeof expr.value === 'string' &&
292 expr.value.length >= POOL_INCLUSION_LENGTH_THRESHOLD_FOR_STRINGS;
293}
294//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"constant_pool.js","sourceRoot":"","sources":["../../../../../../packages/compiler/src/constant_pool.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 * as o from './output/output_ast';\n\nconst CONSTANT_PREFIX = '_c';\n\n/**\n * `ConstantPool` tries to reuse literal factories when two or more literals are identical.\n * We determine whether literals are identical by creating a key out of their AST using the\n * `KeyVisitor`. This constant is used to replace dynamic expressions which can't be safely\n * converted into a key. E.g. given an expression `{foo: bar()}`, since we don't know what\n * the result of `bar` will be, we create a key that looks like `{foo: <unknown>}`. Note\n * that we use a variable, rather than something like `null` in order to avoid collisions.\n */\nconst UNKNOWN_VALUE_KEY = o.variable('<unknown>');\n\nexport const enum DefinitionKind {\n  Injector,\n  Directive,\n  Component,\n  Pipe\n}\n\n/**\n * Context to use when producing a key.\n *\n * This ensures we see the constant not the reference variable when producing\n * a key.\n */\nconst KEY_CONTEXT = {};\n\n/**\n * Generally all primitive values are excluded from the `ConstantPool`, but there is an exclusion\n * for strings that reach a certain length threshold. This constant defines the length threshold for\n * strings.\n */\nconst POOL_INCLUSION_LENGTH_THRESHOLD_FOR_STRINGS = 50;\n\n/**\n * A node that is a place-holder that allows the node to be replaced when the actual\n * node is known.\n *\n * This allows the constant pool to change an expression from a direct reference to\n * a constant to a shared constant. It returns a fix-up node that is later allowed to\n * change the referenced expression.\n */\nclass FixupExpression extends o.Expression {\n  private original: o.Expression;\n\n  // TODO(issue/24571): remove '!'.\n  shared!: boolean;\n\n  constructor(public resolved: o.Expression) {\n    super(resolved.type);\n    this.original = resolved;\n  }\n\n  override visitExpression(visitor: o.ExpressionVisitor, context: any): any {\n    if (context === KEY_CONTEXT) {\n      // When producing a key we want to traverse the constant not the\n      // variable used to refer to it.\n      return this.original.visitExpression(visitor, context);\n    } else {\n      return this.resolved.visitExpression(visitor, context);\n    }\n  }\n\n  override isEquivalent(e: o.Expression): boolean {\n    return e instanceof FixupExpression && this.resolved.isEquivalent(e.resolved);\n  }\n\n  override isConstant() {\n    return true;\n  }\n\n  fixup(expression: o.Expression) {\n    this.resolved = expression;\n    this.shared = true;\n  }\n}\n\n/**\n * A constant pool allows a code emitter to share constant in an output context.\n *\n * The constant pool also supports sharing access to ivy definitions references.\n */\nexport class ConstantPool {\n  statements: o.Statement[] = [];\n  private literals = new Map<string, FixupExpression>();\n  private literalFactories = new Map<string, o.Expression>();\n  private injectorDefinitions = new Map<any, FixupExpression>();\n  private directiveDefinitions = new Map<any, FixupExpression>();\n  private componentDefinitions = new Map<any, FixupExpression>();\n  private pipeDefinitions = new Map<any, FixupExpression>();\n\n  private nextNameIndex = 0;\n\n  constructor(private readonly isClosureCompilerEnabled: boolean = false) {}\n\n  getConstLiteral(literal: o.Expression, forceShared?: boolean): o.Expression {\n    if ((literal instanceof o.LiteralExpr && !isLongStringLiteral(literal)) ||\n        literal instanceof FixupExpression) {\n      // Do no put simple literals into the constant pool or try to produce a constant for a\n      // reference to a constant.\n      return literal;\n    }\n    const key = this.keyOf(literal);\n    let fixup = this.literals.get(key);\n    let newValue = false;\n    if (!fixup) {\n      fixup = new FixupExpression(literal);\n      this.literals.set(key, fixup);\n      newValue = true;\n    }\n\n    if ((!newValue && !fixup.shared) || (newValue && forceShared)) {\n      // Replace the expression with a variable\n      const name = this.freshName();\n      let definition: o.WriteVarExpr;\n      let usage: o.Expression;\n      if (this.isClosureCompilerEnabled && isLongStringLiteral(literal)) {\n        // For string literals, Closure will **always** inline the string at\n        // **all** usages, duplicating it each time. For large strings, this\n        // unnecessarily bloats bundle size. To work around this restriction, we\n        // wrap the string in a function, and call that function for each usage.\n        // This tricks Closure into using inline logic for functions instead of\n        // string literals. Function calls are only inlined if the body is small\n        // enough to be worth it. By doing this, very large strings will be\n        // shared across multiple usages, rather than duplicating the string at\n        // each usage site.\n        //\n        // const myStr = function() { return \"very very very long string\"; };\n        // const usage1 = myStr();\n        // const usage2 = myStr();\n        definition = o.variable(name).set(new o.FunctionExpr(\n            [],  // Params.\n            [\n              // Statements.\n              new o.ReturnStatement(literal),\n            ],\n            ));\n        usage = o.variable(name).callFn([]);\n      } else {\n        // Just declare and use the variable directly, without a function call\n        // indirection. This saves a few bytes and avoids an unncessary call.\n        definition = o.variable(name).set(literal);\n        usage = o.variable(name);\n      }\n\n      this.statements.push(definition.toDeclStmt(o.INFERRED_TYPE, [o.StmtModifier.Final]));\n      fixup.fixup(usage);\n    }\n\n    return fixup;\n  }\n\n  getDefinition(type: any, kind: DefinitionKind, ctx: OutputContext, forceShared: boolean = false):\n      o.Expression {\n    const definitions = this.definitionsOf(kind);\n    let fixup = definitions.get(type);\n    let newValue = false;\n    if (!fixup) {\n      const property = this.propertyNameOf(kind);\n      fixup = new FixupExpression(ctx.importExpr(type).prop(property));\n      definitions.set(type, fixup);\n      newValue = true;\n    }\n\n    if ((!newValue && !fixup.shared) || (newValue && forceShared)) {\n      const name = this.freshName();\n      this.statements.push(\n          o.variable(name).set(fixup.resolved).toDeclStmt(o.INFERRED_TYPE, [o.StmtModifier.Final]));\n      fixup.fixup(o.variable(name));\n    }\n    return fixup;\n  }\n\n  getLiteralFactory(literal: o.LiteralArrayExpr|o.LiteralMapExpr):\n      {literalFactory: o.Expression, literalFactoryArguments: o.Expression[]} {\n    // Create a pure function that builds an array of a mix of constant and variable expressions\n    if (literal instanceof o.LiteralArrayExpr) {\n      const argumentsForKey = literal.entries.map(e => e.isConstant() ? e : UNKNOWN_VALUE_KEY);\n      const key = this.keyOf(o.literalArr(argumentsForKey));\n      return this._getLiteralFactory(key, literal.entries, entries => o.literalArr(entries));\n    } else {\n      const expressionForKey = o.literalMap(\n          literal.entries.map(e => ({\n                                key: e.key,\n                                value: e.value.isConstant() ? e.value : UNKNOWN_VALUE_KEY,\n                                quoted: e.quoted\n                              })));\n      const key = this.keyOf(expressionForKey);\n      return this._getLiteralFactory(\n          key, literal.entries.map(e => e.value),\n          entries => o.literalMap(entries.map((value, index) => ({\n                                                key: literal.entries[index].key,\n                                                value,\n                                                quoted: literal.entries[index].quoted\n                                              }))));\n    }\n  }\n\n  private _getLiteralFactory(\n      key: string, values: o.Expression[], resultMap: (parameters: o.Expression[]) => o.Expression):\n      {literalFactory: o.Expression, literalFactoryArguments: o.Expression[]} {\n    let literalFactory = this.literalFactories.get(key);\n    const literalFactoryArguments = values.filter((e => !e.isConstant()));\n    if (!literalFactory) {\n      const resultExpressions = values.map(\n          (e, index) => e.isConstant() ? this.getConstLiteral(e, true) : o.variable(`a${index}`));\n      const parameters =\n          resultExpressions.filter(isVariable).map(e => new o.FnParam(e.name!, o.DYNAMIC_TYPE));\n      const pureFunctionDeclaration =\n          o.fn(parameters, [new o.ReturnStatement(resultMap(resultExpressions))], o.INFERRED_TYPE);\n      const name = this.freshName();\n      this.statements.push(\n          o.variable(name).set(pureFunctionDeclaration).toDeclStmt(o.INFERRED_TYPE, [\n            o.StmtModifier.Final\n          ]));\n      literalFactory = o.variable(name);\n      this.literalFactories.set(key, literalFactory);\n    }\n    return {literalFactory, literalFactoryArguments};\n  }\n\n  /**\n   * Produce a unique name.\n   *\n   * The name might be unique among different prefixes if any of the prefixes end in\n   * a digit so the prefix should be a constant string (not based on user input) and\n   * must not end in a digit.\n   */\n  uniqueName(prefix: string): string {\n    return `${prefix}${this.nextNameIndex++}`;\n  }\n\n  private definitionsOf(kind: DefinitionKind): Map<any, FixupExpression> {\n    switch (kind) {\n      case DefinitionKind.Component:\n        return this.componentDefinitions;\n      case DefinitionKind.Directive:\n        return this.directiveDefinitions;\n      case DefinitionKind.Injector:\n        return this.injectorDefinitions;\n      case DefinitionKind.Pipe:\n        return this.pipeDefinitions;\n    }\n  }\n\n  public propertyNameOf(kind: DefinitionKind): string {\n    switch (kind) {\n      case DefinitionKind.Component:\n        return 'ɵcmp';\n      case DefinitionKind.Directive:\n        return 'ɵdir';\n      case DefinitionKind.Injector:\n        return 'ɵinj';\n      case DefinitionKind.Pipe:\n        return 'ɵpipe';\n    }\n  }\n\n  private freshName(): string {\n    return this.uniqueName(CONSTANT_PREFIX);\n  }\n\n  private keyOf(expression: o.Expression) {\n    return expression.visitExpression(new KeyVisitor(), KEY_CONTEXT);\n  }\n}\n\nexport interface OutputContext {\n  genFilePath: string;\n  statements: o.Statement[];\n  constantPool: ConstantPool;\n  importExpr(reference: any, typeParams?: o.Type[]|null, useSummaries?: boolean): o.Expression;\n}\n\n/**\n * Visitor used to determine if 2 expressions are equivalent and can be shared in the\n * `ConstantPool`.\n *\n * When the id (string) generated by the visitor is equal, expressions are considered equivalent.\n */\nclass KeyVisitor implements o.ExpressionVisitor {\n  visitLiteralExpr(ast: o.LiteralExpr): string {\n    return `${typeof ast.value === 'string' ? '\"' + ast.value + '\"' : ast.value}`;\n  }\n\n  visitLiteralArrayExpr(ast: o.LiteralArrayExpr, context: object): string {\n    return `[${ast.entries.map(entry => entry.visitExpression(this, context)).join(',')}]`;\n  }\n\n  visitLiteralMapExpr(ast: o.LiteralMapExpr, context: object): string {\n    const mapKey = (entry: o.LiteralMapEntry) => {\n      const quote = entry.quoted ? '\"' : '';\n      return `${quote}${entry.key}${quote}`;\n    };\n    const mapEntry = (entry: o.LiteralMapEntry) =>\n        `${mapKey(entry)}:${entry.value.visitExpression(this, context)}`;\n    return `{${ast.entries.map(mapEntry).join(',')}`;\n  }\n\n  visitExternalExpr(ast: o.ExternalExpr): string {\n    return ast.value.moduleName ? `EX:${ast.value.moduleName}:${ast.value.name}` :\n                                  `EX:${ast.value.runtime.name}`;\n  }\n\n  visitReadVarExpr(node: o.ReadVarExpr) {\n    return `VAR:${node.name}`;\n  }\n\n  visitTypeofExpr(node: o.TypeofExpr, context: any): string {\n    return `TYPEOF:${node.expr.visitExpression(this, context)}`;\n  }\n\n  visitWrappedNodeExpr = invalid;\n  visitWriteVarExpr = invalid;\n  visitWriteKeyExpr = invalid;\n  visitWritePropExpr = invalid;\n  visitInvokeMethodExpr = invalid;\n  visitInvokeFunctionExpr = invalid;\n  visitTaggedTemplateExpr = invalid;\n  visitInstantiateExpr = invalid;\n  visitConditionalExpr = invalid;\n  visitNotExpr = invalid;\n  visitAssertNotNullExpr = invalid;\n  visitCastExpr = invalid;\n  visitFunctionExpr = invalid;\n  visitUnaryOperatorExpr = invalid;\n  visitBinaryOperatorExpr = invalid;\n  visitReadPropExpr = invalid;\n  visitReadKeyExpr = invalid;\n  visitCommaExpr = invalid;\n  visitLocalizedString = invalid;\n}\n\nfunction invalid<T>(this: o.ExpressionVisitor, arg: o.Expression|o.Statement): never {\n  throw new Error(\n      `Invalid state: Visitor ${this.constructor.name} doesn't handle ${arg.constructor.name}`);\n}\n\nfunction isVariable(e: o.Expression): e is o.ReadVarExpr {\n  return e instanceof o.ReadVarExpr;\n}\n\nfunction isLongStringLiteral(expr: o.Expression): boolean {\n  return expr instanceof o.LiteralExpr && typeof expr.value === 'string' &&\n      expr.value.length >= POOL_INCLUSION_LENGTH_THRESHOLD_FOR_STRINGS;\n}\n"]}
Note: See TracBrowser for help on using the repository browser.