Skip to content

Commit 983bcaf

Browse files
authoredApr 5, 2024··
feat(eslint-config): optional rules for module authors (#377)
1 parent 68e1a23 commit 983bcaf

File tree

10 files changed

+199
-23
lines changed

10 files changed

+199
-23
lines changed
 

‎docs/content/1.packages/1.config.md

+32-17
Original file line numberDiff line numberDiff line change
@@ -68,33 +68,48 @@ You might also want to add a script entry to your `package.json`:
6868

6969
### Customizing the Config
7070

71-
Note that `createConfigForNuxt()` returns `Promise<FlatConfig[]>`. ESLint allows you to run the promise directly on the default export. If you want to combine with other configs, you will need await and spread the result. For example:
71+
Note that `createConfigForNuxt()` returns a chainable [`FlatConfigComposer` instance](https://github.com/antfu/eslint-flat-config-utils#composer) from [`eslint-flat-config-utils`](https://github.com/antfu/eslint-flat-config-utils) which allows you to manipulate the ESLint flat config with ease. If you want to combine with other configs, you can use the `.append()` method:
7272

7373
```js [eslint.config.mjs]
7474
import { createConfigForNuxt } from '@nuxt/eslint-config/flat'
7575

76-
export default [
77-
...await createConfigForNuxt({
78-
// options here
79-
}),
80-
// other configs
81-
]
82-
// (which uses Top-level await)
76+
export default createConfigForNuxt({
77+
// options here
78+
})
79+
.prepend(
80+
// ...Prepend some flat configs in front
81+
)
82+
// Override some rules in a specific config, based on their name
83+
.override('nuxt/typescript', {
84+
rules: {
85+
// ...Override rules, for example:
86+
'@typescript-eslint/ban-types': 'off'
87+
}
88+
})
89+
// ...you can chain more operations as needed
8390
```
8491

85-
To make it easier, you can also use our helper `defineFlatConfigs`, which will also resolve and flatten the configs for you:
92+
`FlatConfigComposer` is also a Promise object, that you can use `await` to get the final config object.
93+
94+
### Module Authors
95+
96+
This config also provides rules for module/library authors. You can enable them by setting the `features.tooling` option to `true`:
97+
98+
:::callout{icon="i-ph-crane-tower-duotone" color="amber"}
99+
This feature is experimental and may change in the future.
100+
:::
86101

87102
```js [eslint.config.mjs]
88-
import { defineFlatConfigs, createConfigForNuxt } from '@nuxt/eslint-config/flat'
89-
90-
export default defineFlatConfigs(
91-
createConfigForNuxt({
92-
// options here
93-
}),
94-
// other configs
95-
)
103+
import { createConfigForNuxt } from '@nuxt/eslint-config/flat'
104+
105+
export default createConfigForNuxt({
106+
features: {
107+
tooling: true
108+
}
109+
})
96110
```
97111

112+
This will enable rules with `unicorn` and `jsdoc` plugins, to ensure your module is following best practices.
98113

99114
## Legacy Config Format
100115

‎eslint.config.js

+1
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ import { createConfigForNuxt } from '@nuxt/eslint-config/flat'
44
export default createConfigForNuxt({
55
features: {
66
stylistic: true,
7+
tooling: true,
78
},
89
dirs: {
910
src: [

‎packages/eslint-config/package.json

+2
Original file line numberDiff line numberDiff line change
@@ -45,6 +45,8 @@
4545
"@typescript-eslint/parser": "^7.5.0",
4646
"eslint-config-flat-gitignore": "^0.1.5",
4747
"eslint-flat-config-utils": "^0.2.0",
48+
"eslint-plugin-jsdoc": "^48.2.2",
49+
"eslint-plugin-unicorn": "^51.0.1",
4850
"eslint-plugin-import-x": "^0.5.0",
4951
"eslint-plugin-vue": "^9.24.0",
5052
"globals": "^15.0.0",
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
import type { FlatConfigItem } from 'eslint-flat-config-utils'
2+
import jsdocPlugin from 'eslint-plugin-jsdoc'
3+
import { resolveOptions } from '../utils'
4+
import type { NuxtESLintConfigOptions } from '../types'
5+
6+
export default function jsdoc(options: NuxtESLintConfigOptions = {}): FlatConfigItem[] {
7+
const resolved = resolveOptions(options)
8+
9+
return [
10+
{
11+
name: 'nuxt/tooling/jsdoc',
12+
plugins: {
13+
jsdoc: jsdocPlugin,
14+
},
15+
rules: {
16+
'jsdoc/check-access': 'warn',
17+
'jsdoc/check-param-names': 'warn',
18+
'jsdoc/check-property-names': 'warn',
19+
'jsdoc/check-types': 'warn',
20+
'jsdoc/empty-tags': 'warn',
21+
'jsdoc/implements-on-classes': 'warn',
22+
'jsdoc/no-defaults': 'warn',
23+
'jsdoc/no-multi-asterisks': 'warn',
24+
'jsdoc/require-param-name': 'warn',
25+
'jsdoc/require-property': 'warn',
26+
'jsdoc/require-property-description': 'warn',
27+
'jsdoc/require-property-name': 'warn',
28+
'jsdoc/require-returns-check': 'warn',
29+
'jsdoc/require-returns-description': 'warn',
30+
'jsdoc/require-yields-check': 'warn',
31+
32+
...resolved.features.stylistic
33+
? {
34+
'jsdoc/check-alignment': 'warn',
35+
'jsdoc/multiline-blocks': 'warn',
36+
}
37+
: {},
38+
},
39+
},
40+
]
41+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,42 @@
1+
// @ts-expect-error missing types
2+
import pluginUnicorn from 'eslint-plugin-unicorn'
3+
import type { FlatConfigItem } from 'eslint-flat-config-utils'
4+
5+
export default function unicorn(): FlatConfigItem[] {
6+
return [
7+
{
8+
name: 'nuxt/tooling/unicorn',
9+
plugins: {
10+
unicorn: pluginUnicorn,
11+
},
12+
rules: {
13+
// Pass error message when throwing errors
14+
'unicorn/error-message': 'error',
15+
// Uppercase regex escapes
16+
'unicorn/escape-case': 'error',
17+
// Array.isArray instead of instanceof
18+
'unicorn/no-instanceof-array': 'error',
19+
// Ban `new Array` as `Array` constructor's params are ambiguous
20+
'unicorn/no-new-array': 'error',
21+
// Prevent deprecated `new Buffer()`
22+
'unicorn/no-new-buffer': 'error',
23+
// Lowercase number formatting for octal, hex, binary (0x1'error' instead of 0X1'error')
24+
'unicorn/number-literal-case': 'error',
25+
// textContent instead of innerText
26+
'unicorn/prefer-dom-node-text-content': 'error',
27+
// includes over indexOf when checking for existence
28+
'unicorn/prefer-includes': 'error',
29+
// Prefer using the node: protocol
30+
'unicorn/prefer-node-protocol': 'error',
31+
// Prefer using number properties like `Number.isNaN` rather than `isNaN`
32+
'unicorn/prefer-number-properties': 'error',
33+
// String methods startsWith/endsWith instead of more complicated stuff
34+
'unicorn/prefer-string-starts-ends-with': 'error',
35+
// Enforce throwing type error when throwing error while checking typeof
36+
'unicorn/prefer-type-error': 'error',
37+
// Use new when throwing error
38+
'unicorn/throw-new-error': 'error',
39+
},
40+
},
41+
]
42+
}

‎packages/eslint-config/src/flat/index.ts

+14-5
Original file line numberDiff line numberDiff line change
@@ -52,12 +52,21 @@ export function createConfigForNuxt(options: NuxtESLintConfigOptions = {}): Flat
5252
nuxt(resolved),
5353
)
5454

55+
if (resolved.features.tooling) {
56+
c.append(
57+
import('./configs-tooling/jsdoc').then(m => m.default(resolved)),
58+
import('./configs-tooling/unicorn').then(m => m.default()),
59+
)
60+
}
61+
5562
if (resolved.features.stylistic) {
56-
c.append(import('./configs/stylistic').then(m => m.default(
57-
typeof resolved.features.stylistic === 'boolean'
58-
? {}
59-
: resolved.features.stylistic,
60-
)))
63+
const stylisticOptions = typeof resolved.features.stylistic === 'boolean'
64+
? {}
65+
: resolved.features.stylistic
66+
67+
c.append(
68+
import('./configs/stylistic').then(m => m.default(stylisticOptions)),
69+
)
6170
}
6271

6372
c.append(

‎packages/eslint-config/src/flat/types.ts

+8
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,14 @@ export interface NuxtESLintFeaturesOptions {
1010
*/
1111
standalone?: boolean
1212

13+
/**
14+
* Enable rules for Nuxt module authors or library authors
15+
*
16+
* @experimental Changes might not follow semver
17+
* @default false
18+
*/
19+
tooling?: boolean
20+
1321
/**
1422
* Enable stylistic ESLint rules for formatting and code style check
1523
*

‎packages/eslint-config/src/flat/utils.ts

+1
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,7 @@ export function resolveOptions(
3131
standalone: true,
3232
stylistic: false,
3333
typescript: true,
34+
tooling: false,
3435
...config.features,
3536
},
3637
dirs,

‎packages/module/src/modules/checker.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import { existsSync } from 'fs'
1+
import { existsSync } from 'node:fs'
22
import { addVitePlugin, addWebpackPlugin, useLogger } from '@nuxt/kit'
33
import { relative, resolve } from 'pathe'
44
import { watch } from 'chokidar'

‎pnpm-lock.yaml

+57
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

0 commit comments

Comments
 (0)
Please sign in to comment.