Skip to content

Commit 20afaf4

Browse files
feat: infer base from package json scripts during migration (#993)
<!-- 👋 Hi, thanks for sending a PR to create-typescript-app! 💖. Please fill out all fields below and make sure each item is true and [x] checked. Otherwise we may not be able to review your PR. --> ## PR Checklist - [x] Addresses an existing open issue: fixes #933 - [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 Add a naive solution for checking the existing `package.json` during migration to infer the base. 🐸 --------- Co-authored-by: Josh Goldberg <[email protected]>
1 parent c7217b7 commit 20afaf4

File tree

5 files changed

+140
-0
lines changed

5 files changed

+140
-0
lines changed

src/shared/options/getBase.test.ts

+57
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,57 @@
1+
import { describe, expect, it, vi } from "vitest";
2+
3+
import { getBase } from "./getBase.js";
4+
5+
const mockReadPackageData = vi.fn();
6+
vi.mock("../packages.js", () => ({
7+
get readPackageData() {
8+
return mockReadPackageData;
9+
},
10+
}));
11+
12+
describe("getBase", () => {
13+
it("should return minimum with minimum scripts", async () => {
14+
mockReadPackageData.mockImplementationOnce(() =>
15+
Promise.resolve({
16+
scripts: {
17+
build: "build",
18+
lint: "lint",
19+
test: "test",
20+
},
21+
}),
22+
);
23+
24+
expect(await getBase()).toBe("minimum");
25+
});
26+
it("should return common with common scripts", async () => {
27+
mockReadPackageData.mockImplementationOnce(() =>
28+
Promise.resolve({
29+
scripts: {
30+
build: "build",
31+
lint: "lint",
32+
"lint:knip": "knip",
33+
test: "test",
34+
},
35+
}),
36+
);
37+
38+
expect(await getBase()).toBe("common");
39+
});
40+
it("should return everything with everything scripts", async () => {
41+
mockReadPackageData.mockImplementationOnce(() =>
42+
Promise.resolve({
43+
scripts: {
44+
build: "build",
45+
lint: "lint",
46+
"lint:knip": "knip",
47+
"lint:md": "md",
48+
"lint:package-json": "package-json",
49+
"lint:packages": "packages",
50+
test: "test",
51+
},
52+
}),
53+
);
54+
55+
expect(await getBase()).toBe("everything");
56+
});
57+
});

src/shared/options/getBase.ts

+34
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
import { readPackageData } from "../packages.js";
2+
import { OptionsBase } from "../types.js";
3+
4+
const commonScripts = new Set(["lint:knip", "should-semantic-release", "test"]);
5+
6+
const everythingScripts = new Set([
7+
"lint:md",
8+
"lint:package-json",
9+
"lint:packages",
10+
"lint:spelling",
11+
]);
12+
export async function getBase(): Promise<OptionsBase> {
13+
const scripts = Object.keys((await readPackageData()).scripts ?? {});
14+
15+
if (
16+
scripts.reduce(
17+
(acc, curr) => (everythingScripts.has(curr) ? acc + 1 : acc),
18+
0,
19+
) >= 3
20+
) {
21+
return "everything";
22+
}
23+
24+
if (
25+
scripts.reduce(
26+
(acc, curr) => (commonScripts.has(curr) ? acc + 1 : acc),
27+
0,
28+
) >= 2
29+
) {
30+
return "common";
31+
}
32+
33+
return "minimum";
34+
}

src/shared/options/readOptions.test.ts

+43
Original file line numberDiff line numberDiff line change
@@ -111,6 +111,13 @@ vi.mock("./createOptionDefaults/index.js", () => ({
111111
},
112112
}));
113113

114+
const mockReadPackageData = vi.fn();
115+
vi.mock("../packages.js", () => ({
116+
get readPackageData() {
117+
return mockReadPackageData;
118+
},
119+
}));
120+
114121
describe("readOptions", () => {
115122
it("returns a cancellation when an arg is invalid", async () => {
116123
const validationResult = z
@@ -524,4 +531,40 @@ describe("readOptions", () => {
524531
},
525532
});
526533
});
534+
535+
it("infers base from package scripts during migration", async () => {
536+
mockReadPackageData.mockImplementationOnce(() =>
537+
Promise.resolve({
538+
scripts: {
539+
build: "build",
540+
lint: "lint",
541+
test: "test",
542+
},
543+
}),
544+
);
545+
expect(await readOptions(["--offline"], "migrate")).toStrictEqual({
546+
cancelled: false,
547+
github: mockOptions.github,
548+
options: {
549+
...emptyOptions,
550+
...mockOptions,
551+
access: "public",
552+
base: "minimum",
553+
description: "mock",
554+
directory: "mock",
555+
email: {
556+
github: "mock",
557+
npm: "mock",
558+
},
559+
guide: undefined,
560+
logo: undefined,
561+
mode: "migrate",
562+
offline: true,
563+
owner: "mock",
564+
skipAllContributorsApi: true,
565+
skipGitHubApi: true,
566+
title: "mock",
567+
},
568+
});
569+
});
527570
});

src/shared/options/readOptions.ts

+5
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ import { augmentOptionsWithExcludes } from "./augmentOptionsWithExcludes.js";
1010
import { createOptionDefaults } from "./createOptionDefaults/index.js";
1111
import { detectEmailRedundancy } from "./detectEmailRedundancy.js";
1212
import { ensureRepositoryExists } from "./ensureRepositoryExists.js";
13+
import { getBase } from "./getBase.js";
1314
import { GitHub, getGitHub } from "./getGitHub.js";
1415
import { getPrefillOrPromptedOption } from "./getPrefillOrPromptedOption.js";
1516
import { optionsSchema } from "./optionsSchema.js";
@@ -44,6 +45,10 @@ export async function readOptions(
4445
tokens: true,
4546
});
4647

48+
if (mode === "migrate" && !values.base) {
49+
values.base = await getBase();
50+
}
51+
4752
const mappedOptions = {
4853
access: values.access,
4954
author: values.author,

src/shared/types.ts

+1
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@ export interface PartialPackageData {
1919
email?: string;
2020
name?: string;
2121
repository?: { type: string; url: string } | string;
22+
scripts?: Record<string, string>;
2223
}
2324

2425
export type OptionsAccess = "public" | "restricted";

0 commit comments

Comments
 (0)