Skip to content

Commit eb38a04

Browse files
feat: infer Knip Addons from disk (#2086)
## PR Checklist - [x] Addresses an existing open issue: fixes #2085 - [x] That issue was marked as [`status: accepting prs`](https://github.com/JoshuaKGoldberg/create-typescript-app/issues?q=is%3Aopen+is%3Aissue+label%3A%22status%3A+accepting+prs%22) - [x] Steps in [CONTRIBUTING.md](https://github.com/JoshuaKGoldberg/create-typescript-app/blob/main/.github/CONTRIBUTING.md) were taken ## Overview 🎁
1 parent 1c94da2 commit eb38a04

File tree

4 files changed

+99
-3
lines changed

4 files changed

+99
-3
lines changed

src/blocks/blockKnip.test.ts

+34-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
1-
import { testBlock } from "bingo-stratum-testers";
2-
import { describe, expect, test, vi } from "vitest";
1+
import { testBlock, testIntake } from "bingo-stratum-testers";
2+
import { describe, expect, it, test, vi } from "vitest";
33

44
import { blockKnip } from "./blockKnip.js";
55
import { optionsBase } from "./options.fakes.js";
@@ -206,4 +206,36 @@ describe("blockKnip", () => {
206206
}
207207
`);
208208
});
209+
210+
describe("intake", () => {
211+
it("returns undefined when knip.json does not exist", () => {
212+
const actual = testIntake(blockKnip, {
213+
files: {},
214+
});
215+
216+
expect(actual).toBeUndefined();
217+
});
218+
219+
it("returns undefined when knip.json does not contain ignoreDependencies", () => {
220+
const actual = testIntake(blockKnip, {
221+
files: {
222+
"knip.json": [JSON.stringify({ other: true })],
223+
},
224+
});
225+
226+
expect(actual).toBeUndefined();
227+
});
228+
229+
it("returns ignoreDependencies when knip.json contains ignoreDependencies", () => {
230+
const ignoreDependencies = ["a", "b", "c"];
231+
232+
const actual = testIntake(blockKnip, {
233+
files: {
234+
"knip.json": [JSON.stringify({ ignoreDependencies })],
235+
},
236+
});
237+
238+
expect(actual).toEqual({ ignoreDependencies });
239+
});
240+
});
209241
});

src/blocks/blockKnip.ts

+16-1
Original file line numberDiff line numberDiff line change
@@ -10,13 +10,28 @@ import { blockGitHubActionsCI } from "./blockGitHubActionsCI.js";
1010
import { blockPackageJson } from "./blockPackageJson.js";
1111
import { blockRemoveFiles } from "./blockRemoveFiles.js";
1212
import { blockRemoveWorkflows } from "./blockRemoveWorkflows.js";
13+
import { intakeFileAsJson } from "./intake/intakeFileAsJson.js";
14+
15+
const zIgnoreDependencies = z.array(z.string());
1316

1417
export const blockKnip = base.createBlock({
1518
about: {
1619
name: "Knip",
1720
},
1821
addons: {
19-
ignoreDependencies: z.array(z.string()).optional(),
22+
ignoreDependencies: zIgnoreDependencies.optional(),
23+
},
24+
intake({ files }) {
25+
const knipJson = intakeFileAsJson(files, ["knip.json"]);
26+
if (!knipJson?.ignoreDependencies) {
27+
return undefined;
28+
}
29+
30+
return {
31+
ignoreDependencies: zIgnoreDependencies.safeParse(
32+
knipJson.ignoreDependencies,
33+
).data,
34+
};
2035
},
2136
produce({ addons }) {
2237
const { ignoreDependencies } = addons;
+35
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
import { describe, expect, it } from "vitest";
2+
3+
import { intakeFileAsJson } from "./intakeFileAsJson.js";
4+
5+
describe(intakeFileAsJson, () => {
6+
it("returns undefined when the file does not exist", () => {
7+
const actual = intakeFileAsJson({}, []);
8+
9+
expect(actual).toBeUndefined();
10+
});
11+
12+
it("returns undefined when the file does not have valid JSON", () => {
13+
const actual = intakeFileAsJson(
14+
{
15+
"file.json": ["{"],
16+
},
17+
["file.json"],
18+
);
19+
20+
expect(actual).toBeUndefined();
21+
});
22+
23+
it("returns loaded file contents when the file has valid JSON", () => {
24+
const value = { key: "value" };
25+
26+
const actual = intakeFileAsJson(
27+
{
28+
"file.json": [JSON.stringify(value)],
29+
},
30+
["file.json"],
31+
);
32+
33+
expect(actual).toEqual(value);
34+
});
35+
});

src/blocks/intake/intakeFileAsJson.ts

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
import { IntakeDirectory } from "bingo-fs";
2+
import JSON5 from "json5";
3+
4+
import { intakeFile } from "./intakeFile.js";
5+
6+
export function intakeFileAsJson(files: IntakeDirectory, filePath: string[]) {
7+
const file = intakeFile(files, filePath);
8+
9+
try {
10+
return file && JSON5.parse<Record<string, unknown> | undefined>(file[0]);
11+
} catch {
12+
return undefined;
13+
}
14+
}

0 commit comments

Comments
 (0)