Skip to content

Commit 11d8e05

Browse files
feat: add opt-in blockESLintPlugin (#2131)
## PR Checklist - [x] Addresses an existing open issue: fixes #1936 - [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 Block that devs can use to sync tooling for ESLint plugins built on CTA. It replaces the default example files with its own, so creating a new repo will look like: ```shell npx create-typescript-app --add-eslint-plugin --exclude-example-files ``` 🎁
1 parent c2b0fc7 commit 11d8e05

8 files changed

+659
-13
lines changed

cspell.json

+1
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@
1515
"attw",
1616
"boop",
1717
"dbaeumer",
18+
"eslint-doc-generatorrc.js",
1819
"infile",
1920
"joshuakgoldberg",
2021
"markdownlintignore",

docs/Blocks.md

+1
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@ This table summarizes each block and which base levels they're included in:
2323
| ESLint Node Plugin | `--add-eslint-node-plugin`, `--exclude-eslint-node-plugin` | | | 💯 |
2424
| ESLint package.json Plugin | `--add-eslint-package-json-plugin`, `--exclude-eslint-package-json-plugin` | | | 💯 |
2525
| ESLint Perfectionist Plugin | `--add-eslint-perfectionist-plugin`, `--exclude-eslint-perfectionist-plugin` | | | 💯 |
26+
| ESLint Plugin | `--add-eslint-plugin`, `--exclude-eslint-plugin` | | | |
2627
| ESLint Regexp Plugin | `--add-eslint-regexp-plugin`, `--exclude-eslint-regexp-plugin` | | | 💯 |
2728
| ESLint YML Plugin | `--add-eslint-yml-plugin`, `--exclude-eslint-yml-plugin` | | | 💯 |
2829
| Funding | `--add-funding`, `--exclude-funding` | || 💯 |

src/blocks/blockESLint.test.ts

+6
Original file line numberDiff line numberDiff line change
@@ -465,6 +465,10 @@ describe("blockESLint", () => {
465465
imports: [
466466
{ source: "eslint-plugin-markdown", specifier: "a", types: true },
467467
{ source: "eslint-plugin-regexp", specifier: "b" },
468+
{
469+
source: { packageName: "eslint-plugin-unknown", version: "1.2.3" },
470+
specifier: "c",
471+
},
468472
],
469473
rules: {
470474
"a/b": "error",
@@ -536,6 +540,7 @@ describe("blockESLint", () => {
536540
"eslint": "9.22.0",
537541
"eslint-plugin-markdown": "5.1.0",
538542
"eslint-plugin-regexp": "2.7.0",
543+
"eslint-plugin-unknown": "1.2.3",
539544
"typescript-eslint": "8.26.1",
540545
},
541546
"scripts": {
@@ -586,6 +591,7 @@ describe("blockESLint", () => {
586591
import eslint from "@eslint/js";
587592
import a from "eslint-plugin-markdown"
588593
import b from "eslint-plugin-regexp"
594+
import c from "eslint-plugin-unknown"
589595
import tseslint from "typescript-eslint";
590596
591597
export default tseslint.config(

src/blocks/blockESLint.ts

+35-13
Original file line numberDiff line numberDiff line change
@@ -49,7 +49,10 @@ const zExtension = z.object({
4949
});
5050

5151
const zPackageImport = z.object({
52-
source: z.string(),
52+
source: z.union([
53+
z.string(),
54+
z.object({ packageName: z.string(), version: z.string() }),
55+
]),
5356
specifier: z.string(),
5457
types: z.boolean().optional(),
5558
});
@@ -88,7 +91,7 @@ export const blockESLint = base.createBlock({
8891
'import tseslint from "typescript-eslint";',
8992
...imports.map(
9093
(packageImport) =>
91-
`import ${packageImport.specifier} from "${packageImport.source}"`,
94+
`import ${packageImport.specifier} from "${typeof packageImport.source === "string" ? packageImport.source : packageImport.source.packageName}"`,
9295
),
9396
].sort((a, b) =>
9497
a.replace(/.+from/, "").localeCompare(b.replace(/.+from/, "")),
@@ -183,17 +186,36 @@ Each should be shown in VS Code, and can be run manually on the command-line:
183186
}),
184187
blockPackageJson({
185188
properties: {
186-
devDependencies: getPackageDependencies(
187-
"@eslint/js",
188-
"@types/node",
189-
"eslint",
190-
"typescript-eslint",
191-
...imports.flatMap(({ source, types }) => {
192-
// eslint-disable-next-line @typescript-eslint/no-unsafe-call -- https://github.com/egoist/parse-package-name/issues/30
193-
const { name } = parsePackageName(source) as { name: string };
194-
return types ? [name, `@types/${name}`] : [name];
195-
}),
196-
),
189+
devDependencies: {
190+
...getPackageDependencies(
191+
"@eslint/js",
192+
"@types/node",
193+
"eslint",
194+
"typescript-eslint",
195+
...imports
196+
.filter((imported) => typeof imported.source === "string")
197+
.flatMap(({ source, types }) => {
198+
// eslint-disable-next-line @typescript-eslint/no-unsafe-call -- https://github.com/egoist/parse-package-name/issues/30
199+
const { name } = parsePackageName(source) as {
200+
name: string;
201+
};
202+
return types ? [name, `@types/${name}`] : [name];
203+
}),
204+
),
205+
...Object.fromEntries(
206+
imports
207+
.filter(
208+
(
209+
imported,
210+
): imported is typeof imported & { source: object } =>
211+
typeof imported.source === "object",
212+
)
213+
.map((imported) => [
214+
imported.source.packageName,
215+
imported.source.version,
216+
]),
217+
),
218+
},
197219
scripts: {
198220
lint: "eslint . --max-warnings 0",
199221
},

0 commit comments

Comments
 (0)