mirror of
https://github.com/affaan-m/everything-claude-code.git
synced 2026-06-16 05:01:32 +08:00
feat(rules): add rules/react/ track
Five rule files mirroring per-language convention (coding-style, hooks, patterns, security, testing). Each has `paths:` glob frontmatter for auto-activation when editing matching files. - coding-style.md: file extensions, naming, JSX, RSC boundary - hooks.md: React hooks (NOT Claude Code hooks) — rules-of-hooks, dep arrays, cleanup, memoization, React 19 additions - patterns.md: container/presentational split, state location decision tree, Suspense + error boundaries, forms, data fetching - security.md: dangerouslySetInnerHTML, unsafe URL schemes, server-action validation, env-var leaks, CSP - testing.md: RTL queries, userEvent, async, MSW, axe, anti-patterns Each file extends typescript/* and common/* rules.
This commit is contained in:
@@ -0,0 +1,109 @@
|
||||
---
|
||||
paths:
|
||||
- "**/*.tsx"
|
||||
- "**/*.jsx"
|
||||
- "**/components/**/*.ts"
|
||||
- "**/components/**/*.js"
|
||||
- "**/hooks/**/*.ts"
|
||||
- "**/hooks/**/*.js"
|
||||
---
|
||||
# React Coding Style
|
||||
|
||||
> This file extends [typescript/coding-style.md](../typescript/coding-style.md) and [common/coding-style.md](../common/coding-style.md) with React specific content.
|
||||
|
||||
## File Extensions
|
||||
|
||||
- `.tsx` for any file containing JSX, even one-liner snippets
|
||||
- `.ts` for pure logic, custom hooks without JSX, type definitions, utilities
|
||||
- `.test.tsx` / `.test.ts` mirroring the source file
|
||||
- Use `.jsx` only when the project intentionally avoids TypeScript — flag every new untyped React file in review
|
||||
|
||||
## Naming
|
||||
|
||||
- Components: `PascalCase` for both the symbol and the file (`UserCard.tsx`, default export `UserCard`)
|
||||
- Custom hooks: `useCamelCase` for the symbol, kebab-case for the file when the project convention is kebab-case (`use-debounce.ts` exports `useDebounce`)
|
||||
- Context: `<Domain>Context` symbol, `<Domain>Provider` provider component, `use<Domain>` consumer hook
|
||||
- Event handlers: `handleClick`, `handleSubmit` inside the component; the prop that receives it is `onClick`, `onSubmit`
|
||||
- Boolean props: `isLoading`, `hasError`, `canSubmit` — never `loading` or `error` alone for booleans
|
||||
|
||||
## Component Shape
|
||||
|
||||
```tsx
|
||||
type Props = {
|
||||
user: User;
|
||||
onSelect: (id: string) => void;
|
||||
};
|
||||
|
||||
export function UserCard({ user, onSelect }: Props) {
|
||||
return (
|
||||
<button type="button" onClick={() => onSelect(user.id)}>
|
||||
{user.name}
|
||||
</button>
|
||||
);
|
||||
}
|
||||
```
|
||||
|
||||
- Prefer `type Props = {}` for closed component prop shapes
|
||||
- Use `interface` only when the prop type is extended via declaration merging or exported as a public API extension point
|
||||
- Always destructure props in the parameter list — no `props.user` access inside the body
|
||||
- Type the return implicitly through JSX (`function Foo(): JSX.Element` only when the function returns conditionally and the union confuses inference)
|
||||
|
||||
## JSX
|
||||
|
||||
- Self-close tags with no children: `<img />`, `<UserCard user={u} />`
|
||||
- Use fragments `<>...</>` over wrapper `<div>` when no DOM element is needed
|
||||
- Conditional rendering: `{condition && <Foo />}` for booleans, ternary for either/or, early return for guard clauses
|
||||
- Never put logic inline in JSX when it reads as multi-line — extract to a const above the return or a function
|
||||
|
||||
```tsx
|
||||
// Prefer
|
||||
const greeting = user.isAdmin ? "Welcome, admin" : `Hello ${user.name}`;
|
||||
return <h1>{greeting}</h1>;
|
||||
|
||||
// Over
|
||||
return <h1>{user.isAdmin ? "Welcome, admin" : `Hello ${user.name}`}</h1>;
|
||||
```
|
||||
|
||||
## Server / Client Boundary (Next.js App Router, RSC)
|
||||
|
||||
- Default a new file to Server Component — only add `"use client"` when the file uses state, effects, refs, browser APIs, or event handlers
|
||||
- Place the `"use client"` directive on line 1, before any imports
|
||||
- Never import a Client Component file from inside a `"use server"` action file
|
||||
- Never re-export server-only code through a client module — the bundler will silently include it
|
||||
|
||||
## Imports
|
||||
|
||||
- React imports first: `import { useState } from "react"`
|
||||
- Then third-party libs, then absolute project imports, then relative
|
||||
- Type-only imports: `import type { ReactNode } from "react"` — never mix runtime and type imports in one statement when ESLint's `consistent-type-imports` is configured
|
||||
|
||||
## Hooks Discipline
|
||||
|
||||
See [hooks.md](./hooks.md) for the full ruleset. Style highlights:
|
||||
|
||||
- Custom hooks must start with `use` — enforced by `eslint-plugin-react-hooks`
|
||||
- Group all hook calls at the top of the component, before any conditional logic
|
||||
- Avoid creating ad-hoc hooks for one-line wrappers — inline the call instead
|
||||
|
||||
## State
|
||||
|
||||
- Local first (`useState`), lift only when shared
|
||||
- Context for cross-cutting state read by many components (theme, auth, i18n) — not for high-frequency updates
|
||||
- External store (Zustand, Jotai, Redux Toolkit) when state must persist across route changes, sync across tabs, or be debugged via devtools
|
||||
- Never duplicate state that can be derived — compute during render
|
||||
|
||||
## Class Components
|
||||
|
||||
Forbidden in new code. Convert legacy class components to function components when touching them for non-trivial changes.
|
||||
|
||||
## File Layout per Component
|
||||
|
||||
```
|
||||
components/UserCard/
|
||||
UserCard.tsx
|
||||
UserCard.module.css # or styled-components, or Tailwind classes inline
|
||||
UserCard.test.tsx
|
||||
index.ts # re-export only
|
||||
```
|
||||
|
||||
Inline single-file components are fine for trivial presentational pieces.
|
||||
@@ -0,0 +1,187 @@
|
||||
---
|
||||
paths:
|
||||
- "**/*.tsx"
|
||||
- "**/*.jsx"
|
||||
- "**/hooks/**/*.ts"
|
||||
- "**/hooks/**/*.js"
|
||||
- "**/use-*.ts"
|
||||
- "**/use-*.tsx"
|
||||
---
|
||||
# React Hooks
|
||||
|
||||
> This file covers **React hooks** (`useState`, `useEffect`, `useMemo`, `useCallback`, custom hooks) — NOT the Claude Code `hooks/` runtime system. Naming matches the per-language convention `rules/<lang>/hooks.md` used across this repo.
|
||||
>
|
||||
> Extends [typescript/patterns.md](../typescript/patterns.md) and [common/patterns.md](../common/patterns.md).
|
||||
|
||||
## Rules of Hooks
|
||||
|
||||
Enforce `eslint-plugin-react-hooks` with `react-hooks/rules-of-hooks` set to error.
|
||||
|
||||
1. Hooks only at the top level of a function component or another hook
|
||||
2. Never in loops, conditionals, nested functions, or after early returns
|
||||
3. Always called in the same order on every render
|
||||
4. Only inside React function components or custom hooks (functions starting with `use`)
|
||||
|
||||
```tsx
|
||||
// WRONG: conditional hook
|
||||
function Foo({ enabled }: { enabled: boolean }) {
|
||||
if (enabled) {
|
||||
const [x, setX] = useState(0); // rule violation
|
||||
}
|
||||
}
|
||||
|
||||
// CORRECT: hook unconditional, condition inside
|
||||
function Foo({ enabled }: { enabled: boolean }) {
|
||||
const [x, setX] = useState(0);
|
||||
if (!enabled) return null;
|
||||
return <span>{x}</span>;
|
||||
}
|
||||
```
|
||||
|
||||
## `useEffect` — When NOT to Use
|
||||
|
||||
`useEffect` is for synchronizing with external systems (subscriptions, browser APIs, third-party libraries). It is **not** the right tool for:
|
||||
|
||||
- Derived state — compute it during render
|
||||
- Transforming data for rendering — compute it during render
|
||||
- Resetting state when a prop changes — use a `key` on the parent or derive from props
|
||||
- Notifying parents of state changes — call the callback in the event handler
|
||||
- Initializing app-level singletons — call the function module-side or in `main.tsx`
|
||||
|
||||
```tsx
|
||||
// WRONG: effect for derived state
|
||||
const [fullName, setFullName] = useState("");
|
||||
useEffect(() => {
|
||||
setFullName(`${first} ${last}`);
|
||||
}, [first, last]);
|
||||
|
||||
// CORRECT: derive during render
|
||||
const fullName = `${first} ${last}`;
|
||||
```
|
||||
|
||||
## Dependency Arrays
|
||||
|
||||
- Always include every reactive value referenced inside the effect/callback
|
||||
- Enable `react-hooks/exhaustive-deps` lint rule — never silence it without a comment explaining why
|
||||
- If the dep array grows unwieldy, the effect is doing too much — split it
|
||||
- Stable identity for functions passed in deps: wrap in `useCallback` only when the function is itself a dependency of another hook or passed to a memoized child
|
||||
|
||||
## Cleanup
|
||||
|
||||
Every subscription, interval, listener, or in-flight request must clean up.
|
||||
|
||||
```tsx
|
||||
useEffect(() => {
|
||||
const controller = new AbortController();
|
||||
fetch(url, { signal: controller.signal }).then(handleResponse);
|
||||
return () => controller.abort();
|
||||
}, [url]);
|
||||
```
|
||||
|
||||
```tsx
|
||||
useEffect(() => {
|
||||
const id = setInterval(tick, 1000);
|
||||
return () => clearInterval(id);
|
||||
}, []);
|
||||
```
|
||||
|
||||
Missing cleanup = race conditions when deps change, memory leaks on unmount.
|
||||
|
||||
## `useMemo` and `useCallback` — When Worth It
|
||||
|
||||
Default position: **do not memoize**. Add `useMemo` / `useCallback` only when:
|
||||
|
||||
1. The value is passed to a `React.memo`-wrapped child as a prop, and identity matters
|
||||
2. The value is a dependency of another `useEffect` / `useMemo` / `useCallback`
|
||||
3. The computation is measurably expensive (profile before assuming)
|
||||
|
||||
Premature memoization adds noise, hides bugs, and can be slower than the recompute it replaces.
|
||||
|
||||
## Custom Hooks
|
||||
|
||||
Extract a custom hook when:
|
||||
|
||||
- The same hook sequence (state + effect + computed) appears in 2+ components
|
||||
- The logic has a clear, nameable purpose (`useDebounce`, `useOnClickOutside`, `useLocalStorage`)
|
||||
- You want to test the logic independently of any component
|
||||
|
||||
Do NOT extract when:
|
||||
|
||||
- It would have a single caller — inline it
|
||||
- The "hook" is just `useState` with a different name — adds indirection, no value
|
||||
|
||||
```tsx
|
||||
export function useDebounce<T>(value: T, delay: number): T {
|
||||
const [debounced, setDebounced] = useState(value);
|
||||
useEffect(() => {
|
||||
const id = setTimeout(() => setDebounced(value), delay);
|
||||
return () => clearTimeout(id);
|
||||
}, [value, delay]);
|
||||
return debounced;
|
||||
}
|
||||
```
|
||||
|
||||
## `useState` Patterns
|
||||
|
||||
- Initial state from prop only at mount: pass a function `useState(() => computeInitial(prop))` when computation is expensive
|
||||
- Functional updater when the new state depends on the old: `setCount(c => c + 1)` — never `setCount(count + 1)` inside async or batched contexts
|
||||
- Group related state into one object only when they always change together; otherwise split into multiple `useState` calls
|
||||
- Use `useReducer` once state transitions are conditional on the previous state or there are 3+ related values
|
||||
|
||||
## `useRef` Patterns
|
||||
|
||||
- DOM refs for imperative APIs (focus, scroll, third-party libs)
|
||||
- Mutable container that does not trigger re-render (timer ids, previous values, "is mounted" flags)
|
||||
- Never read or write `ref.current` during render — only inside effects or event handlers
|
||||
- `useImperativeHandle` only when exposing a child API to a parent ref — last-resort escape hatch
|
||||
|
||||
## `useSyncExternalStore`
|
||||
|
||||
Use this hook to subscribe to any external store (browser API, third-party state lib, custom event emitter). It is the supported way to make external state safe with concurrent rendering.
|
||||
|
||||
```tsx
|
||||
const isOnline = useSyncExternalStore(
|
||||
(cb) => {
|
||||
window.addEventListener("online", cb);
|
||||
window.addEventListener("offline", cb);
|
||||
return () => {
|
||||
window.removeEventListener("online", cb);
|
||||
window.removeEventListener("offline", cb);
|
||||
};
|
||||
},
|
||||
() => navigator.onLine,
|
||||
() => true,
|
||||
);
|
||||
```
|
||||
|
||||
## React 19 Additions
|
||||
|
||||
- `use()` — unwrap promises and contexts inline; usable conditionally (only hook with that property)
|
||||
- `useFormStatus()` / `useFormState()` (or `useActionState`) — form submission state without prop drilling
|
||||
- `useOptimistic()` — optimistic UI updates while a server action is pending
|
||||
- `useTransition()` — mark non-urgent state updates so urgent ones stay responsive
|
||||
|
||||
When the project targets React 19+, prefer these over hand-rolled equivalents.
|
||||
|
||||
## Stale Closure Trap
|
||||
|
||||
Async handlers and intervals capture the values from the render where they were created. Fix by:
|
||||
|
||||
1. Using the functional updater form of `setState`
|
||||
2. Putting the changing value in the dep array of `useEffect` and rebuilding the handler
|
||||
3. Reading from a ref that is kept in sync
|
||||
|
||||
## Lint Configuration
|
||||
|
||||
Required rules:
|
||||
|
||||
```json
|
||||
{
|
||||
"rules": {
|
||||
"react-hooks/rules-of-hooks": "error",
|
||||
"react-hooks/exhaustive-deps": "warn"
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
Treat `exhaustive-deps` warnings as errors in CI for new code.
|
||||
@@ -0,0 +1,194 @@
|
||||
---
|
||||
paths:
|
||||
- "**/*.tsx"
|
||||
- "**/*.jsx"
|
||||
- "**/components/**/*.ts"
|
||||
- "**/components/**/*.js"
|
||||
- "**/app/**/*.tsx"
|
||||
- "**/pages/**/*.tsx"
|
||||
---
|
||||
# React Patterns
|
||||
|
||||
> This file extends [typescript/patterns.md](../typescript/patterns.md) and [common/patterns.md](../common/patterns.md) with React specific content. For hook-specific rules see [hooks.md](./hooks.md).
|
||||
|
||||
## Container / Presentational Split
|
||||
|
||||
Container components own data fetching, state, and side effects. Presentational components receive props and render — no service calls, no hooks beyond local UI state.
|
||||
|
||||
```tsx
|
||||
// Container — owns data
|
||||
export function UserPage({ userId }: { userId: string }) {
|
||||
const { data: user, isLoading } = useUser(userId);
|
||||
if (isLoading) return <Spinner />;
|
||||
if (!user) return <NotFound />;
|
||||
return <UserCard user={user} onSelect={handleSelect} />;
|
||||
}
|
||||
|
||||
// Presentational — pure
|
||||
export function UserCard({ user, onSelect }: { user: User; onSelect: (id: string) => void }) {
|
||||
return <button onClick={() => onSelect(user.id)}>{user.name}</button>;
|
||||
}
|
||||
```
|
||||
|
||||
## State Location Decision Tree
|
||||
|
||||
1. Used by one component → `useState` inside it
|
||||
2. Used by parent + a few children → lift to nearest common ancestor, pass via props
|
||||
3. Used across distant branches → React Context **for low-frequency reads only** (theme, auth, locale)
|
||||
4. High-frequency updates shared across the tree → external store (Zustand, Jotai, Redux Toolkit)
|
||||
5. Server-derived data → server-state library (TanStack Query, SWR, RSC fetch) — not application state
|
||||
|
||||
Context misused for frequently changing values causes every consumer to re-render on every update.
|
||||
|
||||
## Server / Client Component Boundary (RSC, Next.js App Router)
|
||||
|
||||
- Server Components are the default — they run on the server, do not ship to the client, and can `await` directly
|
||||
- Client Components opt in with `"use client"` at the top of the file
|
||||
- Data flows down: a Server Component can render a Client Component and pass serializable props
|
||||
- A Client Component cannot import a Server Component, but it can receive one via `children` or named slots
|
||||
|
||||
```tsx
|
||||
// Server (default)
|
||||
export default async function Page() {
|
||||
const user = await fetchUser();
|
||||
return <UserClient user={user} />;
|
||||
}
|
||||
|
||||
// Client
|
||||
"use client";
|
||||
export function UserClient({ user }: { user: User }) {
|
||||
const [tab, setTab] = useState("profile");
|
||||
return <Tabs value={tab} onChange={setTab}>{user.name}</Tabs>;
|
||||
}
|
||||
```
|
||||
|
||||
- Never import `"server-only"` packages (DB clients, secrets) from a Client Component file — wrap them in a Server Component or Server Action
|
||||
- Mark sensitive modules with `import "server-only"` so the bundler errors if a client file imports them
|
||||
|
||||
## Suspense + Error Boundaries
|
||||
|
||||
Every Suspense boundary needs an Error Boundary above it. The pair handles both states.
|
||||
|
||||
```tsx
|
||||
<ErrorBoundary fallback={<ErrorView />}>
|
||||
<Suspense fallback={<Skeleton />}>
|
||||
<UserDetails id={id} />
|
||||
</Suspense>
|
||||
</ErrorBoundary>
|
||||
```
|
||||
|
||||
- Place Suspense boundaries close to where data is needed, not at the route root
|
||||
- Multiple narrower boundaries reveal loaded content progressively
|
||||
- Error Boundary must be a Class Component (React 19 has no functional equivalent yet) OR use a library wrapper such as `react-error-boundary`
|
||||
|
||||
## Forms
|
||||
|
||||
### Uncontrolled (React 19 + form actions)
|
||||
|
||||
Prefer uncontrolled inputs with form actions when the form has a clear submit step. The browser owns the value; React reads it via `FormData` on submit.
|
||||
|
||||
```tsx
|
||||
async function action(formData: FormData) {
|
||||
"use server";
|
||||
await saveUser({ name: String(formData.get("name")) });
|
||||
}
|
||||
|
||||
export function UserForm() {
|
||||
return (
|
||||
<form action={action}>
|
||||
<input name="name" required />
|
||||
<button type="submit">Save</button>
|
||||
</form>
|
||||
);
|
||||
}
|
||||
```
|
||||
|
||||
### Controlled
|
||||
|
||||
Use controlled inputs when the value drives other UI, requires real-time validation, or formatting.
|
||||
|
||||
```tsx
|
||||
const [email, setEmail] = useState("");
|
||||
return <input value={email} onChange={(e) => setEmail(e.target.value)} />;
|
||||
```
|
||||
|
||||
### Form Libraries
|
||||
|
||||
For complex forms (multi-step, dynamic field arrays, cross-field validation), use a library:
|
||||
|
||||
- React Hook Form — minimal re-renders, uncontrolled-first
|
||||
- TanStack Form — typed, framework-agnostic
|
||||
- Final Form — when subscription-based re-renders matter
|
||||
|
||||
## Data Fetching
|
||||
|
||||
| Strategy | When |
|
||||
|---|---|
|
||||
| RSC fetch (`await` in Server Component) | Per-request data in Next.js App Router, no client-side cache needed |
|
||||
| TanStack Query | Client-side cache, mutations, optimistic updates, polling |
|
||||
| SWR | Lightweight cache + revalidation, simpler than TanStack Query |
|
||||
| `fetch` in `useEffect` | Avoid — race conditions, no cache, no retry. Only acceptable for one-off fire-and-forget |
|
||||
|
||||
Never fetch in a `useEffect` when a real cache library is available — they handle deduping, cache invalidation, error retry, and Suspense integration.
|
||||
|
||||
## Lists and Keys
|
||||
|
||||
- `key` must be stable across renders — never `index` for any list that can reorder, insert, or delete
|
||||
- `key` must be unique among siblings, not globally
|
||||
- A reordered list with index keys causes state in child components to attach to the wrong row
|
||||
|
||||
## Composition over Inheritance
|
||||
|
||||
- Pass `children` for slot-style composition
|
||||
- Pass render-prop functions for parameterized rendering
|
||||
- Pass component types for plug-in points: `renderItem={UserRow}`
|
||||
- Never extend a component class to specialize behavior
|
||||
|
||||
## Compound Components
|
||||
|
||||
For related controls (Tabs, Accordion, Menu), use compound components sharing state via Context:
|
||||
|
||||
```tsx
|
||||
<Tabs defaultValue="profile">
|
||||
<Tabs.List>
|
||||
<Tabs.Trigger value="profile">Profile</Tabs.Trigger>
|
||||
<Tabs.Trigger value="settings">Settings</Tabs.Trigger>
|
||||
</Tabs.List>
|
||||
<Tabs.Panel value="profile"><ProfileForm /></Tabs.Panel>
|
||||
<Tabs.Panel value="settings"><SettingsForm /></Tabs.Panel>
|
||||
</Tabs>
|
||||
```
|
||||
|
||||
## Portals
|
||||
|
||||
Use `createPortal` for modals, tooltips, toast containers — anything that must escape the parent's `overflow: hidden` or `z-index` stacking context. Render to a stable DOM node mounted in `index.html`.
|
||||
|
||||
## Refs and Forwarding (React 19+)
|
||||
|
||||
React 19 lets function components accept `ref` as a regular prop — `forwardRef` is no longer required.
|
||||
|
||||
```tsx
|
||||
export function Input({ ref, ...rest }: { ref?: React.Ref<HTMLInputElement> } & InputProps) {
|
||||
return <input ref={ref} {...rest} />;
|
||||
}
|
||||
```
|
||||
|
||||
Older codebases on React 18 still need `forwardRef`.
|
||||
|
||||
## Out of Scope (Pointer Sections)
|
||||
|
||||
### Next.js (App Router)
|
||||
|
||||
- Server Actions, Route Handlers, Middleware, Parallel/Intercepted Routes, streaming Metadata
|
||||
- Treated as a separate framework concern — when adding deep Next-specific patterns, propose a dedicated `rules/nextjs/` track
|
||||
- For now follow Next.js official docs for App Router specifics
|
||||
|
||||
### React Native
|
||||
|
||||
- Platform-specific imports (`Platform.OS`, `.ios.tsx` / `.android.tsx`), `StyleSheet`, navigation libraries (React Navigation, Expo Router)
|
||||
- Treated as a separate track — `rules/react-native/` is not yet present
|
||||
- React core hooks/patterns from this file still apply
|
||||
|
||||
## Skill Reference
|
||||
|
||||
For React-specific deep dives see `skills/react-patterns/SKILL.md`. For cross-framework frontend concerns see `skills/frontend-patterns/SKILL.md`. For accessibility see `skills/accessibility/SKILL.md`.
|
||||
@@ -0,0 +1,180 @@
|
||||
---
|
||||
paths:
|
||||
- "**/*.tsx"
|
||||
- "**/*.jsx"
|
||||
- "**/components/**/*.ts"
|
||||
- "**/app/**/*.ts"
|
||||
- "**/pages/**/*.ts"
|
||||
---
|
||||
# React Security
|
||||
|
||||
> This file extends [typescript/security.md](../typescript/security.md) and [common/security.md](../common/security.md) with React specific content.
|
||||
|
||||
## XSS via `dangerouslySetInnerHTML`
|
||||
|
||||
CRITICAL. The prop name is deliberately scary — treat every usage as a code review halt.
|
||||
|
||||
```tsx
|
||||
// CRITICAL: unsanitized user input
|
||||
<div dangerouslySetInnerHTML={{ __html: userBio }} />
|
||||
|
||||
// CORRECT options:
|
||||
// 1. Render as text
|
||||
<div>{userBio}</div>
|
||||
|
||||
// 2. Render parsed markdown via a library that sanitizes
|
||||
<ReactMarkdown>{userBio}</ReactMarkdown>
|
||||
|
||||
// 3. If raw HTML is required, sanitize first with DOMPurify
|
||||
import DOMPurify from "isomorphic-dompurify";
|
||||
<div dangerouslySetInnerHTML={{ __html: DOMPurify.sanitize(userBio) }} />
|
||||
```
|
||||
|
||||
Audit checklist for every `dangerouslySetInnerHTML` call:
|
||||
|
||||
- Is the input always under our control? Document the source.
|
||||
- If user-derived: is it sanitized at the **same call site**? (Sanitization at the API boundary is acceptable only if every consumer is verified.)
|
||||
- Is the sanitizer config allowlisting tags, not denylisting?
|
||||
|
||||
## Unsafe URL Schemes
|
||||
|
||||
`javascript:` and `data:` URLs in `href`, `src`, and `xlink:href` execute arbitrary code.
|
||||
|
||||
```tsx
|
||||
// CRITICAL: javascript: URL injection
|
||||
<a href={user.website}>Visit</a> // if user.website = "javascript:alert(1)"
|
||||
|
||||
// CORRECT: validate scheme
|
||||
function safeUrl(url: string): string | undefined {
|
||||
try {
|
||||
const parsed = new URL(url);
|
||||
if (["http:", "https:", "mailto:"].includes(parsed.protocol)) return url;
|
||||
} catch {
|
||||
return undefined;
|
||||
}
|
||||
return undefined;
|
||||
}
|
||||
<a href={safeUrl(user.website)}>Visit</a>
|
||||
```
|
||||
|
||||
React 18+ blocks `javascript:` URLs in `href` and logs a warning, but `data:` URLs and other schemes still slip through. Always validate.
|
||||
|
||||
## `target="_blank"` Without `rel`
|
||||
|
||||
`<a target="_blank">` without `rel="noopener noreferrer"` lets the target page access `window.opener` and run navigation hijacks.
|
||||
|
||||
```tsx
|
||||
// WRONG
|
||||
<a href={externalUrl} target="_blank">External</a>
|
||||
|
||||
// CORRECT
|
||||
<a href={externalUrl} target="_blank" rel="noopener noreferrer">External</a>
|
||||
```
|
||||
|
||||
Modern browsers default to `noopener` when `target="_blank"`, but do not rely on browser defaults — be explicit.
|
||||
|
||||
## Server Action Input Validation
|
||||
|
||||
Server Actions (`"use server"`) run with the same trust level as a public API endpoint. Validate every input.
|
||||
|
||||
```tsx
|
||||
"use server";
|
||||
import { z } from "zod";
|
||||
|
||||
const Input = z.object({
|
||||
email: z.string().email(),
|
||||
age: z.number().int().min(0).max(120),
|
||||
});
|
||||
|
||||
export async function updateUser(_state: unknown, formData: FormData) {
|
||||
const parsed = Input.safeParse({
|
||||
email: formData.get("email"),
|
||||
age: Number(formData.get("age")),
|
||||
});
|
||||
if (!parsed.success) return { error: parsed.error.flatten() };
|
||||
// ...
|
||||
}
|
||||
```
|
||||
|
||||
- Authenticate inside the action — do not trust the client-side route gate
|
||||
- Authorize: confirm the current user has permission for the specific record they are mutating
|
||||
- Rate limit sensitive actions
|
||||
|
||||
## Secret Exposure via Env Vars
|
||||
|
||||
Prefixed env vars are bundled into the client. Treat them as public.
|
||||
|
||||
| Framework | Public prefix | Private |
|
||||
|---|---|---|
|
||||
| Next.js | `NEXT_PUBLIC_*` | All others |
|
||||
| Vite | `VITE_*` | `.env` server-side only |
|
||||
| Create React App | `REACT_APP_*` | None — CRA exposes all |
|
||||
| Remix | `process.env` access in `loader`/`action` only | Same |
|
||||
|
||||
```ts
|
||||
// CRITICAL: secret leaked to client bundle
|
||||
const apiKey = process.env.NEXT_PUBLIC_STRIPE_SECRET_KEY;
|
||||
```
|
||||
|
||||
Audit on every PR that touches env vars: would this string in the public bundle be a problem?
|
||||
|
||||
## Authentication / Authorization
|
||||
|
||||
- Never store sessions in `localStorage` — accessible to any XSS. Use httpOnly secure cookies.
|
||||
- Never trust client-set state to gate sensitive UI. Render-gating in JSX prevents display, not access — the API must enforce.
|
||||
- CSRF: cookie-based auth requires CSRF tokens or `SameSite=Strict`/`Lax` cookies
|
||||
- Use double-submit cookies or origin verification for form actions when not using framework defaults
|
||||
|
||||
## Content Security Policy (CSP)
|
||||
|
||||
Configure server-side. The minimum acceptable CSP for a React app:
|
||||
|
||||
```
|
||||
default-src 'self';
|
||||
script-src 'self' 'nonce-{REQUEST_NONCE}';
|
||||
style-src 'self' 'unsafe-inline';
|
||||
img-src 'self' data: https:;
|
||||
connect-src 'self' https://api.example.com;
|
||||
frame-ancestors 'none';
|
||||
```
|
||||
|
||||
- Avoid `unsafe-inline` and `unsafe-eval` in `script-src`
|
||||
- For SSR with inline scripts (Next.js streaming, hydration data), use per-request nonces — both Next.js and Remix support nonce injection
|
||||
- `style-src 'unsafe-inline'` is often unavoidable for CSS-in-JS libraries — document the tradeoff
|
||||
|
||||
## Prototype Pollution via Object Spread
|
||||
|
||||
```tsx
|
||||
// WRONG: untrusted JSON spread directly into state
|
||||
const update = await req.json();
|
||||
setState({ ...state, ...update }); // attacker controls __proto__
|
||||
|
||||
// CORRECT: parse with a schema, or guard keys
|
||||
const Allowed = z.object({ name: z.string(), email: z.string().email() });
|
||||
const parsed = Allowed.parse(await req.json());
|
||||
setState({ ...state, ...parsed });
|
||||
```
|
||||
|
||||
## SSR Template Injection
|
||||
|
||||
When using `renderToString` or `renderToPipeableStream`:
|
||||
|
||||
- All values rendered inside JSX are escaped by React — safe
|
||||
- Values passed to `dangerouslySetInnerHTML` are NOT escaped — same rules as client
|
||||
- Manually constructed HTML wrappers around the React output must be escaped or sanitized — never concatenate user input into the surrounding HTML template
|
||||
|
||||
## Third-Party Components
|
||||
|
||||
- Audit `npm audit` before adding any UI library
|
||||
- Check that the library does not internally use `dangerouslySetInnerHTML` on its input (e.g., rich text editors)
|
||||
- Pin versions, review changelogs before major upgrades
|
||||
- Be wary of components that accept HTML strings as props
|
||||
|
||||
## Source Map Exposure in Production
|
||||
|
||||
Production builds should ship without source maps, or with sourcemaps uploaded to an error tracker (Sentry) and stripped from the public bundle. Public source maps leak internal logic and file structure.
|
||||
|
||||
## Agent Support
|
||||
|
||||
- Use `security-reviewer` agent for comprehensive security audits across the codebase
|
||||
- Use `react-reviewer` agent for React-specific patterns and the above rules in active code review
|
||||
@@ -0,0 +1,208 @@
|
||||
---
|
||||
paths:
|
||||
- "**/*.test.tsx"
|
||||
- "**/*.test.jsx"
|
||||
- "**/*.spec.tsx"
|
||||
- "**/*.spec.jsx"
|
||||
- "**/__tests__/**/*.ts"
|
||||
- "**/__tests__/**/*.tsx"
|
||||
---
|
||||
# React Testing
|
||||
|
||||
> This file extends [typescript/testing.md](../typescript/testing.md) and [common/testing.md](../common/testing.md) with React specific content.
|
||||
|
||||
## Library Choice
|
||||
|
||||
- **React Testing Library (RTL)** — the standard for component testing. Tests behavior through the rendered DOM.
|
||||
- **Vitest** — preferred runner for new Vite-based projects. Faster than Jest, native ESM, same API.
|
||||
- **Jest** — still the default for Next.js / CRA projects. RTL works identically.
|
||||
- **Playwright Component Testing** — when component tests need a real browser engine (animation, layout, complex events)
|
||||
- **Cypress Component Testing** — alternative real-browser component runner
|
||||
|
||||
Pick one component test runner per project — do not mix RTL + Playwright CT in the same repo.
|
||||
|
||||
## Core Principle
|
||||
|
||||
Test what the user sees and does, not implementation details.
|
||||
|
||||
- Query by accessible role first, then label, then text — fall back to `data-testid` only when nothing else fits
|
||||
- Never assert on internal state, props passed to children, or which hooks were called
|
||||
- Refactor without breaking tests = the test was testing behavior; that is the goal
|
||||
|
||||
## Query Priority
|
||||
|
||||
RTL exposes queries in three families. Use this priority order top-down:
|
||||
|
||||
1. **Accessible to everyone**
|
||||
- `getByRole(role, { name })` — primary choice
|
||||
- `getByLabelText` — for form inputs
|
||||
- `getByPlaceholderText` — when no label is available (and add a label)
|
||||
- `getByText` — for non-interactive text
|
||||
- `getByDisplayValue` — for form fields with a current value
|
||||
|
||||
2. **Semantic queries**
|
||||
- `getByAltText` — for images
|
||||
- `getByTitle` — last resort, low accessibility value
|
||||
|
||||
3. **Test IDs**
|
||||
- `getByTestId("some-id")` — escape hatch only, when none of the above work
|
||||
|
||||
`getBy*` throws when no match. `queryBy*` returns null (use for asserting absence). `findBy*` returns a promise (use for async).
|
||||
|
||||
## User Interaction
|
||||
|
||||
Prefer `userEvent` over `fireEvent`. `userEvent` simulates real browser sequences (focus, keydown, beforeinput, input, keyup) — `fireEvent` dispatches a single synthetic event.
|
||||
|
||||
```tsx
|
||||
import userEvent from "@testing-library/user-event";
|
||||
|
||||
test("submits the form", async () => {
|
||||
const user = userEvent.setup();
|
||||
render(<UserForm onSubmit={handleSubmit} />);
|
||||
|
||||
await user.type(screen.getByLabelText("Email"), "user@example.com");
|
||||
await user.click(screen.getByRole("button", { name: /save/i }));
|
||||
|
||||
expect(handleSubmit).toHaveBeenCalledWith({ email: "user@example.com" });
|
||||
});
|
||||
```
|
||||
|
||||
- Always `await` `userEvent` calls — they are async
|
||||
- Call `userEvent.setup()` once at the top of each test, then reuse the returned `user`
|
||||
|
||||
## Async Assertions
|
||||
|
||||
```tsx
|
||||
// WRONG: synchronous query for async-rendered content
|
||||
expect(screen.getByText("Loaded")).toBeInTheDocument(); // throws — not in DOM yet
|
||||
|
||||
// CORRECT: findBy* (returns a promise, retries)
|
||||
expect(await screen.findByText("Loaded")).toBeInTheDocument();
|
||||
|
||||
// CORRECT: waitFor for non-element assertions
|
||||
await waitFor(() => expect(saveSpy).toHaveBeenCalled());
|
||||
```
|
||||
|
||||
- `findBy*` for async element appearance
|
||||
- `waitFor` for async expectations on side effects or other matchers
|
||||
- Never `setTimeout` + assertion — flaky
|
||||
|
||||
## Network Mocking with MSW
|
||||
|
||||
Use Mock Service Worker for any test that hits a network boundary. MSW runs at the network layer, so the component, hooks, and fetch library all behave as in production.
|
||||
|
||||
```tsx
|
||||
// test setup
|
||||
import { setupServer } from "msw/node";
|
||||
import { http, HttpResponse } from "msw";
|
||||
|
||||
const server = setupServer(
|
||||
http.get("/api/users/:id", ({ params }) =>
|
||||
HttpResponse.json({ id: params.id, name: "Alice" }),
|
||||
),
|
||||
);
|
||||
|
||||
beforeAll(() => server.listen());
|
||||
afterEach(() => server.resetHandlers());
|
||||
afterAll(() => server.close());
|
||||
```
|
||||
|
||||
Per-test override:
|
||||
|
||||
```tsx
|
||||
test("renders error on 500", async () => {
|
||||
server.use(http.get("/api/users/:id", () => new HttpResponse(null, { status: 500 })));
|
||||
render(<UserPage id="1" />);
|
||||
expect(await screen.findByText(/something went wrong/i)).toBeInTheDocument();
|
||||
});
|
||||
```
|
||||
|
||||
## Avoid Snapshot Tests for Components
|
||||
|
||||
Snapshots of rendered output are brittle, hard to review, and rubber-stamped by reviewers. Use them only for:
|
||||
|
||||
- Pure data serialization (e.g., a transformer that produces a stable string)
|
||||
- Catching unintended regressions in non-visual output
|
||||
|
||||
For component visual regression, use Playwright / Cypress / Percy screenshots — actual visual diffs, not DOM diffs.
|
||||
|
||||
## Test Setup Helpers
|
||||
|
||||
Wrap providers once:
|
||||
|
||||
```tsx
|
||||
function renderWithProviders(ui: React.ReactElement) {
|
||||
return render(
|
||||
<QueryClientProvider client={new QueryClient()}>
|
||||
<ThemeProvider theme={lightTheme}>
|
||||
<Router>{ui}</Router>
|
||||
</ThemeProvider>
|
||||
</QueryClientProvider>,
|
||||
);
|
||||
}
|
||||
```
|
||||
|
||||
Export from `test-utils.tsx` and use everywhere.
|
||||
|
||||
## Custom Hook Testing
|
||||
|
||||
Use `renderHook` from RTL:
|
||||
|
||||
```tsx
|
||||
import { renderHook, act } from "@testing-library/react";
|
||||
|
||||
test("useCounter increments", () => {
|
||||
const { result } = renderHook(() => useCounter());
|
||||
act(() => result.current.increment());
|
||||
expect(result.current.count).toBe(1);
|
||||
});
|
||||
```
|
||||
|
||||
- Always wrap state-changing calls in `act`
|
||||
- Always test through the public hook API, not internal implementation
|
||||
|
||||
## Accessibility Assertions
|
||||
|
||||
```tsx
|
||||
import { axe } from "vitest-axe"; // or jest-axe
|
||||
|
||||
test("UserCard has no a11y violations", async () => {
|
||||
const { container } = render(<UserCard user={mockUser} />);
|
||||
expect(await axe(container)).toHaveNoViolations();
|
||||
});
|
||||
```
|
||||
|
||||
Run axe assertions in component tests — catches missing labels, ARIA misuse, color contrast (limited).
|
||||
|
||||
## When to Reach for Playwright / Cypress
|
||||
|
||||
Component test with RTL + JSDOM cannot:
|
||||
|
||||
- Test real layout (flexbox, grid, viewport-dependent rendering)
|
||||
- Test scrolling, drag-and-drop, paste from clipboard
|
||||
- Test browser-native animation, CSS transitions
|
||||
- Test cross-frame interactions (iframes, popups)
|
||||
|
||||
For those, use Playwright Component Testing or end-to-end Playwright/Cypress runs. See [e2e-testing skill](../../skills/e2e-testing/SKILL.md).
|
||||
|
||||
## Coverage Targets
|
||||
|
||||
| Layer | Target |
|
||||
|---|---|
|
||||
| Pure utility functions | ≥90% |
|
||||
| Custom hooks | ≥85% |
|
||||
| Components (presentational) | ≥80% — behavior, not lines |
|
||||
| Container components | ≥70% — golden paths + error states |
|
||||
| Pages (E2E covered separately) | Smoke test per route minimum |
|
||||
|
||||
## Anti-Patterns
|
||||
|
||||
- Asserting on `container.querySelector` — bypasses accessibility queries
|
||||
- Asserting on number of renders — implementation detail
|
||||
- Mocking React hooks (`jest.mock("react", ...)`) — refactor the component instead
|
||||
- Mocking child components by default — tests the integration, not the parent in isolation
|
||||
- Manual `act()` warnings ignored — they indicate real bugs
|
||||
|
||||
## Skill Reference
|
||||
|
||||
See `skills/react-testing/SKILL.md` for end-to-end test examples, MSW patterns, and accessibility test scaffolding.
|
||||
Reference in New Issue
Block a user