moai-adk 0.35.1__py3-none-any.whl

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.

Potentially problematic release.


This version of moai-adk might be problematic. Click here for more details.

Files changed (502) hide show
  1. moai_adk/__init__.py +10 -0
  2. moai_adk/__main__.py +199 -0
  3. moai_adk/cli/__init__.py +6 -0
  4. moai_adk/cli/commands/__init__.py +17 -0
  5. moai_adk/cli/commands/analyze.py +116 -0
  6. moai_adk/cli/commands/doctor.py +272 -0
  7. moai_adk/cli/commands/init.py +372 -0
  8. moai_adk/cli/commands/language.py +248 -0
  9. moai_adk/cli/commands/status.py +104 -0
  10. moai_adk/cli/commands/update.py +2686 -0
  11. moai_adk/cli/main.py +13 -0
  12. moai_adk/cli/prompts/__init__.py +5 -0
  13. moai_adk/cli/prompts/init_prompts.py +219 -0
  14. moai_adk/cli/spec_status.py +263 -0
  15. moai_adk/cli/ui/__init__.py +44 -0
  16. moai_adk/cli/ui/progress.py +422 -0
  17. moai_adk/cli/ui/prompts.py +389 -0
  18. moai_adk/cli/ui/theme.py +129 -0
  19. moai_adk/cli/worktree/__init__.py +27 -0
  20. moai_adk/cli/worktree/__main__.py +31 -0
  21. moai_adk/cli/worktree/cli.py +683 -0
  22. moai_adk/cli/worktree/exceptions.py +89 -0
  23. moai_adk/cli/worktree/manager.py +493 -0
  24. moai_adk/cli/worktree/models.py +65 -0
  25. moai_adk/cli/worktree/registry.py +422 -0
  26. moai_adk/core/PHASE2_OPTIMIZATIONS.md +467 -0
  27. moai_adk/core/__init__.py +1 -0
  28. moai_adk/core/analysis/__init__.py +9 -0
  29. moai_adk/core/analysis/session_analyzer.py +400 -0
  30. moai_adk/core/claude_integration.py +393 -0
  31. moai_adk/core/command_helpers.py +270 -0
  32. moai_adk/core/comprehensive_monitoring_system.py +1183 -0
  33. moai_adk/core/config/__init__.py +19 -0
  34. moai_adk/core/config/auto_spec_config.py +340 -0
  35. moai_adk/core/config/migration.py +244 -0
  36. moai_adk/core/config/unified.py +436 -0
  37. moai_adk/core/context_manager.py +273 -0
  38. moai_adk/core/diagnostics/__init__.py +19 -0
  39. moai_adk/core/diagnostics/slash_commands.py +159 -0
  40. moai_adk/core/enterprise_features.py +1404 -0
  41. moai_adk/core/error_recovery_system.py +1902 -0
  42. moai_adk/core/event_driven_hook_system.py +1371 -0
  43. moai_adk/core/git/__init__.py +31 -0
  44. moai_adk/core/git/branch.py +25 -0
  45. moai_adk/core/git/branch_manager.py +129 -0
  46. moai_adk/core/git/checkpoint.py +134 -0
  47. moai_adk/core/git/commit.py +67 -0
  48. moai_adk/core/git/conflict_detector.py +413 -0
  49. moai_adk/core/git/event_detector.py +79 -0
  50. moai_adk/core/git/manager.py +216 -0
  51. moai_adk/core/hooks/post_tool_auto_spec_completion.py +901 -0
  52. moai_adk/core/input_validation_middleware.py +1006 -0
  53. moai_adk/core/integration/__init__.py +22 -0
  54. moai_adk/core/integration/engine.py +157 -0
  55. moai_adk/core/integration/integration_tester.py +226 -0
  56. moai_adk/core/integration/models.py +88 -0
  57. moai_adk/core/integration/utils.py +211 -0
  58. moai_adk/core/issue_creator.py +305 -0
  59. moai_adk/core/jit_context_loader.py +956 -0
  60. moai_adk/core/jit_enhanced_hook_manager.py +1987 -0
  61. moai_adk/core/language_config.py +202 -0
  62. moai_adk/core/language_config_resolver.py +572 -0
  63. moai_adk/core/language_validator.py +543 -0
  64. moai_adk/core/mcp/setup.py +116 -0
  65. moai_adk/core/merge/__init__.py +9 -0
  66. moai_adk/core/merge/analyzer.py +605 -0
  67. moai_adk/core/migration/__init__.py +18 -0
  68. moai_adk/core/migration/alfred_to_moai_migrator.py +383 -0
  69. moai_adk/core/migration/backup_manager.py +277 -0
  70. moai_adk/core/migration/custom_element_scanner.py +358 -0
  71. moai_adk/core/migration/file_migrator.py +209 -0
  72. moai_adk/core/migration/interactive_checkbox_ui.py +488 -0
  73. moai_adk/core/migration/selective_restorer.py +470 -0
  74. moai_adk/core/migration/template_utils.py +74 -0
  75. moai_adk/core/migration/user_selection_ui.py +338 -0
  76. moai_adk/core/migration/version_detector.py +139 -0
  77. moai_adk/core/migration/version_migrator.py +228 -0
  78. moai_adk/core/performance/__init__.py +6 -0
  79. moai_adk/core/performance/cache_system.py +316 -0
  80. moai_adk/core/performance/parallel_processor.py +116 -0
  81. moai_adk/core/phase_optimized_hook_scheduler.py +879 -0
  82. moai_adk/core/project/__init__.py +1 -0
  83. moai_adk/core/project/backup_utils.py +70 -0
  84. moai_adk/core/project/checker.py +300 -0
  85. moai_adk/core/project/detector.py +293 -0
  86. moai_adk/core/project/initializer.py +387 -0
  87. moai_adk/core/project/phase_executor.py +716 -0
  88. moai_adk/core/project/validator.py +139 -0
  89. moai_adk/core/quality/__init__.py +6 -0
  90. moai_adk/core/quality/trust_checker.py +377 -0
  91. moai_adk/core/quality/validators/__init__.py +6 -0
  92. moai_adk/core/quality/validators/base_validator.py +19 -0
  93. moai_adk/core/realtime_monitoring_dashboard.py +1724 -0
  94. moai_adk/core/robust_json_parser.py +611 -0
  95. moai_adk/core/rollback_manager.py +918 -0
  96. moai_adk/core/session_manager.py +651 -0
  97. moai_adk/core/skill_loading_system.py +579 -0
  98. moai_adk/core/spec/confidence_scoring.py +680 -0
  99. moai_adk/core/spec/ears_template_engine.py +1247 -0
  100. moai_adk/core/spec/quality_validator.py +687 -0
  101. moai_adk/core/spec_status_manager.py +478 -0
  102. moai_adk/core/template/__init__.py +7 -0
  103. moai_adk/core/template/backup.py +174 -0
  104. moai_adk/core/template/config.py +191 -0
  105. moai_adk/core/template/languages.py +43 -0
  106. moai_adk/core/template/merger.py +233 -0
  107. moai_adk/core/template/processor.py +1200 -0
  108. moai_adk/core/template_engine.py +310 -0
  109. moai_adk/core/template_variable_synchronizer.py +417 -0
  110. moai_adk/core/unified_permission_manager.py +745 -0
  111. moai_adk/core/user_behavior_analytics.py +851 -0
  112. moai_adk/core/version_sync.py +429 -0
  113. moai_adk/foundation/__init__.py +56 -0
  114. moai_adk/foundation/backend.py +1027 -0
  115. moai_adk/foundation/database.py +1115 -0
  116. moai_adk/foundation/devops.py +1585 -0
  117. moai_adk/foundation/ears.py +431 -0
  118. moai_adk/foundation/frontend.py +870 -0
  119. moai_adk/foundation/git/commit_templates.py +557 -0
  120. moai_adk/foundation/git.py +376 -0
  121. moai_adk/foundation/langs.py +484 -0
  122. moai_adk/foundation/ml_ops.py +1162 -0
  123. moai_adk/foundation/testing.py +1524 -0
  124. moai_adk/foundation/trust/trust_principles.py +676 -0
  125. moai_adk/foundation/trust/validation_checklist.py +1573 -0
  126. moai_adk/project/__init__.py +0 -0
  127. moai_adk/project/configuration.py +1084 -0
  128. moai_adk/project/documentation.py +566 -0
  129. moai_adk/project/schema.py +447 -0
  130. moai_adk/statusline/__init__.py +38 -0
  131. moai_adk/statusline/alfred_detector.py +105 -0
  132. moai_adk/statusline/config.py +376 -0
  133. moai_adk/statusline/enhanced_output_style_detector.py +372 -0
  134. moai_adk/statusline/git_collector.py +190 -0
  135. moai_adk/statusline/main.py +322 -0
  136. moai_adk/statusline/metrics_tracker.py +78 -0
  137. moai_adk/statusline/renderer.py +343 -0
  138. moai_adk/statusline/update_checker.py +129 -0
  139. moai_adk/statusline/version_reader.py +741 -0
  140. moai_adk/templates/.claude/agents/moai/ai-nano-banana.md +714 -0
  141. moai_adk/templates/.claude/agents/moai/builder-agent.md +474 -0
  142. moai_adk/templates/.claude/agents/moai/builder-command.md +1172 -0
  143. moai_adk/templates/.claude/agents/moai/builder-plugin.md +637 -0
  144. moai_adk/templates/.claude/agents/moai/builder-skill.md +666 -0
  145. moai_adk/templates/.claude/agents/moai/expert-backend.md +899 -0
  146. moai_adk/templates/.claude/agents/moai/expert-database.md +777 -0
  147. moai_adk/templates/.claude/agents/moai/expert-debug.md +401 -0
  148. moai_adk/templates/.claude/agents/moai/expert-devops.md +720 -0
  149. moai_adk/templates/.claude/agents/moai/expert-frontend.md +734 -0
  150. moai_adk/templates/.claude/agents/moai/expert-performance.md +657 -0
  151. moai_adk/templates/.claude/agents/moai/expert-security.md +513 -0
  152. moai_adk/templates/.claude/agents/moai/expert-testing.md +733 -0
  153. moai_adk/templates/.claude/agents/moai/expert-uiux.md +1041 -0
  154. moai_adk/templates/.claude/agents/moai/manager-claude-code.md +432 -0
  155. moai_adk/templates/.claude/agents/moai/manager-docs.md +573 -0
  156. moai_adk/templates/.claude/agents/moai/manager-git.md +1060 -0
  157. moai_adk/templates/.claude/agents/moai/manager-project.md +891 -0
  158. moai_adk/templates/.claude/agents/moai/manager-quality.md +624 -0
  159. moai_adk/templates/.claude/agents/moai/manager-spec.md +809 -0
  160. moai_adk/templates/.claude/agents/moai/manager-strategy.md +780 -0
  161. moai_adk/templates/.claude/agents/moai/manager-tdd.md +784 -0
  162. moai_adk/templates/.claude/agents/moai/mcp-context7.md +458 -0
  163. moai_adk/templates/.claude/agents/moai/mcp-figma.md +1607 -0
  164. moai_adk/templates/.claude/agents/moai/mcp-notion.md +789 -0
  165. moai_adk/templates/.claude/agents/moai/mcp-playwright.md +469 -0
  166. moai_adk/templates/.claude/agents/moai/mcp-sequential-thinking.md +1032 -0
  167. moai_adk/templates/.claude/commands/moai/0-project.md +1386 -0
  168. moai_adk/templates/.claude/commands/moai/1-plan.md +1427 -0
  169. moai_adk/templates/.claude/commands/moai/2-run.md +943 -0
  170. moai_adk/templates/.claude/commands/moai/3-sync.md +1324 -0
  171. moai_adk/templates/.claude/commands/moai/9-feedback.md +314 -0
  172. moai_adk/templates/.claude/hooks/__init__.py +8 -0
  173. moai_adk/templates/.claude/hooks/moai/__init__.py +8 -0
  174. moai_adk/templates/.claude/hooks/moai/lib/__init__.py +85 -0
  175. moai_adk/templates/.claude/hooks/moai/lib/checkpoint.py +244 -0
  176. moai_adk/templates/.claude/hooks/moai/lib/common.py +131 -0
  177. moai_adk/templates/.claude/hooks/moai/lib/config_manager.py +446 -0
  178. moai_adk/templates/.claude/hooks/moai/lib/config_validator.py +639 -0
  179. moai_adk/templates/.claude/hooks/moai/lib/example_config.json +104 -0
  180. moai_adk/templates/.claude/hooks/moai/lib/git_operations_manager.py +590 -0
  181. moai_adk/templates/.claude/hooks/moai/lib/language_validator.py +317 -0
  182. moai_adk/templates/.claude/hooks/moai/lib/models.py +102 -0
  183. moai_adk/templates/.claude/hooks/moai/lib/path_utils.py +28 -0
  184. moai_adk/templates/.claude/hooks/moai/lib/project.py +768 -0
  185. moai_adk/templates/.claude/hooks/moai/lib/test_hooks_improvements.py +443 -0
  186. moai_adk/templates/.claude/hooks/moai/lib/timeout.py +160 -0
  187. moai_adk/templates/.claude/hooks/moai/lib/unified_timeout_manager.py +530 -0
  188. moai_adk/templates/.claude/hooks/moai/session_end__auto_cleanup.py +862 -0
  189. moai_adk/templates/.claude/hooks/moai/session_start__show_project_info.py +1083 -0
  190. moai_adk/templates/.claude/output-styles/moai/r2d2.md +560 -0
  191. moai_adk/templates/.claude/output-styles/moai/yoda.md +359 -0
  192. moai_adk/templates/.claude/settings.json +172 -0
  193. moai_adk/templates/.claude/skills/moai-ai-nano-banana/SKILL.md +307 -0
  194. moai_adk/templates/.claude/skills/moai-ai-nano-banana/examples.md +431 -0
  195. moai_adk/templates/.claude/skills/moai-ai-nano-banana/scripts/batch_generate.py +560 -0
  196. moai_adk/templates/.claude/skills/moai-ai-nano-banana/scripts/generate_image.py +362 -0
  197. moai_adk/templates/.claude/skills/moai-docs-generation/SKILL.md +249 -0
  198. moai_adk/templates/.claude/skills/moai-docs-generation/examples.md +406 -0
  199. moai_adk/templates/.claude/skills/moai-docs-generation/modules/README.md +44 -0
  200. moai_adk/templates/.claude/skills/moai-docs-generation/modules/api-documentation.md +130 -0
  201. moai_adk/templates/.claude/skills/moai-docs-generation/modules/code-documentation.md +152 -0
  202. moai_adk/templates/.claude/skills/moai-docs-generation/modules/multi-format-output.md +178 -0
  203. moai_adk/templates/.claude/skills/moai-docs-generation/modules/user-guides.md +147 -0
  204. moai_adk/templates/.claude/skills/moai-docs-generation/reference.md +328 -0
  205. moai_adk/templates/.claude/skills/moai-domain-backend/SKILL.md +320 -0
  206. moai_adk/templates/.claude/skills/moai-domain-backend/examples.md +718 -0
  207. moai_adk/templates/.claude/skills/moai-domain-backend/reference.md +464 -0
  208. moai_adk/templates/.claude/skills/moai-domain-database/SKILL.md +323 -0
  209. moai_adk/templates/.claude/skills/moai-domain-database/examples.md +830 -0
  210. moai_adk/templates/.claude/skills/moai-domain-database/modules/README.md +53 -0
  211. moai_adk/templates/.claude/skills/moai-domain-database/modules/mongodb.md +231 -0
  212. moai_adk/templates/.claude/skills/moai-domain-database/modules/postgresql.md +169 -0
  213. moai_adk/templates/.claude/skills/moai-domain-database/modules/redis.md +262 -0
  214. moai_adk/templates/.claude/skills/moai-domain-database/reference.md +545 -0
  215. moai_adk/templates/.claude/skills/moai-domain-frontend/SKILL.md +497 -0
  216. moai_adk/templates/.claude/skills/moai-domain-frontend/examples.md +968 -0
  217. moai_adk/templates/.claude/skills/moai-domain-frontend/reference.md +664 -0
  218. moai_adk/templates/.claude/skills/moai-domain-uiux/SKILL.md +455 -0
  219. moai_adk/templates/.claude/skills/moai-domain-uiux/examples.md +560 -0
  220. moai_adk/templates/.claude/skills/moai-domain-uiux/modules/accessibility-wcag.md +260 -0
  221. moai_adk/templates/.claude/skills/moai-domain-uiux/modules/component-architecture.md +228 -0
  222. moai_adk/templates/.claude/skills/moai-domain-uiux/modules/icon-libraries.md +401 -0
  223. moai_adk/templates/.claude/skills/moai-domain-uiux/modules/theming-system.md +373 -0
  224. moai_adk/templates/.claude/skills/moai-domain-uiux/reference.md +243 -0
  225. moai_adk/templates/.claude/skills/moai-formats-data/SKILL.md +492 -0
  226. moai_adk/templates/.claude/skills/moai-formats-data/examples.md +804 -0
  227. moai_adk/templates/.claude/skills/moai-formats-data/modules/README.md +98 -0
  228. moai_adk/templates/.claude/skills/moai-formats-data/modules/SKILL-MODULARIZATION-TEMPLATE.md +278 -0
  229. moai_adk/templates/.claude/skills/moai-formats-data/modules/caching-performance.md +459 -0
  230. moai_adk/templates/.claude/skills/moai-formats-data/modules/data-validation.md +485 -0
  231. moai_adk/templates/.claude/skills/moai-formats-data/modules/json-optimization.md +374 -0
  232. moai_adk/templates/.claude/skills/moai-formats-data/modules/toon-encoding.md +308 -0
  233. moai_adk/templates/.claude/skills/moai-formats-data/reference.md +585 -0
  234. moai_adk/templates/.claude/skills/moai-foundation-claude/SKILL.md +202 -0
  235. moai_adk/templates/.claude/skills/moai-foundation-claude/examples.md +732 -0
  236. moai_adk/templates/.claude/skills/moai-foundation-claude/reference/best-practices-checklist.md +616 -0
  237. moai_adk/templates/.claude/skills/moai-foundation-claude/reference/claude-code-custom-slash-commands-official.md +729 -0
  238. moai_adk/templates/.claude/skills/moai-foundation-claude/reference/claude-code-hooks-official.md +560 -0
  239. moai_adk/templates/.claude/skills/moai-foundation-claude/reference/claude-code-iam-official.md +635 -0
  240. moai_adk/templates/.claude/skills/moai-foundation-claude/reference/claude-code-memory-official.md +543 -0
  241. moai_adk/templates/.claude/skills/moai-foundation-claude/reference/claude-code-settings-official.md +663 -0
  242. moai_adk/templates/.claude/skills/moai-foundation-claude/reference/claude-code-skills-official.md +113 -0
  243. moai_adk/templates/.claude/skills/moai-foundation-claude/reference/claude-code-sub-agents-official.md +238 -0
  244. moai_adk/templates/.claude/skills/moai-foundation-claude/reference/complete-configuration-guide.md +175 -0
  245. moai_adk/templates/.claude/skills/moai-foundation-claude/reference/skill-examples.md +1674 -0
  246. moai_adk/templates/.claude/skills/moai-foundation-claude/reference/skill-formatting-guide.md +729 -0
  247. moai_adk/templates/.claude/skills/moai-foundation-claude/reference/sub-agents/sub-agent-examples.md +1513 -0
  248. moai_adk/templates/.claude/skills/moai-foundation-claude/reference/sub-agents/sub-agent-formatting-guide.md +1086 -0
  249. moai_adk/templates/.claude/skills/moai-foundation-claude/reference/sub-agents/sub-agent-integration-patterns.md +1100 -0
  250. moai_adk/templates/.claude/skills/moai-foundation-claude/reference.md +209 -0
  251. moai_adk/templates/.claude/skills/moai-foundation-context/SKILL.md +441 -0
  252. moai_adk/templates/.claude/skills/moai-foundation-context/examples.md +1048 -0
  253. moai_adk/templates/.claude/skills/moai-foundation-context/reference.md +246 -0
  254. moai_adk/templates/.claude/skills/moai-foundation-core/SKILL.md +420 -0
  255. moai_adk/templates/.claude/skills/moai-foundation-core/examples.md +358 -0
  256. moai_adk/templates/.claude/skills/moai-foundation-core/modules/README.md +296 -0
  257. moai_adk/templates/.claude/skills/moai-foundation-core/modules/agents-reference.md +359 -0
  258. moai_adk/templates/.claude/skills/moai-foundation-core/modules/commands-reference.md +432 -0
  259. moai_adk/templates/.claude/skills/moai-foundation-core/modules/delegation-patterns.md +757 -0
  260. moai_adk/templates/.claude/skills/moai-foundation-core/modules/execution-rules.md +687 -0
  261. moai_adk/templates/.claude/skills/moai-foundation-core/modules/modular-system.md +665 -0
  262. moai_adk/templates/.claude/skills/moai-foundation-core/modules/progressive-disclosure.md +649 -0
  263. moai_adk/templates/.claude/skills/moai-foundation-core/modules/spec-first-tdd.md +864 -0
  264. moai_adk/templates/.claude/skills/moai-foundation-core/modules/token-optimization.md +708 -0
  265. moai_adk/templates/.claude/skills/moai-foundation-core/modules/trust-5-framework.md +981 -0
  266. moai_adk/templates/.claude/skills/moai-foundation-core/reference.md +478 -0
  267. moai_adk/templates/.claude/skills/moai-foundation-philosopher/SKILL.md +315 -0
  268. moai_adk/templates/.claude/skills/moai-foundation-philosopher/examples.md +228 -0
  269. moai_adk/templates/.claude/skills/moai-foundation-philosopher/modules/assumption-matrix.md +80 -0
  270. moai_adk/templates/.claude/skills/moai-foundation-philosopher/modules/cognitive-bias.md +199 -0
  271. moai_adk/templates/.claude/skills/moai-foundation-philosopher/modules/first-principles.md +140 -0
  272. moai_adk/templates/.claude/skills/moai-foundation-philosopher/modules/trade-off-analysis.md +154 -0
  273. moai_adk/templates/.claude/skills/moai-foundation-philosopher/reference.md +157 -0
  274. moai_adk/templates/.claude/skills/moai-foundation-quality/SKILL.md +364 -0
  275. moai_adk/templates/.claude/skills/moai-foundation-quality/examples.md +1232 -0
  276. moai_adk/templates/.claude/skills/moai-foundation-quality/modules/best-practices.md +261 -0
  277. moai_adk/templates/.claude/skills/moai-foundation-quality/modules/integration-patterns.md +194 -0
  278. moai_adk/templates/.claude/skills/moai-foundation-quality/modules/proactive-analysis.md +229 -0
  279. moai_adk/templates/.claude/skills/moai-foundation-quality/modules/trust5-validation.md +169 -0
  280. moai_adk/templates/.claude/skills/moai-foundation-quality/reference.md +1266 -0
  281. moai_adk/templates/.claude/skills/moai-foundation-quality/scripts/quality-gate.sh +668 -0
  282. moai_adk/templates/.claude/skills/moai-foundation-quality/templates/github-actions-quality.yml +481 -0
  283. moai_adk/templates/.claude/skills/moai-foundation-quality/templates/quality-config.yaml +519 -0
  284. moai_adk/templates/.claude/skills/moai-lang-cpp/SKILL.md +649 -0
  285. moai_adk/templates/.claude/skills/moai-lang-csharp/SKILL.md +478 -0
  286. moai_adk/templates/.claude/skills/moai-lang-elixir/SKILL.md +612 -0
  287. moai_adk/templates/.claude/skills/moai-lang-flutter/SKILL.md +477 -0
  288. moai_adk/templates/.claude/skills/moai-lang-flutter/examples.md +1090 -0
  289. moai_adk/templates/.claude/skills/moai-lang-flutter/reference.md +686 -0
  290. moai_adk/templates/.claude/skills/moai-lang-go/SKILL.md +376 -0
  291. moai_adk/templates/.claude/skills/moai-lang-go/examples.md +919 -0
  292. moai_adk/templates/.claude/skills/moai-lang-go/reference.md +737 -0
  293. moai_adk/templates/.claude/skills/moai-lang-java/SKILL.md +385 -0
  294. moai_adk/templates/.claude/skills/moai-lang-java/examples.md +864 -0
  295. moai_adk/templates/.claude/skills/moai-lang-java/reference.md +291 -0
  296. moai_adk/templates/.claude/skills/moai-lang-kotlin/SKILL.md +382 -0
  297. moai_adk/templates/.claude/skills/moai-lang-kotlin/examples.md +1006 -0
  298. moai_adk/templates/.claude/skills/moai-lang-kotlin/reference.md +562 -0
  299. moai_adk/templates/.claude/skills/moai-lang-php/SKILL.md +644 -0
  300. moai_adk/templates/.claude/skills/moai-lang-python/SKILL.md +481 -0
  301. moai_adk/templates/.claude/skills/moai-lang-python/examples.md +977 -0
  302. moai_adk/templates/.claude/skills/moai-lang-python/reference.md +804 -0
  303. moai_adk/templates/.claude/skills/moai-lang-r/SKILL.md +579 -0
  304. moai_adk/templates/.claude/skills/moai-lang-ruby/SKILL.md +687 -0
  305. moai_adk/templates/.claude/skills/moai-lang-rust/SKILL.md +372 -0
  306. moai_adk/templates/.claude/skills/moai-lang-rust/examples.md +659 -0
  307. moai_adk/templates/.claude/skills/moai-lang-rust/reference.md +504 -0
  308. moai_adk/templates/.claude/skills/moai-lang-scala/SKILL.md +497 -0
  309. moai_adk/templates/.claude/skills/moai-lang-scala/examples.md +633 -0
  310. moai_adk/templates/.claude/skills/moai-lang-scala/reference.md +423 -0
  311. moai_adk/templates/.claude/skills/moai-lang-swift/SKILL.md +497 -0
  312. moai_adk/templates/.claude/skills/moai-lang-swift/examples.md +918 -0
  313. moai_adk/templates/.claude/skills/moai-lang-swift/reference.md +672 -0
  314. moai_adk/templates/.claude/skills/moai-lang-typescript/SKILL.md +368 -0
  315. moai_adk/templates/.claude/skills/moai-lang-typescript/examples.md +1089 -0
  316. moai_adk/templates/.claude/skills/moai-lang-typescript/reference.md +731 -0
  317. moai_adk/templates/.claude/skills/moai-library-mermaid/SKILL.md +300 -0
  318. moai_adk/templates/.claude/skills/moai-library-mermaid/advanced-patterns.md +465 -0
  319. moai_adk/templates/.claude/skills/moai-library-mermaid/examples.md +270 -0
  320. moai_adk/templates/.claude/skills/moai-library-mermaid/optimization.md +440 -0
  321. moai_adk/templates/.claude/skills/moai-library-mermaid/reference.md +228 -0
  322. moai_adk/templates/.claude/skills/moai-library-nextra/SKILL.md +319 -0
  323. moai_adk/templates/.claude/skills/moai-library-nextra/advanced-patterns.md +336 -0
  324. moai_adk/templates/.claude/skills/moai-library-nextra/examples.md +592 -0
  325. moai_adk/templates/.claude/skills/moai-library-nextra/modules/advanced-deployment-patterns.md +182 -0
  326. moai_adk/templates/.claude/skills/moai-library-nextra/modules/advanced-patterns.md +17 -0
  327. moai_adk/templates/.claude/skills/moai-library-nextra/modules/configuration.md +57 -0
  328. moai_adk/templates/.claude/skills/moai-library-nextra/modules/content-architecture-optimization.md +162 -0
  329. moai_adk/templates/.claude/skills/moai-library-nextra/modules/deployment.md +52 -0
  330. moai_adk/templates/.claude/skills/moai-library-nextra/modules/framework-core-configuration.md +186 -0
  331. moai_adk/templates/.claude/skills/moai-library-nextra/modules/i18n-setup.md +55 -0
  332. moai_adk/templates/.claude/skills/moai-library-nextra/modules/mdx-components.md +52 -0
  333. moai_adk/templates/.claude/skills/moai-library-nextra/optimization.md +303 -0
  334. moai_adk/templates/.claude/skills/moai-library-nextra/reference.md +379 -0
  335. moai_adk/templates/.claude/skills/moai-library-shadcn/SKILL.md +372 -0
  336. moai_adk/templates/.claude/skills/moai-library-shadcn/examples.md +575 -0
  337. moai_adk/templates/.claude/skills/moai-library-shadcn/modules/advanced-patterns.md +394 -0
  338. moai_adk/templates/.claude/skills/moai-library-shadcn/modules/optimization.md +278 -0
  339. moai_adk/templates/.claude/skills/moai-library-shadcn/modules/shadcn-components.md +457 -0
  340. moai_adk/templates/.claude/skills/moai-library-shadcn/modules/shadcn-theming.md +373 -0
  341. moai_adk/templates/.claude/skills/moai-library-shadcn/reference.md +74 -0
  342. moai_adk/templates/.claude/skills/moai-mcp-figma/SKILL.md +402 -0
  343. moai_adk/templates/.claude/skills/moai-mcp-figma/advanced-patterns.md +607 -0
  344. moai_adk/templates/.claude/skills/moai-mcp-notion/SKILL.md +300 -0
  345. moai_adk/templates/.claude/skills/moai-mcp-notion/advanced-patterns.md +537 -0
  346. moai_adk/templates/.claude/skills/moai-platform-auth0/SKILL.md +291 -0
  347. moai_adk/templates/.claude/skills/moai-platform-clerk/SKILL.md +390 -0
  348. moai_adk/templates/.claude/skills/moai-platform-convex/SKILL.md +398 -0
  349. moai_adk/templates/.claude/skills/moai-platform-firebase-auth/SKILL.md +379 -0
  350. moai_adk/templates/.claude/skills/moai-platform-firestore/SKILL.md +358 -0
  351. moai_adk/templates/.claude/skills/moai-platform-neon/SKILL.md +467 -0
  352. moai_adk/templates/.claude/skills/moai-platform-railway/SKILL.md +377 -0
  353. moai_adk/templates/.claude/skills/moai-platform-supabase/SKILL.md +466 -0
  354. moai_adk/templates/.claude/skills/moai-platform-vercel/SKILL.md +482 -0
  355. moai_adk/templates/.claude/skills/moai-plugin-builder/SKILL.md +474 -0
  356. moai_adk/templates/.claude/skills/moai-plugin-builder/examples.md +621 -0
  357. moai_adk/templates/.claude/skills/moai-plugin-builder/migration.md +341 -0
  358. moai_adk/templates/.claude/skills/moai-plugin-builder/reference.md +463 -0
  359. moai_adk/templates/.claude/skills/moai-plugin-builder/validation.md +373 -0
  360. moai_adk/templates/.claude/skills/moai-security-auth0/SKILL.md +275 -0
  361. moai_adk/templates/.claude/skills/moai-security-auth0/modules/adaptive-mfa.md +233 -0
  362. moai_adk/templates/.claude/skills/moai-security-auth0/modules/akamai-integration.md +215 -0
  363. moai_adk/templates/.claude/skills/moai-security-auth0/modules/application-credentials.md +280 -0
  364. moai_adk/templates/.claude/skills/moai-security-auth0/modules/attack-protection-log-events.md +225 -0
  365. moai_adk/templates/.claude/skills/moai-security-auth0/modules/attack-protection-overview.md +140 -0
  366. moai_adk/templates/.claude/skills/moai-security-auth0/modules/bot-detection.md +144 -0
  367. moai_adk/templates/.claude/skills/moai-security-auth0/modules/breached-password-detection.md +187 -0
  368. moai_adk/templates/.claude/skills/moai-security-auth0/modules/brute-force-protection.md +189 -0
  369. moai_adk/templates/.claude/skills/moai-security-auth0/modules/certifications.md +282 -0
  370. moai_adk/templates/.claude/skills/moai-security-auth0/modules/compliance-overview.md +263 -0
  371. moai_adk/templates/.claude/skills/moai-security-auth0/modules/continuous-session-protection.md +307 -0
  372. moai_adk/templates/.claude/skills/moai-security-auth0/modules/customize-mfa.md +178 -0
  373. moai_adk/templates/.claude/skills/moai-security-auth0/modules/dpop-implementation.md +283 -0
  374. moai_adk/templates/.claude/skills/moai-security-auth0/modules/fapi-implementation.md +259 -0
  375. moai_adk/templates/.claude/skills/moai-security-auth0/modules/gdpr-compliance.md +313 -0
  376. moai_adk/templates/.claude/skills/moai-security-auth0/modules/guardian-configuration.md +269 -0
  377. moai_adk/templates/.claude/skills/moai-security-auth0/modules/highly-regulated-identity.md +272 -0
  378. moai_adk/templates/.claude/skills/moai-security-auth0/modules/jwt-fundamentals.md +248 -0
  379. moai_adk/templates/.claude/skills/moai-security-auth0/modules/mdl-verification.md +211 -0
  380. moai_adk/templates/.claude/skills/moai-security-auth0/modules/mfa-api-management.md +278 -0
  381. moai_adk/templates/.claude/skills/moai-security-auth0/modules/mfa-factors.md +226 -0
  382. moai_adk/templates/.claude/skills/moai-security-auth0/modules/mfa-overview.md +174 -0
  383. moai_adk/templates/.claude/skills/moai-security-auth0/modules/mtls-sender-constraining.md +316 -0
  384. moai_adk/templates/.claude/skills/moai-security-auth0/modules/ropg-flow-mfa.md +217 -0
  385. moai_adk/templates/.claude/skills/moai-security-auth0/modules/security-center.md +325 -0
  386. moai_adk/templates/.claude/skills/moai-security-auth0/modules/security-guidance.md +277 -0
  387. moai_adk/templates/.claude/skills/moai-security-auth0/modules/state-parameters.md +178 -0
  388. moai_adk/templates/.claude/skills/moai-security-auth0/modules/step-up-authentication.md +251 -0
  389. moai_adk/templates/.claude/skills/moai-security-auth0/modules/suspicious-ip-throttling.md +240 -0
  390. moai_adk/templates/.claude/skills/moai-security-auth0/modules/tenant-access-control.md +180 -0
  391. moai_adk/templates/.claude/skills/moai-security-auth0/modules/webauthn-fido.md +235 -0
  392. moai_adk/templates/.claude/skills/moai-workflow-jit-docs/SKILL.md +449 -0
  393. moai_adk/templates/.claude/skills/moai-workflow-jit-docs/advanced-patterns.md +379 -0
  394. moai_adk/templates/.claude/skills/moai-workflow-jit-docs/examples.md +544 -0
  395. moai_adk/templates/.claude/skills/moai-workflow-jit-docs/optimization.md +286 -0
  396. moai_adk/templates/.claude/skills/moai-workflow-jit-docs/reference.md +307 -0
  397. moai_adk/templates/.claude/skills/moai-workflow-project/README.md +190 -0
  398. moai_adk/templates/.claude/skills/moai-workflow-project/SKILL.md +390 -0
  399. moai_adk/templates/.claude/skills/moai-workflow-project/__init__.py +520 -0
  400. moai_adk/templates/.claude/skills/moai-workflow-project/complete_workflow_demo_fixed.py +574 -0
  401. moai_adk/templates/.claude/skills/moai-workflow-project/examples/complete_project_setup.py +317 -0
  402. moai_adk/templates/.claude/skills/moai-workflow-project/examples/complete_workflow_demo.py +663 -0
  403. moai_adk/templates/.claude/skills/moai-workflow-project/examples/config-migration-example.json +190 -0
  404. moai_adk/templates/.claude/skills/moai-workflow-project/examples/question-examples.json +175 -0
  405. moai_adk/templates/.claude/skills/moai-workflow-project/examples/quick_start.py +196 -0
  406. moai_adk/templates/.claude/skills/moai-workflow-project/examples.md +547 -0
  407. moai_adk/templates/.claude/skills/moai-workflow-project/modules/__init__.py +17 -0
  408. moai_adk/templates/.claude/skills/moai-workflow-project/modules/advanced-patterns.md +158 -0
  409. moai_adk/templates/.claude/skills/moai-workflow-project/modules/ask_user_integration.py +340 -0
  410. moai_adk/templates/.claude/skills/moai-workflow-project/modules/batch_questions.py +713 -0
  411. moai_adk/templates/.claude/skills/moai-workflow-project/modules/config_manager.py +538 -0
  412. moai_adk/templates/.claude/skills/moai-workflow-project/modules/documentation_manager.py +1336 -0
  413. moai_adk/templates/.claude/skills/moai-workflow-project/modules/language_initializer.py +730 -0
  414. moai_adk/templates/.claude/skills/moai-workflow-project/modules/migration_manager.py +608 -0
  415. moai_adk/templates/.claude/skills/moai-workflow-project/modules/template_optimizer.py +1005 -0
  416. moai_adk/templates/.claude/skills/moai-workflow-project/reference.md +275 -0
  417. moai_adk/templates/.claude/skills/moai-workflow-project/schemas/config-schema.json +316 -0
  418. moai_adk/templates/.claude/skills/moai-workflow-project/schemas/tab_schema.json +1434 -0
  419. moai_adk/templates/.claude/skills/moai-workflow-project/templates/config-template.json +71 -0
  420. moai_adk/templates/.claude/skills/moai-workflow-project/templates/doc-templates/product-template.md +44 -0
  421. moai_adk/templates/.claude/skills/moai-workflow-project/templates/doc-templates/structure-template.md +48 -0
  422. moai_adk/templates/.claude/skills/moai-workflow-project/templates/doc-templates/tech-template.md +92 -0
  423. moai_adk/templates/.claude/skills/moai-workflow-project/templates/question-templates/config-manager-setup.json +109 -0
  424. moai_adk/templates/.claude/skills/moai-workflow-project/templates/question-templates/language-initializer.json +228 -0
  425. moai_adk/templates/.claude/skills/moai-workflow-project/templates/question-templates/menu-project-config.json +130 -0
  426. moai_adk/templates/.claude/skills/moai-workflow-project/templates/question-templates/project-batch-questions.json +97 -0
  427. moai_adk/templates/.claude/skills/moai-workflow-project/templates/question-templates/spec-workflow-setup.json +150 -0
  428. moai_adk/templates/.claude/skills/moai-workflow-project/test_integration_simple.py +436 -0
  429. moai_adk/templates/.claude/skills/moai-workflow-spec/SKILL.md +534 -0
  430. moai_adk/templates/.claude/skills/moai-workflow-spec/examples.md +900 -0
  431. moai_adk/templates/.claude/skills/moai-workflow-spec/reference.md +704 -0
  432. moai_adk/templates/.claude/skills/moai-workflow-templates/SKILL.md +377 -0
  433. moai_adk/templates/.claude/skills/moai-workflow-templates/examples.md +552 -0
  434. moai_adk/templates/.claude/skills/moai-workflow-templates/modules/code-templates.md +124 -0
  435. moai_adk/templates/.claude/skills/moai-workflow-templates/modules/feedback-templates.md +100 -0
  436. moai_adk/templates/.claude/skills/moai-workflow-templates/modules/template-optimizer.md +138 -0
  437. moai_adk/templates/.claude/skills/moai-workflow-templates/reference.md +346 -0
  438. moai_adk/templates/.claude/skills/moai-workflow-testing/LICENSE.txt +202 -0
  439. moai_adk/templates/.claude/skills/moai-workflow-testing/SKILL.md +456 -0
  440. moai_adk/templates/.claude/skills/moai-workflow-testing/advanced-patterns.md +576 -0
  441. moai_adk/templates/.claude/skills/moai-workflow-testing/examples/ai-powered-testing.py +294 -0
  442. moai_adk/templates/.claude/skills/moai-workflow-testing/examples/console_logging.py +35 -0
  443. moai_adk/templates/.claude/skills/moai-workflow-testing/examples/element_discovery.py +40 -0
  444. moai_adk/templates/.claude/skills/moai-workflow-testing/examples/static_html_automation.py +34 -0
  445. moai_adk/templates/.claude/skills/moai-workflow-testing/examples.md +672 -0
  446. moai_adk/templates/.claude/skills/moai-workflow-testing/modules/README.md +220 -0
  447. moai_adk/templates/.claude/skills/moai-workflow-testing/modules/ai-debugging.md +845 -0
  448. moai_adk/templates/.claude/skills/moai-workflow-testing/modules/automated-code-review.md +1416 -0
  449. moai_adk/templates/.claude/skills/moai-workflow-testing/modules/performance-optimization.md +1234 -0
  450. moai_adk/templates/.claude/skills/moai-workflow-testing/modules/smart-refactoring.md +1243 -0
  451. moai_adk/templates/.claude/skills/moai-workflow-testing/modules/tdd-context7.md +1260 -0
  452. moai_adk/templates/.claude/skills/moai-workflow-testing/optimization.md +505 -0
  453. moai_adk/templates/.claude/skills/moai-workflow-testing/reference/playwright-best-practices.md +57 -0
  454. moai_adk/templates/.claude/skills/moai-workflow-testing/reference.md +440 -0
  455. moai_adk/templates/.claude/skills/moai-workflow-testing/scripts/with_server.py +218 -0
  456. moai_adk/templates/.claude/skills/moai-workflow-testing/templates/alfred-integration.md +376 -0
  457. moai_adk/templates/.claude/skills/moai-workflow-testing/workflows/enterprise-testing-workflow.py +571 -0
  458. moai_adk/templates/.claude/skills/moai-worktree/SKILL.md +411 -0
  459. moai_adk/templates/.claude/skills/moai-worktree/examples.md +606 -0
  460. moai_adk/templates/.claude/skills/moai-worktree/modules/integration-patterns.md +982 -0
  461. moai_adk/templates/.claude/skills/moai-worktree/modules/parallel-development.md +778 -0
  462. moai_adk/templates/.claude/skills/moai-worktree/modules/worktree-commands.md +646 -0
  463. moai_adk/templates/.claude/skills/moai-worktree/modules/worktree-management.md +782 -0
  464. moai_adk/templates/.claude/skills/moai-worktree/reference.md +357 -0
  465. moai_adk/templates/.git-hooks/pre-commit +128 -0
  466. moai_adk/templates/.git-hooks/pre-push +365 -0
  467. moai_adk/templates/.github/workflows/ci-universal.yml +513 -0
  468. moai_adk/templates/.github/workflows/security-secrets-check.yml +179 -0
  469. moai_adk/templates/.github/workflows/spec-issue-sync.yml +337 -0
  470. moai_adk/templates/.gitignore +222 -0
  471. moai_adk/templates/.mcp.json +13 -0
  472. moai_adk/templates/.moai/config/config.yaml +58 -0
  473. moai_adk/templates/.moai/config/questions/_schema.yaml +174 -0
  474. moai_adk/templates/.moai/config/questions/tab0-init.yaml +251 -0
  475. moai_adk/templates/.moai/config/questions/tab1-user.yaml +107 -0
  476. moai_adk/templates/.moai/config/questions/tab2-project.yaml +79 -0
  477. moai_adk/templates/.moai/config/questions/tab3-git.yaml +632 -0
  478. moai_adk/templates/.moai/config/questions/tab4-quality.yaml +182 -0
  479. moai_adk/templates/.moai/config/questions/tab5-system.yaml +96 -0
  480. moai_adk/templates/.moai/config/sections/git-strategy.yaml +116 -0
  481. moai_adk/templates/.moai/config/sections/language.yaml +11 -0
  482. moai_adk/templates/.moai/config/sections/project.yaml +13 -0
  483. moai_adk/templates/.moai/config/sections/quality.yaml +17 -0
  484. moai_adk/templates/.moai/config/sections/system.yaml +24 -0
  485. moai_adk/templates/.moai/config/sections/user.yaml +5 -0
  486. moai_adk/templates/.moai/config/statusline-config.yaml +92 -0
  487. moai_adk/templates/.moai/scripts/setup-glm.py +136 -0
  488. moai_adk/templates/CLAUDE.md +642 -0
  489. moai_adk/utils/__init__.py +30 -0
  490. moai_adk/utils/banner.py +38 -0
  491. moai_adk/utils/common.py +294 -0
  492. moai_adk/utils/link_validator.py +241 -0
  493. moai_adk/utils/logger.py +147 -0
  494. moai_adk/utils/safe_file_reader.py +206 -0
  495. moai_adk/utils/timeout.py +160 -0
  496. moai_adk/utils/toon_utils.py +256 -0
  497. moai_adk/version.py +22 -0
  498. moai_adk-0.35.1.dist-info/METADATA +3018 -0
  499. moai_adk-0.35.1.dist-info/RECORD +502 -0
  500. moai_adk-0.35.1.dist-info/WHEEL +4 -0
  501. moai_adk-0.35.1.dist-info/entry_points.txt +3 -0
  502. moai_adk-0.35.1.dist-info/licenses/LICENSE +21 -0
