agentic-orchestrator 0.1.28 → 0.2.1

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 +111 -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 +84 -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 +362 -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 +154 -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 +103 -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 +881 -60
  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 +114 -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
@@ -0,0 +1,1550 @@
1
+ import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest';
2
+ import fs from 'node:fs/promises';
3
+ import os from 'node:os';
4
+ import path from 'node:path';
5
+
6
+ const mergeExecuteMock = vi.hoisted(() => vi.fn());
7
+ const resumeExecuteMock = vi.hoisted(() => vi.fn());
8
+ const toolClientCallMock = vi.hoisted(() => vi.fn());
9
+ const supervisorStartMock = vi.hoisted(() => vi.fn());
10
+ const kernelRuntimeSessionsMock = vi.hoisted(() => vi.fn());
11
+ const kernelIsRunLeaseFreshMock = vi.hoisted(() => vi.fn((_value?: unknown) => false));
12
+ const kernelFeatureStateGetMock = vi.hoisted(() => vi.fn());
13
+ const kernelRedrivePreparedConflictMock = vi.hoisted(() =>
14
+ vi.fn(async (_featureId: string) => false),
15
+ );
16
+
17
+ vi.mock('../src/core/kernel.js', () => ({
18
+ AopKernel: class AopKernel {
19
+ async ensureLoaded() {
20
+ return undefined;
21
+ }
22
+
23
+ getAgentsConfig() {
24
+ return {};
25
+ }
26
+
27
+ getPolicySnapshot() {
28
+ return {};
29
+ }
30
+
31
+ async getRuntimeSessions() {
32
+ return await kernelRuntimeSessionsMock();
33
+ }
34
+
35
+ async featureStateGet(featureId: string) {
36
+ return await kernelFeatureStateGetMock(featureId);
37
+ }
38
+
39
+ isRunLeaseFresh(value: unknown) {
40
+ return kernelIsRunLeaseFreshMock(value);
41
+ }
42
+
43
+ async redrivePreparedReconcilerConflict(featureId: string) {
44
+ return await kernelRedrivePreparedConflictMock(featureId);
45
+ }
46
+
47
+ setProvider() {
48
+ // no-op in tests
49
+ }
50
+ },
51
+ }));
52
+
53
+ vi.mock('../src/mcp/runtime-factory.js', () => ({
54
+ createToolingRuntime: vi.fn(async () => ({})),
55
+ resolveToolClient: vi.fn(() => ({ call: toolClientCallMock })),
56
+ }));
57
+
58
+ vi.mock('../src/providers/providers.js', () => ({
59
+ NullWorkerProvider: class NullWorkerProvider {
60
+ constructor(_selection: unknown) {
61
+ // no-op in tests
62
+ }
63
+ },
64
+ resolveProviderSelection: vi.fn(() => ({
65
+ provider: 'codex',
66
+ model: 'gpt-5.4',
67
+ provider_config_env: null,
68
+ provider_config_ref: null,
69
+ })),
70
+ resolveProviderSelectionsByRole: vi.fn(() => ({})),
71
+ }));
72
+
73
+ vi.mock('../src/providers/worker-provider-factory.js', () => ({
74
+ DefaultWorkerProviderFactory: class DefaultWorkerProviderFactory {
75
+ create() {
76
+ return {
77
+ selection: {
78
+ provider: 'codex',
79
+ model: 'gpt-5.4',
80
+ provider_config_env: null,
81
+ provider_config_ref: null,
82
+ },
83
+ };
84
+ }
85
+ },
86
+ resolveWorkerProviderMode: vi.fn(() => 'live'),
87
+ resolveWorkerProviderObservability: vi.fn(() => ({})),
88
+ resolveWorkerProviderPolicy: vi.fn(() => ({})),
89
+ resolveWorkerProviderRuntime: vi.fn(() => ({})),
90
+ resolvePromptOverlayDir: vi.fn(
91
+ (repoRoot: string) => `${repoRoot}/config/agentic/orchestrator/prompts/overrides`,
92
+ ),
93
+ }));
94
+
95
+ vi.mock('../src/supervisor/runtime.js', () => ({
96
+ SupervisorRuntime: class SupervisorRuntime {
97
+ async start(features: Array<{ feature_id: string }>) {
98
+ return await supervisorStartMock(features);
99
+ }
100
+ },
101
+ }));
102
+
103
+ vi.mock('../src/cli/merge-command-handler.js', () => ({
104
+ MergeCommandHandler: class MergeCommandHandler {
105
+ async execute() {
106
+ return await mergeExecuteMock();
107
+ }
108
+ },
109
+ }));
110
+
111
+ vi.mock('../src/cli/resume-command-handler.js', () => ({
112
+ ResumeCommandHandler: class ResumeCommandHandler {
113
+ async execute() {
114
+ return await resumeExecuteMock();
115
+ }
116
+ },
117
+ }));
118
+
119
+ import {
120
+ buildReconcilerHandoffResponse,
121
+ buildResumeAcceptedResponse,
122
+ buildRuntimeBusyResponse,
123
+ DashboardRuntimeRunner,
124
+ } from '../src/cli/dashboard-runtime-runner.js';
125
+
126
+ describe('dashboard runtime runner helpers', () => {
127
+ it('builds runtime handoff responses for resume submission', () => {
128
+ expect(
129
+ buildResumeAcceptedResponse({
130
+ featureId: 'feature_a',
131
+ started: true,
132
+ alreadyRunning: false,
133
+ }),
134
+ ).toEqual({
135
+ ok: true,
136
+ data: {
137
+ command: 'resume',
138
+ feature_id: 'feature_a',
139
+ handoff_started: true,
140
+ handoff_target: 'runtime',
141
+ started: true,
142
+ already_running: false,
143
+ },
144
+ });
145
+ });
146
+
147
+ it('builds reconciler handoff responses from merge conflicts', () => {
148
+ expect(
149
+ buildReconcilerHandoffResponse(
150
+ 'feature_a',
151
+ {
152
+ ok: false,
153
+ error: {
154
+ code: 'reconciliation_failed',
155
+ message: 'Merge requires reconciler intervention before it can continue',
156
+ },
157
+ },
158
+ {
159
+ attempted: true,
160
+ started: true,
161
+ already_running: false,
162
+ feature_id: 'feature_a',
163
+ },
164
+ ),
165
+ ).toEqual({
166
+ ok: true,
167
+ data: {
168
+ feature_id: 'feature_a',
169
+ handoff_started: true,
170
+ handoff_target: 'reconciler',
171
+ merge: {
172
+ ok: false,
173
+ error: {
174
+ code: 'reconciliation_failed',
175
+ message: 'Merge requires reconciler intervention before it can continue',
176
+ },
177
+ },
178
+ auto_resume: {
179
+ attempted: true,
180
+ started: true,
181
+ already_running: false,
182
+ feature_id: 'feature_a',
183
+ },
184
+ },
185
+ });
186
+ });
187
+
188
+ it('builds a busy response when another background resume is active', () => {
189
+ expect(buildRuntimeBusyResponse('feature_busy')).toEqual({
190
+ ok: false,
191
+ error: {
192
+ code: 'dashboard_runtime_busy',
193
+ message: 'Dashboard runtime is busy resuming feature_busy',
194
+ details: {
195
+ active_command: 'feature.resume',
196
+ active_feature_id: 'feature_busy',
197
+ retryable: true,
198
+ requires_human: false,
199
+ },
200
+ },
201
+ });
202
+ });
203
+
204
+ it('waitForIdle returns immediately when no background task is running', async () => {
205
+ const runner = new DashboardRuntimeRunner({
206
+ repoRoot: '/repo',
207
+ instanceId: 'instance123456',
208
+ env: process.env,
209
+ });
210
+
211
+ await expect(runner.waitForIdle()).resolves.toBeUndefined();
212
+ });
213
+ });
214
+
215
+ describe('DashboardRuntimeRunner.handleCommand', () => {
216
+ let repoRoot: string;
217
+
218
+ beforeEach(async () => {
219
+ repoRoot = await fs.mkdtemp(path.join(os.tmpdir(), 'aop-dashboard-runtime-runner-'));
220
+ });
221
+
222
+ beforeEach(() => {
223
+ mergeExecuteMock.mockReset();
224
+ resumeExecuteMock.mockReset();
225
+ toolClientCallMock.mockReset();
226
+ supervisorStartMock.mockReset();
227
+ kernelRuntimeSessionsMock.mockReset();
228
+ kernelIsRunLeaseFreshMock.mockReset();
229
+ kernelFeatureStateGetMock.mockReset();
230
+ kernelRedrivePreparedConflictMock.mockReset();
231
+ kernelIsRunLeaseFreshMock.mockReturnValue(false);
232
+ kernelRedrivePreparedConflictMock.mockResolvedValue(false);
233
+ kernelRuntimeSessionsMock.mockResolvedValue({
234
+ run_id: 'none',
235
+ owner_instance_id: 'none',
236
+ });
237
+ kernelFeatureStateGetMock.mockResolvedValue({
238
+ data: {
239
+ front_matter: {},
240
+ },
241
+ });
242
+ });
243
+
244
+ afterEach(async () => {
245
+ await fs.rm(repoRoot, { recursive: true, force: true });
246
+ });
247
+
248
+ it('passes successful merge results through unchanged', async () => {
249
+ mergeExecuteMock.mockResolvedValue({
250
+ ok: true,
251
+ data: {
252
+ command: 'merge',
253
+ reconciliation_status: 'merged',
254
+ },
255
+ });
256
+ const runner = new DashboardRuntimeRunner({
257
+ repoRoot,
258
+ instanceId: 'instance123456',
259
+ env: process.env,
260
+ });
261
+
262
+ await expect(
263
+ runner.handleCommand({
264
+ command: 'review.approve',
265
+ feature_id: 'feature_a',
266
+ args: {
267
+ approval_token: 'approved',
268
+ merge_strategy: 'merge_commit',
269
+ commit_message: 'merge feature_a',
270
+ },
271
+ }),
272
+ ).resolves.toEqual({
273
+ ok: true,
274
+ data: {
275
+ command: 'merge',
276
+ reconciliation_status: 'merged',
277
+ },
278
+ });
279
+ });
280
+
281
+ it('returns an invalid-response envelope when a merge command returns a malformed payload', async () => {
282
+ mergeExecuteMock.mockResolvedValue('malformed');
283
+ const runner = new DashboardRuntimeRunner({
284
+ repoRoot,
285
+ instanceId: 'instance123456',
286
+ env: process.env,
287
+ });
288
+
289
+ await expect(
290
+ runner.handleCommand({
291
+ command: 'review.approve',
292
+ feature_id: 'feature_a',
293
+ args: {
294
+ approval_token: 'approved',
295
+ },
296
+ }),
297
+ ).resolves.toEqual({
298
+ ok: false,
299
+ error: {
300
+ code: 'dashboard_runtime_invalid_response',
301
+ message: 'Dashboard runtime command returned an invalid response envelope',
302
+ },
303
+ });
304
+ });
305
+
306
+ it('returns a reconciler handoff when the merge conflict is already dispatched', async () => {
307
+ mergeExecuteMock.mockResolvedValue({
308
+ ok: false,
309
+ error: {
310
+ code: 'reconciliation_failed',
311
+ message: 'Merge requires reconciler intervention before it can continue',
312
+ details: {
313
+ conflict_worktree_prepared: true,
314
+ dispatched_to_reconciler: true,
315
+ },
316
+ },
317
+ });
318
+ const runner = new DashboardRuntimeRunner({
319
+ repoRoot,
320
+ instanceId: 'instance123456',
321
+ env: process.env,
322
+ });
323
+
324
+ const result = await runner.handleCommand({
325
+ command: 'review.approve',
326
+ feature_id: 'feature_a',
327
+ args: {
328
+ approval_token: 'approved',
329
+ merge_strategy: 'merge_commit',
330
+ },
331
+ });
332
+
333
+ expect(result.ok).toBe(true);
334
+ expect(result.data?.handoff_started).toBe(true);
335
+ expect(result.data?.handoff_target).toBe('reconciler');
336
+ expect(result.data?.auto_resume).toEqual({
337
+ attempted: false,
338
+ started: false,
339
+ already_running: false,
340
+ feature_id: 'feature_a',
341
+ });
342
+ });
343
+
344
+ it('redrives the prepared reconciler assignment immediately when an active run is already available', async () => {
345
+ kernelRuntimeSessionsMock.mockResolvedValue({
346
+ run_id: 'run:active',
347
+ owner_instance_id: 'owner:active',
348
+ });
349
+ kernelIsRunLeaseFreshMock.mockReturnValue(true);
350
+ kernelRedrivePreparedConflictMock.mockResolvedValue(true);
351
+ mergeExecuteMock.mockResolvedValue({
352
+ ok: false,
353
+ error: {
354
+ code: 'reconciliation_failed',
355
+ message: 'Merge requires reconciler intervention before it can continue',
356
+ details: {
357
+ conflict_worktree_prepared: true,
358
+ dispatched_to_reconciler: false,
359
+ },
360
+ },
361
+ });
362
+
363
+ const runner = new DashboardRuntimeRunner({
364
+ repoRoot,
365
+ instanceId: 'instance123456',
366
+ env: process.env,
367
+ });
368
+
369
+ const result = await runner.handleCommand({
370
+ command: 'review.approve',
371
+ feature_id: 'feature_a',
372
+ args: {
373
+ approval_token: 'approved',
374
+ },
375
+ });
376
+
377
+ expect(result.ok).toBe(true);
378
+ expect(result.data?.handoff_target).toBe('reconciler');
379
+ expect(result.data?.auto_resume).toEqual({
380
+ attempted: false,
381
+ started: false,
382
+ already_running: false,
383
+ feature_id: 'feature_a',
384
+ });
385
+ expect(result.data?.merge).toEqual({
386
+ ok: false,
387
+ error: {
388
+ code: 'reconciliation_failed',
389
+ message: 'Merge requires reconciler intervention before it can continue',
390
+ details: {
391
+ conflict_worktree_prepared: true,
392
+ dispatched_to_reconciler: true,
393
+ },
394
+ },
395
+ });
396
+ expect(kernelRedrivePreparedConflictMock).toHaveBeenCalledWith('feature_a');
397
+ expect(resumeExecuteMock).not.toHaveBeenCalled();
398
+ });
399
+
400
+ it('treats the dashboard-owned run as active when the lease expiry is stale but the heartbeat is recent', async () => {
401
+ kernelRuntimeSessionsMock.mockResolvedValue({
402
+ run_id: 'run:active',
403
+ owner_instance_id: 'instance123456',
404
+ last_heartbeat_at: new Date(Date.now() - 30_000).toISOString(),
405
+ });
406
+ kernelIsRunLeaseFreshMock.mockReturnValue(false);
407
+ kernelRedrivePreparedConflictMock.mockResolvedValue(true);
408
+ mergeExecuteMock.mockResolvedValue({
409
+ ok: false,
410
+ error: {
411
+ code: 'reconciliation_failed',
412
+ message: 'Merge requires reconciler intervention before it can continue',
413
+ details: {
414
+ conflict_worktree_prepared: true,
415
+ dispatched_to_reconciler: false,
416
+ },
417
+ },
418
+ });
419
+
420
+ const runner = new DashboardRuntimeRunner({
421
+ repoRoot,
422
+ instanceId: 'instance123456',
423
+ env: process.env,
424
+ });
425
+
426
+ const result = await runner.handleCommand({
427
+ command: 'review.approve',
428
+ feature_id: 'feature_a',
429
+ args: {
430
+ approval_token: 'approved',
431
+ },
432
+ });
433
+
434
+ expect(result.ok).toBe(true);
435
+ expect(result.data?.handoff_target).toBe('reconciler');
436
+ expect(result.data?.auto_resume).toEqual({
437
+ attempted: false,
438
+ started: false,
439
+ already_running: false,
440
+ feature_id: 'feature_a',
441
+ });
442
+ expect(kernelRedrivePreparedConflictMock).toHaveBeenCalledWith('feature_a');
443
+ expect(resumeExecuteMock).not.toHaveBeenCalled();
444
+ });
445
+
446
+ it('does not treat the dashboard-owned run as active when its heartbeat is stale', async () => {
447
+ kernelRuntimeSessionsMock.mockResolvedValue({
448
+ run_id: 'run:active',
449
+ owner_instance_id: 'instance123456',
450
+ last_heartbeat_at: new Date(Date.now() - 180_000).toISOString(),
451
+ });
452
+ kernelIsRunLeaseFreshMock.mockReturnValue(false);
453
+ mergeExecuteMock.mockResolvedValue({
454
+ ok: false,
455
+ error: {
456
+ code: 'reconciliation_failed',
457
+ message: 'Merge requires reconciler intervention before it can continue',
458
+ details: {
459
+ conflict_worktree_prepared: true,
460
+ dispatched_to_reconciler: false,
461
+ },
462
+ },
463
+ });
464
+ resumeExecuteMock.mockResolvedValue({
465
+ ok: true,
466
+ data: {
467
+ command: 'resume',
468
+ },
469
+ });
470
+
471
+ const runner = new DashboardRuntimeRunner({
472
+ repoRoot,
473
+ instanceId: 'instance123456',
474
+ env: process.env,
475
+ });
476
+
477
+ const result = await runner.handleCommand({
478
+ command: 'review.approve',
479
+ feature_id: 'feature_a',
480
+ args: {
481
+ approval_token: 'approved',
482
+ },
483
+ });
484
+
485
+ expect(result.ok).toBe(true);
486
+ expect(result.data?.auto_resume).toEqual({
487
+ attempted: true,
488
+ started: true,
489
+ already_running: false,
490
+ feature_id: 'feature_a',
491
+ });
492
+ // Resume now always does best-effort redrive regardless of conflict state
493
+ await runner.waitForIdle();
494
+ });
495
+
496
+ it('starts a background resume when the merge conflict needs reconciler startup', async () => {
497
+ mergeExecuteMock.mockResolvedValue({
498
+ ok: false,
499
+ error: {
500
+ code: 'reconciliation_failed',
501
+ message: 'Merge requires reconciler intervention before it can continue',
502
+ details: {
503
+ conflict_worktree_prepared: true,
504
+ dispatched_to_reconciler: false,
505
+ },
506
+ },
507
+ });
508
+ resumeExecuteMock.mockResolvedValue({
509
+ ok: true,
510
+ data: {
511
+ command: 'resume',
512
+ },
513
+ });
514
+ const runner = new DashboardRuntimeRunner({
515
+ repoRoot,
516
+ instanceId: 'instance123456',
517
+ env: process.env,
518
+ });
519
+
520
+ const result = await runner.handleCommand({
521
+ command: 'review.approve',
522
+ feature_id: 'feature_a',
523
+ args: {
524
+ approval_token: 'approved',
525
+ },
526
+ });
527
+
528
+ expect(result.ok).toBe(true);
529
+ expect(result.data?.handoff_started).toBe(true);
530
+ expect(result.data?.auto_resume).toEqual({
531
+ attempted: true,
532
+ started: true,
533
+ already_running: false,
534
+ feature_id: 'feature_a',
535
+ });
536
+ await runner.waitForIdle();
537
+ expect(kernelRedrivePreparedConflictMock).toHaveBeenCalledWith('feature_a');
538
+ });
539
+
540
+ it('returns a reconciliation failure when an active run cannot redrive a prepared conflict', async () => {
541
+ kernelRuntimeSessionsMock.mockResolvedValue({
542
+ run_id: 'run:active',
543
+ owner_instance_id: 'owner:active',
544
+ });
545
+ kernelIsRunLeaseFreshMock.mockReturnValue(true);
546
+ kernelRedrivePreparedConflictMock.mockResolvedValue(false);
547
+ mergeExecuteMock.mockResolvedValue({
548
+ ok: false,
549
+ error: {
550
+ code: 'reconciliation_failed',
551
+ message: 'Merge requires reconciler intervention before it can continue',
552
+ details: {
553
+ conflict_worktree_prepared: true,
554
+ dispatched_to_reconciler: false,
555
+ },
556
+ },
557
+ });
558
+
559
+ const runner = new DashboardRuntimeRunner({
560
+ repoRoot,
561
+ instanceId: 'instance123456',
562
+ env: process.env,
563
+ });
564
+
565
+ await expect(
566
+ runner.handleCommand({
567
+ command: 'review.approve',
568
+ feature_id: 'feature_a',
569
+ args: {
570
+ approval_token: 'approved',
571
+ },
572
+ }),
573
+ ).resolves.toEqual({
574
+ ok: false,
575
+ error: {
576
+ code: 'reconciliation_failed',
577
+ message: 'Prepared reconciler handoff could not be dispatched to the active run',
578
+ details: {
579
+ conflict_worktree_prepared: true,
580
+ dispatched_to_reconciler: false,
581
+ feature_id: 'feature_a',
582
+ retryable: true,
583
+ requires_human: true,
584
+ },
585
+ },
586
+ });
587
+ expect(resumeExecuteMock).not.toHaveBeenCalled();
588
+ });
589
+
590
+ it('returns busy when another feature is already resuming in the background', async () => {
591
+ let resolveResume: (() => void) | null = null;
592
+ resumeExecuteMock.mockImplementation(
593
+ () =>
594
+ new Promise((resolve) => {
595
+ resolveResume = () => resolve({ ok: true, data: { command: 'resume' } });
596
+ }),
597
+ );
598
+ mergeExecuteMock.mockResolvedValue({
599
+ ok: false,
600
+ error: {
601
+ code: 'reconciliation_failed',
602
+ message: 'Merge requires reconciler intervention before it can continue',
603
+ details: {
604
+ conflict_worktree_prepared: true,
605
+ dispatched_to_reconciler: false,
606
+ },
607
+ },
608
+ });
609
+ const runner = new DashboardRuntimeRunner({
610
+ repoRoot,
611
+ instanceId: 'instance123456',
612
+ env: process.env,
613
+ });
614
+
615
+ const resumeResult = await runner.handleCommand({
616
+ command: 'feature.resume',
617
+ feature_id: 'feature_a',
618
+ args: {},
619
+ });
620
+ expect(resumeResult.ok).toBe(true);
621
+
622
+ const mergeResult = await runner.handleCommand({
623
+ command: 'review.approve',
624
+ feature_id: 'feature_b',
625
+ args: {
626
+ approval_token: 'approved',
627
+ },
628
+ });
629
+
630
+ expect(mergeResult).toEqual(buildRuntimeBusyResponse('feature_a'));
631
+ resolveResume?.();
632
+ await runner.waitForIdle();
633
+ });
634
+
635
+ it('executes feature.resume synchronously when a fresh active run already exists', async () => {
636
+ kernelRuntimeSessionsMock.mockResolvedValue({
637
+ run_id: 'run:active',
638
+ owner_instance_id: 'owner:active',
639
+ });
640
+ kernelIsRunLeaseFreshMock.mockReturnValue(true);
641
+ resumeExecuteMock.mockResolvedValue({
642
+ ok: true,
643
+ data: {
644
+ command: 'resume',
645
+ resumed: true,
646
+ },
647
+ });
648
+
649
+ const runner = new DashboardRuntimeRunner({
650
+ repoRoot,
651
+ instanceId: 'instance123456',
652
+ env: process.env,
653
+ });
654
+
655
+ await expect(
656
+ runner.handleCommand({
657
+ command: 'feature.resume',
658
+ feature_id: 'feature_active',
659
+ args: {},
660
+ }),
661
+ ).resolves.toEqual({
662
+ ok: true,
663
+ data: {
664
+ command: 'resume',
665
+ resumed: true,
666
+ },
667
+ });
668
+ expect(kernelRedrivePreparedConflictMock).toHaveBeenCalledWith('feature_active');
669
+ });
670
+
671
+ it('resumes successfully with best-effort redrive when feature has a pending mainline conflict', async () => {
672
+ kernelRuntimeSessionsMock.mockResolvedValue({
673
+ run_id: 'run:active',
674
+ owner_instance_id: 'owner:active',
675
+ });
676
+ kernelIsRunLeaseFreshMock.mockReturnValue(true);
677
+ kernelRedrivePreparedConflictMock.mockResolvedValue(false);
678
+ resumeExecuteMock.mockResolvedValue({
679
+ ok: true,
680
+ data: {
681
+ command: 'resume',
682
+ resumed: true,
683
+ },
684
+ });
685
+
686
+ const runner = new DashboardRuntimeRunner({
687
+ repoRoot,
688
+ instanceId: 'instance123456',
689
+ env: process.env,
690
+ });
691
+
692
+ await expect(
693
+ runner.handleCommand({
694
+ command: 'feature.resume',
695
+ feature_id: 'feature_active',
696
+ args: {},
697
+ }),
698
+ ).resolves.toEqual({
699
+ ok: true,
700
+ data: {
701
+ command: 'resume',
702
+ resumed: true,
703
+ },
704
+ });
705
+ expect(kernelRedrivePreparedConflictMock).toHaveBeenCalledWith('feature_active');
706
+ });
707
+
708
+ it('resumes successfully and redrives when reconciler conflict can be dispatched', async () => {
709
+ kernelRuntimeSessionsMock.mockResolvedValue({
710
+ run_id: 'run:active',
711
+ owner_instance_id: 'owner:active',
712
+ });
713
+ kernelIsRunLeaseFreshMock.mockReturnValue(true);
714
+ kernelRedrivePreparedConflictMock.mockResolvedValue(true);
715
+ resumeExecuteMock.mockResolvedValue({
716
+ ok: true,
717
+ data: {
718
+ command: 'resume',
719
+ resumed: true,
720
+ },
721
+ });
722
+
723
+ const runner = new DashboardRuntimeRunner({
724
+ repoRoot,
725
+ instanceId: 'instance123456',
726
+ env: process.env,
727
+ });
728
+
729
+ await expect(
730
+ runner.handleCommand({
731
+ command: 'feature.resume',
732
+ feature_id: 'feature_active',
733
+ args: {},
734
+ }),
735
+ ).resolves.toEqual({
736
+ ok: true,
737
+ data: {
738
+ command: 'resume',
739
+ resumed: true,
740
+ },
741
+ });
742
+ expect(kernelRedrivePreparedConflictMock).toHaveBeenCalledWith('feature_active');
743
+ });
744
+
745
+ it('returns invalid_input when launch.submit does not include prepared features', async () => {
746
+ const runner = new DashboardRuntimeRunner({
747
+ repoRoot,
748
+ instanceId: 'instance123456',
749
+ env: process.env,
750
+ });
751
+
752
+ await expect(
753
+ runner.handleCommand({
754
+ command: 'launch.submit',
755
+ feature_id: 'feature_a',
756
+ args: {},
757
+ }),
758
+ ).resolves.toEqual({
759
+ ok: false,
760
+ error: {
761
+ code: 'invalid_input',
762
+ message: 'launch.submit requires at least one prepared feature.',
763
+ },
764
+ });
765
+ });
766
+
767
+ it('rejects malformed launch feature entries after normalizing empty values', async () => {
768
+ const runner = new DashboardRuntimeRunner({
769
+ repoRoot,
770
+ instanceId: 'instance123456',
771
+ env: process.env,
772
+ });
773
+
774
+ await expect(
775
+ runner.handleCommand({
776
+ command: 'launch.submit',
777
+ feature_id: 'feature_a',
778
+ args: {
779
+ features: [null, { feature_id: '', spec_path: ' ' }],
780
+ },
781
+ }),
782
+ ).resolves.toEqual({
783
+ ok: false,
784
+ error: {
785
+ code: 'invalid_input',
786
+ message: 'launch.submit requires at least one prepared feature.',
787
+ },
788
+ });
789
+ });
790
+
791
+ it('submits launch features into the active run when a fresh lease exists', async () => {
792
+ kernelRuntimeSessionsMock.mockResolvedValue({
793
+ run_id: 'run:active',
794
+ owner_instance_id: 'owner:active',
795
+ });
796
+ kernelIsRunLeaseFreshMock.mockReturnValue(true);
797
+ toolClientCallMock.mockResolvedValue({
798
+ ok: true,
799
+ data: {
800
+ run_id: 'run:active',
801
+ request_id: 'exec_req_123',
802
+ request_status: 'pending',
803
+ },
804
+ });
805
+
806
+ const runner = new DashboardRuntimeRunner({
807
+ repoRoot,
808
+ instanceId: 'instance123456',
809
+ env: process.env,
810
+ });
811
+
812
+ await expect(
813
+ runner.handleCommand({
814
+ command: 'launch.submit',
815
+ feature_id: 'feature_a',
816
+ args: {
817
+ features: [
818
+ {
819
+ feature_id: 'feature_b',
820
+ spec_path: '.aop/features/feature_b/spec.md',
821
+ },
822
+ {
823
+ feature_id: 'feature_a',
824
+ spec_path: '.aop/features/feature_a/spec.md',
825
+ },
826
+ ],
827
+ },
828
+ }),
829
+ ).resolves.toEqual({
830
+ ok: true,
831
+ data: {
832
+ command: 'launch.submit',
833
+ mode: 'active_run_enrollment',
834
+ run_id: 'run:active',
835
+ request_id: 'exec_req_123',
836
+ request_status: 'pending',
837
+ },
838
+ });
839
+ });
840
+
841
+ it('returns a dashboard error when active-run launch enrollment fails', async () => {
842
+ kernelRuntimeSessionsMock.mockResolvedValue({
843
+ run_id: 'run:active',
844
+ owner_instance_id: 'owner:active',
845
+ });
846
+ kernelIsRunLeaseFreshMock.mockReturnValue(true);
847
+ toolClientCallMock.mockRejectedValue(new Error('enrollment failed'));
848
+
849
+ const runner = new DashboardRuntimeRunner({
850
+ repoRoot,
851
+ instanceId: 'instance123456',
852
+ env: process.env,
853
+ });
854
+
855
+ await expect(
856
+ runner.handleCommand({
857
+ command: 'launch.submit',
858
+ feature_id: 'feature_a',
859
+ args: {
860
+ features: [
861
+ {
862
+ feature_id: 'feature_a',
863
+ spec_path: '.aop/features/feature_a/spec.md',
864
+ },
865
+ ],
866
+ },
867
+ }),
868
+ ).resolves.toEqual({
869
+ ok: false,
870
+ error: {
871
+ code: 'dashboard_runtime_failed',
872
+ message: 'enrollment failed',
873
+ details: undefined,
874
+ },
875
+ });
876
+ });
877
+
878
+ it('starts a background run for launch features when there is no active lease', async () => {
879
+ let resolveStart: (() => void) | null = null;
880
+ supervisorStartMock.mockImplementation(
881
+ () =>
882
+ new Promise((resolve) => {
883
+ resolveStart = () =>
884
+ resolve({
885
+ status: 'running',
886
+ queue_depth: 1,
887
+ });
888
+ }),
889
+ );
890
+
891
+ const runner = new DashboardRuntimeRunner({
892
+ repoRoot,
893
+ instanceId: 'instance123456',
894
+ env: process.env,
895
+ });
896
+
897
+ const result = await runner.handleCommand({
898
+ command: 'launch.submit',
899
+ feature_id: 'feature_a',
900
+ args: {
901
+ features: [
902
+ {
903
+ feature_id: 'feature_a',
904
+ spec_path: '.aop/features/feature_a/spec.md',
905
+ },
906
+ {
907
+ feature_id: 'feature_b',
908
+ spec_path: '.aop/features/feature_b/spec.md',
909
+ },
910
+ ],
911
+ },
912
+ });
913
+
914
+ expect(result).toEqual({
915
+ ok: true,
916
+ data: {
917
+ command: 'launch.submit',
918
+ mode: 'runtime_start',
919
+ run_id: expect.stringMatching(/^run:dashboard-launch:/),
920
+ feature_id: 'feature_a',
921
+ feature_ids: ['feature_a', 'feature_b'],
922
+ handoff_started: true,
923
+ handoff_target: 'runtime',
924
+ started: true,
925
+ already_running: false,
926
+ request_status: 'started',
927
+ requested_features: [
928
+ {
929
+ feature_id: 'feature_a',
930
+ spec_path: '.aop/features/feature_a/spec.md',
931
+ },
932
+ {
933
+ feature_id: 'feature_b',
934
+ spec_path: '.aop/features/feature_b/spec.md',
935
+ },
936
+ ],
937
+ },
938
+ });
939
+
940
+ resolveStart?.();
941
+ await runner.waitForIdle();
942
+ });
943
+
944
+ it('returns already_running for duplicate launch batches and busy for other launch work', async () => {
945
+ let resolveStart: (() => void) | null = null;
946
+ supervisorStartMock.mockImplementation(
947
+ () =>
948
+ new Promise((resolve) => {
949
+ resolveStart = () => resolve({ status: 'running', queue_depth: 1 });
950
+ }),
951
+ );
952
+
953
+ const runner = new DashboardRuntimeRunner({
954
+ repoRoot,
955
+ instanceId: 'instance123456',
956
+ env: process.env,
957
+ });
958
+
959
+ const first = await runner.handleCommand({
960
+ command: 'launch.submit',
961
+ feature_id: 'feature_a',
962
+ args: {
963
+ features: [{ feature_id: 'feature_a', spec_path: '.aop/features/feature_a/spec.md' }],
964
+ },
965
+ });
966
+ const duplicate = await runner.handleCommand({
967
+ command: 'launch.submit',
968
+ feature_id: 'feature_a',
969
+ args: {
970
+ features: [{ feature_id: 'feature_a', spec_path: '.aop/features/feature_a/spec.md' }],
971
+ },
972
+ });
973
+ const busy = await runner.handleCommand({
974
+ command: 'launch.submit',
975
+ feature_id: 'feature_b',
976
+ args: {
977
+ features: [{ feature_id: 'feature_b', spec_path: '.aop/features/feature_b/spec.md' }],
978
+ },
979
+ });
980
+
981
+ expect(first.ok).toBe(true);
982
+ expect(duplicate).toEqual({
983
+ ok: true,
984
+ data: {
985
+ command: 'launch.submit',
986
+ mode: 'runtime_start',
987
+ run_id: expect.stringMatching(/^run:dashboard-launch:/),
988
+ feature_id: 'feature_a',
989
+ feature_ids: ['feature_a'],
990
+ handoff_started: true,
991
+ handoff_target: 'runtime',
992
+ started: false,
993
+ already_running: true,
994
+ request_status: 'started',
995
+ requested_features: [
996
+ {
997
+ feature_id: 'feature_a',
998
+ spec_path: '.aop/features/feature_a/spec.md',
999
+ },
1000
+ ],
1001
+ },
1002
+ });
1003
+ expect(busy).toEqual({
1004
+ ok: false,
1005
+ error: {
1006
+ code: 'dashboard_runtime_busy',
1007
+ message: 'Dashboard runtime is busy starting execution for feature_a',
1008
+ details: {
1009
+ active_command: 'launch.submit',
1010
+ active_feature_id: 'feature_a',
1011
+ retryable: true,
1012
+ requires_human: false,
1013
+ },
1014
+ },
1015
+ });
1016
+
1017
+ resolveStart?.();
1018
+ await runner.waitForIdle();
1019
+ });
1020
+
1021
+ it('records background resume failures in the runtime state file', async () => {
1022
+ resumeExecuteMock.mockResolvedValue({
1023
+ ok: false,
1024
+ error: {
1025
+ message: 'resume failed',
1026
+ },
1027
+ });
1028
+
1029
+ const runner = new DashboardRuntimeRunner({
1030
+ repoRoot,
1031
+ instanceId: 'instance123456',
1032
+ env: process.env,
1033
+ });
1034
+
1035
+ const response = await runner.handleCommand({
1036
+ command: 'feature.resume',
1037
+ feature_id: 'feature_failed',
1038
+ args: {},
1039
+ });
1040
+
1041
+ expect(response.ok).toBe(true);
1042
+ await runner.waitForIdle();
1043
+
1044
+ const statePath = path.join(
1045
+ repoRoot,
1046
+ '.aop',
1047
+ 'runtime',
1048
+ 'instance123456',
1049
+ 'dashboard-runtime-state.json',
1050
+ );
1051
+ const state = JSON.parse(await fs.readFile(statePath, 'utf8')) as {
1052
+ last_background_error?: { message?: string; feature_id?: string; command?: string } | null;
1053
+ };
1054
+
1055
+ expect(state.last_background_error).toEqual({
1056
+ command: 'feature.resume',
1057
+ feature_id: 'feature_failed',
1058
+ code: 'dashboard_runtime_failed',
1059
+ message: 'resume failed',
1060
+ at: expect.any(String),
1061
+ });
1062
+ });
1063
+
1064
+ it('uses the fallback background failure message when resume errors do not provide one', async () => {
1065
+ resumeExecuteMock.mockResolvedValue({
1066
+ ok: false,
1067
+ error: {},
1068
+ });
1069
+
1070
+ const runner = new DashboardRuntimeRunner({
1071
+ repoRoot,
1072
+ instanceId: 'instance123456',
1073
+ env: process.env,
1074
+ });
1075
+
1076
+ await runner.handleCommand({
1077
+ command: 'feature.resume',
1078
+ feature_id: 'feature_failed',
1079
+ args: {},
1080
+ });
1081
+ await runner.waitForIdle();
1082
+
1083
+ const statePath = path.join(
1084
+ repoRoot,
1085
+ '.aop',
1086
+ 'runtime',
1087
+ 'instance123456',
1088
+ 'dashboard-runtime-state.json',
1089
+ );
1090
+ const state = JSON.parse(await fs.readFile(statePath, 'utf8')) as {
1091
+ last_background_error?: { message?: string } | null;
1092
+ };
1093
+
1094
+ expect(state.last_background_error?.message).toBe('Dashboard background resume failed');
1095
+ });
1096
+
1097
+ it('returns a busy error from feature.resume when launch startup is already occupying the runtime', async () => {
1098
+ let resolveStart: (() => void) | null = null;
1099
+ supervisorStartMock.mockImplementation(
1100
+ () =>
1101
+ new Promise((resolve) => {
1102
+ resolveStart = () => resolve({ status: 'running', queue_depth: 1 });
1103
+ }),
1104
+ );
1105
+
1106
+ const runner = new DashboardRuntimeRunner({
1107
+ repoRoot,
1108
+ instanceId: 'instance123456',
1109
+ env: process.env,
1110
+ });
1111
+
1112
+ await runner.handleCommand({
1113
+ command: 'launch.submit',
1114
+ feature_id: 'feature_a',
1115
+ args: {
1116
+ features: [{ feature_id: 'feature_a', spec_path: '.aop/features/feature_a/spec.md' }],
1117
+ },
1118
+ });
1119
+
1120
+ await expect(
1121
+ runner.handleCommand({
1122
+ command: 'feature.resume',
1123
+ feature_id: 'feature_b',
1124
+ args: {},
1125
+ }),
1126
+ ).resolves.toEqual({
1127
+ ok: false,
1128
+ error: {
1129
+ code: 'dashboard_runtime_busy',
1130
+ message: 'Dashboard runtime is busy starting execution for feature_a',
1131
+ details: {
1132
+ active_command: 'launch.submit',
1133
+ active_feature_id: 'feature_a',
1134
+ retryable: true,
1135
+ requires_human: false,
1136
+ },
1137
+ },
1138
+ });
1139
+
1140
+ resolveStart?.();
1141
+ await runner.waitForIdle();
1142
+ });
1143
+
1144
+ it('processes queued commands and persists command completion state', async () => {
1145
+ const runner = new DashboardRuntimeRunner({
1146
+ repoRoot,
1147
+ instanceId: 'instance123456',
1148
+ env: process.env,
1149
+ });
1150
+ const runnerAny = runner as unknown as {
1151
+ handleCommand: (item: unknown) => Promise<unknown>;
1152
+ stopping: boolean;
1153
+ };
1154
+ vi.spyOn(runnerAny, 'handleCommand').mockImplementation(async () => {
1155
+ runnerAny.stopping = true;
1156
+ return {
1157
+ ok: true,
1158
+ data: {
1159
+ handled: true,
1160
+ },
1161
+ };
1162
+ });
1163
+
1164
+ const runtimeRoot = path.join(repoRoot, '.aop', 'runtime', 'instance123456');
1165
+ await fs.mkdir(runtimeRoot, { recursive: true });
1166
+ await fs.writeFile(
1167
+ path.join(runtimeRoot, 'dashboard-runtime-commands.json'),
1168
+ JSON.stringify({
1169
+ version: 1,
1170
+ instance_id: 'instance123456',
1171
+ updated_at: new Date().toISOString(),
1172
+ items: [
1173
+ {
1174
+ request_id: 'dash_cmd_1',
1175
+ command: 'feature.resume',
1176
+ feature_id: 'feature_a',
1177
+ args: {},
1178
+ status: 'pending',
1179
+ requested_at: new Date().toISOString(),
1180
+ processed_at: null,
1181
+ result: null,
1182
+ },
1183
+ ],
1184
+ }),
1185
+ 'utf8',
1186
+ );
1187
+
1188
+ await runner.start();
1189
+
1190
+ const queue = JSON.parse(
1191
+ await fs.readFile(path.join(runtimeRoot, 'dashboard-runtime-commands.json'), 'utf8'),
1192
+ ) as {
1193
+ items: Array<{ status: string; result: { ok: boolean; data: { handled: boolean } } }>;
1194
+ };
1195
+ const state = JSON.parse(
1196
+ await fs.readFile(path.join(runtimeRoot, 'dashboard-runtime-state.json'), 'utf8'),
1197
+ ) as { status: string };
1198
+
1199
+ expect(queue.items[0]).toMatchObject({
1200
+ status: 'completed',
1201
+ result: {
1202
+ ok: true,
1203
+ data: {
1204
+ handled: true,
1205
+ },
1206
+ },
1207
+ });
1208
+ expect(state.status).toBe('stopped');
1209
+ });
1210
+
1211
+ it('returns without work when processNextCommand sees an empty queue', async () => {
1212
+ const runner = new DashboardRuntimeRunner({
1213
+ repoRoot,
1214
+ instanceId: 'instance123456',
1215
+ env: process.env,
1216
+ });
1217
+
1218
+ await expect(
1219
+ (
1220
+ runner as unknown as {
1221
+ claimNextCommand: () => Promise<null>;
1222
+ processNextCommand: () => Promise<void>;
1223
+ }
1224
+ ).claimNextCommand(),
1225
+ ).resolves.toBeNull();
1226
+ await expect(
1227
+ (
1228
+ runner as unknown as {
1229
+ processNextCommand: () => Promise<void>;
1230
+ }
1231
+ ).processNextCommand(),
1232
+ ).resolves.toBeUndefined();
1233
+ });
1234
+
1235
+ it('creates an empty command snapshot when none exists on disk', async () => {
1236
+ const runner = new DashboardRuntimeRunner({
1237
+ repoRoot,
1238
+ instanceId: 'instance123456',
1239
+ env: process.env,
1240
+ });
1241
+
1242
+ await (
1243
+ runner as unknown as {
1244
+ ensureCommandSnapshot: () => Promise<void>;
1245
+ }
1246
+ ).ensureCommandSnapshot();
1247
+
1248
+ const queue = JSON.parse(
1249
+ await fs.readFile(
1250
+ path.join(repoRoot, '.aop', 'runtime', 'instance123456', 'dashboard-runtime-commands.json'),
1251
+ 'utf8',
1252
+ ),
1253
+ ) as { items: unknown[]; instance_id: string };
1254
+
1255
+ expect(queue).toMatchObject({
1256
+ instance_id: 'instance123456',
1257
+ items: [],
1258
+ });
1259
+ });
1260
+
1261
+ it('marks queued commands as failed when command execution returns an error envelope', async () => {
1262
+ const runner = new DashboardRuntimeRunner({
1263
+ repoRoot,
1264
+ instanceId: 'instance123456',
1265
+ env: process.env,
1266
+ });
1267
+ const runnerAny = runner as unknown as {
1268
+ handleCommand: (item: unknown) => Promise<unknown>;
1269
+ stopping: boolean;
1270
+ };
1271
+ vi.spyOn(runnerAny, 'handleCommand').mockImplementation(async () => {
1272
+ runnerAny.stopping = true;
1273
+ return {
1274
+ ok: false,
1275
+ error: {
1276
+ code: 'dashboard_runtime_failed',
1277
+ message: 'failed to execute queued command',
1278
+ },
1279
+ };
1280
+ });
1281
+
1282
+ const runtimeRoot = path.join(repoRoot, '.aop', 'runtime', 'instance123456');
1283
+ await fs.mkdir(runtimeRoot, { recursive: true });
1284
+ await fs.writeFile(
1285
+ path.join(runtimeRoot, 'dashboard-runtime-commands.json'),
1286
+ JSON.stringify({
1287
+ version: 1,
1288
+ instance_id: 'instance123456',
1289
+ updated_at: new Date().toISOString(),
1290
+ items: [
1291
+ {
1292
+ request_id: 'dash_cmd_fail',
1293
+ command: 'launch.submit',
1294
+ feature_id: 'feature_a',
1295
+ args: {
1296
+ features: [{ feature_id: 'feature_a', spec_path: '.aop/features/feature_a/spec.md' }],
1297
+ },
1298
+ status: 'pending',
1299
+ requested_at: new Date().toISOString(),
1300
+ processed_at: null,
1301
+ result: null,
1302
+ },
1303
+ ],
1304
+ }),
1305
+ 'utf8',
1306
+ );
1307
+
1308
+ await runner.start();
1309
+
1310
+ const queue = JSON.parse(
1311
+ await fs.readFile(path.join(runtimeRoot, 'dashboard-runtime-commands.json'), 'utf8'),
1312
+ ) as {
1313
+ items: Array<{ status: string; result: { ok: boolean; error: { code: string } } }>;
1314
+ };
1315
+
1316
+ expect(queue.items[0]).toMatchObject({
1317
+ status: 'failed',
1318
+ result: {
1319
+ ok: false,
1320
+ error: {
1321
+ code: 'dashboard_runtime_failed',
1322
+ },
1323
+ },
1324
+ });
1325
+ });
1326
+
1327
+ it('preserves unrelated queued commands when completing a single request', async () => {
1328
+ const runner = new DashboardRuntimeRunner({
1329
+ repoRoot,
1330
+ instanceId: 'instance123456',
1331
+ env: process.env,
1332
+ });
1333
+ const runtimeRoot = path.join(repoRoot, '.aop', 'runtime', 'instance123456');
1334
+ await fs.mkdir(runtimeRoot, { recursive: true });
1335
+ await fs.writeFile(
1336
+ path.join(runtimeRoot, 'dashboard-runtime-commands.json'),
1337
+ JSON.stringify({
1338
+ version: 1,
1339
+ instance_id: 'instance123456',
1340
+ updated_at: new Date().toISOString(),
1341
+ items: [
1342
+ {
1343
+ request_id: 'dash_cmd_keep',
1344
+ command: 'feature.resume',
1345
+ feature_id: 'feature_keep',
1346
+ args: {},
1347
+ status: 'pending',
1348
+ requested_at: new Date().toISOString(),
1349
+ processed_at: null,
1350
+ result: null,
1351
+ },
1352
+ {
1353
+ request_id: 'dash_cmd_complete',
1354
+ command: 'feature.resume',
1355
+ feature_id: 'feature_done',
1356
+ args: {},
1357
+ status: 'processing',
1358
+ requested_at: new Date().toISOString(),
1359
+ processed_at: null,
1360
+ result: null,
1361
+ },
1362
+ ],
1363
+ }),
1364
+ 'utf8',
1365
+ );
1366
+
1367
+ await (
1368
+ runner as unknown as {
1369
+ completeCommand: (requestId: string, result: unknown) => Promise<void>;
1370
+ }
1371
+ ).completeCommand('dash_cmd_complete', {
1372
+ ok: true,
1373
+ data: { handled: true },
1374
+ });
1375
+
1376
+ const queue = JSON.parse(
1377
+ await fs.readFile(path.join(runtimeRoot, 'dashboard-runtime-commands.json'), 'utf8'),
1378
+ ) as {
1379
+ items: Array<{ request_id: string; status: string }>;
1380
+ };
1381
+
1382
+ expect(queue.items).toEqual(
1383
+ expect.arrayContaining([
1384
+ expect.objectContaining({
1385
+ request_id: 'dash_cmd_keep',
1386
+ status: 'pending',
1387
+ }),
1388
+ expect.objectContaining({
1389
+ request_id: 'dash_cmd_complete',
1390
+ status: 'completed',
1391
+ }),
1392
+ ]),
1393
+ );
1394
+ });
1395
+
1396
+ it('uses the provider fallback when provider selection throws during tool client initialization', async () => {
1397
+ const { resolveProviderSelection } = await import('../src/providers/providers.js');
1398
+ vi.mocked(resolveProviderSelection).mockImplementationOnce(() => {
1399
+ throw new Error('selection failed');
1400
+ });
1401
+
1402
+ const runner = new DashboardRuntimeRunner({
1403
+ repoRoot,
1404
+ instanceId: 'instance123456',
1405
+ env: process.env,
1406
+ });
1407
+
1408
+ await expect(
1409
+ (
1410
+ runner as unknown as {
1411
+ initializeToolClient: () => Promise<{ call: typeof toolClientCallMock }>;
1412
+ }
1413
+ ).initializeToolClient(),
1414
+ ).resolves.toEqual({
1415
+ call: toolClientCallMock,
1416
+ });
1417
+ });
1418
+
1419
+ it('reuses the cached tool client after initialization', async () => {
1420
+ const runner = new DashboardRuntimeRunner({
1421
+ repoRoot,
1422
+ instanceId: 'instance123456',
1423
+ env: process.env,
1424
+ });
1425
+ const runnerAny = runner as unknown as {
1426
+ getToolClient: () => Promise<{ call: typeof toolClientCallMock }>;
1427
+ };
1428
+
1429
+ const first = await runnerAny.getToolClient();
1430
+ const second = await runnerAny.getToolClient();
1431
+
1432
+ expect(first).toBe(second);
1433
+ });
1434
+
1435
+ it('throws when launch background startup encounters mismatched role providers', async () => {
1436
+ const { resolveProviderSelectionsByRole } = await import('../src/providers/providers.js');
1437
+ vi.mocked(resolveProviderSelectionsByRole).mockImplementationOnce(
1438
+ () =>
1439
+ ({
1440
+ planner: {
1441
+ provider: 'claude',
1442
+ },
1443
+ }) as never,
1444
+ );
1445
+
1446
+ const runner = new DashboardRuntimeRunner({
1447
+ repoRoot,
1448
+ instanceId: 'instance123456',
1449
+ env: process.env,
1450
+ });
1451
+
1452
+ await expect(
1453
+ (
1454
+ runner as unknown as {
1455
+ runLaunchInBackground: (
1456
+ runId: string,
1457
+ features: Array<{ feature_id: string; spec_path: string }>,
1458
+ ) => Promise<void>;
1459
+ }
1460
+ ).runLaunchInBackground('run:launch', [
1461
+ {
1462
+ feature_id: 'feature_a',
1463
+ spec_path: '.aop/features/feature_a/spec.md',
1464
+ },
1465
+ ]),
1466
+ ).rejects.toMatchObject({
1467
+ code: 'invalid_cli_args',
1468
+ details: {
1469
+ role: 'planner',
1470
+ base_provider: 'codex',
1471
+ role_provider: 'claude',
1472
+ },
1473
+ });
1474
+ });
1475
+
1476
+ it('passes object-shaped execution and observability policies into launch startup', async () => {
1477
+ supervisorStartMock.mockResolvedValue({ status: 'running', queue_depth: 1 });
1478
+ const runner = new DashboardRuntimeRunner({
1479
+ repoRoot,
1480
+ instanceId: 'instance123456',
1481
+ env: process.env,
1482
+ });
1483
+ (
1484
+ runner as unknown as {
1485
+ kernel: { getPolicySnapshot: () => Record<string, unknown> };
1486
+ }
1487
+ ).kernel.getPolicySnapshot = () => ({
1488
+ execution: { allow: true },
1489
+ observability: { verbose: true },
1490
+ });
1491
+
1492
+ await expect(
1493
+ (
1494
+ runner as unknown as {
1495
+ runLaunchInBackground: (
1496
+ runId: string,
1497
+ features: Array<{ feature_id: string; spec_path: string }>,
1498
+ ) => Promise<void>;
1499
+ }
1500
+ ).runLaunchInBackground('run:launch', [
1501
+ {
1502
+ feature_id: 'feature_a',
1503
+ spec_path: '.aop/features/feature_a/spec.md',
1504
+ },
1505
+ ]),
1506
+ ).resolves.toBeUndefined();
1507
+ });
1508
+
1509
+ it('returns unsupported_command for unknown dashboard runtime commands', async () => {
1510
+ const runner = new DashboardRuntimeRunner({
1511
+ repoRoot,
1512
+ instanceId: 'instance123456',
1513
+ env: process.env,
1514
+ });
1515
+
1516
+ await expect(
1517
+ runner.handleCommand({
1518
+ command: 'unknown.command' as never,
1519
+ feature_id: 'feature_a',
1520
+ args: {},
1521
+ }),
1522
+ ).resolves.toEqual({
1523
+ ok: false,
1524
+ error: {
1525
+ code: 'dashboard_runtime_unsupported_command',
1526
+ message: 'Unsupported dashboard runtime command: unknown.command',
1527
+ },
1528
+ });
1529
+ });
1530
+
1531
+ it('uses constructor defaults when optional instanceId and env are omitted', async () => {
1532
+ const runner = new DashboardRuntimeRunner({
1533
+ repoRoot,
1534
+ });
1535
+
1536
+ await expect(
1537
+ runner.handleCommand({
1538
+ command: 'unknown.command' as never,
1539
+ feature_id: 'feature_a',
1540
+ args: {},
1541
+ }),
1542
+ ).resolves.toEqual({
1543
+ ok: false,
1544
+ error: {
1545
+ code: 'dashboard_runtime_unsupported_command',
1546
+ message: 'Unsupported dashboard runtime command: unknown.command',
1547
+ },
1548
+ });
1549
+ });
1550
+ });