Skip to content

Commit 810eaad

Browse files
fix: add README.md contributors section in setup (#1958)
## PR Checklist - [x] Addresses an existing open issue: fixes #1956 - [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 The previous `pnpx all-contributors-cli` command was only enough for transition mode to pick up on existing contributors and create a README.md section. For setup mode, there needed to be a default in options if `.all-contributorsrc` doesn't yet exist. That's now pulled from the GitHub API. 🎁
1 parent 8ca464e commit 810eaad

6 files changed

+105
-13
lines changed

src/base.ts

+5-11
Original file line numberDiff line numberDiff line change
@@ -3,14 +3,14 @@ import { execaCommand } from "execa";
33
import gitRemoteOriginUrl from "git-remote-origin-url";
44
import gitUrlParse from "git-url-parse";
55
import { inputFromFile } from "input-from-file";
6-
import { inputFromFileJSON } from "input-from-file-json";
76
import { inputFromScript } from "input-from-script";
87
import lazyValue from "lazy-value";
98
import npmUser from "npm-user";
109
import { z } from "zod";
1110

1211
import { inputFromOctokit } from "./inputs/inputFromOctokit.js";
1312
import { parsePackageAuthor } from "./options/parsePackageAuthor.js";
13+
import { readAllContributors } from "./options/readAllContributors.js";
1414
import { readDefaultsFromReadme } from "./options/readDefaultsFromReadme.js";
1515
import { readDescription } from "./options/readDescription.js";
1616
import { readDocumentation } from "./options/readDocumentation.js";
@@ -19,7 +19,6 @@ import { readFileSafe } from "./options/readFileSafe.js";
1919
import { readFunding } from "./options/readFunding.js";
2020
import { readGuide } from "./options/readGuide.js";
2121
import { readPackageData } from "./options/readPackageData.js";
22-
import { AllContributorsData } from "./types.js";
2322
import { swallowError } from "./utils/swallowError.js";
2423
import { tryCatchLazyValueAsync } from "./utils/tryCatchLazyValueAsync.js";
2524

@@ -141,14 +140,7 @@ export const base = createBase({
141140
.describe("package version to publish as and store in `package.json`"),
142141
},
143142
prepare({ options, take }) {
144-
const allContributors = lazyValue(async () => {
145-
const contributions = (await take(inputFromFileJSON, {
146-
filePath: ".all-contributorsrc",
147-
})) as AllContributorsData;
148-
149-
return contributions.contributors;
150-
});
151-
143+
const allContributors = lazyValue(async () => readAllContributors(take));
152144
const documentation = lazyValue(async () => readDocumentation(take));
153145

154146
const nvmrc = lazyValue(
@@ -232,14 +224,16 @@ export const base = createBase({
232224
options.directory,
233225
);
234226

227+
const email = lazyValue(async () => readEmails(npmDefaults, packageAuthor));
228+
235229
return {
236230
access: "public" as const,
237231
author,
238232
bin: async () => (await packageData()).bin,
239233
contributors: allContributors,
240234
description: async () => await readDescription(packageData, readme),
241235
documentation,
242-
email: async () => readEmails(npmDefaults, packageAuthor),
236+
email,
243237
funding: readFunding,
244238
guide: readGuide,
245239
login: author,

src/blocks/blockAllContributors.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
import _ from "lodash";
22

33
import { base, Contributor } from "../base.js";
4+
import { ownerContributions } from "../data/contributions.js";
45
import { blockREADME } from "./blockREADME.js";
56
import { blockRepositorySecrets } from "./blockRepositorySecrets.js";
67
import { createSoloWorkflowFile } from "./files/createSoloWorkflowFile.js";
@@ -68,7 +69,7 @@ export const blockAllContributors = base.createBlock({
6869
scripts: [
6970
{
7071
commands: [
71-
`pnpx all-contributors-cli add ${options.owner} code,content,doc,ideas,infra,maintenance,projectManagement,tool`,
72+
`pnpx all-contributors-cli add ${options.owner} ${ownerContributions.join(",")}`,
7273
],
7374
phase: CommandPhase.Process,
7475
},

src/data/contributions.ts

+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
export const ownerContributions = [
2+
"code",
3+
"content",
4+
"doc",
5+
"ideas",
6+
"infra",
7+
"maintenance",
8+
"projectManagement",
9+
"tool",
10+
];

src/inputs/inputFromOctokit.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@ import { z } from "zod";
44
export const inputFromOctokit = createInput({
55
args: {
66
endpoint: z.string(),
7-
options: z.record(z.string(), z.unknown()),
7+
options: z.record(z.string(), z.unknown()).optional(),
88
},
99
// TODO: Strongly type this, then push it upstream to Bingo
1010
// This will require smart types around GitHub endpoints, similar to:
+57
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,57 @@
1+
import { createMockSystems } from "bingo-testers";
2+
import { describe, expect, it, vi } from "vitest";
3+
4+
import { ownerContributions } from "../data/contributions.js";
5+
import { readAllContributors } from "./readAllContributors.js";
6+
7+
const mockInputFromFileJSON = vi.fn();
8+
9+
vi.mock("input-from-file-json", () => ({
10+
get inputFromFileJSON() {
11+
return mockInputFromFileJSON;
12+
},
13+
}));
14+
15+
const mockInputFromOctokit = vi.fn();
16+
17+
vi.mock("../inputs/inputFromOctokit.js", () => ({
18+
get inputFromOctokit() {
19+
return mockInputFromOctokit;
20+
},
21+
}));
22+
23+
const { take } = createMockSystems();
24+
25+
describe("readAllContributors", () => {
26+
it("returns contributors from .all-contributorsrc when it can be read", async () => {
27+
const contributors = ["a", "b", "c"];
28+
mockInputFromFileJSON.mockResolvedValueOnce({ contributors });
29+
30+
const actual = await readAllContributors(take);
31+
32+
expect(actual).toEqual(contributors);
33+
expect(mockInputFromOctokit).not.toHaveBeenCalled();
34+
});
35+
36+
it("returns the current user as a contributor when .all-contributorsrc cannot be read", async () => {
37+
mockInputFromFileJSON.mockResolvedValueOnce(new Error("Oh no!"));
38+
mockInputFromOctokit.mockResolvedValueOnce({
39+
avatar_url: "avatar_url",
40+
blog: "blog",
41+
login: "login",
42+
name: "name",
43+
});
44+
45+
const actual = await readAllContributors(take);
46+
47+
expect(actual).toEqual([
48+
{
49+
avatar_url: "avatar_url",
50+
contributions: ownerContributions,
51+
login: "login",
52+
name: "name",
53+
profile: "blog",
54+
},
55+
]);
56+
});
57+
});

src/options/readAllContributors.ts

+30
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
import { TakeInput } from "bingo";
2+
import { inputFromFileJSON } from "input-from-file-json";
3+
4+
import { ownerContributions } from "../data/contributions.js";
5+
import { inputFromOctokit } from "../inputs/inputFromOctokit.js";
6+
import { AllContributorsData } from "../types.js";
7+
8+
export async function readAllContributors(take: TakeInput) {
9+
const contributions = (await take(inputFromFileJSON, {
10+
filePath: ".all-contributorsrc",
11+
})) as AllContributorsData | Error;
12+
13+
if (!(contributions instanceof Error)) {
14+
return contributions.contributors;
15+
}
16+
17+
const user = (await take(inputFromOctokit, {
18+
endpoint: "GET /user",
19+
})) as { avatar_url: string; blog: string; login: string; name: string };
20+
21+
return [
22+
{
23+
avatar_url: user.avatar_url,
24+
contributions: ownerContributions,
25+
login: user.login,
26+
name: user.name,
27+
profile: user.blog,
28+
},
29+
];
30+
}

0 commit comments

Comments
 (0)