-
Notifications
You must be signed in to change notification settings - Fork 56
/
eslint.config.mjs
349 lines (337 loc) · 13.9 KB
/
eslint.config.mjs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
import process from 'node:process';
import babelEslintParser from '@babel/eslint-parser';
import typescriptEslintParser from '@typescript-eslint/parser';
import typescriptEslintPlugin from '@typescript-eslint/eslint-plugin';
import eslintPluginReact from 'eslint-plugin-react';
import eslintPluginImport from 'eslint-plugin-import';
import globals from 'globals';
const commitLint = process.env.COMMITLINT === '1';
const ignores = [
'node_modules/**',
'.yarn/**',
'.husky/**',
'.pnp.js',
'.pnp.cjs',
'packages/**/.sweet/dll/**',
'./app',
'./build',
'lib',
'dist',
'www',
'packages/qqtools/src/QQ/sdk/NIM_Web_SDK_v8.0.0.js',
'packages/qqtools/src/QQ/sdk/eval.js',
'packages/qqtools/src/QQ/sdk/appKey.mjs',
'packages/qqtools/src/QQ/sdk/1',
'packages/qqtools/src/QQ/sdk/bdms.js',
'packages/qqtools/src/utils/toutiao/Signer.js',
'packages/NIMTest/src/sdk/eval.js',
'packages/NIMTest/src/sdk/appKey.mjs',
'packages/NIMTest/src/sdk/NIM_Web_SDK_v8.0.0.js',
'packages/NIMTest/node/NIM_Web_SDK_nodejs_v7.1.0.js',
'packages/NIMTest/node/eval.js',
'qqapi.wk.js'
];
function globalsSettings(keys) {
const globalsObject = {};
keys.forEach((key) => Object.assign(globalsObject, globals[key]));
for (let i = 15; i <= 25; i++) {
Object.assign(globalsObject, globals[`es20${ i }`]);
}
return globalsObject;
}
const languageGlobalsOptions = {
NodeJS: 'readonly',
NodeRequire: 'readonly',
...globalsSettings(['browser', 'chai', 'commonjs', 'es5', 'mocha', 'node', 'nodeBuiltin', 'worker'])
};
const plugins = {
react: eslintPluginReact,
import: eslintPluginImport
};
const settings = {
react: {
version: 'detect'
},
'import/parsers': {
'@typescript-eslint/parser': ['.ts', '.tsx', '.mts', '.cts']
},
'import/resolver': {
typescript: {
alwaysTryTypes: true,
project: 'tsconfig.json'
},
node: {
extensions: ['.js', '.jsx', '.mjs', '.cjs', '.ts', '.tsx', '.mts', '.cts']
}
}
};
const eslintRules = {
// Possible Errors
'no-cond-assign': ['error', 'always'], // 禁止条件表达式中出现赋值操作符
'no-constant-condition': [ // 禁止在条件中使用常量表达式
'error',
{ checkLoops: true }
],
'no-dupe-args': 'error', // 禁止 function 定义中出现重名参数
'no-dupe-keys': 'error', // 禁止对象字面量中出现重复的 key
'no-duplicate-case': 'error', // 禁止出现重复的 case 标签
'no-ex-assign': 'error', // 禁止对 catch 子句的参数重新赋值
'no-extra-boolean-cast': 'error', // 禁止不必要的布尔转换
'no-extra-semi': 'error', // 禁止不必要的分号
'no-func-assign': 'error', // 禁止对 function 声明重新赋值
'no-inner-declarations': 'error', // 禁止在嵌套的块中出现变量声明或 function 声明
'no-invalid-regexp': 'error', // 禁止 RegExp 构造函数中存在无效的正则表达式字符串
'no-obj-calls': 'error', // 禁止把全局对象作为函数调用
'no-prototype-builtins': 'error', // 禁止直接调用 Object.prototypes 的内置属性
'no-sparse-arrays': 'error', // 禁用稀疏数组
'no-unexpected-multiline': 'error', // 禁止出现令人困惑的多行表达式
'use-isnan': 'error', // 要求使用 isNaN() 检查 NaN
'valid-typeof': 'error', // 强制 typeof 表达式与有效的字符串进行比较
// Best Practices
'block-scoped-var': 'error', // 强制把变量的使用限制在其定义的作用域范围内
curly: ['error', 'multi-line'], // 要求遵循大括号约定
'dot-location': ['error', 'property'], // 强制在点号之前和之后一致的换行
eqeqeq: ['error', 'always'], // 要求使用 === 和 !==
'no-empty-function': 'error', // 禁止出现空函数
'no-eval': 'error', // 禁用 eval()
'no-implied-eval': 'error', // 禁用隐式的eval()
'no-multi-spaces': [ // 禁止出现多个空格
'error',
{ ignoreEOLComments: true }
],
'no-new-func': 'error', // 禁止对 Function 对象使用 new 操作符
'no-new-wrappers': 'error', // 禁止对 String,Number 和 Boolean 使用 new 操作符
'no-param-reassign': 'error', // 禁止对 function 的参数进行重新赋值
'no-redeclare': 'error', // 禁止多次声明同一变量
'no-script-url': 'error', // 禁止使用 javascript: url
'no-self-assign': 'error', // 禁止自我赋值
'no-self-compare': 'error', // 禁止自身比较
'no-sequences': 'error', // 禁用逗号操作符
'require-await': 'error', // 禁止使用不带 await 表达式的 async 函数
// Variables
'no-delete-var': 'error', // 禁止删除变量
'no-label-var': 'error', // 禁用与变量同名的标签
'no-shadow': [ // 禁止变量声明覆盖外层作用域的变量
'error',
{ hoist: 'all' }
],
'no-undef': 'error', // 禁用未声明的变量
'no-use-before-define': 'error', // 禁止定义前使用
// Node.js and CommonJS
'no-new-require': 'error', // 禁止调用 require 时使用 new 操作符
// Stylistic Issues
'array-bracket-spacing': 'error', // 强制数组方括号中使用一致的空格
'block-spacing': 'error', // 禁止或强制在代码块中开括号前和闭括号后有空格
'brace-style': 'error', // 强制在代码块中使用一致的大括号风格
'comma-dangle': ['error', 'never'], // 要求或禁止末尾逗号
'comma-spacing': 'error', // 强制在逗号前后使用一致的空格
indent: ['error', 2, { SwitchCase: 1 }], // 强制使用一致的缩进
'jsx-quotes': 'error', // 强制在 JSX 属性中一致地使用双引号或单引号
'key-spacing': [ // 强制在对象字面量的属性中键和值之间使用一致的间距
'error',
{
beforeColon: false,
afterColon: true,
mode: 'strict'
}
],
'keyword-spacing': 'error', // 强制在关键字前后使用一致的空格
'no-mixed-operators': 'error', // 禁止混合使用不同的操作符
'no-mixed-spaces-and-tabs': 'error', // 禁止空格和 tab 的混合缩进
'no-multiple-empty-lines': [
'error',
{
max: 2,
maxEOF: 0,
maxBOF: 0
}
],
'no-new-object': 'error', // 禁用 Object 的构造函数
'no-tabs': 'error', // 禁用 tab
'no-trailing-spaces': [ // 禁用行尾空格
'error',
{
skipBlankLines: true,
ignoreComments: true
}
],
'no-whitespace-before-property': 'error', // 禁止属性前有空白
'object-curly-spacing': ['error', 'always'], // 强制在大括号中使用一致的空格
'operator-linebreak': ['error', 'before'], // 强制操作符使用一致的换行符
'padding-line-between-statements': [ // 要求或禁止在语句间填充空行
'error',
{
blankLine: 'always',
prev: ['const', 'let', 'var'],
next: '*'
},
{
blankLine: 'any',
prev: ['const', 'let', 'var'],
next: ['const', 'let', 'var']
},
{
blankLine: 'always',
prev: '*',
next: 'return'
}
],
quotes: ['error', 'single', { avoidEscape: true }], // 强制使用一致的反勾号、双引号或单引号
semi: ['error', 'always'], // 要求或禁止使用分号代替 ASI
'space-before-blocks': ['error', 'always'], // 强制在块之前使用一致的空格
'space-before-function-paren': [ // 强制在 function 的左括号之前使用一致的空格
'error',
{
anonymous: 'never',
named: 'never',
asyncArrow: 'always'
}
],
'space-infix-ops': 'error', // 要求操作符周围有空格
'space-unary-ops': ['error', { words: true }], // 要求或禁止在一元操作符之前或之后存在空格
'spaced-comment': 'error', // 强制在注释中 // 或 /* 使用一致的空格
// ECMAScript 6
'arrow-parens': ['error', 'always'], // 要求箭头函数的参数使用圆括号
'arrow-spacing': [ // 强制箭头函数的箭头前后使用一致的空格
'error',
{
before: true,
after: true
}
],
'constructor-super': 'error', // 要求在构造函数中有 super() 的调用
'no-this-before-super': 'error', // 禁止在构造函数中,在调用 super() 之前使用 this 或 super
'no-var': 'error', // 要求使用 let 或 const 而不是 var
'prefer-const': 'error', // 要求使用 const 声明那些声明后不再被修改的变量
'object-shorthand': 'error', // 要求或禁止对象字面量中方法和属性使用简写语法
'require-yield': 'error', // 要求 generator 函数内有 yield
'template-curly-spacing': ['error', 'always'], // 要求或禁止模板字符串中的嵌入表达式周围空格的使用
// React
'react/button-has-type': 'error', // 禁止<button>元素没有显式的"type"属性
'react/void-dom-elements-no-children': 'error', // 防止接收子节点的无效DOM元素(例如<img />,<br />)
// JSX
'react/jsx-boolean-value': ['error', 'always'], // 在JSX中使用布尔属性时,可以将属性值设置为true或省略值
'react/jsx-closing-tag-location': 'error', // 验证JSX中的结束标签位置
'react/jsx-curly-brace-presence': ['error', 'never'], // 强制使用大括号或不使用不必要的大括号
'react/jsx-curly-spacing': [ // 在JSX属性和表达式中强制或禁止花括号内的空格
'error',
{
when: 'always',
spacing: { objectLiterals: 'never' }
}
],
'react/jsx-equals-spacing': 'error', // 强制或禁止JSX属性中的等号周围的空格
'react/jsx-first-prop-new-line': ['error', 'never'], // 此规则检查所有JSX元素的第一个属性是否正确放置
'react/jsx-indent': ['error', 2], // 验证JSX缩进
'react/jsx-indent-props': ['error', 2], // 验证JSX中props的缩进
'react/jsx-key': 'error', // 在数组或迭代器中验证JSX具有key属性
'react/jsx-no-comment-textnodes': 'error', // 防止将注释插入为文本节点
'react/jsx-no-duplicate-props': 'error', // 在JSX中防止重复的props
'react/jsx-no-target-blank': [ // 创建具有标记的JSX元素时,通常需要使用target="_blank"属性在新选项卡中打开链接
'error',
{ enforceDynamicLinks: 'never' }
],
'react/jsx-props-no-multi-spaces': 'error', // 禁止内联JSX之间的多个空格
'react/jsx-tag-spacing': [ // 验证JSX左右括号中的空格
'error',
{
closingSlash: 'never',
beforeSelfClosing: 'always',
afterOpening: 'never',
beforeClosing: 'never'
}
],
// import
'import/no-unresolved': [ // 确保导入的模块可以解析为本地文件系统上的模块
commitLint ? 'error' : 'off',
{
commonjs: true,
ignore: ['^worker-loader!', '^@qqtools-api/', '^@qqtools-types/']
}
]
};
const eslintTypescriptRules = {
// Supported Rules
'@typescript-eslint/explicit-function-return-type': 'error', // 函数必须返回值
'@typescript-eslint/member-delimiter-style': [ // 在接口和类型文字中强制使用一致的成员定界符样式
'error',
{
multiline: { delimiter: 'semi', requireLast: true },
singleline: { delimiter: 'semi', requireLast: false }
}
],
'@typescript-eslint/no-empty-interface': 'error', // 禁止空接口
'@typescript-eslint/no-for-in-array': 'error', // 禁止使用for-in循环遍历数组
'@typescript-eslint/type-annotation-spacing': [ // 在类型注释周围需要一致的间距
'error',
{
before: true,
after: true,
overrides: {
colon: { before: false, after: true }
}
}
],
'@typescript-eslint/typedef': [ // 需要存在类型定义
'error',
{
arrayDestructuring: true,
arrowParameter: true,
memberVariableDeclaration: true,
objectDestructuring: true,
parameter: true,
propertyDeclaration: true,
variableDeclaration: true
}
],
// Extension Rules
'@typescript-eslint/no-array-constructor': 'error', // 禁止使用new Array(),但是可以使用new Array<type>()
'no-shadow': 'off',
'@typescript-eslint/no-shadow': [ // 禁止变量声明覆盖外层作用域的变量
'error',
{ hoist: 'all' }
],
'no-use-before-define': 'off',
'@typescript-eslint/no-use-before-define': 'error' // 禁止定义前使用
};
export default [
{ ignores },
{
files: ['**/*.ts', '**/*.tsx', '**/*.mts', '**/*.cts'],
languageOptions: {
parser: typescriptEslintParser,
parserOptions: {
project: 'tsconfig.json',
createDefaultProgram: true,
sourceType: 'module'
},
globals: languageGlobalsOptions
},
settings,
plugins: {
...plugins,
'@typescript-eslint': typescriptEslintPlugin
},
rules: { ...eslintRules, ...eslintTypescriptRules }
},
{
files: ['**/*.js', '**/*.jsx', '**/*.mjs', '**/*.cjs'],
languageOptions: {
parser: babelEslintParser,
parserOptions: {
requireConfigFile: false,
babelOptions: {
presets: [[
'@sweet-milktea/babel-preset-sweet',
{ env: { ecmascript: true } }
]]
},
sourceType: 'module'
},
globals: languageGlobalsOptions
},
settings,
plugins,
rules: eslintRules
}
];