@polymorphism-tech/morph-spec 4.5.0 → 4.7.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 (292) hide show
  1. package/CLAUDE.md +77 -56
  2. package/README.md +394 -700
  3. package/docs/ARCHITECTURE.md +331 -0
  4. package/docs/CHEATSHEET.md +221 -0
  5. package/docs/COMMAND-FLOWS.md +368 -0
  6. package/docs/QUICKSTART.md +212 -0
  7. package/docs/examples/order-management/contracts.cs +84 -0
  8. package/docs/examples/order-management/proposal.md +24 -0
  9. package/docs/examples/order-management/spec.md +162 -0
  10. package/docs/plans/2026-02-23-ddd-architecture-refactor.md +1153 -0
  11. package/docs/plans/2026-02-23-ddd-nextsteps.md +682 -0
  12. package/docs/plans/2026-02-23-infra-architect-refactor.md +437 -0
  13. package/docs/plans/2026-02-23-nextjs-code-review-design.md +156 -0
  14. package/docs/plans/2026-02-23-nextjs-code-review-impl.md +1254 -0
  15. package/docs/plans/2026-02-23-nextjs-standards-design.md +149 -0
  16. package/docs/plans/2026-02-23-nextjs-standards-impl.md +1846 -0
  17. package/framework/{skills/level-2-domains → agents}/README.md +14 -14
  18. package/framework/{skills/level-2-domains → agents}/ai-agents/ai-system-architect.md +1 -4
  19. package/framework/{skills/level-2-domains → agents}/architecture/po-pm-advisor.md +1 -2
  20. package/framework/{skills/level-2-domains → agents}/architecture/prompt-engineer.md +1 -2
  21. package/framework/{skills/level-2-domains → agents}/architecture/seo-growth-hacker.md +1 -2
  22. package/framework/{skills/level-2-domains → agents}/architecture/standards-architect.md +159 -162
  23. package/framework/agents/backend/api-designer.md +103 -0
  24. package/framework/{skills/level-2-domains → agents}/backend/dotnet-senior.md +1 -2
  25. package/framework/agents/backend/ef-modeler.md +119 -0
  26. package/framework/{skills/level-2-domains → agents}/backend/hangfire-orchestrator.md +1 -4
  27. package/framework/{skills/level-2-domains → agents}/backend/ms-agent-expert.md +1 -4
  28. package/framework/{skills/level-2-domains → agents}/frontend/blazor-builder.md +1 -4
  29. package/framework/agents/frontend/nextjs-expert.md +118 -0
  30. package/framework/{skills/level-2-domains → agents}/frontend/ui-ux-designer.md +1 -2
  31. package/framework/{skills/level-2-domains → agents}/infrastructure/azure-architect.md +147 -148
  32. package/framework/{skills/level-2-domains → agents}/infrastructure/azure-deploy-specialist.md +1 -2
  33. package/framework/{skills/level-2-domains → agents}/infrastructure/bicep-architect.md +1 -4
  34. package/framework/{skills/level-2-domains → agents}/infrastructure/container-specialist.md +1 -4
  35. package/framework/{skills/level-2-domains → agents}/infrastructure/devops-engineer.md +1 -4
  36. package/framework/agents/infrastructure/infra-architect.md +45 -0
  37. package/framework/{skills/level-2-domains → agents}/integrations/asaas-financial.md +1 -4
  38. package/framework/{skills/level-2-domains → agents}/integrations/azure-identity.md +1 -4
  39. package/framework/{skills/level-2-domains → agents}/integrations/clerk-auth.md +1 -4
  40. package/framework/{skills/level-2-domains → agents}/integrations/hangfire-integration.md +1 -2
  41. package/framework/{skills/level-2-domains → agents}/integrations/resend-email.md +1 -4
  42. package/framework/{skills/level-2-domains → agents}/quality/code-analyzer.md +1 -4
  43. package/framework/{skills/level-2-domains → agents}/quality/testing-specialist.md +1 -4
  44. package/framework/agents.json +1145 -278
  45. package/framework/hooks/claude-code/statusline.py +384 -85
  46. package/framework/hooks/shared/phase-utils.js +129 -129
  47. package/framework/rules/frontend-standards.md +0 -3
  48. package/framework/rules/nextjs-standards.md +17 -0
  49. package/framework/skills/README.md +66 -0
  50. package/framework/skills/level-0-meta/{brainstorming.md → brainstorming/SKILL.md} +3 -1
  51. package/framework/skills/level-0-meta/brainstorming/references/proposal-example.md +138 -0
  52. package/framework/skills/level-0-meta/{code-review.md → code-review/SKILL.md} +3 -2
  53. package/framework/skills/level-0-meta/code-review/references/review-example.md +164 -0
  54. package/framework/skills/level-0-meta/code-review/scripts/scan-csharp.mjs +121 -0
  55. package/framework/skills/level-0-meta/code-review-nextjs/SKILL.md +147 -0
  56. package/framework/skills/level-0-meta/code-review-nextjs/references/review-example-nextjs.md +254 -0
  57. package/framework/skills/level-0-meta/{morph-checklist.md → morph-checklist/SKILL.md} +2 -5
  58. package/framework/skills/{level-1-workflows/morph-replicate.md → level-0-meta/morph-replicate/SKILL.md} +6 -7
  59. package/framework/skills/level-0-meta/{simulation-checklist.md → simulation-checklist/SKILL.md} +3 -6
  60. package/framework/skills/level-0-meta/{tool-usage-guide.md → tool-usage-guide/SKILL.md} +4 -5
  61. package/framework/skills/level-0-meta/{verification-before-completion.md → verification-before-completion/SKILL.md} +3 -1
  62. package/framework/skills/level-0-meta/verification-before-completion/scripts/check-phase-outputs.mjs +110 -0
  63. package/framework/skills/level-1-workflows/{phase-clarify.md → phase-clarify/SKILL.md} +3 -3
  64. package/framework/skills/level-1-workflows/phase-clarify/references/clarifications-example.md +117 -0
  65. package/framework/skills/level-1-workflows/{phase-codebase-analysis.md → phase-codebase-analysis/SKILL.md} +2 -3
  66. package/framework/skills/level-1-workflows/{phase-design.md → phase-design/SKILL.md} +46 -182
  67. package/framework/skills/level-1-workflows/phase-design/references/spec-example.md +253 -0
  68. package/framework/skills/level-1-workflows/{phase-implement.md → phase-implement/SKILL.md} +3 -3
  69. package/framework/skills/level-1-workflows/phase-implement/references/recap-example.md +132 -0
  70. package/framework/skills/level-1-workflows/{phase-setup.md → phase-setup/SKILL.md} +2 -3
  71. package/framework/skills/level-1-workflows/{phase-tasks.md → phase-tasks/SKILL.md} +42 -3
  72. package/framework/skills/level-1-workflows/phase-tasks/references/tasks-example.md +231 -0
  73. package/framework/skills/level-1-workflows/phase-tasks/scripts/validate-tasks.mjs +112 -0
  74. package/framework/skills/level-1-workflows/{phase-uiux.md → phase-uiux/SKILL.md} +2 -3
  75. package/framework/standards/STANDARDS.json +121 -0
  76. package/framework/standards/architecture/ddd/bounded-contexts.md +105 -0
  77. package/framework/standards/architecture/ddd/complexity-levels.md +108 -0
  78. package/framework/standards/architecture/ddd/ubiquitous-language.md +58 -0
  79. package/framework/standards/frontend/nextjs/app-router.md +123 -0
  80. package/framework/standards/frontend/nextjs/components.md +132 -0
  81. package/framework/standards/frontend/nextjs/data-fetching.md +126 -0
  82. package/framework/standards/frontend/nextjs/forms.md +128 -0
  83. package/framework/standards/frontend/nextjs/naming-conventions.md +67 -0
  84. package/framework/standards/frontend/nextjs/project-structure.md +102 -0
  85. package/framework/standards/frontend/nextjs/state-management.md +72 -0
  86. package/framework/standards/frontend/nextjs/testing.md +111 -0
  87. package/framework/templates/REGISTRY.json +538 -142
  88. package/framework/templates/code/dotnet/contracts/contracts-level1.cs +69 -0
  89. package/framework/templates/code/dotnet/contracts/contracts-level2.cs +86 -0
  90. package/framework/templates/code/dotnet/contracts/contracts-level3.cs +41 -0
  91. package/framework/templates/docs/spec.md +49 -0
  92. package/framework/templates/frontend/nextjs/Dockerfile.nextjs.hbs +43 -0
  93. package/framework/templates/frontend/nextjs/client-component.tsx.hbs +26 -0
  94. package/framework/templates/frontend/nextjs/env.mjs.hbs +32 -0
  95. package/framework/templates/frontend/nextjs/feature-form.tsx.hbs +56 -0
  96. package/framework/templates/frontend/nextjs/page.tsx.hbs +22 -0
  97. package/framework/templates/frontend/nextjs/tsconfig.json.hbs +26 -0
  98. package/framework/templates/frontend/nextjs/use-feature.ts.hbs +54 -0
  99. package/framework/templates/project-structure/dotnet-ddd.md +70 -0
  100. package/framework/workflows/docs/enforcement-pipeline.md +2 -1
  101. package/package.json +1 -1
  102. package/scripts/scan-nextjs.mjs +169 -0
  103. package/src/commands/project/doctor.js +52 -1
  104. package/src/commands/project/init.js +19 -65
  105. package/src/commands/project/update.js +7 -63
  106. package/src/lib/detectors/claude-config-detector.js +1 -3
  107. package/src/lib/standards/standards-context-injector.js +5 -0
  108. package/src/lib/validators/nextjs/index.js +6 -0
  109. package/src/lib/validators/nextjs/next-component-validator.js +181 -0
  110. package/src/lib/validators/validation-runner.js +5 -0
  111. package/src/utils/agents-installer.js +16 -4
  112. package/src/utils/skills-installer.js +59 -15
  113. package/.morph/.morphversion +0 -5
  114. package/.morph/analytics/threads-log.jsonl +0 -44
  115. package/.morph/config/config.json +0 -8
  116. package/.morph/context/README.md +0 -17
  117. package/.morph/framework/agents.json +0 -948
  118. package/.morph/framework/standards/STANDARDS.json +0 -812
  119. package/.morph/framework/standards/ai-agents/blazor-ui.md +0 -364
  120. package/.morph/framework/standards/ai-agents/production.md +0 -415
  121. package/.morph/framework/standards/ai-agents/setup.md +0 -418
  122. package/.morph/framework/standards/ai-agents/team-orchestration.md +0 -479
  123. package/.morph/framework/standards/ai-agents/workflows.md +0 -354
  124. package/.morph/framework/standards/architecture/ddd/aggregates.md +0 -120
  125. package/.morph/framework/standards/architecture/ddd/entities.md +0 -99
  126. package/.morph/framework/standards/architecture/ddd/value-objects.md +0 -124
  127. package/.morph/framework/standards/backend/api/minimal-api.md +0 -494
  128. package/.morph/framework/standards/backend/api/rest.md +0 -492
  129. package/.morph/framework/standards/backend/api/validation.md +0 -88
  130. package/.morph/framework/standards/backend/authentication/passkeys.md +0 -428
  131. package/.morph/framework/standards/backend/database/ef-core.md +0 -199
  132. package/.morph/framework/standards/backend/database/migrations.md +0 -393
  133. package/.morph/framework/standards/backend/database/postgresql/database.md +0 -352
  134. package/.morph/framework/standards/backend/database/repository-patterns.md +0 -528
  135. package/.morph/framework/standards/backend/database/vector-search-rag.md +0 -541
  136. package/.morph/framework/standards/backend/dotnet/async.md +0 -366
  137. package/.morph/framework/standards/backend/dotnet/core.md +0 -117
  138. package/.morph/framework/standards/backend/dotnet/di.md +0 -439
  139. package/.morph/framework/standards/backend/dotnet/program-cs-checklist.md +0 -92
  140. package/.morph/framework/standards/backend/integrations/asaas/asaas-api.md +0 -216
  141. package/.morph/framework/standards/backend/integrations/clerk/clerk-auth.md +0 -290
  142. package/.morph/framework/standards/backend/integrations/hangfire/hangfire-jobs.md +0 -350
  143. package/.morph/framework/standards/backend/integrations/resend/resend-email.md +0 -385
  144. package/.morph/framework/standards/context/analytics.md +0 -96
  145. package/.morph/framework/standards/context/bundles.md +0 -110
  146. package/.morph/framework/standards/context/priming.md +0 -78
  147. package/.morph/framework/standards/core/architecture.md +0 -185
  148. package/.morph/framework/standards/core/coding.md +0 -214
  149. package/.morph/framework/standards/core/git-branching-strategy.md +0 -403
  150. package/.morph/framework/standards/core/git.md +0 -185
  151. package/.morph/framework/standards/core/testing.md +0 -295
  152. package/.morph/framework/standards/data/nosql/blob-storage.md +0 -102
  153. package/.morph/framework/standards/data/nosql/cache/redis.md +0 -97
  154. package/.morph/framework/standards/data/nosql/cosmos-db.md +0 -118
  155. package/.morph/framework/standards/data/vector-search/azure-ai-search.md +0 -121
  156. package/.morph/framework/standards/data/vector-search/rag-chunking.md +0 -104
  157. package/.morph/framework/standards/frontend/blazor/design-checklist.md +0 -222
  158. package/.morph/framework/standards/frontend/blazor/fluent-ui-setup.md +0 -595
  159. package/.morph/framework/standards/frontend/blazor/fluent-ui.md +0 -137
  160. package/.morph/framework/standards/frontend/blazor/html-conversion.md +0 -184
  161. package/.morph/framework/standards/frontend/blazor/lifecycle.md +0 -195
  162. package/.morph/framework/standards/frontend/blazor/pitfalls.md +0 -198
  163. package/.morph/framework/standards/frontend/blazor/state.md +0 -191
  164. package/.morph/framework/standards/frontend/design-system/animations.md +0 -151
  165. package/.morph/framework/standards/frontend/design-system/naming.md +0 -64
  166. package/.morph/framework/standards/frontend/nextjs/nextjs-patterns.md +0 -198
  167. package/.morph/framework/standards/infrastructure/azure/azure.md +0 -624
  168. package/.morph/framework/standards/infrastructure/azure/bicep/bicep-patterns.md +0 -422
  169. package/.morph/framework/standards/infrastructure/azure/devops/azure-devops-setup.md +0 -516
  170. package/.morph/framework/standards/infrastructure/azure/devops/local-development.md +0 -520
  171. package/.morph/framework/standards/infrastructure/azure/services/functions.md +0 -486
  172. package/.morph/framework/standards/infrastructure/azure/services/service-bus.md +0 -459
  173. package/.morph/framework/standards/infrastructure/azure/services/storage.md +0 -407
  174. package/.morph/framework/standards/infrastructure/docker/easypanel-deploy.md +0 -196
  175. package/.morph/framework/standards/infrastructure/supabase/mcp-setup.md +0 -252
  176. package/.morph/framework/standards/infrastructure/supabase/supabase-auth.md +0 -176
  177. package/.morph/framework/standards/infrastructure/supabase/supabase-pgvector.md +0 -169
  178. package/.morph/framework/standards/infrastructure/supabase/supabase-rls.md +0 -184
  179. package/.morph/framework/standards/infrastructure/supabase/supabase-storage.md +0 -153
  180. package/.morph/framework/standards/integration/api/graphql.md +0 -91
  181. package/.morph/framework/standards/integration/api/grpc.md +0 -114
  182. package/.morph/framework/standards/integration/api/rest-design.md +0 -95
  183. package/.morph/framework/standards/integration/event-driven/cqrs.md +0 -101
  184. package/.morph/framework/standards/integration/event-driven/event-sourcing.md +0 -124
  185. package/.morph/framework/standards/integration/event-driven/service-bus.md +0 -95
  186. package/.morph/framework/standards/integration/mcp/mcp-tools.md +0 -384
  187. package/.morph/framework/standards/observability/logging.md +0 -131
  188. package/.morph/framework/standards/observability/metrics.md +0 -121
  189. package/.morph/framework/standards/observability/monitoring.md +0 -114
  190. package/.morph/framework/standards/observability/tracing.md +0 -132
  191. package/.morph/framework/standards/workflows/parallel-execution.md +0 -112
  192. package/.morph/framework/standards/workflows/thread-management.md +0 -113
  193. package/.morph/framework/templates/.idea/morph-templates.xml +0 -92
  194. package/.morph/framework/templates/.vscode/morph-templates.code-snippets +0 -186
  195. package/.morph/framework/templates/IDE-SNIPPETS.md +0 -266
  196. package/.morph/framework/templates/README.md +0 -814
  197. package/.morph/framework/templates/REGISTRY.json +0 -1492
  198. package/.morph/framework/templates/code/dotnet/backend/repository.cs +0 -141
  199. package/.morph/framework/templates/code/dotnet/backend/service.cs +0 -139
  200. package/.morph/framework/templates/code/dotnet/contracts/Commands.cs +0 -74
  201. package/.morph/framework/templates/code/dotnet/contracts/Entities.cs +0 -25
  202. package/.morph/framework/templates/code/dotnet/contracts/Queries.cs +0 -74
  203. package/.morph/framework/templates/code/dotnet/contracts/README.md +0 -74
  204. package/.morph/framework/templates/code/dotnet/contracts/api-contracts.cs +0 -173
  205. package/.morph/framework/templates/code/dotnet/contracts/contracts.cs +0 -217
  206. package/.morph/framework/templates/code/dotnet/contracts/contracts.cs.hbs +0 -172
  207. package/.morph/framework/templates/code/dotnet/database/migration.cs +0 -83
  208. package/.morph/framework/templates/code/dotnet/frontend/component.razor +0 -239
  209. package/.morph/framework/templates/code/dotnet/jobs/agent.cs +0 -163
  210. package/.morph/framework/templates/code/dotnet/jobs/job.cs +0 -171
  211. package/.morph/framework/templates/code/dotnet/test.cs +0 -239
  212. package/.morph/framework/templates/code/sql/rls-policy.sql +0 -57
  213. package/.morph/framework/templates/code/sql/supabase-migration.sql +0 -100
  214. package/.morph/framework/templates/code/sql/supabase-migration.template.sql +0 -113
  215. package/.morph/framework/templates/code/typescript/contracts.ts +0 -168
  216. package/.morph/framework/templates/context/CONTEXT-FEATURE.md +0 -276
  217. package/.morph/framework/templates/context/CONTEXT.md +0 -181
  218. package/.morph/framework/templates/docs/clarifications.md +0 -253
  219. package/.morph/framework/templates/docs/onboarding.md +0 -123
  220. package/.morph/framework/templates/docs/proposal.md +0 -182
  221. package/.morph/framework/templates/docs/schema-analysis.md +0 -119
  222. package/.morph/framework/templates/docs/spec.md +0 -149
  223. package/.morph/framework/templates/docs/ui-components.md +0 -124
  224. package/.morph/framework/templates/docs/ui-design-system.md +0 -76
  225. package/.morph/framework/templates/docs/ui-flows.md +0 -167
  226. package/.morph/framework/templates/docs/ui-mockups.md +0 -98
  227. package/.morph/framework/templates/examples/design-system-examples.md +0 -357
  228. package/.morph/framework/templates/examples/spec-examples.md +0 -90
  229. package/.morph/framework/templates/feature/decisions.md +0 -187
  230. package/.morph/framework/templates/feature/recap.md +0 -146
  231. package/.morph/framework/templates/feature/tasks.md +0 -199
  232. package/.morph/framework/templates/infrastructure/azure/Dockerfile.example +0 -82
  233. package/.morph/framework/templates/infrastructure/azure/README.md +0 -286
  234. package/.morph/framework/templates/infrastructure/azure/app-insights.bicep +0 -63
  235. package/.morph/framework/templates/infrastructure/azure/app-service.bicep +0 -164
  236. package/.morph/framework/templates/infrastructure/azure/container-app-env.bicep +0 -49
  237. package/.morph/framework/templates/infrastructure/azure/container-app.bicep +0 -156
  238. package/.morph/framework/templates/infrastructure/azure/deploy-checklist.md +0 -426
  239. package/.morph/framework/templates/infrastructure/azure/deploy.ps1 +0 -229
  240. package/.morph/framework/templates/infrastructure/azure/deploy.sh +0 -208
  241. package/.morph/framework/templates/infrastructure/azure/key-vault.bicep +0 -91
  242. package/.morph/framework/templates/infrastructure/azure/main.bicep +0 -189
  243. package/.morph/framework/templates/infrastructure/azure/parameters.dev.json +0 -29
  244. package/.morph/framework/templates/infrastructure/azure/parameters.prod.json +0 -29
  245. package/.morph/framework/templates/infrastructure/azure/parameters.staging.json +0 -29
  246. package/.morph/framework/templates/infrastructure/azure/sql-database.bicep +0 -103
  247. package/.morph/framework/templates/infrastructure/azure/storage.bicep +0 -106
  248. package/.morph/framework/templates/infrastructure/docker/Dockerfile.template +0 -58
  249. package/.morph/framework/templates/infrastructure/docker/docker-compose.template.yml +0 -67
  250. package/.morph/framework/templates/infrastructure/docker/dockerfile-api.dockerfile +0 -38
  251. package/.morph/framework/templates/infrastructure/docker/dockerfile-web.dockerfile +0 -48
  252. package/.morph/framework/templates/infrastructure/docker/easypanel.template.json +0 -54
  253. package/.morph/framework/templates/infrastructure/github/README.md +0 -593
  254. package/.morph/framework/templates/infrastructure/github/actions/azure-auth/action.yml.hbs +0 -22
  255. package/.morph/framework/templates/infrastructure/github/actions/docker-build-push/action.yml.hbs +0 -45
  256. package/.morph/framework/templates/infrastructure/github/actions/health-check/action.yml.hbs +0 -27
  257. package/.morph/framework/templates/infrastructure/github/workflows/deploy-azure-app-service.yml.hbs +0 -61
  258. package/.morph/framework/templates/infrastructure/github/workflows/deploy-easypanel.yml.hbs +0 -31
  259. package/.morph/framework/templates/infrastructure/github/workflows/docker-build-push.yml.hbs +0 -59
  260. package/.morph/framework/templates/infrastructure/github/workflows/dotnet-build.yml.hbs +0 -39
  261. package/.morph/framework/templates/integrations/asaas-client.cs +0 -387
  262. package/.morph/framework/templates/integrations/asaas-webhook.cs +0 -351
  263. package/.morph/framework/templates/integrations/azure-identity-config.cs +0 -288
  264. package/.morph/framework/templates/integrations/clerk-config.cs +0 -258
  265. package/.morph/framework/templates/meta-prompts/fusion/fusion-agent.md +0 -76
  266. package/.morph/framework/templates/meta-prompts/fusion/fusion-aggregator.md +0 -100
  267. package/.morph/framework/templates/meta-prompts/hops/hop-retry.md +0 -78
  268. package/.morph/framework/templates/meta-prompts/hops/hop-validation.md +0 -97
  269. package/.morph/framework/templates/meta-prompts/hops/hop-wrapper.md +0 -36
  270. package/.morph/framework/templates/meta-prompts/parallel-workers/parallel-coordinator.md +0 -113
  271. package/.morph/framework/templates/meta-prompts/parallel-workers/parallel-worker.md +0 -80
  272. package/.morph/framework/templates/meta-prompts/squad-leaders/backend-squad.md +0 -90
  273. package/.morph/framework/templates/meta-prompts/squad-leaders/frontend-squad.md +0 -126
  274. package/.morph/framework/templates/meta-prompts/squad-leaders/squad-leader.md +0 -43
  275. package/.morph/framework/templates/meta-prompts/validators/checkpoint-validator.md +0 -107
  276. package/.morph/framework/templates/meta-prompts/validators/pre-commit-validator.md +0 -95
  277. package/.morph/framework/templates/saas/subscription.cs +0 -347
  278. package/.morph/framework/templates/saas/tenant.cs +0 -338
  279. package/.morph/framework/templates/state.template.json +0 -17
  280. package/.morph/framework/templates/ui/FluentDesignTheme.cs +0 -149
  281. package/.morph/framework/templates/ui/MudTheme.cs +0 -281
  282. package/.morph/framework/templates/ui/design-system.css +0 -226
  283. package/.morph/logs/tool-failures.log +0 -51
  284. package/.morph/memory/pre-compact-2026-02-22T17-01-01-658Z.json +0 -16
  285. package/.morph/state.json +0 -48
  286. package/framework/skills/level-2-domains/backend/api-designer.md +0 -66
  287. package/framework/skills/level-2-domains/backend/ef-modeler.md +0 -65
  288. package/framework/skills/level-2-domains/frontend/nextjs-expert.md +0 -161
  289. package/framework/skills/level-3-technologies/README.md +0 -7
  290. package/framework/skills/level-4-patterns/README.md +0 -7
  291. package/framework/templates/code/dotnet/contracts/contracts.cs +0 -217
  292. package/framework/templates/code/dotnet/contracts/contracts.cs.hbs +0 -172
