2016-07-21 20:12:00 -04:00
|
|
|
/**
|
|
|
|
* @license
|
2020-05-19 15:08:49 -04:00
|
|
|
* Copyright Google LLC All Rights Reserved.
|
2016-07-21 20:12:00 -04: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
|
|
|
|
*/
|
|
|
|
|
2020-07-03 19:52:40 -04:00
|
|
|
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';
|
2016-07-06 17:06:47 -04:00
|
|
|
|
|
|
|
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)) {
|
2016-08-30 21:07:40 -04:00
|
|
|
if (this.parentSpan) {
|
2016-11-12 08:08:58 -05:00
|
|
|
const parentSpan = this.parentSpan as ParseSpan;
|
2020-04-08 13:14:18 -04:00
|
|
|
throw Error(`Invalid AST span [expected (${ast.span.start}, ${ast.span.end}) to be in (${
|
|
|
|
parentSpan.start}, ${parentSpan.end}) for ${unparse(ast)}`);
|
2016-08-30 21:07:40 -04:00
|
|
|
} else {
|
|
|
|
throw Error(`Invalid root AST span for ${unparse(ast)}`);
|
|
|
|
}
|
2016-07-06 17:06:47 -04:00
|
|
|
}
|
|
|
|
const oldParent = this.parentSpan;
|
|
|
|
this.parentSpan = ast.span;
|
|
|
|
cb();
|
|
|
|
this.parentSpan = oldParent;
|
|
|
|
}
|
|
|
|
|
2020-07-03 19:52:40 -04:00
|
|
|
visitUnary(ast: Unary, context: any): any {
|
|
|
|
this.validate(ast, () => super.visitUnary(ast, context));
|
|
|
|
}
|
|
|
|
|
2016-07-06 17:06:47 -04:00
|
|
|
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));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-08 13:14:18 -04:00
|
|
|
function inSpan(span: ParseSpan, parentSpan: ParseSpan|undefined): parentSpan is ParseSpan {
|
2016-07-06 17:06:47 -04:00
|
|
|
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;
|
|
|
|
}
|