@@ -0,0 +1,1089 @@
1
+ # TypeScript Production-Ready Examples
2
+
3
+ ## Full-Stack Application Setup
4
+
5
+ ### Next.js 16 + tRPC + Prisma
6
+
7
+ ```
8
+ my-app/
9
+ ├── src/
10
+ │ ├── app/
11
+ │ │ ├── (auth)/
12
+ │ │ │ ├── login/
13
+ │ │ │ │ └── page.tsx
14
+ │ │ │ └── register/
15
+ │ │ │ └── page.tsx
16
+ │ │ ├── (dashboard)/
17
+ │ │ │ ├── layout.tsx
18
+ │ │ │ └── page.tsx
19
+ │ │ ├── api/
20
+ │ │ │ ├── auth/[...nextauth]/
21
+ │ │ │ │ └── route.ts
22
+ │ │ │ └── trpc/[trpc]/
23
+ │ │ │ └── route.ts
24
+ │ │ ├── layout.tsx
25
+ │ │ ├── page.tsx
26
+ │ │ └── providers.tsx
27
+ │ ├── components/
28
+ │ │ ├── ui/
29
+ │ │ └── features/
30
+ │ ├── lib/
31
+ │ │ ├── db.ts
32
+ │ │ ├── auth.ts
33
+ │ │ └── trpc.ts
34
+ │ └── server/
35
+ │ ├── routers/
36
+ │ │ ├── user.ts
37
+ │ │ ├── post.ts
38
+ │ │ └── _app.ts
39
+ │ ├── context.ts
40
+ │ └── trpc.ts
41
+ ├── prisma/
42
+ │ └── schema.prisma
43
+ ├── package.json
44
+ └── tsconfig.json
45
+ ```
46
+
47
+ ### package.json
48
+
49
+ ```json
50
+ {
51
+ "name": "my-app",
52
+ "version": "1.0.0",
53
+ "scripts": {
54
+ "dev": "next dev --turbo",
55
+ "build": "next build",
56
+ "start": "next start",
57
+ "lint": "next lint",
58
+ "test": "vitest",
59
+ "test:e2e": "playwright test",
60
+ "db:push": "prisma db push",
61
+ "db:studio": "prisma studio",
62
+ "typecheck": "tsc --noEmit"
63
+ },
64
+ "dependencies": {
65
+ "next": "^16.0.0",
66
+ "react": "^19.0.0",
67
+ "react-dom": "^19.0.0",
68
+ "@trpc/server": "^11.0.0",
69
+ "@trpc/client": "^11.0.0",
70
+ "@trpc/react-query": "^11.0.0",
71
+ "@tanstack/react-query": "^5.59.0",
72
+ "@prisma/client": "^6.0.0",
73
+ "next-auth": "^5.0.0",
74
+ "zod": "^3.23.8",
75
+ "zustand": "^5.0.0",
76
+ "superjson": "^2.2.1"
77
+ },
78
+ "devDependencies": {
79
+ "typescript": "^5.9.0",
80
+ "@types/react": "^19.0.0",
81
+ "@types/node": "^22.0.0",
82
+ "prisma": "^6.0.0",
83
+ "vitest": "^2.1.0",
84
+ "@testing-library/react": "^16.0.0",
85
+ "@playwright/test": "^1.48.0",
86
+ "eslint": "^9.0.0",
87
+ "eslint-config-next": "^16.0.0"
88
+ }
89
+ }
90
+ ```
91
+
92
+ ### tsconfig.json
93
+
94
+ ```json
95
+ {
96
+ "compilerOptions": {
97
+ "target": "ES2022",
98
+ "lib": ["dom", "dom.iterable", "ES2022"],
99
+ "allowJs": true,
100
+ "skipLibCheck": true,
101
+ "strict": true,
102
+ "noEmit": true,
103
+ "esModuleInterop": true,
104
+ "module": "esnext",
105
+ "moduleResolution": "bundler",
106
+ "resolveJsonModule": true,
107
+ "isolatedModules": true,
108
+ "jsx": "preserve",
109
+ "incremental": true,
110
+ "plugins": [{ "name": "next" }],
111
+ "paths": {
112
+ "@/*": ["./src/*"]
113
+ }
114
+ },
115
+ "include": ["next-env.d.ts", "**/*.ts", "**/*.tsx", ".next/types/**/*.ts"],
116
+ "exclude": ["node_modules"]
117
+ }
118
+ ```
119
+
120
+ ---
121
+
122
+ ## Database Layer (Prisma)
123
+
124
+ ### prisma/schema.prisma
125
+
126
+ ```prisma
127
+ generator client {
128
+ provider = "prisma-client-js"
129
+ }
130
+
131
+ datasource db {
132
+ provider = "postgresql"
133
+ url = env("DATABASE_URL")
134
+ }
135
+
136
+ model User {
137
+ id String @id @default(cuid())
138
+ email String @unique
139
+ name String?
140
+ emailVerified DateTime?
141
+ image String?
142
+ password String?
143
+ role Role @default(USER)
144
+ posts Post[]
145
+ accounts Account[]
146
+ sessions Session[]
147
+ createdAt DateTime @default(now())
148
+ updatedAt DateTime @updatedAt
149
+
150
+ @@index([email])
151
+ }
152
+
153
+ model Post {
154
+ id String @id @default(cuid())
155
+ title String
156
+ content String?
157
+ published Boolean @default(false)
158
+ authorId String
159
+ author User @relation(fields: [authorId], references: [id], onDelete: Cascade)
160
+ tags Tag[]
161
+ createdAt DateTime @default(now())
162
+ updatedAt DateTime @updatedAt
163
+
164
+ @@index([authorId])
165
+ @@index([published])
166
+ }
167
+
168
+ model Tag {
169
+ id String @id @default(cuid())
170
+ name String @unique
171
+ posts Post[]
172
+ }
173
+
174
+ enum Role {
175
+ USER
176
+ ADMIN
177
+ }
178
+
179
+ // NextAuth models
180
+ model Account {
181
+ id String @id @default(cuid())
182
+ userId String
183
+ type String
184
+ provider String
185
+ providerAccountId String
186
+ refresh_token String?
187
+ access_token String?
188
+ expires_at Int?
189
+ token_type String?
190
+ scope String?
191
+ id_token String?
192
+ session_state String?
193
+ user User @relation(fields: [userId], references: [id], onDelete: Cascade)
194
+
195
+ @@unique([provider, providerAccountId])
196
+ }
197
+
198
+ model Session {
199
+ id String @id @default(cuid())
200
+ sessionToken String @unique
201
+ userId String
202
+ expires DateTime
203
+ user User @relation(fields: [userId], references: [id], onDelete: Cascade)
204
+ }
205
+ ```
206
+
207
+ ### src/lib/db.ts
208
+
209
+ ```typescript
210
+ import { PrismaClient } from "@prisma/client";
211
+
212
+ const globalForPrisma = globalThis as unknown as {
213
+ prisma: PrismaClient | undefined;
214
+ };
215
+
216
+ export const db =
217
+ globalForPrisma.prisma ??
218
+ new PrismaClient({
219
+ log: process.env.NODE_ENV === "development" ? ["query", "error", "warn"] : ["error"],
220
+ });
221
+
222
+ if (process.env.NODE_ENV !== "production") {
223
+ globalForPrisma.prisma = db;
224
+ }
225
+ ```
226
+
227
+ ---
228
+
229
+ ## tRPC Setup
230
+
231
+ ### src/server/trpc.ts
232
+
233
+ ```typescript
234
+ import { initTRPC, TRPCError } from "@trpc/server";
235
+ import superjson from "superjson";
236
+ import { ZodError } from "zod";
237
+ import type { Context } from "./context";
238
+
239
+ const t = initTRPC.context<Context>().create({
240
+ transformer: superjson,
241
+ errorFormatter({ shape, error }) {
242
+ return {
243
+ ...shape,
244
+ data: {
245
+ ...shape.data,
246
+ zodError:
247
+ error.cause instanceof ZodError ? error.cause.flatten() : null,
248
+ },
249
+ };
250
+ },
251
+ });
252
+
253
+ export const router = t.router;
254
+ export const publicProcedure = t.procedure;
255
+
256
+ export const protectedProcedure = t.procedure.use(async ({ ctx, next }) => {
257
+ if (!ctx.session?.user) {
258
+ throw new TRPCError({ code: "UNAUTHORIZED" });
259
+ }
260
+ return next({
261
+ ctx: {
262
+ ...ctx,
263
+ user: ctx.session.user,
264
+ },
265
+ });
266
+ });
267
+
268
+ // Admin-only procedure
269
+ export const adminProcedure = protectedProcedure.use(async ({ ctx, next }) => {
270
+ if (ctx.user.role !== "ADMIN") {
271
+ throw new TRPCError({ code: "FORBIDDEN" });
272
+ }
273
+ return next({ ctx });
274
+ });
275
+ ```
276
+
277
+ ### src/server/context.ts
278
+
279
+ ```typescript
280
+ import { getServerSession } from "next-auth";
281
+ import { authOptions } from "@/lib/auth";
282
+ import { db } from "@/lib/db";
283
+ import type { inferAsyncReturnType } from "@trpc/server";
284
+ import type { FetchCreateContextFnOptions } from "@trpc/server/adapters/fetch";
285
+
286
+ export async function createContext(opts: FetchCreateContextFnOptions) {
287
+ const session = await getServerSession(authOptions);
288
+
289
+ return {
290
+ db,
291
+ session,
292
+ headers: opts.req.headers,
293
+ };
294
+ }
295
+
296
+ export type Context = inferAsyncReturnType<typeof createContext>;
297
+ ```
298
+
299
+ ### src/server/routers/user.ts
300
+
301
+ ```typescript
302
+ import { z } from "zod";
303
+ import { router, publicProcedure, protectedProcedure, adminProcedure } from "../trpc";
304
+ import { TRPCError } from "@trpc/server";
305
+ import bcrypt from "bcryptjs";
306
+
307
+ const UserSchema = z.object({
308
+ id: z.string(),
309
+ email: z.string().email(),
310
+ name: z.string().nullable(),
311
+ role: z.enum(["USER", "ADMIN"]),
312
+ createdAt: z.date(),
313
+ });
314
+
315
+ const CreateUserSchema = z.object({
316
+ email: z.string().email(),
317
+ name: z.string().min(2).max(100),
318
+ password: z.string().min(8),
319
+ });
320
+
321
+ const UpdateUserSchema = z.object({
322
+ name: z.string().min(2).max(100).optional(),
323
+ email: z.string().email().optional(),
324
+ });
325
+
326
+ export const userRouter = router({
327
+ // Public: Get user by ID
328
+ getById: publicProcedure
329
+ .input(z.object({ id: z.string() }))
330
+ .query(async ({ input, ctx }) => {
331
+ const user = await ctx.db.user.findUnique({
332
+ where: { id: input.id },
333
+ select: {
334
+ id: true,
335
+ email: true,
336
+ name: true,
337
+ role: true,
338
+ createdAt: true,
339
+ },
340
+ });
341
+
342
+ if (!user) {
343
+ throw new TRPCError({
344
+ code: "NOT_FOUND",
345
+ message: "User not found",
346
+ });
347
+ }
348
+
349
+ return user;
350
+ }),
351
+
352
+ // Protected: Get current user profile
353
+ me: protectedProcedure.query(async ({ ctx }) => {
354
+ return ctx.db.user.findUnique({
355
+ where: { id: ctx.user.id },
356
+ select: {
357
+ id: true,
358
+ email: true,
359
+ name: true,
360
+ role: true,
361
+ createdAt: true,
362
+ _count: { select: { posts: true } },
363
+ },
364
+ });
365
+ }),
366
+
367
+ // Protected: Update current user
368
+ update: protectedProcedure
369
+ .input(UpdateUserSchema)
370
+ .mutation(async ({ input, ctx }) => {
371
+ return ctx.db.user.update({
372
+ where: { id: ctx.user.id },
373
+ data: input,
374
+ });
375
+ }),
376
+
377
+ // Admin: List all users with pagination
378
+ list: adminProcedure
379
+ .input(
380
+ z.object({
381
+ page: z.number().min(1).default(1),
382
+ limit: z.number().min(1).max(100).default(10),
383
+ search: z.string().optional(),
384
+ })
385
+ )
386
+ .query(async ({ input, ctx }) => {
387
+ const { page, limit, search } = input;
388
+ const skip = (page - 1) * limit;
389
+
390
+ const where = search
391
+ ? {
392
+ OR: [
393
+ { name: { contains: search, mode: "insensitive" as const } },
394
+ { email: { contains: search, mode: "insensitive" as const } },
395
+ ],
396
+ }
397
+ : {};
398
+
399
+ const [users, total] = await Promise.all([
400
+ ctx.db.user.findMany({
401
+ where,
402
+ skip,
403
+ take: limit,
404
+ orderBy: { createdAt: "desc" },
405
+ select: {
406
+ id: true,
407
+ email: true,
408
+ name: true,
409
+ role: true,
410
+ createdAt: true,
411
+ },
412
+ }),
413
+ ctx.db.user.count({ where }),
414
+ ]);
415
+
416
+ return {
417
+ users,
418
+ pagination: {
419
+ page,
420
+ limit,
421
+ total,
422
+ totalPages: Math.ceil(total / limit),
423
+ },
424
+ };
425
+ }),
426
+
427
+ // Admin: Delete user
428
+ delete: adminProcedure
429
+ .input(z.object({ id: z.string() }))
430
+ .mutation(async ({ input, ctx }) => {
431
+ if (input.id === ctx.user.id) {
432
+ throw new TRPCError({
433
+ code: "BAD_REQUEST",
434
+ message: "Cannot delete your own account",
435
+ });
436
+ }
437
+
438
+ return ctx.db.user.delete({ where: { id: input.id } });
439
+ }),
440
+ });
441
+ ```
442
+
443
+ ### src/server/routers/post.ts
444
+
445
+ ```typescript
446
+ import { z } from "zod";
447
+ import { router, publicProcedure, protectedProcedure } from "../trpc";
448
+ import { TRPCError } from "@trpc/server";
449
+
450
+ const CreatePostSchema = z.object({
451
+ title: z.string().min(1).max(200),
452
+ content: z.string().optional(),
453
+ tags: z.array(z.string()).optional(),
454
+ });
455
+
456
+ const UpdatePostSchema = CreatePostSchema.partial().extend({
457
+ id: z.string(),
458
+ published: z.boolean().optional(),
459
+ });
460
+
461
+ export const postRouter = router({
462
+ // Public: List published posts
463
+ list: publicProcedure
464
+ .input(
465
+ z.object({
466
+ page: z.number().min(1).default(1),
467
+ limit: z.number().min(1).max(50).default(10),
468
+ tag: z.string().optional(),
469
+ })
470
+ )
471
+ .query(async ({ input, ctx }) => {
472
+ const { page, limit, tag } = input;
473
+ const skip = (page - 1) * limit;
474
+
475
+ const where = {
476
+ published: true,
477
+ ...(tag && { tags: { some: { name: tag } } }),
478
+ };
479
+
480
+ const [posts, total] = await Promise.all([
481
+ ctx.db.post.findMany({
482
+ where,
483
+ skip,
484
+ take: limit,
485
+ orderBy: { createdAt: "desc" },
486
+ include: {
487
+ author: { select: { id: true, name: true } },
488
+ tags: { select: { name: true } },
489
+ },
490
+ }),
491
+ ctx.db.post.count({ where }),
492
+ ]);
493
+
494
+ return { posts, total, page, totalPages: Math.ceil(total / limit) };
495
+ }),
496
+
497
+ // Public: Get single post
498
+ getById: publicProcedure
499
+ .input(z.object({ id: z.string() }))
500
+ .query(async ({ input, ctx }) => {
501
+ const post = await ctx.db.post.findUnique({
502
+ where: { id: input.id },
503
+ include: {
504
+ author: { select: { id: true, name: true } },
505
+ tags: { select: { name: true } },
506
+ },
507
+ });
508
+
509
+ if (!post) {
510
+ throw new TRPCError({ code: "NOT_FOUND", message: "Post not found" });
511
+ }
512
+
513
+ // Only show unpublished to author
514
+ if (!post.published && post.authorId !== ctx.session?.user?.id) {
515
+ throw new TRPCError({ code: "NOT_FOUND", message: "Post not found" });
516
+ }
517
+
518
+ return post;
519
+ }),
520
+
521
+ // Protected: Create post
522
+ create: protectedProcedure
523
+ .input(CreatePostSchema)
524
+ .mutation(async ({ input, ctx }) => {
525
+ const { tags, ...data } = input;
526
+
527
+ return ctx.db.post.create({
528
+ data: {
529
+ ...data,
530
+ authorId: ctx.user.id,
531
+ tags: tags?.length
532
+ ? {
533
+ connectOrCreate: tags.map((name) => ({
534
+ where: { name },
535
+ create: { name },
536
+ })),
537
+ }
538
+ : undefined,
539
+ },
540
+ include: { tags: true },
541
+ });
542
+ }),
543
+
544
+ // Protected: Update own post
545
+ update: protectedProcedure
546
+ .input(UpdatePostSchema)
547
+ .mutation(async ({ input, ctx }) => {
548
+ const { id, tags, ...data } = input;
549
+
550
+ const post = await ctx.db.post.findUnique({ where: { id } });
551
+
552
+ if (!post) {
553
+ throw new TRPCError({ code: "NOT_FOUND" });
554
+ }
555
+
556
+ if (post.authorId !== ctx.user.id) {
557
+ throw new TRPCError({ code: "FORBIDDEN" });
558
+ }
559
+
560
+ return ctx.db.post.update({
561
+ where: { id },
562
+ data: {
563
+ ...data,
564
+ tags: tags
565
+ ? {
566
+ set: [],
567
+ connectOrCreate: tags.map((name) => ({
568
+ where: { name },
569
+ create: { name },
570
+ })),
571
+ }
572
+ : undefined,
573
+ },
574
+ include: { tags: true },
575
+ });
576
+ }),
577
+
578
+ // Protected: Delete own post
579
+ delete: protectedProcedure
580
+ .input(z.object({ id: z.string() }))
581
+ .mutation(async ({ input, ctx }) => {
582
+ const post = await ctx.db.post.findUnique({ where: { id: input.id } });
583
+
584
+ if (!post) {
585
+ throw new TRPCError({ code: "NOT_FOUND" });
586
+ }
587
+
588
+ if (post.authorId !== ctx.user.id) {
589
+ throw new TRPCError({ code: "FORBIDDEN" });
590
+ }
591
+
592
+ return ctx.db.post.delete({ where: { id: input.id } });
593
+ }),
594
+
595
+ // Protected: Get user's drafts
596
+ myDrafts: protectedProcedure.query(async ({ ctx }) => {
597
+ return ctx.db.post.findMany({
598
+ where: { authorId: ctx.user.id, published: false },
599
+ orderBy: { updatedAt: "desc" },
600
+ include: { tags: true },
601
+ });
602
+ }),
603
+ });
604
+ ```
605
+
606
+ ### src/server/routers/_app.ts
607
+
608
+ ```typescript
609
+ import { router } from "../trpc";
610
+ import { userRouter } from "./user";
611
+ import { postRouter } from "./post";
612
+
613
+ export const appRouter = router({
614
+ user: userRouter,
615
+ post: postRouter,
616
+ });
617
+
618
+ export type AppRouter = typeof appRouter;
619
+ ```
620
+
621
+ ---
622
+
623
+ ## Client Setup
624
+
625
+ ### src/lib/trpc.ts
626
+
627
+ ```typescript
628
+ import { createTRPCReact } from "@trpc/react-query";
629
+ import { httpBatchLink, loggerLink } from "@trpc/client";
630
+ import superjson from "superjson";
631
+ import type { AppRouter } from "@/server/routers/_app";
632
+
633
+ export const trpc = createTRPCReact<AppRouter>();
634
+
635
+ function getBaseUrl() {
636
+ if (typeof window !== "undefined") return "";
637
+ if (process.env.VERCEL_URL) return `https://${process.env.VERCEL_URL}`;
638
+ return `http://localhost:${process.env.PORT ?? 3000}`;
639
+ }
640
+
641
+ export const trpcClient = trpc.createClient({
642
+ links: [
643
+ loggerLink({
644
+ enabled: (opts) =>
645
+ process.env.NODE_ENV === "development" ||
646
+ (opts.direction === "down" && opts.result instanceof Error),
647
+ }),
648
+ httpBatchLink({
649
+ url: `${getBaseUrl()}/api/trpc`,
650
+ transformer: superjson,
651
+ }),
652
+ ],
653
+ });
654
+ ```
655
+
656
+ ### src/app/providers.tsx
657
+
658
+ ```typescript
659
+ "use client";
660
+
661
+ import { useState } from "react";
662
+ import { QueryClient, QueryClientProvider } from "@tanstack/react-query";
663
+ import { trpc, trpcClient } from "@/lib/trpc";
664
+ import { SessionProvider } from "next-auth/react";
665
+
666
+ export function Providers({ children }: { children: React.ReactNode }) {
667
+ const [queryClient] = useState(
668
+ () =>
669
+ new QueryClient({
670
+ defaultOptions: {
671
+ queries: {
672
+ staleTime: 60 * 1000,
673
+ refetchOnWindowFocus: false,
674
+ },
675
+ },
676
+ })
677
+ );
678
+
679
+ return (
680
+ <SessionProvider>
681
+ <trpc.Provider client={trpcClient} queryClient={queryClient}>
682
+ <QueryClientProvider client={queryClient}>
683
+ {children}
684
+ </QueryClientProvider>
685
+ </trpc.Provider>
686
+ </SessionProvider>
687
+ );
688
+ }
689
+ ```
690
+
691
+ ---
692
+
693
+ ## React Components
694
+
695
+ ### src/components/features/PostList.tsx
696
+
697
+ ```typescript
698
+ "use client";
699
+
700
+ import { trpc } from "@/lib/trpc";
701
+ import { useState } from "react";
702
+
703
+ interface PostListProps {
704
+ initialTag?: string;
705
+ }
706
+
707
+ export function PostList({ initialTag }: PostListProps) {
708
+ const [page, setPage] = useState(1);
709
+ const [tag, setTag] = useState(initialTag);
710
+
711
+ const { data, isLoading, error } = trpc.post.list.useQuery(
712
+ { page, limit: 10, tag },
713
+ { keepPreviousData: true }
714
+ );
715
+
716
+ if (isLoading) {
717
+ return <PostListSkeleton />;
718
+ }
719
+
720
+ if (error) {
721
+ return <div className="text-red-500">Error: {error.message}</div>;
722
+ }
723
+
724
+ return (
725
+ <div className="space-y-6">
726
+ <div className="grid gap-4">
727
+ {data?.posts.map((post) => (
728
+ <PostCard key={post.id} post={post} />
729
+ ))}
730
+ </div>
731
+
732
+ {data && data.totalPages > 1 && (
733
+ <Pagination
734
+ currentPage={page}
735
+ totalPages={data.totalPages}
736
+ onPageChange={setPage}
737
+ />
738
+ )}
739
+ </div>
740
+ );
741
+ }
742
+
743
+ function PostCard({ post }: { post: Post }) {
744
+ return (
745
+ <article className="p-6 bg-white rounded-lg shadow">
746
+ <h2 className="text-xl font-semibold">
747
+ <a href={`/posts/${post.id}`} className="hover:text-blue-600">
748
+ {post.title}
749
+ </a>
750
+ </h2>
751
+ <p className="mt-2 text-gray-600 line-clamp-2">{post.content}</p>
752
+ <div className="mt-4 flex items-center gap-4">
753
+ <span className="text-sm text-gray-500">By {post.author.name}</span>
754
+ <div className="flex gap-2">
755
+ {post.tags.map((tag) => (
756
+ <span
757
+ key={tag.name}
758
+ className="px-2 py-1 text-xs bg-gray-100 rounded"
759
+ >
760
+ {tag.name}
761
+ </span>
762
+ ))}
763
+ </div>
764
+ </div>
765
+ </article>
766
+ );
767
+ }
768
+ ```
769
+
770
+ ### src/components/features/CreatePostForm.tsx
771
+
772
+ ```typescript
773
+ "use client";
774
+
775
+ import { useRouter } from "next/navigation";
776
+ import { trpc } from "@/lib/trpc";
777
+ import { useForm } from "react-hook-form";
778
+ import { zodResolver } from "@hookform/resolvers/zod";
779
+ import { z } from "zod";
780
+
781
+ const formSchema = z.object({
782
+ title: z.string().min(1, "Title is required").max(200),
783
+ content: z.string().optional(),
784
+ tags: z.string().optional(),
785
+ });
786
+
787
+ type FormData = z.infer<typeof formSchema>;
788
+
789
+ export function CreatePostForm() {
790
+ const router = useRouter();
791
+ const utils = trpc.useUtils();
792
+
793
+ const {
794
+ register,
795
+ handleSubmit,
796
+ formState: { errors, isSubmitting },
797
+ } = useForm<FormData>({
798
+ resolver: zodResolver(formSchema),
799
+ });
800
+
801
+ const createPost = trpc.post.create.useMutation({
802
+ onSuccess: (post) => {
803
+ utils.post.list.invalidate();
804
+ router.push(`/posts/${post.id}`);
805
+ },
806
+ });
807
+
808
+ const onSubmit = async (data: FormData) => {
809
+ const tags = data.tags
810
+ ?.split(",")
811
+ .map((t) => t.trim())
812
+ .filter(Boolean);
813
+
814
+ await createPost.mutateAsync({
815
+ title: data.title,
816
+ content: data.content,
817
+ tags,
818
+ });
819
+ };
820
+
821
+ return (
822
+ <form onSubmit={handleSubmit(onSubmit)} className="space-y-6">
823
+ <div>
824
+ <label htmlFor="title" className="block text-sm font-medium">
825
+ Title
826
+ </label>
827
+ <input
828
+ id="title"
829
+ {...register("title")}
830
+ className="mt-1 block w-full rounded-md border p-2"
831
+ disabled={isSubmitting}
832
+ />
833
+ {errors.title && (
834
+ <p className="mt-1 text-sm text-red-500">{errors.title.message}</p>
835
+ )}
836
+ </div>
837
+
838
+ <div>
839
+ <label htmlFor="content" className="block text-sm font-medium">
840
+ Content
841
+ </label>
842
+ <textarea
843
+ id="content"
844
+ {...register("content")}
845
+ rows={10}
846
+ className="mt-1 block w-full rounded-md border p-2"
847
+ disabled={isSubmitting}
848
+ />
849
+ </div>
850
+
851
+ <div>
852
+ <label htmlFor="tags" className="block text-sm font-medium">
853
+ Tags (comma-separated)
854
+ </label>
855
+ <input
856
+ id="tags"
857
+ {...register("tags")}
858
+ placeholder="react, typescript, nextjs"
859
+ className="mt-1 block w-full rounded-md border p-2"
860
+ disabled={isSubmitting}
861
+ />
862
+ </div>
863
+
864
+ <button
865
+ type="submit"
866
+ disabled={isSubmitting}
867
+ className="px-4 py-2 bg-blue-600 text-white rounded-md hover:bg-blue-700 disabled:opacity-50"
868
+ >
869
+ {isSubmitting ? "Creating..." : "Create Post"}
870
+ </button>
871
+
872
+ {createPost.error && (
873
+ <p className="text-red-500">{createPost.error.message}</p>
874
+ )}
875
+ </form>
876
+ );
877
+ }
878
+ ```
879
+
880
+ ---
881
+
882
+ ## Testing Examples
883
+
884
+ ### vitest.config.ts
885
+
886
+ ```typescript
887
+ import { defineConfig } from "vitest/config";
888
+ import react from "@vitejs/plugin-react";
889
+ import path from "path";
890
+
891
+ export default defineConfig({
892
+ plugins: [react()],
893
+ test: {
894
+ environment: "jsdom",
895
+ setupFiles: ["./src/test/setup.ts"],
896
+ include: ["src/**/*.{test,spec}.{ts,tsx}"],
897
+ coverage: {
898
+ provider: "v8",
899
+ reporter: ["text", "json", "html"],
900
+ exclude: ["node_modules/", "src/test/"],
901
+ },
902
+ },
903
+ resolve: {
904
+ alias: {
905
+ "@": path.resolve(__dirname, "./src"),
906
+ },
907
+ },
908
+ });
909
+ ```
910
+
911
+ ### src/test/setup.ts
912
+
913
+ ```typescript
914
+ import "@testing-library/jest-dom/vitest";
915
+ import { cleanup } from "@testing-library/react";
916
+ import { afterEach, vi } from "vitest";
917
+
918
+ afterEach(() => {
919
+ cleanup();
920
+ });
921
+
922
+ // Mock next/navigation
923
+ vi.mock("next/navigation", () => ({
924
+ useRouter: () => ({
925
+ push: vi.fn(),
926
+ replace: vi.fn(),
927
+ back: vi.fn(),
928
+ }),
929
+ usePathname: () => "/",
930
+ useSearchParams: () => new URLSearchParams(),
931
+ }));
932
+ ```
933
+
934
+ ### src/components/features/__tests__/PostCard.test.tsx
935
+
936
+ ```typescript
937
+ import { describe, it, expect, vi } from "vitest";
938
+ import { render, screen } from "@testing-library/react";
939
+ import userEvent from "@testing-library/user-event";
940
+ import { PostCard } from "../PostCard";
941
+
942
+ const mockPost = {
943
+ id: "1",
944
+ title: "Test Post",
945
+ content: "This is test content",
946
+ published: true,
947
+ author: { id: "1", name: "John Doe" },
948
+ tags: [{ name: "react" }, { name: "typescript" }],
949
+ createdAt: new Date(),
950
+ };
951
+
952
+ describe("PostCard", () => {
953
+ it("renders post title and content", () => {
954
+ render(<PostCard post={mockPost} />);
955
+
956
+ expect(screen.getByText("Test Post")).toBeInTheDocument();
957
+ expect(screen.getByText("This is test content")).toBeInTheDocument();
958
+ });
959
+
960
+ it("displays author name", () => {
961
+ render(<PostCard post={mockPost} />);
962
+
963
+ expect(screen.getByText(/John Doe/)).toBeInTheDocument();
964
+ });
965
+
966
+ it("renders all tags", () => {
967
+ render(<PostCard post={mockPost} />);
968
+
969
+ expect(screen.getByText("react")).toBeInTheDocument();
970
+ expect(screen.getByText("typescript")).toBeInTheDocument();
971
+ });
972
+
973
+ it("links to post detail page", () => {
974
+ render(<PostCard post={mockPost} />);
975
+
976
+ const link = screen.getByRole("link", { name: /Test Post/i });
977
+ expect(link).toHaveAttribute("href", "/posts/1");
978
+ });
979
+ });
980
+ ```
981
+
982
+ ### E2E Test: playwright/posts.spec.ts
983
+
984
+ ```typescript
985
+ import { test, expect } from "@playwright/test";
986
+
987
+ test.describe("Posts", () => {
988
+ test.beforeEach(async ({ page }) => {
989
+ await page.goto("/");
990
+ });
991
+
992
+ test("should display list of posts", async ({ page }) => {
993
+ await expect(page.getByRole("article")).toHaveCount.above(0);
994
+ });
995
+
996
+ test("should navigate to post detail", async ({ page }) => {
997
+ const firstPost = page.getByRole("article").first();
998
+ const title = await firstPost.getByRole("heading").textContent();
999
+
1000
+ await firstPost.getByRole("link").click();
1001
+
1002
+ await expect(page).toHaveURL(/\/posts\/.+/);
1003
+ await expect(page.getByRole("heading", { level: 1 })).toHaveText(title!);
1004
+ });
1005
+
1006
+ test("should filter posts by tag", async ({ page }) => {
1007
+ await page.getByRole("button", { name: "react" }).click();
1008
+
1009
+ const posts = page.getByRole("article");
1010
+ for (const post of await posts.all()) {
1011
+ await expect(post.getByText("react")).toBeVisible();
1012
+ }
1013
+ });
1014
+ });
1015
+
1016
+ test.describe("Authenticated User", () => {
1017
+ test.use({ storageState: "playwright/.auth/user.json" });
1018
+
1019
+ test("should create new post", async ({ page }) => {
1020
+ await page.goto("/posts/new");
1021
+
1022
+ await page.getByLabel("Title").fill("My New Post");
1023
+ await page.getByLabel("Content").fill("This is my new post content.");
1024
+ await page.getByLabel("Tags").fill("test, e2e");
1025
+
1026
+ await page.getByRole("button", { name: "Create Post" }).click();
1027
+
1028
+ await expect(page).toHaveURL(/\/posts\/.+/);
1029
+ await expect(page.getByRole("heading", { level: 1 })).toHaveText(
1030
+ "My New Post"
1031
+ );
1032
+ });
1033
+ });
1034
+ ```
1035
+
1036
+ ---
1037
+
1038
+ ## Environment Configuration
1039
+
1040
+ ### .env.example
1041
+
1042
+ ```env
1043
+ # Database
1044
+ DATABASE_URL="postgresql://user:password@localhost:5432/myapp?schema=public"
1045
+
1046
+ # NextAuth
1047
+ NEXTAUTH_URL="http://localhost:3000"
1048
+ NEXTAUTH_SECRET="your-secret-key-min-32-chars"
1049
+
1050
+ # OAuth Providers
1051
+ GITHUB_ID=""
1052
+ GITHUB_SECRET=""
1053
+ GOOGLE_CLIENT_ID=""
1054
+ GOOGLE_CLIENT_SECRET=""
1055
+
1056
+ # App
1057
+ NEXT_PUBLIC_APP_URL="http://localhost:3000"
1058
+ ```
1059
+
1060
+ ### src/lib/env.ts
1061
+
1062
+ ```typescript
1063
+ import { z } from "zod";
1064
+
1065
+ const envSchema = z.object({
1066
+ DATABASE_URL: z.string().url(),
1067
+ NEXTAUTH_URL: z.string().url(),
1068
+ NEXTAUTH_SECRET: z.string().min(32),
1069
+ NODE_ENV: z.enum(["development", "production", "test"]).default("development"),
1070
+ NEXT_PUBLIC_APP_URL: z.string().url().optional(),
1071
+ });
1072
+
1073
+ const parsed = envSchema.safeParse(process.env);
1074
+
1075
+ if (!parsed.success) {
1076
+ console.error(
1077
+ "Invalid environment variables:",
1078
+ JSON.stringify(parsed.error.format(), null, 2)
1079
+ );
1080
+ throw new Error("Invalid environment variables");
1081
+ }
1082
+
1083
+ export const env = parsed.data;
1084
+ ```
1085
+
1086
+ ---
1087
+
1088
+ Version: 1.0.0
1089
+ Last Updated: 2025-12-07