| 
									
										
										
										
											2016-06-23 09:47:54 -07:00
										 |  |  | /** | 
					
						
							|  |  |  |  * @license | 
					
						
							| 
									
										
										
										
											2020-05-19 12:08:49 -07:00
										 |  |  |  * Copyright Google LLC All Rights Reserved. | 
					
						
							| 
									
										
										
										
											2016-06-23 09:47:54 -07:00
										 |  |  |  * | 
					
						
							|  |  |  |  * 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
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-12 10:12:36 -04:00
										 |  |  | import {AbsoluteSourceSpan, ASTWithSource, BindingPipe, EmptyExpr, Interpolation, MethodCall, ParserError, TemplateBinding, VariableBinding} from '@angular/compiler/src/expression_parser/ast'; | 
					
						
							| 
									
										
										
										
											2016-06-08 16:38:52 -07:00
										 |  |  | import {Lexer} from '@angular/compiler/src/expression_parser/lexer'; | 
					
						
							| 
									
										
										
										
											2020-07-04 09:21:40 +02:00
										 |  |  | import {IvyParser, Parser, SplitInterpolation} from '@angular/compiler/src/expression_parser/parser'; | 
					
						
							| 
									
										
										
										
											2017-03-02 12:12:46 -08:00
										 |  |  | import {expect} from '@angular/platform-browser/testing/src/matchers'; | 
					
						
							| 
									
										
										
										
											2016-06-08 16:38:52 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-27 18:54:30 -07:00
										 |  |  | import {unparse, unparseWithSpan} from './utils/unparser'; | 
					
						
							| 
									
										
										
										
											2018-04-24 14:22:55 -07:00
										 |  |  | import {validate} from './utils/validator'; | 
					
						
							| 
									
										
										
										
											2014-10-28 12:22:38 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  | describe('parser', () => { | 
					
						
							|  |  |  |   describe('parseAction', () => { | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     it('should parse numbers', () => { | 
					
						
							|  |  |  |       checkAction('1'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2014-11-04 09:06:46 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     it('should parse strings', () => { | 
					
						
							|  |  |  |       checkAction('\'1\'', '"1"'); | 
					
						
							|  |  |  |       checkAction('"1"'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2014-11-18 16:38:36 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     it('should parse null', () => { | 
					
						
							|  |  |  |       checkAction('null'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2014-11-03 17:25:16 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     it('should parse undefined', () => { | 
					
						
							|  |  |  |       checkAction('undefined'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-01-08 16:17:56 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-04 01:52:40 +02:00
										 |  |  |     it('should parse unary - and + expressions', () => { | 
					
						
							|  |  |  |       checkAction('-1', '-1'); | 
					
						
							|  |  |  |       checkAction('+1', '+1'); | 
					
						
							|  |  |  |       checkAction(`-'1'`, `-"1"`); | 
					
						
							|  |  |  |       checkAction(`+'1'`, `+"1"`); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2016-10-06 15:22:10 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     it('should parse unary ! expressions', () => { | 
					
						
							|  |  |  |       checkAction('!true'); | 
					
						
							|  |  |  |       checkAction('!!true'); | 
					
						
							|  |  |  |       checkAction('!!!true'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-06-22 08:21:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     it('should parse postfix ! expression', () => { | 
					
						
							|  |  |  |       checkAction('true!'); | 
					
						
							|  |  |  |       checkAction('a!.b'); | 
					
						
							|  |  |  |       checkAction('a!!!!.b'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-12-05 22:06:24 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     it('should parse multiplicative expressions', () => { | 
					
						
							|  |  |  |       checkAction('3*4/2%5', '3 * 4 / 2 % 5'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-06-10 11:11:01 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     it('should parse additive expressions', () => { | 
					
						
							|  |  |  |       checkAction('3 + 6 - 2'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2016-07-06 14:06:47 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     it('should parse relational expressions', () => { | 
					
						
							|  |  |  |       checkAction('2 < 3'); | 
					
						
							|  |  |  |       checkAction('2 > 3'); | 
					
						
							|  |  |  |       checkAction('2 <= 2'); | 
					
						
							|  |  |  |       checkAction('2 >= 2'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2014-11-03 17:25:16 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     it('should parse equality expressions', () => { | 
					
						
							|  |  |  |       checkAction('2 == 3'); | 
					
						
							|  |  |  |       checkAction('2 != 3'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2014-10-28 12:22:38 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     it('should parse strict equality expressions', () => { | 
					
						
							|  |  |  |       checkAction('2 === 3'); | 
					
						
							|  |  |  |       checkAction('2 !== 3'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2014-11-18 16:38:36 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     it('should parse expressions', () => { | 
					
						
							|  |  |  |       checkAction('true && true'); | 
					
						
							|  |  |  |       checkAction('true || false'); | 
					
						
							| 
									
										
										
										
											2021-04-03 18:10:31 +02:00
										 |  |  |       checkAction('null ?? 0'); | 
					
						
							|  |  |  |       checkAction('null ?? undefined ?? 0'); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2014-10-30 23:47:22 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     it('should parse grouped expressions', () => { | 
					
						
							|  |  |  |       checkAction('(1 + 2) * 3', '1 + 2 * 3'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2014-10-30 23:47:22 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     it('should ignore comments in expressions', () => { | 
					
						
							|  |  |  |       checkAction('a //comment', 'a'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2014-10-30 23:47:22 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     it('should retain // in string literals', () => { | 
					
						
							|  |  |  |       checkAction(`"http://www.google.com"`, `"http://www.google.com"`); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2016-08-04 17:04:30 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     it('should parse an empty string', () => { | 
					
						
							|  |  |  |       checkAction(''); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2014-10-30 23:47:22 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     describe('literals', () => { | 
					
						
							|  |  |  |       it('should parse array', () => { | 
					
						
							|  |  |  |         checkAction('[1][0]'); | 
					
						
							|  |  |  |         checkAction('[[1]][0][0]'); | 
					
						
							|  |  |  |         checkAction('[]'); | 
					
						
							|  |  |  |         checkAction('[].length'); | 
					
						
							|  |  |  |         checkAction('[1, 2].length'); | 
					
						
							| 
									
										
										
										
											2018-02-18 00:05:41 +08:00
										 |  |  |         checkAction('[1, 2,]', '[1, 2]'); | 
					
						
							| 
									
										
										
										
											2015-08-12 16:26:21 -07:00
										 |  |  |       }); | 
					
						
							| 
									
										
										
										
											2014-10-30 23:47:22 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |       it('should parse map', () => { | 
					
						
							|  |  |  |         checkAction('{}'); | 
					
						
							|  |  |  |         checkAction('{a: 1, "b": 2}[2]'); | 
					
						
							|  |  |  |         checkAction('{}["a"]'); | 
					
						
							| 
									
										
										
										
											2017-05-11 10:15:54 -07:00
										 |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |       it('should only allow identifier, string, or keyword as map key', () => { | 
					
						
							|  |  |  |         expectActionError('{(:0}', 'expected identifier, keyword, or string'); | 
					
						
							|  |  |  |         expectActionError('{1234:0}', 'expected identifier, keyword, or string'); | 
					
						
							| 
									
										
										
										
											2021-05-10 18:16:45 +02:00
										 |  |  |         expectActionError('{#myField:0}', 'expected identifier, keyword or string'); | 
					
						
							| 
									
										
										
										
											2015-08-12 16:26:21 -07:00
										 |  |  |       }); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2014-10-30 23:47:22 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     describe('member access', () => { | 
					
						
							|  |  |  |       it('should parse field access', () => { | 
					
						
							|  |  |  |         checkAction('a'); | 
					
						
							|  |  |  |         checkAction('this.a', 'a'); | 
					
						
							|  |  |  |         checkAction('a.a'); | 
					
						
							| 
									
										
										
										
											2015-08-12 16:26:21 -07:00
										 |  |  |       }); | 
					
						
							| 
									
										
										
										
											2014-11-03 17:25:16 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-10 18:16:45 +02:00
										 |  |  |       it('should error for private identifiers with implicit receiver', () => { | 
					
						
							|  |  |  |         checkActionWithError( | 
					
						
							|  |  |  |             '#privateField', '', | 
					
						
							|  |  |  |             'Private identifiers are not supported. Unexpected private identifier: #privateField at column 1'); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |       it('should only allow identifier or keyword as member names', () => { | 
					
						
							| 
									
										
										
										
											2020-09-25 11:45:46 -05:00
										 |  |  |         checkActionWithError('x.', 'x.', 'identifier or keyword'); | 
					
						
							|  |  |  |         checkActionWithError('x.(', 'x.', 'identifier or keyword'); | 
					
						
							|  |  |  |         checkActionWithError('x. 1234', 'x.', 'identifier or keyword'); | 
					
						
							|  |  |  |         checkActionWithError('x."foo"', 'x.', 'identifier or keyword'); | 
					
						
							| 
									
										
										
										
											2021-05-10 18:16:45 +02:00
										 |  |  |         checkActionWithError( | 
					
						
							|  |  |  |             'x.#privateField', 'x.', | 
					
						
							|  |  |  |             'Private identifiers are not supported. Unexpected private identifier: #privateField, expected identifier or keyword'); | 
					
						
							| 
									
										
										
										
											2015-08-12 16:26:21 -07:00
										 |  |  |       }); | 
					
						
							| 
									
										
										
										
											2014-11-06 12:00:09 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |       it('should parse safe field access', () => { | 
					
						
							|  |  |  |         checkAction('a?.a'); | 
					
						
							|  |  |  |         checkAction('a.a?.a'); | 
					
						
							| 
									
										
										
										
											2015-08-12 16:26:21 -07:00
										 |  |  |       }); | 
					
						
							| 
									
										
										
										
											2020-09-25 11:45:46 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |       it('should parse incomplete safe field accesses', () => { | 
					
						
							|  |  |  |         checkActionWithError('a?.a.', 'a?.a.', 'identifier or keyword'); | 
					
						
							|  |  |  |         checkActionWithError('a.a?.a.', 'a.a?.a.', 'identifier or keyword'); | 
					
						
							|  |  |  |         checkActionWithError('a.a?.a?. 1234', 'a.a?.a?.', 'identifier or keyword'); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2014-11-06 12:00:09 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-02 23:25:38 -05:00
										 |  |  |     describe('property write', () => { | 
					
						
							|  |  |  |       it('should parse property writes', () => { | 
					
						
							|  |  |  |         checkAction('a.a = 1 + 2'); | 
					
						
							|  |  |  |         checkAction('this.a.a = 1 + 2', 'a.a = 1 + 2'); | 
					
						
							|  |  |  |         checkAction('a.a.a = 1 + 2'); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       describe('malformed property writes', () => { | 
					
						
							|  |  |  |         it('should recover on empty rvalues', () => { | 
					
						
							|  |  |  |           checkActionWithError('a.a = ', 'a.a = ', 'Unexpected end of expression'); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         it('should recover on incomplete rvalues', () => { | 
					
						
							|  |  |  |           checkActionWithError('a.a = 1 + ', 'a.a = 1 + ', 'Unexpected end of expression'); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         it('should recover on missing properties', () => { | 
					
						
							|  |  |  |           checkActionWithError( | 
					
						
							|  |  |  |               'a. = 1', 'a. = 1', 'Expected identifier for property access at column 2'); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         it('should error on writes after a property write', () => { | 
					
						
							|  |  |  |           const ast = parseAction('a.a = 1 = 2'); | 
					
						
							|  |  |  |           expect(unparse(ast)).toEqual('a.a = 1'); | 
					
						
							|  |  |  |           validate(ast); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           expect(ast.errors.length).toBe(1); | 
					
						
							|  |  |  |           expect(ast.errors[0].message).toContain('Unexpected token \'=\''); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     describe('method calls', () => { | 
					
						
							|  |  |  |       it('should parse method calls', () => { | 
					
						
							|  |  |  |         checkAction('fn()'); | 
					
						
							|  |  |  |         checkAction('add(1, 2)'); | 
					
						
							|  |  |  |         checkAction('a.add(1, 2)'); | 
					
						
							|  |  |  |         checkAction('fn().add(1, 2)'); | 
					
						
							| 
									
										
										
										
											2014-11-03 17:25:16 -08:00
										 |  |  |       }); | 
					
						
							| 
									
										
										
										
											2021-04-12 10:22:13 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |       it('should parse an EmptyExpr with a correct span for a trailing empty argument', () => { | 
					
						
							|  |  |  |         const ast = parseAction('fn(1, )').ast as MethodCall; | 
					
						
							|  |  |  |         expect(ast.args[1]).toBeAnInstanceOf(EmptyExpr); | 
					
						
							|  |  |  |         const sourceSpan = (ast.args[1] as EmptyExpr).sourceSpan; | 
					
						
							|  |  |  |         expect([sourceSpan.start, sourceSpan.end]).toEqual([5, 6]); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2014-11-04 09:06:46 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     describe('functional calls', () => { | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |       it('should parse function calls', () => { | 
					
						
							|  |  |  |         checkAction('fn()(1, 2)'); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-05-26 10:19:47 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-25 18:52:06 -05:00
										 |  |  |     describe('keyed read', () => { | 
					
						
							|  |  |  |       it('should parse keyed reads', () => { | 
					
						
							| 
									
										
										
										
											2021-05-01 18:46:34 +02:00
										 |  |  |         checkBinding('a["a"]'); | 
					
						
							|  |  |  |         checkBinding('this.a["a"]', 'a["a"]'); | 
					
						
							|  |  |  |         checkBinding('a.a["a"]'); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       it('should parse safe keyed reads', () => { | 
					
						
							|  |  |  |         checkBinding('a?.["a"]'); | 
					
						
							|  |  |  |         checkBinding('this.a?.["a"]', 'a?.["a"]'); | 
					
						
							|  |  |  |         checkBinding('a.a?.["a"]'); | 
					
						
							|  |  |  |         checkBinding('a.a?.["a" | foo]', 'a.a?.[("a" | foo)]'); | 
					
						
							| 
									
										
										
										
											2020-09-25 18:52:06 -05:00
										 |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       describe('malformed keyed reads', () => { | 
					
						
							|  |  |  |         it('should recover on missing keys', () => { | 
					
						
							|  |  |  |           checkActionWithError('a[]', 'a[]', 'Key access cannot be empty'); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         it('should recover on incomplete expression keys', () => { | 
					
						
							|  |  |  |           checkActionWithError('a[1 + ]', 'a[1 + ]', 'Unexpected token ]'); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         it('should recover on unterminated keys', () => { | 
					
						
							|  |  |  |           checkActionWithError( | 
					
						
							|  |  |  |               'a[1 + 2', 'a[1 + 2]', 'Missing expected ] at the end of the expression'); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         it('should recover on incomplete and unterminated keys', () => { | 
					
						
							|  |  |  |           checkActionWithError( | 
					
						
							|  |  |  |               'a[1 + ', 'a[1 + ]', 'Missing expected ] at the end of the expression'); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     describe('keyed write', () => { | 
					
						
							|  |  |  |       it('should parse keyed writes', () => { | 
					
						
							|  |  |  |         checkAction('a["a"] = 1 + 2'); | 
					
						
							|  |  |  |         checkAction('this.a["a"] = 1 + 2', 'a["a"] = 1 + 2'); | 
					
						
							|  |  |  |         checkAction('a.a["a"] = 1 + 2'); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-01 18:46:34 +02:00
										 |  |  |       it('should report on safe keyed writes', () => { | 
					
						
							|  |  |  |         expectActionError('a?.["a"] = 123', 'cannot be used in the assignment'); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-25 18:52:06 -05:00
										 |  |  |       describe('malformed keyed writes', () => { | 
					
						
							|  |  |  |         it('should recover on empty rvalues', () => { | 
					
						
							|  |  |  |           checkActionWithError('a["a"] = ', 'a["a"] = ', 'Unexpected end of expression'); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         it('should recover on incomplete rvalues', () => { | 
					
						
							|  |  |  |           checkActionWithError('a["a"] = 1 + ', 'a["a"] = 1 + ', 'Unexpected end of expression'); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         it('should recover on missing keys', () => { | 
					
						
							|  |  |  |           checkActionWithError('a[] = 1', 'a[] = 1', 'Key access cannot be empty'); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         it('should recover on incomplete expression keys', () => { | 
					
						
							|  |  |  |           checkActionWithError('a[1 + ] = 1', 'a[1 + ] = 1', 'Unexpected token ]'); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         it('should recover on unterminated keys', () => { | 
					
						
							|  |  |  |           checkActionWithError('a[1 + 2 = 1', 'a[1 + 2] = 1', 'Missing expected ]'); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         it('should recover on incomplete and unterminated keys', () => { | 
					
						
							|  |  |  |           const ast = parseAction('a[1 + = 1'); | 
					
						
							|  |  |  |           expect(unparse(ast)).toEqual('a[1 + ] = 1'); | 
					
						
							|  |  |  |           validate(ast); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           const errors = ast.errors.map(e => e.message); | 
					
						
							|  |  |  |           expect(errors.length).toBe(2); | 
					
						
							|  |  |  |           expect(errors[0]).toContain('Unexpected token ='); | 
					
						
							|  |  |  |           expect(errors[1]).toContain('Missing expected ]'); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         it('should error on writes after a keyed write', () => { | 
					
						
							|  |  |  |           const ast = parseAction('a[1] = 1 = 2'); | 
					
						
							|  |  |  |           expect(unparse(ast)).toEqual('a[1] = 1'); | 
					
						
							|  |  |  |           validate(ast); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           expect(ast.errors.length).toBe(1); | 
					
						
							|  |  |  |           expect(ast.errors[0].message).toContain('Unexpected token \'=\''); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     describe('conditional', () => { | 
					
						
							|  |  |  |       it('should parse ternary/conditional expressions', () => { | 
					
						
							|  |  |  |         checkAction('7 == 3 + 4 ? 10 : 20'); | 
					
						
							|  |  |  |         checkAction('false ? 10 : 20'); | 
					
						
							| 
									
										
										
										
											2014-11-04 16:08:01 -08:00
										 |  |  |       }); | 
					
						
							| 
									
										
										
										
											2014-11-04 10:19:37 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |       it('should report incorrect ternary operator syntax', () => { | 
					
						
							|  |  |  |         expectActionError('true?1', 'Conditional expression true?1 requires all 3 expressions'); | 
					
						
							| 
									
										
										
										
											2016-06-08 16:38:52 -07:00
										 |  |  |       }); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2014-11-04 10:19:37 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     describe('assignment', () => { | 
					
						
							|  |  |  |       it('should support field assignments', () => { | 
					
						
							|  |  |  |         checkAction('a = 12'); | 
					
						
							|  |  |  |         checkAction('a.a.a = 123'); | 
					
						
							|  |  |  |         checkAction('a = 123; b = 234;'); | 
					
						
							| 
									
										
										
										
											2014-11-04 10:19:37 -08:00
										 |  |  |       }); | 
					
						
							| 
									
										
										
										
											2014-11-04 15:51:56 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |       it('should report on safe field assignments', () => { | 
					
						
							|  |  |  |         expectActionError('a?.a = 123', 'cannot be used in the assignment'); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2014-11-06 12:00:09 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |       it('should support array updates', () => { | 
					
						
							|  |  |  |         checkAction('a[0] = 200'); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2014-11-05 15:38:44 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     it('should error when using pipes', () => { | 
					
						
							|  |  |  |       expectActionError('x|blah', 'Cannot have a pipe'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2014-11-18 16:38:36 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     it('should store the source in the result', () => { | 
					
						
							|  |  |  |       expect(parseAction('someExpr', 'someExpr')); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2014-12-10 19:21:15 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     it('should store the passed-in location', () => { | 
					
						
							|  |  |  |       expect(parseAction('someExpr', 'location').location).toBe('location'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-08-20 16:34:47 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     it('should report when encountering interpolation', () => { | 
					
						
							|  |  |  |       expectActionError('{{a()}}', 'Got interpolation ({{}}) where expression was expected'); | 
					
						
							| 
									
										
										
										
											2014-11-06 12:00:09 -08:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2020-12-26 12:45:40 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     it('should not report interpolation inside a string', () => { | 
					
						
							|  |  |  |       expect(parseAction(`"{{a()}}"`).errors).toEqual([]); | 
					
						
							|  |  |  |       expect(parseAction(`'{{a()}}'`).errors).toEqual([]); | 
					
						
							|  |  |  |       expect(parseAction(`"{{a('\\"')}}"`).errors).toEqual([]); | 
					
						
							|  |  |  |       expect(parseAction(`'{{a("\\'")}}'`).errors).toEqual([]); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |   }); | 
					
						
							| 
									
										
										
										
											2014-11-04 09:21:28 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-27 18:54:30 -07:00
										 |  |  |   describe('parse spans', () => { | 
					
						
							|  |  |  |     it('should record property read span', () => { | 
					
						
							|  |  |  |       const ast = parseAction('foo'); | 
					
						
							|  |  |  |       expect(unparseWithSpan(ast)).toContain(['foo', 'foo']); | 
					
						
							|  |  |  |       expect(unparseWithSpan(ast)).toContain(['foo', '[nameSpan] foo']); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     it('should record accessed property read span', () => { | 
					
						
							|  |  |  |       const ast = parseAction('foo.bar'); | 
					
						
							|  |  |  |       expect(unparseWithSpan(ast)).toContain(['foo.bar', 'foo.bar']); | 
					
						
							|  |  |  |       expect(unparseWithSpan(ast)).toContain(['foo.bar', '[nameSpan] bar']); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     it('should record safe property read span', () => { | 
					
						
							|  |  |  |       const ast = parseAction('foo?.bar'); | 
					
						
							|  |  |  |       expect(unparseWithSpan(ast)).toContain(['foo?.bar', 'foo?.bar']); | 
					
						
							|  |  |  |       expect(unparseWithSpan(ast)).toContain(['foo?.bar', '[nameSpan] bar']); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     it('should record method call span', () => { | 
					
						
							|  |  |  |       const ast = parseAction('foo()'); | 
					
						
							|  |  |  |       expect(unparseWithSpan(ast)).toContain(['foo()', 'foo()']); | 
					
						
							|  |  |  |       expect(unparseWithSpan(ast)).toContain(['foo()', '[nameSpan] foo']); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-12 10:12:36 -04:00
										 |  |  |     it('should record method call argument span', () => { | 
					
						
							|  |  |  |       const ast = parseAction('foo(1 + 2)'); | 
					
						
							|  |  |  |       expect(unparseWithSpan(ast)).toContain(['foo(1 + 2)', '[argumentSpan] 1 + 2']); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-27 18:54:30 -07:00
										 |  |  |     it('should record accessed method call span', () => { | 
					
						
							|  |  |  |       const ast = parseAction('foo.bar()'); | 
					
						
							|  |  |  |       expect(unparseWithSpan(ast)).toContain(['foo.bar()', 'foo.bar()']); | 
					
						
							|  |  |  |       expect(unparseWithSpan(ast)).toContain(['foo.bar()', '[nameSpan] bar']); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     it('should record safe method call span', () => { | 
					
						
							|  |  |  |       const ast = parseAction('foo?.bar()'); | 
					
						
							|  |  |  |       expect(unparseWithSpan(ast)).toContain(['foo?.bar()', 'foo?.bar()']); | 
					
						
							|  |  |  |       expect(unparseWithSpan(ast)).toContain(['foo?.bar()', '[nameSpan] bar']); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     it('should record property write span', () => { | 
					
						
							|  |  |  |       const ast = parseAction('a = b'); | 
					
						
							|  |  |  |       expect(unparseWithSpan(ast)).toContain(['a = b', 'a = b']); | 
					
						
							|  |  |  |       expect(unparseWithSpan(ast)).toContain(['a = b', '[nameSpan] a']); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     it('should record accessed property write span', () => { | 
					
						
							|  |  |  |       const ast = parseAction('a.b = c'); | 
					
						
							|  |  |  |       expect(unparseWithSpan(ast)).toContain(['a.b = c', 'a.b = c']); | 
					
						
							|  |  |  |       expect(unparseWithSpan(ast)).toContain(['a.b = c', '[nameSpan] b']); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2021-02-06 20:25:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     it('should include parenthesis in spans', () => { | 
					
						
							|  |  |  |       // When a LHS expression is parenthesized, the parenthesis on the left used to be
 | 
					
						
							|  |  |  |       // excluded from the span. This test verifies that the parenthesis are properly included
 | 
					
						
							|  |  |  |       // in the span for both LHS and RHS expressions.
 | 
					
						
							|  |  |  |       // https://github.com/angular/angular/issues/40721
 | 
					
						
							|  |  |  |       expectSpan('(foo) && (bar)'); | 
					
						
							|  |  |  |       expectSpan('(foo) || (bar)'); | 
					
						
							|  |  |  |       expectSpan('(foo) == (bar)'); | 
					
						
							|  |  |  |       expectSpan('(foo) === (bar)'); | 
					
						
							|  |  |  |       expectSpan('(foo) != (bar)'); | 
					
						
							|  |  |  |       expectSpan('(foo) !== (bar)'); | 
					
						
							|  |  |  |       expectSpan('(foo) > (bar)'); | 
					
						
							|  |  |  |       expectSpan('(foo) >= (bar)'); | 
					
						
							|  |  |  |       expectSpan('(foo) < (bar)'); | 
					
						
							|  |  |  |       expectSpan('(foo) <= (bar)'); | 
					
						
							|  |  |  |       expectSpan('(foo) + (bar)'); | 
					
						
							|  |  |  |       expectSpan('(foo) - (bar)'); | 
					
						
							|  |  |  |       expectSpan('(foo) * (bar)'); | 
					
						
							|  |  |  |       expectSpan('(foo) / (bar)'); | 
					
						
							|  |  |  |       expectSpan('(foo) % (bar)'); | 
					
						
							|  |  |  |       expectSpan('(foo) | pipe'); | 
					
						
							|  |  |  |       expectSpan('(foo)()'); | 
					
						
							|  |  |  |       expectSpan('(foo).bar'); | 
					
						
							|  |  |  |       expectSpan('(foo)?.bar'); | 
					
						
							|  |  |  |       expectSpan('(foo).bar = (baz)'); | 
					
						
							|  |  |  |       expectSpan('(foo | pipe) == false'); | 
					
						
							|  |  |  |       expectSpan('(((foo) && bar) || baz) === true'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       function expectSpan(input: string) { | 
					
						
							|  |  |  |         expect(unparseWithSpan(parseBinding(input))).toContain([jasmine.any(String), input]); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2020-04-27 18:54:30 -07:00
										 |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |   describe('general error handling', () => { | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     it('should report an unexpected token', () => { | 
					
						
							|  |  |  |       expectActionError('[1,2] trac', 'Unexpected token \'trac\''); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-08-12 16:26:21 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     it('should report reasonable error for unconsumed tokens', () => { | 
					
						
							|  |  |  |       expectActionError(')', 'Unexpected token ) at column 1 in [)]'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-08-12 16:26:21 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     it('should report a missing expected token', () => { | 
					
						
							|  |  |  |       expectActionError('a(b', 'Missing expected ) at the end of the expression [a(b]'); | 
					
						
							| 
									
										
										
										
											2015-08-12 16:26:21 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |   }); | 
					
						
							| 
									
										
										
										
											2015-08-12 16:26:21 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |   describe('parseBinding', () => { | 
					
						
							|  |  |  |     describe('pipes', () => { | 
					
						
							|  |  |  |       it('should parse pipes', () => { | 
					
						
							|  |  |  |         checkBinding('a(b | c)', 'a((b | c))'); | 
					
						
							|  |  |  |         checkBinding('a.b(c.d(e) | f)', 'a.b((c.d(e) | f))'); | 
					
						
							|  |  |  |         checkBinding('[1, 2, 3] | a', '([1, 2, 3] | a)'); | 
					
						
							|  |  |  |         checkBinding('{a: 1, "b": 2} | c', '({a: 1, "b": 2} | c)'); | 
					
						
							|  |  |  |         checkBinding('a[b] | c', '(a[b] | c)'); | 
					
						
							|  |  |  |         checkBinding('a?.b | c', '(a?.b | c)'); | 
					
						
							|  |  |  |         checkBinding('true | a', '(true | a)'); | 
					
						
							|  |  |  |         checkBinding('a | b:c | d', '((a | b:c) | d)'); | 
					
						
							|  |  |  |         checkBinding('a | b:(c | d)', '(a | b:(c | d))'); | 
					
						
							| 
									
										
										
										
											2014-11-18 16:38:36 -08:00
										 |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-26 17:40:26 -05:00
										 |  |  |       describe('should parse incomplete pipes', () => { | 
					
						
							|  |  |  |         const cases: Array<[string, string, string, string]> = [ | 
					
						
							|  |  |  |           [ | 
					
						
							|  |  |  |             'should parse missing pipe names: end', | 
					
						
							|  |  |  |             'a | b | ', | 
					
						
							|  |  |  |             '((a | b) | )', | 
					
						
							|  |  |  |             'Unexpected end of input, expected identifier or keyword', | 
					
						
							|  |  |  |           ], | 
					
						
							|  |  |  |           [ | 
					
						
							|  |  |  |             'should parse missing pipe names: middle', | 
					
						
							|  |  |  |             'a | | b', | 
					
						
							|  |  |  |             '((a | ) | b)', | 
					
						
							|  |  |  |             'Unexpected token |, expected identifier or keyword', | 
					
						
							|  |  |  |           ], | 
					
						
							|  |  |  |           [ | 
					
						
							|  |  |  |             'should parse missing pipe names: start', | 
					
						
							|  |  |  |             ' | a | b', | 
					
						
							|  |  |  |             '(( | a) | b)', | 
					
						
							|  |  |  |             'Unexpected token |', | 
					
						
							|  |  |  |           ], | 
					
						
							|  |  |  |           [ | 
					
						
							|  |  |  |             'should parse missing pipe args: end', | 
					
						
							|  |  |  |             'a | b | c: ', | 
					
						
							|  |  |  |             '((a | b) | c:)', | 
					
						
							|  |  |  |             'Unexpected end of expression', | 
					
						
							|  |  |  |           ], | 
					
						
							|  |  |  |           [ | 
					
						
							|  |  |  |             'should parse missing pipe args: middle', | 
					
						
							|  |  |  |             'a | b: | c', | 
					
						
							|  |  |  |             '((a | b:) | c)', | 
					
						
							|  |  |  |             'Unexpected token |', | 
					
						
							|  |  |  |           ], | 
					
						
							|  |  |  |           [ | 
					
						
							|  |  |  |             'should parse incomplete pipe args', | 
					
						
							|  |  |  |             'a | b: (a | ) + | c', | 
					
						
							|  |  |  |             '((a | b:(a | ) + ) | c)', | 
					
						
							|  |  |  |             'Unexpected token |', | 
					
						
							|  |  |  |           ], | 
					
						
							|  |  |  |         ]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (const [name, input, output, err] of cases) { | 
					
						
							|  |  |  |           it(name, () => { | 
					
						
							|  |  |  |             checkBinding(input, output); | 
					
						
							|  |  |  |             expectBindingError(input, err); | 
					
						
							|  |  |  |           }); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-01-07 14:16:05 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         it('should parse an incomplete pipe with a source span that includes trailing whitespace', | 
					
						
							|  |  |  |            () => { | 
					
						
							|  |  |  |              const bindingText = 'foo | '; | 
					
						
							|  |  |  |              const binding = parseBinding(bindingText).ast as BindingPipe; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |              // The sourceSpan should include all characters of the input.
 | 
					
						
							|  |  |  |              expect(rawSpan(binding.sourceSpan)).toEqual([0, bindingText.length]); | 
					
						
							|  |  |  |              // The nameSpan should be positioned at the end of the input.
 | 
					
						
							|  |  |  |              expect(rawSpan(binding.nameSpan)).toEqual([bindingText.length, bindingText.length]); | 
					
						
							|  |  |  |            }); | 
					
						
							| 
									
										
										
										
											2020-10-04 23:31:22 -05:00
										 |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |       it('should only allow identifier or keyword as formatter names', () => { | 
					
						
							|  |  |  |         expectBindingError('"Foo"|(', 'identifier or keyword'); | 
					
						
							|  |  |  |         expectBindingError('"Foo"|1234', 'identifier or keyword'); | 
					
						
							|  |  |  |         expectBindingError('"Foo"|"uppercase"', 'identifier or keyword'); | 
					
						
							| 
									
										
										
										
											2021-05-10 18:16:45 +02:00
										 |  |  |         expectBindingError('"Foo"|#privateIdentifier"', 'identifier or keyword'); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |       }); | 
					
						
							| 
									
										
										
										
											2014-12-10 19:21:15 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |       it('should parse quoted expressions', () => { | 
					
						
							|  |  |  |         checkBinding('a:b', 'a:b'); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2014-11-06 12:00:09 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |       it('should not crash when prefix part is not tokenizable', () => { | 
					
						
							|  |  |  |         checkBinding('"a:b"', '"a:b"'); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2014-11-06 12:00:09 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |       it('should ignore whitespace around quote prefix', () => { | 
					
						
							|  |  |  |         checkBinding(' a :b', 'a:b'); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2015-08-20 16:34:47 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |       it('should refuse prefixes that are not single identifiers', () => { | 
					
						
							|  |  |  |         expectBindingError('a + b:c', ''); | 
					
						
							|  |  |  |         expectBindingError('1:c', ''); | 
					
						
							| 
									
										
										
										
											2015-08-20 16:34:47 +02:00
										 |  |  |       }); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-12-05 22:06:24 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     it('should store the source in the result', () => { | 
					
						
							|  |  |  |       expect(parseBinding('someExpr').source).toBe('someExpr'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2016-04-14 16:16:22 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     it('should store the passed-in location', () => { | 
					
						
							|  |  |  |       expect(parseBinding('someExpr', 'location').location).toBe('location'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2016-04-14 16:16:22 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     it('should report chain expressions', () => { | 
					
						
							|  |  |  |       expectError(parseBinding('1;2'), 'contain chained expression'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2016-04-14 16:16:22 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     it('should report assignment', () => { | 
					
						
							|  |  |  |       expectError(parseBinding('a=2'), 'contain assignments'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2016-04-14 16:16:22 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     it('should report when encountering interpolation', () => { | 
					
						
							|  |  |  |       expectBindingError('{{a.b}}', 'Got interpolation ({{}}) where expression was expected'); | 
					
						
							| 
									
										
										
										
											2014-10-28 12:22:38 -04:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2014-11-18 16:38:36 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-26 12:45:40 +02:00
										 |  |  |     it('should not report interpolation inside a string', () => { | 
					
						
							|  |  |  |       expect(parseBinding(`"{{exp}}"`).errors).toEqual([]); | 
					
						
							|  |  |  |       expect(parseBinding(`'{{exp}}'`).errors).toEqual([]); | 
					
						
							|  |  |  |       expect(parseBinding(`'{{\\"}}'`).errors).toEqual([]); | 
					
						
							|  |  |  |       expect(parseBinding(`'{{\\'}}'`).errors).toEqual([]); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     it('should parse conditional expression', () => { | 
					
						
							|  |  |  |       checkBinding('a < b ? a : b'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-01-27 22:34:25 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     it('should ignore comments in bindings', () => { | 
					
						
							|  |  |  |       checkBinding('a //comment', 'a'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2016-11-10 13:15:09 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     it('should retain // in string literals', () => { | 
					
						
							|  |  |  |       checkBinding(`"http://www.google.com"`, `"http://www.google.com"`); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2014-11-18 16:38:36 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     it('should retain // in : microsyntax', () => { | 
					
						
							|  |  |  |       checkBinding('one:a//b', 'one:a//b'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |   }); | 
					
						
							| 
									
										
										
										
											2014-11-18 16:38:36 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |   describe('parseTemplateBindings', () => { | 
					
						
							| 
									
										
										
										
											2020-03-02 16:38:55 -08:00
										 |  |  |     function humanize(bindings: TemplateBinding[]): Array<[string, string | null, boolean]> { | 
					
						
							|  |  |  |       return bindings.map(binding => { | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |         const key = binding.key.source; | 
					
						
							|  |  |  |         const value = binding.value ? binding.value.source : null; | 
					
						
							|  |  |  |         const keyIsVar = binding instanceof VariableBinding; | 
					
						
							| 
									
										
										
										
											2020-03-02 16:38:55 -08:00
										 |  |  |         return [key, value, keyIsVar]; | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |     function humanizeSpans( | 
					
						
							|  |  |  |         bindings: TemplateBinding[], attr: string): Array<[string, string, string | null]> { | 
					
						
							|  |  |  |       return bindings.map(binding => { | 
					
						
							|  |  |  |         const {sourceSpan, key, value} = binding; | 
					
						
							|  |  |  |         const sourceStr = attr.substring(sourceSpan.start, sourceSpan.end); | 
					
						
							|  |  |  |         const keyStr = attr.substring(key.span.start, key.span.end); | 
					
						
							|  |  |  |         let valueStr = null; | 
					
						
							|  |  |  |         if (value) { | 
					
						
							|  |  |  |           const {start, end} = value instanceof ASTWithSource ? value.ast.sourceSpan : value.span; | 
					
						
							|  |  |  |           valueStr = attr.substring(start, end); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return [sourceStr, keyStr, valueStr]; | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-11-18 16:38:36 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |     it('should parse key and value', () => { | 
					
						
							|  |  |  |       const cases: Array<[string, string, string | null, boolean, string, string, string | null]> = | 
					
						
							|  |  |  |           [ | 
					
						
							|  |  |  |             // expression, key, value, VariableBinding, source span, key span, value span
 | 
					
						
							|  |  |  |             ['*a=""', 'a', null, false, 'a="', 'a', null], | 
					
						
							|  |  |  |             ['*a="b"', 'a', 'b', false, 'a="b', 'a', 'b'], | 
					
						
							|  |  |  |             ['*a-b="c"', 'a-b', 'c', false, 'a-b="c', 'a-b', 'c'], | 
					
						
							|  |  |  |             ['*a="1+1"', 'a', '1+1', false, 'a="1+1', 'a', '1+1'], | 
					
						
							|  |  |  |           ]; | 
					
						
							|  |  |  |       for (const [attr, key, value, keyIsVar, sourceSpan, keySpan, valueSpan] of cases) { | 
					
						
							|  |  |  |         const bindings = parseTemplateBindings(attr); | 
					
						
							|  |  |  |         expect(humanize(bindings)).toEqual([ | 
					
						
							|  |  |  |           [key, value, keyIsVar], | 
					
						
							|  |  |  |         ]); | 
					
						
							|  |  |  |         expect(humanizeSpans(bindings, attr)).toEqual([ | 
					
						
							|  |  |  |           [sourceSpan, keySpan, valueSpan], | 
					
						
							|  |  |  |         ]); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2014-11-18 16:38:36 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |     it('should variable declared via let', () => { | 
					
						
							|  |  |  |       const bindings = parseTemplateBindings('*a="let b"'); | 
					
						
							|  |  |  |       expect(humanize(bindings)).toEqual([ | 
					
						
							|  |  |  |         // key, value, VariableBinding
 | 
					
						
							|  |  |  |         ['a', null, false], | 
					
						
							|  |  |  |         ['b', null, true], | 
					
						
							|  |  |  |       ]); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2014-12-10 19:21:15 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     it('should allow multiple pairs', () => { | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |       const bindings = parseTemplateBindings('*a="1 b 2"'); | 
					
						
							|  |  |  |       expect(humanize(bindings)).toEqual([ | 
					
						
							|  |  |  |         // key, value, VariableBinding
 | 
					
						
							|  |  |  |         ['a', '1', false], | 
					
						
							|  |  |  |         ['aB', '2', false], | 
					
						
							|  |  |  |       ]); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-01-27 22:34:25 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |     it('should allow space and colon as separators', () => { | 
					
						
							|  |  |  |       const bindings = parseTemplateBindings('*a="1,b 2"'); | 
					
						
							|  |  |  |       expect(humanize(bindings)).toEqual([ | 
					
						
							|  |  |  |         // key, value, VariableBinding
 | 
					
						
							|  |  |  |         ['a', '1', false], | 
					
						
							|  |  |  |         ['aB', '2', false], | 
					
						
							|  |  |  |       ]); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-01-27 22:34:25 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |     it('should store the templateUrl', () => { | 
					
						
							|  |  |  |       const bindings = parseTemplateBindings('*a="1,b 2"', '/foo/bar.html'); | 
					
						
							|  |  |  |       expect(humanize(bindings)).toEqual([ | 
					
						
							|  |  |  |         // key, value, VariableBinding
 | 
					
						
							|  |  |  |         ['a', '1', false], | 
					
						
							|  |  |  |         ['aB', '2', false], | 
					
						
							|  |  |  |       ]); | 
					
						
							|  |  |  |       expect((bindings[0].value as ASTWithSource).location).toEqual('/foo/bar.html'); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-05-21 12:17:46 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-02 16:38:55 -08:00
										 |  |  |     it('should support common usage of ngIf', () => { | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |       const bindings = parseTemplateBindings('*ngIf="cond | pipe as foo, let x; ngIf as y"'); | 
					
						
							| 
									
										
										
										
											2020-03-02 16:38:55 -08:00
										 |  |  |       expect(humanize(bindings)).toEqual([ | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |         // [ key, value, VariableBinding ]
 | 
					
						
							|  |  |  |         ['ngIf', 'cond | pipe', false], | 
					
						
							| 
									
										
										
										
											2020-03-02 16:38:55 -08:00
										 |  |  |         ['foo', 'ngIf', true], | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |         ['x', null, true], | 
					
						
							| 
									
										
										
										
											2020-03-02 16:38:55 -08:00
										 |  |  |         ['y', 'ngIf', true], | 
					
						
							|  |  |  |       ]); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     it('should support common usage of ngFor', () => { | 
					
						
							|  |  |  |       let bindings: TemplateBinding[]; | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |       bindings = parseTemplateBindings('*ngFor="let person of people"'); | 
					
						
							|  |  |  |       expect(humanize(bindings)).toEqual([ | 
					
						
							|  |  |  |         // [ key, value, VariableBinding ]
 | 
					
						
							|  |  |  |         ['ngFor', null, false], | 
					
						
							|  |  |  |         ['person', null, true], | 
					
						
							|  |  |  |         ['ngForOf', 'people', false], | 
					
						
							|  |  |  |       ]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-02 16:38:55 -08:00
										 |  |  |       bindings = parseTemplateBindings( | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |           '*ngFor="let item; of items | slice:0:1 as collection, trackBy: func; index as i"'); | 
					
						
							| 
									
										
										
										
											2020-03-02 16:38:55 -08:00
										 |  |  |       expect(humanize(bindings)).toEqual([ | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |         // [ key, value, VariableBinding ]
 | 
					
						
							| 
									
										
										
										
											2020-03-02 16:38:55 -08:00
										 |  |  |         ['ngFor', null, false], | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |         ['item', null, true], | 
					
						
							|  |  |  |         ['ngForOf', 'items | slice:0:1', false], | 
					
						
							| 
									
										
										
										
											2020-03-02 16:38:55 -08:00
										 |  |  |         ['collection', 'ngForOf', true], | 
					
						
							|  |  |  |         ['ngForTrackBy', 'func', false], | 
					
						
							|  |  |  |         ['i', 'index', true], | 
					
						
							|  |  |  |       ]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       bindings = parseTemplateBindings( | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |           '*ngFor="let item, of: [1,2,3] | pipe as items; let i=index, count as len"'); | 
					
						
							| 
									
										
										
										
											2020-03-02 16:38:55 -08:00
										 |  |  |       expect(humanize(bindings)).toEqual([ | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |         // [ key, value, VariableBinding ]
 | 
					
						
							| 
									
										
										
										
											2020-03-02 16:38:55 -08:00
										 |  |  |         ['ngFor', null, false], | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |         ['item', null, true], | 
					
						
							|  |  |  |         ['ngForOf', '[1,2,3] | pipe', false], | 
					
						
							| 
									
										
										
										
											2020-03-02 16:38:55 -08:00
										 |  |  |         ['items', 'ngForOf', true], | 
					
						
							|  |  |  |         ['i', 'index', true], | 
					
						
							|  |  |  |         ['len', 'count', true], | 
					
						
							|  |  |  |       ]); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |     it('should parse pipes', () => { | 
					
						
							|  |  |  |       const bindings = parseTemplateBindings('*key="value|pipe "'); | 
					
						
							|  |  |  |       expect(humanize(bindings)).toEqual([ | 
					
						
							|  |  |  |         // [ key, value, VariableBinding ]
 | 
					
						
							|  |  |  |         ['key', 'value|pipe', false], | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |       ]); | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |       const {value} = bindings[0]; | 
					
						
							|  |  |  |       expect(value).toBeAnInstanceOf(ASTWithSource); | 
					
						
							|  |  |  |       expect((value as ASTWithSource).ast).toBeAnInstanceOf(BindingPipe); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     describe('"let" binding', () => { | 
					
						
							|  |  |  |       it('should support single declaration', () => { | 
					
						
							|  |  |  |         const bindings = parseTemplateBindings('*key="let i"'); | 
					
						
							|  |  |  |         expect(humanize(bindings)).toEqual([ | 
					
						
							|  |  |  |           // [ key, value, VariableBinding ]
 | 
					
						
							|  |  |  |           ['key', null, false], | 
					
						
							|  |  |  |           ['i', null, true], | 
					
						
							|  |  |  |         ]); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |       it('should support multiple declarations', () => { | 
					
						
							|  |  |  |         const bindings = parseTemplateBindings('*key="let a; let b"'); | 
					
						
							|  |  |  |         expect(humanize(bindings)).toEqual([ | 
					
						
							|  |  |  |           // [ key, value, VariableBinding ]
 | 
					
						
							|  |  |  |           ['key', null, false], | 
					
						
							|  |  |  |           ['a', null, true], | 
					
						
							|  |  |  |           ['b', null, true], | 
					
						
							|  |  |  |         ]); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |       it('should support empty string assignment', () => { | 
					
						
							|  |  |  |         const bindings = parseTemplateBindings(`*key="let a=''; let b='';"`); | 
					
						
							|  |  |  |         expect(humanize(bindings)).toEqual([ | 
					
						
							|  |  |  |           // [ key, value, VariableBinding ]
 | 
					
						
							|  |  |  |           ['key', null, false], | 
					
						
							|  |  |  |           ['a', '', true], | 
					
						
							|  |  |  |           ['b', '', true], | 
					
						
							|  |  |  |         ]); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |       it('should support key and value names with dash', () => { | 
					
						
							|  |  |  |         const bindings = parseTemplateBindings('*key="let i-a = j-a,"'); | 
					
						
							|  |  |  |         expect(humanize(bindings)).toEqual([ | 
					
						
							|  |  |  |           // [ key, value, VariableBinding ]
 | 
					
						
							|  |  |  |           ['key', null, false], | 
					
						
							|  |  |  |           ['i-a', 'j-a', true], | 
					
						
							|  |  |  |         ]); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |       it('should support declarations with or without value assignment', () => { | 
					
						
							|  |  |  |         const bindings = parseTemplateBindings('*key="let item; let i = k"'); | 
					
						
							|  |  |  |         expect(humanize(bindings)).toEqual([ | 
					
						
							|  |  |  |           // [ key, value, VariableBinding ]
 | 
					
						
							|  |  |  |           ['key', null, false], | 
					
						
							|  |  |  |           ['item', null, true], | 
					
						
							|  |  |  |           ['i', 'k', true], | 
					
						
							|  |  |  |         ]); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       it('should support declaration before an expression', () => { | 
					
						
							|  |  |  |         const bindings = parseTemplateBindings('*directive="let item in expr; let a = b"'); | 
					
						
							|  |  |  |         expect(humanize(bindings)).toEqual([ | 
					
						
							|  |  |  |           // [ key, value, VariableBinding ]
 | 
					
						
							|  |  |  |           ['directive', null, false], | 
					
						
							|  |  |  |           ['item', null, true], | 
					
						
							|  |  |  |           ['directiveIn', 'expr', false], | 
					
						
							|  |  |  |           ['a', 'b', true], | 
					
						
							|  |  |  |         ]); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-05-21 12:17:46 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |     describe('"as" binding', () => { | 
					
						
							|  |  |  |       it('should support single declaration', () => { | 
					
						
							|  |  |  |         const bindings = parseTemplateBindings('*ngIf="exp as local"'); | 
					
						
							|  |  |  |         expect(humanize(bindings)).toEqual([ | 
					
						
							|  |  |  |           // [ key, value, VariableBinding ]
 | 
					
						
							|  |  |  |           ['ngIf', 'exp', false], | 
					
						
							|  |  |  |           ['local', 'ngIf', true], | 
					
						
							|  |  |  |         ]); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2015-01-27 22:34:25 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |       it('should support declaration after an expression', () => { | 
					
						
							|  |  |  |         const bindings = parseTemplateBindings('*ngFor="let item of items as iter; index as i"'); | 
					
						
							|  |  |  |         expect(humanize(bindings)).toEqual([ | 
					
						
							|  |  |  |           // [ key, value, VariableBinding ]
 | 
					
						
							|  |  |  |           ['ngFor', null, false], | 
					
						
							|  |  |  |           ['item', null, true], | 
					
						
							|  |  |  |           ['ngForOf', 'items', false], | 
					
						
							|  |  |  |           ['iter', 'ngForOf', true], | 
					
						
							|  |  |  |           ['i', 'index', true], | 
					
						
							|  |  |  |         ]); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2015-01-27 22:34:25 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |       it('should support key and value names with dash', () => { | 
					
						
							|  |  |  |         const bindings = parseTemplateBindings('*key="foo, k-b as l-b;"'); | 
					
						
							|  |  |  |         expect(humanize(bindings)).toEqual([ | 
					
						
							|  |  |  |           // [ key, value, VariableBinding ]
 | 
					
						
							|  |  |  |           ['key', 'foo', false], | 
					
						
							|  |  |  |           ['l-b', 'k-b', true], | 
					
						
							|  |  |  |         ]); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-01-27 22:34:25 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |     describe('source, key, value spans', () => { | 
					
						
							|  |  |  |       it('should map empty expression', () => { | 
					
						
							|  |  |  |         const attr = '*ngIf=""'; | 
					
						
							|  |  |  |         const bindings = parseTemplateBindings(attr); | 
					
						
							|  |  |  |         expect(humanizeSpans(bindings, attr)).toEqual([ | 
					
						
							|  |  |  |           // source span, key span, value span
 | 
					
						
							|  |  |  |           ['ngIf="', 'ngIf', null], | 
					
						
							|  |  |  |         ]); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |       }); | 
					
						
							| 
									
										
										
										
											2015-01-27 22:34:25 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |       it('should map variable declaration via "let"', () => { | 
					
						
							|  |  |  |         const attr = '*key="let i"'; | 
					
						
							|  |  |  |         const bindings = parseTemplateBindings(attr); | 
					
						
							|  |  |  |         expect(humanizeSpans(bindings, attr)).toEqual([ | 
					
						
							|  |  |  |           // source span, key span, value span
 | 
					
						
							|  |  |  |           ['key="', 'key', null],  // source span stretches till next binding
 | 
					
						
							|  |  |  |           ['let i', 'i', null], | 
					
						
							| 
									
										
										
										
											2016-06-08 16:38:52 -07:00
										 |  |  |         ]); | 
					
						
							| 
									
										
										
										
											2015-01-27 22:34:25 -08:00
										 |  |  |       }); | 
					
						
							| 
									
										
										
										
											2015-02-23 13:12:49 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |       it('shoud map multiple variable declarations via "let"', () => { | 
					
						
							|  |  |  |         const attr = '*key="let item; let i=index; let e=even;"'; | 
					
						
							|  |  |  |         const bindings = parseTemplateBindings(attr); | 
					
						
							|  |  |  |         expect(humanizeSpans(bindings, attr)).toEqual([ | 
					
						
							|  |  |  |           // source span, key span, value span
 | 
					
						
							|  |  |  |           ['key="', 'key', null], | 
					
						
							|  |  |  |           ['let item; ', 'item', null], | 
					
						
							|  |  |  |           ['let i=index; ', 'i', 'index'], | 
					
						
							|  |  |  |           ['let e=even;', 'e', 'even'], | 
					
						
							|  |  |  |         ]); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       it('shoud map expression with pipe', () => { | 
					
						
							|  |  |  |         const attr = '*ngIf="cond | pipe as foo, let x; ngIf as y"'; | 
					
						
							|  |  |  |         const bindings = parseTemplateBindings(attr); | 
					
						
							|  |  |  |         expect(humanizeSpans(bindings, attr)).toEqual([ | 
					
						
							|  |  |  |           // source span, key span, value span
 | 
					
						
							| 
									
										
										
										
											2019-12-18 19:30:56 -06:00
										 |  |  |           ['ngIf="cond | pipe ', 'ngIf', 'cond | pipe'], | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |           ['ngIf="cond | pipe as foo, ', 'foo', 'ngIf'], | 
					
						
							|  |  |  |           ['let x; ', 'x', null], | 
					
						
							|  |  |  |           ['ngIf as y', 'y', 'ngIf'], | 
					
						
							|  |  |  |         ]); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-07 17:50:44 +02:00
										 |  |  |       it('should report unexpected token when encountering interpolation', () => { | 
					
						
							|  |  |  |         const attr = '*ngIf="name && {{name}}"'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         expectParseTemplateBindingsError( | 
					
						
							|  |  |  |             attr, | 
					
						
							|  |  |  |             'Parser Error: Unexpected token {, expected identifier, keyword, or string at column 10 in [name && {{name}}] in foo.html'); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |       it('should map variable declaration via "as"', () => { | 
					
						
							|  |  |  |         const attr = | 
					
						
							|  |  |  |             '*ngFor="let item; of items | slice:0:1 as collection, trackBy: func; index as i"'; | 
					
						
							|  |  |  |         const bindings = parseTemplateBindings(attr); | 
					
						
							|  |  |  |         expect(humanizeSpans(bindings, attr)).toEqual([ | 
					
						
							|  |  |  |           // source span, key span, value span
 | 
					
						
							|  |  |  |           ['ngFor="', 'ngFor', null], | 
					
						
							|  |  |  |           ['let item; ', 'item', null], | 
					
						
							| 
									
										
										
										
											2019-12-18 19:30:56 -06:00
										 |  |  |           ['of items | slice:0:1 ', 'of', 'items | slice:0:1'], | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |           ['of items | slice:0:1 as collection, ', 'collection', 'of'], | 
					
						
							|  |  |  |           ['trackBy: func; ', 'trackBy', 'func'], | 
					
						
							|  |  |  |           ['index as i', 'i', 'index'], | 
					
						
							|  |  |  |         ]); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       it('should map literal array', () => { | 
					
						
							|  |  |  |         const attr = '*ngFor="let item, of: [1,2,3] | pipe as items; let i=index, count as len, "'; | 
					
						
							|  |  |  |         const bindings = parseTemplateBindings(attr); | 
					
						
							|  |  |  |         expect(humanizeSpans(bindings, attr)).toEqual([ | 
					
						
							|  |  |  |           // source span, key span, value span
 | 
					
						
							|  |  |  |           ['ngFor="', 'ngFor', null], | 
					
						
							|  |  |  |           ['let item, ', 'item', null], | 
					
						
							| 
									
										
										
										
											2019-12-18 19:30:56 -06:00
										 |  |  |           ['of: [1,2,3] | pipe ', 'of', '[1,2,3] | pipe'], | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |           ['of: [1,2,3] | pipe as items; ', 'items', 'of'], | 
					
						
							|  |  |  |           ['let i=index, ', 'i', 'index'], | 
					
						
							| 
									
										
										
										
											2020-04-27 18:54:30 -07:00
										 |  |  |           ['count as len,', 'len', 'count'], | 
					
						
							| 
									
										
										
										
											2017-03-14 20:46:29 -07:00
										 |  |  |         ]); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     }); | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2017-03-14 20:46:29 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |   describe('parseInterpolation', () => { | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     it('should return null if no interpolation', () => { | 
					
						
							|  |  |  |       expect(parseInterpolation('nothing')).toBe(null); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2016-11-10 13:15:09 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-24 11:22:12 -05:00
										 |  |  |     it('should not parse malformed interpolations as strings', () => { | 
					
						
							|  |  |  |       const ast = parseInterpolation('{{a}} {{example}<!--->}')!.ast as Interpolation; | 
					
						
							|  |  |  |       expect(ast.strings).toEqual(['', ' {{example}<!--->}']); | 
					
						
							|  |  |  |       expect(ast.expressions.length).toEqual(1); | 
					
						
							|  |  |  |       expect(ast.expressions[0].name).toEqual('a'); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     it('should parse no prefix/suffix interpolation', () => { | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |       const ast = parseInterpolation('{{a}}')!.ast as Interpolation; | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |       expect(ast.strings).toEqual(['', '']); | 
					
						
							|  |  |  |       expect(ast.expressions.length).toEqual(1); | 
					
						
							|  |  |  |       expect(ast.expressions[0].name).toEqual('a'); | 
					
						
							| 
									
										
										
										
											2014-11-18 16:38:36 -08:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-01-08 16:17:56 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-24 18:50:47 +01:00
										 |  |  |     it('should parse interpolation inside quotes', () => { | 
					
						
							|  |  |  |       const ast = parseInterpolation('"{{a}}"')!.ast as Interpolation; | 
					
						
							|  |  |  |       expect(ast.strings).toEqual(['"', '"']); | 
					
						
							|  |  |  |       expect(ast.expressions.length).toEqual(1); | 
					
						
							|  |  |  |       expect(ast.expressions[0].name).toEqual('a'); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     it('should parse interpolation with interpolation characters inside quotes', () => { | 
					
						
							|  |  |  |       checkInterpolation('{{"{{a}}"}}', '{{ "{{a}}" }}'); | 
					
						
							|  |  |  |       checkInterpolation('{{"{{"}}', '{{ "{{" }}'); | 
					
						
							|  |  |  |       checkInterpolation('{{"}}"}}', '{{ "}}" }}'); | 
					
						
							|  |  |  |       checkInterpolation('{{"{"}}', '{{ "{" }}'); | 
					
						
							|  |  |  |       checkInterpolation('{{"}"}}', '{{ "}" }}'); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     it('should parse interpolation with escaped quotes', () => { | 
					
						
							|  |  |  |       checkInterpolation(`{{'It\\'s just Angular'}}`, `{{ "It's just Angular" }}`); | 
					
						
							|  |  |  |       checkInterpolation(`{{'It\\'s {{ just Angular'}}`, `{{ "It's {{ just Angular" }}`); | 
					
						
							|  |  |  |       checkInterpolation(`{{'It\\'s }} just Angular'}}`, `{{ "It's }} just Angular" }}`); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     it('should parse interpolation with escaped backslashes', () => { | 
					
						
							|  |  |  |       checkInterpolation(`{{foo.split('\\\\')}}`, `{{ foo.split("\\") }}`); | 
					
						
							|  |  |  |       checkInterpolation(`{{foo.split('\\\\\\\\')}}`, `{{ foo.split("\\\\") }}`); | 
					
						
							|  |  |  |       checkInterpolation(`{{foo.split('\\\\\\\\\\\\')}}`, `{{ foo.split("\\\\\\") }}`); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     it('should not parse interpolation with mismatching quotes', () => { | 
					
						
							|  |  |  |       expect(parseInterpolation(`{{ "{{a}}' }}`)).toBeNull(); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     it('should parse prefix/suffix with multiple interpolation', () => { | 
					
						
							|  |  |  |       const originalExp = 'before {{ a }} middle {{ b }} after'; | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |       const ast = parseInterpolation(originalExp)!.ast; | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |       expect(unparse(ast)).toEqual(originalExp); | 
					
						
							|  |  |  |       validate(ast); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-01-08 16:17:56 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     it('should report empty interpolation expressions', () => { | 
					
						
							|  |  |  |       expectError( | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |           parseInterpolation('{{}}')!, 'Blank expressions are not allowed in interpolated strings'); | 
					
						
							| 
									
										
										
										
											2015-01-08 16:17:56 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |       expectError( | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |           parseInterpolation('foo {{  }}')!, | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |           'Parser Error: Blank expressions are not allowed in interpolated strings'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-08-03 12:05:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 12:45:18 -08:00
										 |  |  |     it('should produce an empty expression ast for empty interpolations', () => { | 
					
						
							|  |  |  |       const parsed = parseInterpolation('{{}}')!.ast as Interpolation; | 
					
						
							|  |  |  |       expect(parsed.expressions.length).toBe(1); | 
					
						
							|  |  |  |       expect(parsed.expressions[0]).toBeAnInstanceOf(EmptyExpr); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |     it('should parse conditional expression', () => { | 
					
						
							|  |  |  |       checkInterpolation('{{ a < b ? a : b }}'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-08-03 12:05:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     it('should parse expression with newline characters', () => { | 
					
						
							|  |  |  |       checkInterpolation(`{{ 'foo' +\n 'bar' +\r 'baz' }}`, `{{ "foo" + "bar" + "baz" }}`); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-12-05 22:06:24 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     it('should support custom interpolation', () => { | 
					
						
							|  |  |  |       const parser = new Parser(new Lexer()); | 
					
						
							| 
									
										
										
										
											2020-11-17 10:07:54 +00:00
										 |  |  |       const ast = parser.parseInterpolation('{% a %}', '', 0, {start: '{%', end: '%}'})!.ast as any; | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |       expect(ast.strings).toEqual(['', '']); | 
					
						
							|  |  |  |       expect(ast.expressions.length).toEqual(1); | 
					
						
							|  |  |  |       expect(ast.expressions[0].name).toEqual('a'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-12-21 11:32:23 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     describe('comments', () => { | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |       it('should ignore comments in interpolation expressions', () => { | 
					
						
							|  |  |  |         checkInterpolation('{{a //comment}}', '{{ a }}'); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2016-04-14 16:16:22 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |       it('should retain // in single quote strings', () => { | 
					
						
							|  |  |  |         checkInterpolation(`{{ 'http://www.google.com' }}`, `{{ "http://www.google.com" }}`); | 
					
						
							| 
									
										
										
										
											2016-06-20 09:52:41 -07:00
										 |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |       it('should retain // in double quote strings', () => { | 
					
						
							|  |  |  |         checkInterpolation(`{{ "http://www.google.com" }}`, `{{ "http://www.google.com" }}`); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2016-04-14 16:16:22 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |       it('should ignore comments after string literals', () => { | 
					
						
							|  |  |  |         checkInterpolation(`{{ "a//b" //comment }}`, `{{ "a//b" }}`); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2016-04-14 16:16:22 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |       it('should retain // in complex strings', () => { | 
					
						
							|  |  |  |         checkInterpolation( | 
					
						
							|  |  |  |             `{{"//a\'//b\`//c\`//d\'//e" //comment}}`, `{{ "//a\'//b\`//c\`//d\'//e" }}`); | 
					
						
							| 
									
										
										
										
											2016-04-14 16:16:22 -07:00
										 |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |       it('should retain // in nested, unterminated strings', () => { | 
					
						
							|  |  |  |         checkInterpolation(`{{ "a\'b\`" //comment}}`, `{{ "a\'b\`" }}`); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2020-11-24 18:50:47 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |       it('should ignore quotes inside a comment', () => { | 
					
						
							|  |  |  |         checkInterpolation(`"{{name // " }}"`, `"{{ name }}"`); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2015-01-08 16:17:56 -08:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |   }); | 
					
						
							| 
									
										
										
										
											2015-06-22 08:21:03 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |   describe('parseSimpleBinding', () => { | 
					
						
							|  |  |  |     it('should parse a field access', () => { | 
					
						
							|  |  |  |       const p = parseSimpleBinding('name'); | 
					
						
							|  |  |  |       expect(unparse(p)).toEqual('name'); | 
					
						
							|  |  |  |       validate(p); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-08-20 16:34:47 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     it('should report when encountering pipes', () => { | 
					
						
							|  |  |  |       expectError( | 
					
						
							|  |  |  |           validate(parseSimpleBinding('a | somePipe')), | 
					
						
							|  |  |  |           'Host binding expression cannot contain pipes'); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2016-10-20 15:24:58 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     it('should report when encountering interpolation', () => { | 
					
						
							|  |  |  |       expectError( | 
					
						
							|  |  |  |           validate(parseSimpleBinding('{{exp}}')), | 
					
						
							|  |  |  |           'Got interpolation ({{}}) where expression was expected'); | 
					
						
							| 
									
										
										
										
											2015-06-22 08:21:03 -07:00
										 |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-26 12:45:40 +02:00
										 |  |  |     it('should not report interpolation inside a string', () => { | 
					
						
							|  |  |  |       expect(parseSimpleBinding(`"{{exp}}"`).errors).toEqual([]); | 
					
						
							|  |  |  |       expect(parseSimpleBinding(`'{{exp}}'`).errors).toEqual([]); | 
					
						
							|  |  |  |       expect(parseSimpleBinding(`'{{\\"}}'`).errors).toEqual([]); | 
					
						
							|  |  |  |       expect(parseSimpleBinding(`'{{\\'}}'`).errors).toEqual([]); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     it('should report when encountering field write', () => { | 
					
						
							|  |  |  |       expectError(validate(parseSimpleBinding('a = b')), 'Bindings cannot contain assignments'); | 
					
						
							| 
									
										
										
										
											2015-02-05 20:13:32 +01:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2020-07-04 09:21:40 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     describe('Ivy-only validations', () => { | 
					
						
							|  |  |  |       it('should throw if a pipe is used inside a conditional', () => { | 
					
						
							|  |  |  |         expectError( | 
					
						
							|  |  |  |             validate(parseSimpleBindingIvy('(hasId | myPipe) ? "my-id" : ""')), | 
					
						
							|  |  |  |             'Host binding expression cannot contain pipes'); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       it('should throw if a pipe is used inside a function call', () => { | 
					
						
							|  |  |  |         expectError( | 
					
						
							|  |  |  |             validate(parseSimpleBindingIvy('getId(true, id | myPipe)')), | 
					
						
							|  |  |  |             'Host binding expression cannot contain pipes'); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       it('should throw if a pipe is used inside a method call', () => { | 
					
						
							|  |  |  |         expectError( | 
					
						
							|  |  |  |             validate(parseSimpleBindingIvy('idService.getId(true, id | myPipe)')), | 
					
						
							|  |  |  |             'Host binding expression cannot contain pipes'); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       it('should throw if a pipe is used inside a safe method call', () => { | 
					
						
							|  |  |  |         expectError( | 
					
						
							|  |  |  |             validate(parseSimpleBindingIvy('idService?.getId(true, id | myPipe)')), | 
					
						
							|  |  |  |             'Host binding expression cannot contain pipes'); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       it('should throw if a pipe is used inside a property access', () => { | 
					
						
							|  |  |  |         expectError( | 
					
						
							|  |  |  |             validate(parseSimpleBindingIvy('a[id | myPipe]')), | 
					
						
							|  |  |  |             'Host binding expression cannot contain pipes'); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       it('should throw if a pipe is used inside a keyed read expression', () => { | 
					
						
							|  |  |  |         expectError( | 
					
						
							|  |  |  |             validate(parseSimpleBindingIvy('a[id | myPipe].b')), | 
					
						
							|  |  |  |             'Host binding expression cannot contain pipes'); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       it('should throw if a pipe is used inside a safe property read', () => { | 
					
						
							|  |  |  |         expectError( | 
					
						
							|  |  |  |             validate(parseSimpleBindingIvy('(id | myPipe)?.id')), | 
					
						
							|  |  |  |             'Host binding expression cannot contain pipes'); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       it('should throw if a pipe is used inside a non-null assertion', () => { | 
					
						
							|  |  |  |         expectError( | 
					
						
							|  |  |  |             validate(parseSimpleBindingIvy('[id | myPipe]!')), | 
					
						
							|  |  |  |             'Host binding expression cannot contain pipes'); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       it('should throw if a pipe is used inside a prefix not expression', () => { | 
					
						
							|  |  |  |         expectError( | 
					
						
							|  |  |  |             validate(parseSimpleBindingIvy('!(id | myPipe)')), | 
					
						
							|  |  |  |             'Host binding expression cannot contain pipes'); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       it('should throw if a pipe is used inside a binary expression', () => { | 
					
						
							|  |  |  |         expectError( | 
					
						
							|  |  |  |             validate(parseSimpleBindingIvy('(id | myPipe) === true')), | 
					
						
							|  |  |  |             'Host binding expression cannot contain pipes'); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |   }); | 
					
						
							| 
									
										
										
										
											2016-07-06 14:06:47 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |   describe('wrapLiteralPrimitive', () => { | 
					
						
							|  |  |  |     it('should wrap a literal primitive', () => { | 
					
						
							| 
									
										
										
										
											2020-11-17 10:07:54 +00:00
										 |  |  |       expect(unparse(validate(createParser().wrapLiteralPrimitive('foo', '', 0)))).toEqual('"foo"'); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |     }); | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2016-10-06 15:22:10 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |   describe('error recovery', () => { | 
					
						
							|  |  |  |     function recover(text: string, expected?: string) { | 
					
						
							|  |  |  |       const expr = validate(parseAction(text)); | 
					
						
							|  |  |  |       expect(unparse(expr)).toEqual(expected || text); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     it('should be able to recover from an extra paren', () => recover('((a)))', 'a')); | 
					
						
							|  |  |  |     it('should be able to recover from an extra bracket', () => recover('[[a]]]', '[[a]]')); | 
					
						
							|  |  |  |     it('should be able to recover from a missing )', () => recover('(a;b', 'a; b;')); | 
					
						
							|  |  |  |     it('should be able to recover from a missing ]', () => recover('[a,b', '[a, b]')); | 
					
						
							|  |  |  |     it('should be able to recover from a missing selector', () => recover('a.')); | 
					
						
							|  |  |  |     it('should be able to recover from a missing selector in a array literal', | 
					
						
							|  |  |  |        () => recover('[[a.], b, c]')); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   describe('offsets', () => { | 
					
						
							|  |  |  |     it('should retain the offsets of an interpolation', () => { | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |       const interpolations = splitInterpolation('{{a}}  {{b}}  {{c}}')!; | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |       expect(interpolations.offsets).toEqual([2, 9, 16]); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     it('should retain the offsets into the expression AST of interpolations', () => { | 
					
						
							| 
									
										
										
										
											2020-04-08 10:14:18 -07:00
										 |  |  |       const source = parseInterpolation('{{a}}  {{b}}  {{c}}')!; | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |       const interpolation = source.ast as Interpolation; | 
					
						
							|  |  |  |       expect(interpolation.expressions.map(e => e.span.start)).toEqual([2, 9, 16]); | 
					
						
							| 
									
										
										
										
											2016-10-06 15:22:10 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2014-10-28 12:22:38 -04:00
										 |  |  |   }); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  | }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function createParser() { | 
					
						
							|  |  |  |   return new Parser(new Lexer()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-04 09:21:40 +02:00
										 |  |  | function createIvyParser() { | 
					
						
							|  |  |  |   return new IvyParser(new Lexer()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-16 12:18:32 -07:00
										 |  |  | function parseAction(text: string, location: any = null, offset: number = 0): ASTWithSource { | 
					
						
							|  |  |  |   return createParser().parseAction(text, location, offset); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-16 12:18:32 -07:00
										 |  |  | function parseBinding(text: string, location: any = null, offset: number = 0): ASTWithSource { | 
					
						
							|  |  |  |   return createParser().parseBinding(text, location, offset); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  | function parseTemplateBindings(attribute: string, templateUrl = 'foo.html'): TemplateBinding[] { | 
					
						
							| 
									
										
										
										
											2019-05-07 17:50:44 +02:00
										 |  |  |   const result = _parseTemplateBindings(attribute, templateUrl); | 
					
						
							|  |  |  |   expect(result.errors).toEqual([]); | 
					
						
							|  |  |  |   expect(result.warnings).toEqual([]); | 
					
						
							|  |  |  |   return result.templateBindings; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function expectParseTemplateBindingsError(attribute: string, error: string) { | 
					
						
							|  |  |  |   const result = _parseTemplateBindings(attribute, 'foo.html'); | 
					
						
							|  |  |  |   expect(result.errors[0].message).toEqual(error); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function _parseTemplateBindings(attribute: string, templateUrl: string) { | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |   const match = attribute.match(/^\*(.+)="(.*)"$/); | 
					
						
							|  |  |  |   expect(match).toBeTruthy(`failed to extract key and value from ${attribute}`); | 
					
						
							| 
									
										
										
										
											2020-11-04 19:58:29 -05:00
										 |  |  |   const [_, key, value] = match!; | 
					
						
							| 
									
										
										
										
											2020-03-05 15:38:25 -08:00
										 |  |  |   const absKeyOffset = 1;  // skip the * prefix
 | 
					
						
							|  |  |  |   const absValueOffset = attribute.indexOf('=') + '="'.length; | 
					
						
							|  |  |  |   const parser = createParser(); | 
					
						
							| 
									
										
										
										
											2019-05-07 17:50:44 +02:00
										 |  |  |   return parser.parseTemplateBindings(key, value, templateUrl, absKeyOffset, absValueOffset); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-16 12:18:32 -07:00
										 |  |  | function parseInterpolation(text: string, location: any = null, offset: number = 0): ASTWithSource| | 
					
						
							|  |  |  |     null { | 
					
						
							|  |  |  |   return createParser().parseInterpolation(text, location, offset); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function splitInterpolation(text: string, location: any = null): SplitInterpolation|null { | 
					
						
							|  |  |  |   return createParser().splitInterpolation(text, location); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-16 12:18:32 -07:00
										 |  |  | function parseSimpleBinding(text: string, location: any = null, offset: number = 0): ASTWithSource { | 
					
						
							|  |  |  |   return createParser().parseSimpleBinding(text, location, offset); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-04 09:21:40 +02:00
										 |  |  | function parseSimpleBindingIvy( | 
					
						
							|  |  |  |     text: string, location: any = null, offset: number = 0): ASTWithSource { | 
					
						
							|  |  |  |   return createIvyParser().parseSimpleBinding(text, location, offset); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  | function checkInterpolation(exp: string, expected?: string) { | 
					
						
							| 
									
										
										
										
											2020-11-24 18:50:47 +01:00
										 |  |  |   const ast = parseInterpolation(exp); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |   if (expected == null) expected = exp; | 
					
						
							| 
									
										
										
										
											2020-11-24 18:50:47 +01:00
										 |  |  |   if (ast === null) { | 
					
						
							|  |  |  |     throw Error(`Failed to parse expression "${exp}"`); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-04-19 17:23:27 -07:00
										 |  |  |   expect(unparse(ast)).toEqual(expected); | 
					
						
							|  |  |  |   validate(ast); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function checkBinding(exp: string, expected?: string) { | 
					
						
							|  |  |  |   const ast = parseBinding(exp); | 
					
						
							|  |  |  |   if (expected == null) expected = exp; | 
					
						
							|  |  |  |   expect(unparse(ast)).toEqual(expected); | 
					
						
							|  |  |  |   validate(ast); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function checkAction(exp: string, expected?: string) { | 
					
						
							|  |  |  |   const ast = parseAction(exp); | 
					
						
							|  |  |  |   if (expected == null) expected = exp; | 
					
						
							|  |  |  |   expect(unparse(ast)).toEqual(expected); | 
					
						
							|  |  |  |   validate(ast); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function expectError(ast: {errors: ParserError[]}, message: string) { | 
					
						
							|  |  |  |   for (const error of ast.errors) { | 
					
						
							|  |  |  |     if (error.message.indexOf(message) >= 0) { | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   const errMsgs = ast.errors.map(err => err.message).join('\n'); | 
					
						
							|  |  |  |   throw Error( | 
					
						
							|  |  |  |       `Expected an error containing "${message}" to be reported, but got the errors:\n` + errMsgs); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function expectActionError(text: string, message: string) { | 
					
						
							|  |  |  |   expectError(validate(parseAction(text)), message); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function expectBindingError(text: string, message: string) { | 
					
						
							|  |  |  |   expectError(validate(parseBinding(text)), message); | 
					
						
							| 
									
										
										
										
											2019-07-16 12:18:32 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-09-25 18:52:06 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							| 
									
										
										
										
											2020-09-25 11:45:46 -05:00
										 |  |  |  * Check that a malformed action parses to a recovered AST while emitting an error. | 
					
						
							| 
									
										
										
										
											2020-09-25 18:52:06 -05:00
										 |  |  |  */ | 
					
						
							|  |  |  | function checkActionWithError(text: string, expected: string, error: string) { | 
					
						
							|  |  |  |   checkAction(text, expected); | 
					
						
							|  |  |  |   expectActionError(text, error); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-01-07 14:16:05 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | function rawSpan(span: AbsoluteSourceSpan): [number, number] { | 
					
						
							|  |  |  |   return [span.start, span.end]; | 
					
						
							|  |  |  | } |