cc-devflow 4.3.0 → 4.5.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 (579) hide show
  1. package/.claude/skills/cc-act/CHANGELOG.md +70 -0
  2. package/.claude/skills/cc-act/PLAYBOOK.md +212 -0
  3. package/.claude/skills/cc-act/SKILL.md +330 -0
  4. package/.claude/skills/cc-act/assets/PR_BRIEF_TEMPLATE.md +68 -0
  5. package/.claude/skills/cc-act/assets/RELEASE_NOTE_TEMPLATE.md +29 -0
  6. package/.claude/skills/cc-act/references/closure-contract.md +49 -0
  7. package/.claude/skills/cc-act/references/git-commit-guidelines.md +83 -0
  8. package/.claude/skills/cc-act/scripts/archive-requirement.sh +49 -0
  9. package/.claude/skills/cc-act/scripts/cc-act-common.sh +245 -0
  10. package/.claude/skills/cc-act/scripts/detect-ship-target.sh +82 -0
  11. package/.claude/skills/cc-act/scripts/generate-status-report.sh +83 -0
  12. package/.claude/skills/cc-act/scripts/render-pr-brief.sh +200 -0
  13. package/.claude/skills/cc-act/scripts/sync-act-docs.sh +328 -0
  14. package/.claude/skills/cc-act/scripts/verify-act-gate.sh +63 -0
  15. package/.claude/skills/cc-check/CHANGELOG.md +78 -0
  16. package/.claude/skills/cc-check/PLAYBOOK.md +153 -0
  17. package/.claude/skills/cc-check/SKILL.md +353 -0
  18. package/.claude/skills/cc-check/assets/REPORT_CARD_TEMPLATE.json +53 -0
  19. package/.claude/skills/cc-check/references/gate-contract.md +29 -0
  20. package/.claude/skills/cc-check/references/review-contract.md +45 -0
  21. package/.claude/skills/cc-check/scripts/render-report-card.js +229 -0
  22. package/.claude/skills/cc-check/scripts/run-quality-gates.sh +86 -0
  23. package/.claude/skills/cc-check/scripts/verify-gate.sh +57 -0
  24. package/.claude/skills/cc-do/CHANGELOG.md +54 -0
  25. package/.claude/skills/cc-do/PLAYBOOK.md +119 -0
  26. package/.claude/skills/cc-do/SKILL.md +224 -0
  27. package/.claude/skills/cc-do/references/execution-recovery.md +86 -0
  28. package/.claude/skills/cc-do/references/parallel-dispatch.md +80 -0
  29. package/.claude/skills/cc-do/scripts/build-task-context.sh +184 -0
  30. package/.claude/skills/cc-do/scripts/cc-do-common.sh +79 -0
  31. package/.claude/skills/cc-do/scripts/check-task-status.sh +92 -0
  32. package/.claude/skills/cc-do/scripts/detect-file-conflicts.sh +87 -0
  33. package/.claude/skills/cc-do/scripts/mark-task-complete.sh +119 -0
  34. package/.claude/skills/cc-do/scripts/record-review-decision.sh +89 -0
  35. package/.claude/skills/cc-do/scripts/recover-workflow.sh +84 -0
  36. package/.claude/skills/cc-do/scripts/run-problem-analysis.sh +70 -0
  37. package/.claude/skills/cc-do/scripts/select-ready-tasks.sh +135 -0
  38. package/.claude/skills/cc-do/scripts/verify-task-gates.sh +94 -0
  39. package/.claude/skills/cc-do/scripts/write-task-checkpoint.sh +102 -0
  40. package/.claude/skills/cc-investigate/CHANGELOG.md +31 -0
  41. package/.claude/skills/cc-investigate/PLAYBOOK.md +49 -0
  42. package/.claude/skills/cc-investigate/SKILL.md +229 -0
  43. package/.claude/skills/cc-investigate/assets/ANALYSIS_TEMPLATE.md +56 -0
  44. package/.claude/skills/cc-investigate/assets/TASKS_TEMPLATE.md +53 -0
  45. package/.claude/skills/cc-investigate/assets/TASK_MANIFEST_TEMPLATE.json +106 -0
  46. package/.claude/skills/cc-investigate/references/investigation-contract.md +28 -0
  47. package/.claude/skills/cc-investigate/scripts/bootstrap-analysis.sh +38 -0
  48. package/.claude/skills/cc-plan/CHANGELOG.md +104 -0
  49. package/.claude/skills/cc-plan/PLAYBOOK.md +106 -0
  50. package/.claude/skills/cc-plan/SKILL.md +333 -0
  51. package/.claude/skills/cc-plan/assets/DESIGN_TEMPLATE.md +156 -0
  52. package/.claude/skills/cc-plan/assets/TASKS_TEMPLATE.md +99 -0
  53. package/.claude/skills/cc-plan/assets/TASK_MANIFEST_TEMPLATE.json +111 -0
  54. package/.claude/skills/cc-plan/assets/TINY_DESIGN_TEMPLATE.md +80 -0
  55. package/.claude/skills/cc-plan/references/planning-contract.md +82 -0
  56. package/.claude/skills/cc-plan/scripts/bump-skill-version.sh +103 -0
  57. package/.claude/skills/cc-plan/scripts/parse-task-dependencies.js +75 -0
  58. package/.claude/skills/cc-plan/scripts/validate-scope.sh +78 -0
  59. package/.claude/skills/cc-roadmap/CHANGELOG.md +119 -0
  60. package/.claude/skills/cc-roadmap/PLAYBOOK.md +136 -0
  61. package/.claude/skills/cc-roadmap/SKILL.md +267 -0
  62. package/.claude/skills/cc-roadmap/assets/BACKLOG_TEMPLATE.md +46 -0
  63. package/.claude/skills/cc-roadmap/assets/ROADMAP_TEMPLATE.md +154 -0
  64. package/.claude/skills/cc-roadmap/assets/TRACKING_TEMPLATE.json +51 -0
  65. package/.claude/skills/cc-roadmap/references/roadmap-dialogue.md +37 -0
  66. package/.claude/skills/cc-roadmap/scripts/bump-skill-version.sh +103 -0
  67. package/.claude/skills/cc-roadmap/scripts/lib/roadmap-tracking/markdown.js +604 -0
  68. package/.claude/skills/cc-roadmap/scripts/lib/roadmap-tracking/query.js +63 -0
  69. package/.claude/skills/cc-roadmap/scripts/lib/roadmap-tracking/schema.js +245 -0
  70. package/.claude/skills/cc-roadmap/scripts/lib/roadmap-tracking/store.js +139 -0
  71. package/.claude/skills/cc-roadmap/scripts/locate-roadmap-item.sh +109 -0
  72. package/.claude/skills/cc-roadmap/scripts/roadmap-tracking.js +153 -0
  73. package/.claude/skills/cc-roadmap/scripts/sync-roadmap-progress.sh +100 -0
  74. package/.claude/skills/cc-simplify/CHANGELOG.md +10 -0
  75. package/.claude/skills/cc-simplify/SKILL.md +55 -0
  76. package/.claude/skills/cc-spec-init/CHANGELOG.md +12 -0
  77. package/.claude/skills/cc-spec-init/PLAYBOOK.md +55 -0
  78. package/.claude/skills/cc-spec-init/SKILL.md +135 -0
  79. package/.claude/skills/cc-spec-init/assets/CAPABILITY_TEMPLATE.md +64 -0
  80. package/.claude/skills/cc-spec-init/assets/CHANGE_META_TEMPLATE.json +28 -0
  81. package/.claude/skills/cc-spec-init/assets/INDEX_TEMPLATE.md +35 -0
  82. package/.claude/skills/cc-spec-init/references/spec-contract.md +22 -0
  83. package/.claude/skills/cc-spec-init/scripts/bootstrap-specs.sh +31 -0
  84. package/.claude/skills/cc-spec-init/scripts/validate-spec-links.sh +45 -0
  85. package/CHANGELOG.md +80 -4
  86. package/README.md +148 -741
  87. package/README.zh-CN.md +146 -740
  88. package/bin/adapt.js +2 -6
  89. package/bin/cc-devflow-cli.js +287 -166
  90. package/config/distributable-skills.json +24 -0
  91. package/config/schema/cc-devflow-config.schema.json +45 -0
  92. package/config/user-config.template.yml +16 -0
  93. package/docs/CLAUDE.md +10 -6
  94. package/docs/commands/README.md +19 -46
  95. package/docs/commands/README.zh-CN.md +25 -48
  96. package/docs/examples/BY-ARTIFACT.md +7 -0
  97. package/docs/examples/README.md +38 -0
  98. package/docs/examples/START-HERE.md +149 -0
  99. package/docs/examples/example-bindings.json +38 -0
  100. package/docs/examples/full-design-blocked/BACKLOG.md +45 -0
  101. package/docs/examples/full-design-blocked/README.md +55 -0
  102. package/docs/examples/full-design-blocked/ROADMAP.md +53 -0
  103. package/docs/examples/full-design-blocked/changes/REQ-002-bulk-invite-import/planning/design.md +154 -0
  104. package/docs/examples/full-design-blocked/changes/REQ-002-bulk-invite-import/planning/task-manifest.json +146 -0
  105. package/docs/examples/full-design-blocked/changes/REQ-002-bulk-invite-import/planning/tasks.md +78 -0
  106. package/docs/examples/full-design-blocked/changes/REQ-002-bulk-invite-import/review/report-card.json +52 -0
  107. package/docs/examples/full-design-blocked/roadmap-tracking.json +50 -0
  108. package/docs/examples/local-handoff/BACKLOG.md +45 -0
  109. package/docs/examples/local-handoff/README.md +56 -0
  110. package/docs/examples/local-handoff/ROADMAP.md +45 -0
  111. package/docs/examples/local-handoff/changes/REQ-003-audit-log-export/handoff/resume-index.md +39 -0
  112. package/docs/examples/local-handoff/changes/REQ-003-audit-log-export/handoff/status.md +29 -0
  113. package/docs/examples/local-handoff/changes/REQ-003-audit-log-export/planning/design.md +71 -0
  114. package/docs/examples/local-handoff/changes/REQ-003-audit-log-export/planning/task-manifest.json +98 -0
  115. package/docs/examples/local-handoff/changes/REQ-003-audit-log-export/planning/tasks.md +59 -0
  116. package/docs/examples/local-handoff/changes/REQ-003-audit-log-export/review/report-card.json +44 -0
  117. package/docs/examples/local-handoff/roadmap-tracking.json +48 -0
  118. package/docs/examples/pdca-loop/BACKLOG.md +46 -0
  119. package/docs/examples/pdca-loop/README.md +58 -0
  120. package/docs/examples/pdca-loop/ROADMAP.md +133 -0
  121. package/docs/examples/pdca-loop/changes/REQ-001-copy-invite-link/handoff/pr-brief.md +72 -0
  122. package/docs/examples/pdca-loop/changes/REQ-001-copy-invite-link/handoff/status.md +29 -0
  123. package/docs/examples/pdca-loop/changes/REQ-001-copy-invite-link/planning/design.md +72 -0
  124. package/docs/examples/pdca-loop/changes/REQ-001-copy-invite-link/planning/task-manifest.json +201 -0
  125. package/docs/examples/pdca-loop/changes/REQ-001-copy-invite-link/planning/tasks.md +73 -0
  126. package/docs/examples/pdca-loop/changes/REQ-001-copy-invite-link/review/report-card.json +44 -0
  127. package/docs/examples/pdca-loop/roadmap-tracking.json +114 -0
  128. package/docs/examples/scripts/check-example-bindings.sh +116 -0
  129. package/docs/guides/getting-started.md +94 -129
  130. package/docs/guides/getting-started.zh-CN.md +103 -84
  131. package/docs/skill-runtime-migration.md +46 -0
  132. package/docs/v4.3.0-migration-guide.md +60 -225
  133. package/lib/compiler/CLAUDE.md +55 -84
  134. package/lib/compiler/__tests__/drift.test.js +1 -1
  135. package/lib/compiler/__tests__/errors.test.js +0 -1
  136. package/lib/compiler/__tests__/integration.test.js +49 -3
  137. package/lib/compiler/__tests__/manifest.test.js +0 -156
  138. package/lib/compiler/__tests__/parser.test.js +33 -104
  139. package/lib/compiler/__tests__/schemas.test.js +30 -34
  140. package/lib/compiler/__tests__/skills-registry.test.js +114 -0
  141. package/lib/compiler/__tests__/transformer.test.js +5 -9
  142. package/lib/compiler/emitters/antigravity-emitter.js +5 -213
  143. package/lib/compiler/emitters/base-emitter.js +3 -298
  144. package/lib/compiler/emitters/codex-emitter.js +4 -202
  145. package/lib/compiler/emitters/cursor-emitter.js +3 -287
  146. package/lib/compiler/emitters/qwen-emitter.js +4 -176
  147. package/lib/compiler/index.js +65 -345
  148. package/lib/compiler/manifest.js +6 -68
  149. package/lib/compiler/parser.js +31 -63
  150. package/lib/compiler/platforms.js +32 -277
  151. package/lib/compiler/resource-copier.js +176 -6
  152. package/lib/compiler/rules-emitters/__tests__/antigravity-rules-emitter.test.js +3 -3
  153. package/lib/compiler/rules-emitters/__tests__/cursor-rules-emitter.test.js +14 -4
  154. package/lib/compiler/rules-emitters/__tests__/qwen-rules-emitter.test.js +16 -7
  155. package/lib/compiler/rules-emitters/antigravity-rules-emitter.js +10 -15
  156. package/lib/compiler/rules-emitters/base-rules-emitter.js +5 -5
  157. package/lib/compiler/rules-emitters/cursor-rules-emitter.js +7 -12
  158. package/lib/compiler/rules-emitters/index.js +6 -3
  159. package/lib/compiler/rules-emitters/qwen-rules-emitter.js +7 -7
  160. package/lib/compiler/schemas.js +8 -4
  161. package/lib/compiler/skills-registry.js +213 -15
  162. package/lib/compiler/transformer.js +1 -1
  163. package/lib/skill-runtime/CLAUDE.md +19 -0
  164. package/lib/skill-runtime/__tests__/autopilot.test.js +210 -0
  165. package/lib/skill-runtime/__tests__/cli-bootstrap.integration.test.js +291 -0
  166. package/lib/skill-runtime/__tests__/config.test.js +161 -0
  167. package/lib/skill-runtime/__tests__/delegation.test.js +97 -0
  168. package/lib/skill-runtime/__tests__/dispatch.test.js +267 -0
  169. package/lib/skill-runtime/__tests__/intent.test.js +219 -0
  170. package/lib/skill-runtime/__tests__/lifecycle.test.js +169 -0
  171. package/lib/skill-runtime/__tests__/paths.test.js +42 -0
  172. package/lib/skill-runtime/__tests__/planner.tdd.test.js +250 -0
  173. package/lib/skill-runtime/__tests__/prepare-pr.test.js +139 -0
  174. package/lib/skill-runtime/__tests__/query.test.js +284 -0
  175. package/lib/skill-runtime/__tests__/runtime.integration.test.js +281 -0
  176. package/lib/skill-runtime/__tests__/schemas.test.js +207 -0
  177. package/lib/skill-runtime/__tests__/team-state.test.js +51 -0
  178. package/lib/skill-runtime/__tests__/worker-run.test.js +253 -0
  179. package/lib/skill-runtime/__tests__/worker.test.js +56 -0
  180. package/lib/skill-runtime/artifacts.js +93 -0
  181. package/lib/skill-runtime/config.js +379 -0
  182. package/lib/skill-runtime/delegation.js +533 -0
  183. package/lib/skill-runtime/index.js +36 -0
  184. package/lib/skill-runtime/intent.js +333 -0
  185. package/lib/skill-runtime/lifecycle.js +294 -0
  186. package/lib/skill-runtime/operations/CLAUDE.md +19 -0
  187. package/lib/skill-runtime/operations/approve.js +66 -0
  188. package/lib/skill-runtime/operations/autopilot-core.js +337 -0
  189. package/lib/skill-runtime/operations/autopilot-execution.js +307 -0
  190. package/lib/skill-runtime/operations/autopilot-shared.js +48 -0
  191. package/lib/skill-runtime/operations/autopilot.js +163 -0
  192. package/lib/skill-runtime/operations/dispatch.js +454 -0
  193. package/lib/skill-runtime/operations/init.js +64 -0
  194. package/lib/{harness → skill-runtime}/operations/janitor.js +9 -6
  195. package/lib/skill-runtime/operations/plan.js +59 -0
  196. package/lib/skill-runtime/operations/prepare-pr.js +25 -0
  197. package/lib/skill-runtime/operations/release.js +96 -0
  198. package/lib/skill-runtime/operations/resume.js +143 -0
  199. package/lib/skill-runtime/operations/snapshot.js +45 -0
  200. package/lib/skill-runtime/operations/verify.js +170 -0
  201. package/lib/skill-runtime/operations/worker-run.js +529 -0
  202. package/lib/skill-runtime/operations/worker.js +33 -0
  203. package/lib/skill-runtime/paths.js +213 -0
  204. package/lib/skill-runtime/planner.js +519 -0
  205. package/lib/skill-runtime/query.js +157 -0
  206. package/lib/skill-runtime/review.js +557 -0
  207. package/lib/skill-runtime/schemas.js +400 -0
  208. package/lib/{harness → skill-runtime}/store.js +45 -30
  209. package/lib/skill-runtime/team-state.js +122 -0
  210. package/package.json +13 -11
  211. package/.claude/CLAUDE.md +0 -125
  212. package/.claude/agents/architecture-designer.md +0 -443
  213. package/.claude/agents/bug-analyzer.md +0 -381
  214. package/.claude/agents/checklist-agent.md +0 -175
  215. package/.claude/agents/clarify-analyst.md +0 -50
  216. package/.claude/agents/code-quality-reviewer.md +0 -205
  217. package/.claude/agents/code-reviewer.md +0 -71
  218. package/.claude/agents/codex-analyzer.md +0 -39
  219. package/.claude/agents/compatibility-checker.md +0 -579
  220. package/.claude/agents/consistency-checker.md +0 -532
  221. package/.claude/agents/dev-implementer.md +0 -195
  222. package/.claude/agents/flow-researcher.md +0 -132
  223. package/.claude/agents/impact-analyzer.md +0 -440
  224. package/.claude/agents/planner.md +0 -230
  225. package/.claude/agents/prd-writer.md +0 -320
  226. package/.claude/agents/project-guidelines-generator.md +0 -1329
  227. package/.claude/agents/qa-tester.md +0 -313
  228. package/.claude/agents/release-manager.md +0 -295
  229. package/.claude/agents/security-reviewer.md +0 -314
  230. package/.claude/agents/spec-reviewer.md +0 -221
  231. package/.claude/agents/style-guide-generator.md +0 -458
  232. package/.claude/agents/tech-architect.md +0 -516
  233. package/.claude/agents/ui-designer.md +0 -485
  234. package/.claude/commands/core/architecture.md +0 -459
  235. package/.claude/commands/core/guidelines.md +0 -511
  236. package/.claude/commands/core/roadmap.md +0 -468
  237. package/.claude/commands/core/style.md +0 -83
  238. package/.claude/commands/flow/CLAUDE.md +0 -24
  239. package/.claude/commands/flow/archive.md +0 -280
  240. package/.claude/commands/flow/constitution.md +0 -82
  241. package/.claude/commands/flow/context.md +0 -150
  242. package/.claude/commands/flow/delta.md +0 -245
  243. package/.claude/commands/flow/dev.md +0 -40
  244. package/.claude/commands/flow/fix.md +0 -217
  245. package/.claude/commands/flow/ideate.md +0 -214
  246. package/.claude/commands/flow/init.md +0 -38
  247. package/.claude/commands/flow/release.md +0 -36
  248. package/.claude/commands/flow/restart.md +0 -97
  249. package/.claude/commands/flow/spec.md +0 -36
  250. package/.claude/commands/flow/status.md +0 -64
  251. package/.claude/commands/flow/update.md +0 -111
  252. package/.claude/commands/flow/upgrade.md +0 -115
  253. package/.claude/commands/flow/verify.md +0 -37
  254. package/.claude/commands/flow/workspace.md +0 -155
  255. package/.claude/commands/util/cancel-ralph.md +0 -60
  256. package/.claude/commands/util/code-review.md +0 -58
  257. package/.claude/commands/util/git-commit.md +0 -422
  258. package/.claude/commands/util/problem-analyzer.md +0 -60
  259. package/.claude/config/quality-gates.yml +0 -305
  260. package/.claude/config/quality-rules.yml +0 -161
  261. package/.claude/docs/SPEC_KIT_CONSTITUTION_ANALYSIS.md +0 -426
  262. package/.claude/docs/design/consistency-conflict-detection-algorithms.md +0 -658
  263. package/.claude/docs/design/intent-driven-input-design.md +0 -380
  264. package/.claude/docs/design/prd-version-management-design.md +0 -437
  265. package/.claude/docs/examples/design-inspiration-pool.md +0 -59
  266. package/.claude/docs/examples/ui-prototype-constitution-checklist.md +0 -75
  267. package/.claude/docs/guides/INIT_TROUBLESHOOTING.md +0 -117
  268. package/.claude/docs/guides/NEW_TROUBLESHOOTING.md +0 -99
  269. package/.claude/docs/guides/ROADMAP_TROUBLESHOOTING.md +0 -188
  270. package/.claude/docs/guides/TASK_COMPLETION_MARKING.md +0 -338
  271. package/.claude/docs/guides/TEAM_MODE_GUIDE.md +0 -312
  272. package/.claude/docs/implementation-summary-v7.md +0 -449
  273. package/.claude/docs/spec-format-guide.md +0 -349
  274. package/.claude/docs/state-consolidation-design.md +0 -323
  275. package/.claude/docs/templates/ARCHITECTURE_TEMPLATE.md +0 -332
  276. package/.claude/docs/templates/ATTEMPT_TEMPLATE.md +0 -156
  277. package/.claude/docs/templates/BACKLOG_TEMPLATE.md +0 -261
  278. package/.claude/docs/templates/BRAINSTORM_TEMPLATE.md +0 -148
  279. package/.claude/docs/templates/CHECKLIST_TEMPLATE.md +0 -52
  280. package/.claude/docs/templates/CLARIFICATION_REPORT_TEMPLATE.md +0 -206
  281. package/.claude/docs/templates/CODE_REVIEW_TEMPLATE.md +0 -71
  282. package/.claude/docs/templates/DELTA_SPEC_TEMPLATE.md +0 -91
  283. package/.claude/docs/templates/DESIGN_DECISIONS_TEMPLATE.md +0 -151
  284. package/.claude/docs/templates/DESIGN_TEMPLATE.md +0 -157
  285. package/.claude/docs/templates/ERROR_LOG_TEMPLATE.md +0 -80
  286. package/.claude/docs/templates/INIT_FLOW_TEMPLATE.md +0 -198
  287. package/.claude/docs/templates/INTENT_CLARIFICATION_TEMPLATE.md +0 -57
  288. package/.claude/docs/templates/JOURNAL_TEMPLATE.md +0 -75
  289. package/.claude/docs/templates/NEW_ORCHESTRATION_TEMPLATE.md +0 -76
  290. package/.claude/docs/templates/PROPOSAL_TEMPLATE.md +0 -91
  291. package/.claude/docs/templates/RESEARCH_TEMPLATE.md +0 -276
  292. package/.claude/docs/templates/REVIEW-HIGH.md +0 -57
  293. package/.claude/docs/templates/ROADMAP_DIALOGUE_TEMPLATE.md +0 -198
  294. package/.claude/docs/templates/ROADMAP_TEMPLATE.md +0 -310
  295. package/.claude/docs/templates/SPEC_TEMPLATE_DELTA.md +0 -139
  296. package/.claude/docs/templates/SPEC_TEMPLATE_PROJECT.md +0 -93
  297. package/.claude/docs/templates/STYLE_TEMPLATE.md +0 -479
  298. package/.claude/docs/templates/UI_PROTOTYPE_TEMPLATE.md +0 -373
  299. package/.claude/docs/templates/_shared/CLAUDE.md +0 -36
  300. package/.claude/docs/templates/_shared/CONSTITUTION_CHECK.md +0 -125
  301. package/.claude/docs/templates/_shared/VALIDATION_CHECKLIST.md +0 -187
  302. package/.claude/docs/templates/_shared/YAML_FRONTMATTER.md +0 -164
  303. package/.claude/docs/templates/context/dev.jsonl.template +0 -6
  304. package/.claude/docs/templates/context/epic.jsonl.template +0 -5
  305. package/.claude/docs/templates/context/prd.jsonl.template +0 -4
  306. package/.claude/docs/templates/context/research.jsonl.template +0 -4
  307. package/.claude/docs/templates/context/review.jsonl.template +0 -5
  308. package/.claude/docs/templates/context/tech.jsonl.template +0 -5
  309. package/.claude/guides/agent-guides/agent-coordination-guide.md +0 -459
  310. package/.claude/guides/project-guidelines-system.md +0 -463
  311. package/.claude/guides/technical-guides/datetime-handling-guide.md +0 -563
  312. package/.claude/guides/technical-guides/git-github-guide.md +0 -642
  313. package/.claude/guides/technical-guides/test-execution-guide.md +0 -618
  314. package/.claude/guides/workflow-guides/bug-fix-orchestrator.md +0 -217
  315. package/.claude/guides/workflow-guides/flow-orchestrator.md +0 -48
  316. package/.claude/hooks/CLAUDE.md +0 -342
  317. package/.claude/hooks/checklist-gate.js +0 -397
  318. package/.claude/hooks/error-handling-reminder.sh +0 -12
  319. package/.claude/hooks/error-handling-reminder.ts +0 -459
  320. package/.claude/hooks/hooks.json +0 -15
  321. package/.claude/hooks/inject-agent-context.ts +0 -480
  322. package/.claude/hooks/inject-skill-context.ts +0 -359
  323. package/.claude/hooks/post-tool-use-tracker.sh +0 -280
  324. package/.claude/hooks/pre-tool-use-guardrail.sh +0 -36
  325. package/.claude/hooks/pre-tool-use-guardrail.ts +0 -342
  326. package/.claude/hooks/ralph-loop.ts +0 -931
  327. package/.claude/hooks/ralph-stop-hook.sh +0 -190
  328. package/.claude/hooks/skill-activation-prompt.sh +0 -36
  329. package/.claude/hooks/skill-activation-prompt.ts +0 -214
  330. package/.claude/hooks/state/skills-used-test-guard.json +0 -3
  331. package/.claude/hooks/task-completed-hook.ts +0 -593
  332. package/.claude/hooks/teammate-idle-hook.ts +0 -690
  333. package/.claude/hooks/types/team-types.d.ts +0 -238
  334. package/.claude/rules/devflow-conventions.md +0 -286
  335. package/.claude/rules/project-constitution.md +0 -1002
  336. package/.claude/rules/rationalization-library.md +0 -282
  337. package/.claude/schemas/constitution.schema.json +0 -43
  338. package/.claude/scripts/.claude/commands/flow/export-openspec.md +0 -221
  339. package/.claude/scripts/.claude/commands/flow/import-openspec.md +0 -171
  340. package/.claude/scripts/CLAUDE.md +0 -76
  341. package/.claude/scripts/__tests__/openspec.test.js +0 -212
  342. package/.claude/scripts/analyze-upgrade-impact.sh +0 -200
  343. package/.claude/scripts/archive-requirement.sh +0 -394
  344. package/.claude/scripts/calculate-checklist-completion.sh +0 -243
  345. package/.claude/scripts/calculate-quarter.sh +0 -206
  346. package/.claude/scripts/check-dependencies.sh +0 -409
  347. package/.claude/scripts/check-prerequisites.sh +0 -232
  348. package/.claude/scripts/check-task-status.sh +0 -288
  349. package/.claude/scripts/checklist-errors.sh +0 -131
  350. package/.claude/scripts/common.sh +0 -1102
  351. package/.claude/scripts/consolidate-research.sh +0 -182
  352. package/.claude/scripts/create-requirement.sh +0 -451
  353. package/.claude/scripts/delta-parser.ts +0 -637
  354. package/.claude/scripts/detect-file-conflicts.sh +0 -151
  355. package/.claude/scripts/export-contracts.sh +0 -117
  356. package/.claude/scripts/export-openspec.js +0 -222
  357. package/.claude/scripts/extract-data-model.sh +0 -78
  358. package/.claude/scripts/flow-context-add.sh +0 -134
  359. package/.claude/scripts/flow-context-init.sh +0 -133
  360. package/.claude/scripts/flow-context-validate.sh +0 -144
  361. package/.claude/scripts/flow-delta-apply.sh +0 -297
  362. package/.claude/scripts/flow-delta-archive.sh +0 -71
  363. package/.claude/scripts/flow-delta-create.sh +0 -202
  364. package/.claude/scripts/flow-delta-list.sh +0 -142
  365. package/.claude/scripts/flow-delta-status.sh +0 -235
  366. package/.claude/scripts/flow-quality-full.sh +0 -215
  367. package/.claude/scripts/flow-quality-quick.sh +0 -119
  368. package/.claude/scripts/flow-workspace-init.sh +0 -117
  369. package/.claude/scripts/flow-workspace-record.sh +0 -164
  370. package/.claude/scripts/generate-clarification-questions.sh +0 -377
  371. package/.claude/scripts/generate-clarification-report.sh +0 -463
  372. package/.claude/scripts/generate-quickstart.sh +0 -146
  373. package/.claude/scripts/generate-research-tasks.sh +0 -157
  374. package/.claude/scripts/generate-status-report.sh +0 -523
  375. package/.claude/scripts/generate-tech-analysis.sh +0 -46
  376. package/.claude/scripts/get-workflow-status.sh +0 -415
  377. package/.claude/scripts/import-openspec.js +0 -272
  378. package/.claude/scripts/locate-requirement-in-roadmap.sh +0 -233
  379. package/.claude/scripts/manage-constitution.sh +0 -602
  380. package/.claude/scripts/mark-task-complete.sh +0 -198
  381. package/.claude/scripts/parse-task-dependencies.js +0 -334
  382. package/.claude/scripts/populate-research-tasks.sh +0 -284
  383. package/.claude/scripts/record-quality-error.sh +0 -165
  384. package/.claude/scripts/recover-workflow.sh +0 -463
  385. package/.claude/scripts/run-clarify-scan.sh +0 -601
  386. package/.claude/scripts/run-high-review.sh +0 -62
  387. package/.claude/scripts/run-problem-analysis.sh +0 -68
  388. package/.claude/scripts/run-quality-gates.sh +0 -242
  389. package/.claude/scripts/setup-epic.sh +0 -173
  390. package/.claude/scripts/setup-ralph-loop.sh +0 -155
  391. package/.claude/scripts/sync-roadmap-progress.sh +0 -300
  392. package/.claude/scripts/sync-task-marks.sh +0 -199
  393. package/.claude/scripts/team-dev-init.sh +0 -319
  394. package/.claude/scripts/team-state-recovery.sh +0 -229
  395. package/.claude/scripts/test-clarify-scan.sh +0 -515
  396. package/.claude/scripts/update-agent-context.sh +0 -806
  397. package/.claude/scripts/validate-constitution.sh +0 -567
  398. package/.claude/scripts/validate-hooks.sh +0 -487
  399. package/.claude/scripts/validate-research.sh +0 -332
  400. package/.claude/scripts/validate-scope-boundary.sh +0 -493
  401. package/.claude/scripts/validate-scope.sh +0 -200
  402. package/.claude/scripts/verify-gate.sh +0 -269
  403. package/.claude/scripts/verify-setup.sh +0 -37
  404. package/.claude/scripts/workflow-status.ts +0 -433
  405. package/.claude/settings.json +0 -95
  406. package/.claude/skills/_reference-implementations/README.md +0 -96
  407. package/.claude/skills/_reference-implementations/backend-express-prisma/SKILL.md +0 -302
  408. package/.claude/skills/_reference-implementations/backend-express-prisma/resources/architecture-overview.md +0 -451
  409. package/.claude/skills/_reference-implementations/backend-express-prisma/resources/async-and-errors.md +0 -307
  410. package/.claude/skills/_reference-implementations/backend-express-prisma/resources/complete-examples.md +0 -638
  411. package/.claude/skills/_reference-implementations/backend-express-prisma/resources/configuration.md +0 -275
  412. package/.claude/skills/_reference-implementations/backend-express-prisma/resources/database-patterns.md +0 -224
  413. package/.claude/skills/_reference-implementations/backend-express-prisma/resources/middleware-guide.md +0 -213
  414. package/.claude/skills/_reference-implementations/backend-express-prisma/resources/routing-and-controllers.md +0 -756
  415. package/.claude/skills/_reference-implementations/backend-express-prisma/resources/sentry-and-monitoring.md +0 -336
  416. package/.claude/skills/_reference-implementations/backend-express-prisma/resources/services-and-repositories.md +0 -789
  417. package/.claude/skills/_reference-implementations/backend-express-prisma/resources/testing-guide.md +0 -235
  418. package/.claude/skills/_reference-implementations/backend-express-prisma/resources/validation-patterns.md +0 -754
  419. package/.claude/skills/_reference-implementations/frontend-react-mui/SKILL.md +0 -399
  420. package/.claude/skills/_reference-implementations/frontend-react-mui/resources/common-patterns.md +0 -331
  421. package/.claude/skills/_reference-implementations/frontend-react-mui/resources/complete-examples.md +0 -872
  422. package/.claude/skills/_reference-implementations/frontend-react-mui/resources/component-patterns.md +0 -502
  423. package/.claude/skills/_reference-implementations/frontend-react-mui/resources/data-fetching.md +0 -767
  424. package/.claude/skills/_reference-implementations/frontend-react-mui/resources/file-organization.md +0 -502
  425. package/.claude/skills/_reference-implementations/frontend-react-mui/resources/loading-and-error-states.md +0 -501
  426. package/.claude/skills/_reference-implementations/frontend-react-mui/resources/performance.md +0 -406
  427. package/.claude/skills/_reference-implementations/frontend-react-mui/resources/routing-guide.md +0 -364
  428. package/.claude/skills/_reference-implementations/frontend-react-mui/resources/styling-guide.md +0 -428
  429. package/.claude/skills/_reference-implementations/frontend-react-mui/resources/typescript-standards.md +0 -418
  430. package/.claude/skills/attention-refresh/SKILL.md +0 -170
  431. package/.claude/skills/brainstorming/SKILL.md +0 -161
  432. package/.claude/skills/cc-devflow-orchestrator/SKILL.md +0 -169
  433. package/.claude/skills/constitution-guardian/SKILL.md +0 -306
  434. package/.claude/skills/constitution-quick-ref/SKILL.md +0 -374
  435. package/.claude/skills/debugging/SKILL.md +0 -221
  436. package/.claude/skills/file-standards/SKILL.md +0 -353
  437. package/.claude/skills/finishing-branch/SKILL.md +0 -189
  438. package/.claude/skills/flow-dev/CLAUDE.md +0 -16
  439. package/.claude/skills/flow-dev/SKILL.md +0 -94
  440. package/.claude/skills/flow-dev/assets/IMPLEMENTATION_PLAN_TEMPLATE.md +0 -71
  441. package/.claude/skills/flow-dev/context.jsonl +0 -4
  442. package/.claude/skills/flow-dev/dev-implementer.jsonl +0 -8
  443. package/.claude/skills/flow-dev/scripts/entry-gate.sh +0 -116
  444. package/.claude/skills/flow-dev/scripts/exit-gate.sh +0 -101
  445. package/.claude/skills/flow-dev/scripts/task-orchestrator.sh +0 -106
  446. package/.claude/skills/flow-fix/SKILL.md +0 -105
  447. package/.claude/skills/flow-fix/context.jsonl +0 -6
  448. package/.claude/skills/flow-fix/references/bug-analyzer.md +0 -381
  449. package/.claude/skills/flow-init/SKILL.md +0 -105
  450. package/.claude/skills/flow-init/assets/BRAINSTORM_TEMPLATE.md +0 -148
  451. package/.claude/skills/flow-init/assets/INIT_FLOW_TEMPLATE.md +0 -198
  452. package/.claude/skills/flow-init/assets/RESEARCH_TEMPLATE.md +0 -276
  453. package/.claude/skills/flow-init/context.jsonl +0 -5
  454. package/.claude/skills/flow-init/references/flow-researcher.md +0 -132
  455. package/.claude/skills/flow-init/scripts/check-prerequisites.sh +0 -240
  456. package/.claude/skills/flow-init/scripts/consolidate-research.sh +0 -182
  457. package/.claude/skills/flow-init/scripts/create-requirement.sh +0 -404
  458. package/.claude/skills/flow-init/scripts/generate-research-tasks.sh +0 -157
  459. package/.claude/skills/flow-init/scripts/populate-research-tasks.sh +0 -284
  460. package/.claude/skills/flow-init/scripts/validate-research.sh +0 -340
  461. package/.claude/skills/flow-quality/SKILL.md +0 -98
  462. package/.claude/skills/flow-quality/context.jsonl +0 -6
  463. package/.claude/skills/flow-quality/references/code-quality-reviewer.md +0 -205
  464. package/.claude/skills/flow-quality/references/qa-tester.md +0 -313
  465. package/.claude/skills/flow-quality/references/security-reviewer.md +0 -314
  466. package/.claude/skills/flow-quality/references/spec-reviewer.md +0 -221
  467. package/.claude/skills/flow-release/SKILL.md +0 -60
  468. package/.claude/skills/flow-release/context.jsonl +0 -5
  469. package/.claude/skills/flow-release/references/release-manager.md +0 -295
  470. package/.claude/skills/flow-spec/CLAUDE.md +0 -17
  471. package/.claude/skills/flow-spec/SKILL.md +0 -102
  472. package/.claude/skills/flow-spec/context.jsonl +0 -5
  473. package/.claude/skills/flow-spec/scripts/entry-gate.sh +0 -194
  474. package/.claude/skills/flow-spec/scripts/exit-gate.sh +0 -244
  475. package/.claude/skills/flow-spec/scripts/parallel-orchestrator.sh +0 -205
  476. package/.claude/skills/flow-spec/scripts/team-communication.sh +0 -353
  477. package/.claude/skills/flow-spec/scripts/team-init.sh +0 -195
  478. package/.claude/skills/flow-spec/scripts/test-team-mode.sh +0 -496
  479. package/.claude/skills/flow-spec/team-config.json +0 -165
  480. package/.claude/skills/flow-verify/CLAUDE.md +0 -10
  481. package/.claude/skills/flow-verify/SKILL.md +0 -53
  482. package/.claude/skills/flow-verify/context.jsonl +0 -5
  483. package/.claude/skills/fractal-docs/SKILL.md +0 -45
  484. package/.claude/skills/journey-checker/SKILL.md +0 -199
  485. package/.claude/skills/journey-checker/pressure-scenarios.md +0 -164
  486. package/.claude/skills/receiving-review/SKILL.md +0 -153
  487. package/.claude/skills/skill-creator/LICENSE.txt +0 -202
  488. package/.claude/skills/skill-creator/SKILL.md +0 -356
  489. package/.claude/skills/skill-creator/references/output-patterns.md +0 -82
  490. package/.claude/skills/skill-creator/references/workflows.md +0 -28
  491. package/.claude/skills/skill-creator/scripts/init_skill.py +0 -303
  492. package/.claude/skills/skill-creator/scripts/package_skill.py +0 -110
  493. package/.claude/skills/skill-creator/scripts/quick_validate.py +0 -95
  494. package/.claude/skills/skill-rules.json +0 -359
  495. package/.claude/skills/tdd/SKILL.md +0 -218
  496. package/.claude/skills/tdd-enforcer/SKILL.md +0 -192
  497. package/.claude/skills/utility/npm-release/CLAUDE.md +0 -55
  498. package/.claude/skills/utility/npm-release/SKILL.md +0 -379
  499. package/.claude/skills/utility/npm-release/references/version-decision-guide.md +0 -134
  500. package/.claude/skills/utility/npm-release/scripts/atomic-version-bump.sh +0 -95
  501. package/.claude/skills/utility/npm-release/scripts/validate-version-sync.sh +0 -82
  502. package/.claude/skills/utility/npm-release/scripts/version-decision-tree.sh +0 -44
  503. package/.claude/skills/verification/SKILL.md +0 -158
  504. package/.claude/skills/workflow.yaml +0 -219
  505. package/.claude/tests/README.md +0 -300
  506. package/.claude/tests/TODO.md +0 -69
  507. package/.claude/tests/__pycache__/test_analyze_upgrade_impact.cpython-311-pytest-7.2.2.pyc +0 -0
  508. package/.claude/tests/__pycache__/test_consolidate_research.cpython-311-pytest-7.2.2.pyc +0 -0
  509. package/.claude/tests/__pycache__/test_export_contracts.cpython-311-pytest-7.2.2.pyc +0 -0
  510. package/.claude/tests/__pycache__/test_extract_data_model.cpython-311-pytest-7.2.2.pyc +0 -0
  511. package/.claude/tests/__pycache__/test_generate_quickstart.cpython-311-pytest-7.2.2.pyc +0 -0
  512. package/.claude/tests/__pycache__/test_generate_research_tasks.cpython-311-pytest-7.2.2.pyc +0 -0
  513. package/.claude/tests/constitution/run_all_constitution_tests.sh +0 -111
  514. package/.claude/tests/constitution/test_agent_assignment.sh +0 -207
  515. package/.claude/tests/constitution/test_article_coverage.sh +0 -201
  516. package/.claude/tests/constitution/test_template_completeness.sh +0 -150
  517. package/.claude/tests/constitution/test_version_consistency.sh +0 -120
  518. package/.claude/tests/fixtures/spec_delta_full.md +0 -16
  519. package/.claude/tests/fixtures/tasks_progress_sample.md +0 -5
  520. package/.claude/tests/run-all-tests.sh +0 -229
  521. package/.claude/tests/scripts/run.sh +0 -30
  522. package/.claude/tests/scripts/test-framework.sh +0 -128
  523. package/.claude/tests/scripts/test_check_prerequisites.sh +0 -511
  524. package/.claude/tests/scripts/test_check_prerequisites.sh.bak +0 -504
  525. package/.claude/tests/scripts/test_check_prerequisites.sh.bak2 +0 -505
  526. package/.claude/tests/scripts/test_check_prerequisites.sh.bak3 +0 -506
  527. package/.claude/tests/scripts/test_check_prerequisites.sh.bak4 +0 -507
  528. package/.claude/tests/scripts/test_check_prerequisites.sh.bak5 +0 -508
  529. package/.claude/tests/scripts/test_check_task_status.sh +0 -499
  530. package/.claude/tests/scripts/test_common.sh +0 -244
  531. package/.claude/tests/scripts/test_generate_status_report.sh +0 -71
  532. package/.claude/tests/scripts/test_mark_task_complete.sh +0 -441
  533. package/.claude/tests/scripts/test_mark_task_complete.sh.backup +0 -410
  534. package/.claude/tests/scripts/test_recover_workflow.sh +0 -304
  535. package/.claude/tests/scripts/test_setup_epic.sh +0 -437
  536. package/.claude/tests/scripts/test_sync_task_marks.sh +0 -196
  537. package/.claude/tests/scripts/test_validate_constitution.sh +0 -74
  538. package/.claude/tests/scripts/test_validate_research.sh +0 -462
  539. package/.claude/tests/slugify.bats +0 -82
  540. package/.claude/tests/test-framework.sh +0 -732
  541. package/.claude/tests/test_analyze_upgrade_impact.py +0 -34
  542. package/.claude/tests/test_consolidate_research.py +0 -48
  543. package/.claude/tests/test_export_contracts.py +0 -43
  544. package/.claude/tests/test_extract_data_model.py +0 -33
  545. package/.claude/tests/test_generate_quickstart.py +0 -50
  546. package/.claude/tests/test_generate_research_tasks.py +0 -52
  547. package/.claude/tsc-cache/70d2fc6d-2936-429b-b529-429f1aae8c88/affected-repos.txt +0 -1
  548. package/.claude/tsc-cache/70d2fc6d-2936-429b-b529-429f1aae8c88/edited-files.log +0 -2
  549. package/bin/harness.js +0 -22
  550. package/docs/commands/core-roadmap.md +0 -106
  551. package/docs/commands/core-roadmap.zh-CN.md +0 -102
  552. package/docs/commands/core-style.md +0 -53
  553. package/docs/commands/core-style.zh-CN.md +0 -53
  554. package/docs/commands/flow-init.md +0 -140
  555. package/docs/commands/flow-init.zh-CN.md +0 -169
  556. package/docs/commands/flow-new.md +0 -39
  557. package/docs/commands/flow-new.zh-CN.md +0 -39
  558. package/lib/compiler/__tests__/compile-regression.test.js +0 -103
  559. package/lib/compiler/__tests__/multi-module-emitters.test.js +0 -534
  560. package/lib/compiler/__tests__/resource-copier.test.js +0 -26
  561. package/lib/compiler/__tests__/skill-discovery.test.js +0 -72
  562. package/lib/compiler/context-expander.js +0 -179
  563. package/lib/compiler/rules-emitters/__tests__/codex-rules-emitter.test.js +0 -109
  564. package/lib/compiler/rules-emitters/codex-rules-emitter.js +0 -116
  565. package/lib/compiler/skill-discovery.js +0 -68
  566. package/lib/harness/CLAUDE.md +0 -22
  567. package/lib/harness/__tests__/planner.tdd.test.js +0 -125
  568. package/lib/harness/cli.js +0 -208
  569. package/lib/harness/index.js +0 -18
  570. package/lib/harness/operations/dispatch.js +0 -298
  571. package/lib/harness/operations/init.js +0 -48
  572. package/lib/harness/operations/pack.js +0 -100
  573. package/lib/harness/operations/plan.js +0 -83
  574. package/lib/harness/operations/release.js +0 -170
  575. package/lib/harness/operations/resume.js +0 -44
  576. package/lib/harness/operations/verify.js +0 -177
  577. package/lib/harness/planner.js +0 -272
  578. package/lib/harness/query.js +0 -126
  579. package/lib/harness/schemas.js +0 -129
