PSA/eslint.config.js
Hermes 284313f908
Some checks are pending
Bidi Control Character Guard / bidi-control-guard (push) Waiting to run
Circular Dependency Check / Check for new circular dependencies (push) Waiting to run
Citus Migration Smoke / Combined migrations on single-node Citus (push) Waiting to run
E2E Fresh Install Tests / fresh-install-e2e (push) Waiting to run
ext-v2 guardrails / Run ext-v2 guard and ESLint (push) Waiting to run
Integration Tests / Check for relevant changes (push) Waiting to run
Integration Tests / ${{ (github.event_name == 'schedule' || github.event.inputs.suite == 'full') && 'Full integration suite' || 'Tier-1 integration subset' }} (push) Blocked by required conditions
Mobile checks / Mobile lint + typecheck (push) Waiting to run
Mobile checks / Mobile unit tests (push) Waiting to run
Mobile checks / Mobile dependency audit (report) (push) Waiting to run
Mobile checks / Mobile reproducibility checks (push) Waiting to run
Secrets guard (env backups) / Ensure no tracked env backup files (push) Waiting to run
Temporal Readiness / fast-readiness (push) Waiting to run
Temporal Readiness / docker-parity (push) Waiting to run
TypeScript Type Check / Nx affected typecheck (push) Waiting to run
Unit Tests / Skipped-test budget (push) Waiting to run
Unit Tests / Nx affected unit tests (push) Waiting to run
Unit Tests / Server unit coverage (informational) (push) Waiting to run
Validate Tenant Management Schema / Check for relevant changes (push) Waiting to run
Validate Tenant Management Schema / Validate Tenant Management Schema (push) Blocked by required conditions
EE Workflows Build Guard / ee-workflows-build-guard (push) Waiting to run
Initial import of AlgaPSA codebase from PSA server
Excluded: .git, node_modules, secrets/, compose.env, assemblyscript tgz

Source: /opt/alga-psa on psa.joliet.tech
2026-06-22 16:12:17 -05:00

283 lines
11 KiB
JavaScript

