Skip to content

Commit 2dfb7b5

Browse files
fix: include base in rerun commands (#1301)
## PR Checklist - [x] Addresses an existing open issue: fixes #1276 - [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 Attaches `base` the options returned during augmentation, so they can get printed out later. I noticed that the `exclusionKeys` stuff was exported from the `augmentOptionsWithExcludes.ts` file but never used in its tests. So I extracted it to a new `exclusionKeys.ts`. Adds some unit tests, but doesn't do all of #699. There's still coverage missing. Co-authored-by: @niklas-wortmann
1 parent 8101b51 commit 2dfb7b5

5 files changed

+220
-149
lines changed

src/create/createRerunSuggestion.test.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
import { describe, expect, it } from "vitest";
22

3-
import { getExclusions } from "../shared/options/augmentOptionsWithExcludes.js";
3+
import { getExclusions } from "../shared/options/exclusionKeys.js";
44
import { Options } from "../shared/types.js";
55
import { createRerunSuggestion } from "./createRerunSuggestion.js";
66

src/create/createRerunSuggestion.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@ import { allArgOptions } from "../shared/options/args.js";
22
import {
33
ExclusionKey,
44
getExclusions,
5-
} from "../shared/options/augmentOptionsWithExcludes.js";
5+
} from "../shared/options/exclusionKeys.js";
66
import { Options } from "../shared/types.js";
77

