UNPKG

69.2 kBJavaScriptView Raw
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(function (factory) {
9 if (typeof module === "object" && typeof module.exports === "object") {
10 var v = factory(require, exports);
11 if (v !== undefined) module.exports = v;
12 }
13 else if (typeof define === "function" && define.amd) {
14 define("@angular/language-service/src/expression_type", ["require", "exports", "tslib", "@angular/compiler", "@angular/language-service/src/diagnostic_messages", "@angular/language-service/src/symbols", "@angular/language-service/src/utils"], factory);
15 }
16})(function (require, exports) {
17 "use strict";
18 Object.defineProperty(exports, "__esModule", { value: true });
19 exports.AstType = void 0;
20 var tslib_1 = require("tslib");
21 var compiler_1 = require("@angular/compiler");
22 var diagnostic_messages_1 = require("@angular/language-service/src/diagnostic_messages");
23 var symbols_1 = require("@angular/language-service/src/symbols");
24 var utils_1 = require("@angular/language-service/src/utils");
25 // AstType calculatetype of the ast given AST element.
26 var AstType = /** @class */ (function () {
27 function AstType(scope, query, context, source) {
28 this.scope = scope;
29 this.query = query;
30 this.context = context;
31 this.source = source;
32 this.diagnostics = [];
33 }
34 AstType.prototype.getType = function (ast) {
35 return ast.visit(this);
36 };
37 AstType.prototype.getDiagnostics = function (ast) {
38 var type = ast.visit(this);
39 if (this.context.inEvent && type.callable) {
40 this.diagnostics.push(diagnostic_messages_1.createDiagnostic(refinedSpan(ast), diagnostic_messages_1.Diagnostic.callable_expression_expected_method_call));
41 }
42 return this.diagnostics;
43 };
44 AstType.prototype.visitUnary = function (ast) {
45 // Visit the child to produce diagnostics.
46 ast.expr.visit(this);
47 // The unary plus and minus operator are always of type number.
48 // https://github.com/Microsoft/TypeScript/blob/v1.8.10/doc/spec.md#4.18
49 switch (ast.operator) {
50 case '-':
51 case '+':
52 return this.query.getBuiltinType(symbols_1.BuiltinType.Number);
53 }
54 this.diagnostics.push(diagnostic_messages_1.createDiagnostic(refinedSpan(ast), diagnostic_messages_1.Diagnostic.unrecognized_operator, ast.operator));
55 return this.anyType;
56 };
57 AstType.prototype.visitBinary = function (ast) {
58 var _this_1 = this;
59 var getType = function (ast, operation) {
60 var type = _this_1.getType(ast);
61 if (type.nullable) {
62 switch (operation) {
63 case '&&':
64 case '||':
65 case '==':
66 case '!=':
67 case '===':
68 case '!==':
69 // Nullable allowed.
70 break;
71 default:
72 _this_1.diagnostics.push(diagnostic_messages_1.createDiagnostic(refinedSpan(ast), diagnostic_messages_1.Diagnostic.expression_might_be_null));
73 break;
74 }
75 }
76 return type;
77 };
78 var leftType = getType(ast.left, ast.operation);
79 var rightType = getType(ast.right, ast.operation);
80 var leftKind = this.query.getTypeKind(leftType);
81 var rightKind = this.query.getTypeKind(rightType);
82 // The following swtich implements operator typing similar to the
83 // type production tables in the TypeScript specification.
84 // https://github.com/Microsoft/TypeScript/blob/v1.8.10/doc/spec.md#4.19
85 var operKind = leftKind << 8 | rightKind;
86 switch (ast.operation) {
87 case '*':
88 case '/':
89 case '%':
90 case '-':
91 case '<<':
92 case '>>':
93 case '>>>':
94 case '&':
95 case '^':
96 case '|':
97 switch (operKind) {
98 case symbols_1.BuiltinType.Any << 8 | symbols_1.BuiltinType.Any:
99 case symbols_1.BuiltinType.Number << 8 | symbols_1.BuiltinType.Any:
100 case symbols_1.BuiltinType.Any << 8 | symbols_1.BuiltinType.Number:
101 case symbols_1.BuiltinType.Number << 8 | symbols_1.BuiltinType.Number:
102 return this.query.getBuiltinType(symbols_1.BuiltinType.Number);
103 default:
104 var errorAst = ast.left;
105 switch (leftKind) {
106 case symbols_1.BuiltinType.Any:
107 case symbols_1.BuiltinType.Number:
108 errorAst = ast.right;
109 break;
110 }
111 this.diagnostics.push(diagnostic_messages_1.createDiagnostic(errorAst.span, diagnostic_messages_1.Diagnostic.expected_a_number_type));
112 return this.anyType;
113 }
114 case '+':
115 switch (operKind) {
116 case symbols_1.BuiltinType.Any << 8 | symbols_1.BuiltinType.Any:
117 case symbols_1.BuiltinType.Any << 8 | symbols_1.BuiltinType.Boolean:
118 case symbols_1.BuiltinType.Any << 8 | symbols_1.BuiltinType.Number:
119 case symbols_1.BuiltinType.Any << 8 | symbols_1.BuiltinType.Other:
120 case symbols_1.BuiltinType.Boolean << 8 | symbols_1.BuiltinType.Any:
121 case symbols_1.BuiltinType.Number << 8 | symbols_1.BuiltinType.Any:
122 case symbols_1.BuiltinType.Other << 8 | symbols_1.BuiltinType.Any:
123 return this.anyType;
124 case symbols_1.BuiltinType.Any << 8 | symbols_1.BuiltinType.String:
125 case symbols_1.BuiltinType.Boolean << 8 | symbols_1.BuiltinType.String:
126 case symbols_1.BuiltinType.Number << 8 | symbols_1.BuiltinType.String:
127 case symbols_1.BuiltinType.String << 8 | symbols_1.BuiltinType.Any:
128 case symbols_1.BuiltinType.String << 8 | symbols_1.BuiltinType.Boolean:
129 case symbols_1.BuiltinType.String << 8 | symbols_1.BuiltinType.Number:
130 case symbols_1.BuiltinType.String << 8 | symbols_1.BuiltinType.String:
131 case symbols_1.BuiltinType.String << 8 | symbols_1.BuiltinType.Other:
132 case symbols_1.BuiltinType.Other << 8 | symbols_1.BuiltinType.String:
133 return this.query.getBuiltinType(symbols_1.BuiltinType.String);
134 case symbols_1.BuiltinType.Number << 8 | symbols_1.BuiltinType.Number:
135 return this.query.getBuiltinType(symbols_1.BuiltinType.Number);
136 case symbols_1.BuiltinType.Boolean << 8 | symbols_1.BuiltinType.Number:
137 case symbols_1.BuiltinType.Other << 8 | symbols_1.BuiltinType.Number:
138 this.diagnostics.push(diagnostic_messages_1.createDiagnostic(ast.left.span, diagnostic_messages_1.Diagnostic.expected_a_number_type));
139 return this.anyType;
140 case symbols_1.BuiltinType.Number << 8 | symbols_1.BuiltinType.Boolean:
141 case symbols_1.BuiltinType.Number << 8 | symbols_1.BuiltinType.Other:
142 this.diagnostics.push(diagnostic_messages_1.createDiagnostic(ast.right.span, diagnostic_messages_1.Diagnostic.expected_a_number_type));
143 return this.anyType;
144 default:
145 this.diagnostics.push(diagnostic_messages_1.createDiagnostic(refinedSpan(ast), diagnostic_messages_1.Diagnostic.expected_a_string_or_number_type));
146 return this.anyType;
147 }
148 case '>':
149 case '<':
150 case '<=':
151 case '>=':
152 case '==':
153 case '!=':
154 case '===':
155 case '!==':
156 if (!(leftKind & rightKind) &&
157 !((leftKind | rightKind) & (symbols_1.BuiltinType.Null | symbols_1.BuiltinType.Undefined))) {
158 // Two values are comparable only if
159 // - they have some type overlap, or
160 // - at least one is not defined
161 this.diagnostics.push(diagnostic_messages_1.createDiagnostic(refinedSpan(ast), diagnostic_messages_1.Diagnostic.expected_operands_of_comparable_types_or_any));
162 }
163 return this.query.getBuiltinType(symbols_1.BuiltinType.Boolean);
164 case '&&':
165 return rightType;
166 case '||':
167 return this.query.getTypeUnion(leftType, rightType);
168 }
169 this.diagnostics.push(diagnostic_messages_1.createDiagnostic(refinedSpan(ast), diagnostic_messages_1.Diagnostic.unrecognized_operator, ast.operation));
170 return this.anyType;
171 };
172 AstType.prototype.visitChain = function (ast) {
173 var e_1, _a;
174 try {
175 // If we are producing diagnostics, visit the children
176 for (var _b = tslib_1.__values(ast.expressions), _c = _b.next(); !_c.done; _c = _b.next()) {
177 var expr = _c.value;
178 expr.visit(this);
179 }
180 }
181 catch (e_1_1) { e_1 = { error: e_1_1 }; }
182 finally {
183 try {
184 if (_c && !_c.done && (_a = _b.return)) _a.call(_b);
185 }
186 finally { if (e_1) throw e_1.error; }
187 }
188 // The type of a chain is always undefined.
189 return this.query.getBuiltinType(symbols_1.BuiltinType.Undefined);
190 };
191 AstType.prototype.visitConditional = function (ast) {
192 // The type of a conditional is the union of the true and false conditions.
193 ast.condition.visit(this);
194 ast.trueExp.visit(this);
195 ast.falseExp.visit(this);
196 return this.query.getTypeUnion(this.getType(ast.trueExp), this.getType(ast.falseExp));
197 };
198 AstType.prototype.visitFunctionCall = function (ast) {
199 var _this_1 = this;
200 // The type of a function call is the return type of the selected signature.
201 // The signature is selected based on the types of the arguments. Angular doesn't
202 // support contextual typing of arguments so this is simpler than TypeScript's
203 // version.
204 var args = ast.args.map(function (arg) { return _this_1.getType(arg); });
205 var target = this.getType(ast.target);
206 if (!target || !target.callable) {
207 this.diagnostics.push(diagnostic_messages_1.createDiagnostic(refinedSpan(ast), diagnostic_messages_1.Diagnostic.call_target_not_callable, this.sourceOf(ast.target), target.name));
208 return this.anyType;
209 }
210 var signature = target.selectSignature(args);
211 if (signature) {
212 return signature.result;
213 }
214 // TODO: Consider a better error message here. See `typescript_symbols#selectSignature` for more
215 // details.
216 this.diagnostics.push(diagnostic_messages_1.createDiagnostic(refinedSpan(ast), diagnostic_messages_1.Diagnostic.unable_to_resolve_compatible_call_signature));
217 return this.anyType;
218 };
219 AstType.prototype.visitImplicitReceiver = function (_ast) {
220 var _this = this;
221 // Return a pseudo-symbol for the implicit receiver.
222 // The members of the implicit receiver are what is defined by the
223 // scope passed into this class.
224 return {
225 name: '$implicit',
226 kind: 'component',
227 language: 'ng-template',
228 type: undefined,
229 container: undefined,
230 callable: false,
231 nullable: false,
232 public: true,
233 definition: undefined,
234 documentation: [],
235 members: function () {
236 return _this.scope;
237 },
238 signatures: function () {
239 return [];
240 },
241 selectSignature: function (_types) {
242 return undefined;
243 },
244 indexed: function (_argument) {
245 return undefined;
246 },
247 typeArguments: function () {
248 return undefined;
249 },
250 };
251 };
252 AstType.prototype.visitThisReceiver = function (_ast) {
253 return this.visitImplicitReceiver(_ast);
254 };
255 AstType.prototype.visitInterpolation = function (ast) {
256 var e_2, _a;
257 try {
258 // If we are producing diagnostics, visit the children.
259 for (var _b = tslib_1.__values(ast.expressions), _c = _b.next(); !_c.done; _c = _b.next()) {
260 var expr = _c.value;
261 expr.visit(this);
262 }
263 }
264 catch (e_2_1) { e_2 = { error: e_2_1 }; }
265 finally {
266 try {
267 if (_c && !_c.done && (_a = _b.return)) _a.call(_b);
268 }
269 finally { if (e_2) throw e_2.error; }
270 }
271 return this.undefinedType;
272 };
273 AstType.prototype.visitKeyedRead = function (ast) {
274 var targetType = this.getType(ast.obj);
275 var keyType = this.getType(ast.key);
276 var result = targetType.indexed(keyType, ast.key instanceof compiler_1.LiteralPrimitive ? ast.key.value : undefined);
277 return result || this.anyType;
278 };
279 AstType.prototype.visitKeyedWrite = function (ast) {
280 // The write of a type is the type of the value being written.
281 return this.getType(ast.value);
282 };
283 AstType.prototype.visitLiteralArray = function (ast) {
284 var _a;
285 var _this_1 = this;
286 // A type literal is an array type of the union of the elements
287 return this.query.getArrayType((_a = this.query).getTypeUnion.apply(_a, tslib_1.__spread(ast.expressions.map(function (element) { return _this_1.getType(element); }))));
288 };
289 AstType.prototype.visitLiteralMap = function (ast) {
290 var e_3, _a;
291 try {
292 // If we are producing diagnostics, visit the children
293 for (var _b = tslib_1.__values(ast.values), _c = _b.next(); !_c.done; _c = _b.next()) {
294 var value = _c.value;
295 value.visit(this);
296 }
297 }
298 catch (e_3_1) { e_3 = { error: e_3_1 }; }
299 finally {
300 try {
301 if (_c && !_c.done && (_a = _b.return)) _a.call(_b);
302 }
303 finally { if (e_3) throw e_3.error; }
304 }
305 // TODO: Return a composite type.
306 return this.anyType;
307 };
308 AstType.prototype.visitLiteralPrimitive = function (ast) {
309 // The type of a literal primitive depends on the value of the literal.
310 switch (ast.value) {
311 case true:
312 case false:
313 return this.query.getBuiltinType(symbols_1.BuiltinType.Boolean);
314 case null:
315 return this.query.getBuiltinType(symbols_1.BuiltinType.Null);
316 case undefined:
317 return this.query.getBuiltinType(symbols_1.BuiltinType.Undefined);
318 default:
319 switch (typeof ast.value) {
320 case 'string':
321 return this.query.getBuiltinType(symbols_1.BuiltinType.String);
322 case 'number':
323 return this.query.getBuiltinType(symbols_1.BuiltinType.Number);
324 default:
325 this.diagnostics.push(diagnostic_messages_1.createDiagnostic(refinedSpan(ast), diagnostic_messages_1.Diagnostic.unrecognized_primitive, typeof ast.value));
326 return this.anyType;
327 }
328 }
329 };
330 AstType.prototype.visitMethodCall = function (ast) {
331 return this.resolveMethodCall(this.getType(ast.receiver), ast);
332 };
333 AstType.prototype.visitPipe = function (ast) {
334 var _this_1 = this;
335 // The type of a pipe node is the return type of the pipe's transform method. The table returned
336 // by getPipes() is expected to contain symbols with the corresponding transform method type.
337 var pipe = this.query.getPipes().get(ast.name);
338 if (!pipe) {
339 this.diagnostics.push(diagnostic_messages_1.createDiagnostic(refinedSpan(ast), diagnostic_messages_1.Diagnostic.no_pipe_found, ast.name));
340 return this.anyType;
341 }
342 var expType = this.getType(ast.exp);
343 var signature = pipe.selectSignature([expType].concat(ast.args.map(function (arg) { return _this_1.getType(arg); })));
344 if (!signature) {
345 this.diagnostics.push(diagnostic_messages_1.createDiagnostic(refinedSpan(ast), diagnostic_messages_1.Diagnostic.unable_to_resolve_signature, ast.name));
346 return this.anyType;
347 }
348 return signature.result;
349 };
350 AstType.prototype.visitPrefixNot = function (ast) {
351 // If we are producing diagnostics, visit the children
352 ast.expression.visit(this);
353 // The type of a prefix ! is always boolean.
354 return this.query.getBuiltinType(symbols_1.BuiltinType.Boolean);
355 };
356 AstType.prototype.visitNonNullAssert = function (ast) {
357 var expressionType = this.getType(ast.expression);
358 return this.query.getNonNullableType(expressionType);
359 };
360 AstType.prototype.visitPropertyRead = function (ast) {
361 return this.resolvePropertyRead(this.getType(ast.receiver), ast);
362 };
363 AstType.prototype.visitPropertyWrite = function (ast) {
364 // The type of a write is the type of the value being written.
365 return this.getType(ast.value);
366 };
367 AstType.prototype.visitQuote = function (_ast) {
368 // The type of a quoted expression is any.
369 return this.query.getBuiltinType(symbols_1.BuiltinType.Any);
370 };
371 AstType.prototype.visitSafeMethodCall = function (ast) {
372 return this.resolveMethodCall(this.query.getNonNullableType(this.getType(ast.receiver)), ast);
373 };
374 AstType.prototype.visitSafePropertyRead = function (ast) {
375 return this.resolvePropertyRead(this.query.getNonNullableType(this.getType(ast.receiver)), ast);
376 };
377 /**
378 * Gets the source of an expession AST.
379 * The AST's sourceSpan is relative to the start of the template source code, which is contained
380 * at this.source.
381 */
382 AstType.prototype.sourceOf = function (ast) {
383 return this.source.substring(ast.sourceSpan.start, ast.sourceSpan.end);
384 };
385 Object.defineProperty(AstType.prototype, "anyType", {
386 get: function () {
387 var result = this._anyType;
388 if (!result) {
389 result = this._anyType = this.query.getBuiltinType(symbols_1.BuiltinType.Any);
390 }
391 return result;
392 },
393 enumerable: false,
394 configurable: true
395 });
396 Object.defineProperty(AstType.prototype, "undefinedType", {
397 get: function () {
398 var result = this._undefinedType;
399 if (!result) {
400 result = this._undefinedType = this.query.getBuiltinType(symbols_1.BuiltinType.Undefined);
401 }
402 return result;
403 },
404 enumerable: false,
405 configurable: true
406 });
407 AstType.prototype.resolveMethodCall = function (receiverType, ast) {
408 var _this_1 = this;
409 if (this.isAny(receiverType)) {
410 return this.anyType;
411 }
412 var methodType = this.resolvePropertyRead(receiverType, ast);
413 if (!methodType) {
414 this.diagnostics.push(diagnostic_messages_1.createDiagnostic(refinedSpan(ast), diagnostic_messages_1.Diagnostic.could_not_resolve_type, ast.name));
415 return this.anyType;
416 }
417 if (this.isAny(methodType)) {
418 return this.anyType;
419 }
420 if (!methodType.callable) {
421 this.diagnostics.push(diagnostic_messages_1.createDiagnostic(refinedSpan(ast), diagnostic_messages_1.Diagnostic.identifier_not_callable, ast.name));
422 return this.anyType;
423 }
424 var signature = methodType.selectSignature(ast.args.map(function (arg) { return _this_1.getType(arg); }));
425 if (!signature) {
426 this.diagnostics.push(diagnostic_messages_1.createDiagnostic(refinedSpan(ast), diagnostic_messages_1.Diagnostic.unable_to_resolve_signature, ast.name));
427 return this.anyType;
428 }
429 return signature.result;
430 };
431 AstType.prototype.resolvePropertyRead = function (receiverType, ast) {
432 if (this.isAny(receiverType)) {
433 return this.anyType;
434 }
435 // The type of a property read is the seelcted member's type.
436 var member = receiverType.members().get(ast.name);
437 if (!member) {
438 if (receiverType.name === '$implicit') {
439 this.diagnostics.push(diagnostic_messages_1.createDiagnostic(refinedSpan(ast), diagnostic_messages_1.Diagnostic.identifier_not_defined_in_app_context, ast.name));
440 }
441 else if (receiverType.nullable && ast.receiver instanceof compiler_1.PropertyRead) {
442 var receiver = ast.receiver.name;
443 this.diagnostics.push(diagnostic_messages_1.createDiagnostic(refinedSpan(ast), diagnostic_messages_1.Diagnostic.identifier_possibly_undefined, receiver, receiver + "?." + ast.name, receiver + "!." + ast.name));
444 }
445 else {
446 this.diagnostics.push(diagnostic_messages_1.createDiagnostic(refinedSpan(ast), diagnostic_messages_1.Diagnostic.identifier_not_defined_on_receiver, ast.name, receiverType.name));
447 }
448 return this.anyType;
449 }
450 if (!member.public) {
451 var container = receiverType.name === '$implicit' ? 'the component' : "'" + receiverType.name + "'";
452 this.diagnostics.push(diagnostic_messages_1.createDiagnostic(refinedSpan(ast), diagnostic_messages_1.Diagnostic.identifier_is_private, ast.name, container));
453 }
454 return member.type;
455 };
456 AstType.prototype.isAny = function (symbol) {
457 return !symbol || this.query.getTypeKind(symbol) === symbols_1.BuiltinType.Any ||
458 (!!symbol.type && this.isAny(symbol.type));
459 };
460 return AstType;
461 }());
462 exports.AstType = AstType;
463 function refinedSpan(ast) {
464 // nameSpan is an absolute span, but the spans returned by the expression visitor are expected to
465 // be relative to the start of the expression.
466 // TODO: migrate to only using absolute spans
467 var absoluteOffset = ast.sourceSpan.start - ast.span.start;
468 if (ast instanceof compiler_1.ASTWithName) {
469 return utils_1.offsetSpan(ast.nameSpan, -absoluteOffset);
470 }
471 return utils_1.offsetSpan(ast.sourceSpan, -absoluteOffset);
472 }
473});
474//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"expression_type.js","sourceRoot":"","sources":["../../../../../../packages/language-service/src/expression_type.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 {AST, AstVisitor, ASTWithName, Binary, BindingPipe, Chain, Conditional, FunctionCall, ImplicitReceiver, Interpolation, KeyedRead, KeyedWrite, LiteralArray, LiteralMap, LiteralPrimitive, MethodCall, NonNullAssert, PrefixNot, PropertyRead, PropertyWrite, Quote, SafeMethodCall, SafePropertyRead, ThisReceiver, Unary} from '@angular/compiler';\n\nimport {createDiagnostic, Diagnostic} from './diagnostic_messages';\nimport {BuiltinType, Signature, Symbol, SymbolQuery, SymbolTable} from './symbols';\nimport * as ng from './types';\nimport {offsetSpan} from './utils';\n\ninterface ExpressionDiagnosticsContext {\n  inEvent?: boolean;\n}\n\n// AstType calculatetype of the ast given AST element.\nexport class AstType implements AstVisitor {\n  private readonly diagnostics: ng.Diagnostic[] = [];\n\n  constructor(\n      private scope: SymbolTable, private query: SymbolQuery,\n      private context: ExpressionDiagnosticsContext, private source: string) {}\n\n  getType(ast: AST): Symbol {\n    return ast.visit(this);\n  }\n\n  getDiagnostics(ast: AST): ng.Diagnostic[] {\n    const type: Symbol = ast.visit(this);\n    if (this.context.inEvent && type.callable) {\n      this.diagnostics.push(\n          createDiagnostic(refinedSpan(ast), Diagnostic.callable_expression_expected_method_call));\n    }\n    return this.diagnostics;\n  }\n\n  visitUnary(ast: Unary): Symbol {\n    // Visit the child to produce diagnostics.\n    ast.expr.visit(this);\n\n    // The unary plus and minus operator are always of type number.\n    // https://github.com/Microsoft/TypeScript/blob/v1.8.10/doc/spec.md#4.18\n    switch (ast.operator) {\n      case '-':\n      case '+':\n        return this.query.getBuiltinType(BuiltinType.Number);\n    }\n\n    this.diagnostics.push(\n        createDiagnostic(refinedSpan(ast), Diagnostic.unrecognized_operator, ast.operator));\n    return this.anyType;\n  }\n\n  visitBinary(ast: Binary): Symbol {\n    const getType = (ast: AST, operation: string): Symbol => {\n      const type = this.getType(ast);\n      if (type.nullable) {\n        switch (operation) {\n          case '&&':\n          case '||':\n          case '==':\n          case '!=':\n          case '===':\n          case '!==':\n            // Nullable allowed.\n            break;\n          default:\n            this.diagnostics.push(\n                createDiagnostic(refinedSpan(ast), Diagnostic.expression_might_be_null));\n            break;\n        }\n      }\n      return type;\n    };\n\n    const leftType = getType(ast.left, ast.operation);\n    const rightType = getType(ast.right, ast.operation);\n    const leftKind = this.query.getTypeKind(leftType);\n    const rightKind = this.query.getTypeKind(rightType);\n\n    // The following swtich implements operator typing similar to the\n    // type production tables in the TypeScript specification.\n    // https://github.com/Microsoft/TypeScript/blob/v1.8.10/doc/spec.md#4.19\n    const operKind = leftKind << 8 | rightKind;\n    switch (ast.operation) {\n      case '*':\n      case '/':\n      case '%':\n      case '-':\n      case '<<':\n      case '>>':\n      case '>>>':\n      case '&':\n      case '^':\n      case '|':\n        switch (operKind) {\n          case BuiltinType.Any << 8 | BuiltinType.Any:\n          case BuiltinType.Number << 8 | BuiltinType.Any:\n          case BuiltinType.Any << 8 | BuiltinType.Number:\n          case BuiltinType.Number << 8 | BuiltinType.Number:\n            return this.query.getBuiltinType(BuiltinType.Number);\n          default:\n            let errorAst = ast.left;\n            switch (leftKind) {\n              case BuiltinType.Any:\n              case BuiltinType.Number:\n                errorAst = ast.right;\n                break;\n            }\n            this.diagnostics.push(\n                createDiagnostic(errorAst.span, Diagnostic.expected_a_number_type));\n            return this.anyType;\n        }\n      case '+':\n        switch (operKind) {\n          case BuiltinType.Any << 8 | BuiltinType.Any:\n          case BuiltinType.Any << 8 | BuiltinType.Boolean:\n          case BuiltinType.Any << 8 | BuiltinType.Number:\n          case BuiltinType.Any << 8 | BuiltinType.Other:\n          case BuiltinType.Boolean << 8 | BuiltinType.Any:\n          case BuiltinType.Number << 8 | BuiltinType.Any:\n          case BuiltinType.Other << 8 | BuiltinType.Any:\n            return this.anyType;\n          case BuiltinType.Any << 8 | BuiltinType.String:\n          case BuiltinType.Boolean << 8 | BuiltinType.String:\n          case BuiltinType.Number << 8 | BuiltinType.String:\n          case BuiltinType.String << 8 | BuiltinType.Any:\n          case BuiltinType.String << 8 | BuiltinType.Boolean:\n          case BuiltinType.String << 8 | BuiltinType.Number:\n          case BuiltinType.String << 8 | BuiltinType.String:\n          case BuiltinType.String << 8 | BuiltinType.Other:\n          case BuiltinType.Other << 8 | BuiltinType.String:\n            return this.query.getBuiltinType(BuiltinType.String);\n          case BuiltinType.Number << 8 | BuiltinType.Number:\n            return this.query.getBuiltinType(BuiltinType.Number);\n          case BuiltinType.Boolean << 8 | BuiltinType.Number:\n          case BuiltinType.Other << 8 | BuiltinType.Number:\n            this.diagnostics.push(\n                createDiagnostic(ast.left.span, Diagnostic.expected_a_number_type));\n            return this.anyType;\n          case BuiltinType.Number << 8 | BuiltinType.Boolean:\n          case BuiltinType.Number << 8 | BuiltinType.Other:\n            this.diagnostics.push(\n                createDiagnostic(ast.right.span, Diagnostic.expected_a_number_type));\n            return this.anyType;\n          default:\n            this.diagnostics.push(\n                createDiagnostic(refinedSpan(ast), Diagnostic.expected_a_string_or_number_type));\n            return this.anyType;\n        }\n      case '>':\n      case '<':\n      case '<=':\n      case '>=':\n      case '==':\n      case '!=':\n      case '===':\n      case '!==':\n        if (!(leftKind & rightKind) &&\n            !((leftKind | rightKind) & (BuiltinType.Null | BuiltinType.Undefined))) {\n          // Two values are comparable only if\n          //   - they have some type overlap, or\n          //   - at least one is not defined\n          this.diagnostics.push(createDiagnostic(\n              refinedSpan(ast), Diagnostic.expected_operands_of_comparable_types_or_any));\n        }\n        return this.query.getBuiltinType(BuiltinType.Boolean);\n      case '&&':\n        return rightType;\n      case '||':\n        return this.query.getTypeUnion(leftType, rightType);\n    }\n\n    this.diagnostics.push(\n        createDiagnostic(refinedSpan(ast), Diagnostic.unrecognized_operator, ast.operation));\n    return this.anyType;\n  }\n\n  visitChain(ast: Chain) {\n    // If we are producing diagnostics, visit the children\n    for (const expr of ast.expressions) {\n      expr.visit(this);\n    }\n    // The type of a chain is always undefined.\n    return this.query.getBuiltinType(BuiltinType.Undefined);\n  }\n\n  visitConditional(ast: Conditional) {\n    // The type of a conditional is the union of the true and false conditions.\n    ast.condition.visit(this);\n    ast.trueExp.visit(this);\n    ast.falseExp.visit(this);\n    return this.query.getTypeUnion(this.getType(ast.trueExp), this.getType(ast.falseExp));\n  }\n\n  visitFunctionCall(ast: FunctionCall) {\n    // The type of a function call is the return type of the selected signature.\n    // The signature is selected based on the types of the arguments. Angular doesn't\n    // support contextual typing of arguments so this is simpler than TypeScript's\n    // version.\n    const args = ast.args.map(arg => this.getType(arg));\n    const target = this.getType(ast.target!);\n    if (!target || !target.callable) {\n      this.diagnostics.push(createDiagnostic(\n          refinedSpan(ast), Diagnostic.call_target_not_callable, this.sourceOf(ast.target!),\n          target.name));\n      return this.anyType;\n    }\n    const signature = target.selectSignature(args);\n    if (signature) {\n      return signature.result;\n    }\n    // TODO: Consider a better error message here. See `typescript_symbols#selectSignature` for more\n    // details.\n    this.diagnostics.push(\n        createDiagnostic(refinedSpan(ast), Diagnostic.unable_to_resolve_compatible_call_signature));\n    return this.anyType;\n  }\n\n  visitImplicitReceiver(_ast: ImplicitReceiver): Symbol {\n    const _this = this;\n    // Return a pseudo-symbol for the implicit receiver.\n    // The members of the implicit receiver are what is defined by the\n    // scope passed into this class.\n    return {\n      name: '$implicit',\n      kind: 'component',\n      language: 'ng-template',\n      type: undefined,\n      container: undefined,\n      callable: false,\n      nullable: false,\n      public: true,\n      definition: undefined,\n      documentation: [],\n      members(): SymbolTable {\n        return _this.scope;\n      },\n      signatures(): Signature[] {\n        return [];\n      },\n      selectSignature(_types): Signature |\n          undefined {\n            return undefined;\n          },\n      indexed(_argument): Symbol |\n          undefined {\n            return undefined;\n          },\n      typeArguments(): Symbol[] |\n          undefined {\n            return undefined;\n          },\n    };\n  }\n\n  visitThisReceiver(_ast: ThisReceiver): Symbol {\n    return this.visitImplicitReceiver(_ast);\n  }\n\n  visitInterpolation(ast: Interpolation): Symbol {\n    // If we are producing diagnostics, visit the children.\n    for (const expr of ast.expressions) {\n      expr.visit(this);\n    }\n    return this.undefinedType;\n  }\n\n  visitKeyedRead(ast: KeyedRead): Symbol {\n    const targetType = this.getType(ast.obj);\n    const keyType = this.getType(ast.key);\n    const result = targetType.indexed(\n        keyType, ast.key instanceof LiteralPrimitive ? ast.key.value : undefined);\n    return result || this.anyType;\n  }\n\n  visitKeyedWrite(ast: KeyedWrite): Symbol {\n    // The write of a type is the type of the value being written.\n    return this.getType(ast.value);\n  }\n\n  visitLiteralArray(ast: LiteralArray): Symbol {\n    // A type literal is an array type of the union of the elements\n    return this.query.getArrayType(\n        this.query.getTypeUnion(...ast.expressions.map(element => this.getType(element))));\n  }\n\n  visitLiteralMap(ast: LiteralMap): Symbol {\n    // If we are producing diagnostics, visit the children\n    for (const value of ast.values) {\n      value.visit(this);\n    }\n    // TODO: Return a composite type.\n    return this.anyType;\n  }\n\n  visitLiteralPrimitive(ast: LiteralPrimitive) {\n    // The type of a literal primitive depends on the value of the literal.\n    switch (ast.value) {\n      case true:\n      case false:\n        return this.query.getBuiltinType(BuiltinType.Boolean);\n      case null:\n        return this.query.getBuiltinType(BuiltinType.Null);\n      case undefined:\n        return this.query.getBuiltinType(BuiltinType.Undefined);\n      default:\n        switch (typeof ast.value) {\n          case 'string':\n            return this.query.getBuiltinType(BuiltinType.String);\n          case 'number':\n            return this.query.getBuiltinType(BuiltinType.Number);\n          default:\n            this.diagnostics.push(createDiagnostic(\n                refinedSpan(ast), Diagnostic.unrecognized_primitive, typeof ast.value));\n            return this.anyType;\n        }\n    }\n  }\n\n  visitMethodCall(ast: MethodCall) {\n    return this.resolveMethodCall(this.getType(ast.receiver), ast);\n  }\n\n  visitPipe(ast: BindingPipe) {\n    // The type of a pipe node is the return type of the pipe's transform method. The table returned\n    // by getPipes() is expected to contain symbols with the corresponding transform method type.\n    const pipe = this.query.getPipes().get(ast.name);\n    if (!pipe) {\n      this.diagnostics.push(createDiagnostic(refinedSpan(ast), Diagnostic.no_pipe_found, ast.name));\n      return this.anyType;\n    }\n    const expType = this.getType(ast.exp);\n    const signature =\n        pipe.selectSignature([expType].concat(ast.args.map(arg => this.getType(arg))));\n    if (!signature) {\n      this.diagnostics.push(\n          createDiagnostic(refinedSpan(ast), Diagnostic.unable_to_resolve_signature, ast.name));\n      return this.anyType;\n    }\n    return signature.result;\n  }\n\n  visitPrefixNot(ast: PrefixNot) {\n    // If we are producing diagnostics, visit the children\n    ast.expression.visit(this);\n    // The type of a prefix ! is always boolean.\n    return this.query.getBuiltinType(BuiltinType.Boolean);\n  }\n\n  visitNonNullAssert(ast: NonNullAssert) {\n    const expressionType = this.getType(ast.expression);\n    return this.query.getNonNullableType(expressionType);\n  }\n\n  visitPropertyRead(ast: PropertyRead) {\n    return this.resolvePropertyRead(this.getType(ast.receiver), ast);\n  }\n\n  visitPropertyWrite(ast: PropertyWrite) {\n    // The type of a write is the type of the value being written.\n    return this.getType(ast.value);\n  }\n\n  visitQuote(_ast: Quote) {\n    // The type of a quoted expression is any.\n    return this.query.getBuiltinType(BuiltinType.Any);\n  }\n\n  visitSafeMethodCall(ast: SafeMethodCall) {\n    return this.resolveMethodCall(this.query.getNonNullableType(this.getType(ast.receiver)), ast);\n  }\n\n  visitSafePropertyRead(ast: SafePropertyRead) {\n    return this.resolvePropertyRead(this.query.getNonNullableType(this.getType(ast.receiver)), ast);\n  }\n\n  /**\n   * Gets the source of an expession AST.\n   * The AST's sourceSpan is relative to the start of the template source code, which is contained\n   * at this.source.\n   */\n  private sourceOf(ast: AST): string {\n    return this.source.substring(ast.sourceSpan.start, ast.sourceSpan.end);\n  }\n\n  private _anyType: Symbol|undefined;\n  private get anyType(): Symbol {\n    let result = this._anyType;\n    if (!result) {\n      result = this._anyType = this.query.getBuiltinType(BuiltinType.Any);\n    }\n    return result;\n  }\n\n  private _undefinedType: Symbol|undefined;\n  private get undefinedType(): Symbol {\n    let result = this._undefinedType;\n    if (!result) {\n      result = this._undefinedType = this.query.getBuiltinType(BuiltinType.Undefined);\n    }\n    return result;\n  }\n\n  private resolveMethodCall(receiverType: Symbol, ast: SafeMethodCall|MethodCall) {\n    if (this.isAny(receiverType)) {\n      return this.anyType;\n    }\n    const methodType = this.resolvePropertyRead(receiverType, ast);\n    if (!methodType) {\n      this.diagnostics.push(\n          createDiagnostic(refinedSpan(ast), Diagnostic.could_not_resolve_type, ast.name));\n      return this.anyType;\n    }\n    if (this.isAny(methodType)) {\n      return this.anyType;\n    }\n    if (!methodType.callable) {\n      this.diagnostics.push(\n          createDiagnostic(refinedSpan(ast), Diagnostic.identifier_not_callable, ast.name));\n      return this.anyType;\n    }\n    const signature = methodType.selectSignature(ast.args.map(arg => this.getType(arg)));\n    if (!signature) {\n      this.diagnostics.push(\n          createDiagnostic(refinedSpan(ast), Diagnostic.unable_to_resolve_signature, ast.name));\n      return this.anyType;\n    }\n    return signature.result;\n  }\n\n  private resolvePropertyRead(receiverType: Symbol, ast: SafePropertyRead|PropertyRead) {\n    if (this.isAny(receiverType)) {\n      return this.anyType;\n    }\n    // The type of a property read is the seelcted member's type.\n    const member = receiverType.members().get(ast.name);\n    if (!member) {\n      if (receiverType.name === '$implicit') {\n        this.diagnostics.push(createDiagnostic(\n            refinedSpan(ast), Diagnostic.identifier_not_defined_in_app_context, ast.name));\n      } else if (receiverType.nullable && ast.receiver instanceof PropertyRead) {\n        const receiver = ast.receiver.name;\n        this.diagnostics.push(createDiagnostic(\n            refinedSpan(ast), Diagnostic.identifier_possibly_undefined, receiver,\n            `${receiver}?.${ast.name}`, `${receiver}!.${ast.name}`));\n      } else {\n        this.diagnostics.push(createDiagnostic(\n            refinedSpan(ast), Diagnostic.identifier_not_defined_on_receiver, ast.name,\n            receiverType.name));\n      }\n      return this.anyType;\n    }\n    if (!member.public) {\n      const container =\n          receiverType.name === '$implicit' ? 'the component' : `'${receiverType.name}'`;\n      this.diagnostics.push(createDiagnostic(\n          refinedSpan(ast), Diagnostic.identifier_is_private, ast.name, container));\n    }\n    return member.type;\n  }\n\n  private isAny(symbol: Symbol): boolean {\n    return !symbol || this.query.getTypeKind(symbol) === BuiltinType.Any ||\n        (!!symbol.type && this.isAny(symbol.type));\n  }\n}\n\nfunction refinedSpan(ast: AST): ng.Span {\n  // nameSpan is an absolute span, but the spans returned by the expression visitor are expected to\n  // be relative to the start of the expression.\n  // TODO: migrate to only using absolute spans\n  const absoluteOffset = ast.sourceSpan.start - ast.span.start;\n  if (ast instanceof ASTWithName) {\n    return offsetSpan(ast.nameSpan, -absoluteOffset);\n  }\n  return offsetSpan(ast.sourceSpan, -absoluteOffset);\n}\n"]}
\No newline at end of file