From eb996e5fdb53a9a945f1e41d4d74229b53e01cb1 Mon Sep 17 00:00:00 2001 From: Alexander von Weiss Date: Tue, 28 Nov 2023 11:37:24 +0100 Subject: [PATCH] fix: `RangeError: Maximum call stack size exceeded` on nested templates (#34) Co-authored-by: Alexander von Weiss --- src/parsers/pipe.parser.ts | 90 ++++++++++++++++++++----------- tests/parsers/pipe.parser.spec.ts | 13 +++++ 2 files changed, 73 insertions(+), 30 deletions(-) diff --git a/src/parsers/pipe.parser.ts b/src/parsers/pipe.parser.ts index 7053abdb..54492879 100644 --- a/src/parsers/pipe.parser.ts +++ b/src/parsers/pipe.parser.ts @@ -11,8 +11,11 @@ import { LiteralArray, Interpolation, Call, - TmplAstIfBlockBranch, - TmplAstSwitchBlockCase + TmplAstIfBlock, + TmplAstSwitchBlock, + TmplAstDeferredBlock, + TmplAstForLoopBlock, + TmplAstElement } from '@angular/compiler'; import { ParserInterface } from './parser.interface.js'; @@ -21,6 +24,58 @@ import { isPathAngularComponent, extractComponentInlineTemplate } from '../utils export const TRANSLATE_PIPE_NAMES = ['translate', 'marker']; +function traverseAstNodes( + nodes: (NODE | null)[], + visitor: (node: NODE) => RESULT[], + accumulator: RESULT[] = [] +): RESULT[] { + for (const node of nodes) { + if (node) { + traverseAstNode(node, visitor, accumulator); + } + } + + return accumulator; +} + +function traverseAstNode( + node: NODE, + visitor: (node: NODE) => RESULT[], + accumulator: RESULT[] = [] +): RESULT[] { + accumulator.push(...visitor(node)); + + const children: TmplAstNode[] = []; + // children of templates, html elements or blocks + if ('children' in node && node.children) { + children.push(...node.children); + } + + // contents of @for extra sibling block @empty + if (node instanceof TmplAstForLoopBlock) { + children.push(node.empty); + } + + // contents of @defer extra sibling blocks @error, @placeholder and @loading + if (node instanceof TmplAstDeferredBlock) { + children.push(node.error); + children.push(node.loading); + children.push(node.placeholder); + } + + // contents of @if and @else (ignoring the @if(...) condition statement though) + if (node instanceof TmplAstIfBlock) { + children.push(...node.branches.flatMap((inner) => inner.children)); + } + + // contents of @case blocks (ignoring the @switch(...) statement though) + if (node instanceof TmplAstSwitchBlock) { + children.push(...node.cases.flatMap((inner) => inner.children)); + } + + return traverseAstNodes(children, visitor, accumulator); +} + export class PipeParser implements ParserInterface { public extract(source: string, filePath: string): TranslationCollection { if (filePath && isPathAngularComponent(filePath)) { @@ -29,7 +84,9 @@ export class PipeParser implements ParserInterface { let collection: TranslationCollection = new TranslationCollection(); const nodes: TmplAstNode[] = this.parseTemplate(source, filePath); - const pipes: BindingPipe[] = nodes.map((node) => this.findPipesInNode(node)).flat(); + + const pipes = traverseAstNodes(nodes, (node) => this.findPipesInNode(node)); + pipes.forEach((pipe) => { this.parseTranslationKeysFromPipe(pipe).forEach((key: string) => { collection = collection.add(key, '', filePath); @@ -41,29 +98,6 @@ export class PipeParser implements ParserInterface { protected findPipesInNode(node: any): BindingPipe[] { const ret: BindingPipe[] = []; - const nodeChildren = node?.children ?? []; - - // @if and @switch blocks - const nodeBranchesOrCases: TmplAstIfBlockBranch[] | TmplAstSwitchBlockCase[] = node?.branches ?? node?.cases ?? []; - - // @for blocks - const emptyBlockChildren = node?.empty?.children ?? []; - - // @deferred blocks - const errorBlockChildren = node?.error?.children ?? []; - const loadingBlockChildren = node?.loading?.children ?? []; - const placeholderBlockChildren = node?.placeholder?.children ?? []; - - nodeChildren.push(...emptyBlockChildren, ...errorBlockChildren, ...loadingBlockChildren, ...placeholderBlockChildren); - - if (nodeChildren.length > 0) { - ret.push(...this.extractPipesFromChildNodes(nodeChildren)); - } - - nodeBranchesOrCases.forEach((branch) => { - ret.push(...this.extractPipesFromChildNodes(branch.children)); - }); - if (node?.value?.ast) { ret.push(...this.getTranslatablesFromAst(node.value.ast)); } @@ -93,10 +127,6 @@ export class PipeParser implements ParserInterface { return ret; } - protected extractPipesFromChildNodes(nodeChildren: TmplAstNode[]) { - return nodeChildren.map((childNode) => this.findPipesInNode(childNode)).flat(); - } - protected parseTranslationKeysFromPipe(pipeContent: BindingPipe | LiteralPrimitive | Conditional): string[] { const ret: string[] = []; if (pipeContent instanceof LiteralPrimitive) { diff --git a/tests/parsers/pipe.parser.spec.ts b/tests/parsers/pipe.parser.spec.ts index 42417688..1c0fcee0 100644 --- a/tests/parsers/pipe.parser.spec.ts +++ b/tests/parsers/pipe.parser.spec.ts @@ -369,5 +369,18 @@ describe('PipeParser', () => { 'else.block' ]); }); + + it('should handle ast with arbitrary depth without hitting the call stack limit', () => { + const depth = 500; + const contents = ` + ${Array(depth).fill('').join('')} + {{ 'deep' | translate }} + ${Array(depth).fill('').join('')} + `; + + const keys = parser.extract(contents, templateFilename)?.keys(); + expect(contents).to.contain(''); + expect(keys).to.deep.equal(['deep']); + }); }); });