twenty/packages/twenty-eslint-rules/rules/effect-components.ts
Félix Malfait c737028dd6
Move tools/eslint-rules to packages/twenty-eslint-rules (#17203)
## Summary

Moves the custom ESLint rules from `tools/eslint-rules` to
`packages/twenty-eslint-rules` for better organization within the
monorepo packages structure.

## Changes

- Move `eslint-rules` from `tools/` to `packages/twenty-eslint-rules`
- Use `loadWorkspaceRules` from `@nx/eslint-plugin` to load custom rules
- Update all ESLint configs to use the `twenty/` rule prefix instead of
`@nx/workspace-`
- Update `project.json`, `jest.config.mjs` with new paths
- Update `package.json` workspaces and `nx.json` cache inputs
- Update Dockerfile reference

## Technical Details

The custom ESLint rules are now loaded using Nx's `loadWorkspaceRules`
utility which:
- Handles TypeScript transpilation automatically
- Allows loading workspace rules from any directory
- Provides a cleaner approach than the previous `@nx/workspace-`
convention

## Testing

- Verified all 17 custom ESLint rules load correctly from the new
location
- Verified linting works on dependent packages (twenty-front,
twenty-server, etc.)
2026-01-17 07:37:17 +01:00

115 lines
3.9 KiB
TypeScript

import { ESLintUtils, type TSESTree } from '@typescript-eslint/utils';
import {
isIdentifier,
isVariableDeclarator,
} from '@typescript-eslint/utils/ast-utils';
import { type RuleContext } from '@typescript-eslint/utils/ts-eslint';
// NOTE: The rule will be available in ESLint configs as "@nx/workspace-effect-components"
export const RULE_NAME = 'effect-components';
const isPascalCase = (input: string) => !!input.match(/^[A-Z][a-zA-Z0-9_]*/);
type TargetNode =
| TSESTree.ArrowFunctionExpression
| TSESTree.FunctionDeclaration
| TSESTree.FunctionExpression;
const isReturningEmptyFragmentOrNull = (node: TargetNode) =>
// Direct return of JSX fragment, e.g., () => <></>
(node.body.type === 'JSXFragment' && node.body.children.length === 0) ||
// Direct return of null, e.g., () => null
(node.body.type === 'Literal' && node.body.value === null) ||
// Return JSX fragment or null from block
(node.body.type === 'BlockStatement' &&
node.body.body.some(
(statement) =>
statement.type === 'ReturnStatement' &&
// Empty JSX fragment return, e.g., return <></>;
((statement.argument?.type === 'JSXFragment' &&
statement.argument.children.length === 0) ||
// Empty React.Fragment return, e.g., return <React.Fragment></React.Fragment>;
(statement.argument?.type === 'JSXElement' &&
statement.argument.openingElement.name.type === 'JSXIdentifier' &&
statement.argument.openingElement.name.name === 'React.Fragment' &&
statement.argument.children.length === 0) ||
// Literal null return, e.g., return null;
(statement.argument?.type === 'Literal' &&
statement.argument.value === null)),
));
const checkEffectComponent = ({
context,
identifier,
node,
}: {
context: Readonly<
RuleContext<'addEffectSuffix' | 'removeEffectSuffix', any[]>
>;
identifier: TSESTree.Identifier;
node: TargetNode;
}) => {
const componentName = identifier.name;
if (!isPascalCase(componentName)) return;
const isEffectComponent = isReturningEmptyFragmentOrNull(node);
const hasEffectSuffix = componentName.endsWith('Effect');
if (isEffectComponent && !hasEffectSuffix) {
context.report({
node,
messageId: 'addEffectSuffix',
data: { componentName },
fix: (fixer) => fixer.replaceText(identifier, componentName + 'Effect'),
});
return;
}
if (hasEffectSuffix && !isEffectComponent) {
context.report({
node,
messageId: 'removeEffectSuffix',
data: { componentName },
fix: (fixer) =>
fixer.replaceText(identifier, componentName.replace('Effect', '')),
});
}
};
export const rule = ESLintUtils.RuleCreator(() => __filename)({
name: RULE_NAME,
meta: {
docs: {
description:
'Effect components should end with the Effect suffix. This rule checks only components that are in PascalCase and that return a JSX fragment or null. Any renderProps or camelCase components are ignored.',
},
messages: {
addEffectSuffix:
'Effect component {{ componentName }} should end with the Effect suffix.',
removeEffectSuffix:
"Component {{ componentName }} shouldn't end with the Effect suffix because it doesn't return a JSX fragment or null.",
},
type: 'suggestion',
schema: [],
fixable: 'code',
},
defaultOptions: [],
create: (context) => {
const checkFunctionExpressionEffectComponent = (
node: TSESTree.ArrowFunctionExpression | TSESTree.FunctionExpression,
) =>
isVariableDeclarator(node.parent) && isIdentifier(node.parent.id)
? checkEffectComponent({ context, identifier: node.parent.id, node })
: undefined;
return {
ArrowFunctionExpression: checkFunctionExpressionEffectComponent,
FunctionDeclaration: (node) =>
checkEffectComponent({ context, identifier: node.id, node }),
FunctionExpression: checkFunctionExpressionEffectComponent,
};
},
});