agentic-orchestrator 0.1.28 → 0.2.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 (836) hide show
  1. package/.claude/settings.local.json +46 -1
  2. package/.cortexrc +28 -0
  3. package/.github/agents/copilot-instructions.md +29 -0
  4. package/.github/copilot-instructions.md +93 -0
  5. package/.vscode/settings.json +13 -0
  6. package/.vscode/tms.code-snippets +223 -0
  7. package/AGENTS.md +72 -1
  8. package/Agentic-Orchestrator.iml +12 -11
  9. package/CLAUDE.md +72 -1
  10. package/CONSTITUTION.md +504 -0
  11. package/FUTURE-ENHANCEMENTS.md +85 -0
  12. package/NEXT-TASKS.md +25 -0
  13. package/PROMPTS.md +161 -0
  14. package/README.md +126 -29
  15. package/agentic/orchestrator/agents.yaml +4 -3
  16. package/agentic/orchestrator/defaults/policy.defaults.yaml +39 -3
  17. package/agentic/orchestrator/gates.yaml +15 -3
  18. package/agentic/orchestrator/policy.yaml +47 -3
  19. package/agentic/orchestrator/prompts/builder.system.md +69 -20
  20. package/agentic/orchestrator/prompts/planner-intake.system.md +149 -0
  21. package/agentic/orchestrator/prompts/planner.system.md +113 -40
  22. package/agentic/orchestrator/prompts/qa.system.md +73 -18
  23. package/agentic/orchestrator/prompts/reconciler.system.md +119 -0
  24. package/agentic/orchestrator/schemas/agents.schema.json +89 -1
  25. package/agentic/orchestrator/schemas/execution-control.schema.json +242 -0
  26. package/agentic/orchestrator/schemas/index.schema.json +234 -0
  27. package/agentic/orchestrator/schemas/intake.review.schema.json +82 -0
  28. package/agentic/orchestrator/schemas/organizer-ordering-artifact.schema.json +75 -0
  29. package/agentic/orchestrator/schemas/plan.schema.json +44 -0
  30. package/agentic/orchestrator/schemas/policy.schema.json +238 -9
  31. package/agentic/orchestrator/schemas/policy.user.schema.json +129 -1
  32. package/agentic/orchestrator/schemas/spec.manifest.bootstrap.schema.json +101 -0
  33. package/agentic/orchestrator/schemas/spec.manifest.verified.schema.json +80 -0
  34. package/agentic/orchestrator/schemas/state.schema.json +298 -3
  35. package/agentic/orchestrator/tools/catalog.json +145 -15
  36. package/agentic/orchestrator/tools/schemas/input/doctor.run.input.schema.json +18 -0
  37. package/agentic/orchestrator/tools/schemas/input/evidence.latest.input.schema.json +4 -0
  38. package/agentic/orchestrator/tools/schemas/input/evidence.verify_chain.input.schema.json +13 -0
  39. package/agentic/orchestrator/tools/schemas/input/feature.intake_submit.input.schema.json +11 -0
  40. package/agentic/orchestrator/tools/schemas/input/feature.question_answer.input.schema.json +15 -0
  41. package/agentic/orchestrator/tools/schemas/input/feature.question_create.input.schema.json +21 -0
  42. package/agentic/orchestrator/tools/schemas/input/feature.question_list.input.schema.json +13 -0
  43. package/agentic/orchestrator/tools/schemas/input/feature.ready_to_merge.input.schema.json +5 -0
  44. package/agentic/orchestrator/tools/schemas/input/feature.send_message.input.schema.json +1 -1
  45. package/agentic/orchestrator/tools/schemas/input/replay.timeline_get.input.schema.json +32 -0
  46. package/agentic/orchestrator/tools/schemas/input/repo.conflict_abort.input.schema.json +16 -0
  47. package/agentic/orchestrator/tools/schemas/input/repo.conflict_files.input.schema.json +16 -0
  48. package/agentic/orchestrator/tools/schemas/input/repo.reconcile_mainline.input.schema.json +37 -0
  49. package/agentic/orchestrator/tools/schemas/input/repo.resolve_conflict.input.schema.json +40 -0
  50. package/agentic/orchestrator/tools/schemas/input/runtime.execution_request_list.input.schema.json +7 -0
  51. package/agentic/orchestrator/tools/schemas/input/runtime.execution_request_submit.input.schema.json +25 -0
  52. package/agentic/orchestrator/tools/schemas/output/doctor.run.output.schema.json +34 -0
  53. package/agentic/orchestrator/tools/schemas/output/evidence.verify_chain.output.schema.json +23 -0
  54. package/agentic/orchestrator/tools/schemas/output/feature.get_context.output.schema.json +62 -2
  55. package/agentic/orchestrator/tools/schemas/output/feature.intake_submit.output.schema.json +24 -0
  56. package/agentic/orchestrator/tools/schemas/output/feature.question_answer.output.schema.json +21 -0
  57. package/agentic/orchestrator/tools/schemas/output/feature.question_create.output.schema.json +12 -0
  58. package/agentic/orchestrator/tools/schemas/output/feature.question_list.output.schema.json +14 -0
  59. package/agentic/orchestrator/tools/schemas/output/feature.ready_to_merge.output.schema.json +31 -0
  60. package/agentic/orchestrator/tools/schemas/output/feature.send_message.output.schema.json +8 -18
  61. package/agentic/orchestrator/tools/schemas/output/replay.timeline_get.output.schema.json +64 -0
  62. package/agentic/orchestrator/tools/schemas/output/repo.conflict_abort.output.schema.json +16 -0
  63. package/agentic/orchestrator/tools/schemas/output/repo.conflict_files.output.schema.json +22 -0
  64. package/agentic/orchestrator/tools/schemas/output/repo.reconcile_mainline.output.schema.json +61 -0
  65. package/agentic/orchestrator/tools/schemas/output/repo.resolve_conflict.output.schema.json +19 -0
  66. package/agentic/orchestrator/tools/schemas/output/report.dashboard.output.schema.json +26 -0
  67. package/agentic/orchestrator/tools/schemas/output/runtime.execution_request_list.output.schema.json +17 -0
  68. package/agentic/orchestrator/tools/schemas/output/runtime.execution_request_submit.output.schema.json +24 -0
  69. package/agentic/orchestrator/tools.md +13 -0
  70. package/apps/control-plane/scripts/validate-mcp-contracts.ts +1 -1
  71. package/apps/control-plane/src/application/kernel-tool-wiring.ts +140 -2
  72. package/apps/control-plane/src/application/services/activity-monitor-service.ts +44 -1
  73. package/apps/control-plane/src/application/services/bootstrap-manifest-generator-service.ts +251 -0
  74. package/apps/control-plane/src/application/services/checkpoint-service.ts +87 -27
  75. package/apps/control-plane/src/application/services/collision-override-service.ts +906 -0
  76. package/apps/control-plane/src/application/services/collision-queue-service.ts +129 -38
  77. package/apps/control-plane/src/application/services/cost-tracking-service.ts +94 -0
  78. package/apps/control-plane/src/application/services/execution-control-service.ts +599 -0
  79. package/apps/control-plane/src/application/services/feature-deletion-service.ts +37 -1
  80. package/apps/control-plane/src/application/services/feature-lifecycle-service.ts +182 -4
  81. package/apps/control-plane/src/application/services/feature-send-message-service.ts +17 -8
  82. package/apps/control-plane/src/application/services/feature-state-service.ts +191 -6
  83. package/apps/control-plane/src/application/services/gate-service.ts +121 -2
  84. package/apps/control-plane/src/application/services/git-reconciliation-service.ts +1591 -0
  85. package/apps/control-plane/src/application/services/intake-service.ts +1468 -0
  86. package/apps/control-plane/src/application/services/merge-service.ts +308 -17
  87. package/apps/control-plane/src/application/services/notifier-service.ts +3 -1
  88. package/apps/control-plane/src/application/services/performance-analytics-service.ts +75 -0
  89. package/apps/control-plane/src/application/services/plan-service.ts +336 -20
  90. package/apps/control-plane/src/application/services/question-service.ts +693 -0
  91. package/apps/control-plane/src/application/services/reactions-service.ts +73 -17
  92. package/apps/control-plane/src/application/services/replay-timeline-service.ts +295 -0
  93. package/apps/control-plane/src/application/services/reporting-service.ts +194 -10
  94. package/apps/control-plane/src/application/services/run-lease-service.ts +121 -5
  95. package/apps/control-plane/src/application/services/worktree-watchdog-service.ts +95 -8
  96. package/apps/control-plane/src/application/tools/tool-metadata.ts +7 -0
  97. package/apps/control-plane/src/application/usage-types.ts +138 -0
  98. package/apps/control-plane/src/cli/add-command-handler.ts +162 -0
  99. package/apps/control-plane/src/cli/answer-command-handler.ts +113 -0
  100. package/apps/control-plane/src/cli/attach-command-handler.ts +12 -3
  101. package/apps/control-plane/src/cli/cli-argument-parser.ts +133 -11
  102. package/apps/control-plane/src/cli/collision-command-handler.ts +113 -0
  103. package/apps/control-plane/src/cli/command-catalog.ts +479 -0
  104. package/apps/control-plane/src/cli/complete-command-handler.ts +23 -0
  105. package/apps/control-plane/src/cli/completion-command-handler.ts +25 -0
  106. package/apps/control-plane/src/cli/completion-resolver.ts +319 -0
  107. package/apps/control-plane/src/cli/completion-shell-renderer.ts +58 -0
  108. package/apps/control-plane/src/cli/dashboard-command-handler.ts +110 -1
  109. package/apps/control-plane/src/cli/dashboard-runtime-runner.ts +1036 -0
  110. package/apps/control-plane/src/cli/dashboard-runtime.ts +31 -0
  111. package/apps/control-plane/src/cli/help-command-handler.ts +17 -185
  112. package/apps/control-plane/src/cli/init-command-handler.ts +51 -6
  113. package/apps/control-plane/src/cli/merge-command-handler.ts +200 -0
  114. package/apps/control-plane/src/cli/questions-command-handler.ts +70 -0
  115. package/apps/control-plane/src/cli/replay-command-handler.ts +98 -0
  116. package/apps/control-plane/src/cli/resume-command-handler.ts +231 -16
  117. package/apps/control-plane/src/cli/retry-command-handler.ts +229 -17
  118. package/apps/control-plane/src/cli/retry-resume-decision.ts +75 -0
  119. package/apps/control-plane/src/cli/rollback-command-handler.ts +4 -2
  120. package/apps/control-plane/src/cli/run-command-handler.ts +35 -1
  121. package/apps/control-plane/src/cli/spec-ingestion-service.ts +45 -55
  122. package/apps/control-plane/src/cli/spec-preparation.ts +114 -0
  123. package/apps/control-plane/src/cli/spec-utils.ts +90 -11
  124. package/apps/control-plane/src/cli/status-command-handler.ts +122 -0
  125. package/apps/control-plane/src/cli/types.ts +41 -3
  126. package/apps/control-plane/src/core/collisions.ts +150 -31
  127. package/apps/control-plane/src/core/constants.ts +18 -1
  128. package/apps/control-plane/src/core/error-codes.ts +39 -0
  129. package/apps/control-plane/src/core/execution-control.ts +56 -0
  130. package/apps/control-plane/src/core/feature-resume-phase.ts +118 -0
  131. package/apps/control-plane/src/core/gate-freshness.ts +359 -0
  132. package/apps/control-plane/src/core/gate-log-extractor.ts +97 -0
  133. package/apps/control-plane/src/core/gates.ts +90 -1
  134. package/apps/control-plane/src/core/intake-artifacts.ts +295 -0
  135. package/apps/control-plane/src/core/kernel-types.ts +11 -0
  136. package/apps/control-plane/src/core/kernel.ts +604 -16
  137. package/apps/control-plane/src/core/mainline-conflict.ts +22 -0
  138. package/apps/control-plane/src/core/merge-repair.ts +149 -0
  139. package/apps/control-plane/src/core/path-layout.ts +46 -2
  140. package/apps/control-plane/src/core/path-rules.ts +11 -3
  141. package/apps/control-plane/src/core/plan-submit-recovery.ts +130 -0
  142. package/apps/control-plane/src/core/questions.ts +49 -0
  143. package/apps/control-plane/src/core/runtime-sessions.ts +4 -0
  144. package/apps/control-plane/src/core/schemas.ts +40 -1
  145. package/apps/control-plane/src/core/tool-caller.ts +25 -1
  146. package/apps/control-plane/src/core/utils/index-normalizer.ts +25 -4
  147. package/apps/control-plane/src/core/worktree-diff.ts +66 -0
  148. package/apps/control-plane/src/index.ts +29 -1
  149. package/apps/control-plane/src/interfaces/cli/bootstrap.ts +300 -6
  150. package/apps/control-plane/src/mcp/kernel-tool-executor.ts +17 -0
  151. package/apps/control-plane/src/mcp/tool-runtime.ts +63 -4
  152. package/apps/control-plane/src/providers/api-worker-provider.ts +62 -15
  153. package/apps/control-plane/src/providers/cli-worker-provider.ts +1037 -61
  154. package/apps/control-plane/src/providers/output-parsers/generic-output-parser.ts +99 -1
  155. package/apps/control-plane/src/providers/output-parsers/types.ts +2 -0
  156. package/apps/control-plane/src/providers/provider-defaults.ts +116 -7
  157. package/apps/control-plane/src/providers/providers.ts +225 -21
  158. package/apps/control-plane/src/providers/worker-provider-factory.ts +26 -2
  159. package/apps/control-plane/src/supervisor/artifact-stager.ts +52 -0
  160. package/apps/control-plane/src/supervisor/build-wave-executor.ts +477 -166
  161. package/apps/control-plane/src/supervisor/execution-enrollment-service.ts +408 -0
  162. package/apps/control-plane/src/supervisor/organizer-enrollment-scheduler.ts +117 -0
  163. package/apps/control-plane/src/supervisor/organizer-sidecar-service.ts +394 -0
  164. package/apps/control-plane/src/supervisor/plan-conformance-scorer.ts +2 -5
  165. package/apps/control-plane/src/supervisor/planner-phase.ts +85 -0
  166. package/apps/control-plane/src/supervisor/planning-wave-executor.ts +993 -64
  167. package/apps/control-plane/src/supervisor/prompt-bundle-loader.ts +20 -1
  168. package/apps/control-plane/src/supervisor/qa-wave-executor.ts +384 -177
  169. package/apps/control-plane/src/supervisor/run-coordinator.ts +723 -20
  170. package/apps/control-plane/src/supervisor/runtime.ts +485 -9
  171. package/apps/control-plane/src/supervisor/session-orchestrator.ts +220 -1
  172. package/apps/control-plane/src/supervisor/types.ts +152 -1
  173. package/apps/control-plane/src/supervisor/worker-decision-loop.ts +1030 -92
  174. package/apps/control-plane/test/activity-monitor.spec.ts +76 -0
  175. package/apps/control-plane/test/add-command-handler.spec.ts +189 -0
  176. package/apps/control-plane/test/application/services/feature-state-service.spec.ts +208 -0
  177. package/apps/control-plane/test/artifact-stager.spec.ts +93 -0
  178. package/apps/control-plane/test/batch-operations.spec.ts +58 -0
  179. package/apps/control-plane/test/bootstrap-edge-cases.spec.ts +50 -2
  180. package/apps/control-plane/test/bootstrap-manifest-generator-service.spec.ts +99 -0
  181. package/apps/control-plane/test/bootstrap.spec.ts +177 -4
  182. package/apps/control-plane/test/checkpoint-service.spec.ts +977 -29
  183. package/apps/control-plane/test/cli-argument-parser.spec.ts +119 -0
  184. package/apps/control-plane/test/cli-helpers.spec.ts +1202 -12
  185. package/apps/control-plane/test/cli.unit.spec.ts +797 -16
  186. package/apps/control-plane/test/collision-command-handler.spec.ts +182 -0
  187. package/apps/control-plane/test/collision-override-service.spec.ts +878 -0
  188. package/apps/control-plane/test/collision-queue.spec.ts +430 -2
  189. package/apps/control-plane/test/collisions.spec.ts +209 -1
  190. package/apps/control-plane/test/core-utils.spec.ts +61 -0
  191. package/apps/control-plane/test/cost-tracking.spec.ts +224 -0
  192. package/apps/control-plane/test/dashboard-api.integration.spec.ts +185 -5
  193. package/apps/control-plane/test/dashboard-client.spec.ts +948 -0
  194. package/apps/control-plane/test/dashboard-command.spec.ts +138 -6
  195. package/apps/control-plane/test/dashboard-runtime-runner.spec.ts +1550 -0
  196. package/apps/control-plane/test/dashboard-runtime.spec.ts +138 -0
  197. package/apps/control-plane/test/dashboard-ui-utils.spec.ts +56 -12
  198. package/apps/control-plane/test/dependency-scheduler.spec.ts +7 -1
  199. package/apps/control-plane/test/env-file.spec.ts +76 -0
  200. package/apps/control-plane/test/execution-control-service.spec.ts +535 -0
  201. package/apps/control-plane/test/execution-enrollment-service.spec.ts +648 -0
  202. package/apps/control-plane/test/feature-lifecycle.spec.ts +126 -0
  203. package/apps/control-plane/test/feature-resume-phase.spec.ts +164 -0
  204. package/apps/control-plane/test/feature-send-message-service.spec.ts +161 -0
  205. package/apps/control-plane/test/feature-state-service.spec.ts +295 -0
  206. package/apps/control-plane/test/fs.spec.ts +80 -0
  207. package/apps/control-plane/test/gate-freshness.spec.ts +590 -0
  208. package/apps/control-plane/test/gate-log-extractor.spec.ts +170 -0
  209. package/apps/control-plane/test/gates.spec.ts +108 -0
  210. package/apps/control-plane/test/git-reconciliation-service.spec.ts +2307 -0
  211. package/apps/control-plane/test/helpers.ts +65 -0
  212. package/apps/control-plane/test/incremental-gates.spec.ts +271 -0
  213. package/apps/control-plane/test/index-normalizer.spec.ts +98 -0
  214. package/apps/control-plane/test/init-wizard.spec.ts +17 -0
  215. package/apps/control-plane/test/intake-artifacts.spec.ts +203 -0
  216. package/apps/control-plane/test/intake-service.spec.ts +3176 -0
  217. package/apps/control-plane/test/kernel-collision-replay.spec.ts +3 -2
  218. package/apps/control-plane/test/kernel-tool-executor.spec.ts +77 -0
  219. package/apps/control-plane/test/kernel-tool-wiring.spec.ts +279 -0
  220. package/apps/control-plane/test/kernel.branches.spec.ts +15 -2
  221. package/apps/control-plane/test/kernel.coverage.spec.ts +7 -3
  222. package/apps/control-plane/test/kernel.coverage2.spec.ts +731 -2
  223. package/apps/control-plane/test/kernel.spec.ts +464 -2
  224. package/apps/control-plane/test/mainline-conflict.spec.ts +66 -0
  225. package/apps/control-plane/test/mcp-helpers.spec.ts +79 -0
  226. package/apps/control-plane/test/mcp.spec.ts +177 -13
  227. package/apps/control-plane/test/merge-command-handler.spec.ts +531 -0
  228. package/apps/control-plane/test/merge-service.spec.ts +570 -4
  229. package/apps/control-plane/test/notifier-service.spec.ts +26 -0
  230. package/apps/control-plane/test/organizer-enrollment-scheduler.spec.ts +340 -0
  231. package/apps/control-plane/test/organizer-ordering-artifact.spec.ts +95 -0
  232. package/apps/control-plane/test/organizer-sidecar-service.spec.ts +468 -0
  233. package/apps/control-plane/test/output-loop-detector.spec.ts +6 -0
  234. package/apps/control-plane/test/path-layout.spec.ts +70 -0
  235. package/apps/control-plane/test/performance-analytics.spec.ts +124 -0
  236. package/apps/control-plane/test/plan-conformance-scorer.spec.ts +53 -0
  237. package/apps/control-plane/test/plan-service.spec.ts +686 -4
  238. package/apps/control-plane/test/planning-wave-executor.spec.ts +3272 -86
  239. package/apps/control-plane/test/policy-loader-service.spec.ts +5 -0
  240. package/apps/control-plane/test/prompt-overlay.spec.ts +65 -0
  241. package/apps/control-plane/test/provider-command-runner-epipe.spec.ts +64 -0
  242. package/apps/control-plane/test/providers/api-worker-provider.spec.ts +129 -0
  243. package/apps/control-plane/test/providers/cli-worker-provider.spec.ts +148 -0
  244. package/apps/control-plane/test/providers/usage-types.spec.ts +98 -0
  245. package/apps/control-plane/test/providers.spec.ts +293 -16
  246. package/apps/control-plane/test/question-command-handlers.spec.ts +156 -0
  247. package/apps/control-plane/test/question-service.spec.ts +1119 -0
  248. package/apps/control-plane/test/reactions.spec.ts +114 -0
  249. package/apps/control-plane/test/replay-command-handler.spec.ts +144 -0
  250. package/apps/control-plane/test/replay-timeline-service.spec.ts +459 -0
  251. package/apps/control-plane/test/response.spec.ts +31 -0
  252. package/apps/control-plane/test/resume-command.spec.ts +757 -9
  253. package/apps/control-plane/test/retry-resume-decision.spec.ts +133 -0
  254. package/apps/control-plane/test/rollback-command-handler.spec.ts +334 -0
  255. package/apps/control-plane/test/rollback-command.spec.ts +120 -0
  256. package/apps/control-plane/test/run-coordinator.spec.ts +3062 -404
  257. package/apps/control-plane/test/schemas/state.schema.spec.ts +71 -0
  258. package/apps/control-plane/test/service-retry-paths.spec.ts +112 -0
  259. package/apps/control-plane/test/services.spec.ts +472 -2
  260. package/apps/control-plane/test/session-management.spec.ts +346 -1
  261. package/apps/control-plane/test/spec-ingestion.spec.ts +102 -28
  262. package/apps/control-plane/test/spec-preparation.spec.ts +182 -0
  263. package/apps/control-plane/test/supervisor-collaborators.spec.ts +191 -3
  264. package/apps/control-plane/test/supervisor.calltool.spec.ts +198 -0
  265. package/apps/control-plane/test/supervisor.spec.ts +95 -16
  266. package/apps/control-plane/test/supervisor.unit.spec.ts +385 -18
  267. package/apps/control-plane/test/tool-runtime.spec.ts +122 -0
  268. package/apps/control-plane/test/worker-decision-loop.spec.ts +3479 -476
  269. package/apps/control-plane/test/worker-execution-policy.spec.ts +1416 -6
  270. package/apps/control-plane/test/worker-provider-adapters.spec.ts +1894 -37
  271. package/apps/control-plane/test/worker-provider-factory.spec.ts +81 -0
  272. package/apps/control-plane/test/worktree-watchdog-service.spec.ts +125 -0
  273. package/apps/control-plane/vitest.config.ts +5 -0
  274. package/config/agentic/orchestrator/agents.yaml +22 -1
  275. package/config/agentic/orchestrator/gates.yaml +24 -7
  276. package/config/agentic/orchestrator/policy.yaml +23 -1
  277. package/config/agentic/orchestrator/prompts/builder.system.md +69 -20
  278. package/config/agentic/orchestrator/prompts/organizer.system.md +85 -0
  279. package/config/agentic/orchestrator/prompts/overrides/builder.claude.md +28 -0
  280. package/config/agentic/orchestrator/prompts/overrides/builder.codex.md +28 -0
  281. package/config/agentic/orchestrator/prompts/overrides/planner.claude.md +20 -0
  282. package/config/agentic/orchestrator/prompts/overrides/planner.codex.md +20 -0
  283. package/config/agentic/orchestrator/prompts/planner-intake.system.md +149 -0
  284. package/config/agentic/orchestrator/prompts/planner.system.md +113 -40
  285. package/config/agentic/orchestrator/prompts/qa.system.md +75 -18
  286. package/config/agentic/orchestrator/prompts/reconciler.system.md +119 -0
  287. package/dist/apps/control-plane/application/kernel-tool-wiring.d.ts +26 -2
  288. package/dist/apps/control-plane/application/kernel-tool-wiring.js +40 -2
  289. package/dist/apps/control-plane/application/kernel-tool-wiring.js.map +1 -1
  290. package/dist/apps/control-plane/application/services/activity-monitor-service.js +37 -1
  291. package/dist/apps/control-plane/application/services/activity-monitor-service.js.map +1 -1
  292. package/dist/apps/control-plane/application/services/bootstrap-manifest-generator-service.d.ts +4 -0
  293. package/dist/apps/control-plane/application/services/bootstrap-manifest-generator-service.js +188 -0
  294. package/dist/apps/control-plane/application/services/bootstrap-manifest-generator-service.js.map +1 -0
  295. package/dist/apps/control-plane/application/services/checkpoint-service.d.ts +5 -0
  296. package/dist/apps/control-plane/application/services/checkpoint-service.js +69 -24
  297. package/dist/apps/control-plane/application/services/checkpoint-service.js.map +1 -1
  298. package/dist/apps/control-plane/application/services/collision-override-service.d.ts +139 -0
  299. package/dist/apps/control-plane/application/services/collision-override-service.js +568 -0
  300. package/dist/apps/control-plane/application/services/collision-override-service.js.map +1 -0
  301. package/dist/apps/control-plane/application/services/collision-queue-service.d.ts +15 -0
  302. package/dist/apps/control-plane/application/services/collision-queue-service.js +92 -33
  303. package/dist/apps/control-plane/application/services/collision-queue-service.js.map +1 -1
  304. package/dist/apps/control-plane/application/services/cost-tracking-service.d.ts +11 -0
  305. package/dist/apps/control-plane/application/services/cost-tracking-service.js +75 -0
  306. package/dist/apps/control-plane/application/services/cost-tracking-service.js.map +1 -1
  307. package/dist/apps/control-plane/application/services/execution-control-service.d.ts +75 -0
  308. package/dist/apps/control-plane/application/services/execution-control-service.js +421 -0
  309. package/dist/apps/control-plane/application/services/execution-control-service.js.map +1 -0
  310. package/dist/apps/control-plane/application/services/feature-deletion-service.d.ts +1 -0
  311. package/dist/apps/control-plane/application/services/feature-deletion-service.js +23 -1
  312. package/dist/apps/control-plane/application/services/feature-deletion-service.js.map +1 -1
  313. package/dist/apps/control-plane/application/services/feature-lifecycle-service.d.ts +24 -1
  314. package/dist/apps/control-plane/application/services/feature-lifecycle-service.js +132 -3
  315. package/dist/apps/control-plane/application/services/feature-lifecycle-service.js.map +1 -1
  316. package/dist/apps/control-plane/application/services/feature-send-message-service.js +16 -8
  317. package/dist/apps/control-plane/application/services/feature-send-message-service.js.map +1 -1
  318. package/dist/apps/control-plane/application/services/feature-state-service.d.ts +36 -0
  319. package/dist/apps/control-plane/application/services/feature-state-service.js +163 -6
  320. package/dist/apps/control-plane/application/services/feature-state-service.js.map +1 -1
  321. package/dist/apps/control-plane/application/services/gate-service.d.ts +2 -1
  322. package/dist/apps/control-plane/application/services/gate-service.js +95 -5
  323. package/dist/apps/control-plane/application/services/gate-service.js.map +1 -1
  324. package/dist/apps/control-plane/application/services/git-reconciliation-service.d.ts +92 -0
  325. package/dist/apps/control-plane/application/services/git-reconciliation-service.js +1097 -0
  326. package/dist/apps/control-plane/application/services/git-reconciliation-service.js.map +1 -0
  327. package/dist/apps/control-plane/application/services/intake-service.d.ts +63 -0
  328. package/dist/apps/control-plane/application/services/intake-service.js +1050 -0
  329. package/dist/apps/control-plane/application/services/intake-service.js.map +1 -0
  330. package/dist/apps/control-plane/application/services/merge-service.d.ts +5 -1
  331. package/dist/apps/control-plane/application/services/merge-service.js +233 -18
  332. package/dist/apps/control-plane/application/services/merge-service.js.map +1 -1
  333. package/dist/apps/control-plane/application/services/notifier-service.d.ts +1 -1
  334. package/dist/apps/control-plane/application/services/notifier-service.js +1 -0
  335. package/dist/apps/control-plane/application/services/notifier-service.js.map +1 -1
  336. package/dist/apps/control-plane/application/services/performance-analytics-service.d.ts +11 -0
  337. package/dist/apps/control-plane/application/services/performance-analytics-service.js +59 -0
  338. package/dist/apps/control-plane/application/services/performance-analytics-service.js.map +1 -1
  339. package/dist/apps/control-plane/application/services/plan-service.d.ts +5 -0
  340. package/dist/apps/control-plane/application/services/plan-service.js +254 -15
  341. package/dist/apps/control-plane/application/services/plan-service.js.map +1 -1
  342. package/dist/apps/control-plane/application/services/question-service.d.ts +72 -0
  343. package/dist/apps/control-plane/application/services/question-service.js +507 -0
  344. package/dist/apps/control-plane/application/services/question-service.js.map +1 -0
  345. package/dist/apps/control-plane/application/services/reactions-service.d.ts +2 -0
  346. package/dist/apps/control-plane/application/services/reactions-service.js +60 -17
  347. package/dist/apps/control-plane/application/services/reactions-service.js.map +1 -1
  348. package/dist/apps/control-plane/application/services/replay-timeline-service.d.ts +39 -0
  349. package/dist/apps/control-plane/application/services/replay-timeline-service.js +205 -0
  350. package/dist/apps/control-plane/application/services/replay-timeline-service.js.map +1 -0
  351. package/dist/apps/control-plane/application/services/reporting-service.d.ts +59 -0
  352. package/dist/apps/control-plane/application/services/reporting-service.js +121 -9
  353. package/dist/apps/control-plane/application/services/reporting-service.js.map +1 -1
  354. package/dist/apps/control-plane/application/services/run-lease-service.d.ts +20 -0
  355. package/dist/apps/control-plane/application/services/run-lease-service.js +81 -4
  356. package/dist/apps/control-plane/application/services/run-lease-service.js.map +1 -1
  357. package/dist/apps/control-plane/application/services/worktree-watchdog-service.d.ts +10 -0
  358. package/dist/apps/control-plane/application/services/worktree-watchdog-service.js +65 -8
  359. package/dist/apps/control-plane/application/services/worktree-watchdog-service.js.map +1 -1
  360. package/dist/apps/control-plane/application/tools/tool-metadata.js +7 -0
  361. package/dist/apps/control-plane/application/tools/tool-metadata.js.map +1 -1
  362. package/dist/apps/control-plane/application/usage-types.d.ts +65 -0
  363. package/dist/apps/control-plane/application/usage-types.js +75 -0
  364. package/dist/apps/control-plane/application/usage-types.js.map +1 -0
  365. package/dist/apps/control-plane/cli/add-command-handler.d.ts +18 -0
  366. package/dist/apps/control-plane/cli/add-command-handler.js +110 -0
  367. package/dist/apps/control-plane/cli/add-command-handler.js.map +1 -0
  368. package/dist/apps/control-plane/cli/answer-command-handler.d.ts +8 -0
  369. package/dist/apps/control-plane/cli/answer-command-handler.js +96 -0
  370. package/dist/apps/control-plane/cli/answer-command-handler.js.map +1 -0
  371. package/dist/apps/control-plane/cli/attach-command-handler.js +8 -3
  372. package/dist/apps/control-plane/cli/attach-command-handler.js.map +1 -1
  373. package/dist/apps/control-plane/cli/cli-argument-parser.js +131 -11
  374. package/dist/apps/control-plane/cli/cli-argument-parser.js.map +1 -1
  375. package/dist/apps/control-plane/cli/collision-command-handler.d.ts +8 -0
  376. package/dist/apps/control-plane/cli/collision-command-handler.js +90 -0
  377. package/dist/apps/control-plane/cli/collision-command-handler.js.map +1 -0
  378. package/dist/apps/control-plane/cli/command-catalog.d.ts +21 -0
  379. package/dist/apps/control-plane/cli/command-catalog.js +416 -0
  380. package/dist/apps/control-plane/cli/command-catalog.js.map +1 -0
  381. package/dist/apps/control-plane/cli/complete-command-handler.d.ts +15 -0
  382. package/dist/apps/control-plane/cli/complete-command-handler.js +26 -0
  383. package/dist/apps/control-plane/cli/complete-command-handler.js.map +1 -0
  384. package/dist/apps/control-plane/cli/completion-command-handler.d.ts +8 -0
  385. package/dist/apps/control-plane/cli/completion-command-handler.js +20 -0
  386. package/dist/apps/control-plane/cli/completion-command-handler.js.map +1 -0
  387. package/dist/apps/control-plane/cli/completion-resolver.d.ts +1 -0
  388. package/dist/apps/control-plane/cli/completion-resolver.js +250 -0
  389. package/dist/apps/control-plane/cli/completion-resolver.js.map +1 -0
  390. package/dist/apps/control-plane/cli/completion-shell-renderer.d.ts +3 -0
  391. package/dist/apps/control-plane/cli/completion-shell-renderer.js +53 -0
  392. package/dist/apps/control-plane/cli/completion-shell-renderer.js.map +1 -0
  393. package/dist/apps/control-plane/cli/dashboard-command-handler.d.ts +1 -0
  394. package/dist/apps/control-plane/cli/dashboard-command-handler.js +83 -1
  395. package/dist/apps/control-plane/cli/dashboard-command-handler.js.map +1 -1
  396. package/dist/apps/control-plane/cli/dashboard-runtime-runner.d.ts +81 -0
  397. package/dist/apps/control-plane/cli/dashboard-runtime-runner.js +724 -0
  398. package/dist/apps/control-plane/cli/dashboard-runtime-runner.js.map +1 -0
  399. package/dist/apps/control-plane/cli/dashboard-runtime.d.ts +1 -0
  400. package/dist/apps/control-plane/cli/dashboard-runtime.js +26 -0
  401. package/dist/apps/control-plane/cli/dashboard-runtime.js.map +1 -0
  402. package/dist/apps/control-plane/cli/help-command-handler.js +13 -172
  403. package/dist/apps/control-plane/cli/help-command-handler.js.map +1 -1
  404. package/dist/apps/control-plane/cli/init-command-handler.js +51 -6
  405. package/dist/apps/control-plane/cli/init-command-handler.js.map +1 -1
  406. package/dist/apps/control-plane/cli/merge-command-handler.d.ts +8 -0
  407. package/dist/apps/control-plane/cli/merge-command-handler.js +139 -0
  408. package/dist/apps/control-plane/cli/merge-command-handler.js.map +1 -0
  409. package/dist/apps/control-plane/cli/questions-command-handler.d.ts +8 -0
  410. package/dist/apps/control-plane/cli/questions-command-handler.js +59 -0
  411. package/dist/apps/control-plane/cli/questions-command-handler.js.map +1 -0
  412. package/dist/apps/control-plane/cli/replay-command-handler.d.ts +15 -0
  413. package/dist/apps/control-plane/cli/replay-command-handler.js +55 -0
  414. package/dist/apps/control-plane/cli/replay-command-handler.js.map +1 -0
  415. package/dist/apps/control-plane/cli/resume-command-handler.d.ts +2 -0
  416. package/dist/apps/control-plane/cli/resume-command-handler.js +180 -17
  417. package/dist/apps/control-plane/cli/resume-command-handler.js.map +1 -1
  418. package/dist/apps/control-plane/cli/retry-command-handler.js +202 -16
  419. package/dist/apps/control-plane/cli/retry-command-handler.js.map +1 -1
  420. package/dist/apps/control-plane/cli/retry-resume-decision.d.ts +26 -0
  421. package/dist/apps/control-plane/cli/retry-resume-decision.js +61 -0
  422. package/dist/apps/control-plane/cli/retry-resume-decision.js.map +1 -0
  423. package/dist/apps/control-plane/cli/rollback-command-handler.js +3 -2
  424. package/dist/apps/control-plane/cli/rollback-command-handler.js.map +1 -1
  425. package/dist/apps/control-plane/cli/run-command-handler.js +26 -2
  426. package/dist/apps/control-plane/cli/run-command-handler.js.map +1 -1
  427. package/dist/apps/control-plane/cli/spec-ingestion-service.d.ts +2 -0
  428. package/dist/apps/control-plane/cli/spec-ingestion-service.js +37 -48
  429. package/dist/apps/control-plane/cli/spec-ingestion-service.js.map +1 -1
  430. package/dist/apps/control-plane/cli/spec-preparation.d.ts +14 -0
  431. package/dist/apps/control-plane/cli/spec-preparation.js +81 -0
  432. package/dist/apps/control-plane/cli/spec-preparation.js.map +1 -0
  433. package/dist/apps/control-plane/cli/spec-utils.d.ts +4 -0
  434. package/dist/apps/control-plane/cli/spec-utils.js +70 -11
  435. package/dist/apps/control-plane/cli/spec-utils.js.map +1 -1
  436. package/dist/apps/control-plane/cli/status-command-handler.js +69 -0
  437. package/dist/apps/control-plane/cli/status-command-handler.js.map +1 -1
  438. package/dist/apps/control-plane/cli/types.d.ts +41 -4
  439. package/dist/apps/control-plane/cli/types.js +9 -1
  440. package/dist/apps/control-plane/cli/types.js.map +1 -1
  441. package/dist/apps/control-plane/core/collisions.d.ts +37 -19
  442. package/dist/apps/control-plane/core/collisions.js +87 -12
  443. package/dist/apps/control-plane/core/collisions.js.map +1 -1
  444. package/dist/apps/control-plane/core/constants.d.ts +17 -1
  445. package/dist/apps/control-plane/core/constants.js +18 -1
  446. package/dist/apps/control-plane/core/constants.js.map +1 -1
  447. package/dist/apps/control-plane/core/error-codes.d.ts +39 -0
  448. package/dist/apps/control-plane/core/error-codes.js +39 -0
  449. package/dist/apps/control-plane/core/error-codes.js.map +1 -1
  450. package/dist/apps/control-plane/core/execution-control.d.ts +45 -0
  451. package/dist/apps/control-plane/core/execution-control.js +2 -0
  452. package/dist/apps/control-plane/core/execution-control.js.map +1 -0
  453. package/dist/apps/control-plane/core/feature-resume-phase.d.ts +3 -0
  454. package/dist/apps/control-plane/core/feature-resume-phase.js +88 -0
  455. package/dist/apps/control-plane/core/feature-resume-phase.js.map +1 -0
  456. package/dist/apps/control-plane/core/gate-freshness.d.ts +48 -0
  457. package/dist/apps/control-plane/core/gate-freshness.js +267 -0
  458. package/dist/apps/control-plane/core/gate-freshness.js.map +1 -0
  459. package/dist/apps/control-plane/core/gate-log-extractor.d.ts +22 -0
  460. package/dist/apps/control-plane/core/gate-log-extractor.js +66 -0
  461. package/dist/apps/control-plane/core/gate-log-extractor.js.map +1 -0
  462. package/dist/apps/control-plane/core/gates.d.ts +11 -2
  463. package/dist/apps/control-plane/core/gates.js +67 -3
  464. package/dist/apps/control-plane/core/gates.js.map +1 -1
  465. package/dist/apps/control-plane/core/intake-artifacts.d.ts +109 -0
  466. package/dist/apps/control-plane/core/intake-artifacts.js +143 -0
  467. package/dist/apps/control-plane/core/intake-artifacts.js.map +1 -0
  468. package/dist/apps/control-plane/core/kernel-types.d.ts +8 -0
  469. package/dist/apps/control-plane/core/kernel.d.ts +256 -8
  470. package/dist/apps/control-plane/core/kernel.js +400 -14
  471. package/dist/apps/control-plane/core/kernel.js.map +1 -1
  472. package/dist/apps/control-plane/core/mainline-conflict.d.ts +7 -0
  473. package/dist/apps/control-plane/core/mainline-conflict.js +20 -0
  474. package/dist/apps/control-plane/core/mainline-conflict.js.map +1 -0
  475. package/dist/apps/control-plane/core/merge-repair.d.ts +35 -0
  476. package/dist/apps/control-plane/core/merge-repair.js +99 -0
  477. package/dist/apps/control-plane/core/merge-repair.js.map +1 -0
  478. package/dist/apps/control-plane/core/path-layout.d.ts +10 -0
  479. package/dist/apps/control-plane/core/path-layout.js +32 -2
  480. package/dist/apps/control-plane/core/path-layout.js.map +1 -1
  481. package/dist/apps/control-plane/core/path-rules.js +9 -3
  482. package/dist/apps/control-plane/core/path-rules.js.map +1 -1
  483. package/dist/apps/control-plane/core/plan-submit-recovery.d.ts +22 -0
  484. package/dist/apps/control-plane/core/plan-submit-recovery.js +78 -0
  485. package/dist/apps/control-plane/core/plan-submit-recovery.js.map +1 -0
  486. package/dist/apps/control-plane/core/questions.d.ts +40 -0
  487. package/dist/apps/control-plane/core/questions.js +2 -0
  488. package/dist/apps/control-plane/core/questions.js.map +1 -0
  489. package/dist/apps/control-plane/core/runtime-sessions.d.ts +4 -0
  490. package/dist/apps/control-plane/core/schemas.d.ts +2 -0
  491. package/dist/apps/control-plane/core/schemas.js +31 -1
  492. package/dist/apps/control-plane/core/schemas.js.map +1 -1
  493. package/dist/apps/control-plane/core/tool-caller.d.ts +18 -1
  494. package/dist/apps/control-plane/core/utils/index-normalizer.js +17 -4
  495. package/dist/apps/control-plane/core/utils/index-normalizer.js.map +1 -1
  496. package/dist/apps/control-plane/core/worktree-diff.d.ts +4 -0
  497. package/dist/apps/control-plane/core/worktree-diff.js +52 -0
  498. package/dist/apps/control-plane/core/worktree-diff.js.map +1 -0
  499. package/dist/apps/control-plane/index.d.ts +10 -2
  500. package/dist/apps/control-plane/index.js +9 -2
  501. package/dist/apps/control-plane/index.js.map +1 -1
  502. package/dist/apps/control-plane/interfaces/cli/bootstrap.js +236 -6
  503. package/dist/apps/control-plane/interfaces/cli/bootstrap.js.map +1 -1
  504. package/dist/apps/control-plane/mcp/kernel-tool-executor.js +16 -0
  505. package/dist/apps/control-plane/mcp/kernel-tool-executor.js.map +1 -1
  506. package/dist/apps/control-plane/mcp/tool-runtime.d.ts +5 -0
  507. package/dist/apps/control-plane/mcp/tool-runtime.js +40 -5
  508. package/dist/apps/control-plane/mcp/tool-runtime.js.map +1 -1
  509. package/dist/apps/control-plane/providers/api-worker-provider.d.ts +2 -2
  510. package/dist/apps/control-plane/providers/api-worker-provider.js +40 -9
  511. package/dist/apps/control-plane/providers/api-worker-provider.js.map +1 -1
  512. package/dist/apps/control-plane/providers/cli-worker-provider.d.ts +59 -3
  513. package/dist/apps/control-plane/providers/cli-worker-provider.js +758 -46
  514. package/dist/apps/control-plane/providers/cli-worker-provider.js.map +1 -1
  515. package/dist/apps/control-plane/providers/output-parsers/generic-output-parser.js +91 -1
  516. package/dist/apps/control-plane/providers/output-parsers/generic-output-parser.js.map +1 -1
  517. package/dist/apps/control-plane/providers/output-parsers/types.d.ts +2 -0
  518. package/dist/apps/control-plane/providers/provider-defaults.d.ts +12 -0
  519. package/dist/apps/control-plane/providers/provider-defaults.js +103 -7
  520. package/dist/apps/control-plane/providers/provider-defaults.js.map +1 -1
  521. package/dist/apps/control-plane/providers/providers.d.ts +50 -4
  522. package/dist/apps/control-plane/providers/providers.js +145 -14
  523. package/dist/apps/control-plane/providers/providers.js.map +1 -1
  524. package/dist/apps/control-plane/providers/worker-provider-factory.d.ts +2 -0
  525. package/dist/apps/control-plane/providers/worker-provider-factory.js +8 -1
  526. package/dist/apps/control-plane/providers/worker-provider-factory.js.map +1 -1
  527. package/dist/apps/control-plane/supervisor/artifact-stager.d.ts +5 -0
  528. package/dist/apps/control-plane/supervisor/artifact-stager.js +45 -0
  529. package/dist/apps/control-plane/supervisor/artifact-stager.js.map +1 -0
  530. package/dist/apps/control-plane/supervisor/build-wave-executor.d.ts +24 -1
  531. package/dist/apps/control-plane/supervisor/build-wave-executor.js +362 -150
  532. package/dist/apps/control-plane/supervisor/build-wave-executor.js.map +1 -1
  533. package/dist/apps/control-plane/supervisor/execution-enrollment-service.d.ts +41 -0
  534. package/dist/apps/control-plane/supervisor/execution-enrollment-service.js +311 -0
  535. package/dist/apps/control-plane/supervisor/execution-enrollment-service.js.map +1 -0
  536. package/dist/apps/control-plane/supervisor/organizer-enrollment-scheduler.d.ts +15 -0
  537. package/dist/apps/control-plane/supervisor/organizer-enrollment-scheduler.js +93 -0
  538. package/dist/apps/control-plane/supervisor/organizer-enrollment-scheduler.js.map +1 -0
  539. package/dist/apps/control-plane/supervisor/organizer-sidecar-service.d.ts +44 -0
  540. package/dist/apps/control-plane/supervisor/organizer-sidecar-service.js +311 -0
  541. package/dist/apps/control-plane/supervisor/organizer-sidecar-service.js.map +1 -0
  542. package/dist/apps/control-plane/supervisor/plan-conformance-scorer.js +2 -5
  543. package/dist/apps/control-plane/supervisor/plan-conformance-scorer.js.map +1 -1
  544. package/dist/apps/control-plane/supervisor/planner-phase.d.ts +3 -0
  545. package/dist/apps/control-plane/supervisor/planner-phase.js +70 -0
  546. package/dist/apps/control-plane/supervisor/planner-phase.js.map +1 -0
  547. package/dist/apps/control-plane/supervisor/planning-wave-executor.d.ts +42 -0
  548. package/dist/apps/control-plane/supervisor/planning-wave-executor.js +753 -55
  549. package/dist/apps/control-plane/supervisor/planning-wave-executor.js.map +1 -1
  550. package/dist/apps/control-plane/supervisor/prompt-bundle-loader.js +19 -1
  551. package/dist/apps/control-plane/supervisor/prompt-bundle-loader.js.map +1 -1
  552. package/dist/apps/control-plane/supervisor/qa-wave-executor.d.ts +21 -0
  553. package/dist/apps/control-plane/supervisor/qa-wave-executor.js +287 -156
  554. package/dist/apps/control-plane/supervisor/qa-wave-executor.js.map +1 -1
  555. package/dist/apps/control-plane/supervisor/run-coordinator.d.ts +30 -1
  556. package/dist/apps/control-plane/supervisor/run-coordinator.js +561 -17
  557. package/dist/apps/control-plane/supervisor/run-coordinator.js.map +1 -1
  558. package/dist/apps/control-plane/supervisor/runtime.d.ts +84 -0
  559. package/dist/apps/control-plane/supervisor/runtime.js +393 -3
  560. package/dist/apps/control-plane/supervisor/runtime.js.map +1 -1
  561. package/dist/apps/control-plane/supervisor/session-orchestrator.d.ts +54 -0
  562. package/dist/apps/control-plane/supervisor/session-orchestrator.js +176 -1
  563. package/dist/apps/control-plane/supervisor/session-orchestrator.js.map +1 -1
  564. package/dist/apps/control-plane/supervisor/types.d.ts +142 -1
  565. package/dist/apps/control-plane/supervisor/types.js.map +1 -1
  566. package/dist/apps/control-plane/supervisor/worker-decision-loop.d.ts +68 -2
  567. package/dist/apps/control-plane/supervisor/worker-decision-loop.js +723 -89
  568. package/dist/apps/control-plane/supervisor/worker-decision-loop.js.map +1 -1
  569. package/docs/core/ARCHITECTURE.md +227 -0
  570. package/docs/core/DECISIONS.md +94 -0
  571. package/docs/core/DOMAIN-LOGIC.md +60 -0
  572. package/docs/core/PATTERNS.md +201 -0
  573. package/docs/core/TROUBLESHOOTING.md +347 -0
  574. package/docs/core/intentgraph-dependencies.json +39860 -0
  575. package/docs/core/intentgraph.index.json +46580 -0
  576. package/docs/plans/2026-03-10-gate-failure-targeted-repair-design.md +224 -0
  577. package/docs/plans/2026-03-10-gate-failure-targeted-repair.md +1032 -0
  578. package/docs/superpowers/plans/2026-03-16-provider-cli-config.md +743 -0
  579. package/docs/superpowers/plans/2026-03-23-reconcile-divergence-fix.md +777 -0
  580. package/docs/superpowers/plans/2026-03-28-ordering-agent-implementation.md +1754 -0
  581. package/docs/superpowers/plans/2026-03-29-drop-zone-and-provider-optimization.md +1108 -0
  582. package/docs/superpowers/plans/2026-03-29-merge-target-feature-branch.md +685 -0
  583. package/docs/superpowers/plans/2026-03-29-organizer-sidecar-runtime-loop.md +1289 -0
  584. package/docs/superpowers/specs/2026-03-23-reconcile-divergence-fix-design.md +118 -0
  585. package/docs/superpowers/specs/2026-03-28-ordering-agent-spec-audit-design.md +50 -0
  586. package/docs/superpowers/specs/2026-03-29-drop-zone-and-provider-optimization-design.md +254 -0
  587. package/docs/superpowers/specs/2026-03-29-merge-target-feature-branch-design.md +152 -0
  588. package/docs/superpowers/specs/2026-03-29-organizer-sidecar-runtime-loop-design.md +225 -0
  589. package/package.json +3 -2
  590. package/packages/web-dashboard/package.json +2 -1
  591. package/packages/web-dashboard/src/app/analytics/page.tsx +36 -2
  592. package/packages/web-dashboard/src/app/api/actions/route.ts +274 -63
  593. package/packages/web-dashboard/src/app/api/actions/status/route.ts +35 -0
  594. package/packages/web-dashboard/src/app/api/analytics/provider/route.ts +18 -0
  595. package/packages/web-dashboard/src/app/api/collisions/approve/route.ts +58 -0
  596. package/packages/web-dashboard/src/app/api/features/[id]/checkpoint-diff/route.ts +36 -0
  597. package/packages/web-dashboard/src/app/api/features/[id]/checkpoints/route.ts +29 -0
  598. package/packages/web-dashboard/src/app/api/features/[id]/conflicts/abort/route.ts +29 -0
  599. package/packages/web-dashboard/src/app/api/features/[id]/conflicts/files/route.ts +30 -0
  600. package/packages/web-dashboard/src/app/api/features/[id]/conflicts/resolve/route.ts +51 -0
  601. package/packages/web-dashboard/src/app/api/features/[id]/conflicts/route.ts +75 -0
  602. package/packages/web-dashboard/src/app/api/features/[id]/diff/route.ts +16 -2
  603. package/packages/web-dashboard/src/app/api/features/[id]/files/route.ts +26 -0
  604. package/packages/web-dashboard/src/app/api/features/[id]/gate-history/route.ts +27 -0
  605. package/packages/web-dashboard/src/app/api/features/[id]/genealogy/route.ts +26 -0
  606. package/packages/web-dashboard/src/app/api/features/[id]/history/run/[runId]/route.ts +20 -0
  607. package/packages/web-dashboard/src/app/api/features/[id]/history/runs/route.ts +34 -0
  608. package/packages/web-dashboard/src/app/api/features/[id]/intake-workspace/route.ts +20 -0
  609. package/packages/web-dashboard/src/app/api/features/[id]/live-output/route.ts +74 -0
  610. package/packages/web-dashboard/src/app/api/features/[id]/plan/amend/route.ts +21 -0
  611. package/packages/web-dashboard/src/app/api/features/[id]/plan-progress/route.ts +20 -0
  612. package/packages/web-dashboard/src/app/api/features/[id]/planner-artifacts/[artifact]/route.ts +78 -0
  613. package/packages/web-dashboard/src/app/api/features/[id]/planner-lifecycle/route.ts +20 -0
  614. package/packages/web-dashboard/src/app/api/features/[id]/planning-workspace/route.ts +20 -0
  615. package/packages/web-dashboard/src/app/api/features/[id]/questions/[questionId]/answer/route.ts +27 -0
  616. package/packages/web-dashboard/src/app/api/features/[id]/questions/route.ts +18 -0
  617. package/packages/web-dashboard/src/app/api/features/[id]/review/route.ts +14 -7
  618. package/packages/web-dashboard/src/app/api/features/[id]/route.ts +57 -2
  619. package/packages/web-dashboard/src/app/api/features/[id]/spec/route.ts +30 -0
  620. package/packages/web-dashboard/src/app/api/features/[id]/triage/route.ts +83 -0
  621. package/packages/web-dashboard/src/app/api/features/[id]/worker-events/route.ts +40 -0
  622. package/packages/web-dashboard/src/app/api/launch/preview/route.ts +86 -0
  623. package/packages/web-dashboard/src/app/api/launch/submit/route.ts +180 -0
  624. package/packages/web-dashboard/src/app/api/mainline/status/route.ts +74 -0
  625. package/packages/web-dashboard/src/app/api/merge-queue/route.ts +13 -0
  626. package/packages/web-dashboard/src/app/api/policy/budget/route.ts +14 -0
  627. package/packages/web-dashboard/src/app/api/projects/route.ts +11 -7
  628. package/packages/web-dashboard/src/app/api/reconciler/queue/route.ts +47 -0
  629. package/packages/web-dashboard/src/app/api/run/route.ts +26 -2
  630. package/packages/web-dashboard/src/app/api/runtime/events/route.ts +227 -0
  631. package/packages/web-dashboard/src/app/api/runtime/operations/route.ts +269 -0
  632. package/packages/web-dashboard/src/app/api/runtime/questions/route.ts +11 -0
  633. package/packages/web-dashboard/src/app/api/runtime/runs/route.ts +80 -0
  634. package/packages/web-dashboard/src/app/api/status/route.ts +4 -2
  635. package/packages/web-dashboard/src/app/feature/[id]/page.tsx +32 -42
  636. package/packages/web-dashboard/src/app/globals.css +34 -3
  637. package/packages/web-dashboard/src/app/launch/page.tsx +357 -0
  638. package/packages/web-dashboard/src/app/layout.tsx +23 -1
  639. package/packages/web-dashboard/src/app/page.tsx +263 -272
  640. package/packages/web-dashboard/src/components/dashboard/attention-strip.tsx +52 -0
  641. package/packages/web-dashboard/src/components/dashboard/collision-approval-drawer.tsx +185 -0
  642. package/packages/web-dashboard/src/components/dashboard/command-center-header.tsx +102 -0
  643. package/packages/web-dashboard/src/components/dashboard/mainline-status-banner.tsx +84 -0
  644. package/packages/web-dashboard/src/components/dashboard/merged-archive.tsx +36 -0
  645. package/packages/web-dashboard/src/components/dashboard/prioritized-queues.tsx +98 -0
  646. package/packages/web-dashboard/src/components/dashboard/reconciler-queue-card.tsx +115 -0
  647. package/packages/web-dashboard/src/components/dashboard/secondary-diagnostics-rail.tsx +48 -0
  648. package/packages/web-dashboard/src/components/dashboard/task-filter-bar.tsx +74 -0
  649. package/packages/web-dashboard/src/components/dashboard/triage-drawer.tsx +455 -0
  650. package/packages/web-dashboard/src/components/diff-viewer.tsx +19 -3
  651. package/packages/web-dashboard/src/components/evidence-viewer.tsx +65 -51
  652. package/packages/web-dashboard/src/components/feature-card.tsx +90 -7
  653. package/packages/web-dashboard/src/components/feature-cost-panel.tsx +112 -11
  654. package/packages/web-dashboard/src/components/feature-list-view.tsx +25 -4
  655. package/packages/web-dashboard/src/components/features/runtime-inspector/EventsTimelineView.tsx +260 -0
  656. package/packages/web-dashboard/src/components/features/runtime-inspector/OperationsListView.tsx +172 -0
  657. package/packages/web-dashboard/src/components/features/runtime-inspector/RuntimeInspectorPanel.tsx +896 -0
  658. package/packages/web-dashboard/src/components/filter-bar.tsx +7 -39
  659. package/packages/web-dashboard/src/components/focus/ActionableRiskList.tsx +46 -0
  660. package/packages/web-dashboard/src/components/focus/AgentRolePerformanceCard.tsx +200 -0
  661. package/packages/web-dashboard/src/components/focus/BlockedGuidanceBanner.tsx +149 -0
  662. package/packages/web-dashboard/src/components/focus/CheckpointInspector.tsx +123 -0
  663. package/packages/web-dashboard/src/components/focus/CheckpointRail.tsx +118 -0
  664. package/packages/web-dashboard/src/components/focus/CheckpointScrubber.tsx +249 -0
  665. package/packages/web-dashboard/src/components/focus/CollisionApprovalBanner.tsx +192 -0
  666. package/packages/web-dashboard/src/components/focus/CollisionRadar.tsx +136 -0
  667. package/packages/web-dashboard/src/components/focus/ConflictStatusCard.tsx +52 -0
  668. package/packages/web-dashboard/src/components/focus/ContextSidebar.tsx +108 -0
  669. package/packages/web-dashboard/src/components/focus/DiagnosisPanel.tsx +68 -0
  670. package/packages/web-dashboard/src/components/focus/FeatureDecisionBanner.tsx +68 -0
  671. package/packages/web-dashboard/src/components/focus/FeatureQuestionAnswerPanel.tsx +167 -0
  672. package/packages/web-dashboard/src/components/focus/FocusHeader.tsx +54 -0
  673. package/packages/web-dashboard/src/components/focus/FocusLayout.tsx +283 -0
  674. package/packages/web-dashboard/src/components/focus/GateFlakinessSummary.tsx +144 -0
  675. package/packages/web-dashboard/src/components/focus/GenealogyTree.tsx +34 -0
  676. package/packages/web-dashboard/src/components/focus/HeroBlock.tsx +67 -0
  677. package/packages/web-dashboard/src/components/focus/LiveAgentConsole.tsx +277 -0
  678. package/packages/web-dashboard/src/components/focus/MergeQueueCard.tsx +78 -0
  679. package/packages/web-dashboard/src/components/focus/OperationalSummaryCard.tsx +227 -0
  680. package/packages/web-dashboard/src/components/focus/PinnedActions.tsx +96 -0
  681. package/packages/web-dashboard/src/components/focus/PlanAmendmentPanel.tsx +250 -0
  682. package/packages/web-dashboard/src/components/focus/PlanProgressPanel.tsx +133 -0
  683. package/packages/web-dashboard/src/components/focus/PlannerArtifactViewer.tsx +158 -0
  684. package/packages/web-dashboard/src/components/focus/PlannerLifecycleHeader.tsx +141 -0
  685. package/packages/web-dashboard/src/components/focus/ProgressSnapshotCard.tsx +113 -0
  686. package/packages/web-dashboard/src/components/focus/RecentMaterialChanges.tsx +69 -0
  687. package/packages/web-dashboard/src/components/focus/RoleLogViewer.tsx +436 -0
  688. package/packages/web-dashboard/src/components/focus/RunHistoryBrowser.tsx +62 -0
  689. package/packages/web-dashboard/src/components/focus/SpecViewer.tsx +172 -0
  690. package/packages/web-dashboard/src/components/focus/TabBar.tsx +33 -0
  691. package/packages/web-dashboard/src/components/focus/UsageBurnChart.tsx +212 -0
  692. package/packages/web-dashboard/src/components/focus/VerificationSummaryCard.tsx +122 -0
  693. package/packages/web-dashboard/src/components/focus/tabs/ChangesTab.tsx +325 -0
  694. package/packages/web-dashboard/src/components/focus/tabs/ConflictsTab.tsx +395 -0
  695. package/packages/web-dashboard/src/components/focus/tabs/GatesQaTab.tsx +38 -0
  696. package/packages/web-dashboard/src/components/focus/tabs/HistoryTab.tsx +213 -0
  697. package/packages/web-dashboard/src/components/focus/tabs/IntakeTab.tsx +429 -0
  698. package/packages/web-dashboard/src/components/focus/tabs/OverviewTab.tsx +217 -0
  699. package/packages/web-dashboard/src/components/focus/tabs/PlanningTab.tsx +390 -0
  700. package/packages/web-dashboard/src/components/focus/tabs/ReviewTab.tsx +497 -0
  701. package/packages/web-dashboard/src/components/focus/tabs/RuntimeTab.tsx +213 -0
  702. package/packages/web-dashboard/src/components/focus/tabs/TranscriptTab.tsx +315 -0
  703. package/packages/web-dashboard/src/components/gate-results.tsx +2 -2
  704. package/packages/web-dashboard/src/components/human-input-panel.tsx +33 -57
  705. package/packages/web-dashboard/src/components/kanban-board.tsx +4 -0
  706. package/packages/web-dashboard/src/components/launch/launch-draft-card.tsx +131 -0
  707. package/packages/web-dashboard/src/components/plan-viewer.tsx +147 -69
  708. package/packages/web-dashboard/src/components/quick-launch-panel.tsx +20 -47
  709. package/packages/web-dashboard/src/components/summary-bar.tsx +30 -76
  710. package/packages/web-dashboard/src/lib/aop-client.ts +2484 -36
  711. package/packages/web-dashboard/src/lib/blocked-state-guidance.ts +475 -0
  712. package/packages/web-dashboard/src/lib/collision-radar.ts +136 -0
  713. package/packages/web-dashboard/src/lib/dashboard-action-states.ts +204 -0
  714. package/packages/web-dashboard/src/lib/dashboard-runtime-client.ts +439 -0
  715. package/packages/web-dashboard/src/lib/dashboard-utils.ts +179 -18
  716. package/packages/web-dashboard/src/lib/drop-zone-utils.ts +92 -0
  717. package/packages/web-dashboard/src/lib/focus-detail-derivations.ts +958 -0
  718. package/packages/web-dashboard/src/lib/focus-view.ts +300 -0
  719. package/packages/web-dashboard/src/lib/health-diagnosis.ts +356 -0
  720. package/packages/web-dashboard/src/lib/launch-contracts.ts +77 -0
  721. package/packages/web-dashboard/src/lib/launch-markdown.ts +107 -0
  722. package/packages/web-dashboard/src/lib/launch-page-preview.ts +89 -0
  723. package/packages/web-dashboard/src/lib/live-feed.ts +1 -1
  724. package/packages/web-dashboard/src/lib/multi-project-config.ts +33 -0
  725. package/packages/web-dashboard/src/lib/orchestrator-tools.ts +845 -59
  726. package/packages/web-dashboard/src/lib/planner-workspace.ts +1285 -0
  727. package/packages/web-dashboard/src/lib/review-contracts.ts +5 -3
  728. package/packages/web-dashboard/src/lib/runtime-files.ts +285 -0
  729. package/packages/web-dashboard/src/lib/tool-catalog.ts +51 -0
  730. package/packages/web-dashboard/src/lib/types.ts +731 -3
  731. package/packages/web-dashboard/src/lib/usage-burn.ts +175 -0
  732. package/packages/web-dashboard/src/lib/worktree-diff.ts +128 -0
  733. package/packages/web-dashboard/src/styles/dashboard.module.css +1742 -459
  734. package/packages/web-dashboard/test/api/actions/route.spec.ts +675 -0
  735. package/packages/web-dashboard/test/api/features/diff.route.spec.ts +57 -0
  736. package/packages/web-dashboard/test/api/features/feature.route.spec.ts +99 -0
  737. package/packages/web-dashboard/test/api/features/live-output.route.spec.ts +123 -0
  738. package/packages/web-dashboard/test/api/features/plan-amend.route.spec.ts +95 -0
  739. package/packages/web-dashboard/test/api/features/planner-workspaces.route.spec.ts +162 -0
  740. package/packages/web-dashboard/test/api/features/question-answer.route.spec.ts +99 -0
  741. package/packages/web-dashboard/test/api/features/triage.route.spec.ts +195 -0
  742. package/packages/web-dashboard/test/api/launch/preview.route.spec.ts +149 -0
  743. package/packages/web-dashboard/test/api/launch/submit.route.spec.ts +382 -0
  744. package/packages/web-dashboard/test/api/runtime/events/route.spec.ts +164 -0
  745. package/packages/web-dashboard/test/api/runtime/operations/route.spec.ts +156 -0
  746. package/packages/web-dashboard/test/api/runtime/runs/route.spec.ts +112 -0
  747. package/packages/web-dashboard/test/components/changes-tab.spec.tsx +76 -0
  748. package/packages/web-dashboard/test/components/command-center-root.spec.tsx +87 -0
  749. package/packages/web-dashboard/test/components/diagnosis-panel.spec.tsx +59 -0
  750. package/packages/web-dashboard/test/components/feature-card.spec.tsx +45 -0
  751. package/packages/web-dashboard/test/components/focus-layout.spec.tsx +299 -0
  752. package/packages/web-dashboard/test/components/gate-results.spec.tsx +39 -0
  753. package/packages/web-dashboard/test/components/gates-qa-tab.spec.tsx +118 -0
  754. package/packages/web-dashboard/test/components/human-input-panel.spec.tsx +54 -0
  755. package/packages/web-dashboard/test/components/intake-tab.spec.tsx +210 -0
  756. package/packages/web-dashboard/test/components/kanban-board.spec.tsx +35 -0
  757. package/packages/web-dashboard/test/components/launch-draft-card.spec.tsx +54 -0
  758. package/packages/web-dashboard/test/components/launch-page.spec.tsx +79 -0
  759. package/packages/web-dashboard/test/components/overview-tab.spec.tsx +236 -0
  760. package/packages/web-dashboard/test/components/planning-tab.spec.tsx +202 -0
  761. package/packages/web-dashboard/test/components/review-tab.spec.tsx +169 -0
  762. package/packages/web-dashboard/test/components/role-log-viewer.spec.ts +42 -0
  763. package/packages/web-dashboard/test/components/runtime-inspector.spec.tsx +22 -0
  764. package/packages/web-dashboard/test/components/runtime-tab.spec.tsx +133 -0
  765. package/packages/web-dashboard/test/components/transcript-tab.spec.tsx +46 -0
  766. package/packages/web-dashboard/test/components/triage-drawer.spec.tsx +159 -0
  767. package/packages/web-dashboard/test/lib/aop-client.spec.ts +235 -0
  768. package/packages/web-dashboard/test/lib/dashboard-runtime-client.spec.ts +144 -0
  769. package/packages/web-dashboard/test/lib/focus-detail-derivations.spec.ts +314 -0
  770. package/packages/web-dashboard/test/lib/focus-view.spec.ts +248 -0
  771. package/packages/web-dashboard/test/lib/health-diagnosis.spec.ts +277 -0
  772. package/packages/web-dashboard/test/lib/launch-markdown.spec.ts +36 -0
  773. package/packages/web-dashboard/test/lib/multi-project-config.spec.ts +54 -0
  774. package/packages/web-dashboard/test/lib/orchestrator-tools.spec.ts +352 -0
  775. package/packages/web-dashboard/test/lib/planner-workspace.spec.ts +289 -0
  776. package/packages/web-dashboard/test/lib/worktree-diff.spec.ts +119 -0
  777. package/packages/web-dashboard/vitest.config.ts +2 -0
  778. package/spec-files/completed/agentic_orchestrator_add_feature_to_active_execution_spec.md +557 -0
  779. package/spec-files/completed/agentic_orchestrator_dashboard_command_center_redesign_spec.md +1147 -0
  780. package/spec-files/completed/agentic_orchestrator_execution_mode_spec.md +18 -16
  781. package/spec-files/completed/agentic_orchestrator_feature_focus_view_track_a_spec.md +672 -0
  782. package/spec-files/completed/agentic_orchestrator_feature_focus_view_track_b_spec.md +794 -0
  783. package/spec-files/completed/agentic_orchestrator_feature_focus_view_track_c_decision_centric_remediation_spec.md +1037 -0
  784. package/spec-files/completed/agentic_orchestrator_feature_focus_view_ux_redesign_spec.md +1432 -0
  785. package/spec-files/completed/agentic_orchestrator_focus_plan_tab_intake_planning_workspace_spec.md +921 -0
  786. package/spec-files/completed/agentic_orchestrator_intentional_collision_override_spec.md +584 -0
  787. package/spec-files/completed/agentic_orchestrator_interactive_planning_intake_and_requirements_verification_spec.md +1185 -0
  788. package/spec-files/completed/agentic_orchestrator_reactive_execution_enrollment_spec.md +864 -0
  789. package/spec-files/{outstanding → completed}/agentic_orchestrator_runtime_inspection_spec.md +92 -19
  790. package/spec-files/completed/agentic_orchestrator_scope_aware_run_lease_spec.md +408 -0
  791. package/spec-files/completed/git-reconciliation-engine.md +827 -0
  792. package/spec-files/outstanding/agentic_orchestrator_dashboard_quick_launch_and_control_surface_spec.md +331 -0
  793. package/spec-files/outstanding/agentic_orchestrator_enterprise_governance_dashboard_spec.md +16 -6
  794. package/spec-files/outstanding/agentic_orchestrator_evidence_integrity_doctor_spec.md +60 -9
  795. package/spec-files/outstanding/agentic_orchestrator_focus_plan_tab_execution_contract_workspace_spec.md +616 -0
  796. package/spec-files/outstanding/agentic_orchestrator_headless_standby_dashboard_runtime_spec.md +310 -0
  797. package/spec-files/outstanding/agentic_orchestrator_human_input_interaction_protocol_spec.md +175 -72
  798. package/spec-files/outstanding/agentic_orchestrator_interactive_rename_cleanup_spec.md +197 -0
  799. package/spec-files/outstanding/agentic_orchestrator_interactive_resume_and_reconciliation_disposition_spec.md +412 -0
  800. package/spec-files/outstanding/agentic_orchestrator_knowledge_canary_spec.md +166 -137
  801. package/spec-files/outstanding/agentic_orchestrator_observability_replay_spec.md +3 -3
  802. package/spec-files/outstanding/agentic_orchestrator_phase_specific_agent_profiles_and_token_telemetry_spec.md +303 -0
  803. package/spec-files/outstanding/agentic_orchestrator_planning_review_quality_spec.md +18 -5
  804. package/spec-files/outstanding/agentic_orchestrator_policy_stratification_spec.md +225 -0
  805. package/spec-files/outstanding/agentic_orchestrator_quality_adoption_execution_spec.md +77 -50
  806. package/spec-files/outstanding/agentic_orchestrator_ready_to_merge_branch_handoff_spec.md +724 -0
  807. package/spec-files/outstanding/agentic_orchestrator_remove_deterministic_mode_spec.md +263 -0
  808. package/spec-files/outstanding/agentic_orchestrator_request_more_context_and_dashboard_human_input_spec.md +456 -0
  809. package/spec-files/outstanding/agentic_orchestrator_spec_coverage_and_reconciliation_enforcement_spec.md +1411 -0
  810. package/spec-files/outstanding/agentic_orchestrator_spec_ordering_agent_spec.md +370 -0
  811. package/spec-files/outstanding/shadow_workspace_implementation_spec.md +1 -1
  812. package/spec-files/progress.md +2026 -120
  813. package/specs/001-runtime-inspection/checklists/requirements.md +35 -0
  814. package/specs/001-runtime-inspection/design.md +338 -0
  815. package/specs/001-runtime-inspection/spec.md +95 -0
  816. package/specs/002-scope-aware-lease/checklists/requirements.md +35 -0
  817. package/specs/002-scope-aware-lease/contracts/lease-registry.schema.json +101 -0
  818. package/specs/002-scope-aware-lease/data-model.md +236 -0
  819. package/specs/002-scope-aware-lease/plan.md +766 -0
  820. package/specs/002-scope-aware-lease/quickstart.md +150 -0
  821. package/specs/002-scope-aware-lease/research.md +135 -0
  822. package/specs/002-scope-aware-lease/spec.md +128 -0
  823. package/specs/002-scope-aware-lease/tasks.md +767 -0
  824. package/tsconfig.json +1 -1
  825. package/vitest.config.ts +28 -0
  826. package/ARCHITECTURE_ADHERENCE_ANALYSIS.md +0 -871
  827. package/packages/web-dashboard/next-env.d.ts +0 -6
  828. package/packages/web-dashboard/src/components/detail-panel.tsx +0 -1124
  829. package/packages/web-dashboard/src/components/review-workspace.tsx +0 -1162
  830. /package/spec-files/{outstanding → completed}/agentic_orchestrator_artifact_database_publishing_spec.md +0 -0
  831. /package/spec-files/{outstanding → completed}/agentic_orchestrator_cli_shell_tab_completion_spec.md +0 -0
  832. /package/spec-files/{outstanding → completed}/agentic_orchestrator_dashboard_diff_and_agent_console_spec.md +0 -0
  833. /package/spec-files/{outstanding → completed}/agentic_orchestrator_performance_improvements_spec.md +0 -0
  834. /package/spec-files/{outstanding → completed}/agentic_orchestrator_persistent_worker_runtime_spec.md +0 -0
  835. /package/spec-files/{outstanding → completed}/agentic_orchestrator_provider_auth_bootstrap_spec.md +0 -0
  836. /package/spec-files/{outstanding → completed}/agentic_orchestrator_real_worker_provider_execution_spec.md +0 -0
