@git-validator/eslint-config
Version:
A strict eslint config for better code quality.
394 lines (384 loc) • 14.1 kB
text/typescript
import gitValidatorPlugin from "@git-validator/eslint-plugin";
import confusingKeys from "confusing-browser-globals";
import fpPlugin from "eslint-plugin-fp";
import importPlugin from "eslint-plugin-import";
import nPlugin from "eslint-plugin-n";
import promisePlugin from "eslint-plugin-promise";
import reactPlugin from "eslint-plugin-react";
import reactHooksPlugin from "eslint-plugin-react-hooks";
import simpleImportSortPlugin from "eslint-plugin-simple-import-sort";
import unicornPlugin from "eslint-plugin-unicorn";
import globals from "globals";
// copied from https://github.com/standard/eslint-config-standard/blob/master/src/index.ts
// prettier-ignore
const standardConfigRules = {
'no-var': 'error',
'object-shorthand': ['error', 'properties'],
'accessor-pairs': ['error', { setWithoutGet: true, enforceForClassMembers: true }],
// 'array-bracket-spacing': ['error', 'never'],
'array-callback-return': ['error', {
allowImplicit: false,
checkForEach: false
}],
// 'arrow-spacing': ['error', { before: true, after: true }],
// 'block-spacing': ['error', 'always'],
// 'brace-style': ['error', '1tbs', { allowSingleLine: true }],
camelcase: ['error', {
allow: ['^UNSAFE_'],
properties: 'never',
ignoreGlobals: true
}],
// 'comma-dangle': ['error', {
// arrays: 'never',
// objects: 'never',
// imports: 'never',
// exports: 'never',
// functions: 'never'
// }],
// 'comma-spacing': ['error', { before: false, after: true }],
// 'comma-style': ['error', 'last'],
// 'computed-property-spacing': ['error', 'never', { enforceForClassMembers: true }],
'constructor-super': 'error',
// curly: ['error', 'multi-line'],
'default-case-last': 'error',
// 'dot-location': ['error', 'property'],
// 'dot-notation': ['error', { allowKeywords: true }], // TODO: This should be enabled. Disable it as it conflicts with ts when enabling ts-check
// 'eol-last': 'error',
eqeqeq: ['error', 'always', { null: 'ignore' }],
// 'func-call-spacing': ['error', 'never'],
// 'generator-star-spacing': ['error', { before: true, after: true }],
// indent: ['error', 2, {
// SwitchCase: 1,
// VariableDeclarator: 1,
// outerIIFEBody: 1,
// MemberExpression: 1,
// FunctionDeclaration: { parameters: 1, body: 1 },
// FunctionExpression: { parameters: 1, body: 1 },
// CallExpression: { arguments: 1 },
// ArrayExpression: 1,
// ObjectExpression: 1,
// ImportDeclaration: 1,
// flatTernaryExpressions: false,
// ignoreComments: false,
// ignoredNodes: [
// 'TemplateLiteral *',
// 'JSXElement',
// 'JSXElement > *',
// 'JSXAttribute',
// 'JSXIdentifier',
// 'JSXNamespacedName',
// 'JSXMemberExpression',
// 'JSXSpreadAttribute',
// 'JSXExpressionContainer',
// 'JSXOpeningElement',
// 'JSXClosingElement',
// 'JSXFragment',
// 'JSXOpeningFragment',
// 'JSXClosingFragment',
// 'JSXText',
// 'JSXEmptyExpression',
// 'JSXSpreadChild'
// ],
// offsetTernaryExpressions: true
// }],
// 'key-spacing': ['error', { beforeColon: false, afterColon: true }],
// 'keyword-spacing': ['error', { before: true, after: true }],
'lines-between-class-members': ['error', 'always', { exceptAfterSingleLine: true }],
// 'multiline-ternary': ['error', 'always-multiline'],
'new-cap': ['error', { newIsCap: true, capIsNew: false, properties: true }],
// 'new-parens': 'error',
'no-array-constructor': 'error',
'no-async-promise-executor': 'error',
'no-caller': 'error',
'no-case-declarations': 'error',
'no-class-assign': 'error',
'no-compare-neg-zero': 'error',
'no-cond-assign': 'error',
'no-const-assign': 'error',
'no-constant-condition': ['error', { checkLoops: false }],
'no-control-regex': 'error',
'no-debugger': 'error',
'no-delete-var': 'error',
'no-dupe-args': 'error',
'no-dupe-class-members': 'error',
'no-dupe-keys': 'error',
'no-duplicate-case': 'error',
'no-useless-backreference': 'error',
'no-empty': ['error', { allowEmptyCatch: true }],
'no-empty-character-class': 'error',
'no-empty-pattern': 'error',
'no-eval': 'error',
'no-ex-assign': 'error',
'no-extend-native': 'error',
'no-extra-bind': 'error',
'no-extra-boolean-cast': 'error',
// 'no-extra-parens': ['error', 'functions'],
'no-fallthrough': 'error',
// 'no-floating-decimal': 'error',
'no-func-assign': 'error',
'no-global-assign': 'error',
'no-implied-eval': 'error',
'no-import-assign': 'error',
'no-invalid-regexp': 'error',
'no-irregular-whitespace': 'error',
'no-iterator': 'error',
'no-labels': ['error', { allowLoop: false, allowSwitch: false }],
'no-lone-blocks': 'error',
'no-loss-of-precision': 'error',
'no-misleading-character-class': 'error',
'no-prototype-builtins': 'error',
'no-useless-catch': 'error',
// 'no-mixed-operators': ['error', {
// groups: [
// ['==', '!=', '===', '!==', '>', '>=', '<', '<='],
// ['&&', '||'],
// ['in', 'instanceof']
// ],
// allowSamePrecedence: true
// }],
// 'no-mixed-spaces-and-tabs': 'error',
// 'no-multi-spaces': 'error',
'no-multi-str': 'error',
// 'no-multiple-empty-lines': ['error', { max: 1, maxBOF: 0, maxEOF: 0 }],
'no-new': 'error',
'no-new-func': 'error',
'no-new-object': 'error',
'no-new-symbol': 'error',
'no-new-wrappers': 'error',
'no-obj-calls': 'error',
'no-octal': 'error',
'no-octal-escape': 'error',
'no-proto': 'error',
'no-redeclare': ['error', { builtinGlobals: false }],
'no-regex-spaces': 'error',
'no-return-assign': ['error', 'except-parens'],
'no-self-assign': ['error', { props: true }],
'no-self-compare': 'error',
'no-sequences': 'error',
'no-shadow-restricted-names': 'error',
'no-sparse-arrays': 'error',
// 'no-tabs': 'error',
'no-template-curly-in-string': 'error',
'no-this-before-super': 'error',
'no-throw-literal': 'error',
// 'no-trailing-spaces': 'error',
'no-undef': 'error',
'no-undef-init': 'error',
// 'no-unexpected-multiline': 'error',
'no-unmodified-loop-condition': 'error',
'no-unneeded-ternary': ['error', { defaultAssignment: false }],
'no-unreachable': 'error',
'no-unreachable-loop': 'error',
'no-unsafe-finally': 'error',
'no-unsafe-negation': 'error',
'no-unused-expressions': ['error', {
allowShortCircuit: true,
allowTernary: true,
allowTaggedTemplates: true
}],
'no-unused-vars': ['error', {
args: 'none',
caughtErrors: 'none',
ignoreRestSiblings: true,
vars: 'all'
}],
'no-use-before-define': ['error', { functions: false, classes: false, variables: false }],
'no-useless-call': 'error',
'no-useless-computed-key': 'error',
'no-useless-constructor': 'error',
'no-useless-escape': 'error',
'no-useless-rename': 'error',
'no-useless-return': 'error',
'no-void': 'error',
// 'no-whitespace-before-property': 'error',
'no-with': 'error',
// 'object-curly-newline': ['error', { multiline: true, consistent: true }],
// 'object-curly-spacing': ['error', 'always'],
// 'object-property-newline': ['error', { allowMultiplePropertiesPerLine: true }],
'one-var': ['error', { initialized: 'never' }],
// 'operator-linebreak': ['error', 'after', { overrides: { '?': 'before', ':': 'before', '|>': 'before' } }],
// 'padded-blocks': ['error', { blocks: 'never', switches: 'never', classes: 'never' }],
'prefer-const': ['error', { destructuring: 'all' }],
'prefer-promise-reject-errors': 'error',
'prefer-regex-literals': ['error', { disallowRedundantWrapping: true }],
// 'quote-props': ['error', 'as-needed'],
// quotes: ['error', 'single', { avoidEscape: true, allowTemplateLiterals: false }],
// 'rest-spread-spacing': ['error', 'never'],
// semi: ['error', 'never'],
// 'semi-spacing': ['error', { before: false, after: true }],
// 'space-before-blocks': ['error', 'always'],
// 'space-before-function-paren': ['error', 'always'],
// 'space-in-parens': ['error', 'never'],
// 'space-infix-ops': 'error',
// 'space-unary-ops': ['error', { words: true, nonwords: false }],
'spaced-comment': ['error', 'always', {
line: { markers: ['*package', '!', '/', ',', '='] },
block: { balanced: true, markers: ['*package', '!', ',', ':', '::', 'flow-include'], exceptions: ['*'] }
}],
'symbol-description': 'error',
// 'template-curly-spacing': ['error', 'never'],
// 'template-tag-spacing': ['error', 'never'],
'unicode-bom': ['error', 'never'],
'use-isnan': ['error', {
enforceForSwitchCase: true,
enforceForIndexOf: true
}],
'valid-typeof': ['error', { requireStringLiterals: true }],
// 'wrap-iife': ['error', 'any', { functionPrototypeMethods: true }],
// 'yield-star-spacing': ['error', 'both'],
yoda: ['error', 'never'],
'import/export': 'error',
'import/first': 'error',
'import/no-absolute-path': ['error', { esmodule: true, commonjs: true, amd: false }],
'import/no-duplicates': 'error',
'import/no-named-default': 'error',
'import/no-webpack-loader-syntax': 'error',
'n/handle-callback-err': ['error', '^(err|error)$'],
'n/no-callback-literal': 'error',
'n/no-deprecated-api': 'error',
'n/no-exports-assign': 'error',
'n/no-new-require': 'error',
'n/no-path-concat': 'error',
'n/process-exit-as-throw': 'error',
'promise/param-names': 'error'
};
export default {
files: ["**/*.{js,cjs,mjs,jsx}"],
languageOptions: {
parserOptions: {
ecmaFeatures: {
jsx: true,
},
},
globals: {
// TODO Optimize it. Node code should not use browser's objects.
...Object.fromEntries(
Object.entries(globals.browser).filter(
([k]) => !confusingKeys.includes(k),
),
),
},
},
linterOptions: {
// noInlineConfig: true, // too strict
reportUnusedDisableDirectives: true,
},
plugins: {
fp: fpPlugin,
n: nPlugin,
import: importPlugin,
promise: promisePlugin,
react: reactPlugin,
"react-hooks": reactHooksPlugin,
unicorn: unicornPlugin,
"simple-import-sort": simpleImportSortPlugin,
"@git-validator": gitValidatorPlugin,
},
/**
* @type {Record<string, unknown>}
*/
rules: {
// 1. standard config rules
...standardConfigRules,
// 2. code style for a better readability
"arrow-body-style": ["error", "as-needed"],
"import/newline-after-import": ["error", { count: 1 }],
"simple-import-sort/imports": [
"error",
{ groups: [["^\\u0000", "^node:", "^@?\\w", "^", "^\\."]] },
],
"simple-import-sort/exports": "error",
"unicorn/prefer-node-protocol": "error",
"unicorn/escape-case": "error", // '\ud834' -> '\uD834'
// "unicorn/number-literal-case": "error", // 0XFF -> 0xFF // conflict with prettier
// 3. ban some syntaxes to reduce mistakes
"func-name-matching": "error",
"init-declarations": "error",
"max-params": ["error", { max: 4 }],
"no-duplicate-imports": "error",
"no-implicit-coercion": [
"error",
{ disallowTemplateShorthand: true, allow: ["!!"] },
], // forbid code like `const num = +str`;
"no-invalid-this": "error",
"no-multi-assign": "error",
"no-plusplus": "error",
"no-shadow": ["error", { ignoreOnInitialization: true }],
"no-unused-private-class-members": "error",
"prefer-exponentiation-operator": "error",
"prefer-object-has-own": "error",
// fp
"fp/no-arguments": "error",
"fp/no-delete": "error",
// import
"import/extensions": ["error", "always", { ignorePackages: true }],
"import/no-commonjs": [
"error",
{
allowRequire: false,
allowConditionalRequire: false,
allowPrimitiveModules: false,
},
],
"import/no-dynamic-require": "error",
"import/no-mutable-exports": "error", // forbid code like `export let count = 3`
"import/no-relative-packages": "error", // forbid to import module from other monorepo packages by relative paths
"import/no-self-import": "error",
// n
"n/no-sync": "error",
"n/prefer-global/process": ["error", "never"],
"n/prefer-global/buffer": ["error", "never"],
// react
"react/jsx-key": "error",
"react/jsx-no-duplicate-props": "error",
"react/jsx-no-undef": "error",
// react-hooks
"react-hooks/exhaustive-deps": "error",
"react-hooks/rules-of-hooks": "error",
// unicorn
// 'unicorn/no-null': 'error', // null can be useful when interact with json.
"unicorn/consistent-destructuring": "error",
"unicorn/error-message": "error",
"unicorn/filename-case": [
"error",
{ cases: { kebabCase: true, pascalCase: true } },
],
// TODO remove the options when https://github.com/sindresorhus/eslint-plugin-unicorn/pull/2297 merged
"unicorn/import-style": [
"error",
{
styles: {
"node:path": {
default: true,
},
"node:util": {
named: true,
},
},
},
],
"unicorn/new-for-builtins": "error",
"unicorn/no-abusive-eslint-disable": "error",
"unicorn/no-array-callback-reference": "error",
"unicorn/no-for-loop": "error",
"unicorn/no-instanceof-array": "error",
"unicorn/no-new-array": "error",
"unicorn/no-new-buffer": "error",
"unicorn/no-typeof-undefined": "error",
"unicorn/no-unreadable-iife": "error",
"unicorn/prefer-array-flat-map": "error",
"unicorn/prefer-module": "error",
"unicorn/prefer-number-properties": "error",
"unicorn/prefer-string-slice": "error",
"unicorn/throw-new-error": "error",
"@git-validator/ban-ts-comment": "error",
"@git-validator/import-regex": "error",
"@git-validator/new-parens": "error",
"@git-validator/no-dynamic-import": "error",
"@git-validator/no-for-in": "error",
"@git-validator/no-instanceof-builtin": "error",
"@git-validator/no-legacy-getter-setter": "error",
"@git-validator/prefer-global-this": "error",
"@git-validator/prefer-shortest-relative-path": "error",
},
};