mirror of
https://github.com/affaan-m/everything-claude-code.git
synced 2026-05-30 21:54:17 +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.
* feat(skills): add react-patterns, react-testing, react-performance
Three new skills under skills/ following the SKILL.md convention.
- react-patterns: React 18/19 idioms — hooks discipline, state
location decision tree, server/client component boundary,
Suspense + error boundaries, form actions (React 19), data
fetching matrix, composition recipes, accessibility-first.
- react-testing: React Testing Library + Vitest/Jest, query
priority order, userEvent, MSW network mocking, axe a11y
assertions, RTL vs Playwright CT boundary, TDD workflow.
- react-performance: 70-rule performance ruleset adapted from
Vercel Labs react-best-practices (MIT) across 8 priority
categories — waterfalls, bundle size, server-side, client
fetch, re-render, rendering, JS micro, advanced patterns.
Includes Lighthouse / Web Vitals mapping and attribution to
upstream.
Cross-links between the three skills and out to frontend-patterns,
accessibility, e2e-testing, tdd-workflow.
* feat(agents): add react-reviewer and react-build-resolver
Two new agents covering React-specific code review and build error
resolution, plus matching .kiro/ mirrors and a routing pointer
edit on typescript-reviewer.
- react-reviewer: slim React-only lanes (hooks rules,
dangerouslySetInnerHTML, unsafe URL schemes, key prop, state
mutation, derived-state-in-effect, server/client component
boundary, accessibility, render performance, Server Action
validation, env-var leaks). Explicitly delegates generic
TypeScript/async/Node concerns to typescript-reviewer. Both
agents should be invoked together on .tsx/.jsx PRs.
- react-build-resolver: React build/bundler/runtime hydration
failures across Vite, webpack, Next.js, CRA, Parcel, esbuild,
Bun, Rsbuild. Handles JSX/TSX compile errors, tsconfig fixes,
Next.js App Router server/client boundary errors, hydration
mismatches, duplicated React copies, Tailwind/PostCSS pipeline.
- .kiro/agents/react-reviewer.json + react-build-resolver.json:
Kiro IDE format mirrors following the per-language precedent.
- typescript-reviewer: routing pointer added to its MEDIUM React
block — defers to /react-review for React-specific concerns
while keeping its block as fallback for repos that only invoke
typescript-reviewer.
All agents carry the standard Prompt Defense Baseline stanza.
* feat(commands): add /react-review /react-build /react-test
Three new slash commands invoking the React agents.
- /react-review: invokes react-reviewer. Documents the routing
rule with typescript-reviewer — both should run together on
TSX/JSX PRs. Lists CRITICAL/HIGH/MEDIUM rule categories and
the automated checks (eslint with react-hooks + jsx-a11y,
tsc --noEmit, npm audit).
- /react-build: invokes react-build-resolver. Documents bundler
detection, common failure patterns, fix strategy, and stop
conditions.
- /react-test: enforces TDD with React Testing Library + Vitest
or Jest, behavior-focused queries, userEvent + MSW patterns,
axe accessibility assertions, coverage targets.
Each command file has the required description: frontmatter and
follows the per-language command convention (cpp-test, go-test,
kotlin-test, etc.).
* chore: wire react track into manifests and stack mappings
- agent.yaml: add react-patterns, react-performance, react-testing
to the skills array; add react-build, react-review, react-test to
the commands array (alphabetically inserted to satisfy the
ci/agent-yaml-surface sync test).
- config/project-stack-mappings.json: extend the `react` stack
entry — add "react" to rules array (was ["common","typescript",
"web"]); add react-patterns, react-performance, react-testing,
accessibility to the skills array.
- docs/COMMAND-REGISTRY.json: bump totalCommands 75 -> 78; add
three new entries (react-build, react-review, react-test) with
primaryAgents / allAgents / skills wiring. react-review's
allAgents includes typescript-reviewer to reflect the dual-agent
routing convention.
- CLAUDE.md: add Skills-table row mapping *.tsx / *.jsx /
components/** to react-patterns + react-testing skills and
the /react-review, /react-build, /react-test commands.
* chore(catalog): sync counts to 62 agents / 78 commands / 235 skills
Auto-generated via `node scripts/ci/catalog.js --write --text`
after the react track additions:
- 2 new agents: react-reviewer, react-build-resolver (60 -> 62)
- 3 new commands: react-build, react-review, react-test (75 -> 78)
- 3 new skills: react-patterns, react-performance, react-testing
(232 -> 235)
Files updated by the catalog sync:
- .claude-plugin/plugin.json description string
- .claude-plugin/marketplace.json plugin description
- README.md quick-start summary, project tree, feature parity tables
- README.zh-CN.md quick-start summary
- AGENTS.md project structure summary
- docs/zh-CN/README.md parity table
- docs/zh-CN/AGENTS.md project structure summary
All counts now match the filesystem catalog (verified by
ci/catalog.test.js).
* feat(kiro): add react agent markdown companions to JSON entries
* feat(kiro): add react skills into manifests
* fix(ci): sync catalog counts, registry, and package files for react track
- .claude-plugin/{plugin,marketplace}.json: bump description counts to 62/235/78
- docs/COMMAND-REGISTRY.json: regenerate to include quality-gate and react commands
- package.json: add skills/react-{patterns,performance,testing}/ to files allowlist so npm-publish-surface aligns with install-modules manifest
* fix(react): address PR #2024 review feedback
Critical:
- Remove undefined/.claude/session-aliases.json containing __proto__ prototype-pollution
fixture committed by accident in a7333c14
High:
- agents/react-build-resolver.md: replace brittle `test -o $(grep -l ...)` and
`test -a -n $(grep ...)` detection with explicit `{ ... || grep -q ...; }` so
bundler detection no longer breaks when grep returns empty
- agents/react-build-resolver.md: drop hardcoded `npm i react@^19 react-dom@^19`
remediation; replace with version-agnostic pair-upgrade note that honors the
project's installed major (17/18/19) — surgical fix principle
- commands/react-review.md: guard `tsc --noEmit -p tsconfig.json` with
`[ -f tsconfig.json ] &&` so the review skips cleanly on JS-only projects
Medium:
- rules/react/security.md: correct the React-18-blocks-javascript-URL claim
(React only warns in dev; production navigation is not blocked)
- rules/react/security.md: correct CRA env-var exposure row (CRA exposes
REACT_APP_*, NODE_ENV, PUBLIC_URL — not 'all' variables)
- skills/react-testing/SKILL.md: instantiate QueryClient once outside the
wrapper closure so React Query cache survives re-renders (flaky-test fix)
- skills/react-testing/SKILL.md: restore console.error spy with mockRestore()
in a try/finally so the mock does not leak across tests
- commands/react-test.md: switch outer example-session fence to 4 backticks
so the inner ```tsx/```bash blocks don't prematurely terminate it
* fix(kiro): mirror react-build-resolver react 19 conditional remediation
Discussion r3272907106 flagged the kiro json variant still carrying the hardcoded
'npm i react@^19 react-dom@^19' line that the .md companion already dropped.
Replace with the same conditional, version-agnostic guidance so both variants
stay in sync.
* fix(react): bump react-build example session fence to 4 backticks
Discussion r3272907144 flagged the same nested-fence issue in
commands/react-build.md that we fixed earlier in commands/react-test.md.
The outer triple-backtick text block was being prematurely terminated by
the inner bash/tsx fences inside the Example Session.
* fix(react): bump react-review example usage fence to 4 backticks
Discussion r3272907201 flagged the same nested-fence issue in
commands/react-review.md. The outer triple-backtick text block was
being prematurely terminated by the inner tsx/ts fences inside the
Example Usage transcript.
* fix(docs): clarify commands row as legacy shims in feature parity table
Discussion r3272912003: README comparison table said 'PASS: 78 commands'
while the install-section and quick-start prose use 'legacy command shims'.
Aligned the comparison-table cell to 'PASS: 78 commands (legacy shims)' so
the count word survives the catalog-validator regex while making the legacy
nature explicit.
Widened the catalog comparison-table commands regex to tolerate an optional
parenthetical after the count word, so both the existing 'X commands' and
the new 'X commands (legacy shims)' phrasings validate without breaking
older READMEs/translations.
* Update rules/react/security.md
Co-authored-by: cubic-dev-ai[bot] <191113872+cubic-dev-ai[bot]@users.noreply.github.com>
* fix(react): guard tsc in react-build-resolver diagnostic commands
Discussion r3288910205: the agent prompt instructed an unconditional
'tsc --noEmit -p tsconfig.json', which adds noise (or hard-fails) on
JavaScript-only projects with no tsconfig.json or no installed TypeScript.
Replaced with 'test -f tsconfig.json && npx --yes tsc --noEmit -p tsconfig.json'
in both variants:
- agents/react-build-resolver.md
- .kiro/agents/react-build-resolver.json (prompt string mirrored)
Mirrors the same guard already applied to commands/react-review.md in de135f61.
* fix(react): pin tsc resolution to local install in build resolver
Discussion r3289054157: previous fix used 'npx --yes tsc' which auto-installs
the latest TypeScript from npm when none is local, producing version drift
and non-reproducible typecheck results across machines.
Switched to 'npx --no-install tsc' in both variants so the diagnostic uses
only the project's pinned TypeScript and fails fast if it isn't installed:
- agents/react-build-resolver.md
- .kiro/agents/react-build-resolver.json (prompt string mirrored)
* feat(counts): resolve counts for agents, skills...
* fix(ci): regen command registry for golang-testing entry
Removes stale kotlin-patterns entry to satisfy command-registry:check.
* fix: keep local Claude settings out of React track PR
---------
Co-authored-by: AlexisLeDain <a.ledain@docoon.com>
Co-authored-by: cubic-dev-ai[bot] <191113872+cubic-dev-ai[bot]@users.noreply.github.com>
Co-authored-by: Affaan Mustafa <affaan@dcube.ai>
181 lines
6.2 KiB
Markdown
181 lines
6.2 KiB
Markdown
---
|
|
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 warns about `javascript:` URLs in `href` in development mode, but does not block them at runtime. `data:` URLs and other schemes also 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_*`, plus `NODE_ENV` and `PUBLIC_URL` | All others (anything without the `REACT_APP_` prefix is server-side only) |
|
|
| 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
|