Skip to content

Commit 001882b

Browse files
feat: smarter initial mode prompt (#919)
## PR Checklist - [x] Addresses an existing open issue: fixes #884 - [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 Enhances `promptForMode` to give different options based on the current directory: * If it's empty, offer to `create` a new repository in it or a child directory * If it's a Git directory, offer to `initialize` or `migrate` * If it's not a Git directory, runs `create` for a new repository in a child directory In doing so, adds an optional `--directory` that defaults to the repository's name. Also cleans up `getPrefillOrPromptedOption` a bit. Instead of allowing an `existingValue` parameter, calls to `getPrefillOrPromptedOption` are just put in the right-hand-side of a `??`.
1 parent 1a69169 commit 001882b

38 files changed

+500
-211
lines changed

docs/Options.md

+1
Original file line numberDiff line numberDiff line change
@@ -54,6 +54,7 @@ The setup scripts also allow for optional overrides of the following inputs whos
5454

5555
- `--access` _(`"public" | "restricted"`)_: Which [`npm publish --access`](https://docs.npmjs.com/cli/commands/npm-publish#access) to release npm packages with (by default, `"public"`)
5656
- `--author` _(`string`)_: Username on npm to publish packages under (by default, an existing npm author, or the currently logged in npm user, or `owner.toLowerCase()`)
57+
- `--directory` _(`string`)_: Directory to create the repository in (by default, the same name as the repository)
5758
- `--email` _(`string`)_: Email address to be listed as the point of contact in docs and packages (e.g. `[email protected]`)
5859
- Optionally, `--email-github` _(`string`)_ and/or `--email-npm` _(`string`)_ may be provided to use different emails in `.md` files and `package.json`, respectively
5960
- `--funding` _(`string`)_: GitHub organization or username to mention in `funding.yml` (by default, `owner`)

src/bin/index.test.ts

+18-15
Original file line numberDiff line numberDiff line change
@@ -54,7 +54,7 @@ vi.mock("../migrate/index.js", () => ({
5454

5555
const mockPromptForMode = vi.fn();
5656

57-
vi.mock("./mode.js", () => ({
57+
vi.mock("./promptForMode.js", () => ({
5858
get promptForMode() {
5959
return mockPromptForMode;
6060
},
@@ -65,8 +65,8 @@ describe("bin", () => {
6565
vi.spyOn(console, "clear").mockImplementation(() => undefined);
6666
});
6767

68-
it("returns 1 when promptForMode returns undefined", async () => {
69-
mockPromptForMode.mockResolvedValue(undefined);
68+
it("returns 1 when promptForMode returns an undefined mode", async () => {
69+
mockPromptForMode.mockResolvedValue({});
7070

7171
const result = await bin([]);
7272

@@ -76,9 +76,9 @@ describe("bin", () => {
7676
expect(result).toBe(1);
7777
});
7878

79-
it("returns 1 when promptForMode returns an error", async () => {
79+
it("returns 1 when promptForMode returns an error mode", async () => {
8080
const error = new Error("Oh no!");
81-
mockPromptForMode.mockResolvedValue(error);
81+
mockPromptForMode.mockResolvedValue({ mode: error });
8282

8383
const result = await bin([]);
8484

@@ -90,13 +90,14 @@ describe("bin", () => {
9090
const mode = "create";
9191
const args = ["--owner", "abc123"];
9292
const code = 0;
93+
const promptedOptions = { directory: "." };
9394

94-
mockPromptForMode.mockResolvedValue(mode);
95+
mockPromptForMode.mockResolvedValue({ mode, options: promptedOptions });
9596
mockCreate.mockResolvedValue({ code, options: {} });
9697

9798
const result = await bin(args);
9899

99-
expect(mockCreate).toHaveBeenCalledWith(args);
100+
expect(mockCreate).toHaveBeenCalledWith(args, promptedOptions);
100101
expect(mockCancel).not.toHaveBeenCalled();
101102
expect(mockInitialize).not.toHaveBeenCalled();
102103
expect(mockMigrate).not.toHaveBeenCalled();
@@ -107,13 +108,14 @@ describe("bin", () => {
107108
const mode = "create";
108109
const args = ["--owner", "abc123"];
109110
const code = 2;
111+
const promptedOptions = { directory: "." };
110112

111-
mockPromptForMode.mockResolvedValue(mode);
113+
mockPromptForMode.mockResolvedValue({ mode, options: promptedOptions });
112114
mockCreate.mockResolvedValue({ code, options: {} });
113115

114116
const result = await bin(args);
115117

116-
expect(mockCreate).toHaveBeenCalledWith(args);
118+
expect(mockCreate).toHaveBeenCalledWith(args, promptedOptions);
117119
expect(mockCancel).toHaveBeenCalledWith(
118120
`Operation cancelled. Exiting - maybe another time? 👋`,
119121
);
@@ -126,7 +128,7 @@ describe("bin", () => {
126128
const code = 2;
127129
const error = "Oh no!";
128130

129-
mockPromptForMode.mockResolvedValue(mode);
131+
mockPromptForMode.mockResolvedValue({ mode });
130132
mockInitialize.mockResolvedValue({
131133
code: 2,
132134
error,
@@ -135,7 +137,7 @@ describe("bin", () => {
135137

136138
const result = await bin(args);
137139

138-
expect(mockInitialize).toHaveBeenCalledWith(args);
140+
expect(mockInitialize).toHaveBeenCalledWith(args, undefined);
139141
expect(mockLogLine).toHaveBeenCalledWith(chalk.red(error));
140142
expect(mockCancel).toHaveBeenCalledWith(
141143
`Operation cancelled. Exiting - maybe another time? 👋`,
@@ -152,7 +154,7 @@ describe("bin", () => {
152154
.object({ email: z.string().email() })
153155
.safeParse({ email: "abc123" });
154156

155-
mockPromptForMode.mockResolvedValue(mode);
157+
mockPromptForMode.mockResolvedValue({ mode });
156158
mockInitialize.mockResolvedValue({
157159
code: 2,
158160
error: (validationResult as z.SafeParseError<{ email: string }>).error,
@@ -161,7 +163,7 @@ describe("bin", () => {
161163

162164
const result = await bin(args);
163165

164-
expect(mockInitialize).toHaveBeenCalledWith(args);
166+
expect(mockInitialize).toHaveBeenCalledWith(args, undefined);
165167
expect(mockLogLine).toHaveBeenCalledWith(
166168
chalk.red('Validation error: Invalid email at "email"'),
167169
);
@@ -175,13 +177,14 @@ describe("bin", () => {
175177
const mode = "create";
176178
const args = ["--owner", "abc123"];
177179
const code = 1;
180+
const promptedOptions = { directory: "." };
178181

179-
mockPromptForMode.mockResolvedValue(mode);
182+
mockPromptForMode.mockResolvedValue({ mode, options: promptedOptions });
180183
mockCreate.mockResolvedValue({ code, options: {} });
181184

182185
const result = await bin(args);
183186

184-
expect(mockCreate).toHaveBeenCalledWith(args);
187+
expect(mockCreate).toHaveBeenCalledWith(args, promptedOptions);
185188
expect(mockCancel).toHaveBeenCalledWith(
186189
`Operation failed. Exiting - maybe another time? 👋`,
187190
);

src/bin/index.ts

+3-3
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ import { initialize } from "../initialize/index.js";
99
import { migrate } from "../migrate/index.js";
1010
import { logLine } from "../shared/cli/lines.js";
1111
import { StatusCodes } from "../shared/codes.js";
12-
import { promptForMode } from "./mode.js";
12+
import { promptForMode } from "./promptForMode.js";
1313

1414
const operationMessage = (verb: string) =>
1515
`Operation ${verb}. Exiting - maybe another time? 👋`;
@@ -45,14 +45,14 @@ export async function bin(args: string[]) {
4545
strict: false,
4646
});
4747

48-
const mode = await promptForMode(values.mode);
48+
const { mode, options: promptedOptions } = await promptForMode(values.mode);
4949
if (typeof mode !== "string") {
5050
prompts.outro(chalk.red(mode?.message ?? operationMessage("cancelled")));
5151
return 1;
5252
}
5353

5454
const runners = { create, initialize, migrate };
55-
const { code, error, options } = await runners[mode](args);
55+
const { code, error, options } = await runners[mode](args, promptedOptions);
5656

5757
prompts.log.info(
5858
[

src/bin/mode.test.ts

-39
This file was deleted.

src/bin/mode.ts

-67
This file was deleted.

src/bin/promptForMode.test.ts

+117
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,117 @@
1+
import chalk from "chalk";
2+
import { describe, expect, it, vi } from "vitest";
3+
4+
import { promptForMode } from "./promptForMode.js";
5+
6+
const mockSelect = vi.fn();
7+
8+
vi.mock("@clack/prompts", () => ({
9+
isCancel: () => false,
10+
get select() {
11+
return mockSelect;
12+
},
13+
}));
14+
15+
const mockReaddir = vi.fn();
16+
17+
vi.mock("node:fs/promises", () => ({
18+
get readdir() {
19+
return mockReaddir;
20+
},
21+
}));
22+
23+
const mockCwd = vi.fn();
24+
25+
vi.mock("node:process", () => ({
26+
get cwd() {
27+
return mockCwd;
28+
},
29+
}));
30+
31+
const mockLogLine = vi.fn();
32+
33+
vi.mock("../shared/cli/lines.js", () => ({
34+
get logLine() {
35+
return mockLogLine;
36+
},
37+
}));
38+
describe("promptForMode", () => {
39+
it("returns an error when the input exists and is not a mode", async () => {
40+
const mode = await promptForMode("other");
41+
42+
expect(mode).toMatchInlineSnapshot(
43+
`
44+
{
45+
"mode": [Error: Unknown --mode: other. Allowed modes are: create, initialize, migrate.],
46+
}
47+
`,
48+
);
49+
});
50+
51+
it("returns the input when it is a mode", async () => {
52+
const input = "create";
53+
54+
const mode = await promptForMode(input);
55+
56+
expect(mode).toEqual({ mode: input });
57+
});
58+
59+
it("returns creating in the current directory when the current directory is empty and the user selects create-current", async () => {
60+
mockSelect.mockResolvedValueOnce("create-current");
61+
const directory = "test-directory";
62+
63+
mockReaddir.mockResolvedValueOnce([]);
64+
mockCwd.mockReturnValueOnce(`/path/to/${directory}`);
65+
66+
const actual = await promptForMode(undefined);
67+
68+
expect(actual).toEqual({
69+
mode: "create",
70+
options: { directory: ".", repository: directory },
71+
});
72+
expect(mockLogLine).not.toHaveBeenCalled();
73+
});
74+
75+
it("returns creating in a child directory when the current directory is empty and the user selects create-child", async () => {
76+
mockSelect.mockResolvedValueOnce("create-child");
77+
const directory = "test-directory";
78+
79+
mockReaddir.mockResolvedValueOnce([]);
80+
mockCwd.mockReturnValueOnce(`/path/to/${directory}`);
81+
82+
const actual = await promptForMode(undefined);
83+
84+
expect(actual).toEqual({
85+
mode: "create",
86+
});
87+
expect(mockLogLine).not.toHaveBeenCalled();
88+
});
89+
90+
it("returns the user selection when the current directory is a Git directory", async () => {
91+
const mode = "initialize";
92+
mockSelect.mockResolvedValueOnce(mode);
93+
94+
mockReaddir.mockResolvedValueOnce([".git"]);
95+
96+
const actual = await promptForMode(undefined);
97+
98+
expect(actual).toEqual({ mode });
99+
expect(mockLogLine).not.toHaveBeenCalled();
100+
});
101+
102+
it("returns create without prompting when the current directory contains children but is not a Git directory", async () => {
103+
const mode = "create";
104+
105+
mockReaddir.mockResolvedValueOnce(["file"]);
106+
107+
const actual = await promptForMode(undefined);
108+
109+
expect(actual).toEqual({ mode });
110+
expect(mockSelect).not.toHaveBeenCalled();
111+
expect(mockLogLine).toHaveBeenCalledWith(
112+
chalk.gray(
113+
"Defaulting to --mode create because the directory contains children and isn't a Git repository.",
114+
),
115+
);
116+
});
117+
});

0 commit comments

Comments
 (0)