@@ -1,7 +1,7 @@
1
1
  import fs from 'node:fs/promises';
2
2
  import os from 'node:os';
3
3
  import path from 'node:path';
4
- import { describe, expect, it, vi } from 'vitest';
4
+ import { afterEach, describe, expect, it, vi } from 'vitest';
5
5
  import { ERROR_CODES } from '../src/core/error-codes.js';
6
6
  import { STATUS, TOOLS } from '../src/core/constants.js';
7
7
  import { BuildWaveExecutor } from '../src/supervisor/build-wave-executor.js';
@@ -11,9 +11,15 @@ import type { WorkerDecisionResult } from '../src/supervisor/worker-decision-loo
11
11
  function decision(overrides: Partial<WorkerDecisionResult>): WorkerDecisionResult {
12
12
  return {
13
13
  planSubmission: false,
14
+ intakeSubmission: false,
14
15
  patchApplied: false,
15
16
  noteLogged: false,
16
17
  requestHandled: false,
18
+ questionRequested: false,
19
+ contextStall: false,
20
+ contextRequestCount: 0,
21
+ lastContextRequestAt: null,
22
+ lastContextRequestRole: null,
17
23
  invalidOutput: false,
18
24
  noProgress: false,
19
25
  outputTypes: [],
@@ -169,6 +175,112 @@ describe('BuildWaveExecutor live policy branches', () => {
169
175
  });
170
176
  });