@@ -1,789 +0,0 @@
1
- # Services and Repositories - Business Logic Layer
2
-
3
- Complete guide to organizing business logic with services and data access with repositories.
4
-
5
- ## Table of Contents
6
-
7
- - [Service Layer Overview](#service-layer-overview)
8
- - [Dependency Injection Pattern](#dependency-injection-pattern)
9
- - [Singleton Pattern](#singleton-pattern)
10
- - [Repository Pattern](#repository-pattern)
11
- - [Service Design Principles](#service-design-principles)
12
- - [Caching Strategies](#caching-strategies)
13
- - [Testing Services](#testing-services)
14
-
15
- ---
16
-
17
- ## Service Layer Overview
18
-
19
- ### Purpose of Services
20
-
21
- **Services contain business logic** - the 'what' and 'why' of your application:
22
-
23
- ```
24
- Controller asks: "Should I do this?"
25
- Service answers: "Yes/No, here's why, and here's what happens"
26
- Repository executes: "Here's the data you requested"
27
- ```
28
-
29
- **Services are responsible for:**
30
- - ✅ Business rules enforcement
31
- - ✅ Orchestrating multiple repositories
32
- - ✅ Transaction management
33
- - ✅ Complex calculations
34
- - ✅ External service integration
35
- - ✅ Business validations
36
-
37
- **Services should NOT:**
38
- - ❌ Know about HTTP (Request/Response)
39
- - ❌ Direct Prisma access (use repositories)
40
- - ❌ Handle route-specific logic
41
- - ❌ Format HTTP responses
42
-
43
- ---
44
-
45
- ## Dependency Injection Pattern
46
-
47
- ### Why Dependency Injection?
48
-
49
- **Benefits:**
50
- - Easy to test (inject mocks)
51
- - Clear dependencies
52
- - Flexible configuration
53
- - Promotes loose coupling
54
-
55
- ### Excellent Example: NotificationService
56
-
57
- **File:** `/blog-api/src/services/NotificationService.ts`
58
-
59
- ```typescript
60
- // Define dependencies interface for clarity
61
- export interface NotificationServiceDependencies {
62
- prisma: PrismaClient;
63
- batchingService: BatchingService;
64
- emailComposer: EmailComposer;
65
- }
66
-
67
- // Service with dependency injection
68
- export class NotificationService {
69
- private prisma: PrismaClient;
70
- private batchingService: BatchingService;
71
- private emailComposer: EmailComposer;
72
- private preferencesCache: Map<string, { preferences: UserPreference; timestamp: number }> = new Map();
73
- private CACHE_TTL = (notificationConfig.preferenceCacheTTLMinutes || 5) * 60 * 1000;
74
-
75
- // Dependencies injected via constructor
76
- constructor(dependencies: NotificationServiceDependencies) {
77
- this.prisma = dependencies.prisma;
78
- this.batchingService = dependencies.batchingService;
79
- this.emailComposer = dependencies.emailComposer;
80
- }
81
-
82
- /**
83
- * Create a notification and route it appropriately
84
- */
85
- async createNotification(params: CreateNotificationParams) {
86
- const { recipientID, type, title, message, link, context = {}, channel = 'both', priority = NotificationPriority.NORMAL } = params;
87
-
88
- try {
89
- // Get template and render content
90
- const template = getNotificationTemplate(type);
91
- const rendered = renderNotificationContent(template, context);
92
-
93
- // Create in-app notification record
94
- const notificationId = await createNotificationRecord({
95
- instanceId: parseInt(context.instanceId || '0', 10),
96
- template: type,
97
- recipientUserId: recipientID,
98
- channel: channel === 'email' ? 'email' : 'inApp',
99
- contextData: context,
100
- title: finalTitle,
101
- message: finalMessage,
102
- link: finalLink,
103
- });
104
-
105
- // Route notification based on channel
106
- if (channel === 'email' || channel === 'both') {
107
- await this.routeNotification({
108
- notificationId,
109
- userId: recipientID,
110
- type,
111
- priority,
112
- title: finalTitle,
113
- message: finalMessage,
114
- link: finalLink,
115
- context,
116
- });
117
- }
118
-
119
- return notification;
120
- } catch (error) {
121
- ErrorLogger.log(error, {
122
- context: {
123
- '[NotificationService] createNotification': {
124
- type: params.type,
125
- recipientID: params.recipientID,
126
- },
127
- },
128
- });
129
- throw error;
130
- }
131
- }
132
-
133
- /**
134
- * Route notification based on user preferences
135
- */
136
- private async routeNotification(params: { notificationId: number; userId: string; type: string; priority: NotificationPriority; title: string; message: string; link?: string; context?: Record<string, any> }) {
137
- // Get user preferences with caching
138
- const preferences = await this.getUserPreferences(params.userId);
139
-
140
- // Check if we should batch or send immediately
141
- if (this.shouldBatchEmail(preferences, params.type, params.priority)) {
142
- await this.batchingService.queueNotificationForBatch({
143
- notificationId: params.notificationId,
144
- userId: params.userId,
145
- userPreference: preferences,
146
- priority: params.priority,
147
- });
148
- } else {
149
- // Send immediately via EmailComposer
150
- await this.sendImmediateEmail({
151
- userId: params.userId,
152
- title: params.title,
153
- message: params.message,
154
- link: params.link,
155
- context: params.context,
156
- type: params.type,
157
- });
158
- }
159
- }
160
-
161
- /**
162
- * Determine if email should be batched
163
- */
164
- shouldBatchEmail(preferences: UserPreference, notificationType: string, priority: NotificationPriority): boolean {
165
- // HIGH priority always immediate
166
- if (priority === NotificationPriority.HIGH) {
167
- return false;
168
- }
169
-
170
- // Check batch mode
171
- const batchMode = preferences.emailBatchMode || BatchMode.IMMEDIATE;
172
- return batchMode !== BatchMode.IMMEDIATE;
173
- }
174
-
175
- /**
176
- * Get user preferences with caching
177
- */
178
- async getUserPreferences(userId: string): Promise<UserPreference> {
179
- // Check cache first
180
- const cached = this.preferencesCache.get(userId);
181
- if (cached && Date.now() - cached.timestamp < this.CACHE_TTL) {
182
- return cached.preferences;
183
- }
184
-
185
- const preference = await this.prisma.userPreference.findUnique({
186
- where: { userID: userId },
187
- });
188
-
189
- const finalPreferences = preference || DEFAULT_PREFERENCES;
190
-
191
- // Update cache
192
- this.preferencesCache.set(userId, {
193
- preferences: finalPreferences,
194
- timestamp: Date.now(),
195
- });
196
-
197
- return finalPreferences;
198
- }
199
- }
200
- ```
201
-
202
- **Usage in Controller:**
203
-
204
- ```typescript
205
- // Instantiate with dependencies
206
- const notificationService = new NotificationService({
207
- prisma: PrismaService.main,
208
- batchingService: new BatchingService(PrismaService.main),
209
- emailComposer: new EmailComposer(),
210
- });
211
-
212
- // Use in controller
213
- const notification = await notificationService.createNotification({
214
- recipientID: 'user-123',
215
- type: 'AFRLWorkflowNotification',
216
- context: { workflowName: 'AFRL Monthly Report' },
217
- });
218
- ```
219
-
220
- **Key Takeaways:**
221
- - Dependencies passed via constructor
222
- - Clear interface defines required dependencies
223
- - Easy to test (inject mocks)
224
- - Encapsulated caching logic
225
- - Business rules isolated from HTTP
226
-
227
- ---
228
-
229
- ## Singleton Pattern
230
-
231
- ### When to Use Singletons
232
-
233
- **Use for:**
234
- - Services with expensive initialization
235
- - Services with shared state (caching)
236
- - Services accessed from many places
237
- - Permission services
238
- - Configuration services
239
-
240
- ### Example: PermissionService (Singleton)
241
-
242
- **File:** `/blog-api/src/services/permissionService.ts`
243
-
244
- ```typescript
245
- import { PrismaClient } from '@prisma/client';
246
-
247
- class PermissionService {
248
- private static instance: PermissionService;
249
- private prisma: PrismaClient;
250
- private permissionCache: Map<string, { canAccess: boolean; timestamp: number }> = new Map();
251
- private CACHE_TTL = 5 * 60 * 1000; // 5 minutes
252
-
253
- // Private constructor prevents direct instantiation
254
- private constructor() {
255
- this.prisma = PrismaService.main;
256
- }
257
-
258
- // Get singleton instance
259
- public static getInstance(): PermissionService {
260
- if (!PermissionService.instance) {
261
- PermissionService.instance = new PermissionService();
262
- }
263
- return PermissionService.instance;
264
- }
265
-
266
- /**
267
- * Check if user can complete a workflow step
268
- */
269
- async canCompleteStep(userId: string, stepInstanceId: number): Promise<boolean> {
270
- const cacheKey = `${userId}:${stepInstanceId}`;
271
-
272
- // Check cache
273
- const cached = this.permissionCache.get(cacheKey);
274
- if (cached && Date.now() - cached.timestamp < this.CACHE_TTL) {
275
- return cached.canAccess;
276
- }
277
-
278
- try {
279
- const post = await this.prisma.post.findUnique({
280
- where: { id: postId },
281
- include: {
282
- author: true,
283
- comments: {
284
- include: {
285
- user: true,
286
- },
287
- },
288
- },
289
- });
290
-
291
- if (!post) {
292
- return false;
293
- }
294
-
295
- // Check if user has permission
296
- const canEdit = post.authorId === userId ||
297
- await this.isUserAdmin(userId);
298
-
299
- // Cache result
300
- this.permissionCache.set(cacheKey, {
301
- canAccess: isAssigned,
302
- timestamp: Date.now(),
303
- });
304
-
305
- return isAssigned;
306
- } catch (error) {
307
- console.error('[PermissionService] Error checking step permission:', error);
308
- return false;
309
- }
310
- }
311
-
312
- /**
313
- * Clear cache for user
314
- */
315
- clearUserCache(userId: string): void {
316
- for (const [key] of this.permissionCache) {
317
- if (key.startsWith(`${userId}:`)) {
318
- this.permissionCache.delete(key);
319
- }
320
- }
321
- }
322
-
323
- /**
324
- * Clear all cache
325
- */
326
- clearCache(): void {
327
- this.permissionCache.clear();
328
- }
329
- }
330
-
331
- // Export singleton instance
332
- export const permissionService = PermissionService.getInstance();
333
- ```
334
-
335
- **Usage:**
336
-
337
- ```typescript
338
- import { permissionService } from '../services/permissionService';
339
-
340
- // Use anywhere in the codebase
341
- const canComplete = await permissionService.canCompleteStep(userId, stepId);
342
-
343
- if (!canComplete) {
344
- throw new ForbiddenError('You do not have permission to complete this step');
345
- }
346
- ```
347
-
348
- ---
349
-
350
- ## Repository Pattern
351
-
352
- ### Purpose of Repositories
353
-
354
- **Repositories abstract data access** - the 'how' of data operations:
355
-
356
- ```
357
- Service: "Get me all active users sorted by name"
358
- Repository: "Here's the Prisma query that does that"
359
- ```
360
-
361
- **Repositories are responsible for:**
362
- - ✅ All Prisma operations
363
- - ✅ Query construction
364
- - ✅ Query optimization (select, include)
365
- - ✅ Database error handling
366
- - ✅ Caching database results
367
-
368
- **Repositories should NOT:**
369
- - ❌ Contain business logic
370
- - ❌ Know about HTTP
371
- - ❌ Make decisions (that's service layer)
372
-
373
- ### Repository Template
374
-
375
- ```typescript
376
- // repositories/UserRepository.ts
377
- import { PrismaService } from '@project-lifecycle-portal/database';
378
- import type { User, Prisma } from '@project-lifecycle-portal/database';
379
-
380
- export class UserRepository {
381
- /**
382
- * Find user by ID with optimized query
383
- */
384
- async findById(userId: string): Promise<User | null> {
385
- try {
386
- return await PrismaService.main.user.findUnique({
387
- where: { userID: userId },
388
- select: {
389
- userID: true,
390
- email: true,
391
- name: true,
392
- isActive: true,
393
- roles: true,
394
- createdAt: true,
395
- updatedAt: true,
396
- },
397
- });
398
- } catch (error) {
399
- console.error('[UserRepository] Error finding user by ID:', error);
400
- throw new Error(`Failed to find user: ${userId}`);
401
- }
402
- }
403
-
404
- /**
405
- * Find all active users
406
- */
407
- async findActive(options?: { orderBy?: Prisma.UserOrderByWithRelationInput }): Promise<User[]> {
408
- try {
409
- return await PrismaService.main.user.findMany({
410
- where: { isActive: true },
411
- orderBy: options?.orderBy || { name: 'asc' },
412
- select: {
413
- userID: true,
414
- email: true,
415
- name: true,
416
- roles: true,
417
- },
418
- });
419
- } catch (error) {
420
- console.error('[UserRepository] Error finding active users:', error);
421
- throw new Error('Failed to find active users');
422
- }
423
- }
424
-
425
- /**
426
- * Find user by email
427
- */
428
- async findByEmail(email: string): Promise<User | null> {
429
- try {
430
- return await PrismaService.main.user.findUnique({
431
- where: { email },
432
- });
433
- } catch (error) {
434
- console.error('[UserRepository] Error finding user by email:', error);
435
- throw new Error(`Failed to find user with email: ${email}`);
436
- }
437
- }
438
-
439
- /**
440
- * Create new user
441
- */
442
- async create(data: Prisma.UserCreateInput): Promise<User> {
443
- try {
444
- return await PrismaService.main.user.create({ data });
445
- } catch (error) {
446
- console.error('[UserRepository] Error creating user:', error);
447
- throw new Error('Failed to create user');
448
- }
449
- }
450
-
451
- /**
452
- * Update user
453
- */
454
- async update(userId: string, data: Prisma.UserUpdateInput): Promise<User> {
455
- try {
456
- return await PrismaService.main.user.update({
457
- where: { userID: userId },
458
- data,
459
- });
460
- } catch (error) {
461
- console.error('[UserRepository] Error updating user:', error);
462
- throw new Error(`Failed to update user: ${userId}`);
463
- }
464
- }
465
-
466
- /**
467
- * Delete user (soft delete by setting isActive = false)
468
- */
469
- async delete(userId: string): Promise<User> {
470
- try {
471
- return await PrismaService.main.user.update({
472
- where: { userID: userId },
473
- data: { isActive: false },
474
- });
475
- } catch (error) {
476
- console.error('[UserRepository] Error deleting user:', error);
477
- throw new Error(`Failed to delete user: ${userId}`);
478
- }
479
- }
480
-
481
- /**
482
- * Check if email exists
483
- */
484
- async emailExists(email: string): Promise<boolean> {
485
- try {
486
- const count = await PrismaService.main.user.count({
487
- where: { email },
488
- });
489
- return count > 0;
490
- } catch (error) {
491
- console.error('[UserRepository] Error checking email exists:', error);
492
- throw new Error('Failed to check if email exists');
493
- }
494
- }
495
- }
496
-
497
- // Export singleton instance
498
- export const userRepository = new UserRepository();
499
- ```
500
-
501
- **Using Repository in Service:**
502
-
503
- ```typescript
504
- // services/userService.ts
505
- import { userRepository } from '../repositories/UserRepository';
506
- import { ConflictError, NotFoundError } from '../utils/errors';
507
-
508
- export class UserService {
509
- /**
510
- * Create new user with business rules
511
- */
512
- async createUser(data: { email: string; name: string; roles: string[] }): Promise<User> {
513
- // Business rule: Check if email already exists
514
- const emailExists = await userRepository.emailExists(data.email);
515
- if (emailExists) {
516
- throw new ConflictError('Email already exists');
517
- }
518
-
519
- // Business rule: Validate roles
520
- const validRoles = ['admin', 'operations', 'user'];
521
- const invalidRoles = data.roles.filter((role) => !validRoles.includes(role));
522
- if (invalidRoles.length > 0) {
523
- throw new ValidationError(`Invalid roles: ${invalidRoles.join(', ')}`);
524
- }
525
-
526
- // Create user via repository
527
- return await userRepository.create({
528
- email: data.email,
529
- name: data.name,
530
- roles: data.roles,
531
- isActive: true,
532
- });
533
- }
534
-
535
- /**
536
- * Get user by ID
537
- */
538
- async getUser(userId: string): Promise<User> {
539
- const user = await userRepository.findById(userId);
540
-
541
- if (!user) {
542
- throw new NotFoundError(`User not found: ${userId}`);
543
- }
544
-
545
- return user;
546
- }
547
- }
548
- ```
549
-
550
- ---
551
-
552
- ## Service Design Principles
553
-
554
- ### 1. Single Responsibility
555
-
556
- Each service should have ONE clear purpose:
557
-
558
- ```typescript
559
- // ✅ GOOD - Single responsibility
560
- class UserService {
561
- async createUser() {}
562
- async updateUser() {}
563
- async deleteUser() {}
564
- }
565
-
566
- class EmailService {
567
- async sendEmail() {}
568
- async sendBulkEmails() {}
569
- }
570
-
571
- // ❌ BAD - Too many responsibilities
572
- class UserService {
573
- async createUser() {}
574
- async sendWelcomeEmail() {} // Should be EmailService
575
- async logUserActivity() {} // Should be AuditService
576
- async processPayment() {} // Should be PaymentService
577
- }
578
- ```
579
-
580
- ### 2. Clear Method Names
581
-
582
- Method names should describe WHAT they do:
583
-
584
- ```typescript
585
- // ✅ GOOD - Clear intent
586
- async createNotification()
587
- async getUserPreferences()
588
- async shouldBatchEmail()
589
- async routeNotification()
590
-
591
- // ❌ BAD - Vague or misleading
592
- async process()
593
- async handle()
594
- async doIt()
595
- async execute()
596
- ```
597
-
598
- ### 3. Return Types
599
-
600
- Always use explicit return types:
601
-
602
- ```typescript
603
- // ✅ GOOD - Explicit types
604
- async createUser(data: CreateUserDTO): Promise<User> {}
605
- async findUsers(): Promise<User[]> {}
606
- async deleteUser(id: string): Promise<void> {}
607
-
608
- // ❌ BAD - Implicit any
609
- async createUser(data) {} // No types!
610
- ```
611
-
612
- ### 4. Error Handling
613
-
614
- Services should throw meaningful errors:
615
-
616
- ```typescript
617
- // ✅ GOOD - Meaningful errors
618
- if (!user) {
619
- throw new NotFoundError(`User not found: ${userId}`);
620
- }
621
-
622
- if (emailExists) {
623
- throw new ConflictError('Email already exists');
624
- }
625
-
626
- // ❌ BAD - Generic errors
627
- if (!user) {
628
- throw new Error('Error'); // What error?
629
- }
630
- ```
631
-
632
- ### 5. Avoid God Services
633
-
634
- Don't create services that do everything:
635
-
636
- ```typescript
637
- // ❌ BAD - God service
638
- class WorkflowService {
639
- async startWorkflow() {}
640
- async completeStep() {}
641
- async assignRoles() {}
642
- async sendNotifications() {} // Should be NotificationService
643
- async validatePermissions() {} // Should be PermissionService
644
- async logAuditTrail() {} // Should be AuditService
645
- // ... 50 more methods
646
- }
647
-
648
- // ✅ GOOD - Focused services
649
- class WorkflowService {
650
- constructor(
651
- private notificationService: NotificationService,
652
- private permissionService: PermissionService,
653
- private auditService: AuditService
654
- ) {}
655
-
656
- async startWorkflow() {
657
- // Orchestrate other services
658
- await this.permissionService.checkPermission();
659
- await this.workflowRepository.create();
660
- await this.notificationService.notify();
661
- await this.auditService.log();
662
- }
663
- }
664
- ```
665
-
666
- ---
667
-
668
- ## Caching Strategies
669
-
670
- ### 1. In-Memory Caching
671
-
672
- ```typescript
673
- class UserService {
674
- private cache: Map<string, { user: User; timestamp: number }> = new Map();
675
- private CACHE_TTL = 5 * 60 * 1000; // 5 minutes
676
-
677
- async getUser(userId: string): Promise<User> {
678
- // Check cache
679
- const cached = this.cache.get(userId);
680
- if (cached && Date.now() - cached.timestamp < this.CACHE_TTL) {
681
- return cached.user;
682
- }
683
-
684
- // Fetch from database
685
- const user = await userRepository.findById(userId);
686
-
687
- // Update cache
688
- if (user) {
689
- this.cache.set(userId, { user, timestamp: Date.now() });
690
- }
691
-
692
- return user;
693
- }
694
-
695
- clearUserCache(userId: string): void {
696
- this.cache.delete(userId);
697
- }
698
- }
699
- ```
700
-
701
- ### 2. Cache Invalidation
702
-
703
- ```typescript
704
- class UserService {
705
- async updateUser(userId: string, data: UpdateUserDTO): Promise<User> {
706
- // Update in database
707
- const user = await userRepository.update(userId, data);
708
-
709
- // Invalidate cache
710
- this.clearUserCache(userId);
711
-
712
- return user;
713
- }
714
- }
715
- ```
716
-
717
- ---
718
-
719
- ## Testing Services
720
-
721
- ### Unit Tests
722
-
723
- ```typescript
724
- // tests/userService.test.ts
725
- import { UserService } from '../services/userService';
726
- import { userRepository } from '../repositories/UserRepository';
727
- import { ConflictError } from '../utils/errors';
728
-
729
- // Mock repository
730
- jest.mock('../repositories/UserRepository');
731
-
732
- describe('UserService', () => {
733
- let userService: UserService;
734
-
735
- beforeEach(() => {
736
- userService = new UserService();
737
- jest.clearAllMocks();
738
- });
739
-
740
- describe('createUser', () => {
741
- it('should create user when email does not exist', async () => {
742
- // Arrange
743
- const userData = {
744
- email: 'test@example.com',
745
- name: 'Test User',
746
- roles: ['user'],
747
- };
748
-
749
- (userRepository.emailExists as jest.Mock).mockResolvedValue(false);
750
- (userRepository.create as jest.Mock).mockResolvedValue({
751
- userID: '123',
752
- ...userData,
753
- });
754
-
755
- // Act
756
- const user = await userService.createUser(userData);
757
-
758
- // Assert
759
- expect(user).toBeDefined();
760
- expect(user.email).toBe(userData.email);
761
- expect(userRepository.emailExists).toHaveBeenCalledWith(userData.email);
762
- expect(userRepository.create).toHaveBeenCalled();
763
- });
764
-
765
- it('should throw ConflictError when email exists', async () => {
766
- // Arrange
767
- const userData = {
768
- email: 'existing@example.com',
769
- name: 'Test User',
770
- roles: ['user'],
771
- };
772
-
773
- (userRepository.emailExists as jest.Mock).mockResolvedValue(true);
774
-
775
- // Act & Assert
776
- await expect(userService.createUser(userData)).rejects.toThrow(ConflictError);
777
- expect(userRepository.create).not.toHaveBeenCalled();
778
- });
779
- });
780
- });
781
- ```
782
-
783
- ---
784
-
785
- **Related Files:**
786
- - [SKILL.md](SKILL.md) - Main guide
787
- - [routing-and-controllers.md](routing-and-controllers.md) - Controllers that use services
788
- - [database-patterns.md](database-patterns.md) - Prisma and repository patterns
789
- - [complete-examples.md](complete-examples.md) - Full service/repository examples