From b047e02e6c3b60a250545b2ff6071009a7a5b942 Mon Sep 17 00:00:00 2001 From: Jordan Pittman Date: Tue, 11 Jun 2024 10:44:09 -0400 Subject: [PATCH] Improve detection of string concatenation (#288) * Improve `visit()` types * Check for ancestor concat expressions * Rename global usage of `createRequire` * Update changelog * Update src/index.ts Co-authored-by: Robin Malfait * Update src/index.ts Co-authored-by: Robin Malfait * Update src/index.ts Co-authored-by: Robin Malfait --------- Co-authored-by: Robin Malfait --- CHANGELOG.md | 1 + build.mjs | 4 +- src/index.ts | 273 +++++++++++++++++++++++-------------------- src/utils.ts | 80 ++++++++++--- tests/format.test.js | 5 + 5 files changed, 219 insertions(+), 144 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 1338e975..2a0327b4 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -15,6 +15,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Only remove duplicate Tailwind classes ([#277](https://github.com/tailwindlabs/prettier-plugin-tailwindcss/pull/277)) - Make sure escapes in classes are preserved in string literals ([#286](https://github.com/tailwindlabs/prettier-plugin-tailwindcss/pull/286)) +- Improve detection of string concatenation ([#288](https://github.com/tailwindlabs/prettier-plugin-tailwindcss/pull/288)) ## [0.6.1] - 2024-05-31 diff --git a/build.mjs b/build.mjs index 7adfa8b1..705f5953 100644 --- a/build.mjs +++ b/build.mjs @@ -44,12 +44,12 @@ function patchCjsInterop() { // Prepend `createRequire` let code = [ - `import {createRequire} from 'module'`, + `import {createRequire as __global__createRequire__} from 'module'`, `import {dirname as __global__dirname__} from 'path'`, `import {fileURLToPath} from 'url'`, // CJS interop fixes - `const require=createRequire(import.meta.url)`, + `const require=__global__createRequire__(import.meta.url)`, `const __filename=fileURLToPath(import.meta.url)`, `const __dirname=__global__dirname__(__filename)`, ] diff --git a/src/index.ts b/src/index.ts index 8813cb81..05a9d1d9 100644 --- a/src/index.ts +++ b/src/index.ts @@ -117,11 +117,16 @@ function transformDynamicAngularAttribute(attr: any, env: TransformerEnv) { let changes: StringChange[] = [] visit(directiveAst, { - StringLiteral(node: any, parent: any, key: string | null) { + StringLiteral(node, path) { if (!node.value) return - let isConcat = - parent.type === 'BinaryExpression' && parent.operator === '+' + let concat = path.find((entry) => { + return ( + entry.parent && + entry.parent.type === 'BinaryExpression' && + entry.parent.operator === '+' + ) + }) changes.push({ start: node.start + 1, @@ -130,8 +135,8 @@ function transformDynamicAngularAttribute(attr: any, env: TransformerEnv) { after: sortClasses(node.value, { env, collapseWhitespace: { - start: !(isConcat && key === 'right'), - end: !(isConcat && key === 'left'), + start: concat?.key !== 'right', + end: concat?.key !== 'left', }, }), }) @@ -148,21 +153,37 @@ function transformDynamicJsAttribute(attr: any, env: TransformerEnv) { parser: prettierParserBabel.parsers['babel-ts'], }) + function* ancestors( + path: import('ast-types/lib/node-path').NodePath, + ) { + yield path + + while (path.parentPath) { + path = path.parentPath + yield path + } + } + let didChange = false astTypes.visit(ast, { visitLiteral(path) { - let isConcat = - path.parent.value.type === 'BinaryExpression' && - path.parent.value.operator === '+' - let key = path.name + let entries = Array.from(ancestors(path)) + let concat = entries.find((entry) => { + return ( + entry.parent && + entry.parent.value && + entry.parent.value.type === 'BinaryExpression' && + entry.parent.value.operator === '+' + ) + }) if (isStringLiteral(path.node)) { let sorted = sortStringLiteral(path.node, { env, collapseWhitespace: { - start: !(isConcat && key === 'right'), - end: !(isConcat && key === 'left'), + start: concat?.name !== 'right', + end: concat?.name !== 'left', }, }) @@ -183,15 +204,21 @@ function transformDynamicJsAttribute(attr: any, env: TransformerEnv) { }, visitTemplateLiteral(path) { - let isConcat = - path.parent.value.type === 'BinaryExpression' && - path.parent.value.operator === '+' - let key = path.name + let entries = Array.from(ancestors(path)) + let concat = entries.find((entry) => { + return ( + entry.parent && + entry.parent.value && + entry.parent.value.type === 'BinaryExpression' && + entry.parent.value.operator === '+' + ) + }) + let sorted = sortTemplateLiteral(path.node, { env, collapseWhitespace: { - start: !(isConcat && key === 'right'), - end: !(isConcat && key === 'left'), + start: concat?.name !== 'right', + end: concat?.name !== 'left', }, }) @@ -203,17 +230,22 @@ function transformDynamicJsAttribute(attr: any, env: TransformerEnv) { }, visitTaggedTemplateExpression(path) { - let isConcat = - path.parent.value.type === 'BinaryExpression' && - path.parent.value.operator === '+' - let key = path.name + let entries = Array.from(ancestors(path)) + let concat = entries.find((entry) => { + return ( + entry.parent && + entry.parent.value && + entry.parent.value.type === 'BinaryExpression' && + entry.parent.value.operator === '+' + ) + }) if (isSortableTemplateExpression(path.node, functions)) { let sorted = sortTemplateLiteral(path.node.quasi, { env, collapseWhitespace: { - start: !(isConcat && key === 'right'), - end: !(isConcat && key === 'left'), + start: concat?.name !== 'right', + end: concat?.name !== 'left', }, }) @@ -260,68 +292,56 @@ function transformGlimmer(ast: any, { env }: TransformerContext) { let { staticAttrs } = env.customizations visit(ast, { - AttrNode( - attr: any, - _parent: any, - _key: string | null, - _index: number, - meta: Record, - ) { + AttrNode(attr, _path, meta) { if (staticAttrs.has(attr.name) && attr.value) { meta.sortTextNodes = true } }, - TextNode( - node: any, - parent: any, - _key: string | null, - index: number, - meta: Record, - ) { + TextNode(node, path, meta) { if (!meta.sortTextNodes) { return } - let siblings = - parent?.type === 'ConcatStatement' - ? { - prev: parent.parts[index - 1], - next: parent.parts[index + 1], - } - : null + let concat = path.find((entry) => { + return entry.parent && entry.parent.type === 'ConcatStatement' + }) + + let siblings = { + prev: concat?.parent.parts[concat.index! - 1], + next: concat?.parent.parts[concat.index! + 1], + } node.chars = sortClasses(node.chars, { env, - ignoreFirst: siblings?.prev && !/^\s/.test(node.chars), - ignoreLast: siblings?.next && !/\s$/.test(node.chars), + ignoreFirst: siblings.prev && !/^\s/.test(node.chars), + ignoreLast: siblings.next && !/\s$/.test(node.chars), collapseWhitespace: { - start: !siblings?.prev, - end: !siblings?.next, + start: !siblings.prev, + end: !siblings.next, }, }) }, - StringLiteral( - node: any, - parent: any, - _key: string | null, - _index: number, - meta: Record, - ) { + StringLiteral(node, path, meta) { if (!meta.sortTextNodes) { return } - const isConcat = - parent.type === 'SubExpression' && parent.path.original === 'concat' + let concat = path.find((entry) => { + return ( + entry.parent && + entry.parent.type === 'SubExpression' && + entry.parent.path.original === 'concat' + ) + }) node.value = sortClasses(node.value, { env, - ignoreLast: isConcat && !/[^\S\r\n]$/.test(node.value), + ignoreLast: Boolean(concat) && !/[^\S\r\n]$/.test(node.value), collapseWhitespace: { start: false, - end: !isConcat, + end: !concat, }, }) }, @@ -597,49 +617,49 @@ function transformJavaScript( let { staticAttrs, functions } = env.customizations function sortInside(ast: import('@babel/types').Node) { - visit( - ast, - ( - node: import('@babel/types').Node, - parent: import('@babel/types').Node | null, - key: string, - ) => { - let isConcat = - parent?.type === 'BinaryExpression' && parent?.operator === '+' - - if (isStringLiteral(node)) { - sortStringLiteral(node, { - env, - collapseWhitespace: { - start: !(isConcat && key === 'right'), - end: !(isConcat && key === 'left'), - }, - }) - } else if (node.type === 'TemplateLiteral') { - sortTemplateLiteral(node, { + visit(ast, (node, path) => { + let concat = path.find((entry) => { + return ( + entry.parent && + entry.parent.type === 'BinaryExpression' && + entry.parent.operator === '+' + ) + }) + + if (isStringLiteral(node)) { + sortStringLiteral(node, { + env, + collapseWhitespace: { + start: concat?.key !== 'right', + end: concat?.key !== 'left', + }, + }) + } else if (node.type === 'TemplateLiteral') { + sortTemplateLiteral(node, { + env, + collapseWhitespace: { + start: concat?.key !== 'right', + end: concat?.key !== 'left', + }, + }) + } else if (node.type === 'TaggedTemplateExpression') { + if (isSortableTemplateExpression(node, functions)) { + sortTemplateLiteral(node.quasi, { env, collapseWhitespace: { - start: !(isConcat && key === 'right'), - end: !(isConcat && key === 'left'), + start: concat?.key !== 'right', + end: concat?.key !== 'left', }, }) - } else if (node.type === 'TaggedTemplateExpression') { - if (isSortableTemplateExpression(node, functions)) { - sortTemplateLiteral(node.quasi, { - env, - collapseWhitespace: { - start: !(isConcat && key === 'right'), - end: !(isConcat && key === 'left'), - }, - }) - } } - }, - ) + } + }) } visit(ast, { - JSXAttribute(node: import('@babel/types').JSXAttribute) { + JSXAttribute(node) { + node = node as import('@babel/types').JSXAttribute + if (!node.value) { return } @@ -661,7 +681,9 @@ function transformJavaScript( } }, - CallExpression(node: import('@babel/types').CallExpression) { + CallExpression(node) { + node = node as import('@babel/types').CallExpression + if (!isSortableCallExpression(node, functions)) { return } @@ -669,23 +691,26 @@ function transformJavaScript( node.arguments.forEach((arg) => sortInside(arg)) }, - TaggedTemplateExpression( - node: import('@babel/types').TaggedTemplateExpression, - parent: import('@babel/types').Expression | null, - key: string | null, - ) { + TaggedTemplateExpression(node, path) { + node = node as import('@babel/types').TaggedTemplateExpression + if (!isSortableTemplateExpression(node, functions)) { return } - let isConcat = - parent?.type === 'BinaryExpression' && parent?.operator === '+' + let concat = path.find((entry) => { + return ( + entry.parent && + entry.parent.type === 'BinaryExpression' && + entry.parent.operator === '+' + ) + }) sortTemplateLiteral(node.quasi, { env, collapseWhitespace: { - start: !(isConcat && key === 'right'), - end: !(isConcat && key === 'left'), + start: concat?.key !== 'right', + end: concat?.key !== 'left', }, }) }, @@ -794,42 +819,32 @@ function transformMelody(ast: any, { env, changes }: TransformerContext) { } visit(ast, { - Attribute( - node: any, - _parent: any, - _key: string | null, - _index: number, - meta: Record, - ) { + Attribute(node, _path, meta) { if (!staticAttrs.has(node.name.name)) return meta.sortTextNodes = true }, - StringLiteral( - node: any, - parent: any, - _key: string | null, - _index: number, - meta: Record, - ) { + StringLiteral(node, path, meta) { if (!meta.sortTextNodes) { return } - const isConcat = - parent.type === 'BinaryConcatExpression' || - parent.type === 'BinaryAddExpression' + const concat = path.find((entry) => { + return ( + entry.parent && + (entry.parent.type === 'BinaryConcatExpression' || + entry.parent.type === 'BinaryAddExpression') + ) + }) node.value = sortClasses(node.value, { env, - ignoreFirst: - isConcat && _key === 'right' && !/^[^\S\r\n]/.test(node.value), - ignoreLast: - isConcat && _key === 'left' && !/[^\S\r\n]$/.test(node.value), + ignoreFirst: concat?.key === 'right' && !/^[^\S\r\n]/.test(node.value), + ignoreLast: concat?.key === 'left' && !/[^\S\r\n]$/.test(node.value), collapseWhitespace: { - start: !(isConcat && _key === 'right'), - end: !(isConcat && _key === 'left'), + start: concat?.key !== 'right', + end: concat?.key !== 'left', }, }) }, @@ -926,7 +941,7 @@ function transformSvelte(ast: any, { env, changes }: TransformerContext) { }) } else if (value.type === 'MustacheTag') { visit(value.expression, { - Literal(node: any, _parent: any, _key: string | null) { + Literal(node) { if (isStringLiteral(node)) { let before = node.raw let sorted = sortStringLiteral(node, { @@ -945,7 +960,7 @@ function transformSvelte(ast: any, { env, changes }: TransformerContext) { } } }, - TemplateLiteral(node: any, _parent: any, _key: string | null) { + TemplateLiteral(node) { let before = node.quasis.map((quasi: any) => quasi.value.raw) let sorted = sortTemplateLiteral(node, { env, diff --git a/src/utils.ts b/src/utils.ts index 5230a2dd..78e5bd7b 100644 --- a/src/utils.ts +++ b/src/utils.ts @@ -11,21 +11,39 @@ export function loadIfExists(name: string): any { } } +interface PathEntry { + node: T + parent: T | null + key: string | null + index: number | null + meta: Meta +} + +type Path = PathEntry[] + +type Visitor> = ( + node: T, + path: Path, + meta: Partial, +) => void | false + +type Visitors> = Record< + string, + Visitor +> + // https://lihautan.com/manipulating-ast-with-javascript/ -export function visit(ast: any, callbackMap: any) { - function _visit( - node: any, - parent?: any, - key?: any, - index?: any, - meta: Record = {}, - ) { +export function visit>( + ast: T, + callbackMap: Visitors | Visitor, +) { + function _visit(node: any, path: Path, meta: Meta) { if (typeof callbackMap === 'function') { - if (callbackMap(node, parent, key, index, meta) === false) { + if (callbackMap(node, path, meta) === false) { return } } else if (node.type in callbackMap) { - if (callbackMap[node.type](node, parent, key, index, meta) === false) { + if (callbackMap[node.type](node, path, meta) === false) { return } } @@ -36,15 +54,51 @@ export function visit(ast: any, callbackMap: any) { if (Array.isArray(child)) { for (let j = 0; j < child.length; j++) { if (child[j] !== null) { - _visit(child[j], node, keys[i], j, { ...meta }) + let newMeta = { ...meta } + let newPath = [ + { + node: child[j], + parent: node, + key: keys[i], + index: j, + meta: newMeta, + }, + ...path, + ] + + _visit(child[j], newPath, newMeta) } } } else if (typeof child?.type === 'string') { - _visit(child, node, keys[i], i, { ...meta }) + let newMeta = { ...meta } + let newPath = [ + { + node: child, + parent: node, + key: keys[i], + index: i, + meta: newMeta, + }, + ...path, + ] + + _visit(child, newPath, newMeta) } } } - _visit(ast) + + let newMeta: Meta = {} as any + let newPath: Path = [ + { + node: ast, + parent: null, + key: null, + index: null, + meta: newMeta, + }, + ] + + _visit(ast, newPath, newMeta) } /** diff --git a/tests/format.test.js b/tests/format.test.js index 41af0bf1..76c41e28 100644 --- a/tests/format.test.js +++ b/tests/format.test.js @@ -103,6 +103,11 @@ let javascript = [ `;
`, `;
`, ], + + [ + `;
`, + `;
`, + ], ] javascript = javascript.concat( javascript.map((test) => [