From ce44a3d1dc89d396b6488ed9a983334a5ba0f6ca Mon Sep 17 00:00:00 2001 From: Pete Bacon Darwin Date: Mon, 8 Feb 2021 21:29:41 +0000 Subject: [PATCH] fix(localize): support downleveled inlined helper localize calls (#40754) When the downleveling helper function has been inlined into the `$localize` call, it is a bit more tricky to parse out the cooked and raw strings. There was already code to do this but it assumed that the `cooked` and `raw` items were both arrays. Sometimes the `raw` array is just a copy of the `cooked` array via an expression similar to `raw || (raw=tcookedslice(0))`. This commit changes the `unwrapMessagePartsFromLocalizeCall()` function to be able to handle such a situation. Fixes #40702 PR Close #40754 --- .../src/tools/src/source_file_utils.ts | 17 +++- .../src/tools/test/source_file_utils_spec.ts | 98 +++++++++++++++++++ 2 files changed, 110 insertions(+), 5 deletions(-) diff --git a/packages/localize/src/tools/src/source_file_utils.ts b/packages/localize/src/tools/src/source_file_utils.ts index e0814955e8..55cd10c5fa 100644 --- a/packages/localize/src/tools/src/source_file_utils.ts +++ b/packages/localize/src/tools/src/source_file_utils.ts @@ -89,7 +89,7 @@ export function unwrapMessagePartsFromLocalizeCall( // If there is no call to `__makeTemplateObject(...)`, then `raw` must be the same as `cooked`. let raw = cooked; - // Check for cached call of the form `x || x = __makeTemplateObject(...)` + // Check for a memoized form: `x || x = ...` if (cooked.isLogicalExpression() && cooked.node.operator === '||' && cooked.get('left').isIdentifier()) { const right = cooked.get('right'); @@ -105,15 +105,22 @@ export function unwrapMessagePartsFromLocalizeCall( // This is a minified sequence expression, where the first two expressions in the sequence // are assignments of the cooked and raw arrays respectively. const [first, second] = expressions; - if (first.isAssignmentExpression() && second.isAssignmentExpression()) { + if (first.isAssignmentExpression()) { cooked = first.get('right'); if (!cooked.isExpression()) { throw new BabelParseError( first.node, 'Unexpected cooked value, expected an expression.'); } - raw = second.get('right'); - if (!raw.isExpression()) { - throw new BabelParseError(second.node, 'Unexpected raw value, expected an expression.'); + if (second.isAssignmentExpression()) { + raw = second.get('right'); + if (!raw.isExpression()) { + throw new BabelParseError( + second.node, 'Unexpected raw value, expected an expression.'); + } + } else { + // If the second expression is not an assignment then it is probably code to take a copy + // of the cooked array. For example: `raw || (raw=cooked.slice(0))`. + raw = cooked; } } } diff --git a/packages/localize/src/tools/test/source_file_utils_spec.ts b/packages/localize/src/tools/test/source_file_utils_spec.ts index 1f8421fcd7..249c4beef8 100644 --- a/packages/localize/src/tools/test/source_file_utils_spec.ts +++ b/packages/localize/src/tools/test/source_file_utils_spec.ts @@ -132,6 +132,104 @@ runInEachFileSystem(() => { ]); }); + it('should return an array of string literals and locations from a (Babel helper) downleveled tagged template', + () => { + let localizeCall = getLocalizeCall( + `$localize(babelHelpers.taggedTemplateLiteral(['a', 'b\\t', 'c'], ['a', 'b\\\\t', 'c']), 1, 2)`); + const [parts, locations] = unwrapMessagePartsFromLocalizeCall(localizeCall, fs); + expect(parts).toEqual(['a', 'b\t', 'c']); + expect(parts.raw).toEqual(['a', 'b\\t', 'c']); + expect(locations).toEqual([ + { + start: {line: 0, column: 65}, + end: {line: 0, column: 68}, + file: absoluteFrom('/test/file.js'), + text: `'a'`, + }, + { + start: {line: 0, column: 70}, + end: {line: 0, column: 76}, + file: absoluteFrom('/test/file.js'), + text: `'b\\\\t'`, + }, + { + start: {line: 0, column: 78}, + end: {line: 0, column: 81}, + file: absoluteFrom('/test/file.js'), + text: `'c'`, + }, + ]); + }); + + it('should return an array of string literals and locations from a memoized downleveled tagged template', + () => { + let localizeCall = getLocalizeCall(` + var _templateObject; + $localize(_templateObject || (_templateObject = __makeTemplateObject(['a', 'b\\t', 'c'], ['a', 'b\\\\t', 'c'])), 1, 2)`); + const [parts, locations] = unwrapMessagePartsFromLocalizeCall(localizeCall, fs); + expect(parts).toEqual(['a', 'b\t', 'c']); + expect(parts.raw).toEqual(['a', 'b\\t', 'c']); + expect(locations).toEqual([ + { + start: {line: 2, column: 105}, + end: {line: 2, column: 108}, + file: absoluteFrom('/test/file.js'), + text: `'a'`, + }, + { + start: {line: 2, column: 110}, + end: {line: 2, column: 116}, + file: absoluteFrom('/test/file.js'), + text: `'b\\\\t'`, + }, + { + start: {line: 2, column: 118}, + end: {line: 2, column: 121}, + file: absoluteFrom('/test/file.js'), + text: `'c'`, + }, + ]); + }); + + it('should return an array of string literals and locations from a memoized (inlined Babel helper) downleveled tagged template', + () => { + let localizeCall = getLocalizeCall(` + var e,t,n; + $localize(e || + ( + t=["a","b\t","c"], + n || (n=t.slice(0)), + e = Object.freeze( + Object.defineProperties(t, { raw: { value: Object.freeze(n) } }) + ) + ), + 1,2 + )`); + const [parts, locations] = unwrapMessagePartsFromLocalizeCall(localizeCall, fs); + expect(parts).toEqual(['a', 'b\t', 'c']); + expect(parts.raw).toEqual(['a', 'b\t', 'c']); + expect(locations).toEqual([ + { + start: {line: 4, column: 21}, + end: {line: 4, column: 24}, + file: absoluteFrom('/test/file.js'), + text: `"a"`, + }, + { + start: {line: 4, column: 25}, + end: {line: 4, column: 29}, + file: absoluteFrom('/test/file.js'), + text: `"b\t"`, + }, + { + start: {line: 4, column: 30}, + end: {line: 4, column: 33}, + file: absoluteFrom('/test/file.js'), + text: `"c"`, + }, + ]); + }); + it('should return an array of string literals and locations from a lazy load template helper', () => { let localizeCall = getLocalizeCall(`