171
177
 
178
+ it('GIVEN_checkpoint_invalid_WHEN_run_THEN_continues_without_blocking_the_feature', async () => {
179
+ const toolCaller = buildToolCaller();
180
+ const executor = new BuildWaveExecutor({
181
+ toolCaller: toolCaller as never,
182
+ providerMode: 'live',
183
+ malformedOutputAction: 'fail_run',
184
+ workerDecisionRunner: {
185
+ execute: vi.fn(async () =>
186
+ decision({
187
+ noProgress: true,
188
+ interactiveOutcome: 'progress_applied',
189
+ checkpoint: {
190
+ checkpoint_id: 'checkpoint-1',
191
+ timestamp: '2026-03-12T16:00:00.000Z',
192
+ files_changed: ['apps/control-plane/src/supervisor/build-wave-executor.ts'],
193
+ validation_status: 'invalid',
194
+ severity: 'warning',
195
+ diff_snapshot: '.aop/features/feature-a/checkpoints/checkpoint-1.diff',
196
+ violations: ['outside_allowed_areas'],
197
+ },
198
+ }),
199
+ ),
200
+ },
201
+ });
202
+
203
+ await expect(executor.run(['feature-a'], 1)).resolves.toBeUndefined();
204
+
205
+ const statePatches = toolCaller.callTool.mock.calls.filter(
206
+ (call) => call[1] === TOOLS.FEATURE_STATE_PATCH,
207
+ );
208
+ expect(statePatches).toHaveLength(0);
209
+ const gateRuns = toolCaller.callTool.mock.calls.filter((call) => call[1] === TOOLS.GATES_RUN);
210
+ expect(gateRuns).toHaveLength(0);
211
+ });
212
+
213
+ it('GIVEN_checkpoint_invalid_with_empty_violations_WHEN_run_THEN_blocks_with_fallback_message', async () => {
214
+ const toolCaller = buildToolCaller();
215
+ const executor = new BuildWaveExecutor({
216
+ toolCaller: toolCaller as never,
217
+ providerMode: 'live',
218
+ workerDecisionRunner: {
219
+ execute: vi.fn(async () =>
220
+ decision({
221
+ interactiveOutcome: 'checkpoint_invalid',
222
+ checkpoint: {
223
+ checkpoint_id: 'checkpoint-synthetic',
224
+ timestamp: '2026-03-21T00:00:00.000Z',
225
+ files_changed: [],
226
+ validation_status: 'skipped',
227
+ diff_snapshot: '.aop/features/feature-a/checkpoints/synthetic.diff',
228
+ violations: [],
229
+ },
230
+ }),
231
+ ),
232
+ },
233
+ });
234
+
235
+ await executor.run(['feature-a'], 1);
236
+
237
+ const statePatches = toolCaller.callTool.mock.calls.filter(
238
+ (call) => call[1] === TOOLS.FEATURE_STATE_PATCH,
239
+ );
240
+ expect(statePatches).toHaveLength(1);
241
+ const args = statePatches[0][2];
242
+ const patch = args?.patch as Record<string, unknown> | undefined;
243
+ const fm = patch?.front_matter as Record<string, unknown> | undefined;
244
+ const reason = fm?.status_reason as string | undefined;
245
+ expect(reason).toContain('checkpoint validation failed');
246
+ expect(reason?.endsWith(': ')).toBe(false);
247
+ });
248
+
249
+ it('GIVEN_checkpoint_invalid_with_violations_WHEN_run_THEN_blocks_with_joined_violations', async () => {
250
+ const toolCaller = buildToolCaller();
251
+ const executor = new BuildWaveExecutor({
252
+ toolCaller: toolCaller as never,
253
+ providerMode: 'live',
254
+ workerDecisionRunner: {
255
+ execute: vi.fn(async () =>
256
+ decision({
257
+ interactiveOutcome: 'checkpoint_invalid',
258
+ checkpoint: {
259
+ checkpoint_id: 'checkpoint-1',
260
+ timestamp: '2026-03-21T00:00:00.000Z',
261
+ files_changed: ['src/foo.ts'],
262
+ validation_status: 'invalid',
263
+ diff_snapshot: '.aop/features/feature-a/checkpoints/checkpoint-1.diff',
264
+ violations: ['lint failure in foo.ts', 'type error in bar.ts'],
265
+ },
266
+ }),
267
+ ),
268
+ },
269
+ });
270
+
271
+ await executor.run(['feature-a'], 1);
272
+
273
+ const statePatches = toolCaller.callTool.mock.calls.filter(
274
+ (call) => call[1] === TOOLS.FEATURE_STATE_PATCH,
275
+ );
276
+ expect(statePatches).toHaveLength(1);
277
+ const args = statePatches[0][2];
278
+ const patch = args?.patch as Record<string, unknown> | undefined;
279
+ const fm = patch?.front_matter as Record<string, unknown> | undefined;
280
+ const reason = fm?.status_reason as string | undefined;
281
+ expect(reason).toContain('lint failure in foo.ts; type error in bar.ts');
282
+ });
283
+
172
284
  it('GIVEN_no_progress_below_limit_WHEN_run_THEN_skips_current_cycle_without_blocking', async () => {
173
285
  const toolCaller = buildToolCaller();
174
286
  const executor = new BuildWaveExecutor({
@@ -190,6 +302,76 @@ describe('BuildWaveExecutor live policy branches', () => {
190
302
  expect(statePatches).toHaveLength(0);
191
303
  });
192
304
 
305
+ it('GIVEN_question_requested_WHEN_build_wave_runs_THEN_skips_gate_without_blocking', async () => {
306
+ const toolCaller = buildToolCaller();
307
+ const executor = new BuildWaveExecutor({
308
+ toolCaller: toolCaller as never,
309
+ providerMode: 'live',
310
+ workerDecisionRunner: {
311
+ execute: vi.fn(async () => decision({ questionRequested: true })),
312
+ },
313
+ });
314
+
315
+ await executor.run(['feature-a'], 1);
316
+
317
+ const gateRuns = toolCaller.callTool.mock.calls.filter((call) => call[1] === TOOLS.GATES_RUN);
318
+ const statePatches = toolCaller.callTool.mock.calls.filter(
319
+ (call) => call[1] === TOOLS.FEATURE_STATE_PATCH,
320
+ );
321
+ expect(gateRuns).toHaveLength(0);
322
+ expect(statePatches).toHaveLength(0);
323
+ });
324
+
325
+ it('GIVEN_context_stall_WHEN_build_wave_runs_THEN_blocks_feature_with_request_metadata', async () => {
326
+ const toolCaller = buildToolCaller();
327
+ const executor = new BuildWaveExecutor({
328
+ toolCaller: toolCaller as never,
329
+ providerMode: 'live',
330
+ workerDecisionRunner: {
331
+ execute: vi.fn(async () =>
332
+ decision({
333
+ contextStall: true,
334
+ contextRequestCount: 2,
335
+ lastContextRequestAt: '2026-03-16T10:00:00.000Z',
336
+ lastContextRequestRole: 'builder',
337
+ }),
338
+ ),
339
+ },
340
+ });
341
+
342
+ await executor.run(['feature-a'], 1);
343
+
344
+ expect(toolCaller.callTool).toHaveBeenCalledWith(
345
+ 'orchestrator',
346
+ TOOLS.FEATURE_STATE_PATCH,
347
+ expect.objectContaining({
348
+ feature_id: 'feature-a',
349
+ expected_version: 7,
350
+ patch: {
351
+ front_matter: {
352
+ status: STATUS.BLOCKED,
353
+ status_reason: `${ERROR_CODES.PROVIDER_CONTEXT_STALL}: repeated context refresh with no progress`,
354
+ context_request_count: 2,
355
+ last_context_request_at: '2026-03-16T10:00:00.000Z',
356
+ last_context_request_role: 'builder',
357
+ },
358
+ },
359
+ }),
360
+ );
361
+ const logCall = toolCaller.callTool.mock.calls.find(
362
+ (call) => call[1] === TOOLS.FEATURE_LOG_APPEND,
363
+ );
364
+ expect(logCall?.[2]).toMatchObject({
365
+ feature_id: 'feature-a',
366
+ });
367
+ expect(JSON.parse(String(logCall?.[2]?.note ?? '{}'))).toMatchObject({
368
+ phase: 'build',
369
+ error_code: ERROR_CODES.PROVIDER_CONTEXT_STALL,
370
+ context_request_count: 2,
371
+ last_context_request_role: 'builder',
372
+ });
373
+ });
374
+
193
375
  it('GIVEN_no_progress_at_limit_and_fail_run_policy_WHEN_run_THEN_throws_provider_no_progress', async () => {
194
376
  const toolCaller = buildToolCaller();
195
377
  const executor = new BuildWaveExecutor({
@@ -207,7 +389,7 @@ describe('BuildWaveExecutor live policy branches', () => {
207
389
  });
208
390
  });
209
391
 
210
- it('GIVEN_runtime_hard_timeout_and_block_policy_WHEN_run_THEN_blocks_feature', async () => {
392
+ it('GIVEN_runtime_hard_timeout_and_block_policy_WHEN_run_twice_THEN_blocks_feature_on_second_timeout', async () => {
211
393
  const toolCaller = buildToolCaller();
212
394
  const hardTimeoutError = new Error('hard timeout') as Error & { code?: string };
213
395
  hardTimeoutError.code = ERROR_CODES.PROVIDER_HARD_TIMEOUT;
@@ -222,6 +404,11 @@ describe('BuildWaveExecutor live policy branches', () => {
222
404
  },
223
405
  });
224
406
 
407
+ // First run: auto-retry once returns false, error re-thrown
408
+ await expect(executor.run(['feature-a'], 1)).rejects.toMatchObject({
409
+ code: ERROR_CODES.PROVIDER_HARD_TIMEOUT,
410
+ });
411
+ // Second run: timeout retry already used, now blocks
225
412
  await executor.run(['feature-a'], 1);
226
413
  const statePatches = toolCaller.callTool.mock.calls.filter(
227
414
  (call) => call[1] === TOOLS.FEATURE_STATE_PATCH,
@@ -257,6 +444,11 @@ describe('BuildWaveExecutor live policy branches', () => {
257
444
  },
258
445
  });
259
446
 
447
+ // First run: auto-retry once at same phase, error re-thrown
448
+ await expect(executor.run(['feature-a'], 1)).rejects.toMatchObject({
449
+ code: ERROR_CODES.PROVIDER_HARD_TIMEOUT,
450
+ });
451
+ // Second run: timeout retry used, now blocks and journals
260
452
  await executor.run(['feature-a'], 1);
261
453
  const events = await readWorkerEvents(repoRoot, 'run:build-policy-journal');
262
454
  expect(events).toEqual(
@@ -341,7 +533,7 @@ describe('BuildWaveExecutor live policy branches', () => {
341
533
  }
342
534
  });
343
535
 
344
- it('GIVEN_runtime_spawn_timeout_and_block_policy_WHEN_run_THEN_blocks_feature', async () => {
536
+ it('GIVEN_runtime_spawn_timeout_and_block_policy_WHEN_run_twice_THEN_blocks_feature_on_second_timeout', async () => {
345
537
  const toolCaller = buildToolCaller();
346
538
  const spawnTimeoutError = new Error('spawn timeout') as Error & { code?: string };
347
539
  spawnTimeoutError.code = ERROR_CODES.PROVIDER_SPAWN_TIMEOUT;
@@ -356,6 +548,11 @@ describe('BuildWaveExecutor live policy branches', () => {
356
548
  },
357
549
  });
358
550
 
551
+ // First run: auto-retry once at same phase, error re-thrown
552
+ await expect(executor.run(['feature-a'], 1)).rejects.toMatchObject({
553
+ code: ERROR_CODES.PROVIDER_SPAWN_TIMEOUT,
554
+ });
555
+ // Second run: timeout retry used, now blocks
359
556
  await executor.run(['feature-a'], 1);
360
557
  const statePatches = toolCaller.callTool.mock.calls.filter(
361
558
  (call) => call[1] === TOOLS.FEATURE_STATE_PATCH,
@@ -570,6 +767,328 @@ describe('BuildWaveExecutor live policy branches', () => {
570
767
  code: ERROR_CODES.PROVIDER_OUTPUT_LOOP,
571
768
  });
572
769
  });
770
+
771
+ describe('BuildWaveExecutor repair loop log extraction', () => {
772
+ afterEach(() => {
773
+ vi.restoreAllMocks();
774
+ });
775
+
776
+ it('GIVEN_gate_fails_and_log_exists_WHEN_repair_runs_THEN_context_is_refreshed_each_iteration', async () => {
777
+ let contextCallCount = 0;
778
+ const toolCaller = {
779
+ callTool: vi.fn(async (_role: string, toolName: string) => {
780
+ if (toolName === TOOLS.FEATURE_STATE_GET) {
781
+ return {
782
+ ok: true,
783
+ data: { front_matter: { status: STATUS.BUILDING, gate_retry_count: 0, version: 1 } },
784
+ };
785
+ }
786
+ if (toolName === TOOLS.FEATURE_GET_CONTEXT) {
787
+ contextCallCount += 1;
788
+ return {
789
+ ok: true,
790
+ data: {
791
+ state: { front_matter: { status: STATUS.BUILDING, gates: { fast: 'fail' } } },
792
+ latest_evidence: {
793
+ overall: 'fail',
794
+ mode: 'fast',
795
+ coverage_status: 'unknown',
796
+ step_results: [{ name: 'typecheck', exit_code: 1, log_path: '/tmp/test.log' }],
797
+ },
798
+ },
799
+ };
800
+ }
801
+ if (toolName === TOOLS.GATES_RUN) {
802
+ return { ok: true, data: { overall: 'fail', evidence_path: 'evidence.json' } };
803
+ }
804
+ if (toolName === TOOLS.FEATURE_STATE_PATCH) {
805
+ return { ok: true, data: {} };
806
+ }
807
+ if (toolName === TOOLS.FEATURE_LOG_APPEND) {
808
+ return { ok: true, data: {} };
809
+ }
810
+ return { ok: true, data: {} };
811
+ }),
812
+ };
813
+ vi.spyOn(fs, 'readFile').mockResolvedValue(
814
+ 'src/foo.ts:1:1: error TS2345: bad type\n1 error' as never,
815
+ );
816
+
817
+ const workerDecisionRunner = {
818
+ execute: vi.fn(async () => ({
819
+ planSubmission: false,
820
+ patchApplied: true,
821
+ noteLogged: false,
822
+ requestHandled: false,
823
+ invalidOutput: false,
824
+ noProgress: false,
825
+ outputTypes: ['PATCH'],
826
+ rawOutputs: [],
827
+ priorityOrder: [],
828
+ toolResults: [],
829
+ })),
830
+ };
831
+ const reactionsService = {
832
+ shouldRetry: vi.fn().mockReturnValueOnce(true).mockReturnValue(false),
833
+ shouldEscalate: vi.fn().mockReturnValue(false),
834
+ retryDelayMs: vi.fn().mockReturnValue(0),
835
+ waitBeforeRetry: vi.fn().mockResolvedValue(undefined),
836
+ buildRepairPrompt: vi.fn().mockReturnValue('repair prompt'),
837
+ recordRetry: vi.fn().mockResolvedValue(undefined),
838
+ };
839
+
840
+ const executor = new BuildWaveExecutor({
841
+ toolCaller: toolCaller as never,
842
+ workerDecisionRunner: workerDecisionRunner as never,
843
+ reactionsService: reactionsService as never,
844
+ providerMode: 'live',
845
+ repoRoot: '/repo',
846
+ });
847
+
848
+ await executor.run(['feature_a'], 1);
849
+
850
+ // context should be fetched: once pre-work + once for repair (at minimum 2 total)
851
+ expect(contextCallCount).toBeGreaterThanOrEqual(2);
852
+ });
853
+
854
+ it('GIVEN_gate_fails_and_log_readable_WHEN_repair_runs_THEN_buildRepairPrompt_receives_failureAnalysis', async () => {
855
+ const toolCaller = {
856
+ callTool: vi.fn(async (_role: string, toolName: string) => {
857
+ if (toolName === TOOLS.FEATURE_STATE_GET) {
858
+ return {
859
+ ok: true,
860
+ data: { front_matter: { status: STATUS.BUILDING, gate_retry_count: 0, version: 1 } },
861
+ };
862
+ }
863
+ if (toolName === TOOLS.FEATURE_GET_CONTEXT) {
864
+ return {
865
+ ok: true,
866
+ data: {
867
+ state: { front_matter: { status: STATUS.BUILDING, gates: {} } },
868
+ latest_evidence: {
869
+ overall: 'fail',
870
+ mode: 'fast',
871
+ coverage_status: 'unknown',
872
+ step_results: [{ name: 'lint', exit_code: 1, log_path: '/tmp/lint.log' }],
873
+ },
874
+ },
875
+ };
876
+ }
877
+ if (toolName === TOOLS.GATES_RUN) {
878
+ return { ok: true, data: { overall: 'fail', evidence_path: 'ev.json' } };
879
+ }
880
+ if (toolName === TOOLS.FEATURE_STATE_PATCH) {
881
+ return { ok: true, data: {} };
882
+ }
883
+ if (toolName === TOOLS.FEATURE_LOG_APPEND) {
884
+ return { ok: true, data: {} };
885
+ }
886
+ return { ok: true, data: {} };
887
+ }),
888
+ };
889
+ vi.spyOn(fs, 'readFile').mockResolvedValue(
890
+ 'src/a.ts:5:3: error no-unused-vars\n1 error' as never,
891
+ );
892
+
893
+ const workerDecisionRunner = {
894
+ execute: vi.fn(async () => ({
895
+ planSubmission: false,
896
+ patchApplied: true,
897
+ noteLogged: false,
898
+ requestHandled: false,
899
+ invalidOutput: false,
900
+ noProgress: false,
901
+ outputTypes: ['PATCH'],
902
+ rawOutputs: [],
903
+ priorityOrder: [],
904
+ toolResults: [],
905
+ })),
906
+ };
907
+ const capturedCtxs: unknown[] = [];
908
+ const reactionsService = {
909
+ shouldRetry: vi.fn().mockReturnValueOnce(true).mockReturnValue(false),
910
+ shouldEscalate: vi.fn().mockReturnValue(false),
911
+ retryDelayMs: vi.fn().mockReturnValue(0),
912
+ waitBeforeRetry: vi.fn().mockResolvedValue(undefined),
913
+ buildRepairPrompt: vi.fn((ctx: unknown) => {
914
+ capturedCtxs.push(ctx);
915
+ return 'repair';
916
+ }),
917
+ recordRetry: vi.fn().mockResolvedValue(undefined),
918
+ };
919
+
920
+ const executor = new BuildWaveExecutor({
921
+ toolCaller: toolCaller as never,
922
+ workerDecisionRunner: workerDecisionRunner as never,
923
+ reactionsService: reactionsService as never,
924
+ providerMode: 'live',
925
+ repoRoot: '/repo',
926
+ });
927
+
928
+ await executor.run(['feature_a'], 1);
929
+
930
+ expect(reactionsService.buildRepairPrompt).toHaveBeenCalled();
931
+ const ctx = capturedCtxs[0] as Record<string, unknown>;
932
+ expect(ctx).toHaveProperty('failureAnalysis');
933
+ const analysis = ctx.failureAnalysis as Record<string, unknown>;
934
+ expect(analysis.stepName).toBe('lint');
935
+ expect(analysis.exitCode).toBe(1);
936
+ expect(Array.isArray(analysis.affectedFiles)).toBe(true);
937
+ });
938
+
939
+ it('GIVEN_log_file_unreadable_WHEN_repair_runs_THEN_failureAnalysis_absent_and_no_crash', async () => {
940
+ const toolCaller = {
941
+ callTool: vi.fn(async (_role: string, toolName: string) => {
942
+ if (toolName === TOOLS.FEATURE_STATE_GET) {
943
+ return {
944
+ ok: true,
945
+ data: { front_matter: { status: STATUS.BUILDING, gate_retry_count: 0, version: 1 } },
946
+ };
947
+ }
948
+ if (toolName === TOOLS.FEATURE_GET_CONTEXT) {
949
+ return {
950
+ ok: true,
951
+ data: {
952
+ state: { front_matter: { status: STATUS.BUILDING, gates: {} } },
953
+ latest_evidence: {
954
+ overall: 'fail',
955
+ mode: 'fast',
956
+ coverage_status: 'unknown',
957
+ step_results: [{ name: 'lint', exit_code: 1, log_path: '/tmp/missing.log' }],
958
+ },
959
+ },
960
+ };
961
+ }
962
+ if (toolName === TOOLS.GATES_RUN) {
963
+ return { ok: true, data: { overall: 'fail', evidence_path: 'ev.json' } };
964
+ }
965
+ if (toolName === TOOLS.FEATURE_STATE_PATCH) {
966
+ return { ok: true, data: {} };
967
+ }
968
+ if (toolName === TOOLS.FEATURE_LOG_APPEND) {
969
+ return { ok: true, data: {} };
970
+ }
971
+ return { ok: true, data: {} };
972
+ }),
973
+ };
974
+ vi.spyOn(fs, 'readFile').mockRejectedValue(new Error('ENOENT'));
975
+
976
+ const workerDecisionRunner = {
977
+ execute: vi.fn(async () => ({
978
+ planSubmission: false,
979
+ patchApplied: true,
980
+ noteLogged: false,
981
+ requestHandled: false,
982
+ invalidOutput: false,
983
+ noProgress: false,
984
+ outputTypes: ['PATCH'],
985
+ rawOutputs: [],
986
+ priorityOrder: [],
987
+ toolResults: [],
988
+ })),
989
+ };
990
+ const capturedCtxs: unknown[] = [];
991
+ const reactionsService = {
992
+ shouldRetry: vi.fn().mockReturnValueOnce(true).mockReturnValue(false),
993
+ shouldEscalate: vi.fn().mockReturnValue(false),
994
+ retryDelayMs: vi.fn().mockReturnValue(0),
995
+ waitBeforeRetry: vi.fn().mockResolvedValue(undefined),
996
+ buildRepairPrompt: vi.fn((ctx: unknown) => {
997
+ capturedCtxs.push(ctx);
998
+ return 'repair';
999
+ }),
1000
+ recordRetry: vi.fn().mockResolvedValue(undefined),
1001
+ };
1002
+
1003
+ const executor = new BuildWaveExecutor({
1004
+ toolCaller: toolCaller as never,
1005
+ workerDecisionRunner: workerDecisionRunner as never,
1006
+ reactionsService: reactionsService as never,
1007
+ providerMode: 'live',
1008
+ repoRoot: '/repo',
1009
+ });
1010
+
1011
+ await expect(executor.run(['feature_a'], 1)).resolves.not.toThrow();
1012
+ const ctx = capturedCtxs[0] as Record<string, unknown>;
1013
+ expect(ctx.failureAnalysis).toBeUndefined();
1014
+ });
1015
+
1016
+ it('GIVEN_gate_fails_with_empty_step_results_WHEN_repair_runs_THEN_failureAnalysis_absent_and_no_crash', async () => {
1017
+ const toolCaller = {
1018
+ callTool: vi.fn(async (_role: string, toolName: string) => {
1019
+ if (toolName === TOOLS.FEATURE_STATE_GET) {
1020
+ return {
1021
+ ok: true,
1022
+ data: { front_matter: { status: STATUS.BUILDING, gate_retry_count: 0, version: 1 } },
1023
+ };
1024
+ }
1025
+ if (toolName === TOOLS.FEATURE_GET_CONTEXT) {
1026
+ return {
1027
+ ok: true,
1028
+ data: {
1029
+ state: { front_matter: { status: STATUS.BUILDING, gates: {} } },
1030
+ latest_evidence: {
1031
+ overall: 'fail',
1032
+ mode: 'fast',
1033
+ coverage_status: 'unknown',
1034
+ step_results: [],
1035
+ },
1036
+ },
1037
+ };
1038
+ }
1039
+ if (toolName === TOOLS.GATES_RUN) {
1040
+ return { ok: true, data: { overall: 'fail', evidence_path: 'ev.json' } };
1041
+ }
1042
+ if (toolName === TOOLS.FEATURE_STATE_PATCH) {
1043
+ return { ok: true, data: {} };
1044
+ }
1045
+ if (toolName === TOOLS.FEATURE_LOG_APPEND) {
1046
+ return { ok: true, data: {} };
1047
+ }
1048
+ return { ok: true, data: {} };
1049
+ }),
1050
+ };
1051
+
1052
+ const workerDecisionRunner = {
1053
+ execute: vi.fn(async () => ({
1054
+ planSubmission: false,
1055
+ patchApplied: true,
1056
+ noteLogged: false,
1057
+ requestHandled: false,
1058
+ invalidOutput: false,
1059
+ noProgress: false,
1060
+ outputTypes: ['PATCH'],
1061
+ rawOutputs: [],
1062
+ priorityOrder: [],
1063
+ toolResults: [],
1064
+ })),
1065
+ };
1066
+ const capturedCtxs: unknown[] = [];
1067
+ const reactionsService = {
1068
+ shouldRetry: vi.fn().mockReturnValueOnce(true).mockReturnValue(false),
1069
+ shouldEscalate: vi.fn().mockReturnValue(false),
1070
+ retryDelayMs: vi.fn().mockReturnValue(0),
1071
+ waitBeforeRetry: vi.fn().mockResolvedValue(undefined),
1072
+ buildRepairPrompt: vi.fn((ctx: unknown) => {
1073
+ capturedCtxs.push(ctx);
1074
+ return 'repair';
1075
+ }),
1076
+ recordRetry: vi.fn().mockResolvedValue(undefined),
1077
+ };
1078
+
1079
+ const executor = new BuildWaveExecutor({
1080
+ toolCaller: toolCaller as never,
1081
+ workerDecisionRunner: workerDecisionRunner as never,
1082
+ reactionsService: reactionsService as never,
1083
+ providerMode: 'live',
1084
+ repoRoot: '/repo',
1085
+ });
1086
+
1087
+ await expect(executor.run(['feature_a'], 1)).resolves.not.toThrow();
1088
+ const ctx = capturedCtxs[0] as Record<string, unknown>;
1089
+ expect(ctx.failureAnalysis).toBeUndefined();
1090
+ });
1091
+ });
573
1092
  });
574
1093
 
575
1094
  describe('QaWaveExecutor live policy branches', () => {
@@ -622,6 +1141,125 @@ describe('QaWaveExecutor live policy branches', () => {
622
1141
  expect(deps.kernel.updateFeatureSessionAssignment).not.toHaveBeenCalled();
623
1142
  });
624
1143
 
1144
+ it('GIVEN_qa_checkpoint_invalid_WHEN_run_THEN_continues_without_blocking_the_feature', async () => {
1145
+ const deps = qaDependencies(qaToolCaller());
1146
+ const executor = new QaWaveExecutor({
1147
+ kernel: deps.kernel as never,
1148
+ provider: deps.provider as never,
1149
+ toolCaller: deps.toolCaller as never,
1150
+ promptProvider: deps.promptProvider as never,
1151
+ featureClusterPatcher: deps.featureClusterPatcher as never,
1152
+ state: deps.state as never,
1153
+ providerMode: 'live',
1154
+ malformedOutputAction: 'fail_run',
1155
+ workerDecisionRunner: {
1156
+ execute: vi.fn(async () =>
1157
+ decision({
1158
+ noProgress: true,
1159
+ interactiveOutcome: 'progress_applied',
1160
+ checkpoint: {
1161
+ checkpoint_id: 'checkpoint-1',
1162
+ timestamp: '2026-03-12T16:00:00.000Z',
1163
+ files_changed: ['apps/control-plane/src/supervisor/qa-wave-executor.ts'],
1164
+ validation_status: 'invalid',
1165
+ severity: 'warning',
1166
+ diff_snapshot: '.aop/features/feature-a/checkpoints/checkpoint-1.diff',
1167
+ violations: ['outside_allowed_areas'],
1168
+ },
1169
+ }),
1170
+ ),
1171
+ },
1172
+ });
1173
+
1174
+ await expect(executor.run(['feature-a'], 1)).resolves.toBeUndefined();
1175
+
1176
+ const statePatches = deps.toolCaller.callTool.mock.calls.filter(
1177
+ (call) => call[1] === TOOLS.FEATURE_STATE_PATCH,
1178
+ );
1179
+ expect(statePatches).toHaveLength(0);
1180
+ });
1181
+
1182
+ it('GIVEN_qa_checkpoint_invalid_with_empty_violations_WHEN_run_THEN_blocks_with_fallback_message', async () => {
1183
+ const deps = qaDependencies(qaToolCaller());
1184
+ const executor = new QaWaveExecutor({
1185
+ kernel: deps.kernel as never,
1186
+ provider: deps.provider as never,
1187
+ toolCaller: deps.toolCaller as never,
1188
+ promptProvider: deps.promptProvider as never,
1189
+ featureClusterPatcher: deps.featureClusterPatcher as never,
1190
+ state: deps.state as never,
1191
+ providerMode: 'live',
1192
+ workerDecisionRunner: {
1193
+ execute: vi.fn(async () =>
1194
+ decision({
1195
+ interactiveOutcome: 'checkpoint_invalid',
1196
+ checkpoint: {
1197
+ checkpoint_id: 'checkpoint-synthetic',
1198
+ timestamp: '2026-03-21T00:00:00.000Z',
1199
+ files_changed: [],
1200
+ validation_status: 'skipped',
1201
+ diff_snapshot: '.aop/features/feature-a/checkpoints/synthetic.diff',
1202
+ violations: [],
1203
+ },
1204
+ }),
1205
+ ),
1206
+ },
1207
+ });
1208
+
1209
+ await executor.run(['feature-a'], 1);
1210
+
1211
+ const statePatches = deps.toolCaller.callTool.mock.calls.filter(
1212
+ (call) => call[1] === TOOLS.FEATURE_STATE_PATCH,
1213
+ );
1214
+ expect(statePatches).toHaveLength(1);
1215
+ const args = statePatches[0][2];
1216
+ const patch = args?.patch as Record<string, unknown> | undefined;
1217
+ const fm = patch?.front_matter as Record<string, unknown> | undefined;
1218
+ const reason = fm?.status_reason as string | undefined;
1219
+ expect(reason).toContain('checkpoint validation failed');
1220
+ expect(reason?.endsWith(': ')).toBe(false);
1221
+ });
1222
+
1223
+ it('GIVEN_qa_checkpoint_invalid_with_violations_WHEN_run_THEN_blocks_with_joined_violations', async () => {
1224
+ const deps = qaDependencies(qaToolCaller());
1225
+ const executor = new QaWaveExecutor({
1226
+ kernel: deps.kernel as never,
1227
+ provider: deps.provider as never,
1228
+ toolCaller: deps.toolCaller as never,
1229
+ promptProvider: deps.promptProvider as never,
1230
+ featureClusterPatcher: deps.featureClusterPatcher as never,
1231
+ state: deps.state as never,
1232
+ providerMode: 'live',
1233
+ workerDecisionRunner: {
1234
+ execute: vi.fn(async () =>
1235
+ decision({
1236
+ interactiveOutcome: 'checkpoint_invalid',
1237
+ checkpoint: {
1238
+ checkpoint_id: 'checkpoint-1',
1239
+ timestamp: '2026-03-21T00:00:00.000Z',
1240
+ files_changed: ['src/foo.ts'],
1241
+ validation_status: 'invalid',
1242
+ diff_snapshot: '.aop/features/feature-a/checkpoints/checkpoint-1.diff',
1243
+ violations: ['lint failure', 'type error'],
1244
+ },
1245
+ }),
1246
+ ),
1247
+ },
1248
+ });
1249
+
1250
+ await executor.run(['feature-a'], 1);
1251
+
1252
+ const statePatches = deps.toolCaller.callTool.mock.calls.filter(
1253
+ (call) => call[1] === TOOLS.FEATURE_STATE_PATCH,
1254
+ );
1255
+ expect(statePatches).toHaveLength(1);
1256
+ const args = statePatches[0][2];
1257
+ const patch = args?.patch as Record<string, unknown> | undefined;
1258
+ const fm = patch?.front_matter as Record<string, unknown> | undefined;
1259
+ const reason = fm?.status_reason as string | undefined;
1260
+ expect(reason).toContain('lint failure; type error');
1261
+ });
1262
+
625
1263
  it('GIVEN_no_progress_below_limit_WHEN_run_THEN_skips_current_cycle', async () => {
626
1264
  const toolCaller = qaToolCaller();
627
1265
  const executor = new QaWaveExecutor({
@@ -637,7 +1275,7 @@ describe('QaWaveExecutor live policy branches', () => {
637
1275
  runId: 'run:test',
638
1276
  ownerInstanceId: 'owner:test',
639
1277
  orchestratorSessionId: 'orch-session',
640
- sessionsByFeature: new Map(),
1278
+ sessionsByFeature: new Map([['feature-a', { planner: 'p-1', builder: 'b-1', qa: 'q-1' }]]),
641
1279
  } as never,
642
1280
  providerMode: 'live',
643
1281
  noProgressLimit: 3,
@@ -651,6 +1289,402 @@ describe('QaWaveExecutor live policy branches', () => {
651
1289
  expect(gateRuns).toHaveLength(0);
652
1290
  });
653
1291
 
1292
+ it('GIVEN_question_requested_WHEN_qa_wave_runs_THEN_skips_gate_and_session_rotation', async () => {
1293
+ const deps = qaDependencies(qaToolCaller());
1294
+ const executor = new QaWaveExecutor({
1295
+ kernel: deps.kernel as never,
1296
+ provider: deps.provider as never,
1297
+ toolCaller: deps.toolCaller as never,
1298
+ promptProvider: deps.promptProvider as never,
1299
+ featureClusterPatcher: deps.featureClusterPatcher as never,
1300
+ state: deps.state as never,
1301
+ providerMode: 'live',
1302
+ workerDecisionRunner: {
1303
+ execute: vi.fn(async () => decision({ questionRequested: true })),
1304
+ },
1305
+ });
1306
+
1307
+ await executor.run(['feature-a'], 1);
1308
+
1309
+ const gateRuns = deps.toolCaller.callTool.mock.calls.filter(
1310
+ (call) => call[1] === TOOLS.GATES_RUN,
1311
+ );
1312
+ expect(gateRuns).toHaveLength(0);
1313
+ expect(deps.provider.closeSession).not.toHaveBeenCalled();
1314
+ expect(deps.kernel.updateFeatureSessionAssignment).not.toHaveBeenCalled();
1315
+ });
1316
+
1317
+ it('GIVEN_context_stall_WHEN_qa_wave_runs_THEN_blocks_feature_with_request_metadata', async () => {
1318
+ const deps = qaDependencies(qaToolCaller());
1319
+ const executor = new QaWaveExecutor({
1320
+ kernel: deps.kernel as never,
1321
+ provider: deps.provider as never,
1322
+ toolCaller: deps.toolCaller as never,
1323
+ promptProvider: deps.promptProvider as never,
1324
+ featureClusterPatcher: deps.featureClusterPatcher as never,
1325
+ state: deps.state as never,
1326
+ providerMode: 'live',
1327
+ workerDecisionRunner: {
1328
+ execute: vi.fn(async () =>
1329
+ decision({
1330
+ contextStall: true,
1331
+ contextRequestCount: 3,
1332
+ lastContextRequestAt: '2026-03-16T11:00:00.000Z',
1333
+ lastContextRequestRole: 'qa',
1334
+ }),
1335
+ ),
1336
+ },
1337
+ });
1338
+
1339
+ await executor.run(['feature-a'], 1);
1340
+
1341
+ expect(deps.toolCaller.callTool).toHaveBeenCalledWith(
1342
+ 'orchestrator',
1343
+ TOOLS.FEATURE_STATE_PATCH,
1344
+ expect.objectContaining({
1345
+ feature_id: 'feature-a',
1346
+ expected_version: 9,
1347
+ patch: {
1348
+ front_matter: {
1349
+ status: STATUS.BLOCKED,
1350
+ status_reason: `${ERROR_CODES.PROVIDER_CONTEXT_STALL}: repeated context refresh with no progress`,
1351
+ context_request_count: 3,
1352
+ last_context_request_at: '2026-03-16T11:00:00.000Z',
1353
+ last_context_request_role: 'qa',
1354
+ },
1355
+ },
1356
+ }),
1357
+ );
1358
+ const logCall = deps.toolCaller.callTool.mock.calls.find(
1359
+ (call) => call[1] === TOOLS.FEATURE_LOG_APPEND,
1360
+ );
1361
+ expect(JSON.parse(String(logCall?.[2]?.note ?? '{}'))).toMatchObject({
1362
+ phase: 'qa',
1363
+ error_code: ERROR_CODES.PROVIDER_CONTEXT_STALL,
1364
+ context_request_count: 3,
1365
+ last_context_request_role: 'qa',
1366
+ });
1367
+ });
1368
+
1369
+ it('GIVEN_interactive_no_new_work_with_stale_fast_evidence_WHEN_run_THEN_reruns_fast_gate', async () => {
1370
+ const toolCaller = {
1371
+ callTool: vi.fn(async (role: string, toolName: string) => {
1372
+ if (toolName === TOOLS.FEATURE_STATE_GET && role === 'builder') {
1373
+ return {
1374
+ ok: true,
1375
+ data: {
1376
+ front_matter: {
1377
+ status: STATUS.BUILDING,
1378
+ gate_retry_count: 0,
1379
+ version: 3,
1380
+ checkpoints: [
1381
+ {
1382
+ checkpoint_id: 'checkpoint-new',
1383
+ timestamp: '2026-03-12T16:00:00.000Z',
1384
+ files_changed: ['apps/control-plane/src/supervisor/run-coordinator.ts'],
1385
+ validation_status: 'valid',
1386
+ diff_snapshot: '.aop/features/feature-a/checkpoints/checkpoint-new.diff',
1387
+ diff_hash: 'diff-new',
1388
+ },
1389
+ ],
1390
+ },
1391
+ },
1392
+ };
1393
+ }
1394
+ if (toolName === TOOLS.FEATURE_STATE_GET && role === 'orchestrator') {
1395
+ return { ok: true, data: { front_matter: { version: 7 } } };
1396
+ }
1397
+ if (toolName === TOOLS.FEATURE_GET_CONTEXT) {
1398
+ return {
1399
+ ok: true,
1400
+ data: {
1401
+ state: {
1402
+ front_matter: {
1403
+ checkpoints: [
1404
+ {
1405
+ checkpoint_id: 'checkpoint-new',
1406
+ timestamp: '2026-03-12T16:00:00.000Z',
1407
+ files_changed: ['apps/control-plane/src/supervisor/run-coordinator.ts'],
1408
+ validation_status: 'valid',
1409
+ diff_snapshot: '.aop/features/feature-a/checkpoints/checkpoint-new.diff',
1410
+ diff_hash: 'diff-new',
1411
+ },
1412
+ ],
1413
+ },
1414
+ },
1415
+ gate_evidence_by_mode: {
1416
+ fast: {
1417
+ mode: 'fast',
1418
+ overall: 'fail',
1419
+ input_diff_hash: 'diff-old',
1420
+ },
1421
+ },
1422
+ latest_evidence: {
1423
+ mode: 'fast',
1424
+ overall: 'fail',
1425
+ input_diff_hash: 'diff-old',
1426
+ },
1427
+ },
1428
+ };
1429
+ }
1430
+ if (toolName === TOOLS.GATES_RUN) {
1431
+ return { ok: true, data: { overall: 'pass', evidence_path: 'evidence.json' } };
1432
+ }
1433
+ return { ok: true, data: {} };
1434
+ }),
1435
+ };
1436
+ const executor = new BuildWaveExecutor({
1437
+ toolCaller: toolCaller as never,
1438
+ providerMode: 'live',
1439
+ workerDecisionRunner: {
1440
+ execute: vi.fn(async () =>
1441
+ decision({
1442
+ noProgress: true,
1443
+ interactiveOutcome: 'no_new_work',
1444
+ }),
1445
+ ),
1446
+ },
1447
+ });
1448
+
1449
+ await executor.run(['feature-a'], 1);
1450
+
1451
+ const gateRuns = toolCaller.callTool.mock.calls.filter((call) => call[1] === TOOLS.GATES_RUN);
1452
+ expect(gateRuns).toHaveLength(1);
1453
+ const patchCalls = toolCaller.callTool.mock.calls.filter(
1454
+ (call) => call[1] === TOOLS.FEATURE_STATE_PATCH,
1455
+ );
1456
+ expect(patchCalls).toHaveLength(0);
1457
+ });
1458
+
1459
+ it('GIVEN_interactive_no_new_work_with_empty_fast_checkpoint_diff_WHEN_run_THEN_skips_fast_gate_refresh', async () => {
1460
+ const toolCaller = {
1461
+ callTool: vi.fn(async (role: string, toolName: string) => {
1462
+ if (toolName === TOOLS.FEATURE_STATE_GET && role === 'builder') {
1463
+ return {
1464
+ ok: true,
1465
+ data: {
1466
+ front_matter: {
1467
+ status: STATUS.BUILDING,
1468
+ gate_retry_count: 0,
1469
+ version: 3,
1470
+ checkpoints: [
1471
+ {
1472
+ checkpoint_id: 'checkpoint-empty',
1473
+ timestamp: '2026-03-12T16:00:00.000Z',
1474
+ files_changed: [],
1475
+ validation_status: 'valid',
1476
+ diff_snapshot: '.aop/features/feature-a/checkpoints/checkpoint-empty.diff',
1477
+ diff_hash: 'e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855',
1478
+ net_new_worktree_change: false,
1479
+ },
1480
+ ],
1481
+ },
1482
+ },
1483
+ };
1484
+ }
1485
+ if (toolName === TOOLS.FEATURE_GET_CONTEXT) {
1486
+ return {
1487
+ ok: true,
1488
+ data: {
1489
+ state: {
1490
+ front_matter: {
1491
+ checkpoints: [
1492
+ {
1493
+ checkpoint_id: 'checkpoint-empty',
1494
+ timestamp: '2026-03-12T16:00:00.000Z',
1495
+ files_changed: [],
1496
+ validation_status: 'valid',
1497
+ diff_snapshot: '.aop/features/feature-a/checkpoints/checkpoint-empty.diff',
1498
+ diff_hash: 'e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855',
1499
+ net_new_worktree_change: false,
1500
+ },
1501
+ ],
1502
+ },
1503
+ },
1504
+ gate_evidence_by_mode: {},
1505
+ latest_evidence: null,
1506
+ },
1507
+ };
1508
+ }
1509
+ return { ok: true, data: {} };
1510
+ }),
1511
+ };
1512
+ const executor = new BuildWaveExecutor({
1513
+ toolCaller: toolCaller as never,
1514
+ providerMode: 'live',
1515
+ workerDecisionRunner: {
1516
+ execute: vi.fn(async () =>
1517
+ decision({
1518
+ noProgress: true,
1519
+ interactiveOutcome: 'no_new_work',
1520
+ }),
1521
+ ),
1522
+ },
1523
+ });
1524
+
1525
+ await executor.run(['feature-a'], 1);
1526
+
1527
+ const gateRuns = toolCaller.callTool.mock.calls.filter((call) => call[1] === TOOLS.GATES_RUN);
1528
+ expect(gateRuns).toHaveLength(0);
1529
+ });
1530
+
1531
+ it('GIVEN_recovered_fast_gate_refresh_reason_WHEN_run_THEN_bypasses_worker_and_reruns_gate', async () => {
1532
+ const workerDecisionRunner = {
1533
+ execute: vi.fn(async () =>
1534
+ decision({
1535
+ noProgress: true,
1536
+ interactiveOutcome: 'no_new_work',
1537
+ }),
1538
+ ),
1539
+ };
1540
+ const toolCaller = {
1541
+ callTool: vi.fn(async (role: string, toolName: string) => {
1542
+ if (toolName === TOOLS.FEATURE_STATE_GET && role === 'builder') {
1543
+ return {
1544
+ ok: true,
1545
+ data: {
1546
+ front_matter: {
1547
+ status: STATUS.BUILDING,
1548
+ status_reason: 'recovered_from_blocked:gate_refresh:fast',
1549
+ gate_retry_count: 0,
1550
+ version: 3,
1551
+ checkpoints: [
1552
+ {
1553
+ checkpoint_id: 'checkpoint-new',
1554
+ timestamp: '2026-03-12T16:00:00.000Z',
1555
+ files_changed: ['apps/control-plane/src/supervisor/build-wave-executor.ts'],
1556
+ validation_status: 'valid',
1557
+ diff_snapshot: '.aop/features/feature-a/checkpoints/checkpoint-new.diff',
1558
+ diff_hash: 'diff-new',
1559
+ },
1560
+ ],
1561
+ },
1562
+ },
1563
+ };
1564
+ }
1565
+ if (toolName === TOOLS.FEATURE_GET_CONTEXT) {
1566
+ return {
1567
+ ok: true,
1568
+ data: {
1569
+ state: {
1570
+ front_matter: {
1571
+ checkpoints: [
1572
+ {
1573
+ checkpoint_id: 'checkpoint-new',
1574
+ timestamp: '2026-03-12T16:00:00.000Z',
1575
+ files_changed: ['apps/control-plane/src/supervisor/build-wave-executor.ts'],
1576
+ validation_status: 'valid',
1577
+ diff_snapshot: '.aop/features/feature-a/checkpoints/checkpoint-new.diff',
1578
+ diff_hash: 'diff-new',
1579
+ },
1580
+ ],
1581
+ },
1582
+ },
1583
+ gate_evidence_by_mode: {
1584
+ fast: {
1585
+ mode: 'fast',
1586
+ overall: 'fail',
1587
+ input_diff_hash: 'diff-old',
1588
+ },
1589
+ },
1590
+ },
1591
+ };
1592
+ }
1593
+ if (toolName === TOOLS.GATES_RUN) {
1594
+ return { ok: true, data: { overall: 'pass', evidence_path: 'evidence.json' } };
1595
+ }
1596
+ return { ok: true, data: {} };
1597
+ }),
1598
+ };
1599
+
1600
+ const executor = new BuildWaveExecutor({
1601
+ toolCaller: toolCaller as never,
1602
+ providerMode: 'live',
1603
+ workerDecisionRunner: workerDecisionRunner as never,
1604
+ });
1605
+
1606
+ await executor.run(['feature-a'], 1);
1607
+
1608
+ expect(workerDecisionRunner.execute).not.toHaveBeenCalled();
1609
+ const gateRuns = toolCaller.callTool.mock.calls.filter((call) => call[1] === TOOLS.GATES_RUN);
1610
+ expect(gateRuns).toHaveLength(1);
1611
+ });
1612
+
1613
+ it('GIVEN_recovered_fast_gate_refresh_reason_with_current_evidence_WHEN_run_THEN_worker_still_executes', async () => {
1614
+ const workerDecisionRunner = {
1615
+ execute: vi.fn(async () => decision({ patchApplied: true })),
1616
+ };
1617
+ const toolCaller = {
1618
+ callTool: vi.fn(async (role: string, toolName: string) => {
1619
+ if (toolName === TOOLS.FEATURE_STATE_GET && role === 'builder') {
1620
+ return {
1621
+ ok: true,
1622
+ data: {
1623
+ front_matter: {
1624
+ status: STATUS.BUILDING,
1625
+ status_reason: 'recovered_from_blocked:gate_refresh:fast',
1626
+ gate_retry_count: 0,
1627
+ version: 3,
1628
+ checkpoints: [
1629
+ {
1630
+ checkpoint_id: 'checkpoint-new',
1631
+ timestamp: '2026-03-12T16:00:00.000Z',
1632
+ files_changed: ['apps/control-plane/src/supervisor/build-wave-executor.ts'],
1633
+ validation_status: 'valid',
1634
+ diff_snapshot: '.aop/features/feature-a/checkpoints/checkpoint-new.diff',
1635
+ diff_hash: 'diff-new',
1636
+ },
1637
+ ],
1638
+ },
1639
+ },
1640
+ };
1641
+ }
1642
+ if (toolName === TOOLS.FEATURE_GET_CONTEXT) {
1643
+ return {
1644
+ ok: true,
1645
+ data: {
1646
+ state: {
1647
+ front_matter: {
1648
+ checkpoints: [
1649
+ {
1650
+ checkpoint_id: 'checkpoint-new',
1651
+ timestamp: '2026-03-12T16:00:00.000Z',
1652
+ files_changed: ['apps/control-plane/src/supervisor/build-wave-executor.ts'],
1653
+ validation_status: 'valid',
1654
+ diff_snapshot: '.aop/features/feature-a/checkpoints/checkpoint-new.diff',
1655
+ diff_hash: 'diff-new',
1656
+ },
1657
+ ],
1658
+ },
1659
+ },
1660
+ gate_evidence_by_mode: {
1661
+ fast: {
1662
+ mode: 'fast',
1663
+ overall: 'pass',
1664
+ input_diff_hash: 'diff-new',
1665
+ },
1666
+ },
1667
+ },
1668
+ };
1669
+ }
1670
+ if (toolName === TOOLS.GATES_RUN) {
1671
+ return { ok: true, data: { overall: 'pass', evidence_path: 'evidence.json' } };
1672
+ }
1673
+ return { ok: true, data: {} };
1674
+ }),
1675
+ };
1676
+
1677
+ const executor = new BuildWaveExecutor({
1678
+ toolCaller: toolCaller as never,
1679
+ providerMode: 'live',
1680
+ workerDecisionRunner: workerDecisionRunner as never,
1681
+ });
1682
+
1683
+ await executor.run(['feature-a'], 1);
1684
+
1685
+ expect(workerDecisionRunner.execute).toHaveBeenCalledTimes(1);
1686
+ });
1687
+
654
1688
  it('GIVEN_no_progress_at_limit_and_fail_run_policy_WHEN_run_THEN_throws_provider_no_progress', async () => {
655
1689
  const toolCaller = qaToolCaller();
656
1690
  const executor = new QaWaveExecutor({
@@ -666,7 +1700,7 @@ describe('QaWaveExecutor live policy branches', () => {
666
1700
  runId: 'run:test',
667
1701
  ownerInstanceId: 'owner:test',
668
1702
  orchestratorSessionId: 'orch-session',
669
- sessionsByFeature: new Map(),
1703
+ sessionsByFeature: new Map([['feature-a', { planner: 'p-1', builder: 'b-1', qa: 'q-1' }]]),
670
1704
  } as never,
671
1705
  providerMode: 'live',
672
1706
  noProgressLimit: 1,
@@ -681,6 +1715,377 @@ describe('QaWaveExecutor live policy branches', () => {
681
1715
  });
682
1716
  });
683
1717
 
1718
+ it('GIVEN_interactive_no_new_work_with_stale_full_evidence_WHEN_run_THEN_reruns_full_gate', async () => {
1719
+ const toolCaller = {
1720
+ callTool: vi.fn(async (role: string, toolName: string) => {
1721
+ if (toolName === TOOLS.FEATURE_STATE_GET && role === 'qa') {
1722
+ return {
1723
+ ok: true,
1724
+ data: {
1725
+ front_matter: {
1726
+ status: STATUS.QA,
1727
+ gate_retry_count: 0,
1728
+ version: 5,
1729
+ checkpoints: [
1730
+ {
1731
+ checkpoint_id: 'checkpoint-qa',
1732
+ timestamp: '2026-03-12T16:00:00.000Z',
1733
+ files_changed: ['apps/control-plane/src/supervisor/qa-wave-executor.ts'],
1734
+ validation_status: 'valid',
1735
+ diff_snapshot: '.aop/features/feature-a/checkpoints/checkpoint-qa.diff',
1736
+ diff_hash: 'qa-diff-new',
1737
+ },
1738
+ ],
1739
+ },
1740
+ },
1741
+ };
1742
+ }
1743
+ if (toolName === TOOLS.FEATURE_STATE_GET && role === 'orchestrator') {
1744
+ return { ok: true, data: { front_matter: { version: 9 } } };
1745
+ }
1746
+ if (toolName === TOOLS.FEATURE_GET_CONTEXT) {
1747
+ return {
1748
+ ok: true,
1749
+ data: {
1750
+ state: {
1751
+ front_matter: {
1752
+ checkpoints: [
1753
+ {
1754
+ checkpoint_id: 'checkpoint-qa',
1755
+ timestamp: '2026-03-12T16:00:00.000Z',
1756
+ files_changed: ['apps/control-plane/src/supervisor/qa-wave-executor.ts'],
1757
+ validation_status: 'valid',
1758
+ diff_snapshot: '.aop/features/feature-a/checkpoints/checkpoint-qa.diff',
1759
+ diff_hash: 'qa-diff-new',
1760
+ },
1761
+ ],
1762
+ },
1763
+ },
1764
+ gate_evidence_by_mode: {
1765
+ full: {
1766
+ mode: 'full',
1767
+ overall: 'fail',
1768
+ input_diff_hash: 'qa-diff-old',
1769
+ },
1770
+ },
1771
+ },
1772
+ };
1773
+ }
1774
+ if (toolName === TOOLS.GATES_RUN) {
1775
+ return { ok: true, data: { overall: 'pass', evidence_path: 'qa-evidence.json' } };
1776
+ }
1777
+ return { ok: true, data: {} };
1778
+ }),
1779
+ };
1780
+ const executor = new QaWaveExecutor({
1781
+ kernel: { updateFeatureSessionAssignment: vi.fn() } as never,
1782
+ provider: {
1783
+ closeSession: vi.fn(),
1784
+ createSession: vi.fn(async () => ({ session_id: 'new-qa-session' })),
1785
+ } as never,
1786
+ toolCaller: toolCaller as never,
1787
+ promptProvider: {
1788
+ loadRolePrompts: vi.fn(async () => ({ planner: null, builder: null, qa: null })),
1789
+ } as never,
1790
+ featureClusterPatcher: { patchFeatureCluster: vi.fn() } as never,
1791
+ state: {
1792
+ runId: 'run:test',
1793
+ ownerInstanceId: 'owner:test',
1794
+ orchestratorSessionId: 'orch-session',
1795
+ sessionsByFeature: new Map(),
1796
+ } as never,
1797
+ providerMode: 'live',
1798
+ workerDecisionRunner: {
1799
+ execute: vi.fn(async () =>
1800
+ decision({
1801
+ noProgress: true,
1802
+ interactiveOutcome: 'no_new_work',
1803
+ }),
1804
+ ),
1805
+ },
1806
+ });
1807
+
1808
+ await executor.run(['feature-a'], 1);
1809
+
1810
+ const gateRuns = toolCaller.callTool.mock.calls.filter((call) => call[1] === TOOLS.GATES_RUN);
1811
+ expect(gateRuns).toHaveLength(1);
1812
+ const patchCalls = toolCaller.callTool.mock.calls.filter(
1813
+ (call) => call[1] === TOOLS.FEATURE_STATE_PATCH,
1814
+ );
1815
+ expect(patchCalls).toHaveLength(0);
1816
+ });
1817
+
1818
+ it('GIVEN_interactive_no_new_work_with_empty_full_checkpoint_diff_WHEN_run_THEN_skips_full_gate_refresh', async () => {
1819
+ const toolCaller = {
1820
+ callTool: vi.fn(async (role: string, toolName: string) => {
1821
+ if (toolName === TOOLS.FEATURE_STATE_GET && role === 'qa') {
1822
+ return {
1823
+ ok: true,
1824
+ data: {
1825
+ front_matter: {
1826
+ status: STATUS.QA,
1827
+ gate_retry_count: 0,
1828
+ version: 5,
1829
+ checkpoints: [
1830
+ {
1831
+ checkpoint_id: 'checkpoint-empty',
1832
+ timestamp: '2026-03-12T16:00:00.000Z',
1833
+ files_changed: [],
1834
+ validation_status: 'valid',
1835
+ diff_snapshot: '.aop/features/feature-a/checkpoints/checkpoint-empty.diff',
1836
+ diff_hash: 'e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855',
1837
+ net_new_worktree_change: false,
1838
+ },
1839
+ ],
1840
+ },
1841
+ },
1842
+ };
1843
+ }
1844
+ if (toolName === TOOLS.FEATURE_GET_CONTEXT) {
1845
+ return {
1846
+ ok: true,
1847
+ data: {
1848
+ state: {
1849
+ front_matter: {
1850
+ checkpoints: [
1851
+ {
1852
+ checkpoint_id: 'checkpoint-empty',
1853
+ timestamp: '2026-03-12T16:00:00.000Z',
1854
+ files_changed: [],
1855
+ validation_status: 'valid',
1856
+ diff_snapshot: '.aop/features/feature-a/checkpoints/checkpoint-empty.diff',
1857
+ diff_hash: 'e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855',
1858
+ net_new_worktree_change: false,
1859
+ },
1860
+ ],
1861
+ },
1862
+ },
1863
+ gate_evidence_by_mode: {},
1864
+ },
1865
+ };
1866
+ }
1867
+ return { ok: true, data: {} };
1868
+ }),
1869
+ };
1870
+ const executor = new QaWaveExecutor({
1871
+ kernel: { updateFeatureSessionAssignment: vi.fn() } as never,
1872
+ provider: {
1873
+ closeSession: vi.fn(),
1874
+ createSession: vi.fn(async () => ({ session_id: 'new-qa-session' })),
1875
+ } as never,
1876
+ toolCaller: toolCaller as never,
1877
+ promptProvider: {
1878
+ loadRolePrompts: vi.fn(async () => ({ planner: null, builder: null, qa: null })),
1879
+ } as never,
1880
+ featureClusterPatcher: { patchFeatureCluster: vi.fn() } as never,
1881
+ state: {
1882
+ runId: 'run:test',
1883
+ ownerInstanceId: 'owner:test',
1884
+ orchestratorSessionId: 'orch-session',
1885
+ sessionsByFeature: new Map(),
1886
+ } as never,
1887
+ providerMode: 'live',
1888
+ workerDecisionRunner: {
1889
+ execute: vi.fn(async () =>
1890
+ decision({
1891
+ noProgress: true,
1892
+ interactiveOutcome: 'no_new_work',
1893
+ }),
1894
+ ),
1895
+ },
1896
+ });
1897
+
1898
+ await executor.run(['feature-a'], 1);
1899
+
1900
+ const gateRuns = toolCaller.callTool.mock.calls.filter((call) => call[1] === TOOLS.GATES_RUN);
1901
+ expect(gateRuns).toHaveLength(0);
1902
+ });
1903
+
1904
+ it('GIVEN_recovered_full_gate_refresh_reason_WHEN_run_THEN_bypasses_worker_and_reruns_gate', async () => {
1905
+ const workerDecisionRunner = {
1906
+ execute: vi.fn(async () =>
1907
+ decision({
1908
+ noProgress: true,
1909
+ interactiveOutcome: 'no_new_work',
1910
+ }),
1911
+ ),
1912
+ };
1913
+ const toolCaller = {
1914
+ callTool: vi.fn(async (role: string, toolName: string) => {
1915
+ if (toolName === TOOLS.FEATURE_STATE_GET && role === 'qa') {
1916
+ return {
1917
+ ok: true,
1918
+ data: {
1919
+ front_matter: {
1920
+ status: STATUS.QA,
1921
+ status_reason: 'recovered_from_blocked:gate_refresh:full',
1922
+ gate_retry_count: 0,
1923
+ version: 5,
1924
+ checkpoints: [
1925
+ {
1926
+ checkpoint_id: 'checkpoint-qa',
1927
+ timestamp: '2026-03-12T16:00:00.000Z',
1928
+ files_changed: ['apps/control-plane/src/supervisor/qa-wave-executor.ts'],
1929
+ validation_status: 'valid',
1930
+ diff_snapshot: '.aop/features/feature-a/checkpoints/checkpoint-qa.diff',
1931
+ diff_hash: 'qa-diff-new',
1932
+ },
1933
+ ],
1934
+ },
1935
+ },
1936
+ };
1937
+ }
1938
+ if (toolName === TOOLS.FEATURE_GET_CONTEXT) {
1939
+ return {
1940
+ ok: true,
1941
+ data: {
1942
+ state: {
1943
+ front_matter: {
1944
+ checkpoints: [
1945
+ {
1946
+ checkpoint_id: 'checkpoint-qa',
1947
+ timestamp: '2026-03-12T16:00:00.000Z',
1948
+ files_changed: ['apps/control-plane/src/supervisor/qa-wave-executor.ts'],
1949
+ validation_status: 'valid',
1950
+ diff_snapshot: '.aop/features/feature-a/checkpoints/checkpoint-qa.diff',
1951
+ diff_hash: 'qa-diff-new',
1952
+ },
1953
+ ],
1954
+ },
1955
+ },
1956
+ gate_evidence_by_mode: {
1957
+ full: {
1958
+ mode: 'full',
1959
+ overall: 'fail',
1960
+ input_diff_hash: 'qa-diff-old',
1961
+ },
1962
+ },
1963
+ },
1964
+ };
1965
+ }
1966
+ if (toolName === TOOLS.GATES_RUN) {
1967
+ return { ok: true, data: { overall: 'pass', evidence_path: 'qa-evidence.json' } };
1968
+ }
1969
+ return { ok: true, data: {} };
1970
+ }),
1971
+ };
1972
+ const executor = new QaWaveExecutor({
1973
+ kernel: { updateFeatureSessionAssignment: vi.fn() } as never,
1974
+ provider: {
1975
+ closeSession: vi.fn(),
1976
+ createSession: vi.fn(async () => ({ session_id: 'new-qa-session' })),
1977
+ } as never,
1978
+ toolCaller: toolCaller as never,
1979
+ promptProvider: {
1980
+ loadRolePrompts: vi.fn(async () => ({ planner: null, builder: null, qa: null })),
1981
+ } as never,
1982
+ featureClusterPatcher: { patchFeatureCluster: vi.fn() } as never,
1983
+ state: {
1984
+ runId: 'run:test',
1985
+ ownerInstanceId: 'owner:test',
1986
+ orchestratorSessionId: 'orch-session',
1987
+ sessionsByFeature: new Map(),
1988
+ } as never,
1989
+ providerMode: 'live',
1990
+ workerDecisionRunner: workerDecisionRunner as never,
1991
+ });
1992
+
1993
+ await executor.run(['feature-a'], 1);
1994
+
1995
+ expect(workerDecisionRunner.execute).not.toHaveBeenCalled();
1996
+ const gateRuns = toolCaller.callTool.mock.calls.filter((call) => call[1] === TOOLS.GATES_RUN);
1997
+ expect(gateRuns).toHaveLength(1);
1998
+ });
1999
+
2000
+ it('GIVEN_recovered_full_gate_refresh_reason_with_current_evidence_WHEN_run_THEN_worker_still_executes', async () => {
2001
+ const workerDecisionRunner = {
2002
+ execute: vi.fn(async () => decision({ patchApplied: true })),
2003
+ };
2004
+ const toolCaller = {
2005
+ callTool: vi.fn(async (role: string, toolName: string) => {
2006
+ if (toolName === TOOLS.FEATURE_STATE_GET && role === 'qa') {
2007
+ return {
2008
+ ok: true,
2009
+ data: {
2010
+ front_matter: {
2011
+ status: STATUS.QA,
2012
+ status_reason: 'recovered_from_blocked:gate_refresh:full',
2013
+ gate_retry_count: 0,
2014
+ version: 5,
2015
+ checkpoints: [
2016
+ {
2017
+ checkpoint_id: 'checkpoint-qa',
2018
+ timestamp: '2026-03-12T16:00:00.000Z',
2019
+ files_changed: ['apps/control-plane/src/supervisor/qa-wave-executor.ts'],
2020
+ validation_status: 'valid',
2021
+ diff_snapshot: '.aop/features/feature-a/checkpoints/checkpoint-qa.diff',
2022
+ diff_hash: 'qa-diff-new',
2023
+ },
2024
+ ],
2025
+ },
2026
+ },
2027
+ };
2028
+ }
2029
+ if (toolName === TOOLS.FEATURE_GET_CONTEXT) {
2030
+ return {
2031
+ ok: true,
2032
+ data: {
2033
+ state: {
2034
+ front_matter: {
2035
+ checkpoints: [
2036
+ {
2037
+ checkpoint_id: 'checkpoint-qa',
2038
+ timestamp: '2026-03-12T16:00:00.000Z',
2039
+ files_changed: ['apps/control-plane/src/supervisor/qa-wave-executor.ts'],
2040
+ validation_status: 'valid',
2041
+ diff_snapshot: '.aop/features/feature-a/checkpoints/checkpoint-qa.diff',
2042
+ diff_hash: 'qa-diff-new',
2043
+ },
2044
+ ],
2045
+ },
2046
+ },
2047
+ gate_evidence_by_mode: {
2048
+ full: {
2049
+ mode: 'full',
2050
+ overall: 'pass',
2051
+ input_diff_hash: 'qa-diff-new',
2052
+ },
2053
+ },
2054
+ },
2055
+ };
2056
+ }
2057
+ if (toolName === TOOLS.GATES_RUN) {
2058
+ return { ok: true, data: { overall: 'pass', evidence_path: 'qa-evidence.json' } };
2059
+ }
2060
+ return { ok: true, data: {} };
2061
+ }),
2062
+ };
2063
+ const executor = new QaWaveExecutor({
2064
+ kernel: { updateFeatureSessionAssignment: vi.fn() } as never,
2065
+ provider: {
2066
+ closeSession: vi.fn(),
2067
+ createSession: vi.fn(async () => ({ session_id: 'new-qa-session' })),
2068
+ } as never,
2069
+ toolCaller: toolCaller as never,
2070
+ promptProvider: {
2071
+ loadRolePrompts: vi.fn(async () => ({ planner: null, builder: null, qa: null })),
2072
+ } as never,
2073
+ featureClusterPatcher: { patchFeatureCluster: vi.fn() } as never,
2074
+ state: {
2075
+ runId: 'run:test',
2076
+ ownerInstanceId: 'owner:test',
2077
+ orchestratorSessionId: 'orch-session',
2078
+ sessionsByFeature: new Map(),
2079
+ } as never,
2080
+ providerMode: 'live',
2081
+ workerDecisionRunner: workerDecisionRunner as never,
2082
+ });
2083
+
2084
+ await executor.run(['feature-a'], 1);
2085
+
2086
+ expect(workerDecisionRunner.execute).toHaveBeenCalledTimes(1);
2087
+ });
2088
+
684
2089
  it('GIVEN_plan_conformance_stall_and_fail_run_policy_WHEN_run_THEN_throws_provider_plan_conformance_stall', async () => {
685
2090
  const toolCaller = qaToolCaller();
686
2091
  const executor = new QaWaveExecutor({
@@ -737,7 +2142,7 @@ describe('QaWaveExecutor live policy branches', () => {
737
2142
  });
738
2143
  });
739
2144
 
740
- it('GIVEN_runtime_stall_timeout_and_block_policy_WHEN_run_THEN_blocks_feature_and_skips_rotation', async () => {
2145
+ it('GIVEN_runtime_stall_timeout_and_block_policy_WHEN_run_twice_THEN_blocks_feature_and_skips_rotation', async () => {
741
2146
  const deps = qaDependencies(qaToolCaller());
742
2147
  const stallTimeoutError = new Error('stall timeout') as Error & { code?: string };
743
2148
  stallTimeoutError.code = ERROR_CODES.PROVIDER_STALL_TIMEOUT;
@@ -757,6 +2162,11 @@ describe('QaWaveExecutor live policy branches', () => {
757
2162
  },
758
2163
  });
759
2164
 
2165
+ // First run: auto-retry once at same phase, error re-thrown
2166
+ await expect(executor.run(['feature-a'], 1)).rejects.toMatchObject({
2167
+ code: ERROR_CODES.PROVIDER_STALL_TIMEOUT,
2168
+ });
2169
+ // Second run: timeout retry used, now blocks
760
2170
  await executor.run(['feature-a'], 1);
761
2171
  const statePatches = deps.toolCaller.callTool.mock.calls.filter(
762
2172
  (call) => call[1] === TOOLS.FEATURE_STATE_PATCH,