From ed02f2c1b296c7c8991c217f6f63cd14a952542c Mon Sep 17 00:00:00 2001 From: Tanguy <13471753+Menduist@users.noreply.github.com> Date: Thu, 26 Sep 2024 15:00:49 +0200 Subject: [PATCH] fix: expressions scopes and member expressions --- source/components/JsxParser.test.tsx | 28 +++++++++ source/components/JsxParser.tsx | 90 +++++++++++++--------------- source/types/acorn-jsx.d.ts | 5 +- 3 files changed, 71 insertions(+), 52 deletions(-) diff --git a/source/components/JsxParser.test.tsx b/source/components/JsxParser.test.tsx index 14f3458..5402d13 100644 --- a/source/components/JsxParser.test.tsx +++ b/source/components/JsxParser.test.tsx @@ -958,6 +958,24 @@ describe('JsxParser Component', () => { expect(node.childNodes[0].textContent).toEqual(bindings.array[bindings.index].of) expect(instance.ParsedChildren[0].props.foo).toEqual(bindings.array[bindings.index].of) }) + it('can evaluate a[b]', () => { + const { node } = render( + , + ) + expect(node.innerHTML).toMatch('hello') + }) + it('handles optional chaining', () => { + const { node } = render( + , + ) + expect(node.innerHTML).toMatch('baz') + }) /* eslint-enable dot-notation,no-useless-concat */ }) }) @@ -1264,5 +1282,15 @@ describe('JsxParser Component', () => { ) expect(node.outerHTML).toEqual('

from-container

') }) + + it('supports math with scope', () => { + const { node } = render() + expect(node.innerHTML).toEqual('246') + }) + + it('supports conditional with scope', () => { + const { node } = render() + expect(node.innerHTML).toEqual('1-13') + }) }) }) diff --git a/source/components/JsxParser.tsx b/source/components/JsxParser.tsx index 1efe906..9930fc5 100644 --- a/source/components/JsxParser.tsx +++ b/source/components/JsxParser.tsx @@ -94,27 +94,29 @@ export default class JsxParser extends React.Component { case 'ArrayExpression': return expression.elements.map(ele => this.#parseExpression(ele, scope)) as ParsedTree case 'BinaryExpression': + const binaryLeft = this.#parseExpression(expression.left, scope) + const binaryRight = this.#parseExpression(expression.right, scope) /* eslint-disable eqeqeq,max-len */ switch (expression.operator) { - case '-': return this.#parseExpression(expression.left) - this.#parseExpression(expression.right) - case '!=': return this.#parseExpression(expression.left) != this.#parseExpression(expression.right) - case '!==': return this.#parseExpression(expression.left) !== this.#parseExpression(expression.right) - case '*': return this.#parseExpression(expression.left) * this.#parseExpression(expression.right) - case '**': return this.#parseExpression(expression.left) ** this.#parseExpression(expression.right) - case '/': return this.#parseExpression(expression.left) / this.#parseExpression(expression.right) - case '%': return this.#parseExpression(expression.left) % this.#parseExpression(expression.right) - case '+': return this.#parseExpression(expression.left) + this.#parseExpression(expression.right) - case '<': return this.#parseExpression(expression.left) < this.#parseExpression(expression.right) - case '<=': return this.#parseExpression(expression.left) <= this.#parseExpression(expression.right) - case '==': return this.#parseExpression(expression.left) == this.#parseExpression(expression.right) - case '===': return this.#parseExpression(expression.left) === this.#parseExpression(expression.right) - case '>': return this.#parseExpression(expression.left) > this.#parseExpression(expression.right) - case '>=': return this.#parseExpression(expression.left) >= this.#parseExpression(expression.right) + case '-': return binaryLeft - binaryRight + case '!=': return binaryLeft != binaryRight + case '!==': return binaryLeft !== binaryRight + case '*': return binaryLeft * binaryRight + case '**': return binaryLeft ** binaryRight + case '/': return binaryLeft / binaryRight + case '%': return binaryLeft % binaryRight + case '+': return binaryLeft + binaryRight + case '<': return binaryLeft < binaryRight + case '<=': return binaryLeft <= binaryRight + case '==': return binaryLeft == binaryRight + case '===': return binaryLeft === binaryRight + case '>': return binaryLeft > binaryRight + case '>=': return binaryLeft >= binaryRight /* eslint-enable eqeqeq,max-len */ } return undefined case 'CallExpression': - const parsedCallee = this.#parseExpression(expression.callee) + const parsedCallee = this.#parseExpression(expression.callee, scope) if (parsedCallee === undefined) { this.props.onError!(new Error(`The expression '${expression.callee}' could not be resolved, resulting in an undefined return value.`)) return undefined @@ -123,11 +125,11 @@ export default class JsxParser extends React.Component { arg => this.#parseExpression(arg, expression.callee), )) case 'ConditionalExpression': - return this.#parseExpression(expression.test) - ? this.#parseExpression(expression.consequent) - : this.#parseExpression(expression.alternate) + return this.#parseExpression(expression.test, scope) + ? this.#parseExpression(expression.consequent, scope) + : this.#parseExpression(expression.alternate, scope) case 'ExpressionStatement': - return this.#parseExpression(expression.expression) + return this.#parseExpression(expression.expression, scope) case 'Identifier': if (scope && expression.name in scope) { return scope[expression.name] @@ -137,10 +139,10 @@ export default class JsxParser extends React.Component { case 'Literal': return expression.value case 'LogicalExpression': - const left = this.#parseExpression(expression.left) + const left = this.#parseExpression(expression.left, scope) if (expression.operator === '||' && left) return left if ((expression.operator === '&&' && left) || (expression.operator === '||' && !left)) { - return this.#parseExpression(expression.right) + return this.#parseExpression(expression.right, scope) } return false case 'MemberExpression': @@ -148,7 +150,7 @@ export default class JsxParser extends React.Component { case 'ObjectExpression': const object: Record = {} expression.properties.forEach(prop => { - object[prop.key.name! || prop.key.value!] = this.#parseExpression(prop.value) + object[prop.key.name! || prop.key.value!] = this.#parseExpression(prop.value, scope) }) return object case 'TemplateElement': @@ -159,7 +161,7 @@ export default class JsxParser extends React.Component { if (a.start < b.start) return -1 return 1 }) - .map(item => this.#parseExpression(item)) + .map(item => this.#parseExpression(item, scope)) .join('') case 'UnaryExpression': switch (expression.operator) { @@ -183,37 +185,25 @@ export default class JsxParser extends React.Component { } #parseMemberExpression = (expression: AcornJSX.MemberExpression, scope?: Scope): any => { - // eslint-disable-next-line prefer-destructuring - let { object } = expression - const path = [expression.property?.name ?? JSON.parse(expression.property?.raw ?? '""')] - - if (expression.object.type !== 'Literal') { - while (object && ['MemberExpression', 'Literal'].includes(object?.type)) { - const { property } = (object as AcornJSX.MemberExpression) - if ((object as AcornJSX.MemberExpression).computed) { - path.unshift(this.#parseExpression(property!, scope)) - } else { - path.unshift(property?.name ?? JSON.parse(property?.raw ?? '""')) - } + const object = this.#parseExpression(expression.object, scope) - object = (object as AcornJSX.MemberExpression).object - } + let property + if (expression.computed) { + property = this.#parseExpression(expression.property, scope) + } else if (expression.property.type === 'Identifier') { + property = expression.property.name + } else { + this.props.onError!(new Error('Only simple MemberExpressions are supported.')) + return undefined } - const target = this.#parseExpression(object, scope) - try { - let parent = target - const member = path.reduce((value, next) => { - parent = value - return value[next] - }, target) - if (typeof member === 'function') return member.bind(parent) - - return member - } catch { - const name = (object as AcornJSX.MemberExpression)?.name || 'unknown' - this.props.onError!(new Error(`Unable to parse ${name}["${path.join('"]["')}"]}`)) + if (expression.optional) { + if (object === null || object === undefined) return undefined } + + const member = object[property] + if (typeof member === 'function') return member.bind(object) + return member } #parseName = (element: AcornJSX.JSXIdentifier | AcornJSX.JSXMemberExpression): string => { diff --git a/source/types/acorn-jsx.d.ts b/source/types/acorn-jsx.d.ts index 5af688c..2477f57 100644 --- a/source/types/acorn-jsx.d.ts +++ b/source/types/acorn-jsx.d.ts @@ -122,9 +122,10 @@ declare module 'acorn-jsx' { export interface MemberExpression extends BaseExpression { type: 'MemberExpression'; computed: boolean; + optional: boolean; name?: string; - object: Literal | MemberExpression; - property?: MemberExpression; + object: Expression; + property: Expression; raw?: string; }