Prior to this change, the unary + and - operators would be parsed as `x - 0`
and `0 - x` respectively. The runtime semantics of these expressions are
equivalent, however they may introduce inaccurate template type checking
errors as the literal type is lost, for example:
```ts
@Component({
  template: `<button [disabled]="isAdjacent(-1)"></button>`
})
export class Example {
  isAdjacent(direction: -1 | 1): boolean { return false; }
}
```
would incorrectly report a type-check error:
> error TS2345: Argument of type 'number' is not assignable to parameter
  of type '-1 | 1'.
Additionally, the translated expression for the unary + operator would be
considered as arithmetic expression with an incompatible left-hand side:
> error TS2362: The left-hand side of an arithmetic operation must be of
  type 'any', 'number', 'bigint' or an enum type.
To resolve this issues, the implicit transformation should be avoided.
This commit adds a new unary AST node to represent these expressions,
allowing for more accurate type-checking.
Fixes #20845
Fixes #36178
PR Close #37918
		
	
			
		
			
				
	
	
		
			128 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			128 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| /**
 | |
|  * @license
 | |
|  * Copyright Google LLC All Rights Reserved.
 | |
|  *
 | |
|  * Use of this source code is governed by an MIT-style license that can be
 | |
|  * found in the LICENSE file at https://angular.io/license
 | |
|  */
 | |
| 
 | |
| import {AST, Binary, BindingPipe, Chain, Conditional, FunctionCall, ImplicitReceiver, Interpolation, KeyedRead, KeyedWrite, LiteralArray, LiteralMap, LiteralPrimitive, MethodCall, ParseSpan, PrefixNot, PropertyRead, PropertyWrite, Quote, RecursiveAstVisitor, SafeMethodCall, SafePropertyRead, Unary} from '../../../src/expression_parser/ast';
 | |
| 
 | |
| import {unparse} from './unparser';
 | |
| 
 | |
| class ASTValidator extends RecursiveAstVisitor {
 | |
|   private parentSpan: ParseSpan|undefined;
 | |
| 
 | |
|   visit(ast: AST) {
 | |
|     this.parentSpan = undefined;
 | |
|     ast.visit(this);
 | |
|   }
 | |
| 
 | |
|   validate(ast: AST, cb: () => void): void {
 | |
|     if (!inSpan(ast.span, this.parentSpan)) {
 | |
|       if (this.parentSpan) {
 | |
|         const parentSpan = this.parentSpan as ParseSpan;
 | |
|         throw Error(`Invalid AST span [expected (${ast.span.start}, ${ast.span.end}) to be in (${
 | |
|             parentSpan.start},  ${parentSpan.end}) for ${unparse(ast)}`);
 | |
|       } else {
 | |
|         throw Error(`Invalid root AST span for ${unparse(ast)}`);
 | |
|       }
 | |
|     }
 | |
|     const oldParent = this.parentSpan;
 | |
|     this.parentSpan = ast.span;
 | |
|     cb();
 | |
|     this.parentSpan = oldParent;
 | |
|   }
 | |
| 
 | |
|   visitUnary(ast: Unary, context: any): any {
 | |
|     this.validate(ast, () => super.visitUnary(ast, context));
 | |
|   }
 | |
| 
 | |
|   visitBinary(ast: Binary, context: any): any {
 | |
|     this.validate(ast, () => super.visitBinary(ast, context));
 | |
|   }
 | |
| 
 | |
|   visitChain(ast: Chain, context: any): any {
 | |
|     this.validate(ast, () => super.visitChain(ast, context));
 | |
|   }
 | |
| 
 | |
|   visitConditional(ast: Conditional, context: any): any {
 | |
|     this.validate(ast, () => super.visitConditional(ast, context));
 | |
|   }
 | |
| 
 | |
|   visitFunctionCall(ast: FunctionCall, context: any): any {
 | |
|     this.validate(ast, () => super.visitFunctionCall(ast, context));
 | |
|   }
 | |
| 
 | |
|   visitImplicitReceiver(ast: ImplicitReceiver, context: any): any {
 | |
|     this.validate(ast, () => super.visitImplicitReceiver(ast, context));
 | |
|   }
 | |
| 
 | |
|   visitInterpolation(ast: Interpolation, context: any): any {
 | |
|     this.validate(ast, () => super.visitInterpolation(ast, context));
 | |
|   }
 | |
| 
 | |
|   visitKeyedRead(ast: KeyedRead, context: any): any {
 | |
|     this.validate(ast, () => super.visitKeyedRead(ast, context));
 | |
|   }
 | |
| 
 | |
|   visitKeyedWrite(ast: KeyedWrite, context: any): any {
 | |
|     this.validate(ast, () => super.visitKeyedWrite(ast, context));
 | |
|   }
 | |
| 
 | |
|   visitLiteralArray(ast: LiteralArray, context: any): any {
 | |
|     this.validate(ast, () => super.visitLiteralArray(ast, context));
 | |
|   }
 | |
| 
 | |
|   visitLiteralMap(ast: LiteralMap, context: any): any {
 | |
|     this.validate(ast, () => super.visitLiteralMap(ast, context));
 | |
|   }
 | |
| 
 | |
|   visitLiteralPrimitive(ast: LiteralPrimitive, context: any): any {
 | |
|     this.validate(ast, () => super.visitLiteralPrimitive(ast, context));
 | |
|   }
 | |
| 
 | |
|   visitMethodCall(ast: MethodCall, context: any): any {
 | |
|     this.validate(ast, () => super.visitMethodCall(ast, context));
 | |
|   }
 | |
| 
 | |
|   visitPipe(ast: BindingPipe, context: any): any {
 | |
|     this.validate(ast, () => super.visitPipe(ast, context));
 | |
|   }
 | |
| 
 | |
|   visitPrefixNot(ast: PrefixNot, context: any): any {
 | |
|     this.validate(ast, () => super.visitPrefixNot(ast, context));
 | |
|   }
 | |
| 
 | |
|   visitPropertyRead(ast: PropertyRead, context: any): any {
 | |
|     this.validate(ast, () => super.visitPropertyRead(ast, context));
 | |
|   }
 | |
| 
 | |
|   visitPropertyWrite(ast: PropertyWrite, context: any): any {
 | |
|     this.validate(ast, () => super.visitPropertyWrite(ast, context));
 | |
|   }
 | |
| 
 | |
|   visitQuote(ast: Quote, context: any): any {
 | |
|     this.validate(ast, () => super.visitQuote(ast, context));
 | |
|   }
 | |
| 
 | |
|   visitSafeMethodCall(ast: SafeMethodCall, context: any): any {
 | |
|     this.validate(ast, () => super.visitSafeMethodCall(ast, context));
 | |
|   }
 | |
| 
 | |
|   visitSafePropertyRead(ast: SafePropertyRead, context: any): any {
 | |
|     this.validate(ast, () => super.visitSafePropertyRead(ast, context));
 | |
|   }
 | |
| }
 | |
| 
 | |
| function inSpan(span: ParseSpan, parentSpan: ParseSpan|undefined): parentSpan is ParseSpan {
 | |
|   return !parentSpan || (span.start >= parentSpan.start && span.end <= parentSpan.end);
 | |
| }
 | |
| 
 | |
| const sharedValidator = new ASTValidator();
 | |
| 
 | |
| export function validate<T extends AST>(ast: T): T {
 | |
|   sharedValidator.visit(ast);
 | |
|   return ast;
 | |
| }
 |