import globals from "globals";
import pluginJs from "@eslint/js";
import tseslint from "@typescript-eslint/eslint-plugin";
import tsParser from "@typescript-eslint/parser";
import pluginReact from "eslint-plugin-react";
import pluginReactHooks from "eslint-plugin-react-hooks";
import customRules from "./eslint-plugin-custom-rules/index.js";
import { fileURLToPath } from 'url';
import path from 'path';
// Ensure tsconfig resolution works regardless of process.cwd()
const __dirname = path.dirname(fileURLToPath(import.meta.url));
export default [
// Global ignores (apply to all config blocks)
{
ignores: [
"eslint-plugin-custom-rules/**/*",
".ai/**/*",
"tools/**/*",
"ee/extensions/**/*",
"**/.next/**/*",
"**/dist/**/*",
"**/out/**/*",
"**/build/**/*",
"**/coverage/**/*",
"server/public/**/*",
"server/src/invoice-templates/assemblyscript/**/*"
],
},
// Configuration for migration files - enforce naming conventions
// IMPORTANT: This must come first before other configs that might ignore these files
{
files: ["**/migrations/**/*.cjs"],
ignores: [
"**/migrations/**/utils/**", // Exclude utility files in utils directories
"ee/server/migrations/**/*", // Exclude EE migrations entirely
],
languageOptions: {
globals: {
...globals.node
},
ecmaVersion: 2022,
sourceType: "commonjs"
},
plugins: {
"custom-rules": customRules,
},
rules: {
"custom-rules/migration-filename": "error",
"no-unused-vars": "warn",
}
},
// Configuration for JavaScript files (no TypeScript)
{
files: ["**/*.{js,mjs,cjs}"],
ignores: [
"eslint-plugin-custom-rules/**/*",
"eslint.config.js",
"**/eslint.config.js",
"ee/server/migrations/**/*", // Ignore EE migration files (usually .cjs)
// Mirror packages under ee/server that aren't primary sources (avoid duplicate lint targets)
"ee/server/packages/extension-iframe-sdk/**/*",
"ee/extensions/**/*", // Exclude extension bundles/examples from lint to reduce load and avoid false positives
"**/dist/**/*",
"**/.next/**/*",
"**/out/**/*",
"tools/**/*",
".ai/**/*",
"**/build/**/*",
"server/public/**/*",
"services/workflow-worker/src/workflows/system-email-processing-workflow.ts", // Plain JS for workflow runtime compatibility
"server/src/invoice-templates/assemblyscript/**/*" // AssemblyScript files have different syntax
],
// Define language options for JS files
languageOptions: {
globals: {
...globals.browser,
...globals.node,
React: true,
JSX: true
},
ecmaVersion: 2022,
sourceType: "module"
},
plugins: {
"custom-rules": customRules,
},
rules: {
// Base ESLint rules for JS
"no-unused-vars": "warn",
"no-undef": "warn",
"no-console": "off",
// Custom rules
"custom-rules/map-return-type": "off",
"custom-rules/check-required-props": "error",
"custom-rules/no-legacy-ext-imports": "error",
"custom-rules/no-feature-to-feature-imports": "error",
}
},
// Configuration for TypeScript files
{
files: ["**/*.{ts,tsx}"],
ignores: [
"eslint-plugin-custom-rules/**/*",
"eslint.config.js",
"**/eslint.config.js",
"ee/server/migrations/**/*",
// Mirror packages under ee/server that aren't primary sources (avoid duplicate lint targets)
"ee/server/packages/extension-iframe-sdk/**/*",
"ee/extensions/**/*",
"**/dist/**/*",
"**/.next/**/*",
"**/out/**/*",
"tools/**/*",
".ai/**/*",
"**/build/**/*",
"server/public/**/*",
"shared/workflow/workflows/system-email-processing-workflow.ts",
"server/src/invoice-templates/assemblyscript/**/*" // AssemblyScript files have different syntax
],
languageOptions: {
globals: {
...globals.browser,
React: true,
JSX: true
},
parser: tsParser,
parserOptions: {
ecmaVersion: 2022,
sourceType: "module",
ecmaFeatures: {
jsx: true
},
},
},
// Add base rules and plugins
plugins: {
"@typescript-eslint": tseslint,
"react-hooks": pluginReactHooks,
"custom-rules": customRules,
},
rules: {
// TypeScript rules
"@typescript-eslint/explicit-function-return-type": [
"warn",
{
allowExpressions: true,
allowTypedFunctionExpressions: true,
allowFunctionsWithoutTypeParameters: true,
},
],
"@typescript-eslint/no-explicit-any": "off",
"@typescript-eslint/no-unused-vars": ["off", { argsIgnorePattern: "^_" }],
// NOTE: These rules require type information (parserOptions.project).
// This repo runs ESLint without typed linting to avoid OOM on large workspaces.
"@typescript-eslint/no-unsafe-assignment": "off",
"@typescript-eslint/no-unsafe-member-access": "off",
"@typescript-eslint/no-unsafe-return": "off",
"@typescript-eslint/no-unsafe-call": "off",
"@typescript-eslint/no-unsafe-argument": "off",
"@typescript-eslint/no-misused-promises": "off",
"@typescript-eslint/await-thenable": "off",
"@typescript-eslint/no-floating-promises": "off",
"@typescript-eslint/no-unnecessary-type-assertion": "off",
"@typescript-eslint/restrict-plus-operands": "off",
"@typescript-eslint/restrict-template-expressions": "off",
"@typescript-eslint/unbound-method": "off",
"@typescript-eslint/no-non-null-assertion": "warn",
"@typescript-eslint/no-redundant-type-constituents": "off", // Disabled due to infinite recursion with complex types
// React hooks rules
"react-hooks/rules-of-hooks": "error",
"react-hooks/exhaustive-deps": "warn",
// Custom rules as warnings
"custom-rules/map-return-type": "off",
"custom-rules/check-required-props": "error",
"custom-rules/no-legacy-ext-imports": "error",
"custom-rules/no-feature-to-feature-imports": "error",
// Base ESLint rules
"no-unused-vars": "off", // Turn off in favor of @typescript-eslint/no-unused-vars
"react/react-in-jsx-scope": "off", // Not needed in Next.js
"no-undef": "off", // TypeScript handles this
"react/prop-types": "off", // TypeScript handles this
// Override recommended configs to use warnings
// NOTE: Disabled "recommended-requiring-type-checking" due to memory exhaustion during type checking
...Object.fromEntries(Object.entries({
...tseslint.configs.recommended.rules,
// ...tseslint.configs["recommended-requiring-type-checking"].rules, // DISABLED: Causes OOM
}).map(([key, value]) => [
key,
typeof value === 'string' ? 'warn' : ['warn', ...(Array.isArray(value) ? value.slice(1) : [])],
])),
},
settings: {
typescript: {
alwaysTryTypes: true,
}
}
},
// Add plugin-specific configurations with warnings
{
...pluginJs.configs.recommended,
rules: Object.fromEntries(Object.entries(pluginJs.configs.recommended.rules || {}).map(([key, value]) => [
key,
typeof value === 'string' ? 'warn' : ['warn', ...(Array.isArray(value) ? value.slice(1) : [])],
])),
},
{
...pluginReact.configs.flat.recommended,
rules: Object.fromEntries(Object.entries(pluginReact.configs.flat.recommended.rules || {}).map(([key, value]) => [
key,
typeof value === 'string' ? 'warn' : ['warn', ...(Array.isArray(value) ? value.slice(1) : [])],
])),
settings: {
react: {
version: "18.2"
}
}
},
// Runtime boundary guardrails for workflow worker and shared runtime.
{
files: [
"services/workflow-worker/src/**/*.{js,mjs,cjs,ts,tsx}",
"shared/workflow/runtime/**/*.{js,mjs,cjs,ts,tsx}"
],
rules: {
"no-restricted-imports": [
"error",
{
paths: [
{
name: "@alga-psa/auth",
message: "Auth package root imports are not allowed in worker/runtime code."
},
{
name: "@alga-psa/documents",
message: "Use @alga-psa/documents/runtime or @alga-psa/storage in worker/runtime code."
},
{
name: "@alga-psa/integrations",
message: "Use @alga-psa/integrations/runtime in worker/runtime code."
},
{
name: "@alga-psa/billing",
message: "Use @alga-psa/billing/runtime in worker/runtime code."
},
{
name: "@alga-psa/ui",
message: "UI package imports are not allowed in worker/runtime code."
}
],
patterns: [
{
group: [
"@alga-psa/ui/*",
"@alga-psa/*/components",
"@alga-psa/*/components/*"
],
message: "Component imports are not allowed in worker/runtime code."
}
]
}
]
}
},
// Configuration for test files
{
files: ["**/*.test.ts", "**/*.test.tsx", "**/*.spec.ts", "**/*.spec.tsx", "**/*.playwright.test.ts", "**/__tests__/**/*.ts", "**/__tests__/**/*.tsx"],
languageOptions: {
globals: {
...globals.node,
...globals.browser,
React: true,
JSX: true
}
}
},
{
ignores: [
"**/vitest.config.*.timestamp*"
]
}
];