tribunal-kit 3.0.0 → 3.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (226) hide show
  1. package/.agent/ARCHITECTURE.md +99 -99
  2. package/.agent/GEMINI.md +52 -52
  3. package/.agent/agents/accessibility-reviewer.md +187 -220
  4. package/.agent/agents/ai-code-reviewer.md +199 -233
  5. package/.agent/agents/backend-specialist.md +215 -238
  6. package/.agent/agents/code-archaeologist.md +161 -181
  7. package/.agent/agents/database-architect.md +184 -207
  8. package/.agent/agents/debugger.md +191 -218
  9. package/.agent/agents/dependency-reviewer.md +103 -136
  10. package/.agent/agents/devops-engineer.md +218 -238
  11. package/.agent/agents/documentation-writer.md +201 -221
  12. package/.agent/agents/explorer-agent.md +160 -180
  13. package/.agent/agents/frontend-reviewer.md +160 -194
  14. package/.agent/agents/frontend-specialist.md +248 -237
  15. package/.agent/agents/game-developer.md +48 -52
  16. package/.agent/agents/logic-reviewer.md +116 -149
  17. package/.agent/agents/mobile-developer.md +200 -223
  18. package/.agent/agents/mobile-reviewer.md +162 -195
  19. package/.agent/agents/orchestrator.md +181 -211
  20. package/.agent/agents/penetration-tester.md +157 -174
  21. package/.agent/agents/performance-optimizer.md +183 -203
  22. package/.agent/agents/performance-reviewer.md +178 -211
  23. package/.agent/agents/product-manager.md +142 -162
  24. package/.agent/agents/product-owner.md +6 -25
  25. package/.agent/agents/project-planner.md +142 -162
  26. package/.agent/agents/qa-automation-engineer.md +225 -242
  27. package/.agent/agents/security-auditor.md +174 -194
  28. package/.agent/agents/seo-specialist.md +193 -213
  29. package/.agent/agents/sql-reviewer.md +161 -194
  30. package/.agent/agents/supervisor-agent.md +184 -203
  31. package/.agent/agents/swarm-worker-contracts.md +17 -17
  32. package/.agent/agents/swarm-worker-registry.md +46 -46
  33. package/.agent/agents/test-coverage-reviewer.md +160 -193
  34. package/.agent/agents/test-engineer.md +0 -21
  35. package/.agent/agents/type-safety-reviewer.md +175 -208
  36. package/.agent/patterns/generator.md +9 -9
  37. package/.agent/patterns/inversion.md +12 -12
  38. package/.agent/patterns/pipeline.md +9 -9
  39. package/.agent/patterns/reviewer.md +13 -13
  40. package/.agent/patterns/tool-wrapper.md +9 -9
  41. package/.agent/rules/GEMINI.md +63 -63
  42. package/.agent/scripts/compress_skills.py +167 -0
  43. package/.agent/scripts/consolidate_skills.py +173 -0
  44. package/.agent/scripts/deep_compress.py +202 -0
  45. package/.agent/scripts/minify_context.py +80 -0
  46. package/.agent/scripts/security_scan.py +1 -1
  47. package/.agent/scripts/strip_tribunal.py +41 -0
  48. package/.agent/skills/agent-organizer/SKILL.md +92 -126
  49. package/.agent/skills/agentic-patterns/SKILL.md +0 -70
  50. package/.agent/skills/ai-prompt-injection-defense/SKILL.md +126 -160
  51. package/.agent/skills/api-patterns/SKILL.md +123 -215
  52. package/.agent/skills/api-security-auditor/SKILL.md +143 -177
  53. package/.agent/skills/app-builder/SKILL.md +326 -50
  54. package/.agent/skills/app-builder/templates/SKILL.md +13 -15
  55. package/.agent/skills/app-builder/templates/astro-static/TEMPLATE.md +16 -16
  56. package/.agent/skills/app-builder/templates/chrome-extension/TEMPLATE.md +22 -22
  57. package/.agent/skills/app-builder/templates/cli-tool/TEMPLATE.md +18 -18
  58. package/.agent/skills/app-builder/templates/electron-desktop/TEMPLATE.md +20 -20
  59. package/.agent/skills/app-builder/templates/express-api/TEMPLATE.md +17 -17
  60. package/.agent/skills/app-builder/templates/flutter-app/TEMPLATE.md +18 -18
  61. package/.agent/skills/app-builder/templates/monorepo-turborepo/TEMPLATE.md +21 -21
  62. package/.agent/skills/app-builder/templates/nextjs-fullstack/TEMPLATE.md +19 -19
  63. package/.agent/skills/app-builder/templates/nextjs-saas/TEMPLATE.md +26 -26
  64. package/.agent/skills/app-builder/templates/nextjs-static/TEMPLATE.md +26 -26
  65. package/.agent/skills/app-builder/templates/nuxt-app/TEMPLATE.md +19 -19
  66. package/.agent/skills/app-builder/templates/python-fastapi/TEMPLATE.md +18 -18
  67. package/.agent/skills/app-builder/templates/react-native-app/TEMPLATE.md +20 -20
  68. package/.agent/skills/appflow-wireframe/SKILL.md +87 -121
  69. package/.agent/skills/architecture/SKILL.md +82 -252
  70. package/.agent/skills/authentication-best-practices/SKILL.md +139 -173
  71. package/.agent/skills/bash-linux/SKILL.md +120 -154
  72. package/.agent/skills/behavioral-modes/SKILL.md +8 -69
  73. package/.agent/skills/brainstorming/SKILL.md +428 -104
  74. package/.agent/skills/building-native-ui/SKILL.md +143 -174
  75. package/.agent/skills/clean-code/SKILL.md +323 -360
  76. package/.agent/skills/code-review-checklist/SKILL.md +0 -62
  77. package/.agent/skills/config-validator/SKILL.md +107 -141
  78. package/.agent/skills/csharp-developer/SKILL.md +468 -528
  79. package/.agent/skills/database-design/SKILL.md +104 -369
  80. package/.agent/skills/deployment-procedures/SKILL.md +111 -145
  81. package/.agent/skills/devops-engineer/SKILL.md +295 -332
  82. package/.agent/skills/devops-incident-responder/SKILL.md +79 -113
  83. package/.agent/skills/doc.md +5 -5
  84. package/.agent/skills/documentation-templates/SKILL.md +19 -63
  85. package/.agent/skills/edge-computing/SKILL.md +123 -157
  86. package/.agent/skills/extract-design-system/SKILL.md +100 -134
  87. package/.agent/skills/framer-motion-expert/SKILL.md +111 -855
  88. package/.agent/skills/frontend-design/SKILL.md +151 -499
  89. package/.agent/skills/game-design-expert/SKILL.md +71 -105
  90. package/.agent/skills/game-engineering-expert/SKILL.md +88 -122
  91. package/.agent/skills/geo-fundamentals/SKILL.md +89 -124
  92. package/.agent/skills/github-operations/SKILL.md +279 -314
  93. package/.agent/skills/gsap-expert/SKILL.md +119 -826
  94. package/.agent/skills/i18n-localization/SKILL.md +104 -138
  95. package/.agent/skills/intelligent-routing/SKILL.md +159 -127
  96. package/.agent/skills/lint-and-validate/SKILL.md +8 -52
  97. package/.agent/skills/llm-engineering/SKILL.md +344 -357
  98. package/.agent/skills/local-first/SKILL.md +120 -154
  99. package/.agent/skills/mcp-builder/SKILL.md +84 -118
  100. package/.agent/skills/mobile-design/SKILL.md +213 -219
  101. package/.agent/skills/motion-engineering/SKILL.md +184 -0
  102. package/.agent/skills/nextjs-react-expert/SKILL.md +99 -698
  103. package/.agent/skills/nodejs-best-practices/SKILL.md +498 -559
  104. package/.agent/skills/observability/SKILL.md +293 -330
  105. package/.agent/skills/parallel-agents/SKILL.md +88 -122
  106. package/.agent/skills/performance-profiling/SKILL.md +217 -254
  107. package/.agent/skills/plan-writing/SKILL.md +84 -118
  108. package/.agent/skills/platform-engineer/SKILL.md +89 -123
  109. package/.agent/skills/playwright-best-practices/SKILL.md +128 -162
  110. package/.agent/skills/powershell-windows/SKILL.md +112 -146
  111. package/.agent/skills/python-patterns/SKILL.md +7 -35
  112. package/.agent/skills/python-pro/SKILL.md +148 -754
  113. package/.agent/skills/react-specialist/SKILL.md +123 -827
  114. package/.agent/skills/readme-builder/SKILL.md +15 -85
  115. package/.agent/skills/realtime-patterns/SKILL.md +269 -304
  116. package/.agent/skills/red-team-tactics/SKILL.md +10 -51
  117. package/.agent/skills/rust-pro/SKILL.md +623 -701
  118. package/.agent/skills/seo-fundamentals/SKILL.md +120 -154
  119. package/.agent/skills/server-management/SKILL.md +156 -190
  120. package/.agent/skills/shadcn-ui-expert/SKILL.md +172 -206
  121. package/.agent/skills/skill-creator/SKILL.md +18 -58
  122. package/.agent/skills/sql-pro/SKILL.md +579 -633
  123. package/.agent/skills/supabase-postgres-best-practices/SKILL.md +28 -68
  124. package/.agent/skills/swiftui-expert/SKILL.md +142 -176
  125. package/.agent/skills/systematic-debugging/SKILL.md +84 -118
  126. package/.agent/skills/tailwind-patterns/SKILL.md +516 -576
  127. package/.agent/skills/tdd-workflow/SKILL.md +103 -137
  128. package/.agent/skills/test-result-analyzer/SKILL.md +33 -73
  129. package/.agent/skills/testing-patterns/SKILL.md +512 -573
  130. package/.agent/skills/trend-researcher/SKILL.md +30 -71
  131. package/.agent/skills/ui-ux-pro-max/SKILL.md +0 -41
  132. package/.agent/skills/ui-ux-researcher/SKILL.md +51 -91
  133. package/.agent/skills/vue-expert/SKILL.md +127 -866
  134. package/.agent/skills/vulnerability-scanner/SKILL.md +354 -269
  135. package/.agent/skills/web-accessibility-auditor/SKILL.md +159 -193
  136. package/.agent/skills/web-design-guidelines/SKILL.md +17 -61
  137. package/.agent/skills/webapp-testing/SKILL.md +111 -145
  138. package/.agent/skills/whimsy-injector/SKILL.md +58 -132
  139. package/.agent/skills/workflow-optimizer/SKILL.md +28 -68
  140. package/.agent/workflows/api-tester.md +151 -151
  141. package/.agent/workflows/audit.md +127 -138
  142. package/.agent/workflows/brainstorm.md +110 -110
  143. package/.agent/workflows/changelog.md +112 -112
  144. package/.agent/workflows/create.md +124 -124
  145. package/.agent/workflows/debug.md +165 -189
  146. package/.agent/workflows/deploy.md +180 -189
  147. package/.agent/workflows/enhance.md +128 -151
  148. package/.agent/workflows/fix.md +114 -135
  149. package/.agent/workflows/generate.md +12 -4
  150. package/.agent/workflows/migrate.md +160 -160
  151. package/.agent/workflows/orchestrate.md +168 -168
  152. package/.agent/workflows/performance-benchmarker.md +114 -123
  153. package/.agent/workflows/plan.md +173 -173
  154. package/.agent/workflows/preview.md +80 -80
  155. package/.agent/workflows/refactor.md +161 -183
  156. package/.agent/workflows/review-ai.md +101 -129
  157. package/.agent/workflows/review.md +116 -116
  158. package/.agent/workflows/session.md +94 -94
  159. package/.agent/workflows/status.md +79 -79
  160. package/.agent/workflows/strengthen-skills.md +138 -139
  161. package/.agent/workflows/swarm.md +179 -179
  162. package/.agent/workflows/test.md +189 -211
  163. package/.agent/workflows/tribunal-backend.md +93 -113
  164. package/.agent/workflows/tribunal-database.md +94 -115
  165. package/.agent/workflows/tribunal-frontend.md +95 -118
  166. package/.agent/workflows/tribunal-full.md +92 -133
  167. package/.agent/workflows/tribunal-mobile.md +94 -119
  168. package/.agent/workflows/tribunal-performance.md +109 -133
  169. package/.agent/workflows/ui-ux-pro-max.md +122 -143
  170. package/package.json +1 -1
  171. package/.agent/skills/api-patterns/api-style.md +0 -42
  172. package/.agent/skills/api-patterns/auth.md +0 -24
  173. package/.agent/skills/api-patterns/documentation.md +0 -26
  174. package/.agent/skills/api-patterns/graphql.md +0 -41
  175. package/.agent/skills/api-patterns/rate-limiting.md +0 -31
  176. package/.agent/skills/api-patterns/response.md +0 -37
  177. package/.agent/skills/api-patterns/rest.md +0 -40
  178. package/.agent/skills/api-patterns/security-testing.md +0 -122
  179. package/.agent/skills/api-patterns/trpc.md +0 -41
  180. package/.agent/skills/api-patterns/versioning.md +0 -22
  181. package/.agent/skills/app-builder/agent-coordination.md +0 -71
  182. package/.agent/skills/app-builder/feature-building.md +0 -53
  183. package/.agent/skills/app-builder/project-detection.md +0 -34
  184. package/.agent/skills/app-builder/scaffolding.md +0 -118
  185. package/.agent/skills/app-builder/tech-stack.md +0 -40
  186. package/.agent/skills/architecture/context-discovery.md +0 -43
  187. package/.agent/skills/architecture/examples.md +0 -94
  188. package/.agent/skills/architecture/pattern-selection.md +0 -68
  189. package/.agent/skills/architecture/patterns-reference.md +0 -50
  190. package/.agent/skills/architecture/trade-off-analysis.md +0 -77
  191. package/.agent/skills/brainstorming/dynamic-questioning.md +0 -360
  192. package/.agent/skills/database-design/database-selection.md +0 -43
  193. package/.agent/skills/database-design/indexing.md +0 -39
  194. package/.agent/skills/database-design/migrations.md +0 -48
  195. package/.agent/skills/database-design/optimization.md +0 -36
  196. package/.agent/skills/database-design/orm-selection.md +0 -30
  197. package/.agent/skills/database-design/schema-design.md +0 -56
  198. package/.agent/skills/frontend-design/animation-guide.md +0 -331
  199. package/.agent/skills/frontend-design/color-system.md +0 -329
  200. package/.agent/skills/frontend-design/decision-trees.md +0 -418
  201. package/.agent/skills/frontend-design/motion-graphics.md +0 -306
  202. package/.agent/skills/frontend-design/typography-system.md +0 -363
  203. package/.agent/skills/frontend-design/ux-psychology.md +0 -1116
  204. package/.agent/skills/frontend-design/visual-effects.md +0 -383
  205. package/.agent/skills/intelligent-routing/router-manifest.md +0 -65
  206. package/.agent/skills/mobile-design/decision-trees.md +0 -516
  207. package/.agent/skills/mobile-design/mobile-backend.md +0 -491
  208. package/.agent/skills/mobile-design/mobile-color-system.md +0 -420
  209. package/.agent/skills/mobile-design/mobile-debugging.md +0 -122
  210. package/.agent/skills/mobile-design/mobile-design-thinking.md +0 -357
  211. package/.agent/skills/mobile-design/mobile-navigation.md +0 -458
  212. package/.agent/skills/mobile-design/mobile-performance.md +0 -767
  213. package/.agent/skills/mobile-design/mobile-testing.md +0 -356
  214. package/.agent/skills/mobile-design/mobile-typography.md +0 -433
  215. package/.agent/skills/mobile-design/platform-android.md +0 -666
  216. package/.agent/skills/mobile-design/platform-ios.md +0 -561
  217. package/.agent/skills/mobile-design/touch-psychology.md +0 -537
  218. package/.agent/skills/nextjs-react-expert/1-async-eliminating-waterfalls.md +0 -312
  219. package/.agent/skills/nextjs-react-expert/2-bundle-bundle-size-optimization.md +0 -240
  220. package/.agent/skills/nextjs-react-expert/3-server-server-side-performance.md +0 -490
  221. package/.agent/skills/nextjs-react-expert/4-client-client-side-data-fetching.md +0 -264
  222. package/.agent/skills/nextjs-react-expert/5-rerender-re-render-optimization.md +0 -581
  223. package/.agent/skills/nextjs-react-expert/6-rendering-rendering-performance.md +0 -432
  224. package/.agent/skills/nextjs-react-expert/7-js-javascript-performance.md +0 -684
  225. package/.agent/skills/nextjs-react-expert/8-advanced-advanced-patterns.md +0 -150
  226. package/.agent/skills/vulnerability-scanner/checklists.md +0 -121
