Skip to content

Commit 38b017f

Browse files
committed
fix: allow string literals and template literals without expressions to match
1 parent 625bcb5 commit 38b017f

7 files changed

+101
-18
lines changed

src/compileMatcher/Placeholder.ts

+9-8
Original file line numberDiff line numberDiff line change
@@ -151,26 +151,27 @@ export default function compilePlaceholderMatcher(
151151
)
152152
}
153153

154+
function getString(node: Node): string | undefined {
155+
if (node.type === 'StringLiteral') return node.value
156+
if (node.type === 'TemplateLiteral' && node.quasis.length === 1)
157+
return node.quasis[0].value.cooked ?? node.quasis[0].value.raw
158+
return undefined
159+
}
160+
154161
export function compileStringPlaceholderMatcher<N extends Node>(
155162
pattern: NodePath<N>,
156-
getString: (node: N) => string | null,
157-
compileOptions: CompileOptions,
158-
otherOptions?: {
159-
nodeType?: NodeType | NodeType[]
160-
}
163+
compileOptions: CompileOptions
161164
): CompiledMatcher | void {
162165
const { debug } = compileOptions
163166
const string = getString(pattern.value)
164167
if (!string) return
165168
const placeholder = getPlaceholder(string)
166-
const nodeType = otherOptions?.nodeType
167169
if (placeholder) {
168170
return {
169171
pattern,
170172
placeholder,
171-
nodeType,
173+
nodeType: ['StringLiteral', 'TemplateLiteral'],
172174
match: (path: NodePath, matchSoFar: MatchResult): MatchResult => {
173-
if (path.value?.type !== pattern.value.type) return null
174175
debug('String Placeholder', placeholder)
175176
const string = getString((path as NodePath<N>).value)
176177
if (!string) return null

src/compileMatcher/StringLiteral.ts

+17-6
Original file line numberDiff line numberDiff line change
@@ -10,27 +10,38 @@ export default function matchStringLiteral(
1010
compileOptions: CompileOptions
1111
): CompiledMatcher {
1212
const pattern: StringLiteral = path.value
13-
const n = compileOptions.backend.t.namedTypes
1413
const placeholderMatcher = compileStringPlaceholderMatcher(
1514
path,
16-
(pattern) => pattern.value,
17-
compileOptions,
18-
{ nodeType: 'StringLiteral' }
15+
compileOptions
1916
)
2017

2118
if (placeholderMatcher) return placeholderMatcher
2219

2320
pattern.value = unescapeIdentifier(pattern.value)
2421

22+
return createStringLiteralMatcher(path, pattern.value, compileOptions)
23+
}
24+
25+
export function createStringLiteralMatcher(
26+
path: NodePath,
27+
value: string,
28+
compileOptions: CompileOptions
29+
): CompiledMatcher {
30+
const n = compileOptions.backend.t.namedTypes
2531
return convertPredicateMatcher(
2632
path,
2733
{
2834
match: (path: NodePath): boolean => {
2935
const { value: node } = path
3036

31-
return n.StringLiteral.check(node) && pattern.value === node.value
37+
return (
38+
(n.StringLiteral.check(node) && value === node.value) ||
39+
(n.TemplateLiteral.check(node) &&
40+
node.quasis.length === 1 &&
41+
value === (node.quasis[0].value.cooked ?? node.quasis[0].value.raw))
42+
)
3243
},
33-
nodeType: 'StringLiteral',
44+
nodeType: ['StringLiteral', 'TemplateLiteral'],
3445
},
3546
compileOptions
3647
)

src/compileMatcher/TemplateLiteral.ts

+7-4
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ import {
44
compileStringPlaceholderMatcher,
55
unescapeIdentifier,
66
} from './Placeholder'
7+
import { createStringLiteralMatcher } from './StringLiteral'
78

89
function generateValue(cooked: string): { raw: string; cooked: string } {
910
return { raw: cooked.replace(/\\|`|\${/g, '\\$&'), cooked }
@@ -17,10 +18,7 @@ export default function matchTemplateLiteral(
1718

1819
const placeholderMatcher = compileStringPlaceholderMatcher(
1920
path,
20-
(node: TemplateLiteral) =>
21-
node.quasis.length === 1 ? node.quasis[0].value.cooked ?? null : null,
22-
compileOptions,
23-
{ nodeType: 'TemplateLiteral' }
21+
compileOptions
2422
)
2523

2624
if (placeholderMatcher) return placeholderMatcher
@@ -33,5 +31,10 @@ export default function matchTemplateLiteral(
3331
quasi.value = generateValue(unescaped)
3432
}
3533
}
34+
return createStringLiteralMatcher(
35+
path,
36+
quasi.value.cooked ?? quasi.value.raw,
37+
compileOptions
38+
)
3639
}
3740
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
import { findReplaceTestcase } from '../findReplaceTestcase'
2+
import dedent from 'dedent-js'
3+
4+
findReplaceTestcase({
5+
file: __filename,
6+
input: dedent`
7+
'foo'
8+
\`foo\`
9+
'bar'
10+
`,
11+
find: dedent`
12+
\`foo\`
13+
`,
14+
expectedFind: [{ node: "'foo'" }, { node: '`foo`' }],
15+
})
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
import { findReplaceTestcase } from '../findReplaceTestcase'
2+
import dedent from 'dedent-js'
3+
4+
findReplaceTestcase({
5+
file: __filename,
6+
input: dedent`
7+
'foo'
8+
\`bar\`
9+
`,
10+
find: dedent`
11+
\`$s\`
12+
`,
13+
expectedFind: ["'foo'", '`bar`'].map((node) => ({
14+
node,
15+
captures: { $s: node },
16+
stringCaptures: { $s: node.substring(1, node.length - 1) },
17+
})),
18+
})
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
import { findReplaceTestcase } from '../findReplaceTestcase'
2+
import dedent from 'dedent-js'
3+
4+
findReplaceTestcase({
5+
file: __filename,
6+
input: dedent`
7+
'foo'
8+
\`foo\`
9+
\`foo\${a}\`
10+
\`bar\`
11+
`,
12+
find: dedent`
13+
'foo'
14+
`,
15+
expectedFind: [{ node: "'foo'" }, { node: '`foo`' }],
16+
})
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
import { findReplaceTestcase } from '../findReplaceTestcase'
2+
import dedent from 'dedent-js'
3+
4+
findReplaceTestcase({
5+
file: __filename,
6+
input: dedent`
7+
'foo'
8+
\`bar\`
9+
\`bar\${baz}\`
10+
`,
11+
find: dedent`
12+
'$s'
13+
`,
14+
expectedFind: ["'foo'", '`bar`'].map((node) => ({
15+
node,
16+
captures: { $s: node },
17+
stringCaptures: { $s: node.substring(1, node.length - 1) },
18+
})),
19+
})

0 commit comments

Comments
 (0)