@@ -0,0 +1,126 @@
1
+ # Next.js Data Fetching Standard
2
+
3
+ > **Scope:** frontend/nextjs/data-fetching
4
+ > **Layer:** 2 (on keyword)
5
+ > **Keywords:** data fetching, tanstack query, react query, server component, fetch, api, useQuery, useMutation
6
+ > **Load When:** fetching data from .NET API
7
+
8
+ Server Components for initial page load. TanStack Query for client mutations and interactive data. Never `useEffect` for fetching.
9
+
10
+ ## Core Rules
11
+
12
+ - ALWAYS use Server Components for initial data that doesn't require interactivity
13
+ - ALWAYS use TanStack Query (`useQuery`, `useMutation`) for client-side data needs
14
+ - NEVER use `useEffect(() => { fetch(...) }, [])` — this is the old pattern
15
+ - ALWAYS validate API responses with Zod before using them
16
+ - ALWAYS use query key factories for consistent cache invalidation
17
+
18
+ ## TanStack Query Setup
19
+
20
+ ```tsx
21
+ // lib/query-client.tsx
22
+ 'use client';
23
+
24
+ import { QueryClient, QueryClientProvider } from '@tanstack/react-query';
25
+ import { useState } from 'react';
26
+
27
+ export function QueryProvider({ children }: { children: React.ReactNode }) {
28
+ const [queryClient] = useState(() => new QueryClient({
29
+ defaultOptions: { queries: { staleTime: 60 * 1000, retry: 1 } },
30
+ }));
31
+ return <QueryClientProvider client={queryClient}>{children}</QueryClientProvider>;
32
+ }
33
+ ```
34
+
35
+ ## Query Key Factory Pattern
36
+
37
+ ```ts
38
+ // features/users/hooks/query-keys.ts
39
+ export const userKeys = {
40
+ all: ['users'] as const,
41
+ lists: () => [...userKeys.all, 'list'] as const,
42
+ list: (filters: Record<string, unknown>) => [...userKeys.lists(), filters] as const,
43
+ details: () => [...userKeys.all, 'detail'] as const,
44
+ detail: (id: string) => [...userKeys.details(), id] as const,
45
+ };
46
+ ```
47
+
48
+ ## useQuery Hook Pattern
49
+
50
+ ```ts
51
+ // features/users/hooks/use-users.ts
52
+ import { useQuery } from '@tanstack/react-query';
53
+ import { userKeys } from './query-keys';
54
+ import { z } from 'zod';
55
+ import { userSchema } from '@/features/users/types/user.schemas';
56
+
57
+ const usersResponseSchema = z.array(userSchema);
58
+
59
+ export function useUsers() {
60
+ return useQuery({
61
+ queryKey: userKeys.lists(),
62
+ queryFn: async () => {
63
+ const res = await fetch('/api/proxy/users');
64
+ if (!res.ok) throw new Error('Failed to fetch users');
65
+ return usersResponseSchema.parse(await res.json());
66
+ },
67
+ });
68
+ }
69
+ ```
70
+
71
+ ## useMutation Hook Pattern
72
+
73
+ ```ts
74
+ // features/users/hooks/use-create-user.ts
75
+ import { useMutation, useQueryClient } from '@tanstack/react-query';
76
+ import { userKeys } from './query-keys';
77
+ import type { CreateUserInput } from '@/features/users/types/user.types';
78
+
79
+ export function useCreateUser() {
80
+ const queryClient = useQueryClient();
81
+ return useMutation({
82
+ mutationFn: async (input: CreateUserInput) => {
83
+ const res = await fetch('/api/proxy/users', {
84
+ method: 'POST',
85
+ headers: { 'Content-Type': 'application/json' },
86
+ body: JSON.stringify(input),
87
+ });
88
+ if (!res.ok) throw new Error('Failed to create user');
89
+ return res.json();
90
+ },
91
+ onSuccess: () => {
92
+ queryClient.invalidateQueries({ queryKey: userKeys.lists() });
93
+ },
94
+ });
95
+ }
96
+ ```
97
+
98
+ ## Server Component + TanStack Query Handoff
99
+
100
+ ```tsx
101
+ // app/(dashboard)/users/page.tsx — Server Component: initial fetch
102
+ export default async function UsersPage() {
103
+ const initialUsers = await fetch(`${process.env.API_URL}/api/users`).then(r => r.json());
104
+ return <UserListClient initialData={initialUsers} />;
105
+ }
106
+
107
+ // features/users/components/user-list-client.tsx — Client Component
108
+ 'use client';
109
+ export function UserListClient({ initialData }: { initialData: User[] }) {
110
+ const { data: users = initialData } = useUsers();
111
+ // mutations, filtering, etc.
112
+ }
113
+ ```
114
+
115
+ ## Common Mistakes
116
+
117
+ | Wrong | Right | Why |
118
+ |-------|-------|-----|
119
+ | `useEffect(() => { fetch(...) }, [])` | `useQuery(...)` | Two renders, no cache, no SSR |
120
+ | No Zod on API response | `schema.parse(await res.json())` | Type safety at runtime |
121
+ | Hardcoded query keys: `['users']` | Key factory: `userKeys.lists()` | Consistent invalidation |
122
+ | `queryClient.invalidateQueries(['users'])` | `{ queryKey: userKeys.lists() }` | Object syntax required in v5 |
123
+
124
+ ---
125
+
126
+ *MORPH-SPEC by Polymorphism Tech*
@@ -0,0 +1,128 @@
1
+ # Next.js Forms Standard
2
+
3
+ > **Scope:** frontend/nextjs/forms
4
+ > **Layer:** 2 (on keyword)
5
+ > **Keywords:** form, react-hook-form, zod, validation, input, submit, zodResolver
6
+ > **Load When:** implementing any form in Next.js
7
+
8
+ react-hook-form + Zod + shadcn Form components. Schema defines both validation rules and TypeScript types.
9
+
10
+ ## Core Rules
11
+
12
+ - ALWAYS define the Zod schema first — derive the TypeScript type from it with `z.infer<>`
13
+ - ALWAYS use `zodResolver` to connect schema to react-hook-form
14
+ - ALWAYS use shadcn `<Form>`, `<FormField>`, `<FormItem>`, `<FormMessage>` for consistent UI
15
+ - NEVER use `useState` for form field values — react-hook-form handles this
16
+ - ALWAYS use `useMutation` from TanStack Query for form submission
17
+
18
+ ## Complete Form Pattern
19
+
20
+ ```tsx
21
+ // features/users/types/user.schemas.ts
22
+ import { z } from 'zod';
23
+
24
+ export const createUserSchema = z.object({
25
+ name: z.string().min(2, 'Name must be at least 2 characters'),
26
+ email: z.string().email('Invalid email address'),
27
+ role: z.enum(['admin', 'user'], { required_error: 'Role is required' }),
28
+ });
29
+
30
+ export type CreateUserInput = z.infer<typeof createUserSchema>;
31
+ ```
32
+
33
+ ```tsx
34
+ // features/users/components/create-user-form.tsx
35
+ 'use client';
36
+
37
+ import { useForm } from 'react-hook-form';
38
+ import { zodResolver } from '@hookform/resolvers/zod';
39
+ import {
40
+ Form, FormControl, FormField, FormItem, FormLabel, FormMessage
41
+ } from '@/components/ui/form';
42
+ import { Input } from '@/components/ui/input';
43
+ import { Button } from '@/components/ui/button';
44
+ import { useCreateUser } from '@/features/users/hooks/use-create-user';
45
+ import { createUserSchema, type CreateUserInput } from '@/features/users/types/user.schemas';
46
+
47
+ export function CreateUserForm({ onSuccess }: { onSuccess?: () => void }) {
48
+ const form = useForm<CreateUserInput>({
49
+ resolver: zodResolver(createUserSchema),
50
+ defaultValues: { name: '', email: '', role: 'user' },
51
+ });
52
+
53
+ const { mutate: createUser, isPending } = useCreateUser();
54
+
55
+ function onSubmit(values: CreateUserInput) {
56
+ createUser(values, {
57
+ onSuccess: () => { form.reset(); onSuccess?.(); },
58
+ onError: (error) => form.setError('root', { message: error.message }),
59
+ });
60
+ }
61
+
62
+ return (
63
+ <Form {...form}>
64
+ <form onSubmit={form.handleSubmit(onSubmit)} className="space-y-4">
65
+ <FormField
66
+ control={form.control}
67
+ name="name"
68
+ render={({ field }) => (
69
+ <FormItem>
70
+ <FormLabel>Name</FormLabel>
71
+ <FormControl><Input placeholder="John Doe" {...field} /></FormControl>
72
+ <FormMessage />
73
+ </FormItem>
74
+ )}
75
+ />
76
+ <FormField
77
+ control={form.control}
78
+ name="email"
79
+ render={({ field }) => (
80
+ <FormItem>
81
+ <FormLabel>Email</FormLabel>
82
+ <FormControl><Input type="email" {...field} /></FormControl>
83
+ <FormMessage />
84
+ </FormItem>
85
+ )}
86
+ />
87
+ {form.formState.errors.root && (
88
+ <p className="text-sm text-destructive">{form.formState.errors.root.message}</p>
89
+ )}
90
+ <Button type="submit" disabled={isPending}>
91
+ {isPending ? 'Creating...' : 'Create User'}
92
+ </Button>
93
+ </form>
94
+ </Form>
95
+ );
96
+ }
97
+ ```
98
+
99
+ ## Edit Form Pattern
100
+
101
+ ```tsx
102
+ // Edit forms reuse the same schema with partial()
103
+ const form = useForm<UpdateUserInput>({
104
+ resolver: zodResolver(updateUserSchema),
105
+ defaultValues: { name: user.name, email: user.email },
106
+ });
107
+ ```
108
+
109
+ ## Schema Composition
110
+
111
+ ```ts
112
+ export const userSchema = z.object({ id: z.string(), name: z.string(), email: z.string() });
113
+ export const createUserSchema = userSchema.omit({ id: true });
114
+ export const updateUserSchema = createUserSchema.partial();
115
+ ```
116
+
117
+ ## Common Mistakes
118
+
119
+ | Wrong | Right | Why |
120
+ |-------|-------|-----|
121
+ | `useState` for each field | `useForm` | Unnecessary re-renders on every keystroke |
122
+ | Manual error display | `<FormMessage />` | Consistent UI, auto-connects to field errors |
123
+ | `fetch` in submit handler | `useMutation` | Loading state, error handling, cache invalidation |
124
+ | Type written manually | `z.infer<typeof schema>` | Single source of truth |
125
+
126
+ ---
127
+
128
+ *MORPH-SPEC by Polymorphism Tech*
@@ -0,0 +1,67 @@
1
+ # Next.js Naming Conventions Standard
2
+
3
+ > **Scope:** frontend/nextjs/naming-conventions
4
+ > **Layer:** 1 (always)
5
+ > **Keywords:** naming, conventions, file names, component names, hooks, typescript
6
+ > **Load When:** creating any new file in a Next.js project
7
+
8
+ Consistent naming prevents filesystem bugs (Linux case-sensitivity) and keeps the codebase predictable.
9
+
10
+ ## Core Rules
11
+
12
+ - ALWAYS use kebab-case for file names — never PascalCase or camelCase
13
+ - ALWAYS use PascalCase for React component exports
14
+ - ALWAYS use camelCase starting with `use` for hook exports
15
+ - ALWAYS suffix schema files with `.schemas.ts` and type files with `.types.ts`
16
+ - NEVER mix cases in the same category — no `UserCard.tsx` alongside `user-profile.tsx`
17
+
18
+ ## Complete Reference Table
19
+
20
+ | Artifact | File Name | Export Name | Example |
21
+ |----------|-----------|-------------|---------|
22
+ | React Component | `user-card.tsx` | `UserCard` | `export function UserCard()` |
23
+ | Client Component | `user-form.tsx` | `UserForm` | `export function UserForm()` + `'use client'` |
24
+ | TanStack Query hook | `use-users.ts` | `useUsers` | `export function useUsers()` |
25
+ | Mutation hook | `use-create-user.ts` | `useCreateUser` | `export function useCreateUser()` |
26
+ | Utility hook | `use-debounce.ts` | `useDebounce` | `export function useDebounce()` |
27
+ | Zod schema file | `user.schemas.ts` | `createUserSchema` | `export const createUserSchema = z.object(...)` |
28
+ | TypeScript types | `user.types.ts` | `User`, `CreateUserInput` | `export type User = z.infer<typeof userSchema>` |
29
+ | Feature index | `index.ts` | (re-exports) | `export { UserCard } from './components/user-card'` |
30
+ | Page file | `page.tsx` | `default` | `export default function UsersPage()` |
31
+
32
+ ## Hook Naming Rules
33
+
34
+ ```
35
+ GET list: use-users.ts → useUsers()
36
+ GET single: use-user.ts → useUser(id: string)
37
+ POST: use-create-user.ts → useCreateUser()
38
+ PUT/PATCH: use-update-user.ts → useUpdateUser()
39
+ DELETE: use-delete-user.ts → useDeleteUser()
40
+ ```
41
+
42
+ ## Schema and Type Naming Rules
43
+
44
+ ```typescript
45
+ // user.schemas.ts
46
+ export const userSchema = z.object({ id: z.string(), name: z.string() });
47
+ export const createUserSchema = z.object({ name: z.string().min(2), email: z.string().email() });
48
+ export const updateUserSchema = createUserSchema.partial();
49
+
50
+ // user.types.ts — derive from schemas, don't duplicate
51
+ export type User = z.infer<typeof userSchema>;
52
+ export type CreateUserInput = z.infer<typeof createUserSchema>;
53
+ export type UpdateUserInput = z.infer<typeof updateUserSchema>;
54
+ ```
55
+
56
+ ## Common Mistakes
57
+
58
+ | Wrong | Right | Why |
59
+ |-------|-------|-----|
60
+ | `UserCard.tsx` | `user-card.tsx` | Linux servers are case-sensitive |
61
+ | `useUsers.ts` | `use-users.ts` | Inconsistent with Next.js file conventions |
62
+ | `export default function UserCard` | `export function UserCard` | Named exports are more refactor-safe |
63
+ | `type User = { id: string }` | `type User = z.infer<typeof userSchema>` | Single source of truth |
64
+
65
+ ---
66
+
67
+ *MORPH-SPEC by Polymorphism Tech*
@@ -0,0 +1,102 @@
1
+ # Next.js Project Structure Standard
2
+
3
+ > **Scope:** frontend/nextjs/project-structure
4
+ > **Layer:** 2 (on keyword)
5
+ > **Keywords:** project structure, folder structure, feature folder, src layout
6
+ > **Load When:** creating new files or features
7
+
8
+ Feature-based architecture with a shared core. Features are self-contained; shared code is explicit.
9
+
10
+ ## Core Rules
11
+
12
+ - ALWAYS use feature-based folders: `features/{feature-name}/`
13
+ - ALWAYS keep `app/` for routing only — import from `features/`, not the other way
14
+ - NEVER import from one feature into another — extract to `components/` or `lib/` if shared
15
+ - ALWAYS use `src/` directory (configured in `tsconfig.json` with `@/` alias)
16
+ - NEVER put business logic in `components/` — it is presentation only
17
+
18
+ ## Canonical Folder Tree
19
+
20
+ ```
21
+ src/
22
+ ├── app/ # Next.js App Router — routes only
23
+ │ ├── layout.tsx # Root layout + providers
24
+ │ ├── page.tsx # Home page
25
+ │ └── (dashboard)/
26
+ │ ├── layout.tsx
27
+ │ ├── users/
28
+ │ │ ├── page.tsx
29
+ │ │ ├── loading.tsx
30
+ │ │ └── error.tsx
31
+ │ └── billing/
32
+ │ └── page.tsx
33
+
34
+ ├── features/ # Domain features
35
+ │ └── {feature-name}/
36
+ │ ├── components/ # Feature-specific UI components
37
+ │ │ ├── user-list.tsx
38
+ │ │ └── user-card.tsx
39
+ │ ├── hooks/ # TanStack Query hooks for this feature
40
+ │ │ ├── use-users.ts
41
+ │ │ └── use-create-user.ts
42
+ │ ├── types/
43
+ │ │ ├── user.types.ts # TypeScript types
44
+ │ │ └── user.schemas.ts # Zod schemas + z.infer<> types
45
+ │ └── index.ts # Public API — only export what's needed
46
+
47
+ ├── components/ # Shared UI — no business logic
48
+ │ ├── ui/ # shadcn/ui — never edit directly
49
+ │ │ ├── button.tsx
50
+ │ │ └── card.tsx
51
+ │ ├── data-table.tsx
52
+ │ ├── page-header.tsx
53
+ │ └── empty-state.tsx
54
+
55
+ ├── hooks/ # Shared utility hooks — no API calls
56
+ │ ├── use-debounce.ts
57
+ │ └── use-media-query.ts
58
+
59
+ ├── lib/
60
+ │ ├── api-client.ts # Typed fetch wrapper for .NET API
61
+ │ └── query-client.tsx # TanStack Query client + provider
62
+
63
+ ├── types/
64
+ │ ├── api.ts # Shared API shapes: PaginatedResponse<T>, ApiError
65
+ │ └── env.d.ts
66
+
67
+ └── env.mjs # Zod-validated env vars
68
+ ```
69
+
70
+ ## Feature Index Pattern
71
+
72
+ ```ts
73
+ // features/users/index.ts — explicit public API
74
+ export { UserList } from './components/user-list';
75
+ export { UserCard } from './components/user-card';
76
+ export { useUsers } from './hooks/use-users';
77
+ export { useCreateUser } from './hooks/use-create-user';
78
+ export type { User, CreateUserInput } from './types/user.types';
79
+ ```
80
+
81
+ Import from the index, not deep paths:
82
+ ```ts
83
+ // GOOD
84
+ import { UserList, useUsers } from '@/features/users';
85
+
86
+ // BAD
87
+ import { UserList } from '@/features/users/components/user-list';
88
+ ```
89
+
90
+ ## Feature Boundary Rule
91
+
92
+ ```
93
+ app/ → imports from features/ ✓
94
+ features/users/ → imports from components/ and lib/ ✓
95
+ features/users/ → imports from features/billing/ ✗ (extract shared instead)
96
+ components/ → imports from components/ui/ ✓
97
+ components/ → imports from features/ ✗
98
+ ```
99
+
100
+ ---
101
+
102
+ *MORPH-SPEC by Polymorphism Tech*
@@ -0,0 +1,72 @@
1
+ # Next.js State Management Standard
2
+
3
+ > **Scope:** frontend/nextjs/state-management
4
+ > **Layer:** 2 (on keyword)
5
+ > **Keywords:** state, zustand, context, redux, client state, server state, useState
6
+ > **Load When:** deciding how to manage state in Next.js
7
+
8
+ Server state (API data) lives in TanStack Query. UI state lives in React. No global state library unless genuinely needed.
9
+
10
+ ## Core Rules
11
+
12
+ - ALWAYS use TanStack Query for anything that comes from the .NET API
13
+ - ALWAYS use `useState` for local UI state (modal open, selected tab, form step)
14
+ - NEVER install Zustand or Redux without first exhausting Server Components + TanStack Query
15
+ - NEVER use React Context for server state — Context does not cache or deduplicate
16
+ - ALWAYS derive state from server data rather than syncing it to local state
17
+
18
+ ## State Decision Tree
19
+
20
+ ```
21
+ Is this data from the .NET API?
22
+ YES → TanStack Query (useQuery / useMutation)
23
+
24
+ Is this local UI state (open/closed, selected, current step)?
25
+ YES → useState in the component that needs it
26
+
27
+ Does multiple components need this UI state?
28
+ YES — is it parent-child? → prop drilling (2-3 levels is fine)
29
+ YES — is it truly global? → React Context (theme, auth user, locale)
30
+ YES — complex with many actions? → Consider Zustand (last resort)
31
+ ```
32
+
33
+ ## Server State (TanStack Query)
34
+
35
+ ```ts
36
+ // GOOD — server state in TanStack Query
37
+ const { data: users } = useUsers(); // cached, deduplicated, background-refetched
38
+
39
+ // BAD — copying server state to useState
40
+ const [users, setUsers] = useState([]);
41
+ useEffect(() => { fetchUsers().then(setUsers); }, []);
42
+ ```
43
+
44
+ ## Local UI State (useState)
45
+
46
+ ```tsx
47
+ function UserList() {
48
+ const [isCreateDialogOpen, setIsCreateDialogOpen] = useState(false);
49
+ const [selectedUserId, setSelectedUserId] = useState<string | null>(null);
50
+ }
51
+ ```
52
+
53
+ ## When Zustand Is Justified
54
+
55
+ Only add Zustand if ALL of these are true:
56
+ 1. State is needed in 5+ unrelated components
57
+ 2. State has complex update logic (not just toggle/set)
58
+ 3. Context causes measurable re-render performance issues
59
+ 4. State is not from the server
60
+
61
+ ## Common Mistakes
62
+
63
+ | Wrong | Right | Why |
64
+ |-------|-------|-----|
65
+ | `useEffect` + `useState` for API data | TanStack Query | Stale data, no cache, double-render |
66
+ | Global store for user list | `useUsers()` | TanStack Query handles sync/cache |
67
+ | Context for everything | useState + prop drilling first | Context causes all consumers to re-render |
68
+ | Zustand by default | Start with useState | YAGNI — most state is local |
69
+
70
+ ---
71
+
72
+ *MORPH-SPEC by Polymorphism Tech*
@@ -0,0 +1,111 @@
1
+ # Next.js Testing Standard
2
+
3
+ > **Scope:** frontend/nextjs/testing
4
+ > **Layer:** 2 (on keyword)
5
+ > **Keywords:** testing, jest, vitest, testing-library, msw, component test, unit test
6
+ > **Load When:** writing tests for Next.js components or hooks
7
+
8
+ Jest + React Testing Library for component tests. MSW for API mocking. Co-locate tests with the code they test.
9
+
10
+ ## Core Rules
11
+
12
+ - ALWAYS co-locate test files: `user-card.test.tsx` next to `user-card.tsx`
13
+ - ALWAYS use `@testing-library/user-event` for interactions — not `fireEvent`
14
+ - ALWAYS mock the API layer with MSW — never mock `fetch` directly
15
+ - NEVER test implementation details — test what the user sees
16
+ - ALWAYS test the happy path + one error path per component
17
+
18
+ ## Test File Co-location
19
+
20
+ ```
21
+ features/users/
22
+ ├── components/
23
+ │ ├── user-card.tsx
24
+ │ ├── user-card.test.tsx ← co-located
25
+ │ ├── user-list.tsx
26
+ │ └── user-list.test.tsx
27
+ ├── hooks/
28
+ │ ├── use-users.ts
29
+ │ └── use-users.test.ts ← co-located
30
+ ```
31
+
32
+ ## Component Test Pattern
33
+
34
+ ```tsx
35
+ // features/users/components/user-card.test.tsx
36
+ import { render, screen } from '@testing-library/react';
37
+ import userEvent from '@testing-library/user-event';
38
+ import { UserCard } from './user-card';
39
+
40
+ const mockUser = { id: '1', name: 'João Silva', email: 'joao@example.com', role: 'user' as const };
41
+
42
+ describe('UserCard', () => {
43
+ it('renders user name and email', () => {
44
+ render(<UserCard user={mockUser} />);
45
+ expect(screen.getByText('João Silva')).toBeInTheDocument();
46
+ expect(screen.getByText('joao@example.com')).toBeInTheDocument();
47
+ });
48
+
49
+ it('calls onEdit with user id when edit button clicked', async () => {
50
+ const onEdit = vi.fn();
51
+ render(<UserCard user={mockUser} onEdit={onEdit} />);
52
+ await userEvent.click(screen.getByRole('button', { name: /edit/i }));
53
+ expect(onEdit).toHaveBeenCalledWith('1');
54
+ });
55
+ });
56
+ ```
57
+
58
+ ## Hook Test with MSW
59
+
60
+ ```ts
61
+ // features/users/hooks/use-users.test.ts
62
+ import { renderHook, waitFor } from '@testing-library/react';
63
+ import { http, HttpResponse } from 'msw';
64
+ import { server } from '@/test/msw-server';
65
+ import { QueryClientWrapper } from '@/test/helpers';
66
+ import { useUsers } from './use-users';
67
+
68
+ describe('useUsers', () => {
69
+ it('returns users from API', async () => {
70
+ server.use(
71
+ http.get('/api/proxy/users', () =>
72
+ HttpResponse.json([{ id: '1', name: 'João', email: 'j@ex.com' }])
73
+ )
74
+ );
75
+ const { result } = renderHook(() => useUsers(), { wrapper: QueryClientWrapper });
76
+ await waitFor(() => expect(result.current.isSuccess).toBe(true));
77
+ expect(result.current.data).toHaveLength(1);
78
+ });
79
+
80
+ it('returns error state when API fails', async () => {
81
+ server.use(http.get('/api/proxy/users', () => new HttpResponse(null, { status: 500 })));
82
+ const { result } = renderHook(() => useUsers(), { wrapper: QueryClientWrapper });
83
+ await waitFor(() => expect(result.current.isError).toBe(true));
84
+ });
85
+ });
86
+ ```
87
+
88
+ ## Test Helper Setup
89
+
90
+ ```tsx
91
+ // test/helpers.tsx
92
+ import { QueryClient, QueryClientProvider } from '@tanstack/react-query';
93
+
94
+ export function QueryClientWrapper({ children }: { children: React.ReactNode }) {
95
+ const queryClient = new QueryClient({ defaultOptions: { queries: { retry: false } } });
96
+ return <QueryClientProvider client={queryClient}>{children}</QueryClientProvider>;
97
+ }
98
+ ```
99
+
100
+ ## Common Mistakes
101
+
102
+ | Wrong | Right | Why |
103
+ |-------|-------|-----|
104
+ | `fireEvent.click(button)` | `await userEvent.click(button)` | userEvent simulates real browser events |
105
+ | Mock `global.fetch` | Mock with MSW | MSW intercepts at network level |
106
+ | `expect(component).toMatchSnapshot()` | `expect(screen.getByText(...))` | Snapshots break on any change |
107
+ | Test file in `__tests__/` folder | Co-locate with source | Easier to find, deleted with the component |
108
+
109
+ ---
110
+
111
+ *MORPH-SPEC by Polymorphism Tech*