Skip to content

Commit 778e5eb

Browse files
fix: update existing ruleset when in transition mode (#1951)
## PR Checklist - [x] Addresses an existing open issue: fixes #1950 - [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 Previously, `blockRepositoryBranchRuleset` would _always_ create a ruleset on `main` with the hardcoded name. This is fine for _setup_ mode but crashes on a duplicate ID creation error in _transition_ mode. Now, its creations are split between the two modes: * Setup mode `POST` creates the ruleset as before * Transition mode `PUT` updates the existing ruleset The `PUT` update requires a `ruleset_id`, so I added that as a new option in the Base populated by a new `inputFromOctkit`. 🎁
1 parent 5387d42 commit 778e5eb

4 files changed

+202
-64
lines changed

src/base.ts

+28-4
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@ import lazyValue from "lazy-value";
99
import npmUser from "npm-user";
1010
import { z } from "zod";
1111

12+
import { inputFromOctokit } from "./inputs/inputFromGitHub.js";
1213
import { parsePackageAuthor } from "./options/parsePackageAuthor.js";
1314
import { readDefaultsFromReadme } from "./options/readDefaultsFromReadme.js";
1415
import { readDescription } from "./options/readDescription.js";
@@ -125,6 +126,10 @@ export const base = createBase({
125126
repository: z
126127
.string()
127128
.describe("'kebab-case' or 'PascalCase' title of the repository"),
129+
rulesetId: z
130+
.string()
131+
.optional()
132+
.describe("GitHub branch ruleset ID for main branch protections"),
128133
title: z.string().describe("'Title Case' title for the repository"),
129134
usage: z
130135
.string()
@@ -163,6 +168,20 @@ export const base = createBase({
163168

164169
const readme = lazyValue(async () => await readFileSafe("README.md", ""));
165170

171+
const rulesetId = lazyValue(async () => {
172+
const rulesets = (await take(inputFromOctokit, {
173+
endpoint: "GET /repos/{owner}/{repo}/rulesets",
174+
options: {
175+
owner: await owner(),
176+
repo: await repository(),
177+
},
178+
})) as { id: string; name: string }[];
179+
180+
return rulesets.find(
181+
(ruleset) => ruleset.name === "Branch protection for main",
182+
)?.id;
183+
});
184+
166185
// TODO: Make these all use take
167186

168187
const gitDefaults = tryCatchLazyValueAsync(async () =>
@@ -198,6 +217,13 @@ export const base = createBase({
198217

199218
const version = lazyValue(async () => (await packageData()).version);
200219

220+
const owner = lazyValue(
221+
async () =>
222+
(await gitDefaults())?.organization ??
223+
(await packageAuthor()).author ??
224+
(await githubCliUser()),
225+
);
226+
201227
const repository = lazyValue(
202228
async () =>
203229
options.repository ??
@@ -218,10 +244,7 @@ export const base = createBase({
218244
guide: readGuide,
219245
login: author,
220246
node,
221-
owner: async () =>
222-
(await gitDefaults())?.organization ??
223-
(await packageAuthor()).author ??
224-
(await githubCliUser()),
247+
owner,
225248
packageData: async () => {
226249
const original = await packageData();
227250

@@ -232,6 +255,7 @@ export const base = createBase({
232255
};
233256
},
234257
repository,
258+
rulesetId,
235259
...readDefaultsFromReadme(readme, repository),
236260
version,
237261
};

src/blocks/blockRepositoryBranchRuleset.test.ts

+66-6
Original file line numberDiff line numberDiff line change
@@ -5,39 +5,99 @@ import { blockRepositoryBranchRuleset } from "./blockRepositoryBranchRuleset.js"
55
import { optionsBase } from "./options.fakes.js";
66

77
describe("blockRepositoryBranchRuleset", () => {
8-
test("without addons", () => {
8+
test("without addons when mode is undefined", () => {
99
const creation = testBlock(blockRepositoryBranchRuleset, {
1010
options: optionsBase,
1111
});
1212

13+
expect(creation).toMatchInlineSnapshot(`{}`);
14+
});
15+
16+
// TODO for improving the "requests" snapshots:
17+
// https://github.com/JoshuaKGoldberg/create/issues/65
18+
19+
test("without addons when mode is setup", () => {
20+
const creation = testBlock(blockRepositoryBranchRuleset, {
21+
mode: "setup",
22+
options: optionsBase,
23+
});
24+
1325
expect(creation).toMatchInlineSnapshot(`
1426
{
1527
"requests": [
1628
{
17-
"id": "branch-ruleset",
29+
"id": "branch-ruleset-create",
1830
"send": [Function],
1931
},
2032
],
2133
}
2234
`);
2335
});
2436

25-
// TODO for improving the "requests" snapshots:
26-
// https://github.com/JoshuaKGoldberg/create/issues/65
37+
test("without addons when mode is transition", () => {
38+
const creation = testBlock(blockRepositoryBranchRuleset, {
39+
mode: "transition",
40+
options: optionsBase,
41+
});
42+
43+
expect(creation).toMatchInlineSnapshot(`
44+
{
45+
"requests": [
46+
{
47+
"id": "branch-ruleset-update",
48+
"send": [Function],
49+
},
50+
],
51+
}
52+
`);
53+
});
54+
55+
test("with addons when mode is undefined", () => {
56+
const creation = testBlock(blockRepositoryBranchRuleset, {
57+
addons: {
58+
requiredStatusChecks: ["build", "test"],
59+
},
60+
options: optionsBase,
61+
});
62+
63+
expect(creation).toMatchInlineSnapshot(`{}`);
64+
});
65+
66+
test("with addons when mode is setup", () => {
67+
const creation = testBlock(blockRepositoryBranchRuleset, {
68+
addons: {
69+
requiredStatusChecks: ["build", "test"],
70+
},
71+
mode: "setup",
72+
options: optionsBase,
73+
});
74+
75+
expect(creation).toMatchInlineSnapshot(`
76+
{
77+
"requests": [
78+
{
79+
"id": "branch-ruleset-create",
80+
"send": [Function],
81+
},
82+
],
83+
}
84+
`);
85+
});
2786

28-
test("with addons", () => {
87+
test("with addons when mode is transition", () => {
2988
const creation = testBlock(blockRepositoryBranchRuleset, {
3089
addons: {
3190
requiredStatusChecks: ["build", "test"],
3291
},
92+
mode: "setup",
3393
options: optionsBase,
3494
});
3595

3696
expect(creation).toMatchInlineSnapshot(`
3797
{
3898
"requests": [
3999
{
40-
"id": "branch-ruleset",
100+
"id": "branch-ruleset-create",
41101
"send": [Function],
42102
},
43103
],
+86-54
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
1+
import { BlockCreation } from "bingo-stratum";
12
import { z } from "zod";
23

3-
import { base } from "../base.js";
4+
import { base, BaseOptions } from "../base.js";
45

56
export const blockRepositoryBranchRuleset = base.createBlock({
67
about: {
@@ -9,60 +10,91 @@ export const blockRepositoryBranchRuleset = base.createBlock({
910
addons: {
1011
requiredStatusChecks: z.array(z.string()).default([]),
1112
},
12-
produce({ addons, options }) {
13-
return {
14-
requests: [
15-
{
16-
id: "branch-ruleset",
17-
async send({ octokit }) {
18-
await octokit.request("POST /repos/{owner}/{repo}/rulesets", {
19-
bypass_actors: [
20-
{
21-
// This *seems* to be the Repository Admin role always?
22-
// https://github.com/github/rest-api-description/issues/4406
23-
actor_id: 5,
24-
actor_type: "RepositoryRole",
25-
bypass_mode: "always",
26-
},
27-
],
28-
conditions: {
29-
ref_name: {
30-
exclude: [],
31-
include: ["refs/heads/main"],
32-
},
13+
setup({ addons, options }) {
14+
return createRequestSend(
15+
addons.requiredStatusChecks,
16+
"POST /repos/{owner}/{repo}/rulesets",
17+
options,
18+
"branch-ruleset-create",
19+
undefined,
20+
);
21+
},
22+
transition({ addons, options }) {
23+
return createRequestSend(
24+
addons.requiredStatusChecks,
25+
`PUT /repos/{owner}/{repo}/rulesets/{ruleset_id}`,
26+
options,
27+
"branch-ruleset-update",
28+
options.rulesetId,
29+
);
30+
},
31+
// TODO: Make produce() optional
32+
// This needs createBlock to be generic to know if block.produce({}) is ok
33+
produce() {
34+
return {};
35+
},
36+
});
37+
38+
function createRequestSend(
39+
contexts: string[],
40+
endpoint: string,
41+
options: BaseOptions,
42+
requestId: string,
43+
rulesetId: string | undefined,
44+
): Partial<BlockCreation<BaseOptions>> {
45+
return {
46+
requests: [
47+
{
48+
id: requestId,
49+
async send({ octokit }) {
50+
await octokit.request(endpoint, {
51+
bypass_actors: [
52+
{
53+
// This *seems* to be the Repository Admin role always?
54+
// https://github.com/github/rest-api-description/issues/4406
55+
actor_id: 5,
56+
actor_type: "RepositoryRole",
57+
bypass_mode: "always",
58+
},
59+
],
60+
conditions: {
61+
ref_name: {
62+
exclude: [],
63+
include: ["refs/heads/main"],
3364
},
34-
enforcement: "active",
35-
name: "Branch protection for main",
36-
owner: options.owner,
37-
repo: options.repository,
38-
rules: [
39-
{ type: "deletion" },
40-
{
41-
parameters: {
42-
allowed_merge_methods: ["squash"],
43-
dismiss_stale_reviews_on_push: false,
44-
require_code_owner_review: false,
45-
require_last_push_approval: false,
46-
required_approving_review_count: 0,
47-
required_review_thread_resolution: false,
48-
},
49-
type: "pull_request",
65+
},
66+
enforcement: "active",
67+
name: "Branch protection for main",
68+
owner: options.owner,
69+
repo: options.repository,
70+
rules: [
71+
{ type: "deletion" },
72+
{
73+
parameters: {
74+
allowed_merge_methods: ["squash"],
75+
dismiss_stale_reviews_on_push: false,
76+
require_code_owner_review: false,
77+
require_last_push_approval: false,
78+
required_approving_review_count: 0,
79+
required_review_thread_resolution: false,
5080
},
51-
{
52-
parameters: {
53-
required_status_checks: addons.requiredStatusChecks.map(
54-
(context) => ({ context }),
55-
),
56-
strict_required_status_checks_policy: false,
57-
},
58-
type: "required_status_checks",
81+
type: "pull_request",
82+
},
83+
{
84+
parameters: {
85+
required_status_checks: contexts.map((context) => ({
86+
context,
87+
})),
88+
strict_required_status_checks_policy: false,
5989
},
60-
],
61-
target: "branch",
62-
});
63-
},
90+
type: "required_status_checks",
91+
},
92+
],
93+
ruleset_id: rulesetId,
94+
target: "branch",
95+
});
6496
},
65-
],
66-
};
67-
},
68-
});
97+
},
98+
],
99+
};
100+
}

src/inputs/inputFromGitHub.ts

+22
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
import { createInput } from "bingo";
2+
import { z } from "zod";
3+
4+
export const inputFromOctokit = createInput({
5+
args: {
6+
endpoint: z.string(),
7+
options: z.record(z.string(), z.unknown()),
8+
},
9+
// TODO: Strongly type this, then push it upstream to Bingo
10+
// This will require smart types around GitHub endpoints, similar to:
11+
// https://github.com/JoshuaKGoldberg/bingo/issues/65
12+
async produce({ args, fetchers }): Promise<unknown> {
13+
const response = await fetchers.octokit.request(args.endpoint, {
14+
headers: {
15+
"X-GitHub-Api-Version": "2022-11-28",
16+
},
17+
...args.options,
18+
});
19+
20+
return response.data;
21+
},
22+
});

0 commit comments

Comments
 (0)