@@ -1,490 +0,0 @@
1
- # 3. Server-Side Performance
2
-
3
- > **Impact:** HIGH
4
- > **Focus:** Optimizing server-side rendering and data fetching eliminates server-side waterfalls and reduces response times.
5
-
6
- ---
7
-
8
- ## Overview
9
-
10
- This section contains **7 rules** focused on server-side performance.
11
-
12
- ---
13
-
14
- ## Rule 3.1: Authenticate Server Actions Like API Routes
15
-
16
- **Impact:** CRITICAL
17
- **Tags:** server, server-actions, authentication, security, authorization
18
-
19
- ## Authenticate Server Actions Like API Routes
20
-
21
- **Impact: CRITICAL (prevents unauthorized access to server mutations)**
22
-
23
- Server Actions (functions with `"use server"`) are exposed as public endpoints, just like API routes. Always verify authentication and authorization **inside** each Server Action—do not rely solely on middleware, layout guards, or page-level checks, as Server Actions can be invoked directly.
24
-
25
- Next.js documentation explicitly states: "Treat Server Actions with the same security considerations as public-facing API endpoints, and verify if the user is allowed to perform a mutation."
26
-
27
- **Incorrect (no authentication check):**
28
-
29
- ```typescript
30
- 'use server'
31
-
32
- export async function deleteUser(userId: string) {
33
- // Anyone can call this! No auth check
34
- await db.user.delete({ where: { id: userId } })
35
- return { success: true }
36
- }
37
- ```
38
-
39
- **Correct (authentication inside the action):**
40
-
41
- ```typescript
42
- 'use server'
43
-
44
- import { verifySession } from '@/lib/auth'
45
- import { unauthorized } from '@/lib/errors'
46
-
47
- export async function deleteUser(userId: string) {
48
- // Always check auth inside the action
49
- const session = await verifySession()
50
-
51
- if (!session) {
52
- throw unauthorized('Must be logged in')
53
- }
54
-
55
- // Check authorization too
56
- if (session.user.role !== 'admin' && session.user.id !== userId) {
57
- throw unauthorized('Cannot delete other users')
58
- }
59
-
60
- await db.user.delete({ where: { id: userId } })
61
- return { success: true }
62
- }
63
- ```
64
-
65
- **With input validation:**
66
-
67
- ```typescript
68
- 'use server'
69
-
70
- import { verifySession } from '@/lib/auth'
71
- import { z } from 'zod'
72
-
73
- const updateProfileSchema = z.object({
74
- userId: z.string().uuid(),
75
- name: z.string().min(1).max(100),
76
- email: z.string().email()
77
- })
78
-
79
- export async function updateProfile(data: unknown) {
80
- // Validate input first
81
- const validated = updateProfileSchema.parse(data)
82
-
83
- // Then authenticate
84
- const session = await verifySession()
85
- if (!session) {
86
- throw new Error('Unauthorized')
87
- }
88
-
89
- // Then authorize
90
- if (session.user.id !== validated.userId) {
91
- throw new Error('Can only update own profile')
92
- }
93
-
94
- // Finally perform the mutation
95
- await db.user.update({
96
- where: { id: validated.userId },
97
- data: {
98
- name: validated.name,
99
- email: validated.email
100
- }
101
- })
102
-
103
- return { success: true }
104
- }
105
- ```
106
-
107
- Reference: [https://nextjs.org/docs/app/guides/authentication](https://nextjs.org/docs/app/guides/authentication)
108
-
109
- ---
110
-
111
- ## Rule 3.2: Avoid Duplicate Serialization in RSC Props
112
-
113
- **Impact:** LOW
114
- **Tags:** server, rsc, serialization, props, client-components
115
-
116
- ## Avoid Duplicate Serialization in RSC Props
117
-
118
- **Impact: LOW (reduces network payload by avoiding duplicate serialization)**
119
-
120
- RSC→client serialization deduplicates by object reference, not value. Same reference = serialized once; new reference = serialized again. Do transformations (`.toSorted()`, `.filter()`, `.map()`) in client, not server.
121
-
122
- **Incorrect (duplicates array):**
123
-
124
- ```tsx
125
- // RSC: sends 6 strings (2 arrays × 3 items)
126
- <ClientList usernames={usernames} usernamesOrdered={usernames.toSorted()} />
127
- ```
128
-
129
- **Correct (sends 3 strings):**
130
-
131
- ```tsx
132
- // RSC: send once
133
- <ClientList usernames={usernames} />
134
-
135
- // Client: transform there
136
- 'use client'
137
- const sorted = useMemo(() => [...usernames].sort(), [usernames])
138
- ```
139
-
140
- **Nested deduplication behavior:**
141
-
142
- Deduplication works recursively. Impact varies by data type:
143
-
144
- - `string[]`, `number[]`, `boolean[]`: **HIGH impact** - array + all primitives fully duplicated
145
- - `object[]`: **LOW impact** - array duplicated, but nested objects deduplicated by reference
146
-
147
- ```tsx
148
- // string[] - duplicates everything
149
- usernames={['a','b']} sorted={usernames.toSorted()} // sends 4 strings
150
-
151
- // object[] - duplicates array structure only
152
- users={[{id:1},{id:2}]} sorted={users.toSorted()} // sends 2 arrays + 2 unique objects (not 4)
153
- ```
154
-
155
- **Operations breaking deduplication (create new references):**
156
-
157
- - Arrays: `.toSorted()`, `.filter()`, `.map()`, `.slice()`, `[...arr]`
158
- - Objects: `{...obj}`, `Object.assign()`, `structuredClone()`, `JSON.parse(JSON.stringify())`
159
-
160
- **More examples:**
161
-
162
- ```tsx
163
- // ❌ Bad
164
- <C users={users} active={users.filter(u => u.active)} />
165
- <C product={product} productName={product.name} />
166
-
167
- // ✅ Good
168
- <C users={users} />
169
- <C product={product} />
170
- // Do filtering/destructuring in client
171
- ```
172
-
173
- **Exception:** Pass derived data when transformation is expensive or client doesn't need original.
174
-
175
- ---
176
-
177
- ## Rule 3.3: Cross-Request LRU Caching
178
-
179
- **Impact:** HIGH
180
- **Tags:** server, cache, lru, cross-request
181
-
182
- ## Cross-Request LRU Caching
183
-
184
- `React.cache()` only works within one request. For data shared across sequential requests (user clicks button A then button B), use an LRU cache.
185
-
186
- **Implementation:**
187
-
188
- ```typescript
189
- import { LRUCache } from 'lru-cache'
190
-
191
- const cache = new LRUCache<string, any>({
192
- max: 1000,
193
- ttl: 5 * 60 * 1000 // 5 minutes
194
- })
195
-
196
- export async function getUser(id: string) {
197
- const cached = cache.get(id)
198
- if (cached) return cached
199
-
200
- const user = await db.user.findUnique({ where: { id } })
201
- cache.set(id, user)
202
- return user
203
- }
204
-
205
- // Request 1: DB query, result cached
206
- // Request 2: cache hit, no DB query
207
- ```
208
-
209
- Use when sequential user actions hit multiple endpoints needing the same data within seconds.
210
-
211
- **With Vercel's [Fluid Compute](https://vercel.com/docs/fluid-compute):** LRU caching is especially effective because multiple concurrent requests can share the same function instance and cache. This means the cache persists across requests without needing external storage like Redis.
212
-
213
- **In traditional serverless:** Each invocation runs in isolation, so consider Redis for cross-process caching.
214
-
215
- Reference: [https://github.com/isaacs/node-lru-cache](https://github.com/isaacs/node-lru-cache)
216
-
217
- ---
218
-
219
- ## Rule 3.4: Minimize Serialization at RSC Boundaries
220
-
221
- **Impact:** HIGH
222
- **Tags:** server, rsc, serialization, props
223
-
224
- ## Minimize Serialization at RSC Boundaries
225
-
226
- The React Server/Client boundary serializes all object properties into strings and embeds them in the HTML response and subsequent RSC requests. This serialized data directly impacts page weight and load time, so **size matters a lot**. Only pass fields that the client actually uses.
227
-
228
- **Incorrect (serializes all 50 fields):**
229
-
230
- ```tsx
231
- async function Page() {
232
- const user = await fetchUser() // 50 fields
233
- return <Profile user={user} />
234
- }
235
-
236
- 'use client'
237
- function Profile({ user }: { user: User }) {
238
- return <div>{user.name}</div> // uses 1 field
239
- }
240
- ```
241
-
242
- **Correct (serializes only 1 field):**
243
-
244
- ```tsx
245
- async function Page() {
246
- const user = await fetchUser()
247
- return <Profile name={user.name} />
248
- }
249
-
250
- 'use client'
251
- function Profile({ name }: { name: string }) {
252
- return <div>{name}</div>
253
- }
254
- ```
255
-
256
- ---
257
-
258
- ## Rule 3.5: Parallel Data Fetching with Component Composition
259
-
260
- **Impact:** CRITICAL
261
- **Tags:** server, rsc, parallel-fetching, composition
262
-
263
- ## Parallel Data Fetching with Component Composition
264
-
265
- React Server Components execute sequentially within a tree. Restructure with composition to parallelize data fetching.
266
-
267
- **Incorrect (Sidebar waits for Page's fetch to complete):**
268
-
269
- ```tsx
270
- export default async function Page() {
271
- const header = await fetchHeader()
272
- return (
273
- <div>
274
- <div>{header}</div>
275
- <Sidebar />
276
- </div>
277
- )
278
- }
279
-
280
- async function Sidebar() {
281
- const items = await fetchSidebarItems()
282
- return <nav>{items.map(renderItem)}</nav>
283
- }
284
- ```
285
-
286
- **Correct (both fetch simultaneously):**
287
-
288
- ```tsx
289
- async function Header() {
290
- const data = await fetchHeader()
291
- return <div>{data}</div>
292
- }
293
-
294
- async function Sidebar() {
295
- const items = await fetchSidebarItems()
296
- return <nav>{items.map(renderItem)}</nav>
297
- }
298
-
299
- export default function Page() {
300
- return (
301
- <div>
302
- <Header />
303
- <Sidebar />
304
- </div>
305
- )
306
- }
307
- ```
308
-
309
- **Alternative with children prop:**
310
-
311
- ```tsx
312
- async function Header() {
313
- const data = await fetchHeader()
314
- return <div>{data}</div>
315
- }
316
-
317
- async function Sidebar() {
318
- const items = await fetchSidebarItems()
319
- return <nav>{items.map(renderItem)}</nav>
320
- }
321
-
322
- function Layout({ children }: { children: ReactNode }) {
323
- return (
324
- <div>
325
- <Header />
326
- {children}
327
- </div>
328
- )
329
- }
330
-
331
- export default function Page() {
332
- return (
333
- <Layout>
334
- <Sidebar />
335
- </Layout>
336
- )
337
- }
338
- ```
339
-
340
- ---
341
-
342
- ## Rule 3.6: Per-Request Deduplication with React.cache()
343
-
344
- **Impact:** MEDIUM
345
- **Tags:** server, cache, react-cache, deduplication
346
-
347
- ## Per-Request Deduplication with React.cache()
348
-
349
- Use `React.cache()` for server-side request deduplication. Authentication and database queries benefit most.
350
-
351
- **Usage:**
352
-
353
- ```typescript
354
- import { cache } from 'react'
355
-
356
- export const getCurrentUser = cache(async () => {
357
- const session = await auth()
358
- if (!session?.user?.id) return null
359
- return await db.user.findUnique({
360
- where: { id: session.user.id }
361
- })
362
- })
363
- ```
364
-
365
- Within a single request, multiple calls to `getCurrentUser()` execute the query only once.
366
-
367
- **Avoid inline objects as arguments:**
368
-
369
- `React.cache()` uses shallow equality (`Object.is`) to determine cache hits. Inline objects create new references each call, preventing cache hits.
370
-
371
- **Incorrect (always cache miss):**
372
-
373
- ```typescript
374
- const getUser = cache(async (params: { uid: number }) => {
375
- return await db.user.findUnique({ where: { id: params.uid } })
376
- })
377
-
378
- // Each call creates new object, never hits cache
379
- getUser({ uid: 1 })
380
- getUser({ uid: 1 }) // Cache miss, runs query again
381
- ```
382
-
383
- **Correct (cache hit):**
384
-
385
- ```typescript
386
- const getUser = cache(async (uid: number) => {
387
- return await db.user.findUnique({ where: { id: uid } })
388
- })
389
-
390
- // Primitive args use value equality
391
- getUser(1)
392
- getUser(1) // Cache hit, returns cached result
393
- ```
394
-
395
- If you must pass objects, pass the same reference:
396
-
397
- ```typescript
398
- const params = { uid: 1 }
399
- getUser(params) // Query runs
400
- getUser(params) // Cache hit (same reference)
401
- ```
402
-
403
- **Next.js-Specific Note:**
404
-
405
- In Next.js, the `fetch` API is automatically extended with request memoization. Requests with the same URL and options are automatically deduplicated within a single request, so you don't need `React.cache()` for `fetch` calls. However, `React.cache()` is still essential for other async tasks:
406
-
407
- - Database queries (Prisma, Drizzle, etc.)
408
- - Heavy computations
409
- - Authentication checks
410
- - File system operations
411
- - Any non-fetch async work
412
-
413
- Use `React.cache()` to deduplicate these operations across your component tree.
414
-
415
- Reference: [React.cache documentation](https://react.dev/reference/react/cache)
416
-
417
- ---
418
-
419
- ## Rule 3.7: Use after() for Non-Blocking Operations
420
-
421
- **Impact:** MEDIUM
422
- **Tags:** server, async, logging, analytics, side-effects
423
-
424
- ## Use after() for Non-Blocking Operations
425
-
426
- Use Next.js's `after()` to schedule work that should execute after a response is sent. This prevents logging, analytics, and other side effects from blocking the response.
427
-
428
- **Incorrect (blocks response):**
429
-
430
- ```tsx
431
- import { logUserAction } from '@/app/utils'
432
-
433
- export async function POST(request: Request) {
434
- // Perform mutation
435
- await updateDatabase(request)
436
-
437
- // Logging blocks the response
438
- const userAgent = request.headers.get('user-agent') || 'unknown'
439
- await logUserAction({ userAgent })
440
-
441
- return new Response(JSON.stringify({ status: 'success' }), {
442
- status: 200,
443
- headers: { 'Content-Type': 'application/json' }
444
- })
445
- }
446
- ```
447
-
448
- **Correct (non-blocking):**
449
-
450
- ```tsx
451
- import { after } from 'next/server'
452
- import { headers, cookies } from 'next/headers'
453
- import { logUserAction } from '@/app/utils'
454
-
455
- export async function POST(request: Request) {
456
- // Perform mutation
457
- await updateDatabase(request)
458
-
459
- // Log after response is sent
460
- after(async () => {
461
- const userAgent = (await headers()).get('user-agent') || 'unknown'
462
- const sessionCookie = (await cookies()).get('session-id')?.value || 'anonymous'
463
-
464
- logUserAction({ sessionCookie, userAgent })
465
- })
466
-
467
- return new Response(JSON.stringify({ status: 'success' }), {
468
- status: 200,
469
- headers: { 'Content-Type': 'application/json' }
470
- })
471
- }
472
- ```
473
-
474
- The response is sent immediately while logging happens in the background.
475
-
476
- **Common use cases:**
477
-
478
- - Analytics tracking
479
- - Audit logging
480
- - Sending notifications
481
- - Cache invalidation
482
- - Cleanup tasks
483
-
484
- **Important notes:**
485
-
486
- - `after()` runs even if the response fails or redirects
487
- - Works in Server Actions, Route Handlers, and Server Components
488
-
489
- Reference: [https://nextjs.org/docs/app/api-reference/functions/after](https://nextjs.org/docs/app/api-reference/functions/after)
490
-