Skip to content

Commit f3ab953

Browse files
feat: add --words with default from cspell.json words (#2021)
## PR Checklist - [x] Addresses an existing open issue: fixes #2020 - [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 Adds a `words?: string[]` option that defaults to reading `cspell.json` as JSON. `blockCSpell` will add in any words from it to what it produces. 🎁
1 parent bab5313 commit f3ab953

File tree

7 files changed

+150
-17
lines changed

7 files changed

+150
-17
lines changed

docs/CLI.md

+15-14
Original file line numberDiff line numberDiff line change
@@ -43,20 +43,21 @@ See [Bingo > Stratum > Concepts > Templates > `--preset`](https://www.create.bin
4343
The following flags may be provided on the CLI to customize their values.
4444
Each defaults to a value based on the running system, including an repository if transitioning one.
4545

46-
| Flag | Type | Description | Default |
47-
| -------------- | ---------- | ------------------------------------------------------------------------------------------------------------------- | ------------------------------------------------------------------------------------- |
48-
| `--access` | `string` | Which [`npm publish --access`](https://docs.npmjs.com/cli/commands/npm-publish#access) to release npm packages with | `"public"` |
49-
| `--author` | `string` | Username on npm to publish packages under | An existing npm author, or the currently logged in npm user, or `owner.toLowerCase()` |
50-
| `--bin` | `string` | Value to set in `package.json`'s `"bin"` property, per [FAQs > How can I use `bin`?](./FAQs.md#how-can-i-use-bin) | _(none)_ |
51-
| `--email` | `string` | Email address to be listed as the point of contact in docs and packages (e.g. `[email protected]`) | Yours from `gh`, `git config`, or `npm whoami` |
52-
| `--emoji` | `string` | decorative emoji to use in descriptions and docs | The last emoji from `description`, or `"💖"` |
53-
| `--funding` | `string` | GitHub organization or username to mention in `funding.yml` | The same as `owner` |
54-
| `--keywords` | `string[]` | Any number of keywords to include in `package.json` | _(none)_ |
55-
| `--owner` | `string` | Organization or user owning the repository | Yours from `gh` or `git config` |
56-
| `--pnpm` | `string` | pnpm version for `package.json`'s `packageManager` field | Existing value in `package.json` if it exists |
57-
| `--repository` | `string` | Name for the new repository | The same as `--directory` |
58-
| `--title` | `string` | 'Title Case' title for the repository | Title-cased `repository` |
59-
| `--version` | `string` | package version to publish as and store in `package.json` | Existing value in `package.json` if it exists, or `"0.0.0"` |
46+
| Flag | Type | Description | Default |
47+
| -------------- | ---------- | ------------------------------------------------------------------------------------------------------------------- | ---------------------------------------------------------------------------------------------- |
48+
| `--access` | `string` | Which [`npm publish --access`](https://docs.npmjs.com/cli/commands/npm-publish#access) to release npm packages with | `"public"` |
49+
| `--author` | `string` | Username on npm to publish packages under | An existing npm author, or the currently logged in npm user, or `owner.toLowerCase()` |
50+
| `--bin` | `string` | Value to set in `package.json`'s `"bin"` property, per [FAQs > How can I use `bin`?](./FAQs.md#how-can-i-use-bin) | _(none)_ |
51+
| `--email` | `string` | Email address to be listed as the point of contact in docs and packages (e.g. `[email protected]`) | Yours from `gh`, `git config`, or `npm whoami` |
52+
| `--emoji` | `string` | decorative emoji to use in descriptions and docs | The last emoji from `description`, or `"💖"` |
53+
| `--funding` | `string` | GitHub organization or username to mention in `funding.yml` | The same as `owner` |
54+
| `--keywords` | `string[]` | Any number of keywords to include in `package.json` | _(none)_ |
55+
| `--owner` | `string` | Organization or user owning the repository | Yours from `gh` or `git config` |
56+
| `--pnpm` | `string` | pnpm version for `package.json`'s `packageManager` field | Existing value in `package.json` if it exists |
57+
| `--repository` | `string` | Name for the new repository | The same as `--directory` |
58+
| `--title` | `string` | 'Title Case' title for the repository | Title-cased `repository` |
59+
| `--version` | `string` | package version to publish as and store in `package.json` | Existing value in `package.json` if it exists, or `"0.0.0"` |
60+
| `--words` | `string[]` | additional words to add to the CSpell dictionary | Existing `words` in a `cspell.json` file if it exists, and any new words in from other options |
6061

6162
For example, customizing the npm author and funding source to different values than what would be inferred:
6263

src/base.test.ts

+2
Original file line numberDiff line numberDiff line change
@@ -53,6 +53,8 @@ describe("base", () => {
5353
title: "Create TypeScript App",
5454
usage: expect.any(String),
5555
version: expect.any(String),
56+
// eslint-disable-next-line @typescript-eslint/no-unsafe-member-access, @typescript-eslint/no-require-imports
57+
words: require("../cspell.json").words,
5658
workflowsVersions: expect.any(Object),
5759
});
5860
});

src/base.ts

+8
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,7 @@ import { readRepository } from "./options/readRepository.js";
3131
import { readRulesetId } from "./options/readRulesetId.js";
3232
import { readTitle } from "./options/readTitle.js";
3333
import { readUsage } from "./options/readUsage.js";
34+
import { readWords } from "./options/readWords.js";
3435
import { readWorkflowsVersions } from "./options/readWorkflowsVersions.js";
3536
import { zContributor, zWorkflowsVersions } from "./schemas.js";
3637

@@ -158,6 +159,10 @@ export const base = createBase({
158159
.string()
159160
.optional()
160161
.describe("package version to publish as and store in `package.json`"),
162+
words: z
163+
.array(z.string())
164+
.optional()
165+
.describe("additional words to add to the CSpell dictionary"),
161166
workflowsVersions: zWorkflowsVersions
162167
.optional()
163168
.describe("existing versions of GitHub Actions workflows used"),
@@ -273,6 +278,8 @@ export const base = createBase({
273278

274279
const getVersion = lazyValue(async () => (await getPackageData()).version);
275280

281+
const getWords = lazyValue(async () => await readWords(take));
282+
276283
const getWorkflowData = lazyValue(
277284
async () => await readWorkflowsVersions(take),
278285
);
@@ -300,6 +307,7 @@ export const base = createBase({
300307
title: getTitle,
301308
usage: getUsage,
302309
version: getVersion,
310+
words: getWords,
303311
workflowsVersions: getWorkflowData,
304312
};
305313
},

src/blocks/blockCSpell.test.ts

+74-1
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ vi.mock("../utils/resolveBin.js", () => ({
99
}));
1010

1111
describe("blockCSpell", () => {
12-
test("without addons", () => {
12+
test("without addons or options", () => {
1313
const creation = testBlock(blockCSpell, {
1414
options: optionsBase,
1515
});
@@ -153,6 +153,79 @@ describe("blockCSpell", () => {
153153
`);
154154
});
155155

156+
test("with options", () => {
157+
const creation = testBlock(blockCSpell, {
158+
options: {
159+
...optionsBase,
160+
words: ["joshuakgoldberg"],
161+
},
162+
});
163+
164+
expect(creation).toMatchInlineSnapshot(`
165+
{
166+
"addons": [
167+
{
168+
"addons": {
169+
"sections": {
170+
"Linting": {
171+
"contents": {
172+
"items": [
173+
"- \`pnpm lint:spelling\` ([cspell](https://cspell.org)): Spell checks across all source files",
174+
],
175+
},
176+
},
177+
},
178+
},
179+
"block": [Function],
180+
},
181+
{
182+
"addons": {
183+
"extensions": [
184+
"streetsidesoftware.code-spell-checker",
185+
],
186+
},
187+
"block": [Function],
188+
},
189+
{
190+
"addons": {
191+
"jobs": [
192+
{
193+
"name": "Lint Spelling",
194+
"steps": [
195+
{
196+
"run": "pnpm lint:spelling",
197+
},
198+
],
199+
},
200+
],
201+
"removedWorkflows": [
202+
"lint-spelling",
203+
"spelling",
204+
],
205+
},
206+
"block": [Function],
207+
},
208+
{
209+
"addons": {
210+
"properties": {
211+
"devDependencies": {
212+
"cspell": "8.17.5",
213+
},
214+
"scripts": {
215+
"lint:spelling": "cspell "**" ".github/**/*"",
216+
},
217+
},
218+
},
219+
"block": [Function],
220+
},
221+
],
222+
"files": {
223+
"cspell.json": "{"dictionaries":["npm","node","typescript"],"ignorePaths":[".github","CHANGELOG.md","lib","node_modules","pnpm-lock.yaml"],"words":["joshuakgoldberg"]}",
224+
},
225+
}
226+
`);
227+
});
228+
156229
test("setup mode", () => {
157230
const creation = testBlock(blockCSpell, {
158231
mode: "setup",

src/blocks/blockCSpell.ts

+6-2
Original file line numberDiff line numberDiff line change
@@ -20,9 +20,13 @@ export const blockCSpell = base.createBlock({
2020
ignores: z.array(z.string()).default([]),
2121
words: z.array(z.string()).default([]),
2222
},
23-
produce({ addons }) {
23+
produce({ addons, options }) {
2424
const { ignores, words } = addons;
2525

26+
const allWords = Array.from(
27+
new Set([...(options.words ?? []), ...words]),
28+
).sort();
29+
2630
return {
2731
addons: [
2832
blockDevelopmentDocs({
@@ -68,7 +72,7 @@ export const blockCSpell = base.createBlock({
6872
"pnpm-lock.yaml",
6973
...ignores,
7074
].sort(),
71-
...(words.length && { words: words.sort() }),
75+
...(allWords.length && { words: allWords }),
7276
}),
7377
},
7478
};

src/options/readWords.test.ts

+30
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
import { describe, expect, it, vi } from "vitest";
2+
3+
import { readWords } from "./readWords.js";
4+
5+
describe(readWords, () => {
6+
it("returns undefined when the file does not exist", async () => {
7+
const take = vi.fn().mockRejectedValueOnce(new Error("Oh no!"));
8+
9+
const actual = await readWords(take);
10+
11+
expect(actual).toBeUndefined();
12+
});
13+
14+
it("returns undefined when the file has no words", async () => {
15+
const take = vi.fn().mockResolvedValueOnce({});
16+
17+
const actual = await readWords(take);
18+
19+
expect(actual).toBeUndefined();
20+
});
21+
22+
it("returns the words when the file has words", async () => {
23+
const words = ["abc", "def"];
24+
const take = vi.fn().mockResolvedValueOnce({ words });
25+
26+
const actual = await readWords(take);
27+
28+
expect(actual).toEqual(words);
29+
});
30+
});

src/options/readWords.ts

+15
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
import { TakeInput } from "bingo";
2+
import { inputFromFileJSON } from "input-from-file-json";
3+
4+
import { swallowErrorAsync } from "../utils/swallowErrorAsync.js";
5+
6+
export async function readWords(take: TakeInput) {
7+
const cspell =
8+
(await swallowErrorAsync(
9+
take(inputFromFileJSON, {
10+
filePath: "./cspell.json",
11+
}),
12+
)) || {};
13+
14+
return (cspell as { words?: string[] }).words;
15+
}

0 commit comments

Comments
 (0)