mirror of
https://github.com/danny-avila/LibreChat.git
synced 2026-06-09 17:31:19 +00:00
* 🔧 chore: Update ESLint config, add import sorting script, Test Sharding, Bump `@librechat/agents`
* Change 'no-nested-ternary' rule from 'warn' to 'error' in ESLint config
* Add new scripts for sorting imports in the project
* Update lint-staged configuration to include import sorting
* Modify GitHub Actions workflows to support sharding for unit tests
* chore: remove nested ternary expressions
* refactor: Extract scale multiplier logic into a separate function in CircleRender component
* refactor: Simplify auto-refill rendering logic in Balance component for better readability
* refactor: Improve width style handling in DataTable components for clarity and maintainability
* chore: remove CircleRender component
* delete: Remove CircleRender component as it is no longer needed in the project
* chore: Bump @librechat/agents to version 3.2.31 and update Node.js engine requirement
* Update @librechat/agents dependency from 3.2.2 to 3.2.31 in package-lock.json, api/package.json, and packages/api/package.json
* Change Node.js engine requirement from >=20.0.0 to >=24.0.0 in @librechat/agents
* chore: Add import sorting check to ESLint CI workflow
* Implement a new job in the GitHub Actions workflow to verify import ordering on changed files.
* The job checks for changes in specific file types and reports any import order drift, providing instructions for local fixes.
393 lines
11 KiB
JavaScript
393 lines
11 KiB
JavaScript
import { fileURLToPath } from 'node:url';
|
|
import path from 'node:path';
|
|
import typescriptEslintEslintPlugin from '@typescript-eslint/eslint-plugin';
|
|
import { fixupConfigRules, fixupPluginRules } from '@eslint/compat';
|
|
import reactHooks from 'eslint-plugin-react-hooks';
|
|
import tsParser from '@typescript-eslint/parser';
|
|
import importPlugin from 'eslint-plugin-import';
|
|
import prettier from 'eslint-plugin-prettier';
|
|
import { FlatCompat } from '@eslint/eslintrc';
|
|
import jsxA11Y from 'eslint-plugin-jsx-a11y';
|
|
import i18next from 'eslint-plugin-i18next';
|
|
import react from 'eslint-plugin-react';
|
|
import jest from 'eslint-plugin-jest';
|
|
import globals from 'globals';
|
|
import js from '@eslint/js';
|
|
|
|
const __filename = fileURLToPath(import.meta.url);
|
|
const __dirname = path.dirname(__filename);
|
|
const compat = new FlatCompat({
|
|
baseDirectory: __dirname,
|
|
recommendedConfig: js.configs.recommended,
|
|
allConfig: js.configs.all,
|
|
});
|
|
|
|
export default [
|
|
{
|
|
ignores: [
|
|
'client/dist/**/*',
|
|
'client/public/**/*',
|
|
'client/coverage/**/*',
|
|
'e2e/playwright-report/**/*',
|
|
'packages/api/types/**/*',
|
|
'packages/api/dist/**/*',
|
|
'packages/api/test_bundle/**/*',
|
|
'api/demo/**/*',
|
|
'packages/client/dist/**/*',
|
|
'packages/data-provider/types/**/*',
|
|
'packages/data-provider/dist/**/*',
|
|
'packages/data-provider/test_bundle/**/*',
|
|
'packages/data-schemas/dist/**/*',
|
|
'packages/data-schemas/misc/**/*',
|
|
'data-node/**/*',
|
|
'meili_data/**/*',
|
|
'**/node_modules/**/*',
|
|
'.devcontainer/**/*',
|
|
],
|
|
},
|
|
...fixupConfigRules(
|
|
compat.extends(
|
|
'eslint:recommended',
|
|
'plugin:react/recommended',
|
|
'plugin:react-hooks/recommended',
|
|
'plugin:jest/recommended',
|
|
'prettier',
|
|
'plugin:jsx-a11y/recommended',
|
|
),
|
|
),
|
|
{
|
|
plugins: {
|
|
react: fixupPluginRules(react),
|
|
'react-hooks': fixupPluginRules(reactHooks),
|
|
'@typescript-eslint': typescriptEslintEslintPlugin,
|
|
import: importPlugin,
|
|
'jsx-a11y': fixupPluginRules(jsxA11Y),
|
|
'import/parsers': tsParser,
|
|
i18next,
|
|
prettier: fixupPluginRules(prettier),
|
|
},
|
|
|
|
languageOptions: {
|
|
globals: {
|
|
...globals.browser,
|
|
...globals.node,
|
|
...globals.commonjs,
|
|
},
|
|
parser: tsParser,
|
|
ecmaVersion: 'latest',
|
|
sourceType: 'module',
|
|
parserOptions: {
|
|
ecmaFeatures: {
|
|
jsx: true,
|
|
},
|
|
},
|
|
},
|
|
|
|
settings: {
|
|
react: {
|
|
createClass: 'createReactClass',
|
|
pragma: 'React',
|
|
fragment: 'Fragment',
|
|
version: 'detect',
|
|
},
|
|
'import/parsers': {
|
|
'@typescript-eslint/parser': ['.ts', '.tsx'],
|
|
},
|
|
'import/resolver': {
|
|
typescript: {
|
|
project: ['./client/tsconfig.json'],
|
|
},
|
|
node: {
|
|
project: ['./client/tsconfig.json'],
|
|
},
|
|
},
|
|
},
|
|
|
|
rules: {
|
|
'prettier/prettier': 'error',
|
|
'react/react-in-jsx-scope': 'off',
|
|
|
|
'@typescript-eslint/ban-ts-comment': [
|
|
'error',
|
|
{
|
|
'ts-ignore': false,
|
|
},
|
|
],
|
|
// Disable a11y features to be enabled later on.
|
|
'jsx-a11y/no-static-element-interactions': 'off',
|
|
'jsx-a11y/click-events-have-key-events': 'off',
|
|
'jsx-a11y/alt-text': 'off',
|
|
'jsx-a11y/img-redundant-alt': 'off',
|
|
'jsx-a11y/no-noninteractive-tabindex': 'off',
|
|
// common rules
|
|
'no-nested-ternary': 'error',
|
|
'no-constant-binary-expression': 'warn',
|
|
'no-unused-vars': [
|
|
'warn',
|
|
{
|
|
argsIgnorePattern: '^_',
|
|
varsIgnorePattern: '^_',
|
|
caughtErrorsIgnorePattern: '^_',
|
|
},
|
|
],
|
|
'no-console': 'off',
|
|
'import/no-cycle': 'error',
|
|
'import/no-self-import': 'error',
|
|
'import/extensions': 'off',
|
|
'no-promise-executor-return': 'off',
|
|
'no-param-reassign': 'off',
|
|
'no-continue': 'off',
|
|
'no-restricted-syntax': 'off',
|
|
'react/prop-types': 'off',
|
|
'react/display-name': 'off',
|
|
},
|
|
},
|
|
{
|
|
files: ['api/**/*.js', 'config/**/*.js'],
|
|
rules: {
|
|
// API
|
|
'no-async-promise-executor': 'off',
|
|
},
|
|
},
|
|
{
|
|
files: [
|
|
'client/src/**/*.tsx',
|
|
'client/src/**/*.ts',
|
|
'client/src/**/*.jsx',
|
|
'client/src/**/*.js',
|
|
],
|
|
rules: {
|
|
// Client a11y
|
|
// TODO: maybe later to error.
|
|
'jsx-a11y/no-noninteractive-element-interactions': 'off',
|
|
'jsx-a11y/label-has-associated-control': 'off',
|
|
'jsx-a11y/no-static-element-interactions': 'off',
|
|
'jsx-a11y/click-events-have-key-events': 'off',
|
|
'jsx-a11y/interactive-supports-focus': 'off',
|
|
'jsx-a11y/no-noninteractive-tabindex': 'off',
|
|
'jsx-a11y/img-redundant-alt': 'off',
|
|
},
|
|
},
|
|
{
|
|
files: ['**/rollup.config.js', '**/.eslintrc.js', '**/jest.config.js', 'client/vite.config.ts'],
|
|
languageOptions: {
|
|
globals: {
|
|
...globals.node,
|
|
},
|
|
},
|
|
rules: {
|
|
'import/no-cycle': 'off',
|
|
},
|
|
},
|
|
{
|
|
files: [
|
|
'**/*.test.js',
|
|
'**/*.test.jsx',
|
|
'**/*.test.ts',
|
|
'**/*.test.tsx',
|
|
'**/*.spec.js',
|
|
'**/*.spec.jsx',
|
|
'**/*.spec.ts',
|
|
'**/*.spec.tsx',
|
|
'**/setupTests.js',
|
|
],
|
|
languageOptions: {
|
|
globals: {
|
|
...globals.jest,
|
|
...globals.node,
|
|
},
|
|
},
|
|
rules: {
|
|
// TEST
|
|
'react/display-name': 'off',
|
|
'react/prop-types': 'off',
|
|
'jest/no-commented-out-tests': 'off',
|
|
'react/no-unescaped-entities': 'off',
|
|
'jest/no-conditional-expect': 'off',
|
|
'jest/no-disabled-tests': 'off',
|
|
'@typescript-eslint/no-unused-vars': 'off',
|
|
},
|
|
},
|
|
...compat
|
|
.extends(
|
|
'plugin:@typescript-eslint/eslint-recommended',
|
|
'plugin:@typescript-eslint/recommended',
|
|
)
|
|
.map((config) => ({
|
|
...config,
|
|
files: ['**/*.ts', '**/*.tsx'],
|
|
})),
|
|
{
|
|
files: ['**/*.ts', '**/*.tsx'],
|
|
ignores: ['packages/**/*', 'client/vite.config.ts'],
|
|
plugins: {
|
|
'@typescript-eslint': typescriptEslintEslintPlugin,
|
|
jest: fixupPluginRules(jest),
|
|
},
|
|
languageOptions: {
|
|
parser: tsParser,
|
|
ecmaVersion: 5,
|
|
sourceType: 'script',
|
|
parserOptions: {
|
|
project: './client/tsconfig.json',
|
|
},
|
|
},
|
|
rules: {
|
|
// i18n
|
|
'i18next/no-literal-string': [
|
|
'error',
|
|
{
|
|
mode: 'jsx-text-only',
|
|
'should-validate-template': true,
|
|
},
|
|
],
|
|
//
|
|
'lines-between-class-members': ['error', 'always', { exceptAfterSingleLine: true }],
|
|
'@typescript-eslint/no-unused-expressions': 'off',
|
|
'@typescript-eslint/no-unused-vars': [
|
|
'warn',
|
|
{
|
|
argsIgnorePattern: '^_',
|
|
varsIgnorePattern: '^_',
|
|
caughtErrorsIgnorePattern: '^_',
|
|
},
|
|
],
|
|
'@typescript-eslint/no-explicit-any': 'off',
|
|
'@typescript-eslint/no-unnecessary-condition': 'off',
|
|
'@typescript-eslint/strict-boolean-expressions': 'off',
|
|
'@typescript-eslint/ban-ts-comment': 'off',
|
|
// React
|
|
'react/no-unknown-property': 'warn',
|
|
'react-hooks/rules-of-hooks': 'error',
|
|
'react-hooks/exhaustive-deps': 'warn',
|
|
// General
|
|
'no-constant-binary-expression': 'off',
|
|
'import/no-cycle': 'off',
|
|
},
|
|
},
|
|
{
|
|
// **Data-provider specific configuration block**
|
|
files: ['./packages/data-provider/**/*.ts'],
|
|
languageOptions: {
|
|
parser: tsParser,
|
|
ecmaVersion: 'latest',
|
|
sourceType: 'module',
|
|
parserOptions: {
|
|
project: './packages/data-provider/tsconfig.json',
|
|
},
|
|
},
|
|
rules: {
|
|
'@typescript-eslint/no-unused-vars': [
|
|
'warn',
|
|
{
|
|
argsIgnorePattern: '^_',
|
|
varsIgnorePattern: '^_',
|
|
caughtErrorsIgnorePattern: '^_',
|
|
},
|
|
],
|
|
},
|
|
},
|
|
{
|
|
files: ['./api/demo/**/*.ts'],
|
|
},
|
|
{
|
|
files: ['./packages/api/**/*.ts'],
|
|
rules: {
|
|
'lines-between-class-members': ['error', 'always', { exceptAfterSingleLine: true }],
|
|
'@typescript-eslint/no-unused-vars': [
|
|
'warn',
|
|
{
|
|
argsIgnorePattern: '^_',
|
|
varsIgnorePattern: '^_',
|
|
caughtErrorsIgnorePattern: '^_',
|
|
destructuredArrayIgnorePattern: '^_',
|
|
},
|
|
],
|
|
},
|
|
},
|
|
{
|
|
files: ['./config/translations/**/*.ts'],
|
|
languageOptions: {
|
|
parser: tsParser,
|
|
ecmaVersion: 5,
|
|
sourceType: 'script',
|
|
parserOptions: {
|
|
project: './config/translations/tsconfig.json',
|
|
},
|
|
},
|
|
},
|
|
{
|
|
files: ['./packages/data-provider/specs/**/*.ts'],
|
|
languageOptions: {
|
|
ecmaVersion: 5,
|
|
sourceType: 'script',
|
|
parserOptions: {
|
|
project: './packages/data-provider/tsconfig.spec.json',
|
|
},
|
|
},
|
|
},
|
|
{
|
|
files: ['./api/demo/specs/**/*.ts'],
|
|
languageOptions: {
|
|
ecmaVersion: 5,
|
|
sourceType: 'script',
|
|
parserOptions: {
|
|
project: './packages/data-provider/tsconfig.spec.json',
|
|
},
|
|
},
|
|
},
|
|
{
|
|
files: ['./packages/api/specs/**/*.ts'],
|
|
languageOptions: {
|
|
ecmaVersion: 5,
|
|
sourceType: 'script',
|
|
parserOptions: {
|
|
project: './packages/api/tsconfig.spec.json',
|
|
},
|
|
},
|
|
},
|
|
{
|
|
// **Data-schemas — shared rules for all TS files**
|
|
files: ['./packages/data-schemas/**/*.ts'],
|
|
languageOptions: {
|
|
parser: tsParser,
|
|
ecmaVersion: 'latest',
|
|
sourceType: 'module',
|
|
parserOptions: {
|
|
project: './packages/data-schemas/tsconfig.json',
|
|
},
|
|
},
|
|
rules: {
|
|
'@typescript-eslint/no-unused-vars': [
|
|
'warn',
|
|
{
|
|
argsIgnorePattern: '^_',
|
|
varsIgnorePattern: '^_',
|
|
caughtErrorsIgnorePattern: '^_',
|
|
destructuredArrayIgnorePattern: '^_',
|
|
},
|
|
],
|
|
},
|
|
},
|
|
{
|
|
// **Data-schemas — ban raw bulkWrite/collection.* in production code**
|
|
// Tests and the tenantSafeBulkWrite wrapper itself are excluded.
|
|
files: ['./packages/data-schemas/**/*.ts'],
|
|
ignores: ['**/*.spec.ts', '**/*.test.ts', '**/utils/tenantBulkWrite.ts'],
|
|
rules: {
|
|
'no-restricted-syntax': [
|
|
'error',
|
|
{
|
|
selector: "CallExpression[callee.property.name='bulkWrite']",
|
|
message:
|
|
'Use tenantSafeBulkWrite() instead of Model.bulkWrite() — Mongoose middleware does not fire for bulkWrite, so the tenant isolation plugin cannot intercept it.',
|
|
},
|
|
{
|
|
selector: "MemberExpression[property.name='collection'][parent.type='MemberExpression']",
|
|
message:
|
|
'Avoid Model.collection.* — raw driver calls bypass all Mongoose middleware including tenant isolation. Use Mongoose model methods or tenantSafeBulkWrite() instead.',
|
|
},
|
|
],
|
|
},
|
|
},
|
|
];
|