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,931 +0,0 @@
1
- #!/usr/bin/env npx ts-node
2
- /**
3
- * [INPUT]: 依赖 quality-gates.yml 的 verify 命令配置, orchestration_status.json 的 Team 状态
4
- * [OUTPUT]: 对外提供 SubagentStop 钩子,拦截子 Agent 停止并执行程序化验证
5
- * [POS]: hooks/ 的 Ralph Loop 控制器,支持单 Agent 和多 Teammate 模式
6
- * [PROTOCOL]: 变更时更新此头部,然后检查 CLAUDE.md
7
- *
8
- * =============================================================================
9
- * CC-DevFlow Ralph Loop - SubagentStop Hook (借鉴 Trellis)
10
- * =============================================================================
11
- *
12
- * 核心设计哲学:
13
- * - 子 Agent 尝试停止时执行程序化验证
14
- * - 验证失败则阻止停止,返回错误信息
15
- * - 最多迭代 N 次,防止无限循环
16
- * - 状态持久化到 .ralph-state.json (单 Agent) 或 orchestration_status.json (Team)
17
- *
18
- * v4.7 Team 模式支持:
19
- * - Teammate 级别验证:每个 teammate 停止时验证自己的改动
20
- * - 全局验证:最后一个 teammate 停止时执行全局验证
21
- * - 分布式迭代控制:每个 teammate 独立计数 + 全局计数
22
- *
23
- * 触发条件:SubagentStop 事件
24
- */
25
-
26
- import * as fs from 'fs';
27
- import * as path from 'path';
28
- import { execSync } from 'child_process';
29
-
30
- // =============================================================================
31
- // 类型定义
32
- // =============================================================================
33
-
34
- interface RalphState {
35
- agent_id: string;
36
- iteration: number;
37
- last_failures: FailureRecord[];
38
- started_at: string;
39
- }
40
-
41
- interface FailureRecord {
42
- command: string;
43
- output: string;
44
- timestamp: string;
45
- }
46
-
47
- interface VerifyCommand {
48
- name?: string;
49
- command: string;
50
- required?: boolean;
51
- }
52
-
53
- // Team 模式类型
54
- interface TeammateRalphState {
55
- iteration: number;
56
- lastVerifyResult: 'passed' | 'failed' | 'skipped';
57
- lastVerifyAt?: string;
58
- lastFailures?: FailureRecord[];
59
- }
60
-
61
- interface RalphLoopTeamState {
62
- enabled: boolean;
63
- teammates: Record<string, TeammateRalphState>;
64
- globalIteration: number;
65
- maxIterations: number;
66
- startedAt: string;
67
- }
68
-
69
- interface TeamState {
70
- mode: 'sequential' | 'parallel';
71
- lead: string;
72
- teammates: Array<{
73
- id: string;
74
- role: string;
75
- status: string;
76
- currentTask: string | null;
77
- completedTasks: string[];
78
- lastActiveAt: string;
79
- }>;
80
- taskAssignments: Record<string, string>;
81
- createdAt: string;
82
- updatedAt: string;
83
- }
84
-
85
- interface OrchestrationStatus {
86
- reqId: string;
87
- status: string;
88
- phase: string;
89
- team?: TeamState;
90
- ralphLoop?: RalphLoopTeamState;
91
- updatedAt: string;
92
- }
93
-
94
- interface TeamModeConfig {
95
- enabled?: boolean;
96
- scope?: 'teammate' | 'global';
97
- teammate_verify?: Record<string, string[]>;
98
- global_verify?: string[];
99
- max_iterations_per_teammate?: number;
100
- max_global_iterations?: number;
101
- }
102
-
103
- interface QualityGatesConfig {
104
- verify?: (string | VerifyCommand)[];
105
- ralph_loop?: {
106
- max_iterations?: number;
107
- timeout_minutes?: number;
108
- team_mode?: TeamModeConfig;
109
- };
110
- global?: {
111
- max_iterations?: number;
112
- timeout_minutes?: number;
113
- };
114
- }
115
-
116
- interface HookInput {
117
- hook_event_name: string;
118
- subagent_type?: string;
119
- agent_output?: string;
120
- prompt?: string;
121
- cwd?: string;
122
- session_id?: string;
123
- // Team 模式扩展字段
124
- teammate_id?: string;
125
- teammate_role?: string;
126
- }
127
-
128
- interface HookOutput {
129
- decision: 'allow' | 'block';
130
- reason: string;
131
- systemMessage?: string;
132
- }
133
-
134
- // =============================================================================
135
- // 配置常量
136
- // =============================================================================
137
-
138
- const DEFAULT_MAX_ITERATIONS = 5;
139
- const DEFAULT_TIMEOUT_MINUTES = 30;
140
- const DEFAULT_MAX_ITERATIONS_PER_TEAMMATE = 3;
141
- const DEFAULT_MAX_GLOBAL_ITERATIONS = 10;
142
- const STATE_FILE = '.ralph-state.json';
143
- const CONFIG_FILE = '.claude/config/quality-gates.yml';
144
- const COMMAND_TIMEOUT_MS = 120000; // 2 minutes per command
145
-
146
- // =============================================================================
147
- // 工具函数
148
- // =============================================================================
149
-
150
- /**
151
- * 查找 Git 仓库根目录
152
- */
153
- function findRepoRoot(startPath: string): string | null {
154
- let current = path.resolve(startPath);
155
- while (current !== path.dirname(current)) {
156
- if (fs.existsSync(path.join(current, '.git'))) {
157
- return current;
158
- }
159
- current = path.dirname(current);
160
- }
161
- return null;
162
- }
163
-
164
- /**
165
- * 简单 YAML 解析器 (无外部依赖)
166
- * 只解析 verify 列表和 ralph_loop 配置
167
- */
168
- function parseSimpleYaml(content: string): QualityGatesConfig {
169
- const config: QualityGatesConfig = {};
170
- const lines = content.split('\n');
171
-
172
- let currentSection = '';
173
- let inVerifySection = false;
174
- let inRalphLoopSection = false;
175
- let inGlobalSection = false;
176
- let inTeamModeSection = false;
177
- let inTeammateVerifySection = false;
178
- let inGlobalVerifySection = false;
179
- let currentTeammateRole = '';
180
- const verifyCommands: (string | VerifyCommand)[] = [];
181
-
182
- for (const line of lines) {
183
- const trimmed = line.trim();
184
-
185
- // 跳过注释和空行
186
- if (trimmed.startsWith('#') || trimmed === '') continue;
187
-
188
- // 检测顶级 section
189
- if (!line.startsWith(' ') && !line.startsWith('\t') && trimmed.endsWith(':')) {
190
- currentSection = trimmed.slice(0, -1);
191
- inVerifySection = false;
192
- inRalphLoopSection = currentSection === 'ralph_loop';
193
- inGlobalSection = currentSection === 'global';
194
- inTeamModeSection = false;
195
- inTeammateVerifySection = false;
196
- inGlobalVerifySection = false;
197
- continue;
198
- }
199
-
200
- // 检测 verify 子 section (顶级)
201
- if (trimmed === 'verify:' && !inRalphLoopSection) {
202
- inVerifySection = true;
203
- continue;
204
- }
205
-
206
- // 解析 verify 列表项
207
- if (inVerifySection && trimmed.startsWith('- ')) {
208
- const value = trimmed.slice(2).trim();
209
- // 简单字符串命令
210
- if (!value.startsWith('name:') && !value.includes(':')) {
211
- verifyCommands.push(value);
212
- }
213
- continue;
214
- }
215
-
216
- // 解析 ralph_loop 配置
217
- if (inRalphLoopSection) {
218
- if (trimmed.startsWith('max_iterations:')) {
219
- const val = parseInt(trimmed.split(':')[1].trim(), 10);
220
- if (!isNaN(val)) {
221
- config.ralph_loop = config.ralph_loop || {};
222
- config.ralph_loop.max_iterations = val;
223
- }
224
- } else if (trimmed.startsWith('timeout_minutes:')) {
225
- const val = parseInt(trimmed.split(':')[1].trim(), 10);
226
- if (!isNaN(val)) {
227
- config.ralph_loop = config.ralph_loop || {};
228
- config.ralph_loop.timeout_minutes = val;
229
- }
230
- } else if (trimmed === 'team_mode:') {
231
- inTeamModeSection = true;
232
- config.ralph_loop = config.ralph_loop || {};
233
- config.ralph_loop.team_mode = {};
234
- }
235
- }
236
-
237
- // 解析 team_mode 配置
238
- if (inTeamModeSection) {
239
- if (trimmed.startsWith('enabled:')) {
240
- const val = trimmed.split(':')[1].trim();
241
- config.ralph_loop!.team_mode!.enabled = val === 'true';
242
- } else if (trimmed.startsWith('scope:')) {
243
- const val = trimmed.split(':')[1].trim() as 'teammate' | 'global';
244
- config.ralph_loop!.team_mode!.scope = val;
245
- } else if (trimmed.startsWith('max_iterations_per_teammate:')) {
246
- const val = parseInt(trimmed.split(':')[1].trim(), 10);
247
- if (!isNaN(val)) {
248
- config.ralph_loop!.team_mode!.max_iterations_per_teammate = val;
249
- }
250
- } else if (trimmed.startsWith('max_global_iterations:')) {
251
- const val = parseInt(trimmed.split(':')[1].trim(), 10);
252
- if (!isNaN(val)) {
253
- config.ralph_loop!.team_mode!.max_global_iterations = val;
254
- }
255
- } else if (trimmed === 'teammate_verify:') {
256
- inTeammateVerifySection = true;
257
- inGlobalVerifySection = false;
258
- config.ralph_loop!.team_mode!.teammate_verify = {};
259
- } else if (trimmed === 'global_verify:') {
260
- inGlobalVerifySection = true;
261
- inTeammateVerifySection = false;
262
- config.ralph_loop!.team_mode!.global_verify = [];
263
- }
264
- }
265
-
266
- // 解析 teammate_verify 配置
267
- if (inTeammateVerifySection && !inGlobalVerifySection) {
268
- // 检测 teammate role (e.g., "dev-frontend:")
269
- if (trimmed.endsWith(':') && !trimmed.startsWith('-')) {
270
- currentTeammateRole = trimmed.slice(0, -1);
271
- config.ralph_loop!.team_mode!.teammate_verify![currentTeammateRole] = [];
272
- } else if (trimmed.startsWith('- ') && currentTeammateRole) {
273
- const cmd = trimmed.slice(2).trim();
274
- config.ralph_loop!.team_mode!.teammate_verify![currentTeammateRole].push(cmd);
275
- }
276
- }
277
-
278
- // 解析 global_verify 配置
279
- if (inGlobalVerifySection && trimmed.startsWith('- ')) {
280
- const cmd = trimmed.slice(2).trim();
281
- config.ralph_loop!.team_mode!.global_verify!.push(cmd);
282
- }
283
-
284
- // 解析 global 配置
285
- if (inGlobalSection) {
286
- if (trimmed.startsWith('max_iterations:')) {
287
- const val = parseInt(trimmed.split(':')[1].trim(), 10);
288
- if (!isNaN(val)) {
289
- config.global = config.global || {};
290
- config.global.max_iterations = val;
291
- }
292
- } else if (trimmed.startsWith('timeout_minutes:')) {
293
- const val = parseInt(trimmed.split(':')[1].trim(), 10);
294
- if (!isNaN(val)) {
295
- config.global = config.global || {};
296
- config.global.timeout_minutes = val;
297
- }
298
- }
299
- }
300
- }
301
-
302
- if (verifyCommands.length > 0) {
303
- config.verify = verifyCommands;
304
- }
305
-
306
- return config;
307
- }
308
-
309
- /**
310
- * 读取 quality-gates.yml 配置
311
- */
312
- function loadConfig(repoRoot: string): QualityGatesConfig {
313
- const configPath = path.join(repoRoot, CONFIG_FILE);
314
- if (!fs.existsSync(configPath)) {
315
- return {};
316
- }
317
-
318
- try {
319
- const content = fs.readFileSync(configPath, 'utf-8');
320
- return parseSimpleYaml(content);
321
- } catch {
322
- return {};
323
- }
324
- }
325
-
326
- /**
327
- * 获取 verify 命令列表
328
- */
329
- function getVerifyCommands(config: QualityGatesConfig): string[] {
330
- if (!config.verify) {
331
- // 默认验证命令
332
- return [
333
- 'npm run lint --if-present',
334
- 'npm run typecheck --if-present',
335
- 'npm test -- --passWithNoTests'
336
- ];
337
- }
338
-
339
- return config.verify.map(cmd => {
340
- if (typeof cmd === 'string') return cmd;
341
- return cmd.command;
342
- });
343
- }
344
-
345
- /**
346
- * 获取最大迭代次数
347
- */
348
- function getMaxIterations(config: QualityGatesConfig): number {
349
- return config.ralph_loop?.max_iterations
350
- ?? config.global?.max_iterations
351
- ?? DEFAULT_MAX_ITERATIONS;
352
- }
353
-
354
- /**
355
- * 获取超时时间 (分钟)
356
- */
357
- function getTimeoutMinutes(config: QualityGatesConfig): number {
358
- return config.ralph_loop?.timeout_minutes
359
- ?? config.global?.timeout_minutes
360
- ?? DEFAULT_TIMEOUT_MINUTES;
361
- }
362
-
363
- /**
364
- * 加载 Ralph 状态
365
- */
366
- function loadState(repoRoot: string, agentId: string): RalphState {
367
- const statePath = path.join(repoRoot, STATE_FILE);
368
-
369
- if (!fs.existsSync(statePath)) {
370
- return createNewState(agentId);
371
- }
372
-
373
- try {
374
- const content = fs.readFileSync(statePath, 'utf-8');
375
- const state: RalphState = JSON.parse(content);
376
-
377
- // 检查是否是同一个 agent 会话
378
- if (state.agent_id !== agentId) {
379
- return createNewState(agentId);
380
- }
381
-
382
- return state;
383
- } catch {
384
- return createNewState(agentId);
385
- }
386
- }
387
-
388
- /**
389
- * 创建新状态
390
- */
391
- function createNewState(agentId: string): RalphState {
392
- return {
393
- agent_id: agentId,
394
- iteration: 0,
395
- last_failures: [],
396
- started_at: new Date().toISOString()
397
- };
398
- }
399
-
400
- /**
401
- * 保存 Ralph 状态
402
- */
403
- function saveState(repoRoot: string, state: RalphState): void {
404
- const statePath = path.join(repoRoot, STATE_FILE);
405
- try {
406
- fs.writeFileSync(statePath, JSON.stringify(state, null, 2), 'utf-8');
407
- } catch {
408
- // 忽略写入错误
409
- }
410
- }
411
-
412
- /**
413
- * 清理 Ralph 状态
414
- */
415
- function clearState(repoRoot: string): void {
416
- const statePath = path.join(repoRoot, STATE_FILE);
417
- try {
418
- if (fs.existsSync(statePath)) {
419
- fs.unlinkSync(statePath);
420
- }
421
- } catch {
422
- // 忽略删除错误
423
- }
424
- }
425
-
426
- /**
427
- * 检查是否超时
428
- */
429
- function isTimedOut(state: RalphState, timeoutMinutes: number): boolean {
430
- const startedAt = new Date(state.started_at);
431
- const now = new Date();
432
- const elapsedMinutes = (now.getTime() - startedAt.getTime()) / (1000 * 60);
433
- return elapsedMinutes >= timeoutMinutes;
434
- }
435
-
436
- /**
437
- * 执行验证命令
438
- */
439
- function runVerifyCommands(
440
- repoRoot: string,
441
- commands: string[]
442
- ): { passed: boolean; failures: FailureRecord[] } {
443
- const failures: FailureRecord[] = [];
444
-
445
- for (const cmd of commands) {
446
- try {
447
- execSync(cmd, {
448
- cwd: repoRoot,
449
- timeout: COMMAND_TIMEOUT_MS,
450
- stdio: ['pipe', 'pipe', 'pipe'],
451
- encoding: 'utf-8'
452
- });
453
- } catch (error: unknown) {
454
- let output = '';
455
- if (error && typeof error === 'object') {
456
- const execError = error as { stderr?: string; stdout?: string; message?: string };
457
- output = execError.stderr || execError.stdout || execError.message || 'Unknown error';
458
- }
459
-
460
- // 截断过长输出
461
- if (output.length > 500) {
462
- output = output.slice(0, 500) + '...';
463
- }
464
-
465
- failures.push({
466
- command: cmd,
467
- output,
468
- timestamp: new Date().toISOString()
469
- });
470
- }
471
- }
472
-
473
- return {
474
- passed: failures.length === 0,
475
- failures
476
- };
477
- }
478
-
479
- /**
480
- * 格式化失败信息
481
- */
482
- function formatFailures(failures: FailureRecord[]): string {
483
- return failures.map(f => `Command: ${f.command}\nOutput: ${f.output}`).join('\n\n');
484
- }
485
-
486
- // =============================================================================
487
- // Team 模式工具函数
488
- // =============================================================================
489
-
490
- /**
491
- * 查找当前需求的 orchestration_status.json
492
- */
493
- function findOrchestrationStatus(repoRoot: string): string | null {
494
- const devflowDir = path.join(repoRoot, 'devflow', 'requirements');
495
- if (!fs.existsSync(devflowDir)) return null;
496
-
497
- // 查找最近修改的需求目录
498
- const reqDirs = fs.readdirSync(devflowDir)
499
- .filter(d => d.startsWith('REQ-') || d.startsWith('BUG-'))
500
- .map(d => ({
501
- name: d,
502
- path: path.join(devflowDir, d),
503
- mtime: fs.statSync(path.join(devflowDir, d)).mtime.getTime()
504
- }))
505
- .sort((a, b) => b.mtime - a.mtime);
506
-
507
- for (const reqDir of reqDirs) {
508
- const statusFile = path.join(reqDir.path, 'orchestration_status.json');
509
- if (fs.existsSync(statusFile)) {
510
- return statusFile;
511
- }
512
- }
513
-
514
- return null;
515
- }
516
-
517
- /**
518
- * 加载 orchestration_status.json
519
- */
520
- function loadOrchestrationStatus(statusPath: string): OrchestrationStatus | null {
521
- try {
522
- const content = fs.readFileSync(statusPath, 'utf-8');
523
- return JSON.parse(content);
524
- } catch {
525
- return null;
526
- }
527
- }
528
-
529
- /**
530
- * 保存 orchestration_status.json
531
- */
532
- function saveOrchestrationStatus(statusPath: string, status: OrchestrationStatus): void {
533
- try {
534
- status.updatedAt = new Date().toISOString();
535
- fs.writeFileSync(statusPath, JSON.stringify(status, null, 2), 'utf-8');
536
- } catch {
537
- // 忽略写入错误
538
- }
539
- }
540
-
541
- /**
542
- * 检查是否启用 Team 模式
543
- */
544
- function isTeamModeEnabled(status: OrchestrationStatus | null, config: QualityGatesConfig): boolean {
545
- // 检查 orchestration_status.json 中是否有 team 状态
546
- if (!status?.team) return false;
547
-
548
- // 检查配置中是否启用 team_mode
549
- return config.ralph_loop?.team_mode?.enabled ?? false;
550
- }
551
-
552
- /**
553
- * 获取 Teammate 特定的验证命令
554
- */
555
- function getTeammateVerifyCommands(
556
- config: QualityGatesConfig,
557
- teammateRole: string
558
- ): string[] {
559
- const teamModeConfig = config.ralph_loop?.team_mode;
560
-
561
- // 优先使用 teammate 特定命令
562
- if (teamModeConfig?.teammate_verify?.[teammateRole]) {
563
- return teamModeConfig.teammate_verify[teammateRole];
564
- }
565
-
566
- // 回退到默认验证命令
567
- return getVerifyCommands(config);
568
- }
569
-
570
- /**
571
- * 获取全局验证命令
572
- */
573
- function getGlobalVerifyCommands(config: QualityGatesConfig): string[] {
574
- const teamModeConfig = config.ralph_loop?.team_mode;
575
-
576
- if (teamModeConfig?.global_verify && teamModeConfig.global_verify.length > 0) {
577
- return teamModeConfig.global_verify;
578
- }
579
-
580
- // 回退到默认验证命令
581
- return getVerifyCommands(config);
582
- }
583
-
584
- /**
585
- * 检查是否是最后一个活跃的 Teammate
586
- */
587
- function isLastActiveTeammate(
588
- status: OrchestrationStatus,
589
- currentTeammateId: string
590
- ): boolean {
591
- if (!status.team?.teammates) return true;
592
-
593
- const activeTeammates = status.team.teammates.filter(
594
- t => t.status === 'working' && t.id !== currentTeammateId
595
- );
596
-
597
- return activeTeammates.length === 0;
598
- }
599
-
600
- /**
601
- * 初始化 Teammate 的 Ralph 状态
602
- */
603
- function initTeammateRalphState(): TeammateRalphState {
604
- return {
605
- iteration: 0,
606
- lastVerifyResult: 'skipped'
607
- };
608
- }
609
-
610
- /**
611
- * 更新 Teammate 的 Ralph 状态
612
- */
613
- function updateTeammateRalphState(
614
- status: OrchestrationStatus,
615
- teammateId: string,
616
- verifyResult: 'passed' | 'failed',
617
- failures?: FailureRecord[]
618
- ): void {
619
- if (!status.ralphLoop) {
620
- status.ralphLoop = {
621
- enabled: true,
622
- teammates: {},
623
- globalIteration: 0,
624
- maxIterations: DEFAULT_MAX_GLOBAL_ITERATIONS,
625
- startedAt: new Date().toISOString()
626
- };
627
- }
628
-
629
- if (!status.ralphLoop.teammates[teammateId]) {
630
- status.ralphLoop.teammates[teammateId] = initTeammateRalphState();
631
- }
632
-
633
- const teammateState = status.ralphLoop.teammates[teammateId];
634
- teammateState.iteration += 1;
635
- teammateState.lastVerifyResult = verifyResult;
636
- teammateState.lastVerifyAt = new Date().toISOString();
637
-
638
- if (failures && failures.length > 0) {
639
- teammateState.lastFailures = failures;
640
- } else {
641
- delete teammateState.lastFailures;
642
- }
643
-
644
- // 更新全局迭代计数
645
- status.ralphLoop.globalIteration += 1;
646
- }
647
-
648
- /**
649
- * 获取 Teammate 的迭代次数
650
- */
651
- function getTeammateIteration(status: OrchestrationStatus, teammateId: string): number {
652
- return status.ralphLoop?.teammates?.[teammateId]?.iteration ?? 0;
653
- }
654
-
655
- /**
656
- * 获取全局迭代次数
657
- */
658
- function getGlobalIteration(status: OrchestrationStatus): number {
659
- return status.ralphLoop?.globalIteration ?? 0;
660
- }
661
-
662
- /**
663
- * 获取 Teammate 最大迭代次数
664
- */
665
- function getMaxIterationsPerTeammate(config: QualityGatesConfig): number {
666
- return config.ralph_loop?.team_mode?.max_iterations_per_teammate
667
- ?? DEFAULT_MAX_ITERATIONS_PER_TEAMMATE;
668
- }
669
-
670
- /**
671
- * 获取全局最大迭代次数
672
- */
673
- function getMaxGlobalIterations(config: QualityGatesConfig): number {
674
- return config.ralph_loop?.team_mode?.max_global_iterations
675
- ?? DEFAULT_MAX_GLOBAL_ITERATIONS;
676
- }
677
-
678
- // =============================================================================
679
- // 主函数
680
- // =============================================================================
681
-
682
- function main(): void {
683
- let inputData: HookInput;
684
-
685
- try {
686
- const stdin = fs.readFileSync(0, 'utf-8');
687
- inputData = JSON.parse(stdin);
688
- } catch {
689
- // 无法解析输入,允许停止
690
- process.exit(0);
691
- }
692
-
693
- const hookEvent = inputData.hook_event_name || '';
694
-
695
- // 只处理 SubagentStop 事件
696
- if (hookEvent !== 'SubagentStop') {
697
- process.exit(0);
698
- }
699
-
700
- const cwd = inputData.cwd || process.cwd();
701
- const agentId = inputData.session_id || 'default';
702
- const teammateId = inputData.teammate_id;
703
- const teammateRole = inputData.teammate_role || 'default';
704
-
705
- // 查找仓库根目录
706
- const repoRoot = findRepoRoot(cwd);
707
- if (!repoRoot) {
708
- process.exit(0);
709
- }
710
-
711
- // 加载配置
712
- const config = loadConfig(repoRoot);
713
-
714
- // 查找 orchestration_status.json
715
- const statusPath = findOrchestrationStatus(repoRoot);
716
- const orchestrationStatus = statusPath ? loadOrchestrationStatus(statusPath) : null;
717
-
718
- // 检查是否启用 Team 模式
719
- if (teammateId && isTeamModeEnabled(orchestrationStatus, config)) {
720
- handleTeamMode(
721
- repoRoot,
722
- config,
723
- statusPath!,
724
- orchestrationStatus!,
725
- teammateId,
726
- teammateRole
727
- );
728
- } else {
729
- handleSingleAgentMode(repoRoot, config, agentId);
730
- }
731
- }
732
-
733
- /**
734
- * 处理 Team 模式的 SubagentStop
735
- */
736
- function handleTeamMode(
737
- repoRoot: string,
738
- config: QualityGatesConfig,
739
- statusPath: string,
740
- status: OrchestrationStatus,
741
- teammateId: string,
742
- teammateRole: string
743
- ): void {
744
- const maxIterationsPerTeammate = getMaxIterationsPerTeammate(config);
745
- const maxGlobalIterations = getMaxGlobalIterations(config);
746
- const timeoutMinutes = getTimeoutMinutes(config);
747
-
748
- // 检查超时
749
- if (status.ralphLoop?.startedAt) {
750
- const startedAt = new Date(status.ralphLoop.startedAt);
751
- const now = new Date();
752
- const elapsedMinutes = (now.getTime() - startedAt.getTime()) / (1000 * 60);
753
- if (elapsedMinutes >= timeoutMinutes) {
754
- const output: HookOutput = {
755
- decision: 'allow',
756
- reason: `Team mode timeout (${timeoutMinutes} minutes) reached. Allowing stop.`
757
- };
758
- console.log(JSON.stringify(output, null, 0));
759
- process.exit(0);
760
- }
761
- }
762
-
763
- // 检查 Teammate 迭代次数
764
- const teammateIteration = getTeammateIteration(status, teammateId);
765
- if (teammateIteration >= maxIterationsPerTeammate) {
766
- const output: HookOutput = {
767
- decision: 'allow',
768
- reason: `Teammate ${teammateId} max iterations (${maxIterationsPerTeammate}) reached. Allowing stop.`
769
- };
770
- console.log(JSON.stringify(output, null, 0));
771
- process.exit(0);
772
- }
773
-
774
- // 检查全局迭代次数
775
- const globalIteration = getGlobalIteration(status);
776
- if (globalIteration >= maxGlobalIterations) {
777
- const output: HookOutput = {
778
- decision: 'allow',
779
- reason: `Global max iterations (${maxGlobalIterations}) reached. Allowing stop.`
780
- };
781
- console.log(JSON.stringify(output, null, 0));
782
- process.exit(0);
783
- }
784
-
785
- // 获取验证命令
786
- const verifyCommands = getTeammateVerifyCommands(config, teammateRole);
787
-
788
- // 执行 Teammate 级别验证
789
- const { passed: teammatePassed, failures: teammateFailures } = runVerifyCommands(
790
- repoRoot,
791
- verifyCommands
792
- );
793
-
794
- // 更新 Teammate Ralph 状态
795
- updateTeammateRalphState(
796
- status,
797
- teammateId,
798
- teammatePassed ? 'passed' : 'failed',
799
- teammateFailures
800
- );
801
-
802
- if (!teammatePassed) {
803
- // Teammate 验证失败
804
- saveOrchestrationStatus(statusPath, status);
805
-
806
- const failureDetails = formatFailures(teammateFailures);
807
- const currentIteration = getTeammateIteration(status, teammateId);
808
- const output: HookOutput = {
809
- decision: 'block',
810
- reason: `Teammate ${teammateId} iteration ${currentIteration}/${maxIterationsPerTeammate}. Verification failed:\n\n${failureDetails}\n\nPlease fix the issues and try again.`,
811
- systemMessage: `Ralph Loop (Team Mode) - Teammate ${teammateId} iteration ${currentIteration}/${maxIterationsPerTeammate}`
812
- };
813
- console.log(JSON.stringify(output, null, 0));
814
- process.exit(0);
815
- }
816
-
817
- // Teammate 验证通过,检查是否是最后一个活跃的 Teammate
818
- const isLast = isLastActiveTeammate(status, teammateId);
819
-
820
- if (isLast) {
821
- // 执行全局验证
822
- const globalVerifyCommands = getGlobalVerifyCommands(config);
823
- const { passed: globalPassed, failures: globalFailures } = runVerifyCommands(
824
- repoRoot,
825
- globalVerifyCommands
826
- );
827
-
828
- if (!globalPassed) {
829
- // 全局验证失败
830
- saveOrchestrationStatus(statusPath, status);
831
-
832
- const failureDetails = formatFailures(globalFailures);
833
- const output: HookOutput = {
834
- decision: 'block',
835
- reason: `Global verification failed (last teammate ${teammateId}):\n\n${failureDetails}\n\nPlease fix the issues and try again.`,
836
- systemMessage: `Ralph Loop (Team Mode) - Global verification failed`
837
- };
838
- console.log(JSON.stringify(output, null, 0));
839
- process.exit(0);
840
- }
841
-
842
- // 全局验证通过
843
- saveOrchestrationStatus(statusPath, status);
844
- const output: HookOutput = {
845
- decision: 'allow',
846
- reason: `All verifications passed. Teammate ${teammateId} (last active) completed successfully.`
847
- };
848
- console.log(JSON.stringify(output, null, 0));
849
- process.exit(0);
850
- }
851
-
852
- // 不是最后一个 Teammate,允许停止
853
- saveOrchestrationStatus(statusPath, status);
854
- const output: HookOutput = {
855
- decision: 'allow',
856
- reason: `Teammate ${teammateId} verification passed. Other teammates still active.`
857
- };
858
- console.log(JSON.stringify(output, null, 0));
859
- process.exit(0);
860
- }
861
-
862
- /**
863
- * 处理单 Agent 模式的 SubagentStop (原有逻辑)
864
- */
865
- function handleSingleAgentMode(
866
- repoRoot: string,
867
- config: QualityGatesConfig,
868
- agentId: string
869
- ): void {
870
- const maxIterations = getMaxIterations(config);
871
- const timeoutMinutes = getTimeoutMinutes(config);
872
- const verifyCommands = getVerifyCommands(config);
873
-
874
- // 加载状态
875
- const state = loadState(repoRoot, agentId);
876
-
877
- // 检查超时
878
- if (isTimedOut(state, timeoutMinutes)) {
879
- clearState(repoRoot);
880
- const output: HookOutput = {
881
- decision: 'allow',
882
- reason: `Timeout (${timeoutMinutes} minutes) reached. Stopping Ralph loop.`
883
- };
884
- console.log(JSON.stringify(output, null, 0));
885
- process.exit(0);
886
- }
887
-
888
- // 增加迭代计数
889
- state.iteration += 1;
890
-
891
- // 检查最大迭代次数
892
- if (state.iteration >= maxIterations) {
893
- clearState(repoRoot);
894
- const output: HookOutput = {
895
- decision: 'allow',
896
- reason: `Max iterations (${maxIterations}) reached. Stopping to prevent infinite loop.`
897
- };
898
- console.log(JSON.stringify(output, null, 0));
899
- process.exit(0);
900
- }
901
-
902
- // 执行验证命令
903
- const { passed, failures } = runVerifyCommands(repoRoot, verifyCommands);
904
-
905
- if (passed) {
906
- // 所有验证通过,允许停止
907
- clearState(repoRoot);
908
- const output: HookOutput = {
909
- decision: 'allow',
910
- reason: 'All verify commands passed. Quality check complete.'
911
- };
912
- console.log(JSON.stringify(output, null, 0));
913
- process.exit(0);
914
- }
915
-
916
- // 验证失败,更新状态并阻止停止
917
- state.last_failures = failures;
918
- saveState(repoRoot, state);
919
-
920
- const failureDetails = formatFailures(failures);
921
- const output: HookOutput = {
922
- decision: 'block',
923
- reason: `Iteration ${state.iteration}/${maxIterations}. Verification failed:\n\n${failureDetails}\n\nPlease fix the issues and try again.`,
924
- systemMessage: `Ralph Loop iteration ${state.iteration}/${maxIterations} - Fix the errors above to proceed.`
925
- };
926
-
927
- console.log(JSON.stringify(output, null, 0));
928
- process.exit(0);
929
- }
930
-
931
- main();