Skip to content

Commit 2793e4c

Browse files
feat: include URLs in GitHub app and secret suggestions (#1964)
## PR Checklist - [x] Addresses an existing open issue: fixes #1621 - [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 Refactors the two blocks to use a shared `getInstallationSuggestions` function because I didn't want to keep updating both. 🎁
1 parent 360f887 commit 2793e4c

6 files changed

+183
-40
lines changed

src/blocks/blockGitHubApps.test.ts

+47
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
import { testBlock } from "bingo-stratum-testers";
2+
import { describe, expect, test } from "vitest";
3+
4+
import { blockGitHubApps } from "./blockGitHubApps.js";
5+
import { optionsBase } from "./options.fakes.js";
6+
7+
describe("blockGitHubApps", () => {
8+
test("without addons", () => {
9+
const creation = testBlock(blockGitHubApps, {
10+
options: optionsBase,
11+
});
12+
13+
expect(creation).toMatchInlineSnapshot(`
14+
{
15+
"suggestions": undefined,
16+
}
17+
`);
18+
});
19+
20+
test("with addons", () => {
21+
const creation = testBlock(blockGitHubApps, {
22+
addons: {
23+
apps: [
24+
{
25+
name: "Secret A.",
26+
url: "https://example.com?a",
27+
},
28+
{
29+
name: "Secret B.",
30+
url: "https://example.com?b",
31+
},
32+
],
33+
},
34+
options: optionsBase,
35+
});
36+
37+
expect(creation).toMatchInlineSnapshot(`
38+
{
39+
"suggestions": [
40+
"- enable the GitHub apps on https://github.com/test-owner/test-repository/settings/installations:
41+
- Secret A. (https://example.com?a)
42+
- Secret B. (https://example.com?b)",
43+
],
44+
}
45+
`);
46+
});
47+
});

src/blocks/blockGitHubApps.ts

+15-20
Original file line numberDiff line numberDiff line change
@@ -1,34 +1,29 @@
11
import { z } from "zod";
22

33
import { base } from "../base.js";
4+
import { getInstallationSuggestions } from "./getInstallationSuggestions.js";
45

5-
const zApp = z.object({
6-
name: z.string(),
7-
url: z.string(),
8-
});
96
export const blockGitHubApps = base.createBlock({
107
about: {
118
name: "GitHub Apps",
129
},
1310
addons: {
14-
apps: z.array(zApp).default([]),
11+
apps: z
12+
.array(
13+
z.object({
14+
name: z.string(),
15+
url: z.string(),
16+
}),
17+
)
18+
.default([]),
1519
},
16-
produce({ addons }) {
20+
produce({ addons, options }) {
1721
return {
18-
suggestions: addons.apps.length
19-
? [
20-
[
21-
`- enable the GitHub app`,
22-
addons.apps.length === 1 ? "" : "s",
23-
`:\n`,
24-
addons.apps.map(printApp).join("\n"),
25-
].join(""),
26-
]
27-
: undefined,
22+
suggestions: getInstallationSuggestions(
23+
"enable the GitHub app",
24+
addons.apps.map((app) => `${app.name} (${app.url})`),
25+
`https://github.com/${options.owner}/${options.repository}/settings/installations`,
26+
),
2827
};
2928
},
3029
});
31-
32-
function printApp(app: z.infer<typeof zApp>) {
33-
return ` - ${app.name} (${app.url})`;
34-
}
+47
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
import { testBlock } from "bingo-stratum-testers";
2+
import { describe, expect, test } from "vitest";
3+
4+
import { blockRepositorySecrets } from "./blockRepositorySecrets.js";
5+
import { optionsBase } from "./options.fakes.js";
6+
7+
describe("blockRepositorySecrets", () => {
8+
test("without addons", () => {
9+
const creation = testBlock(blockRepositorySecrets, {
10+
options: optionsBase,
11+
});
12+
13+
expect(creation).toMatchInlineSnapshot(`
14+
{
15+
"suggestions": undefined,
16+
}
17+
`);
18+
});
19+
20+
test("with addons", () => {
21+
const creation = testBlock(blockRepositorySecrets, {
22+
addons: {
23+
secrets: [
24+
{
25+
description: "Secret description a.",
26+
name: "Secret Name A",
27+
},
28+
{
29+
description: "Secret description b.",
30+
name: "Secret Name B",
31+
},
32+
],
33+
},
34+
options: optionsBase,
35+
});
36+
37+
expect(creation).toMatchInlineSnapshot(`
38+
{
39+
"suggestions": [
40+
"- populate the secrets on https://github.com/test-owner/test-repository/settings/secrets/actions:
41+
- Secret Name A (Secret description a.)
42+
- Secret Name B (Secret description b.)",
43+
],
44+
}
45+
`);
46+
});
47+
});

src/blocks/blockRepositorySecrets.ts

+17-20
Original file line numberDiff line numberDiff line change
@@ -1,34 +1,31 @@
11
import { z } from "zod";
22

33
import { base } from "../base.js";
4+
import { getInstallationSuggestions } from "./getInstallationSuggestions.js";
45

5-
const zSecret = z.object({
6-
description: z.string(),
7-
name: z.string(),
8-
});
96
export const blockRepositorySecrets = base.createBlock({
107
about: {
118
name: "Repository Secrets",
129
},
1310
addons: {
14-
secrets: z.array(zSecret).default([]),
11+
secrets: z
12+
.array(
13+
z.object({
14+
description: z.string(),
15+
name: z.string(),
16+
}),
17+
)
18+
.default([]),
1519
},
16-
produce({ addons }) {
20+
produce({ addons, options }) {
1721
return {
18-
suggestions: addons.secrets.length
19-
? [
20-
[
21-
`- populate the secret`,
22-
addons.secrets.length === 1 ? "" : "s",
23-
`:\n`,
24-
addons.secrets.map(printSecret).join("\n"),
25-
].join(""),
26-
]
27-
: undefined,
22+
suggestions: getInstallationSuggestions(
23+
"populate the secret",
24+
addons.secrets.map(
25+
(secret) => `${secret.name} (${secret.description})`,
26+
),
27+
`https://github.com/${options.owner}/${options.repository}/settings/secrets/actions`,
28+
),
2829
};
2930
},
3031
});
31-
32-
function printSecret(app: z.infer<typeof zSecret>) {
33-
return ` - ${app.name} (${app.description})`;
34-
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
import { describe, expect, it } from "vitest";
2+
3+
import { getInstallationSuggestions } from "./getInstallationSuggestions.js";
4+
5+
const description = "do the action";
6+
const url = "https://example.com";
7+
8+
describe(getInstallationSuggestions, () => {
9+
it("returns undefined when there are no entries", () => {
10+
const actual = getInstallationSuggestions(description, [], url);
11+
12+
expect(actual).toBeUndefined();
13+
});
14+
15+
it("returns a non-plural list when there is one entry", () => {
16+
const actual = getInstallationSuggestions(description, ["entry"], url);
17+
18+
expect(actual).toMatchInlineSnapshot(`
19+
[
20+
"- do the action on https://example.com:
21+
- entry",
22+
]
23+
`);
24+
});
25+
26+
it("returns a plural list when there are multiple entries", () => {
27+
const actual = getInstallationSuggestions(
28+
description,
29+
["entry a", "entry b"],
30+
url,
31+
);
32+
33+
expect(actual).toMatchInlineSnapshot(`
34+
[
35+
"- do the actions on https://example.com:
36+
- entry a
37+
- entry b",
38+
]
39+
`);
40+
});
41+
});
+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
export function getInstallationSuggestions(
2+
description: string,
3+
entries: string[],
4+
url: string,
5+
) {
6+
return entries.length
7+
? [
8+
[
9+
`- ${description}`,
10+
entries.length === 1 ? "" : "s",
11+
` on ${url}:\n`,
12+
entries.map((entry) => ` - ${entry}`).join("\n"),
13+
].join(""),
14+
]
15+
: undefined;
16+
}

0 commit comments

Comments
 (0)