Skip to content

Commit 5f2cf34

Browse files
committedJan 15, 2025
Support more execa methods in visitor
1 parent 1731ee5 commit 5f2cf34

File tree

8 files changed

+56
-11
lines changed

8 files changed

+56
-11
lines changed
 

‎packages/knip/fixtures/script-visitors/execa/execa-docs.mjs

+1-1
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@ await $`cat package.json | grep name`;
55
let branch = await $`git branch --show-current`;
66
await $`dep deploy --branch=${branch}`;
77

8-
await Promise.all([$`sleep 1; echo 1`, $`sleep 2; echo 2`, $`sleep 3; echo 3`]);
8+
await Promise.all([$`executable1; echo 1`, $`executable2; echo 2`, $`executable3; echo 3`]);
99

1010
let name = 'foo bar';
1111
await $`mkdir /tmp/${name}`;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
import { execa, execaSync, execaCommand, execaCommandSync, execaNode, $sync } from 'execa';
2+
3+
$sync`pnpm dlx executable4`;
4+
5+
await execa('bun x', ['executable5']);
6+
7+
execaSync('npx', ['executable6']);
8+
9+
await execaCommand('bunx executable7');
10+
11+
execaCommandSync('pnpx executable8');

‎packages/knip/fixtures/script-visitors/execa/package.json

+12-2
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,8 @@
44
"execa1": "node ./execa-docs.mjs",
55
"execa2": "node ./script.js",
66
"execa3": "node ./node.mjs",
7-
"execa4": "node ./options.mjs"
7+
"execa4": "node ./options.mjs",
8+
"execa5": "node ./methods.mjs"
89
},
910
"dependencies": {
1011
"dep": "*"
@@ -16,6 +17,15 @@
1617
"execa": "*"
1718
},
1819
"knip": {
19-
"ignoreBinaries": ["sleep"]
20+
"ignoreBinaries": [
21+
"executable1",
22+
"executable2",
23+
"executable3",
24+
"executable4",
25+
"executable5",
26+
"executable6",
27+
"executable7",
28+
"executable8"
29+
]
2030
}
2131
}
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,9 @@
1-
import { $ } from 'execa';
1+
import { $, $sync } from 'execa';
22

33
/* global $ */
44
import { EOL } from 'node:os';
55
import { Octokit } from 'octokit';
66

77
await $`pnpm all-contributors generate`;
88

9-
await $`npx -y all-contributors-cli@6.25 add user`;
9+
await $sync`npx -y all-contributors-cli@6.25 add user`;

‎packages/knip/src/typescript/visitors/helpers.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -38,13 +38,13 @@ export function getImportsFromPragmas(sourceFile: BoundSourceFile) {
3838
return importNodes;
3939
}
4040

41-
export function hasImportSpecifier(node: ts.Statement, name: string): boolean {
41+
export function hasImportSpecifier(node: ts.Statement, name: string, id?: string): boolean {
4242
return (
4343
ts.isImportDeclaration(node) &&
4444
ts.isStringLiteral(node.moduleSpecifier) &&
4545
node.moduleSpecifier.text === name &&
4646
!!node.importClause?.namedBindings &&
4747
ts.isNamedImports(node.importClause.namedBindings) &&
48-
node.importClause.namedBindings.elements.some(element => element.name.text === '$')
48+
(!id || node.importClause.namedBindings.elements.some(element => element.name.text === id))
4949
);
5050
}

‎packages/knip/src/typescript/visitors/scripts/bun.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@ import { hasImportSpecifier } from '../helpers.js';
44
import { scriptVisitor as visit } from '../index.js';
55

66
export default visit(
7-
sourceFile => sourceFile.statements.some(node => hasImportSpecifier(node, 'bun')),
7+
sourceFile => sourceFile.statements.some(node => hasImportSpecifier(node, 'bun', '$')),
88
node => {
99
if (ts.isTaggedTemplateExpression(node) && node.tag.getText() === '$') {
1010
return stripQuotes(node.template.getText());

‎packages/knip/src/typescript/visitors/scripts/execa.ts

+25-1
Original file line numberDiff line numberDiff line change
@@ -3,13 +3,37 @@ import { stripQuotes } from '../../ast-helpers.js';
33
import { hasImportSpecifier } from '../helpers.js';
44
import { scriptVisitor as visit } from '../index.js';
55

6+
const tags = new Set(['$', '$sync']);
7+
const methods = new Set(['execa', 'execaSync', 'execaCommand', 'execaCommandSync', '$sync']);
8+
69
export default visit(
710
sourceFile => sourceFile.statements.some(node => hasImportSpecifier(node, 'execa')),
811
node => {
912
if (ts.isTaggedTemplateExpression(node)) {
10-
if (node.tag.getText() === '$' || (ts.isCallExpression(node.tag) && node.tag.expression.getText() === '$')) {
13+
if (tags.has(node.tag.getText()) || (ts.isCallExpression(node.tag) && tags.has(node.tag.expression.getText()))) {
1114
return stripQuotes(node.template.getText());
1215
}
1316
}
17+
18+
if (ts.isCallExpression(node)) {
19+
const functionName = node.expression.getText();
20+
if (methods.has(functionName)) {
21+
if (functionName.startsWith('execaCommand')) {
22+
if (node.arguments[0] && ts.isStringLiteral(node.arguments[0])) {
23+
return stripQuotes(node.arguments[0].getText());
24+
}
25+
} else {
26+
const [executable, args] = node.arguments;
27+
if (executable && ts.isStringLiteral(executable)) {
28+
const executableStr = stripQuotes(executable.getText());
29+
if (args && ts.isArrayLiteralExpression(args)) {
30+
const argStrings = args.elements.filter(ts.isStringLiteral).map(arg => stripQuotes(arg.getText()));
31+
return [executableStr, ...argStrings].join(' ');
32+
}
33+
return executableStr;
34+
}
35+
}
36+
}
37+
}
1438
}
1539
);

‎packages/knip/test/script-visitors-execa.test.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ test('Find dependencies with custom script visitors (execa)', async () => {
1818

1919
assert.deepEqual(counters, {
2020
...baseCounters,
21-
processed: 5,
22-
total: 5,
21+
processed: 6,
22+
total: 6,
2323
});
2424
});

0 commit comments

Comments
 (0)
Please sign in to comment.