From e7c2dff4d1977cad09eabc4673fa8623bbe2a6ba Mon Sep 17 00:00:00 2001 From: "Alexander S." Date: Sun, 5 Jan 2025 17:36:26 +0100 Subject: [PATCH] build: use `oxlint --rules --format=json` for building rules (#287) --- .github/workflows/bump_oxlint.yml | 3 +- .github/workflows/generate.yml | 3 - README.md | 14 +- package.json | 1 - scripts/constants.ts | 21 --- scripts/generate.ts | 8 +- scripts/sparse-clone.ts | 95 ---------- scripts/traverse-rules.test.ts | 237 ----------------------- scripts/traverse-rules.ts | 289 +++++++---------------------- src/generated/rules-by-category.ts | 56 +++--- src/generated/rules-by-scope.ts | 70 +++---- 11 files changed, 142 insertions(+), 655 deletions(-) delete mode 100644 scripts/sparse-clone.ts delete mode 100644 scripts/traverse-rules.test.ts diff --git a/.github/workflows/bump_oxlint.yml b/.github/workflows/bump_oxlint.yml index bd5a879..3970012 100644 --- a/.github/workflows/bump_oxlint.yml +++ b/.github/workflows/bump_oxlint.yml @@ -25,8 +25,7 @@ jobs: OXLINT_VERSION: ${{ inputs.version }} run: | pnpm install oxlint@${OXLINT_VERSION} - pnpm run clone ${OXLINT_VERSION} - pnpm run generate # Generate rules from source code + pnpm run generate # Generate rules pnpm run format # run prettier over it - name: Test and update snapshot diff --git a/.github/workflows/generate.yml b/.github/workflows/generate.yml index ca10366..d03024f 100644 --- a/.github/workflows/generate.yml +++ b/.github/workflows/generate.yml @@ -28,9 +28,6 @@ jobs: - uses: ./.github/actions/pnpm - - name: Clone oxc_linter project - run: pnpm run clone - - name: Remove current generated code run: rm -r ./src/generated/ diff --git a/README.md b/README.md index c3e18c6..f2b3b35 100644 --- a/README.md +++ b/README.md @@ -132,17 +132,21 @@ You need to install both the [oxc](https://marketplace.visualstudio.com/items?it ## Contributing -sparse clone the oxlint repository to have a local copy +### Generate rules + +Generates the rules from installed oxlint version ```shell -pnpm clone +pnpm generate +pnpm format ``` -generates the rules from the sparse cloned Rust library, only for the latest version, -new rules that haven't been released will not be included. +### Test + +Tests the source code ```shell -pnpm generate +pnpm test ``` ## License diff --git a/package.json b/package.json index d98dcea..aafcdb6 100644 --- a/package.json +++ b/package.json @@ -39,7 +39,6 @@ "scripts": { "prepare": "husky", "generate": "node --import @oxc-node/core/register ./scripts/generate.ts", - "clone": "node --import @oxc-node/core/register ./scripts/sparse-clone.ts", "build": "vite build", "lint": "npx oxlint --tsconfig=tsconfig.json && npx eslint --flag unstable_ts_config", "format": "npx prettier --write .", diff --git a/scripts/constants.ts b/scripts/constants.ts index 8dd0572..368e165 100644 --- a/scripts/constants.ts +++ b/scripts/constants.ts @@ -1,26 +1,5 @@ -import path from 'node:path'; -import { aliasPluginNames } from '../src/constants.js'; - -const __dirname = new URL('.', import.meta.url).pathname; - -export const TARGET_DIRECTORY = path.resolve(__dirname, '..', '.oxc_sparse'); -export const VERSION_PREFIX = 'oxlint_v'; -export const SPARSE_CLONE_DIRECTORY = 'crates/oxc_linter/src'; - // these are the rules that don't have a direct equivalent in the eslint rules export const ignoreScope = new Set(['oxc', 'deepscan', 'security']); // these are the rules that are not fully implemented in oxc export const ignoreCategories = new Set(['nursery']); - -export function convertScope(scope: string) { - return Reflect.has(aliasPluginNames, scope) - ? aliasPluginNames[scope as 'eslint'] - : scope.replace('_', '-'); -} - -export function prefixScope(scope: string) { - const _scope = convertScope(scope); - - return _scope ? `${_scope}/` : ''; -} diff --git a/scripts/generate.ts b/scripts/generate.ts index 25914cd..4afd85f 100644 --- a/scripts/generate.ts +++ b/scripts/generate.ts @@ -6,13 +6,7 @@ import { traverseRules } from './traverse-rules.js'; const __dirname = new URL('.', import.meta.url).pathname; -const { successResultArray, failureResultArray } = await traverseRules(); - -if (failureResultArray.length > 0) { - console.error( - `Failed to generate rules for the following rules ${JSON.stringify(failureResultArray)}` - ); -} +const successResultArray = traverseRules(); const rulesGenerator = new RulesGenerator(successResultArray); const configGenerator = new ConfigGenerator(successResultArray); diff --git a/scripts/sparse-clone.ts b/scripts/sparse-clone.ts deleted file mode 100644 index 03c7201..0000000 --- a/scripts/sparse-clone.ts +++ /dev/null @@ -1,95 +0,0 @@ -import shell from 'shelljs'; -import fs from 'node:fs'; -import { - TARGET_DIRECTORY, - SPARSE_CLONE_DIRECTORY, - VERSION_PREFIX, -} from './constants.js'; -import packageJson from '../package.json' with { type: 'json' }; - -/** - * Run this file in CLI like `pnpm run clone` - * It clones the oxc_linter git project into your local file. - * - * You can run it with a version argument like `pnpm run clone 0.10.0`. - * When no argument is provided, the current package.json version is used. - * - */ - -const checkoutVersion = process.argv[2] ?? packageJson.version; - -// Function to initialize or reconfigure sparse-checkout -function configureSparseCheckout(cloneDirectory: string) { - console.log('Configuring sparse-checkout...'); - // Initialize sparse-checkout if not already initialized - if ( - !fs.existsSync('.oxc_sparse/.git/info/sparse-checkout') && - shell.exec('git sparse-checkout init --cone').code !== 0 - ) { - shell.echo('Error: Failed to initialize sparse-checkout'); - shell.exit(1); - } - - // Set the directory to be checked out - if (shell.exec(`git sparse-checkout set ${cloneDirectory}`).code !== 0) { - shell.echo('Error: Failed to configure sparse-checkout'); - shell.exit(1); - } -} - -function checkoutVersionTag(version: string) { - const tag = `${VERSION_PREFIX}${version}`; - - // Checkout the specified directory - if (shell.exec(`git checkout ${tag}`, { silent: true }).code !== 0) { - shell.echo('Error: Git checkout failed'); - shell.exit(1); - } - - console.log(`Successfully checkout git tag ${tag}`); -} - -// Function to clone or update a repository -function cloneOrUpdateRepo( - repositoryUrl: string, - targetDirectory: string, - cloneDirectory: string, - version: string -) { - // Check if the target directory exists and is a Git repository - if ( - fs.existsSync(targetDirectory) && - fs.existsSync(`${targetDirectory}/.git`) - ) { - console.log(`Repository exists, updating ${targetDirectory}...`); - - shell.cd(targetDirectory); - - configureSparseCheckout(cloneDirectory); - checkoutVersionTag(version); - } else { - console.log(`Cloning new repository into ${targetDirectory}...`); - - // Clone the repository without checking out files - if ( - shell.exec( - `git clone --filter=blob:none --no-checkout ${repositoryUrl} ${targetDirectory}` - ).code !== 0 - ) { - shell.echo('Error: Git clone failed'); - shell.exit(1); - } - - shell.cd(targetDirectory); - - configureSparseCheckout(cloneDirectory); - checkoutVersionTag(version); - } -} - -cloneOrUpdateRepo( - 'https://github.com/oxc-project/oxc.git', - TARGET_DIRECTORY, - SPARSE_CLONE_DIRECTORY, - checkoutVersion -); diff --git a/scripts/traverse-rules.test.ts b/scripts/traverse-rules.test.ts deleted file mode 100644 index 850108c..0000000 --- a/scripts/traverse-rules.test.ts +++ /dev/null @@ -1,237 +0,0 @@ -import type { Rule } from './traverse-rules.js'; -import { - getFileNameWithoutExtension, - getFolderNameUnderRules, -} from './traverse-rules.js'; -import { suite, expect, test, vi, afterEach, beforeEach } from 'vitest'; -import { readFilesRecursively } from './traverse-rules.js'; -import { type fs, vol } from 'memfs'; -import dedent from 'dedent'; - -suite('getFileNameWithoutExtension', () => { - test('getFileNameWithoutExtension returns correct file name without extension', () => { - const filePath = '/path/to/file.rs'; - const currentDirectory = '/path/to'; - const expectedFileName = 'file'; - - const result = getFileNameWithoutExtension(filePath, currentDirectory); - - expect(result).toEqual(expectedFileName); - }); - - test("getFileNameWithoutExtension returns current directory name when file name is 'mod.rs'", () => { - const filePath = '/path/to/mod.rs'; - const currentDirectory = '/path/to'; - const expectedFileName = 'to'; - - const result = getFileNameWithoutExtension(filePath, currentDirectory); - - expect(result).toEqual(expectedFileName); - }); -}); - -suite('getFolderNameUnderRules', () => { - test("getFolderNameUnderRules returns empty string when 'rules' directory not found", () => { - const filePath = '/path/to/file.ts'; - const expectedFolderName = ''; - - const result = getFolderNameUnderRules(filePath); - - expect(result).toEqual(expectedFolderName); - }); - - test("getFolderNameUnderRules returns folder name directly under 'rules'", () => { - const filePath = '/path/to/rules/folder/file.ts'; - const expectedFolderName = 'folder'; - - const result = getFolderNameUnderRules(filePath); - - expect(result).toEqual(expectedFolderName); - }); - - test("getFolderNameUnderRules returns remaining path if there's no additional '/'", () => { - const filePath = '/path/to/rules/file.ts'; - const expectedFolderName = 'file.ts'; - - const result = getFolderNameUnderRules(filePath); - - expect(result).toEqual(expectedFolderName); - }); -}); - -suite('readFilesRecursively', () => { - beforeEach(() => { - vi.mock('node:fs', async () => { - const memfs: { fs: typeof fs } = await vi.importActual('memfs'); - return { - promises: memfs.fs.promises, - }; - }); - }); - - afterEach(() => { - vol.reset(); - vi.restoreAllMocks(); - }); - - test('readFilesRecursively recursively reads files and directories', async () => { - // Prepare test data - const successResultArray: Rule[] = []; - const skippedResultArray: Rule[] = []; - const failureResultArray: Rule[] = []; - - vol.fromJSON({ - 'crates/src/rules/eslint/rulename-with-mod/mod.rs': 'content', - 'crates/src/rules/typescript/rulename-without-mod.rs': 'content', - 'crates/src/rules/oxc/rulename-which-will-be-skipped.rs': 'content', - }); - - // Call the function - await readFilesRecursively( - '.', - successResultArray, - skippedResultArray, - failureResultArray - ); - - expect(successResultArray.length).toEqual(0); - expect(skippedResultArray.length).toEqual(1); - expect(failureResultArray.length).toEqual(2); - - expect(successResultArray).toEqual([]); - expect(skippedResultArray).toEqual([ - { - category: 'unknown', - scope: 'oxc', - value: 'oxc/rulename-which-will-be-skipped', - }, - ]); - expect(failureResultArray).toContainEqual({ - category: 'unknown', - error: 'No match block for `declare_oxc_lint`', - scope: 'eslint', - value: 'rulename-with-mod', - }); - expect(failureResultArray).toContainEqual({ - category: 'unknown', - error: 'No match block for `declare_oxc_lint`', - scope: 'typescript', - value: '@typescript-eslint/rulename-without-mod', - }); - }); - - test('readFilesRecursively returns parsed rules correctly', async () => { - // Prepare test data - const successResultArray: Rule[] = []; - const skippedResultArray: Rule[] = []; - const failureResultArray: Rule[] = []; - - const ruleNameWithModuleContent = dedent(`declare_oxc_lint!( - /// Some Block Content - /// ) extra parenthesis to make sure it doesn't catch - DefaultCaseLast, - style - )`); - - const ruleNameWithoutModuleContent = dedent(`declare_oxc_lint!( - /// ### What it does - /// Disallow calling some global objects as functions - NoObjCalls, - correctness - )`); - - const ruleWithFixabilityModuleContent = dedent(`declare_oxc_lint!( - /// Some Block Content - /// ) extra parenthesis to make sure it doesn't catch - DefaultCaseLast, - correctness, - fix - )`); - - vol.fromJSON({ - 'crates/src/rules/eslint/rulename-with-mod/mod.rs': - ruleNameWithModuleContent, - 'crates/src/rules/typescript/rulename-without-mod.rs': - ruleNameWithoutModuleContent, - 'crates/src/rules/unicorn/rule-with-fixability.rs': - ruleWithFixabilityModuleContent, - }); - - // Call the function - await readFilesRecursively('.', successResultArray, [], []); - expect(successResultArray).toContainEqual({ - category: 'style', - scope: 'eslint', - value: 'rulename-with-mod', - }); - expect(successResultArray).toContainEqual({ - category: 'correctness', - scope: 'typescript', - value: '@typescript-eslint/rulename-without-mod', - }); - expect(successResultArray).toContainEqual({ - category: 'correctness', - scope: 'unicorn', - value: 'unicorn/rule-with-fixability', - }); - - expect(skippedResultArray).toEqual([]); - expect(failureResultArray).toEqual([]); - }); - - test('readFilesRecursively returns non-parsed rules correctly', async () => { - // Prepare test data - const successResultArray: Rule[] = []; - const skippedResultArray: Rule[] = []; - const failureResultArray: Rule[] = []; - - const badContent = 'bad content'; - - vol.fromJSON({ - 'crates/src/rules/eslint/rulename-that-will-be-skipped-because-no-match-block.rs': - badContent, - 'crates/src/rules/oxc/rulename-that-will-be-skipped-because-bad-content-or-scope.rs': - badContent, - 'crates/src/rules/oxc/rulename-that-will-be-skipped-because-skip-scope.rs': - badContent, - 'crates/src/rules/typescript/rulename-that-will-error.rs': badContent, - }); - - // Call the function - await readFilesRecursively( - '.', - successResultArray, - skippedResultArray, - failureResultArray - ); - - expect(successResultArray).toEqual([]); - expect(skippedResultArray).toEqual([ - { - category: 'unknown', - scope: 'oxc', - value: 'oxc/rulename-that-will-be-skipped-because-bad-content-or-scope', - }, - { - category: 'unknown', - scope: 'oxc', - value: 'oxc/rulename-that-will-be-skipped-because-skip-scope', - }, - ]); - - expect(failureResultArray).toEqual([ - { - category: 'unknown', - error: 'No match block for `declare_oxc_lint`', - scope: 'eslint', - value: 'rulename-that-will-be-skipped-because-no-match-block', - }, - { - category: 'unknown', - error: 'No match block for `declare_oxc_lint`', - scope: 'typescript', - value: '@typescript-eslint/rulename-that-will-error', - }, - ]); - }); -}); diff --git a/scripts/traverse-rules.ts b/scripts/traverse-rules.ts index 5ecffaa..ffae7c4 100644 --- a/scripts/traverse-rules.ts +++ b/scripts/traverse-rules.ts @@ -1,254 +1,101 @@ -import { promises } from 'node:fs'; -import path from 'node:path'; -import { - ignoreCategories, - ignoreScope, - prefixScope, - SPARSE_CLONE_DIRECTORY, - TARGET_DIRECTORY, -} from './constants.js'; +import { execSync } from 'node:child_process'; +import { ignoreCategories, ignoreScope } from './constants.js'; import { + aliasPluginNames, reactHookRulesInsideReactScope, typescriptRulesExtendEslintRules, viteTestCompatibleRules, } from '../src/constants.js'; -// Recursive function to read files in a directory, this currently assumes that the directory -// structure is semi-consistent within the oxc_linter crate -export async function readFilesRecursively( - directory: string, - successResultArray: Rule[], - skippedResultArray: Rule[], - failureResultArray: Rule[] -): Promise { - const entries = await promises.readdir(directory, { withFileTypes: true }); - - // Check if the current directory contains a 'mod.rs' file - // eslint-disable-next-line unicorn/prevent-abbreviations - const containsModRs = entries.some( - (entry) => entry.isFile() && entry.name === 'mod.rs' - ); - - await Promise.all( - entries.map(async (entry) => { - const entryPath = path.join(directory, entry.name); - if (entry.isDirectory()) { - await readFilesRecursively( - entryPath, - successResultArray, - skippedResultArray, - failureResultArray - ); // Recursive call for directories - } else if ( - entry.isFile() && - (!containsModRs || entry.name === 'mod.rs') - ) { - await processFile( - entryPath, - directory, - successResultArray, - skippedResultArray, - failureResultArray - ); // Process each file - } - }) - ); -} - -export interface Rule { +export type Rule = { value: string; scope: string; category: string; - error?: string; -} - -// Function to process each file and extract the desired word -async function processFile( - filePath: string, - currentDirectory: string, - successResultArray: Rule[], - skippedResultArray: Rule[], - failureResultArray: Rule[] -): Promise { - const content = await promises.readFile(filePath, 'utf8'); - - // 'ok' way to get the scope, depends on the directory structure - let scope = getFolderNameUnderRules(filePath); - const shouldIgnoreRule = ignoreScope.has(scope); +}; + +/** + * Read the rules from oxlint command and returns an array of Rule-Objects + */ +function readRulesFromCommand(): Rule[] { + // do not handle the exception + const oxlintOutput = execSync(`npx oxlint --rules --format=json`, { + encoding: 'utf8', + stdio: 'pipe', + }); - // when the file is called `mod.rs` we want to use the parent directory name as the rule name - // Note that this is fairly brittle, as relying on the directory structure can be risky - const ruleNameWithoutScope = getFileNameWithoutExtension( - filePath, - currentDirectory - ).replaceAll('_', '-'); + // do not handle the exception + return JSON.parse(oxlintOutput); +} - // All rules from `eslint-plugin-react-hooks` - // Since oxlint supports these rules under react/*, we need to remap them. +/** + * Some rules are in a different scope then in eslint + */ +function fixScopeOfRule(rule: Rule): void { if ( - scope === 'react' && - reactHookRulesInsideReactScope.includes(ruleNameWithoutScope) + rule.scope === 'react' && + reactHookRulesInsideReactScope.includes(rule.value) ) { - scope = 'react_hooks'; - } - - const effectiveRuleName = - `${prefixScope(scope)}${ruleNameWithoutScope}`.replaceAll('_', '-'); - - // add the rule to the skipped array and continue to see if there's a match regardless - if (shouldIgnoreRule) { - skippedResultArray.push({ - value: effectiveRuleName, - scope: scope, - category: 'unknown', - }); - - return; - } - - // Remove comments to prevent them from affecting the regex - const cleanContent = content.replaceAll(/^\s*\/\/.*$/gm, ''); - - // find the correct macro block where `);` or `}` is the end of the block - // ensure that the `);` or `}` is on its own line, with no characters before it - const blockRegex = /declare_oxc_lint!\s*([({]([\S\s]*?)\s*[)}]\s*;?)/gm; - - const match = blockRegex.exec(cleanContent); - - if (match === null) { - failureResultArray.push({ - value: effectiveRuleName, - scope: scope, - category: 'unknown', - error: 'No match block for `declare_oxc_lint`', - }); - return; - } - - const block = match[2]; - - // Remove comments to prevent them from affecting the regex - const cleanBlock = block.replaceAll(/\/\/.*$|\/\*[\S\s]*?\*\//gm, '').trim(); - - // Extract the keyword, skipping the optional fixability metadata, - // and correctly handling optional trailing characters - // since trailing commas are optional in Rust and the last keyword may not have one - const keywordRegex = /,\s*(\w+)\s*,?\s*(?:(\w+)\s*,?\s*)?$/; - const keywordMatch = keywordRegex.exec(cleanBlock); - - if (keywordMatch === null) { - failureResultArray.push({ - value: effectiveRuleName, - scope: `unknown: ${scope}`, - category: 'unknown', - error: 'Could not extract keyword from macro block', - }); - return; + rule.scope = 'react_hooks'; } +} - if (ignoreCategories.has(keywordMatch[1])) { - skippedResultArray.push({ - value: effectiveRuleName, - scope: scope, - category: keywordMatch[1], - }); +/** + * oxlint returns the value without a scope name + */ +function fixValueOfRule(rule: Rule): void { + if (rule.scope === 'eslint') { return; } - successResultArray.push({ - value: effectiveRuleName, - scope: scope, - category: keywordMatch[1], - }); - - // special case for eslint and typescript alias rules - if (scope === 'eslint') { - const ruleName = effectiveRuleName.replace(/^.*\//, ''); + const scope = + rule.scope in aliasPluginNames ? aliasPluginNames[rule.scope] : rule.scope; - if (typescriptRulesExtendEslintRules.includes(ruleName)) { - successResultArray.push({ - value: `@typescript-eslint/${ruleName}`, - scope: 'typescript', - category: keywordMatch[1], - }); - } - - // special case for jest and vitest alias rules - } else if (scope === 'jest') { - const ruleName = effectiveRuleName.replace(/^.*\//, ''); - - if (viteTestCompatibleRules.includes(ruleName)) { - successResultArray.push({ - value: `vitest/${ruleName}`, - scope: 'vitest', - category: keywordMatch[1], - }); - } - } + rule.value = `${scope}/${rule.value}`; } -export function getFolderNameUnderRules(filePath: string) { - const sourceIndex = filePath.indexOf('/rules/'); - if (sourceIndex === -1) { - return ''; // 'rules' directory not found +/** + * some rules are reimplemented in another scope + * remap them so we can disable all the reimplemented too + */ +function getAliasRules(rule: Rule): Rule | undefined { + if ( + rule.scope === 'eslint' && + typescriptRulesExtendEslintRules.includes(rule.value) + ) { + return { + value: `@typescript-eslint/${rule.value}`, + scope: 'typescript', + category: rule.category, + }; } - // Extract the substring starting after '/src/' - const subPath = filePath.slice(Math.max(0, sourceIndex + 7)); - - // Find the next '/' to isolate the folder name directly under 'src' - const nextSlashIndex = subPath.indexOf('/'); - if (nextSlashIndex === -1) { - return subPath; // Return the remaining path if there's no additional '/' + if (rule.scope === 'jest' && viteTestCompatibleRules.includes(rule.value)) { + return { + value: `vitest/${rule.value}`, + scope: 'vitest', + category: rule.category, + }; } - - return subPath.slice(0, Math.max(0, nextSlashIndex)); } -export function getFileNameWithoutExtension( - filePath: string, - currentDirectory: string -) { - return path.basename(filePath) === 'mod.rs' - ? path.basename(currentDirectory) - : path.basename(filePath, path.extname(filePath)); -} - -export async function traverseRules(): Promise<{ - successResultArray: Rule[]; - skippedResultArray: Rule[]; - failureResultArray: Rule[]; -}> { - const successResultArray: Rule[] = []; - const skippedResultArray: Rule[] = []; - const failureResultArray: Rule[] = []; - - const startDirectory = path.join( - TARGET_DIRECTORY, - SPARSE_CLONE_DIRECTORY, - 'rules' +export function traverseRules(): Rule[] { + // get all rules and filter the ignored one + const rules = readRulesFromCommand().filter( + (rule) => + !ignoreCategories.has(rule.category) && !ignoreScope.has(rule.scope) ); - await readFilesRecursively( - startDirectory, - successResultArray, - skippedResultArray, - failureResultArray - ); - - successResultArray.sort((aRule, bRule) => { - const scopeCompare = aRule.scope.localeCompare(bRule.scope); + const aliasRules: Rule[] = []; - if (scopeCompare !== 0) { - return scopeCompare; + for (const rule of rules) { + const aliasRule = getAliasRules(rule); + if (aliasRule) { + aliasRules.push(aliasRule); } - return aRule.value.localeCompare(bRule.value); - }); - - console.log( - `>> Parsed ${successResultArray.length} rules, skipped ${skippedResultArray.length} and encountered ${failureResultArray.length} failures\n` - ); + fixScopeOfRule(rule); + fixValueOfRule(rule); + } - return { successResultArray, skippedResultArray, failureResultArray }; + return [...rules, ...aliasRules]; } diff --git a/src/generated/rules-by-category.ts b/src/generated/rules-by-category.ts index a2f5d26..9a381e9 100644 --- a/src/generated/rules-by-category.ts +++ b/src/generated/rules-by-category.ts @@ -5,6 +5,7 @@ const pedanticRules = { eqeqeq: 'off', 'max-classes-per-file': 'off', 'max-lines': 'off', + 'no-object-constructor': 'off', 'no-array-constructor': 'off', 'no-case-declarations': 'off', 'no-constructor-return': 'off', @@ -13,7 +14,6 @@ const pedanticRules = { 'no-inner-declarations': 'off', 'no-negated-condition': 'off', 'no-new-wrappers': 'off', - 'no-object-constructor': 'off', 'no-prototype-builtins': 'off', 'no-redeclare': 'off', 'no-self-compare': 'off', @@ -37,8 +37,6 @@ const pedanticRules = { 'react-hooks/rules-of-hooks': 'off', '@typescript-eslint/ban-ts-comment': 'off', '@typescript-eslint/ban-types': 'off', - '@typescript-eslint/no-array-constructor': 'off', - '@typescript-eslint/no-redeclare': 'off', '@typescript-eslint/no-unsafe-function-type': 'off', '@typescript-eslint/prefer-enum-initializers': 'off', '@typescript-eslint/prefer-ts-expect-error': 'off', @@ -78,6 +76,8 @@ const pedanticRules = { 'unicorn/prefer-string-slice': 'off', 'unicorn/prefer-type-error': 'off', 'unicorn/require-number-to-fixed-digits-argument': 'off', + '@typescript-eslint/no-array-constructor': 'off', + '@typescript-eslint/no-redeclare': 'off', 'vitest/no-conditional-in-test': 'off', } as const; @@ -87,8 +87,8 @@ const restrictionRules = { 'no-bitwise': 'off', 'no-console': 'off', 'no-div-regex': 'off', - 'no-empty': 'off', 'no-empty-function': 'off', + 'no-empty': 'off', 'no-eq-null': 'off', 'no-eval': 'off', 'no-iterator': 'off', @@ -119,7 +119,6 @@ const restrictionRules = { 'react/no-unknown-property': 'off', '@typescript-eslint/explicit-function-return-type': 'off', '@typescript-eslint/no-dynamic-delete': 'off', - '@typescript-eslint/no-empty-function': 'off', '@typescript-eslint/no-empty-object-type': 'off', '@typescript-eslint/no-explicit-any': 'off', '@typescript-eslint/no-import-type-side-effects': 'off', @@ -127,7 +126,6 @@ const restrictionRules = { '@typescript-eslint/no-non-null-asserted-nullish-coalescing': 'off', '@typescript-eslint/no-non-null-assertion': 'off', '@typescript-eslint/no-require-imports': 'off', - '@typescript-eslint/no-unused-expressions': 'off', '@typescript-eslint/no-var-requires': 'off', '@typescript-eslint/prefer-literal-enum-member': 'off', 'unicorn/no-abusive-eslint-disable': 'off', @@ -141,6 +139,8 @@ const restrictionRules = { 'unicorn/prefer-modern-math-apis': 'off', 'unicorn/prefer-node-protocol': 'off', 'unicorn/prefer-number-properties': 'off', + '@typescript-eslint/no-empty-function': 'off', + '@typescript-eslint/no-unused-expressions': 'off', } as const; const styleRules = { @@ -150,24 +150,24 @@ const styleRules = { 'guard-for-in': 'off', 'max-params': 'off', 'new-cap': 'off', - 'no-continue': 'off', - 'no-duplicate-imports': 'off', 'no-extra-label': 'off', - 'no-label-var': 'off', + 'no-multi-assign': 'off', + 'no-nested-ternary': 'off', 'no-labels': 'off', + 'no-duplicate-imports': 'off', + 'no-continue': 'off', + 'no-label-var': 'off', 'no-magic-numbers': 'off', - 'no-multi-assign': 'off', 'no-multi-str': 'off', - 'no-nested-ternary': 'off', 'no-new-func': 'off', 'no-return-assign': 'off', 'no-script-url': 'off', 'no-template-curly-in-string': 'off', 'no-ternary': 'off', + 'prefer-rest-params': 'off', 'prefer-exponentiation-operator': 'off', 'prefer-numeric-literals': 'off', 'prefer-object-has-own': 'off', - 'prefer-rest-params': 'off', 'prefer-spread': 'off', 'sort-imports': 'off', 'sort-keys': 'off', @@ -228,11 +228,8 @@ const styleRules = { '@typescript-eslint/consistent-generic-constructors': 'off', '@typescript-eslint/consistent-indexed-object-style': 'off', '@typescript-eslint/consistent-type-definitions': 'off', - '@typescript-eslint/default-param-last': 'off', - '@typescript-eslint/max-params': 'off', - '@typescript-eslint/no-empty-interface': 'off', '@typescript-eslint/no-inferrable-types': 'off', - '@typescript-eslint/no-magic-numbers': 'off', + '@typescript-eslint/no-empty-interface': 'off', '@typescript-eslint/prefer-for-of': 'off', '@typescript-eslint/prefer-function-type': 'off', '@typescript-eslint/prefer-namespace-keyword': 'off', @@ -263,17 +260,20 @@ const styleRules = { 'unicorn/switch-case-braces': 'off', 'unicorn/text-encoding-identifier-case': 'off', 'unicorn/throw-new-error': 'off', + 'vitest/no-import-node-test': 'off', + 'vitest/prefer-each': 'off', + 'vitest/prefer-to-be-falsy': 'off', + 'vitest/prefer-to-be-object': 'off', + 'vitest/prefer-to-be-truthy': 'off', + '@typescript-eslint/default-param-last': 'off', + '@typescript-eslint/max-params': 'off', + '@typescript-eslint/no-magic-numbers': 'off', 'vitest/consistent-test-it': 'off', 'vitest/no-alias-methods': 'off', 'vitest/no-identical-title': 'off', - 'vitest/no-import-node-test': 'off', 'vitest/no-restricted-jest-methods': 'off', 'vitest/no-test-prefixes': 'off', - 'vitest/prefer-each': 'off', 'vitest/prefer-hooks-in-order': 'off', - 'vitest/prefer-to-be-falsy': 'off', - 'vitest/prefer-to-be-object': 'off', - 'vitest/prefer-to-be-truthy': 'off', } as const; const correctnessRules = { @@ -363,11 +363,11 @@ const correctnessRules = { 'jsx-a11y/lang': 'off', 'jsx-a11y/media-has-caption': 'off', 'jsx-a11y/mouse-events-have-key-events': 'off', + 'jsx-a11y/no-noninteractive-tabindex': 'off', 'jsx-a11y/no-access-key': 'off', 'jsx-a11y/no-aria-hidden-on-focusable': 'off', 'jsx-a11y/no-autofocus': 'off', 'jsx-a11y/no-distracting-elements': 'off', - 'jsx-a11y/no-noninteractive-tabindex': 'off', 'jsx-a11y/no-redundant-roles': 'off', 'jsx-a11y/prefer-tag-over-role': 'off', 'jsx-a11y/role-has-required-aria-props': 'off', @@ -410,15 +410,12 @@ const correctnessRules = { 'react/no-render-return-value': 'off', 'react/no-string-refs': 'off', 'react/void-dom-elements-no-children': 'off', - '@typescript-eslint/no-dupe-class-members': 'off', '@typescript-eslint/no-duplicate-enum-values': 'off', '@typescript-eslint/no-extra-non-null-assertion': 'off', - '@typescript-eslint/no-loss-of-precision': 'off', '@typescript-eslint/no-misused-new': 'off', '@typescript-eslint/no-non-null-asserted-optional-chain': 'off', '@typescript-eslint/no-this-alias': 'off', '@typescript-eslint/no-unsafe-declaration-merging': 'off', - '@typescript-eslint/no-unused-vars': 'off', '@typescript-eslint/no-useless-empty-export': 'off', '@typescript-eslint/no-wrapper-object-types': 'off', '@typescript-eslint/prefer-as-const': 'off', @@ -436,12 +433,15 @@ const correctnessRules = { 'unicorn/no-useless-spread': 'off', 'unicorn/prefer-set-size': 'off', 'unicorn/prefer-string-starts-ends-with': 'off', + 'vitest/no-conditional-tests': 'off', + 'vitest/require-local-test-context-for-concurrent-snapshots': 'off', + '@typescript-eslint/no-dupe-class-members': 'off', + '@typescript-eslint/no-loss-of-precision': 'off', + '@typescript-eslint/no-unused-vars': 'off', 'vitest/expect-expect': 'off', 'vitest/no-conditional-expect': 'off', - 'vitest/no-conditional-tests': 'off', 'vitest/no-disabled-tests': 'off', 'vitest/no-focused-tests': 'off', - 'vitest/require-local-test-context-for-concurrent-snapshots': 'off', 'vitest/valid-describe-callback': 'off', 'vitest/valid-expect': 'off', } as const; @@ -476,9 +476,9 @@ const suspiciousRules = { '@typescript-eslint/no-confusing-non-null-assertion': 'off', '@typescript-eslint/no-extraneous-class': 'off', '@typescript-eslint/no-unnecessary-type-constraint': 'off', - '@typescript-eslint/no-useless-constructor': 'off', 'unicorn/consistent-function-scoping': 'off', 'unicorn/prefer-add-event-listener': 'off', + '@typescript-eslint/no-useless-constructor': 'off', 'vitest/no-commented-out-tests': 'off', } as const; diff --git a/src/generated/rules-by-scope.ts b/src/generated/rules-by-scope.ts index 2219efb..0728a88 100644 --- a/src/generated/rules-by-scope.ts +++ b/src/generated/rules-by-scope.ts @@ -13,6 +13,12 @@ const eslintRules = { 'max-lines': 'off', 'max-params': 'off', 'new-cap': 'off', + 'no-extra-label': 'off', + 'no-multi-assign': 'off', + 'no-nested-ternary': 'off', + 'no-labels': 'off', + 'no-object-constructor': 'off', + 'no-duplicate-imports': 'off', 'no-alert': 'off', 'no-array-constructor': 'off', 'no-async-promise-executor': 'off', @@ -37,19 +43,17 @@ const eslintRules = { 'no-dupe-else-if': 'off', 'no-dupe-keys': 'off', 'no-duplicate-case': 'off', - 'no-duplicate-imports': 'off', 'no-else-return': 'off', - 'no-empty': 'off', 'no-empty-character-class': 'off', 'no-empty-function': 'off', 'no-empty-pattern': 'off', 'no-empty-static-block': 'off', + 'no-empty': 'off', 'no-eq-null': 'off', 'no-eval': 'off', 'no-ex-assign': 'off', 'no-extend-native': 'off', 'no-extra-boolean-cast': 'off', - 'no-extra-label': 'off', 'no-fallthrough': 'off', 'no-func-assign': 'off', 'no-global-assign': 'off', @@ -59,20 +63,16 @@ const eslintRules = { 'no-irregular-whitespace': 'off', 'no-iterator': 'off', 'no-label-var': 'off', - 'no-labels': 'off', 'no-loss-of-precision': 'off', 'no-magic-numbers': 'off', - 'no-multi-assign': 'off', - 'no-multi-str': 'off', 'no-negated-condition': 'off', - 'no-nested-ternary': 'off', - 'no-new': 'off', + 'no-multi-str': 'off', 'no-new-func': 'off', 'no-new-native-nonconstructor': 'off', 'no-new-wrappers': 'off', + 'no-new': 'off', 'no-nonoctal-decimal-escape': 'off', 'no-obj-calls': 'off', - 'no-object-constructor': 'off', 'no-plusplus': 'off', 'no-proto': 'off', 'no-prototype-builtins': 'off', @@ -107,10 +107,10 @@ const eslintRules = { 'no-var': 'off', 'no-void': 'off', 'no-with': 'off', + 'prefer-rest-params': 'off', 'prefer-exponentiation-operator': 'off', 'prefer-numeric-literals': 'off', 'prefer-object-has-own': 'off', - 'prefer-rest-params': 'off', 'prefer-spread': 'off', radix: 'off', 'require-await': 'off', @@ -129,6 +129,8 @@ const eslintRules = { const importRules = { 'import/default': 'off', 'import/first': 'off', + 'import/no-named-default': 'off', + 'import/no-namespace': 'off', 'import/max-dependencies': 'off', 'import/namespace': 'off', 'import/no-amd': 'off', @@ -139,8 +141,6 @@ const importRules = { 'import/no-dynamic-require': 'off', 'import/no-named-as-default': 'off', 'import/no-named-as-default-member': 'off', - 'import/no-named-default': 'off', - 'import/no-namespace': 'off', 'import/no-self-import': 'off', 'import/no-webpack-loader-syntax': 'off', 'import/unambiguous': 'off', @@ -220,7 +220,6 @@ const jsdocRules = { const jsxA11yRules = { 'jsx-a11y/alt-text': 'off', - 'jsx-a11y/anchor-ambiguous-text': 'off', 'jsx-a11y/anchor-has-content': 'off', 'jsx-a11y/anchor-is-valid': 'off', 'jsx-a11y/aria-activedescendant-has-tabindex': 'off', @@ -237,17 +236,18 @@ const jsxA11yRules = { 'jsx-a11y/lang': 'off', 'jsx-a11y/media-has-caption': 'off', 'jsx-a11y/mouse-events-have-key-events': 'off', + 'jsx-a11y/no-noninteractive-tabindex': 'off', 'jsx-a11y/no-access-key': 'off', 'jsx-a11y/no-aria-hidden-on-focusable': 'off', 'jsx-a11y/no-autofocus': 'off', 'jsx-a11y/no-distracting-elements': 'off', - 'jsx-a11y/no-noninteractive-tabindex': 'off', 'jsx-a11y/no-redundant-roles': 'off', 'jsx-a11y/prefer-tag-over-role': 'off', 'jsx-a11y/role-has-required-aria-props': 'off', 'jsx-a11y/role-supports-aria-props': 'off', 'jsx-a11y/scope': 'off', 'jsx-a11y/tabindex-no-positive': 'off', + 'jsx-a11y/anchor-ambiguous-text': 'off', } as const; const nextjsRules = { @@ -281,9 +281,9 @@ const nodeRules = { const promiseRules = { 'promise/avoid-new': 'off', 'promise/catch-or-return': 'off', + 'promise/no-promise-in-callback': 'off', 'promise/no-callback-in-promise': 'off', 'promise/no-new-statics': 'off', - 'promise/no-promise-in-callback': 'off', 'promise/param-names': 'off', 'promise/prefer-await-to-callbacks': 'off', 'promise/prefer-await-to-then': 'off', @@ -307,8 +307,8 @@ const reactRules = { 'react/jsx-props-no-spread-multi': 'off', 'react/no-array-index-key': 'off', 'react/no-children-prop': 'off', - 'react/no-danger': 'off', 'react/no-danger-with-children': 'off', + 'react/no-danger': 'off', 'react/no-direct-mutation-state': 'off', 'react/no-find-dom-node': 'off', 'react/no-is-mounted': 'off', @@ -344,38 +344,27 @@ const typescriptRules = { '@typescript-eslint/consistent-generic-constructors': 'off', '@typescript-eslint/consistent-indexed-object-style': 'off', '@typescript-eslint/consistent-type-definitions': 'off', - '@typescript-eslint/default-param-last': 'off', '@typescript-eslint/explicit-function-return-type': 'off', - '@typescript-eslint/max-params': 'off', - '@typescript-eslint/no-array-constructor': 'off', + '@typescript-eslint/no-inferrable-types': 'off', '@typescript-eslint/no-confusing-non-null-assertion': 'off', - '@typescript-eslint/no-dupe-class-members': 'off', '@typescript-eslint/no-duplicate-enum-values': 'off', '@typescript-eslint/no-dynamic-delete': 'off', - '@typescript-eslint/no-empty-function': 'off', '@typescript-eslint/no-empty-interface': 'off', '@typescript-eslint/no-empty-object-type': 'off', '@typescript-eslint/no-explicit-any': 'off', '@typescript-eslint/no-extra-non-null-assertion': 'off', '@typescript-eslint/no-extraneous-class': 'off', '@typescript-eslint/no-import-type-side-effects': 'off', - '@typescript-eslint/no-inferrable-types': 'off', - '@typescript-eslint/no-loss-of-precision': 'off', - '@typescript-eslint/no-magic-numbers': 'off', '@typescript-eslint/no-misused-new': 'off', '@typescript-eslint/no-namespace': 'off', '@typescript-eslint/no-non-null-asserted-nullish-coalescing': 'off', '@typescript-eslint/no-non-null-asserted-optional-chain': 'off', '@typescript-eslint/no-non-null-assertion': 'off', - '@typescript-eslint/no-redeclare': 'off', '@typescript-eslint/no-require-imports': 'off', '@typescript-eslint/no-this-alias': 'off', '@typescript-eslint/no-unnecessary-type-constraint': 'off', '@typescript-eslint/no-unsafe-declaration-merging': 'off', '@typescript-eslint/no-unsafe-function-type': 'off', - '@typescript-eslint/no-unused-expressions': 'off', - '@typescript-eslint/no-unused-vars': 'off', - '@typescript-eslint/no-useless-constructor': 'off', '@typescript-eslint/no-useless-empty-export': 'off', '@typescript-eslint/no-var-requires': 'off', '@typescript-eslint/no-wrapper-object-types': 'off', @@ -387,6 +376,17 @@ const typescriptRules = { '@typescript-eslint/prefer-namespace-keyword': 'off', '@typescript-eslint/prefer-ts-expect-error': 'off', '@typescript-eslint/triple-slash-reference': 'off', + '@typescript-eslint/default-param-last': 'off', + '@typescript-eslint/max-params': 'off', + '@typescript-eslint/no-array-constructor': 'off', + '@typescript-eslint/no-dupe-class-members': 'off', + '@typescript-eslint/no-empty-function': 'off', + '@typescript-eslint/no-loss-of-precision': 'off', + '@typescript-eslint/no-magic-numbers': 'off', + '@typescript-eslint/no-redeclare': 'off', + '@typescript-eslint/no-unused-expressions': 'off', + '@typescript-eslint/no-unused-vars': 'off', + '@typescript-eslint/no-useless-constructor': 'off', } as const; const unicornRules = { @@ -483,25 +483,25 @@ const unicornRules = { } as const; const vitestRules = { + 'vitest/no-conditional-tests': 'off', + 'vitest/no-import-node-test': 'off', + 'vitest/prefer-each': 'off', + 'vitest/prefer-to-be-falsy': 'off', + 'vitest/prefer-to-be-object': 'off', + 'vitest/prefer-to-be-truthy': 'off', + 'vitest/require-local-test-context-for-concurrent-snapshots': 'off', 'vitest/consistent-test-it': 'off', 'vitest/expect-expect': 'off', 'vitest/no-alias-methods': 'off', 'vitest/no-commented-out-tests': 'off', 'vitest/no-conditional-expect': 'off', 'vitest/no-conditional-in-test': 'off', - 'vitest/no-conditional-tests': 'off', 'vitest/no-disabled-tests': 'off', 'vitest/no-focused-tests': 'off', 'vitest/no-identical-title': 'off', - 'vitest/no-import-node-test': 'off', 'vitest/no-restricted-jest-methods': 'off', 'vitest/no-test-prefixes': 'off', - 'vitest/prefer-each': 'off', 'vitest/prefer-hooks-in-order': 'off', - 'vitest/prefer-to-be-falsy': 'off', - 'vitest/prefer-to-be-object': 'off', - 'vitest/prefer-to-be-truthy': 'off', - 'vitest/require-local-test-context-for-concurrent-snapshots': 'off', 'vitest/valid-describe-callback': 'off', 'vitest/valid-expect': 'off', } as const;