/** * @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 {Injectable} from '@angular/core'; import * as chars from '../chars'; import {ListWrapper} from '../facade/collection'; import {BaseException} from '../facade/exceptions'; import {RegExpWrapper, StringWrapper, escapeRegExp, isBlank, isPresent} from '../facade/lang'; import {DEFAULT_INTERPOLATION_CONFIG, InterpolationConfig} from '../interpolation_config'; import {AST, ASTWithSource, AstVisitor, Binary, BindingPipe, Chain, Conditional, EmptyExpr, FunctionCall, ImplicitReceiver, Interpolation, KeyedRead, KeyedWrite, LiteralArray, LiteralMap, LiteralPrimitive, MethodCall, PrefixNot, PropertyRead, PropertyWrite, Quote, SafeMethodCall, SafePropertyRead, TemplateBinding} from './ast'; import {EOF, Lexer, Token, isIdentifier, isQuote} from './lexer'; var _implicitReceiver = new ImplicitReceiver(); class ParseException extends BaseException { constructor(message: string, input: string, errLocation: string, ctxLocation?: any) { super(`Parser Error: ${message} ${errLocation} [${input}] in ${ctxLocation}`); } } export class SplitInterpolation { constructor(public strings: string[], public expressions: string[]) {} } export class TemplateBindingParseResult { constructor(public templateBindings: TemplateBinding[], public warnings: string[]) {} } function _createInterpolateRegExp(config: InterpolationConfig): RegExp { const regexp = escapeRegExp(config.start) + '([\\s\\S]*?)' + escapeRegExp(config.end); return RegExpWrapper.create(regexp, 'g'); } @Injectable() export class Parser { constructor(/** @internal */ public _lexer: Lexer) {} parseAction( input: string, location: any, interpolationConfig: InterpolationConfig = DEFAULT_INTERPOLATION_CONFIG): ASTWithSource { this._checkNoInterpolation(input, location, interpolationConfig); var tokens = this._lexer.tokenize(this._stripComments(input)); var ast = new _ParseAST(input, location, tokens, true).parseChain(); return new ASTWithSource(ast, input, location); } parseBinding( input: string, location: any, interpolationConfig: InterpolationConfig = DEFAULT_INTERPOLATION_CONFIG): ASTWithSource { var ast = this._parseBindingAst(input, location, interpolationConfig); return new ASTWithSource(ast, input, location); } parseSimpleBinding( input: string, location: string, interpolationConfig: InterpolationConfig = DEFAULT_INTERPOLATION_CONFIG): ASTWithSource { var ast = this._parseBindingAst(input, location, interpolationConfig); if (!SimpleExpressionChecker.check(ast)) { throw new ParseException( 'Host binding expression can only contain field access and constants', input, location); } return new ASTWithSource(ast, input, location); } private _parseBindingAst( input: string, location: string, interpolationConfig: InterpolationConfig): AST { // Quotes expressions use 3rd-party expression language. We don't want to use // our lexer or parser for that, so we check for that ahead of time. var quote = this._parseQuote(input, location); if (isPresent(quote)) { return quote; } this._checkNoInterpolation(input, location, interpolationConfig); var tokens = this._lexer.tokenize(this._stripComments(input)); return new _ParseAST(input, location, tokens, false).parseChain(); } private _parseQuote(input: string, location: any): AST { if (isBlank(input)) return null; var prefixSeparatorIndex = input.indexOf(':'); if (prefixSeparatorIndex == -1) return null; var prefix = input.substring(0, prefixSeparatorIndex).trim(); if (!isIdentifier(prefix)) return null; var uninterpretedExpression = input.substring(prefixSeparatorIndex + 1); return new Quote(prefix, uninterpretedExpression, location); } parseTemplateBindings(input: string, location: any): TemplateBindingParseResult { var tokens = this._lexer.tokenize(input); return new _ParseAST(input, location, tokens, false).parseTemplateBindings(); } parseInterpolation( input: string, location: any, interpolationConfig: InterpolationConfig = DEFAULT_INTERPOLATION_CONFIG): ASTWithSource { let split = this.splitInterpolation(input, location, interpolationConfig); if (split == null) return null; let expressions: AST[] = []; for (let i = 0; i < split.expressions.length; ++i) { var tokens = this._lexer.tokenize(this._stripComments(split.expressions[i])); var ast = new _ParseAST(input, location, tokens, false).parseChain(); expressions.push(ast); } return new ASTWithSource(new Interpolation(split.strings, expressions), input, location); } splitInterpolation( input: string, location: string, interpolationConfig: InterpolationConfig = DEFAULT_INTERPOLATION_CONFIG): SplitInterpolation { const regexp = _createInterpolateRegExp(interpolationConfig); const parts = StringWrapper.split(input, regexp); if (parts.length <= 1) { return null; } var strings: string[] = []; var expressions: string[] = []; for (var i = 0; i < parts.length; i++) { var part: string = parts[i]; if (i % 2 === 0) { // fixed string strings.push(part); } else if (part.trim().length > 0) { expressions.push(part); } else { throw new ParseException( 'Blank expressions are not allowed in interpolated strings', input, `at column ${this._findInterpolationErrorColumn(parts, i, interpolationConfig)} in`, location); } } return new SplitInterpolation(strings, expressions); } wrapLiteralPrimitive(input: string, location: any): ASTWithSource { return new ASTWithSource(new LiteralPrimitive(input), input, location); } private _stripComments(input: string): string { let i = this._commentStart(input); return isPresent(i) ? input.substring(0, i).trim() : input; } private _commentStart(input: string): number { var outerQuote: number = null; for (var i = 0; i < input.length - 1; i++) { let char = StringWrapper.charCodeAt(input, i); let nextChar = StringWrapper.charCodeAt(input, i + 1); if (char === chars.$SLASH && nextChar == chars.$SLASH && isBlank(outerQuote)) return i; if (outerQuote === char) { outerQuote = null; } else if (isBlank(outerQuote) && isQuote(char)) { outerQuote = char; } } return null; } private _checkNoInterpolation( input: string, location: any, interpolationConfig: InterpolationConfig): void { var regexp = _createInterpolateRegExp(interpolationConfig); var parts = StringWrapper.split(input, regexp); if (parts.length > 1) { throw new ParseException( `Got interpolation (${interpolationConfig.start}${interpolationConfig.end}) where expression was expected`, input, `at column ${this._findInterpolationErrorColumn(parts, 1, interpolationConfig)} in`, location); } } private _findInterpolationErrorColumn( parts: string[], partInErrIdx: number, interpolationConfig: InterpolationConfig): number { var errLocation = ''; for (var j = 0; j < partInErrIdx; j++) { errLocation += j % 2 === 0 ? parts[j] : `${interpolationConfig.start}${parts[j]}${interpolationConfig.end}`; } return errLocation.length; } } export class _ParseAST { index: number = 0; constructor( public input: string, public location: any, public tokens: any[], public parseAction: boolean) {} peek(offset: number): Token { var i = this.index + offset; return i < this.tokens.length ? this.tokens[i] : EOF; } get next(): Token { return this.peek(0); } get inputIndex(): number { return (this.index < this.tokens.length) ? this.next.index : this.input.length; } advance() { this.index++; } optionalCharacter(code: number): boolean { if (this.next.isCharacter(code)) { this.advance(); return true; } else { return false; } } peekKeywordLet(): boolean { return this.next.isKeywordLet(); } peekDeprecatedKeywordVar(): boolean { return this.next.isKeywordDeprecatedVar(); } peekDeprecatedOperatorHash(): boolean { return this.next.isOperator('#'); } expectCharacter(code: number) { if (this.optionalCharacter(code)) return; this.error(`Missing expected ${StringWrapper.fromCharCode(code)}`); } optionalOperator(op: string): boolean { if (this.next.isOperator(op)) { this.advance(); return true; } else { return false; } } expectOperator(operator: string) { if (this.optionalOperator(operator)) return; this.error(`Missing expected operator ${operator}`); } expectIdentifierOrKeyword(): string { var n = this.next; if (!n.isIdentifier() && !n.isKeyword()) { this.error(`Unexpected token ${n}, expected identifier or keyword`); } this.advance(); return n.toString(); } expectIdentifierOrKeywordOrString(): string { var n = this.next; if (!n.isIdentifier() && !n.isKeyword() && !n.isString()) { this.error(`Unexpected token ${n}, expected identifier, keyword, or string`); } this.advance(); return n.toString(); } parseChain(): AST { var exprs: AST[] = []; while (this.index < this.tokens.length) { var expr = this.parsePipe(); exprs.push(expr); if (this.optionalCharacter(chars.$SEMICOLON)) { if (!this.parseAction) { this.error('Binding expression cannot contain chained expression'); } while (this.optionalCharacter(chars.$SEMICOLON)) { } // read all semicolons } else if (this.index < this.tokens.length) { this.error(`Unexpected token '${this.next}'`); } } if (exprs.length == 0) return new EmptyExpr(); if (exprs.length == 1) return exprs[0]; return new Chain(exprs); } parsePipe(): AST { var result = this.parseExpression(); if (this.optionalOperator('|')) { if (this.parseAction) { this.error('Cannot have a pipe in an action expression'); } do { var name = this.expectIdentifierOrKeyword(); var args: AST[] = []; while (this.optionalCharacter(chars.$COLON)) { args.push(this.parseExpression()); } result = new BindingPipe(result, name, args); } while (this.optionalOperator('|')); } return result; } parseExpression(): AST { return this.parseConditional(); } parseConditional(): AST { var start = this.inputIndex; var result = this.parseLogicalOr(); if (this.optionalOperator('?')) { var yes = this.parsePipe(); if (!this.optionalCharacter(chars.$COLON)) { var end = this.inputIndex; var expression = this.input.substring(start, end); this.error(`Conditional expression ${expression} requires all 3 expressions`); } var no = this.parsePipe(); return new Conditional(result, yes, no); } else { return result; } } parseLogicalOr(): AST { // '||' var result = this.parseLogicalAnd(); while (this.optionalOperator('||')) { result = new Binary('||', result, this.parseLogicalAnd()); } return result; } parseLogicalAnd(): AST { // '&&' var result = this.parseEquality(); while (this.optionalOperator('&&')) { result = new Binary('&&', result, this.parseEquality()); } return result; } parseEquality(): AST { // '==','!=','===','!==' var result = this.parseRelational(); while (true) { if (this.optionalOperator('==')) { result = new Binary('==', result, this.parseRelational()); } else if (this.optionalOperator('===')) { result = new Binary('===', result, this.parseRelational()); } else if (this.optionalOperator('!=')) { result = new Binary('!=', result, this.parseRelational()); } else if (this.optionalOperator('!==')) { result = new Binary('!==', result, this.parseRelational()); } else { return result; } } } parseRelational(): AST { // '<', '>', '<=', '>=' var result = this.parseAdditive(); while (true) { if (this.optionalOperator('<')) { result = new Binary('<', result, this.parseAdditive()); } else if (this.optionalOperator('>')) { result = new Binary('>', result, this.parseAdditive()); } else if (this.optionalOperator('<=')) { result = new Binary('<=', result, this.parseAdditive()); } else if (this.optionalOperator('>=')) { result = new Binary('>=', result, this.parseAdditive()); } else { return result; } } } parseAdditive(): AST { // '+', '-' var result = this.parseMultiplicative(); while (true) { if (this.optionalOperator('+')) { result = new Binary('+', result, this.parseMultiplicative()); } else if (this.optionalOperator('-')) { result = new Binary('-', result, this.parseMultiplicative()); } else { return result; } } } parseMultiplicative(): AST { // '*', '%', '/' var result = this.parsePrefix(); while (true) { if (this.optionalOperator('*')) { result = new Binary('*', result, this.parsePrefix()); } else if (this.optionalOperator('%')) { result = new Binary('%', result, this.parsePrefix()); } else if (this.optionalOperator('/')) { result = new Binary('/', result, this.parsePrefix()); } else { return result; } } } parsePrefix(): AST { if (this.optionalOperator('+')) { return this.parsePrefix(); } else if (this.optionalOperator('-')) { return new Binary('-', new LiteralPrimitive(0), this.parsePrefix()); } else if (this.optionalOperator('!')) { return new PrefixNot(this.parsePrefix()); } else { return this.parseCallChain(); } } parseCallChain(): AST { var result = this.parsePrimary(); while (true) { if (this.optionalCharacter(chars.$PERIOD)) { result = this.parseAccessMemberOrMethodCall(result, false); } else if (this.optionalOperator('?.')) { result = this.parseAccessMemberOrMethodCall(result, true); } else if (this.optionalCharacter(chars.$LBRACKET)) { var key = this.parsePipe(); this.expectCharacter(chars.$RBRACKET); if (this.optionalOperator('=')) { var value = this.parseConditional(); result = new KeyedWrite(result, key, value); } else { result = new KeyedRead(result, key); } } else if (this.optionalCharacter(chars.$LPAREN)) { var args = this.parseCallArguments(); this.expectCharacter(chars.$RPAREN); result = new FunctionCall(result, args); } else { return result; } } } parsePrimary(): AST { if (this.optionalCharacter(chars.$LPAREN)) { let result = this.parsePipe(); this.expectCharacter(chars.$RPAREN); return result; } else if (this.next.isKeywordNull() || this.next.isKeywordUndefined()) { this.advance(); return new LiteralPrimitive(null); } else if (this.next.isKeywordTrue()) { this.advance(); return new LiteralPrimitive(true); } else if (this.next.isKeywordFalse()) { this.advance(); return new LiteralPrimitive(false); } else if (this.optionalCharacter(chars.$LBRACKET)) { var elements = this.parseExpressionList(chars.$RBRACKET); this.expectCharacter(chars.$RBRACKET); return new LiteralArray(elements); } else if (this.next.isCharacter(chars.$LBRACE)) { return this.parseLiteralMap(); } else if (this.next.isIdentifier()) { return this.parseAccessMemberOrMethodCall(_implicitReceiver, false); } else if (this.next.isNumber()) { var value = this.next.toNumber(); this.advance(); return new LiteralPrimitive(value); } else if (this.next.isString()) { var literalValue = this.next.toString(); this.advance(); return new LiteralPrimitive(literalValue); } else if (this.index >= this.tokens.length) { this.error(`Unexpected end of expression: ${this.input}`); } else { this.error(`Unexpected token ${this.next}`); } // error() throws, so we don't reach here. throw new BaseException('Fell through all cases in parsePrimary'); } parseExpressionList(terminator: number): AST[] { var result: AST[] = []; if (!this.next.isCharacter(terminator)) { do { result.push(this.parsePipe()); } while (this.optionalCharacter(chars.$COMMA)); } return result; } parseLiteralMap(): LiteralMap { var keys: string[] = []; var values: AST[] = []; this.expectCharacter(chars.$LBRACE); if (!this.optionalCharacter(chars.$RBRACE)) { do { var key = this.expectIdentifierOrKeywordOrString(); keys.push(key); this.expectCharacter(chars.$COLON); values.push(this.parsePipe()); } while (this.optionalCharacter(chars.$COMMA)); this.expectCharacter(chars.$RBRACE); } return new LiteralMap(keys, values); } parseAccessMemberOrMethodCall(receiver: AST, isSafe: boolean = false): AST { let id = this.expectIdentifierOrKeyword(); if (this.optionalCharacter(chars.$LPAREN)) { let args = this.parseCallArguments(); this.expectCharacter(chars.$RPAREN); return isSafe ? new SafeMethodCall(receiver, id, args) : new MethodCall(receiver, id, args); } else { if (isSafe) { if (this.optionalOperator('=')) { this.error('The \'?.\' operator cannot be used in the assignment'); } else { return new SafePropertyRead(receiver, id); } } else { if (this.optionalOperator('=')) { if (!this.parseAction) { this.error('Bindings cannot contain assignments'); } let value = this.parseConditional(); return new PropertyWrite(receiver, id, value); } else { return new PropertyRead(receiver, id); } } } return null; } parseCallArguments(): BindingPipe[] { if (this.next.isCharacter(chars.$RPAREN)) return []; var positionals: AST[] = []; do { positionals.push(this.parsePipe()); } while (this.optionalCharacter(chars.$COMMA)); return positionals as BindingPipe[]; } /** * An identifier, a keyword, a string with an optional `-` inbetween. */ expectTemplateBindingKey(): string { var result = ''; var operatorFound = false; do { result += this.expectIdentifierOrKeywordOrString(); operatorFound = this.optionalOperator('-'); if (operatorFound) { result += '-'; } } while (operatorFound); return result.toString(); } parseTemplateBindings(): TemplateBindingParseResult { var bindings: TemplateBinding[] = []; var prefix: string = null; var warnings: string[] = []; while (this.index < this.tokens.length) { var keyIsVar: boolean = this.peekKeywordLet(); if (!keyIsVar && this.peekDeprecatedKeywordVar()) { keyIsVar = true; warnings.push(`"var" inside of expressions is deprecated. Use "let" instead!`); } if (!keyIsVar && this.peekDeprecatedOperatorHash()) { keyIsVar = true; warnings.push(`"#" inside of expressions is deprecated. Use "let" instead!`); } if (keyIsVar) { this.advance(); } var key = this.expectTemplateBindingKey(); if (!keyIsVar) { if (prefix == null) { prefix = key; } else { key = prefix + key[0].toUpperCase() + key.substring(1); } } this.optionalCharacter(chars.$COLON); var name: string = null; var expression: ASTWithSource = null; if (keyIsVar) { if (this.optionalOperator('=')) { name = this.expectTemplateBindingKey(); } else { name = '\$implicit'; } } else if ( this.next !== EOF && !this.peekKeywordLet() && !this.peekDeprecatedKeywordVar() && !this.peekDeprecatedOperatorHash()) { var start = this.inputIndex; var ast = this.parsePipe(); var source = this.input.substring(start, this.inputIndex); expression = new ASTWithSource(ast, source, this.location); } bindings.push(new TemplateBinding(key, keyIsVar, name, expression)); if (!this.optionalCharacter(chars.$SEMICOLON)) { this.optionalCharacter(chars.$COMMA); } } return new TemplateBindingParseResult(bindings, warnings); } error(message: string, index: number = null) { if (isBlank(index)) index = this.index; var location = (index < this.tokens.length) ? `at column ${this.tokens[index].index + 1} in` : `at the end of the expression`; throw new ParseException(message, this.input, location, this.location); } } class SimpleExpressionChecker implements AstVisitor { static check(ast: AST): boolean { var s = new SimpleExpressionChecker(); ast.visit(s); return s.simple; } simple = true; visitImplicitReceiver(ast: ImplicitReceiver, context: any) {} visitInterpolation(ast: Interpolation, context: any) { this.simple = false; } visitLiteralPrimitive(ast: LiteralPrimitive, context: any) {} visitPropertyRead(ast: PropertyRead, context: any) {} visitPropertyWrite(ast: PropertyWrite, context: any) { this.simple = false; } visitSafePropertyRead(ast: SafePropertyRead, context: any) { this.simple = false; } visitMethodCall(ast: MethodCall, context: any) { this.simple = false; } visitSafeMethodCall(ast: SafeMethodCall, context: any) { this.simple = false; } visitFunctionCall(ast: FunctionCall, context: any) { this.simple = false; } visitLiteralArray(ast: LiteralArray, context: any) { this.visitAll(ast.expressions); } visitLiteralMap(ast: LiteralMap, context: any) { this.visitAll(ast.values); } visitBinary(ast: Binary, context: any) { this.simple = false; } visitPrefixNot(ast: PrefixNot, context: any) { this.simple = false; } visitConditional(ast: Conditional, context: any) { this.simple = false; } visitPipe(ast: BindingPipe, context: any) { this.simple = false; } visitKeyedRead(ast: KeyedRead, context: any) { this.simple = false; } visitKeyedWrite(ast: KeyedWrite, context: any) { this.simple = false; } visitAll(asts: any[]): any[] { var res = ListWrapper.createFixedSize(asts.length); for (var i = 0; i < asts.length; ++i) { res[i] = asts[i].visit(this); } return res; } visitChain(ast: Chain, context: any) { this.simple = false; } visitQuote(ast: Quote, context: any) { this.simple = false; } }