88
function getFirstMatchingArg(key: string) {

src/shared/options/augmentOptionsWithExcludes.test.ts

+62-3
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,25 @@
1-
import { describe, expect, it } from "vitest";
1+
import { describe, expect, it, vi } from "vitest";
22

33
import { Options } from "../types.js";
44
import { augmentOptionsWithExcludes } from "./augmentOptionsWithExcludes.js";
55

6+
const mockMultiselect = vi.fn();
7+
const mockSelect = vi.fn();
8+
9+
vi.mock("@clack/prompts", () => ({
10+
isCancel: () => false,
11+
get multiselect() {
12+
return mockMultiselect;
13+
},
14+
get select() {
15+
return mockSelect;
16+
},
17+
}));
18+
619
const optionsBase = {
720
access: "public",
821
author: undefined,
9-
base: "everything",
22+
base: undefined,
1023
description: "",
1124
directory: ".",
1225
email: {
@@ -47,7 +60,53 @@ const optionsBase = {
4760
} satisfies Options;
4861

4962
describe("augmentOptionsWithExcludes", () => {
50-
it("returns options without exclusions and skips prompting when exclusions are provided manually", async () => {
63+
it("returns options directly when no exclusions are provided and 'base' is provided for the prompt", async () => {
64+
const base = "everything";
65+
66+
mockSelect.mockResolvedValueOnce(base);
67+
68+
const actual = await augmentOptionsWithExcludes(optionsBase);
69+
70+
expect(actual).toEqual({
71+
...optionsBase,
72+
base,
73+
});
74+
});
75+
76+
it("returns options based on the select when no exclusions are provided and 'prompt' is provided for the prompt", async () => {
77+
const base = "prompt";
78+
79+
mockSelect.mockResolvedValueOnce(base);
80+
mockMultiselect.mockResolvedValue([]);
81+
82+
const actual = await augmentOptionsWithExcludes(optionsBase);
83+
84+
expect(actual).toEqual({
85+
...optionsBase,
86+
base,
87+
excludeAllContributors: true,
88+
excludeCompliance: true,
89+
excludeLintDeprecation: true,
90+
excludeLintESLint: true,
91+
excludeLintJSDoc: true,
92+
excludeLintJson: true,
93+
excludeLintKnip: true,
94+
excludeLintMd: true,
95+
excludeLintPackageJson: true,
96+
excludeLintPackages: true,
97+
excludeLintPerfectionist: true,
98+
excludeLintRegex: true,
99+
excludeLintSpelling: true,
100+
excludeLintStrict: true,
101+
excludeLintStylistic: true,
102+
excludeLintYml: true,
103+
excludeReleases: true,
104+
excludeRenovate: true,
105+
excludeTests: true,
106+
});
107+
});
108+
109+
it("skips prompting returns options directly when exclusions are provided manually", async () => {
51110
const options = {
52111
...optionsBase,
53112
excludeCompliance: true,

src/shared/options/augmentOptionsWithExcludes.ts

+8-144
Original file line numberDiff line numberDiff line change
@@ -3,150 +3,12 @@ import chalk from "chalk";
33

44
import { filterPromptCancel } from "../prompts.js";
55
import { Options, OptionsBase } from "../types.js";
6-
7-
interface ExclusionDescription {
8-
hint: string;
9-
label: string;
10-
uncommon?: true;
11-
}
12-
13-
export type ExclusionKey = keyof Options & `exclude${string}`;
14-
15-
const exclusionDescriptions: Record<ExclusionKey, ExclusionDescription> = {
16-
excludeAllContributors: {
17-
hint: "--exclude-all-contributors",
18-
label:
19-
"Add all-contributors to track contributions and display them in a README.md table.",
20-
},
21-
excludeCompliance: {
22-
hint: "--exclude-compliance",
23-
label:
24-
"Add a GitHub Actions workflow to verify that PRs match an expected format.",
25-
uncommon: true,
26-
},
27-
excludeLintDeprecation: {
28-
hint: "--exclude-lint-deprecation",
29-
label:
30-
"Include an eslint-plugin-deprecation to reports on usage of code marked as @deprecated.",
31-
uncommon: true,
32-
},
33-
excludeLintESLint: {
34-
hint: "--exclude-lint-eslint",
35-
label:
36-
"Include eslint-plugin-eslint-comment to enforce good practices around ESLint comment directives.",
37-
uncommon: true,
38-
},
39-
excludeLintJSDoc: {
40-
hint: "--exclude-lint-jsdoc",
41-
label:
42-
"Include eslint-plugin-jsdoc to enforce good practices around JSDoc comments.",
43-
uncommon: true,
44-
},
45-
excludeLintJson: {
46-
hint: "--exclude-lint-json",
47-
label: "Apply linting and sorting to *.json and *.jsonc files.",
48-
uncommon: true,
49-
},
50-
excludeLintKnip: {
51-
hint: "--exclude-lint-knip",
52-
label: "Add Knip to detect unused files, dependencies, and code exports.",
53-
},
54-
excludeLintMd: {
55-
hint: "--exclude-lint-md",
56-
label: "Apply linting to *.md files.",
57-
uncommon: true,
58-
},
59-
excludeLintPackageJson: {
60-
hint: "--exclude-lint-package-json",
61-
label:
62-
"Add eslint-plugin-package-json to lint for package.json correctness.",
63-
uncommon: true,
64-
},
65-
excludeLintPackages: {
66-
hint: "--exclude-lint-packages",
67-
label:
68-
"Add a pnpm dedupe workflow to ensure packages aren't duplicated unnecessarily.",
69-
uncommon: true,
70-
},
71-
excludeLintPerfectionist: {
72-
hint: "--exclude-lint-perfectionist",
73-
label:
74-
"Apply eslint-plugin-perfectionist to ensure imports, keys, and so on are in sorted order.",
75-
uncommon: true,
76-
},
77-
excludeLintRegex: {
78-
hint: "--exclude-lint-regex",
79-
label:
80-
"Include eslint-plugin-regex to enforce good practices around regular expressions.",
81-
uncommon: true,
82-
},
83-
excludeLintSpelling: {
84-
hint: "--exclude-lint-spelling",
85-
label: "Add cspell to spell check against dictionaries of known words.",
86-
uncommon: true,
87-
},
88-
excludeLintStrict: {
89-
hint: "--exclude-lint-strict",
90-
label:
91-
"Include strict logical lint rules such as typescript-eslint's strict config. ",
92-
uncommon: true,
93-
},
94-
excludeLintStylistic: {
95-
hint: "--exclude-lint-stylistic",
96-
label:
97-
"Include stylistic lint rules such as typescript-eslint's stylistic config.",
98-
uncommon: true,
99-
},
100-
excludeLintYml: {
101-
hint: "--exclude-lint-yml",
102-
label: "Apply linting and sorting to *.yaml and *.yml files.",
103-
uncommon: true,
104-
},
105-
excludeReleases: {
106-
hint: "--exclude-releases",
107-
label:
108-
"Add release-it to generate changelogs, package bumps, and publishes based on conventional commits.",
109-
},
110-
excludeRenovate: {
111-
hint: "--exclude-renovate",
112-
label: "Add a Renovate config to keep dependencies up-to-date with PRs.",
113-
},
114-
excludeTests: {
115-
hint: "--exclude-tests",
116-
label:
117-
"Add Vitest tooling for fast unit tests, configured with coverage tracking.",
118-
},
119-
};
120-
121-
const exclusionKeys = Object.keys(exclusionDescriptions) as ExclusionKey[];
122-
123-
export function getExclusions(
124-
options: Partial<Options>,
125-
base?: OptionsBase,
126-
): Partial<Options> {
127-
switch (base) {
128-
case "common":
129-
return {
130-
...Object.fromEntries(
131-
exclusionKeys
132-
.filter((exclusion) => exclusionDescriptions[exclusion].uncommon)
133-
.map((exclusion) => [exclusion, options[exclusion] ?? true]),
134-
),
135-
};
136-
case "minimum":
137-
return {
138-
...Object.fromEntries(
139-
exclusionKeys.map((exclusion) => [
140-
exclusion,
141-
options[exclusion] ?? true,
142-
]),
143-
),
144-
};
145-
// We only really care about exclusions on the common and minimum bases
146-
default:
147-
return {};
148-
}
149-
}
6+
import {
7+
ExclusionKey,
8+
exclusionDescriptions,
9+
exclusionKeys,
10+
getExclusions,
11+
} from "./exclusionKeys.js";
15012

15113
export async function augmentOptionsWithExcludes(
15214
options: Options,
@@ -206,6 +68,7 @@ export async function augmentOptionsWithExcludes(
20668
case "everything":
20769
return {
20870
...options,
71+
base,
20972
...getExclusions(options, base),
21073
};
21174
case "prompt":
@@ -228,6 +91,7 @@ export async function augmentOptionsWithExcludes(
22891

22992
return {
23093
...options,
94+
base,
23195
...Object.fromEntries(
23296
exclusionKeys.map(
23397
(exclusionKey) =>

0 commit comments

Comments
 (0)