agentic-orchestrator 0.1.27 → 0.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (837) hide show
  1. package/.claude/settings.local.json +46 -1
  2. package/.cortexrc +28 -0
  3. package/.github/agents/copilot-instructions.md +29 -0
  4. package/.github/copilot-instructions.md +93 -0
  5. package/.vscode/settings.json +13 -0
  6. package/.vscode/tms.code-snippets +223 -0
  7. package/AGENTS.md +72 -1
  8. package/Agentic-Orchestrator.iml +12 -11
  9. package/CLAUDE.md +72 -1
  10. package/CONSTITUTION.md +504 -0
  11. package/FUTURE-ENHANCEMENTS.md +85 -0
  12. package/NEXT-TASKS.md +25 -0
  13. package/PROMPTS.md +161 -0
  14. package/README.md +126 -29
  15. package/agentic/orchestrator/agents.yaml +4 -3
  16. package/agentic/orchestrator/defaults/policy.defaults.yaml +39 -3
  17. package/agentic/orchestrator/gates.yaml +15 -3
  18. package/agentic/orchestrator/policy.yaml +47 -3
  19. package/agentic/orchestrator/prompts/builder.system.md +69 -20
  20. package/agentic/orchestrator/prompts/planner-intake.system.md +149 -0
  21. package/agentic/orchestrator/prompts/planner.system.md +113 -40
  22. package/agentic/orchestrator/prompts/qa.system.md +73 -18
  23. package/agentic/orchestrator/prompts/reconciler.system.md +119 -0
  24. package/agentic/orchestrator/schemas/agents.schema.json +89 -1
  25. package/agentic/orchestrator/schemas/execution-control.schema.json +242 -0
  26. package/agentic/orchestrator/schemas/index.schema.json +234 -0
  27. package/agentic/orchestrator/schemas/intake.review.schema.json +82 -0
  28. package/agentic/orchestrator/schemas/organizer-ordering-artifact.schema.json +75 -0
  29. package/agentic/orchestrator/schemas/plan.schema.json +44 -0
  30. package/agentic/orchestrator/schemas/policy.schema.json +238 -9
  31. package/agentic/orchestrator/schemas/policy.user.schema.json +129 -1
  32. package/agentic/orchestrator/schemas/spec.manifest.bootstrap.schema.json +101 -0
  33. package/agentic/orchestrator/schemas/spec.manifest.verified.schema.json +80 -0
  34. package/agentic/orchestrator/schemas/state.schema.json +298 -3
  35. package/agentic/orchestrator/tools/catalog.json +145 -15
  36. package/agentic/orchestrator/tools/schemas/input/doctor.run.input.schema.json +18 -0
  37. package/agentic/orchestrator/tools/schemas/input/evidence.latest.input.schema.json +4 -0
  38. package/agentic/orchestrator/tools/schemas/input/evidence.verify_chain.input.schema.json +13 -0
  39. package/agentic/orchestrator/tools/schemas/input/feature.intake_submit.input.schema.json +11 -0
  40. package/agentic/orchestrator/tools/schemas/input/feature.question_answer.input.schema.json +15 -0
  41. package/agentic/orchestrator/tools/schemas/input/feature.question_create.input.schema.json +21 -0
  42. package/agentic/orchestrator/tools/schemas/input/feature.question_list.input.schema.json +13 -0
  43. package/agentic/orchestrator/tools/schemas/input/feature.ready_to_merge.input.schema.json +5 -0
  44. package/agentic/orchestrator/tools/schemas/input/feature.send_message.input.schema.json +1 -1
  45. package/agentic/orchestrator/tools/schemas/input/replay.timeline_get.input.schema.json +32 -0
  46. package/agentic/orchestrator/tools/schemas/input/repo.conflict_abort.input.schema.json +16 -0
  47. package/agentic/orchestrator/tools/schemas/input/repo.conflict_files.input.schema.json +16 -0
  48. package/agentic/orchestrator/tools/schemas/input/repo.reconcile_mainline.input.schema.json +37 -0
  49. package/agentic/orchestrator/tools/schemas/input/repo.resolve_conflict.input.schema.json +40 -0
  50. package/agentic/orchestrator/tools/schemas/input/runtime.execution_request_list.input.schema.json +7 -0
  51. package/agentic/orchestrator/tools/schemas/input/runtime.execution_request_submit.input.schema.json +25 -0
  52. package/agentic/orchestrator/tools/schemas/output/doctor.run.output.schema.json +34 -0
  53. package/agentic/orchestrator/tools/schemas/output/evidence.verify_chain.output.schema.json +23 -0
  54. package/agentic/orchestrator/tools/schemas/output/feature.get_context.output.schema.json +62 -2
  55. package/agentic/orchestrator/tools/schemas/output/feature.intake_submit.output.schema.json +24 -0
  56. package/agentic/orchestrator/tools/schemas/output/feature.question_answer.output.schema.json +21 -0
  57. package/agentic/orchestrator/tools/schemas/output/feature.question_create.output.schema.json +12 -0
  58. package/agentic/orchestrator/tools/schemas/output/feature.question_list.output.schema.json +14 -0
  59. package/agentic/orchestrator/tools/schemas/output/feature.ready_to_merge.output.schema.json +31 -0
  60. package/agentic/orchestrator/tools/schemas/output/feature.send_message.output.schema.json +8 -18
  61. package/agentic/orchestrator/tools/schemas/output/replay.timeline_get.output.schema.json +64 -0
  62. package/agentic/orchestrator/tools/schemas/output/repo.conflict_abort.output.schema.json +16 -0
  63. package/agentic/orchestrator/tools/schemas/output/repo.conflict_files.output.schema.json +22 -0
  64. package/agentic/orchestrator/tools/schemas/output/repo.reconcile_mainline.output.schema.json +61 -0
  65. package/agentic/orchestrator/tools/schemas/output/repo.resolve_conflict.output.schema.json +19 -0
  66. package/agentic/orchestrator/tools/schemas/output/report.dashboard.output.schema.json +26 -0
  67. package/agentic/orchestrator/tools/schemas/output/runtime.execution_request_list.output.schema.json +17 -0
  68. package/agentic/orchestrator/tools/schemas/output/runtime.execution_request_submit.output.schema.json +24 -0
  69. package/agentic/orchestrator/tools.md +13 -0
  70. package/apps/control-plane/scripts/validate-mcp-contracts.ts +1 -1
  71. package/apps/control-plane/src/application/kernel-tool-wiring.ts +140 -2
  72. package/apps/control-plane/src/application/services/activity-monitor-service.ts +44 -1
  73. package/apps/control-plane/src/application/services/bootstrap-manifest-generator-service.ts +251 -0
  74. package/apps/control-plane/src/application/services/checkpoint-service.ts +87 -27
  75. package/apps/control-plane/src/application/services/collision-override-service.ts +906 -0
  76. package/apps/control-plane/src/application/services/collision-queue-service.ts +129 -38
  77. package/apps/control-plane/src/application/services/cost-tracking-service.ts +94 -0
  78. package/apps/control-plane/src/application/services/execution-control-service.ts +599 -0
  79. package/apps/control-plane/src/application/services/feature-deletion-service.ts +37 -1
  80. package/apps/control-plane/src/application/services/feature-lifecycle-service.ts +182 -4
  81. package/apps/control-plane/src/application/services/feature-send-message-service.ts +17 -8
  82. package/apps/control-plane/src/application/services/feature-state-service.ts +191 -6
  83. package/apps/control-plane/src/application/services/gate-service.ts +121 -2
  84. package/apps/control-plane/src/application/services/git-reconciliation-service.ts +1591 -0
  85. package/apps/control-plane/src/application/services/intake-service.ts +1468 -0
  86. package/apps/control-plane/src/application/services/merge-service.ts +308 -17
  87. package/apps/control-plane/src/application/services/notifier-service.ts +3 -1
  88. package/apps/control-plane/src/application/services/performance-analytics-service.ts +75 -0
  89. package/apps/control-plane/src/application/services/plan-service.ts +336 -20
  90. package/apps/control-plane/src/application/services/question-service.ts +693 -0
  91. package/apps/control-plane/src/application/services/reactions-service.ts +73 -17
  92. package/apps/control-plane/src/application/services/replay-timeline-service.ts +295 -0
  93. package/apps/control-plane/src/application/services/reporting-service.ts +194 -10
  94. package/apps/control-plane/src/application/services/run-lease-service.ts +121 -5
  95. package/apps/control-plane/src/application/services/worktree-watchdog-service.ts +95 -8
  96. package/apps/control-plane/src/application/tools/tool-metadata.ts +7 -0
  97. package/apps/control-plane/src/application/usage-types.ts +138 -0
  98. package/apps/control-plane/src/cli/add-command-handler.ts +162 -0
  99. package/apps/control-plane/src/cli/answer-command-handler.ts +113 -0
  100. package/apps/control-plane/src/cli/attach-command-handler.ts +12 -3
  101. package/apps/control-plane/src/cli/cli-argument-parser.ts +133 -11
  102. package/apps/control-plane/src/cli/collision-command-handler.ts +113 -0
  103. package/apps/control-plane/src/cli/command-catalog.ts +479 -0
  104. package/apps/control-plane/src/cli/complete-command-handler.ts +23 -0
  105. package/apps/control-plane/src/cli/completion-command-handler.ts +25 -0
  106. package/apps/control-plane/src/cli/completion-resolver.ts +319 -0
  107. package/apps/control-plane/src/cli/completion-shell-renderer.ts +58 -0
  108. package/apps/control-plane/src/cli/dashboard-command-handler.ts +110 -1
  109. package/apps/control-plane/src/cli/dashboard-runtime-runner.ts +1036 -0
  110. package/apps/control-plane/src/cli/dashboard-runtime.ts +31 -0
  111. package/apps/control-plane/src/cli/help-command-handler.ts +17 -185
  112. package/apps/control-plane/src/cli/init-command-handler.ts +51 -6
  113. package/apps/control-plane/src/cli/merge-command-handler.ts +200 -0
  114. package/apps/control-plane/src/cli/questions-command-handler.ts +70 -0
  115. package/apps/control-plane/src/cli/replay-command-handler.ts +98 -0
  116. package/apps/control-plane/src/cli/resume-command-handler.ts +252 -18
  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 +801 -43
  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/path-normalizers.spec.ts +41 -0
  236. package/apps/control-plane/test/performance-analytics.spec.ts +124 -0
  237. package/apps/control-plane/test/plan-conformance-scorer.spec.ts +53 -0
  238. package/apps/control-plane/test/plan-service.spec.ts +686 -4
  239. package/apps/control-plane/test/planning-wave-executor.spec.ts +3272 -86
  240. package/apps/control-plane/test/policy-loader-service.spec.ts +5 -0
  241. package/apps/control-plane/test/prompt-overlay.spec.ts +65 -0
  242. package/apps/control-plane/test/provider-command-runner-epipe.spec.ts +64 -0
  243. package/apps/control-plane/test/providers/api-worker-provider.spec.ts +129 -0
  244. package/apps/control-plane/test/providers/cli-worker-provider.spec.ts +148 -0
  245. package/apps/control-plane/test/providers/usage-types.spec.ts +98 -0
  246. package/apps/control-plane/test/providers.spec.ts +293 -16
  247. package/apps/control-plane/test/question-command-handlers.spec.ts +156 -0
  248. package/apps/control-plane/test/question-service.spec.ts +1119 -0
  249. package/apps/control-plane/test/reactions.spec.ts +114 -0
  250. package/apps/control-plane/test/replay-command-handler.spec.ts +144 -0
  251. package/apps/control-plane/test/replay-timeline-service.spec.ts +459 -0
  252. package/apps/control-plane/test/response.spec.ts +31 -0
  253. package/apps/control-plane/test/resume-command.spec.ts +786 -9
  254. package/apps/control-plane/test/retry-resume-decision.spec.ts +133 -0
  255. package/apps/control-plane/test/rollback-command-handler.spec.ts +334 -0
  256. package/apps/control-plane/test/rollback-command.spec.ts +120 -0
  257. package/apps/control-plane/test/run-coordinator.spec.ts +3141 -364
  258. package/apps/control-plane/test/schemas/state.schema.spec.ts +71 -0
  259. package/apps/control-plane/test/service-retry-paths.spec.ts +112 -0
  260. package/apps/control-plane/test/services.spec.ts +472 -2
  261. package/apps/control-plane/test/session-management.spec.ts +346 -1
  262. package/apps/control-plane/test/spec-ingestion.spec.ts +102 -28
  263. package/apps/control-plane/test/spec-preparation.spec.ts +182 -0
  264. package/apps/control-plane/test/supervisor-collaborators.spec.ts +191 -3
  265. package/apps/control-plane/test/supervisor.calltool.spec.ts +198 -0
  266. package/apps/control-plane/test/supervisor.spec.ts +95 -16
  267. package/apps/control-plane/test/supervisor.unit.spec.ts +385 -18
  268. package/apps/control-plane/test/tool-runtime.spec.ts +122 -0
  269. package/apps/control-plane/test/worker-decision-loop.spec.ts +3479 -476
  270. package/apps/control-plane/test/worker-execution-policy.spec.ts +1416 -6
  271. package/apps/control-plane/test/worker-provider-adapters.spec.ts +1894 -37
  272. package/apps/control-plane/test/worker-provider-factory.spec.ts +81 -0
  273. package/apps/control-plane/test/worktree-watchdog-service.spec.ts +125 -0
  274. package/apps/control-plane/vitest.config.ts +5 -0
  275. package/config/agentic/orchestrator/agents.yaml +23 -2
  276. package/config/agentic/orchestrator/gates.yaml +24 -7
  277. package/config/agentic/orchestrator/policy.yaml +23 -1
  278. package/config/agentic/orchestrator/prompts/builder.system.md +69 -20
  279. package/config/agentic/orchestrator/prompts/organizer.system.md +85 -0
  280. package/config/agentic/orchestrator/prompts/overrides/builder.claude.md +28 -0
  281. package/config/agentic/orchestrator/prompts/overrides/builder.codex.md +28 -0
  282. package/config/agentic/orchestrator/prompts/overrides/planner.claude.md +20 -0
  283. package/config/agentic/orchestrator/prompts/overrides/planner.codex.md +20 -0
  284. package/config/agentic/orchestrator/prompts/planner-intake.system.md +149 -0
  285. package/config/agentic/orchestrator/prompts/planner.system.md +113 -40
  286. package/config/agentic/orchestrator/prompts/qa.system.md +75 -18
  287. package/config/agentic/orchestrator/prompts/reconciler.system.md +119 -0
  288. package/dist/apps/control-plane/application/kernel-tool-wiring.d.ts +26 -2
  289. package/dist/apps/control-plane/application/kernel-tool-wiring.js +40 -2
  290. package/dist/apps/control-plane/application/kernel-tool-wiring.js.map +1 -1
  291. package/dist/apps/control-plane/application/services/activity-monitor-service.js +37 -1
  292. package/dist/apps/control-plane/application/services/activity-monitor-service.js.map +1 -1
  293. package/dist/apps/control-plane/application/services/bootstrap-manifest-generator-service.d.ts +4 -0
  294. package/dist/apps/control-plane/application/services/bootstrap-manifest-generator-service.js +188 -0
  295. package/dist/apps/control-plane/application/services/bootstrap-manifest-generator-service.js.map +1 -0
  296. package/dist/apps/control-plane/application/services/checkpoint-service.d.ts +5 -0
  297. package/dist/apps/control-plane/application/services/checkpoint-service.js +69 -24
  298. package/dist/apps/control-plane/application/services/checkpoint-service.js.map +1 -1
  299. package/dist/apps/control-plane/application/services/collision-override-service.d.ts +139 -0
  300. package/dist/apps/control-plane/application/services/collision-override-service.js +568 -0
  301. package/dist/apps/control-plane/application/services/collision-override-service.js.map +1 -0
  302. package/dist/apps/control-plane/application/services/collision-queue-service.d.ts +15 -0
  303. package/dist/apps/control-plane/application/services/collision-queue-service.js +92 -33
  304. package/dist/apps/control-plane/application/services/collision-queue-service.js.map +1 -1
  305. package/dist/apps/control-plane/application/services/cost-tracking-service.d.ts +11 -0
  306. package/dist/apps/control-plane/application/services/cost-tracking-service.js +75 -0
  307. package/dist/apps/control-plane/application/services/cost-tracking-service.js.map +1 -1
  308. package/dist/apps/control-plane/application/services/execution-control-service.d.ts +75 -0
  309. package/dist/apps/control-plane/application/services/execution-control-service.js +421 -0
  310. package/dist/apps/control-plane/application/services/execution-control-service.js.map +1 -0
  311. package/dist/apps/control-plane/application/services/feature-deletion-service.d.ts +1 -0
  312. package/dist/apps/control-plane/application/services/feature-deletion-service.js +23 -1
  313. package/dist/apps/control-plane/application/services/feature-deletion-service.js.map +1 -1
  314. package/dist/apps/control-plane/application/services/feature-lifecycle-service.d.ts +24 -1
  315. package/dist/apps/control-plane/application/services/feature-lifecycle-service.js +132 -3
  316. package/dist/apps/control-plane/application/services/feature-lifecycle-service.js.map +1 -1
  317. package/dist/apps/control-plane/application/services/feature-send-message-service.js +16 -8
  318. package/dist/apps/control-plane/application/services/feature-send-message-service.js.map +1 -1
  319. package/dist/apps/control-plane/application/services/feature-state-service.d.ts +36 -0
  320. package/dist/apps/control-plane/application/services/feature-state-service.js +163 -6
  321. package/dist/apps/control-plane/application/services/feature-state-service.js.map +1 -1
  322. package/dist/apps/control-plane/application/services/gate-service.d.ts +2 -1
  323. package/dist/apps/control-plane/application/services/gate-service.js +95 -5
  324. package/dist/apps/control-plane/application/services/gate-service.js.map +1 -1
  325. package/dist/apps/control-plane/application/services/git-reconciliation-service.d.ts +92 -0
  326. package/dist/apps/control-plane/application/services/git-reconciliation-service.js +1097 -0
  327. package/dist/apps/control-plane/application/services/git-reconciliation-service.js.map +1 -0
  328. package/dist/apps/control-plane/application/services/intake-service.d.ts +63 -0
  329. package/dist/apps/control-plane/application/services/intake-service.js +1050 -0
  330. package/dist/apps/control-plane/application/services/intake-service.js.map +1 -0
  331. package/dist/apps/control-plane/application/services/merge-service.d.ts +5 -1
  332. package/dist/apps/control-plane/application/services/merge-service.js +233 -18
  333. package/dist/apps/control-plane/application/services/merge-service.js.map +1 -1
  334. package/dist/apps/control-plane/application/services/notifier-service.d.ts +1 -1
  335. package/dist/apps/control-plane/application/services/notifier-service.js +1 -0
  336. package/dist/apps/control-plane/application/services/notifier-service.js.map +1 -1
  337. package/dist/apps/control-plane/application/services/performance-analytics-service.d.ts +11 -0
  338. package/dist/apps/control-plane/application/services/performance-analytics-service.js +59 -0
  339. package/dist/apps/control-plane/application/services/performance-analytics-service.js.map +1 -1
  340. package/dist/apps/control-plane/application/services/plan-service.d.ts +5 -0
  341. package/dist/apps/control-plane/application/services/plan-service.js +254 -15
  342. package/dist/apps/control-plane/application/services/plan-service.js.map +1 -1
  343. package/dist/apps/control-plane/application/services/question-service.d.ts +72 -0
  344. package/dist/apps/control-plane/application/services/question-service.js +507 -0
  345. package/dist/apps/control-plane/application/services/question-service.js.map +1 -0
  346. package/dist/apps/control-plane/application/services/reactions-service.d.ts +2 -0
  347. package/dist/apps/control-plane/application/services/reactions-service.js +60 -17
  348. package/dist/apps/control-plane/application/services/reactions-service.js.map +1 -1
  349. package/dist/apps/control-plane/application/services/replay-timeline-service.d.ts +39 -0
  350. package/dist/apps/control-plane/application/services/replay-timeline-service.js +205 -0
  351. package/dist/apps/control-plane/application/services/replay-timeline-service.js.map +1 -0
  352. package/dist/apps/control-plane/application/services/reporting-service.d.ts +59 -0
  353. package/dist/apps/control-plane/application/services/reporting-service.js +121 -9
  354. package/dist/apps/control-plane/application/services/reporting-service.js.map +1 -1
  355. package/dist/apps/control-plane/application/services/run-lease-service.d.ts +20 -0
  356. package/dist/apps/control-plane/application/services/run-lease-service.js +81 -4
  357. package/dist/apps/control-plane/application/services/run-lease-service.js.map +1 -1
  358. package/dist/apps/control-plane/application/services/worktree-watchdog-service.d.ts +10 -0
  359. package/dist/apps/control-plane/application/services/worktree-watchdog-service.js +65 -8
  360. package/dist/apps/control-plane/application/services/worktree-watchdog-service.js.map +1 -1
  361. package/dist/apps/control-plane/application/tools/tool-metadata.js +7 -0
  362. package/dist/apps/control-plane/application/tools/tool-metadata.js.map +1 -1
  363. package/dist/apps/control-plane/application/usage-types.d.ts +65 -0
  364. package/dist/apps/control-plane/application/usage-types.js +75 -0
  365. package/dist/apps/control-plane/application/usage-types.js.map +1 -0
  366. package/dist/apps/control-plane/cli/add-command-handler.d.ts +18 -0
  367. package/dist/apps/control-plane/cli/add-command-handler.js +110 -0
  368. package/dist/apps/control-plane/cli/add-command-handler.js.map +1 -0
  369. package/dist/apps/control-plane/cli/answer-command-handler.d.ts +8 -0
  370. package/dist/apps/control-plane/cli/answer-command-handler.js +96 -0
  371. package/dist/apps/control-plane/cli/answer-command-handler.js.map +1 -0
  372. package/dist/apps/control-plane/cli/attach-command-handler.js +8 -3
  373. package/dist/apps/control-plane/cli/attach-command-handler.js.map +1 -1
  374. package/dist/apps/control-plane/cli/cli-argument-parser.js +131 -11
  375. package/dist/apps/control-plane/cli/cli-argument-parser.js.map +1 -1
  376. package/dist/apps/control-plane/cli/collision-command-handler.d.ts +8 -0
  377. package/dist/apps/control-plane/cli/collision-command-handler.js +90 -0
  378. package/dist/apps/control-plane/cli/collision-command-handler.js.map +1 -0
  379. package/dist/apps/control-plane/cli/command-catalog.d.ts +21 -0
  380. package/dist/apps/control-plane/cli/command-catalog.js +416 -0
  381. package/dist/apps/control-plane/cli/command-catalog.js.map +1 -0
  382. package/dist/apps/control-plane/cli/complete-command-handler.d.ts +15 -0
  383. package/dist/apps/control-plane/cli/complete-command-handler.js +26 -0
  384. package/dist/apps/control-plane/cli/complete-command-handler.js.map +1 -0
  385. package/dist/apps/control-plane/cli/completion-command-handler.d.ts +8 -0
  386. package/dist/apps/control-plane/cli/completion-command-handler.js +20 -0
  387. package/dist/apps/control-plane/cli/completion-command-handler.js.map +1 -0
  388. package/dist/apps/control-plane/cli/completion-resolver.d.ts +1 -0
  389. package/dist/apps/control-plane/cli/completion-resolver.js +250 -0
  390. package/dist/apps/control-plane/cli/completion-resolver.js.map +1 -0
  391. package/dist/apps/control-plane/cli/completion-shell-renderer.d.ts +3 -0
  392. package/dist/apps/control-plane/cli/completion-shell-renderer.js +53 -0
  393. package/dist/apps/control-plane/cli/completion-shell-renderer.js.map +1 -0
  394. package/dist/apps/control-plane/cli/dashboard-command-handler.d.ts +1 -0
  395. package/dist/apps/control-plane/cli/dashboard-command-handler.js +83 -1
  396. package/dist/apps/control-plane/cli/dashboard-command-handler.js.map +1 -1
  397. package/dist/apps/control-plane/cli/dashboard-runtime-runner.d.ts +81 -0
  398. package/dist/apps/control-plane/cli/dashboard-runtime-runner.js +724 -0
  399. package/dist/apps/control-plane/cli/dashboard-runtime-runner.js.map +1 -0
  400. package/dist/apps/control-plane/cli/dashboard-runtime.d.ts +1 -0
  401. package/dist/apps/control-plane/cli/dashboard-runtime.js +26 -0
  402. package/dist/apps/control-plane/cli/dashboard-runtime.js.map +1 -0
  403. package/dist/apps/control-plane/cli/help-command-handler.js +13 -172
  404. package/dist/apps/control-plane/cli/help-command-handler.js.map +1 -1
  405. package/dist/apps/control-plane/cli/init-command-handler.js +51 -6
  406. package/dist/apps/control-plane/cli/init-command-handler.js.map +1 -1
  407. package/dist/apps/control-plane/cli/merge-command-handler.d.ts +8 -0
  408. package/dist/apps/control-plane/cli/merge-command-handler.js +139 -0
  409. package/dist/apps/control-plane/cli/merge-command-handler.js.map +1 -0
  410. package/dist/apps/control-plane/cli/questions-command-handler.d.ts +8 -0
  411. package/dist/apps/control-plane/cli/questions-command-handler.js +59 -0
  412. package/dist/apps/control-plane/cli/questions-command-handler.js.map +1 -0
  413. package/dist/apps/control-plane/cli/replay-command-handler.d.ts +15 -0
  414. package/dist/apps/control-plane/cli/replay-command-handler.js +55 -0
  415. package/dist/apps/control-plane/cli/replay-command-handler.js.map +1 -0
  416. package/dist/apps/control-plane/cli/resume-command-handler.d.ts +2 -0
  417. package/dist/apps/control-plane/cli/resume-command-handler.js +196 -19
  418. package/dist/apps/control-plane/cli/resume-command-handler.js.map +1 -1
  419. package/dist/apps/control-plane/cli/retry-command-handler.js +202 -16
  420. package/dist/apps/control-plane/cli/retry-command-handler.js.map +1 -1
  421. package/dist/apps/control-plane/cli/retry-resume-decision.d.ts +26 -0
  422. package/dist/apps/control-plane/cli/retry-resume-decision.js +61 -0
  423. package/dist/apps/control-plane/cli/retry-resume-decision.js.map +1 -0
  424. package/dist/apps/control-plane/cli/rollback-command-handler.js +3 -2
  425. package/dist/apps/control-plane/cli/rollback-command-handler.js.map +1 -1
  426. package/dist/apps/control-plane/cli/run-command-handler.js +26 -2
  427. package/dist/apps/control-plane/cli/run-command-handler.js.map +1 -1
  428. package/dist/apps/control-plane/cli/spec-ingestion-service.d.ts +2 -0
  429. package/dist/apps/control-plane/cli/spec-ingestion-service.js +37 -48
  430. package/dist/apps/control-plane/cli/spec-ingestion-service.js.map +1 -1
  431. package/dist/apps/control-plane/cli/spec-preparation.d.ts +14 -0
  432. package/dist/apps/control-plane/cli/spec-preparation.js +81 -0
  433. package/dist/apps/control-plane/cli/spec-preparation.js.map +1 -0
  434. package/dist/apps/control-plane/cli/spec-utils.d.ts +4 -0
  435. package/dist/apps/control-plane/cli/spec-utils.js +70 -11
  436. package/dist/apps/control-plane/cli/spec-utils.js.map +1 -1
  437. package/dist/apps/control-plane/cli/status-command-handler.js +69 -0
  438. package/dist/apps/control-plane/cli/status-command-handler.js.map +1 -1
  439. package/dist/apps/control-plane/cli/types.d.ts +41 -4
  440. package/dist/apps/control-plane/cli/types.js +9 -1
  441. package/dist/apps/control-plane/cli/types.js.map +1 -1
  442. package/dist/apps/control-plane/core/collisions.d.ts +37 -19
  443. package/dist/apps/control-plane/core/collisions.js +87 -12
  444. package/dist/apps/control-plane/core/collisions.js.map +1 -1
  445. package/dist/apps/control-plane/core/constants.d.ts +17 -1
  446. package/dist/apps/control-plane/core/constants.js +18 -1
  447. package/dist/apps/control-plane/core/constants.js.map +1 -1
  448. package/dist/apps/control-plane/core/error-codes.d.ts +39 -0
  449. package/dist/apps/control-plane/core/error-codes.js +39 -0
  450. package/dist/apps/control-plane/core/error-codes.js.map +1 -1
  451. package/dist/apps/control-plane/core/execution-control.d.ts +45 -0
  452. package/dist/apps/control-plane/core/execution-control.js +2 -0
  453. package/dist/apps/control-plane/core/execution-control.js.map +1 -0
  454. package/dist/apps/control-plane/core/feature-resume-phase.d.ts +3 -0
  455. package/dist/apps/control-plane/core/feature-resume-phase.js +88 -0
  456. package/dist/apps/control-plane/core/feature-resume-phase.js.map +1 -0
  457. package/dist/apps/control-plane/core/gate-freshness.d.ts +48 -0
  458. package/dist/apps/control-plane/core/gate-freshness.js +267 -0
  459. package/dist/apps/control-plane/core/gate-freshness.js.map +1 -0
  460. package/dist/apps/control-plane/core/gate-log-extractor.d.ts +22 -0
  461. package/dist/apps/control-plane/core/gate-log-extractor.js +66 -0
  462. package/dist/apps/control-plane/core/gate-log-extractor.js.map +1 -0
  463. package/dist/apps/control-plane/core/gates.d.ts +11 -2
  464. package/dist/apps/control-plane/core/gates.js +67 -3
  465. package/dist/apps/control-plane/core/gates.js.map +1 -1
  466. package/dist/apps/control-plane/core/intake-artifacts.d.ts +109 -0
  467. package/dist/apps/control-plane/core/intake-artifacts.js +143 -0
  468. package/dist/apps/control-plane/core/intake-artifacts.js.map +1 -0
  469. package/dist/apps/control-plane/core/kernel-types.d.ts +8 -0
  470. package/dist/apps/control-plane/core/kernel.d.ts +256 -8
  471. package/dist/apps/control-plane/core/kernel.js +400 -14
  472. package/dist/apps/control-plane/core/kernel.js.map +1 -1
  473. package/dist/apps/control-plane/core/mainline-conflict.d.ts +7 -0
  474. package/dist/apps/control-plane/core/mainline-conflict.js +20 -0
  475. package/dist/apps/control-plane/core/mainline-conflict.js.map +1 -0
  476. package/dist/apps/control-plane/core/merge-repair.d.ts +35 -0
  477. package/dist/apps/control-plane/core/merge-repair.js +99 -0
  478. package/dist/apps/control-plane/core/merge-repair.js.map +1 -0
  479. package/dist/apps/control-plane/core/path-layout.d.ts +10 -0
  480. package/dist/apps/control-plane/core/path-layout.js +32 -2
  481. package/dist/apps/control-plane/core/path-layout.js.map +1 -1
  482. package/dist/apps/control-plane/core/path-rules.js +9 -3
  483. package/dist/apps/control-plane/core/path-rules.js.map +1 -1
  484. package/dist/apps/control-plane/core/plan-submit-recovery.d.ts +22 -0
  485. package/dist/apps/control-plane/core/plan-submit-recovery.js +78 -0
  486. package/dist/apps/control-plane/core/plan-submit-recovery.js.map +1 -0
  487. package/dist/apps/control-plane/core/questions.d.ts +40 -0
  488. package/dist/apps/control-plane/core/questions.js +2 -0
  489. package/dist/apps/control-plane/core/questions.js.map +1 -0
  490. package/dist/apps/control-plane/core/runtime-sessions.d.ts +4 -0
  491. package/dist/apps/control-plane/core/schemas.d.ts +2 -0
  492. package/dist/apps/control-plane/core/schemas.js +31 -1
  493. package/dist/apps/control-plane/core/schemas.js.map +1 -1
  494. package/dist/apps/control-plane/core/tool-caller.d.ts +18 -1
  495. package/dist/apps/control-plane/core/utils/index-normalizer.js +17 -4
  496. package/dist/apps/control-plane/core/utils/index-normalizer.js.map +1 -1
  497. package/dist/apps/control-plane/core/worktree-diff.d.ts +4 -0
  498. package/dist/apps/control-plane/core/worktree-diff.js +52 -0
  499. package/dist/apps/control-plane/core/worktree-diff.js.map +1 -0
  500. package/dist/apps/control-plane/index.d.ts +10 -2
  501. package/dist/apps/control-plane/index.js +9 -2
  502. package/dist/apps/control-plane/index.js.map +1 -1
  503. package/dist/apps/control-plane/interfaces/cli/bootstrap.js +236 -6
  504. package/dist/apps/control-plane/interfaces/cli/bootstrap.js.map +1 -1
  505. package/dist/apps/control-plane/mcp/kernel-tool-executor.js +16 -0
  506. package/dist/apps/control-plane/mcp/kernel-tool-executor.js.map +1 -1
  507. package/dist/apps/control-plane/mcp/tool-runtime.d.ts +5 -0
  508. package/dist/apps/control-plane/mcp/tool-runtime.js +40 -5
  509. package/dist/apps/control-plane/mcp/tool-runtime.js.map +1 -1
  510. package/dist/apps/control-plane/providers/api-worker-provider.d.ts +2 -2
  511. package/dist/apps/control-plane/providers/api-worker-provider.js +40 -9
  512. package/dist/apps/control-plane/providers/api-worker-provider.js.map +1 -1
  513. package/dist/apps/control-plane/providers/cli-worker-provider.d.ts +59 -3
  514. package/dist/apps/control-plane/providers/cli-worker-provider.js +758 -46
  515. package/dist/apps/control-plane/providers/cli-worker-provider.js.map +1 -1
  516. package/dist/apps/control-plane/providers/output-parsers/generic-output-parser.js +91 -1
  517. package/dist/apps/control-plane/providers/output-parsers/generic-output-parser.js.map +1 -1
  518. package/dist/apps/control-plane/providers/output-parsers/types.d.ts +2 -0
  519. package/dist/apps/control-plane/providers/provider-defaults.d.ts +12 -0
  520. package/dist/apps/control-plane/providers/provider-defaults.js +103 -7
  521. package/dist/apps/control-plane/providers/provider-defaults.js.map +1 -1
  522. package/dist/apps/control-plane/providers/providers.d.ts +50 -4
  523. package/dist/apps/control-plane/providers/providers.js +145 -14
  524. package/dist/apps/control-plane/providers/providers.js.map +1 -1
  525. package/dist/apps/control-plane/providers/worker-provider-factory.d.ts +2 -0
  526. package/dist/apps/control-plane/providers/worker-provider-factory.js +8 -1
  527. package/dist/apps/control-plane/providers/worker-provider-factory.js.map +1 -1
  528. package/dist/apps/control-plane/supervisor/artifact-stager.d.ts +5 -0
  529. package/dist/apps/control-plane/supervisor/artifact-stager.js +45 -0
  530. package/dist/apps/control-plane/supervisor/artifact-stager.js.map +1 -0
  531. package/dist/apps/control-plane/supervisor/build-wave-executor.d.ts +24 -1
  532. package/dist/apps/control-plane/supervisor/build-wave-executor.js +362 -150
  533. package/dist/apps/control-plane/supervisor/build-wave-executor.js.map +1 -1
  534. package/dist/apps/control-plane/supervisor/execution-enrollment-service.d.ts +41 -0
  535. package/dist/apps/control-plane/supervisor/execution-enrollment-service.js +311 -0
  536. package/dist/apps/control-plane/supervisor/execution-enrollment-service.js.map +1 -0
  537. package/dist/apps/control-plane/supervisor/organizer-enrollment-scheduler.d.ts +15 -0
  538. package/dist/apps/control-plane/supervisor/organizer-enrollment-scheduler.js +93 -0
  539. package/dist/apps/control-plane/supervisor/organizer-enrollment-scheduler.js.map +1 -0
  540. package/dist/apps/control-plane/supervisor/organizer-sidecar-service.d.ts +44 -0
  541. package/dist/apps/control-plane/supervisor/organizer-sidecar-service.js +311 -0
  542. package/dist/apps/control-plane/supervisor/organizer-sidecar-service.js.map +1 -0
  543. package/dist/apps/control-plane/supervisor/plan-conformance-scorer.js +2 -5
  544. package/dist/apps/control-plane/supervisor/plan-conformance-scorer.js.map +1 -1
  545. package/dist/apps/control-plane/supervisor/planner-phase.d.ts +3 -0
  546. package/dist/apps/control-plane/supervisor/planner-phase.js +70 -0
  547. package/dist/apps/control-plane/supervisor/planner-phase.js.map +1 -0
  548. package/dist/apps/control-plane/supervisor/planning-wave-executor.d.ts +42 -0
  549. package/dist/apps/control-plane/supervisor/planning-wave-executor.js +753 -55
  550. package/dist/apps/control-plane/supervisor/planning-wave-executor.js.map +1 -1
  551. package/dist/apps/control-plane/supervisor/prompt-bundle-loader.js +19 -1
  552. package/dist/apps/control-plane/supervisor/prompt-bundle-loader.js.map +1 -1
  553. package/dist/apps/control-plane/supervisor/qa-wave-executor.d.ts +21 -0
  554. package/dist/apps/control-plane/supervisor/qa-wave-executor.js +287 -156
  555. package/dist/apps/control-plane/supervisor/qa-wave-executor.js.map +1 -1
  556. package/dist/apps/control-plane/supervisor/run-coordinator.d.ts +33 -1
  557. package/dist/apps/control-plane/supervisor/run-coordinator.js +631 -39
  558. package/dist/apps/control-plane/supervisor/run-coordinator.js.map +1 -1
  559. package/dist/apps/control-plane/supervisor/runtime.d.ts +84 -0
  560. package/dist/apps/control-plane/supervisor/runtime.js +393 -3
  561. package/dist/apps/control-plane/supervisor/runtime.js.map +1 -1
  562. package/dist/apps/control-plane/supervisor/session-orchestrator.d.ts +54 -0
  563. package/dist/apps/control-plane/supervisor/session-orchestrator.js +176 -1
  564. package/dist/apps/control-plane/supervisor/session-orchestrator.js.map +1 -1
  565. package/dist/apps/control-plane/supervisor/types.d.ts +142 -1
  566. package/dist/apps/control-plane/supervisor/types.js.map +1 -1
  567. package/dist/apps/control-plane/supervisor/worker-decision-loop.d.ts +68 -2
  568. package/dist/apps/control-plane/supervisor/worker-decision-loop.js +723 -89
  569. package/dist/apps/control-plane/supervisor/worker-decision-loop.js.map +1 -1
  570. package/docs/core/ARCHITECTURE.md +227 -0
  571. package/docs/core/DECISIONS.md +94 -0
  572. package/docs/core/DOMAIN-LOGIC.md +60 -0
  573. package/docs/core/PATTERNS.md +201 -0
  574. package/docs/core/TROUBLESHOOTING.md +347 -0
  575. package/docs/core/intentgraph-dependencies.json +39860 -0
  576. package/docs/core/intentgraph.index.json +46580 -0
  577. package/docs/plans/2026-03-10-gate-failure-targeted-repair-design.md +224 -0
  578. package/docs/plans/2026-03-10-gate-failure-targeted-repair.md +1032 -0
  579. package/docs/superpowers/plans/2026-03-16-provider-cli-config.md +743 -0
  580. package/docs/superpowers/plans/2026-03-23-reconcile-divergence-fix.md +777 -0
  581. package/docs/superpowers/plans/2026-03-28-ordering-agent-implementation.md +1754 -0
  582. package/docs/superpowers/plans/2026-03-29-drop-zone-and-provider-optimization.md +1108 -0
  583. package/docs/superpowers/plans/2026-03-29-merge-target-feature-branch.md +685 -0
  584. package/docs/superpowers/plans/2026-03-29-organizer-sidecar-runtime-loop.md +1289 -0
  585. package/docs/superpowers/specs/2026-03-23-reconcile-divergence-fix-design.md +118 -0
  586. package/docs/superpowers/specs/2026-03-28-ordering-agent-spec-audit-design.md +50 -0
  587. package/docs/superpowers/specs/2026-03-29-drop-zone-and-provider-optimization-design.md +254 -0
  588. package/docs/superpowers/specs/2026-03-29-merge-target-feature-branch-design.md +152 -0
  589. package/docs/superpowers/specs/2026-03-29-organizer-sidecar-runtime-loop-design.md +225 -0
  590. package/package.json +3 -2
  591. package/packages/web-dashboard/package.json +2 -1
  592. package/packages/web-dashboard/src/app/analytics/page.tsx +36 -2
  593. package/packages/web-dashboard/src/app/api/actions/route.ts +274 -63
  594. package/packages/web-dashboard/src/app/api/actions/status/route.ts +35 -0
  595. package/packages/web-dashboard/src/app/api/analytics/provider/route.ts +18 -0
  596. package/packages/web-dashboard/src/app/api/collisions/approve/route.ts +58 -0
  597. package/packages/web-dashboard/src/app/api/features/[id]/checkpoint-diff/route.ts +36 -0
  598. package/packages/web-dashboard/src/app/api/features/[id]/checkpoints/route.ts +29 -0
  599. package/packages/web-dashboard/src/app/api/features/[id]/conflicts/abort/route.ts +29 -0
  600. package/packages/web-dashboard/src/app/api/features/[id]/conflicts/files/route.ts +30 -0
  601. package/packages/web-dashboard/src/app/api/features/[id]/conflicts/resolve/route.ts +51 -0
  602. package/packages/web-dashboard/src/app/api/features/[id]/conflicts/route.ts +75 -0
  603. package/packages/web-dashboard/src/app/api/features/[id]/diff/route.ts +16 -2
  604. package/packages/web-dashboard/src/app/api/features/[id]/files/route.ts +26 -0
  605. package/packages/web-dashboard/src/app/api/features/[id]/gate-history/route.ts +27 -0
  606. package/packages/web-dashboard/src/app/api/features/[id]/genealogy/route.ts +26 -0
  607. package/packages/web-dashboard/src/app/api/features/[id]/history/run/[runId]/route.ts +20 -0
  608. package/packages/web-dashboard/src/app/api/features/[id]/history/runs/route.ts +34 -0
  609. package/packages/web-dashboard/src/app/api/features/[id]/intake-workspace/route.ts +20 -0
  610. package/packages/web-dashboard/src/app/api/features/[id]/live-output/route.ts +74 -0
  611. package/packages/web-dashboard/src/app/api/features/[id]/plan/amend/route.ts +21 -0
  612. package/packages/web-dashboard/src/app/api/features/[id]/plan-progress/route.ts +20 -0
  613. package/packages/web-dashboard/src/app/api/features/[id]/planner-artifacts/[artifact]/route.ts +78 -0
  614. package/packages/web-dashboard/src/app/api/features/[id]/planner-lifecycle/route.ts +20 -0
  615. package/packages/web-dashboard/src/app/api/features/[id]/planning-workspace/route.ts +20 -0
  616. package/packages/web-dashboard/src/app/api/features/[id]/questions/[questionId]/answer/route.ts +27 -0
  617. package/packages/web-dashboard/src/app/api/features/[id]/questions/route.ts +18 -0
  618. package/packages/web-dashboard/src/app/api/features/[id]/review/route.ts +14 -7
  619. package/packages/web-dashboard/src/app/api/features/[id]/route.ts +57 -2
  620. package/packages/web-dashboard/src/app/api/features/[id]/spec/route.ts +30 -0
  621. package/packages/web-dashboard/src/app/api/features/[id]/triage/route.ts +83 -0
  622. package/packages/web-dashboard/src/app/api/features/[id]/worker-events/route.ts +40 -0
  623. package/packages/web-dashboard/src/app/api/launch/preview/route.ts +86 -0
  624. package/packages/web-dashboard/src/app/api/launch/submit/route.ts +180 -0
  625. package/packages/web-dashboard/src/app/api/mainline/status/route.ts +74 -0
  626. package/packages/web-dashboard/src/app/api/merge-queue/route.ts +13 -0
  627. package/packages/web-dashboard/src/app/api/policy/budget/route.ts +14 -0
  628. package/packages/web-dashboard/src/app/api/projects/route.ts +11 -7
  629. package/packages/web-dashboard/src/app/api/reconciler/queue/route.ts +47 -0
  630. package/packages/web-dashboard/src/app/api/run/route.ts +26 -2
  631. package/packages/web-dashboard/src/app/api/runtime/events/route.ts +227 -0
  632. package/packages/web-dashboard/src/app/api/runtime/operations/route.ts +269 -0
  633. package/packages/web-dashboard/src/app/api/runtime/questions/route.ts +11 -0
  634. package/packages/web-dashboard/src/app/api/runtime/runs/route.ts +80 -0
  635. package/packages/web-dashboard/src/app/api/status/route.ts +4 -2
  636. package/packages/web-dashboard/src/app/feature/[id]/page.tsx +32 -42
  637. package/packages/web-dashboard/src/app/globals.css +34 -3
  638. package/packages/web-dashboard/src/app/launch/page.tsx +357 -0
  639. package/packages/web-dashboard/src/app/layout.tsx +23 -1
  640. package/packages/web-dashboard/src/app/page.tsx +263 -272
  641. package/packages/web-dashboard/src/components/dashboard/attention-strip.tsx +52 -0
  642. package/packages/web-dashboard/src/components/dashboard/collision-approval-drawer.tsx +185 -0
  643. package/packages/web-dashboard/src/components/dashboard/command-center-header.tsx +102 -0
  644. package/packages/web-dashboard/src/components/dashboard/mainline-status-banner.tsx +84 -0
  645. package/packages/web-dashboard/src/components/dashboard/merged-archive.tsx +36 -0
  646. package/packages/web-dashboard/src/components/dashboard/prioritized-queues.tsx +98 -0
  647. package/packages/web-dashboard/src/components/dashboard/reconciler-queue-card.tsx +115 -0
  648. package/packages/web-dashboard/src/components/dashboard/secondary-diagnostics-rail.tsx +48 -0
  649. package/packages/web-dashboard/src/components/dashboard/task-filter-bar.tsx +74 -0
  650. package/packages/web-dashboard/src/components/dashboard/triage-drawer.tsx +455 -0
  651. package/packages/web-dashboard/src/components/diff-viewer.tsx +19 -3
  652. package/packages/web-dashboard/src/components/evidence-viewer.tsx +65 -51
  653. package/packages/web-dashboard/src/components/feature-card.tsx +90 -7
  654. package/packages/web-dashboard/src/components/feature-cost-panel.tsx +112 -11
  655. package/packages/web-dashboard/src/components/feature-list-view.tsx +25 -4
  656. package/packages/web-dashboard/src/components/features/runtime-inspector/EventsTimelineView.tsx +260 -0
  657. package/packages/web-dashboard/src/components/features/runtime-inspector/OperationsListView.tsx +172 -0
  658. package/packages/web-dashboard/src/components/features/runtime-inspector/RuntimeInspectorPanel.tsx +896 -0
  659. package/packages/web-dashboard/src/components/filter-bar.tsx +7 -39
  660. package/packages/web-dashboard/src/components/focus/ActionableRiskList.tsx +46 -0
  661. package/packages/web-dashboard/src/components/focus/AgentRolePerformanceCard.tsx +200 -0
  662. package/packages/web-dashboard/src/components/focus/BlockedGuidanceBanner.tsx +149 -0
  663. package/packages/web-dashboard/src/components/focus/CheckpointInspector.tsx +123 -0
  664. package/packages/web-dashboard/src/components/focus/CheckpointRail.tsx +118 -0
  665. package/packages/web-dashboard/src/components/focus/CheckpointScrubber.tsx +249 -0
  666. package/packages/web-dashboard/src/components/focus/CollisionApprovalBanner.tsx +192 -0
  667. package/packages/web-dashboard/src/components/focus/CollisionRadar.tsx +136 -0
  668. package/packages/web-dashboard/src/components/focus/ConflictStatusCard.tsx +52 -0
  669. package/packages/web-dashboard/src/components/focus/ContextSidebar.tsx +108 -0
  670. package/packages/web-dashboard/src/components/focus/DiagnosisPanel.tsx +68 -0
  671. package/packages/web-dashboard/src/components/focus/FeatureDecisionBanner.tsx +68 -0
  672. package/packages/web-dashboard/src/components/focus/FeatureQuestionAnswerPanel.tsx +167 -0
  673. package/packages/web-dashboard/src/components/focus/FocusHeader.tsx +54 -0
  674. package/packages/web-dashboard/src/components/focus/FocusLayout.tsx +283 -0
  675. package/packages/web-dashboard/src/components/focus/GateFlakinessSummary.tsx +144 -0
  676. package/packages/web-dashboard/src/components/focus/GenealogyTree.tsx +34 -0
  677. package/packages/web-dashboard/src/components/focus/HeroBlock.tsx +67 -0
  678. package/packages/web-dashboard/src/components/focus/LiveAgentConsole.tsx +277 -0
  679. package/packages/web-dashboard/src/components/focus/MergeQueueCard.tsx +78 -0
  680. package/packages/web-dashboard/src/components/focus/OperationalSummaryCard.tsx +227 -0
  681. package/packages/web-dashboard/src/components/focus/PinnedActions.tsx +96 -0
  682. package/packages/web-dashboard/src/components/focus/PlanAmendmentPanel.tsx +250 -0
  683. package/packages/web-dashboard/src/components/focus/PlanProgressPanel.tsx +133 -0
  684. package/packages/web-dashboard/src/components/focus/PlannerArtifactViewer.tsx +158 -0
  685. package/packages/web-dashboard/src/components/focus/PlannerLifecycleHeader.tsx +141 -0
  686. package/packages/web-dashboard/src/components/focus/ProgressSnapshotCard.tsx +113 -0
  687. package/packages/web-dashboard/src/components/focus/RecentMaterialChanges.tsx +69 -0
  688. package/packages/web-dashboard/src/components/focus/RoleLogViewer.tsx +436 -0
  689. package/packages/web-dashboard/src/components/focus/RunHistoryBrowser.tsx +62 -0
  690. package/packages/web-dashboard/src/components/focus/SpecViewer.tsx +172 -0
  691. package/packages/web-dashboard/src/components/focus/TabBar.tsx +33 -0
  692. package/packages/web-dashboard/src/components/focus/UsageBurnChart.tsx +212 -0
  693. package/packages/web-dashboard/src/components/focus/VerificationSummaryCard.tsx +122 -0
  694. package/packages/web-dashboard/src/components/focus/tabs/ChangesTab.tsx +325 -0
  695. package/packages/web-dashboard/src/components/focus/tabs/ConflictsTab.tsx +395 -0
  696. package/packages/web-dashboard/src/components/focus/tabs/GatesQaTab.tsx +38 -0
  697. package/packages/web-dashboard/src/components/focus/tabs/HistoryTab.tsx +213 -0
  698. package/packages/web-dashboard/src/components/focus/tabs/IntakeTab.tsx +429 -0
  699. package/packages/web-dashboard/src/components/focus/tabs/OverviewTab.tsx +217 -0
  700. package/packages/web-dashboard/src/components/focus/tabs/PlanningTab.tsx +390 -0
  701. package/packages/web-dashboard/src/components/focus/tabs/ReviewTab.tsx +497 -0
  702. package/packages/web-dashboard/src/components/focus/tabs/RuntimeTab.tsx +213 -0
  703. package/packages/web-dashboard/src/components/focus/tabs/TranscriptTab.tsx +315 -0
  704. package/packages/web-dashboard/src/components/gate-results.tsx +2 -2
  705. package/packages/web-dashboard/src/components/human-input-panel.tsx +33 -57
  706. package/packages/web-dashboard/src/components/kanban-board.tsx +4 -0
  707. package/packages/web-dashboard/src/components/launch/launch-draft-card.tsx +131 -0
  708. package/packages/web-dashboard/src/components/plan-viewer.tsx +147 -69
  709. package/packages/web-dashboard/src/components/quick-launch-panel.tsx +20 -47
  710. package/packages/web-dashboard/src/components/summary-bar.tsx +30 -76
  711. package/packages/web-dashboard/src/lib/aop-client.ts +2484 -36
  712. package/packages/web-dashboard/src/lib/blocked-state-guidance.ts +475 -0
  713. package/packages/web-dashboard/src/lib/collision-radar.ts +136 -0
  714. package/packages/web-dashboard/src/lib/dashboard-action-states.ts +204 -0
  715. package/packages/web-dashboard/src/lib/dashboard-runtime-client.ts +439 -0
  716. package/packages/web-dashboard/src/lib/dashboard-utils.ts +179 -18
  717. package/packages/web-dashboard/src/lib/drop-zone-utils.ts +92 -0
  718. package/packages/web-dashboard/src/lib/focus-detail-derivations.ts +958 -0
  719. package/packages/web-dashboard/src/lib/focus-view.ts +300 -0
  720. package/packages/web-dashboard/src/lib/health-diagnosis.ts +356 -0
  721. package/packages/web-dashboard/src/lib/launch-contracts.ts +77 -0
  722. package/packages/web-dashboard/src/lib/launch-markdown.ts +107 -0
  723. package/packages/web-dashboard/src/lib/launch-page-preview.ts +89 -0
  724. package/packages/web-dashboard/src/lib/live-feed.ts +1 -1
  725. package/packages/web-dashboard/src/lib/multi-project-config.ts +33 -0
  726. package/packages/web-dashboard/src/lib/orchestrator-tools.ts +845 -59
  727. package/packages/web-dashboard/src/lib/planner-workspace.ts +1285 -0
  728. package/packages/web-dashboard/src/lib/review-contracts.ts +5 -3
  729. package/packages/web-dashboard/src/lib/runtime-files.ts +285 -0
  730. package/packages/web-dashboard/src/lib/tool-catalog.ts +51 -0
  731. package/packages/web-dashboard/src/lib/types.ts +731 -3
  732. package/packages/web-dashboard/src/lib/usage-burn.ts +175 -0
  733. package/packages/web-dashboard/src/lib/worktree-diff.ts +128 -0
  734. package/packages/web-dashboard/src/styles/dashboard.module.css +1742 -459
  735. package/packages/web-dashboard/test/api/actions/route.spec.ts +675 -0
  736. package/packages/web-dashboard/test/api/features/diff.route.spec.ts +57 -0
  737. package/packages/web-dashboard/test/api/features/feature.route.spec.ts +99 -0
  738. package/packages/web-dashboard/test/api/features/live-output.route.spec.ts +123 -0
  739. package/packages/web-dashboard/test/api/features/plan-amend.route.spec.ts +95 -0
  740. package/packages/web-dashboard/test/api/features/planner-workspaces.route.spec.ts +162 -0
  741. package/packages/web-dashboard/test/api/features/question-answer.route.spec.ts +99 -0
  742. package/packages/web-dashboard/test/api/features/triage.route.spec.ts +195 -0
  743. package/packages/web-dashboard/test/api/launch/preview.route.spec.ts +149 -0
  744. package/packages/web-dashboard/test/api/launch/submit.route.spec.ts +382 -0
  745. package/packages/web-dashboard/test/api/runtime/events/route.spec.ts +164 -0
  746. package/packages/web-dashboard/test/api/runtime/operations/route.spec.ts +156 -0
  747. package/packages/web-dashboard/test/api/runtime/runs/route.spec.ts +112 -0
  748. package/packages/web-dashboard/test/components/changes-tab.spec.tsx +76 -0
  749. package/packages/web-dashboard/test/components/command-center-root.spec.tsx +87 -0
  750. package/packages/web-dashboard/test/components/diagnosis-panel.spec.tsx +59 -0
  751. package/packages/web-dashboard/test/components/feature-card.spec.tsx +45 -0
  752. package/packages/web-dashboard/test/components/focus-layout.spec.tsx +299 -0
  753. package/packages/web-dashboard/test/components/gate-results.spec.tsx +39 -0
  754. package/packages/web-dashboard/test/components/gates-qa-tab.spec.tsx +118 -0
  755. package/packages/web-dashboard/test/components/human-input-panel.spec.tsx +54 -0
  756. package/packages/web-dashboard/test/components/intake-tab.spec.tsx +210 -0
  757. package/packages/web-dashboard/test/components/kanban-board.spec.tsx +35 -0
  758. package/packages/web-dashboard/test/components/launch-draft-card.spec.tsx +54 -0
  759. package/packages/web-dashboard/test/components/launch-page.spec.tsx +79 -0
  760. package/packages/web-dashboard/test/components/overview-tab.spec.tsx +236 -0
  761. package/packages/web-dashboard/test/components/planning-tab.spec.tsx +202 -0
  762. package/packages/web-dashboard/test/components/review-tab.spec.tsx +169 -0
  763. package/packages/web-dashboard/test/components/role-log-viewer.spec.ts +42 -0
  764. package/packages/web-dashboard/test/components/runtime-inspector.spec.tsx +22 -0
  765. package/packages/web-dashboard/test/components/runtime-tab.spec.tsx +133 -0
  766. package/packages/web-dashboard/test/components/transcript-tab.spec.tsx +46 -0
  767. package/packages/web-dashboard/test/components/triage-drawer.spec.tsx +159 -0
  768. package/packages/web-dashboard/test/lib/aop-client.spec.ts +235 -0
  769. package/packages/web-dashboard/test/lib/dashboard-runtime-client.spec.ts +144 -0
  770. package/packages/web-dashboard/test/lib/focus-detail-derivations.spec.ts +314 -0
  771. package/packages/web-dashboard/test/lib/focus-view.spec.ts +248 -0
  772. package/packages/web-dashboard/test/lib/health-diagnosis.spec.ts +277 -0
  773. package/packages/web-dashboard/test/lib/launch-markdown.spec.ts +36 -0
  774. package/packages/web-dashboard/test/lib/multi-project-config.spec.ts +54 -0
  775. package/packages/web-dashboard/test/lib/orchestrator-tools.spec.ts +352 -0
  776. package/packages/web-dashboard/test/lib/planner-workspace.spec.ts +289 -0
  777. package/packages/web-dashboard/test/lib/worktree-diff.spec.ts +119 -0
  778. package/packages/web-dashboard/vitest.config.ts +2 -0
  779. package/spec-files/completed/agentic_orchestrator_add_feature_to_active_execution_spec.md +557 -0
  780. package/spec-files/completed/agentic_orchestrator_dashboard_command_center_redesign_spec.md +1147 -0
  781. package/spec-files/completed/agentic_orchestrator_execution_mode_spec.md +18 -16
  782. package/spec-files/completed/agentic_orchestrator_feature_focus_view_track_a_spec.md +672 -0
  783. package/spec-files/completed/agentic_orchestrator_feature_focus_view_track_b_spec.md +794 -0
  784. package/spec-files/completed/agentic_orchestrator_feature_focus_view_track_c_decision_centric_remediation_spec.md +1037 -0
  785. package/spec-files/completed/agentic_orchestrator_feature_focus_view_ux_redesign_spec.md +1432 -0
  786. package/spec-files/completed/agentic_orchestrator_focus_plan_tab_intake_planning_workspace_spec.md +921 -0
  787. package/spec-files/completed/agentic_orchestrator_intentional_collision_override_spec.md +584 -0
  788. package/spec-files/completed/agentic_orchestrator_interactive_planning_intake_and_requirements_verification_spec.md +1185 -0
  789. package/spec-files/completed/agentic_orchestrator_reactive_execution_enrollment_spec.md +864 -0
  790. package/spec-files/{outstanding → completed}/agentic_orchestrator_runtime_inspection_spec.md +92 -19
  791. package/spec-files/completed/agentic_orchestrator_scope_aware_run_lease_spec.md +408 -0
  792. package/spec-files/completed/git-reconciliation-engine.md +827 -0
  793. package/spec-files/outstanding/agentic_orchestrator_dashboard_quick_launch_and_control_surface_spec.md +331 -0
  794. package/spec-files/outstanding/agentic_orchestrator_enterprise_governance_dashboard_spec.md +16 -6
  795. package/spec-files/outstanding/agentic_orchestrator_evidence_integrity_doctor_spec.md +60 -9
  796. package/spec-files/outstanding/agentic_orchestrator_focus_plan_tab_execution_contract_workspace_spec.md +616 -0
  797. package/spec-files/outstanding/agentic_orchestrator_headless_standby_dashboard_runtime_spec.md +310 -0
  798. package/spec-files/outstanding/agentic_orchestrator_human_input_interaction_protocol_spec.md +175 -72
  799. package/spec-files/outstanding/agentic_orchestrator_interactive_rename_cleanup_spec.md +197 -0
  800. package/spec-files/outstanding/agentic_orchestrator_interactive_resume_and_reconciliation_disposition_spec.md +412 -0
  801. package/spec-files/outstanding/agentic_orchestrator_knowledge_canary_spec.md +166 -137
  802. package/spec-files/outstanding/agentic_orchestrator_observability_replay_spec.md +3 -3
  803. package/spec-files/outstanding/agentic_orchestrator_phase_specific_agent_profiles_and_token_telemetry_spec.md +303 -0
  804. package/spec-files/outstanding/agentic_orchestrator_planning_review_quality_spec.md +18 -5
  805. package/spec-files/outstanding/agentic_orchestrator_policy_stratification_spec.md +225 -0
  806. package/spec-files/outstanding/agentic_orchestrator_quality_adoption_execution_spec.md +77 -50
  807. package/spec-files/outstanding/agentic_orchestrator_ready_to_merge_branch_handoff_spec.md +724 -0
  808. package/spec-files/outstanding/agentic_orchestrator_remove_deterministic_mode_spec.md +263 -0
  809. package/spec-files/outstanding/agentic_orchestrator_request_more_context_and_dashboard_human_input_spec.md +456 -0
  810. package/spec-files/outstanding/agentic_orchestrator_spec_coverage_and_reconciliation_enforcement_spec.md +1411 -0
  811. package/spec-files/outstanding/agentic_orchestrator_spec_ordering_agent_spec.md +370 -0
  812. package/spec-files/outstanding/shadow_workspace_implementation_spec.md +1 -1
  813. package/spec-files/progress.md +2045 -87
  814. package/specs/001-runtime-inspection/checklists/requirements.md +35 -0
  815. package/specs/001-runtime-inspection/design.md +338 -0
  816. package/specs/001-runtime-inspection/spec.md +95 -0
  817. package/specs/002-scope-aware-lease/checklists/requirements.md +35 -0
  818. package/specs/002-scope-aware-lease/contracts/lease-registry.schema.json +101 -0
  819. package/specs/002-scope-aware-lease/data-model.md +236 -0
  820. package/specs/002-scope-aware-lease/plan.md +766 -0
  821. package/specs/002-scope-aware-lease/quickstart.md +150 -0
  822. package/specs/002-scope-aware-lease/research.md +135 -0
  823. package/specs/002-scope-aware-lease/spec.md +128 -0
  824. package/specs/002-scope-aware-lease/tasks.md +767 -0
  825. package/tsconfig.json +1 -1
  826. package/vitest.config.ts +28 -0
  827. package/ARCHITECTURE_ADHERENCE_ANALYSIS.md +0 -871
  828. package/packages/web-dashboard/next-env.d.ts +0 -6
  829. package/packages/web-dashboard/src/components/detail-panel.tsx +0 -1124
  830. package/packages/web-dashboard/src/components/review-workspace.tsx +0 -1162
  831. /package/spec-files/{outstanding → completed}/agentic_orchestrator_artifact_database_publishing_spec.md +0 -0
  832. /package/spec-files/{outstanding → completed}/agentic_orchestrator_cli_shell_tab_completion_spec.md +0 -0
  833. /package/spec-files/{outstanding → completed}/agentic_orchestrator_dashboard_diff_and_agent_console_spec.md +0 -0
  834. /package/spec-files/{outstanding → completed}/agentic_orchestrator_performance_improvements_spec.md +0 -0
  835. /package/spec-files/{outstanding → completed}/agentic_orchestrator_persistent_worker_runtime_spec.md +0 -0
  836. /package/spec-files/{outstanding → completed}/agentic_orchestrator_provider_auth_bootstrap_spec.md +0 -0
  837. /package/spec-files/{outstanding → completed}/agentic_orchestrator_real_worker_provider_execution_spec.md +0 -0
@@ -0,0 +1,1591 @@
1
+ import path from 'node:path';
2
+ import fs from 'node:fs/promises';
3
+ import { ensureDir, nowIso } from '../../core/fs.js';
4
+ import { runGit } from '../../core/git.js';
5
+ import { STATUS } from '../../core/constants.js';
6
+ import { ERROR_CODES } from '../../core/error-codes.js';
7
+ import { ok, fail, type ToolResponse } from '../../core/response.js';
8
+ import type { RuntimeSessionsSnapshot } from '../../core/runtime-sessions.js';
9
+
10
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
11
+ type AnyRecord = Record<string, any>;
12
+
13
+ type MainlineConflictEntry = AnyRecord & {
14
+ type: 'mainline_divergence';
15
+ merge_base: string;
16
+ mainline_head: string;
17
+ worktree_head: string;
18
+ conflicting_files: string[];
19
+ resolution_status: 'pending' | 'in_progress' | 'resolved';
20
+ };
21
+
22
+ interface ReconciliationControlProvider {
23
+ sendMessage?(sessionId: string, message: string): Promise<void>;
24
+ getSessionInfo?(sessionId: string): Promise<{ active: boolean; provider: string }>;
25
+ }
26
+
27
+ function readFrontMatterVersion(frontMatter: AnyRecord): number {
28
+ const raw = frontMatter.version;
29
+ return typeof raw === 'number' && Number.isFinite(raw) ? raw : 0;
30
+ }
31
+
32
+ type MaterializedConflictResult =
33
+ | { kind: 'merged'; strategy: 'merge_commit' }
34
+ | { kind: 'conflict'; conflictingFiles: string[] }
35
+ | { kind: 'error'; response: ToolResponse<ReconciliationResult> };
36
+
37
+ type ReconcilerLaneClaim = {
38
+ activeFeatureId: string | null;
39
+ sessionId: string | null;
40
+ queuePosition: number | null;
41
+ dispatchNow: boolean;
42
+ };
43
+
44
+ type ReconcilerLaneRelease = {
45
+ nextFeatureId: string | null;
46
+ sessionId: string | null;
47
+ };
48
+
49
+ type CollisionOverride = {
50
+ override_id: string;
51
+ feature_ids: [string, string];
52
+ rationale: string;
53
+ approved_paths: string[];
54
+ approved_collision_fingerprint: string;
55
+ status: 'active' | 'stale' | 'revoked' | 'applied';
56
+ source_collision_id: string;
57
+ applied_at?: string;
58
+ };
59
+
60
+ type CollisionRecord = {
61
+ collision_id: string;
62
+ detected_at: string;
63
+ [key: string]: unknown;
64
+ };
65
+
66
+ type ApprovedCollisionContext = {
67
+ override_id: string;
68
+ counterpart_feature_id: string;
69
+ rationale: string;
70
+ approved_paths: string[];
71
+ approved_collision_fingerprint: string;
72
+ source_collision_id: string;
73
+ detected_at: string;
74
+ applied_at?: string;
75
+ source_collision_record: CollisionRecord;
76
+ };
77
+
78
+ /**
79
+ * Result of a remote/local mainline divergence check.
80
+ */
81
+ export type RemoteLocalDivergenceResult = {
82
+ diverged: boolean;
83
+ local_ahead: boolean;
84
+ local_head: string;
85
+ remote_head: string;
86
+ };
87
+
88
+ /**
89
+ * Result of a mainline reconciliation operation.
90
+ */
91
+ export type ReconciliationResult = {
92
+ status: 'up_to_date' | 'merged' | 'conflict_detected';
93
+ divergence_detected: boolean;
94
+ files_changed_mainline: string[];
95
+ files_changed_worktree: string[];
96
+ conflicting_files: string[];
97
+ merge_base: string;
98
+ mainline_head: string;
99
+ worktree_head: string;
100
+ reconciler_active_feature_id?: string | null;
101
+ reconciler_queue_position?: number | null;
102
+ dispatched_to_reconciler?: boolean;
103
+ conflict_worktree_prepared?: boolean;
104
+ };
105
+
106
+ /**
107
+ * Port interface for the reconciliation service dependencies.
108
+ */
109
+ export interface GitReconciliationServicePort {
110
+ getRepoRoot(): string;
111
+ worktreePath(featureId: string): string;
112
+ readState(featureId: string): Promise<{ frontMatter: AnyRecord; body: string }>;
113
+ writeState(featureId: string, frontMatter: AnyRecord, body?: string): Promise<void>;
114
+ withFeatureLock<T>(featureId: string, operation: () => Promise<T>): Promise<T>;
115
+ readIndex?(): Promise<AnyRecord>;
116
+ evidencePath(featureId: string): string;
117
+ logsPath(featureId: string): string;
118
+ withIndexLock<T>(operation: () => Promise<T>): Promise<T>;
119
+ readRunLease(): Promise<RuntimeSessionsSnapshot>;
120
+ writeRunLease(data: RuntimeSessionsSnapshot): Promise<void>;
121
+ normalizeRuntimeSessions(value: unknown, at?: string): RuntimeSessionsSnapshot;
122
+ getProvider(): ReconciliationControlProvider | null;
123
+ refreshReconcilerSession?(): Promise<string | null>;
124
+ }
125
+
126
+ function parseDiffNameOnly(stdout: string): string[] {
127
+ return stdout
128
+ .split('\n')
129
+ .map((line) => line.trim())
130
+ .filter((line) => line.length > 0);
131
+ }
132
+
133
+ function parseStatusPaths(statusPorcelain: string): string[] {
134
+ const files: string[] = [];
135
+ const lines = statusPorcelain
136
+ .split('\n')
137
+ .map((line) => line.trimEnd())
138
+ .filter((line) => line.length >= 4);
139
+
140
+ for (const line of lines) {
141
+ const candidate = line.slice(3).trim();
142
+ if (!candidate) {
143
+ continue;
144
+ }
145
+ const renameMarker = candidate.indexOf(' -> ');
146
+ const pathValue = renameMarker >= 0 ? candidate.slice(renameMarker + 4) : candidate;
147
+ const normalized = pathValue.replaceAll('\\\\', '/').trim();
148
+ if (normalized.length > 0) {
149
+ files.push(normalized);
150
+ }
151
+ }
152
+
153
+ return [...new Set(files)].sort((a, b) => a.localeCompare(b));
154
+ }
155
+
156
+ function isIgnoredReconciliationStatusPath(filePath: string): boolean {
157
+ return (
158
+ filePath.startsWith('.aop/') ||
159
+ filePath.startsWith('.worktrees/') ||
160
+ filePath === 'agentic/orchestrator/tools.md'
161
+ );
162
+ }
163
+
164
+ function intersect(a: string[], b: string[]): string[] {
165
+ const setB = new Set(b);
166
+ return a.filter((f) => setB.has(f));
167
+ }
168
+
169
+ function asStringArray(value: unknown): string[] {
170
+ if (!Array.isArray(value)) {
171
+ return [];
172
+ }
173
+ return value.filter((item): item is string => typeof item === 'string' && item.length > 0);
174
+ }
175
+
176
+ function normalizeQueue(queue: string[]): string[] {
177
+ const seen = new Set<string>();
178
+ const normalized: string[] = [];
179
+ for (const featureId of queue) {
180
+ if (seen.has(featureId)) {
181
+ continue;
182
+ }
183
+ seen.add(featureId);
184
+ normalized.push(featureId);
185
+ }
186
+ return normalized;
187
+ }
188
+
189
+ function asCollisionOverrides(value: unknown): CollisionOverride[] {
190
+ if (!Array.isArray(value)) {
191
+ return [];
192
+ }
193
+ return value.filter(
194
+ (entry): entry is CollisionOverride => entry !== null && typeof entry === 'object',
195
+ );
196
+ }
197
+
198
+ function asCollisionRecords(value: unknown): CollisionRecord[] {
199
+ if (!Array.isArray(value)) {
200
+ return [];
201
+ }
202
+ return value.filter(
203
+ (entry): entry is CollisionRecord => entry !== null && typeof entry === 'object',
204
+ );
205
+ }
206
+
207
+ function defined<T>(value: T | null): value is T {
208
+ return value !== null;
209
+ }
210
+
211
+ function findLatestUnresolvedMainlineConflict(frontMatter: AnyRecord): {
212
+ index: number;
213
+ entry: MainlineConflictEntry | null;
214
+ } {
215
+ const conflicts = Array.isArray(frontMatter.conflicts) ? frontMatter.conflicts : [];
216
+ for (let index = conflicts.length - 1; index >= 0; index -= 1) {
217
+ const entry = conflicts[index];
218
+ if (!entry || typeof entry !== 'object' || entry.type !== 'mainline_divergence') {
219
+ continue;
220
+ }
221
+ const typed = entry as MainlineConflictEntry;
222
+ if (typed.resolution_status === 'resolved') {
223
+ continue;
224
+ }
225
+ return { index, entry: typed };
226
+ }
227
+ return { index: -1, entry: null };
228
+ }
229
+
230
+ function readConflictInfo(conflict: MainlineConflictEntry | null): {
231
+ mergeBase: string;
232
+ mainlineHead: string;
233
+ worktreeHead: string;
234
+ conflictingFiles: string[];
235
+ } {
236
+ return {
237
+ mergeBase: typeof conflict?.merge_base === 'string' ? conflict.merge_base : 'unknown',
238
+ mainlineHead: typeof conflict?.mainline_head === 'string' ? conflict.mainline_head : 'unknown',
239
+ worktreeHead: typeof conflict?.worktree_head === 'string' ? conflict.worktree_head : 'unknown',
240
+ conflictingFiles: asStringArray(conflict?.conflicting_files),
241
+ };
242
+ }
243
+
244
+ function buildQueueStatusReason(
245
+ activeFeatureId: string | null,
246
+ queuePosition: number | null,
247
+ ): string {
248
+ if (queuePosition === null || queuePosition <= 0) {
249
+ return 'Mainline divergence assigned to the shared reconciler lane';
250
+ }
251
+ const ordinal = queuePosition + 1;
252
+ const active = activeFeatureId ? ` behind ${activeFeatureId}` : '';
253
+ return `Queued for shared reconciler lane at position ${ordinal}${active}`;
254
+ }
255
+
256
+ function errorMessage(error: unknown): string {
257
+ if (error instanceof Error && typeof error.message === 'string' && error.message.length > 0) {
258
+ const details =
259
+ 'details' in error && error.details && typeof error.details === 'object'
260
+ ? (error.details as Record<string, unknown>)
261
+ : null;
262
+ const stderr =
263
+ details && typeof details.stderr === 'string' && details.stderr.trim().length > 0
264
+ ? details.stderr.trim()
265
+ : null;
266
+ if (stderr) {
267
+ return `${error.message}: ${stderr}`;
268
+ }
269
+ return error.message;
270
+ }
271
+ if (typeof error === 'string' && error.length > 0) {
272
+ return error;
273
+ }
274
+ return 'unknown reconciler dispatch failure';
275
+ }
276
+
277
+ function isMissingReconcilerConversation(error: unknown): boolean {
278
+ if (error instanceof Error && error.message.includes('No conversation found with session ID')) {
279
+ return true;
280
+ }
281
+ if (typeof error === 'string') {
282
+ return error.includes('No conversation found with session ID');
283
+ }
284
+ const details =
285
+ error &&
286
+ typeof error === 'object' &&
287
+ 'details' in error &&
288
+ error.details &&
289
+ typeof error.details === 'object'
290
+ ? (error.details as Record<string, unknown>)
291
+ : null;
292
+ const stderr = details && typeof details.stderr === 'string' ? details.stderr : '';
293
+ return stderr.includes('No conversation found with session ID');
294
+ }
295
+
296
+ /**
297
+ * Service that reconciles feature worktrees with the mainline branch.
298
+ */
299
+ export class GitReconciliationService {
300
+ private readonly port: GitReconciliationServicePort;
301
+
302
+ constructor(port: GitReconciliationServicePort) {
303
+ this.port = port;
304
+ }
305
+
306
+ async checkRemoteLocalDivergence(
307
+ baseRef: string,
308
+ ): Promise<ToolResponse<RemoteLocalDivergenceResult>> {
309
+ const repoRoot = this.port.getRepoRoot();
310
+
311
+ await runGit(repoRoot, ['fetch', 'origin', baseRef]);
312
+
313
+ const localResult = await runGit(repoRoot, ['rev-parse', baseRef]);
314
+ const remoteResult = await runGit(repoRoot, ['rev-parse', `origin/${baseRef}`]);
315
+
316
+ if (localResult.code !== 0) {
317
+ return fail(ERROR_CODES.GIT_FAILURE, `Cannot resolve local ref ${baseRef}`, {
318
+ retryable: false,
319
+ requires_human: true,
320
+ });
321
+ }
322
+
323
+ const localHead = localResult.stdout.trim();
324
+
325
+ // Preserve fallback: if remote ref fails to resolve, treat as no divergence
326
+ if (remoteResult.code !== 0) {
327
+ return ok<RemoteLocalDivergenceResult>({
328
+ diverged: false,
329
+ local_ahead: false,
330
+ local_head: localHead,
331
+ remote_head: localHead,
332
+ });
333
+ }
334
+
335
+ const remoteHead = remoteResult.stdout.trim();
336
+
337
+ // Identical SHAs — short-circuit without merge-base call
338
+ if (localHead === remoteHead) {
339
+ return ok<RemoteLocalDivergenceResult>({
340
+ diverged: false,
341
+ local_ahead: false,
342
+ local_head: localHead,
343
+ remote_head: remoteHead,
344
+ });
345
+ }
346
+
347
+ // Check if remote is an ancestor of local (local is ahead)
348
+ const ancestorResult = await runGit(repoRoot, [
349
+ 'merge-base',
350
+ '--is-ancestor',
351
+ `origin/${baseRef}`,
352
+ baseRef,
353
+ ]);
354
+
355
+ if (ancestorResult.code === 0) {
356
+ // Remote is ancestor of local — local is ahead, no real divergence
357
+ return ok<RemoteLocalDivergenceResult>({
358
+ diverged: false,
359
+ local_ahead: true,
360
+ local_head: localHead,
361
+ remote_head: remoteHead,
362
+ });
363
+ }
364
+
365
+ // True divergence — remote has commits local doesn't
366
+ return ok<RemoteLocalDivergenceResult>({
367
+ diverged: true,
368
+ local_ahead: false,
369
+ local_head: localHead,
370
+ remote_head: remoteHead,
371
+ });
372
+ }
373
+
374
+ async syncMainlineFromRemote(
375
+ baseRef: string,
376
+ ): Promise<ToolResponse<{ merged: boolean; sha: string }>> {
377
+ const repoRoot = this.port.getRepoRoot();
378
+
379
+ const fetchResult = await runGit(repoRoot, ['fetch', 'origin', baseRef]);
380
+ if (fetchResult.code !== 0) {
381
+ return fail(ERROR_CODES.GIT_FAILURE, `Failed to fetch origin/${baseRef}`, {
382
+ stderr: fetchResult.stderr,
383
+ retryable: true,
384
+ requires_human: false,
385
+ });
386
+ }
387
+
388
+ const checkoutResult = await runGit(repoRoot, ['checkout', baseRef]);
389
+ if (checkoutResult.code !== 0) {
390
+ return fail(ERROR_CODES.GIT_FAILURE, `Failed to checkout ${baseRef} for remote sync`, {
391
+ stderr: checkoutResult.stderr,
392
+ retryable: false,
393
+ requires_human: true,
394
+ });
395
+ }
396
+
397
+ const mergeResult = await runGit(repoRoot, ['merge', '-X', 'ours', `origin/${baseRef}`]);
398
+ if (mergeResult.code !== 0) {
399
+ await runGit(repoRoot, ['merge', '--abort']);
400
+ return fail(
401
+ ERROR_CODES.RECONCILIATION_FAILED,
402
+ `Failed to sync origin/${baseRef} into local ${baseRef}`,
403
+ {
404
+ stderr: mergeResult.stderr,
405
+ retryable: false,
406
+ requires_human: true,
407
+ },
408
+ );
409
+ }
410
+
411
+ const shaResult = await runGit(repoRoot, ['rev-parse', 'HEAD']);
412
+ return ok({ merged: true, sha: shaResult.stdout.trim() });
413
+ }
414
+
415
+ async reconcileWithMainline(
416
+ featureId: string,
417
+ baseRef: string,
418
+ checkpointCommitMessage: string | null = null,
419
+ ): Promise<ToolResponse<ReconciliationResult>> {
420
+ const repoRoot = this.port.getRepoRoot();
421
+ const worktreePath = this.port.worktreePath(featureId);
422
+
423
+ const existingConflictResult = await this.reuseExistingConflictIfPresent(
424
+ featureId,
425
+ worktreePath,
426
+ baseRef,
427
+ );
428
+ if (existingConflictResult) {
429
+ return existingConflictResult;
430
+ }
431
+
432
+ const fetchResult = await runGit(worktreePath, ['fetch', 'origin', baseRef]);
433
+ if (fetchResult.code !== 0) {
434
+ // Non-fatal: continue with local refs
435
+ }
436
+
437
+ const localMergeBase = await runGit(worktreePath, ['merge-base', baseRef, 'HEAD']);
438
+ if (localMergeBase.code !== 0) {
439
+ return fail(
440
+ ERROR_CODES.MERGE_BASE_NOT_FOUND,
441
+ `Cannot find merge base between ${baseRef} and HEAD`,
442
+ { featureId, baseRef, retryable: false, requires_human: true },
443
+ );
444
+ }
445
+ const mergeBase = localMergeBase.stdout.trim();
446
+
447
+ if (!mergeBase) {
448
+ return fail(ERROR_CODES.MERGE_BASE_NOT_FOUND, `Empty merge base for ${baseRef}`, {
449
+ featureId,
450
+ baseRef,
451
+ retryable: false,
452
+ requires_human: true,
453
+ });
454
+ }
455
+
456
+ const checkpointResult = await this.checkpointFeatureWorktreeIfRequested(
457
+ featureId,
458
+ worktreePath,
459
+ checkpointCommitMessage,
460
+ );
461
+ if (!checkpointResult.ok) {
462
+ return checkpointResult as ToolResponse<ReconciliationResult>;
463
+ }
464
+
465
+ const localMainlineResult = await runGit(repoRoot, ['rev-parse', baseRef]);
466
+ const originMainlineResult = await runGit(worktreePath, ['rev-parse', `origin/${baseRef}`]);
467
+ const mainlineHead =
468
+ localMainlineResult.code === 0
469
+ ? localMainlineResult.stdout.trim()
470
+ : originMainlineResult.stdout.trim();
471
+
472
+ const worktreeHeadResult = await runGit(worktreePath, ['rev-parse', 'HEAD']);
473
+ const worktreeHead = worktreeHeadResult.stdout.trim();
474
+
475
+ if (mainlineHead === mergeBase || mainlineHead === worktreeHead) {
476
+ await this.updateReconciliationState(featureId, mainlineHead);
477
+ await this.resolveMainlineConflictState(featureId, mainlineHead);
478
+ await this.releaseAndDispatchNext(featureId, baseRef);
479
+ return ok<ReconciliationResult>({
480
+ status: 'up_to_date',
481
+ divergence_detected: false,
482
+ files_changed_mainline: [],
483
+ files_changed_worktree: [],
484
+ conflicting_files: [],
485
+ merge_base: mergeBase,
486
+ mainline_head: mainlineHead,
487
+ worktree_head: worktreeHead,
488
+ });
489
+ }
490
+
491
+ const mainlineDiffResult = await runGit(worktreePath, [
492
+ 'diff',
493
+ '--name-only',
494
+ mergeBase,
495
+ mainlineHead,
496
+ ]);
497
+ const filesChangedMainline = parseDiffNameOnly(mainlineDiffResult.stdout);
498
+
499
+ const worktreeDiffResult = await runGit(worktreePath, [
500
+ 'diff',
501
+ '--name-only',
502
+ mergeBase,
503
+ worktreeHead,
504
+ ]);
505
+ const filesChangedWorktree = parseDiffNameOnly(worktreeDiffResult.stdout);
506
+
507
+ const overlappingFiles = intersect(filesChangedMainline, filesChangedWorktree);
508
+
509
+ await this.writeEvidenceArtifacts(
510
+ featureId,
511
+ worktreePath,
512
+ mergeBase,
513
+ mainlineHead,
514
+ worktreeHead,
515
+ overlappingFiles,
516
+ );
517
+
518
+ if (overlappingFiles.length > 0) {
519
+ const materialized = await this.materializeConflictWorktree(
520
+ featureId,
521
+ worktreePath,
522
+ baseRef,
523
+ mainlineHead,
524
+ );
525
+ if (materialized.kind === 'error') {
526
+ return materialized.response;
527
+ }
528
+
529
+ if (materialized.kind === 'conflict') {
530
+ const claim = await this.claimReconcilerLane(featureId);
531
+ const queuePosition = claim.activeFeatureId === featureId ? 0 : (claim.queuePosition ?? 1);
532
+ await this.upsertMainlineConflictState(
533
+ featureId,
534
+ {
535
+ mergeBase,
536
+ mainlineHead,
537
+ worktreeHead,
538
+ conflictingFiles: materialized.conflictingFiles,
539
+ },
540
+ {
541
+ baseRef,
542
+ resolutionStatus: claim.activeFeatureId === featureId ? 'in_progress' : 'pending',
543
+ queuePosition,
544
+ statusReason: buildQueueStatusReason(claim.activeFeatureId, queuePosition),
545
+ },
546
+ );
547
+ const dispatched =
548
+ claim.activeFeatureId === featureId
549
+ ? await this.dispatchReconcilerTurn(featureId, baseRef, claim.sessionId, {
550
+ mergeBase,
551
+ mainlineHead,
552
+ worktreeHead,
553
+ conflictingFiles: materialized.conflictingFiles,
554
+ })
555
+ : false;
556
+ await this.appendReconciliationLog(
557
+ featureId,
558
+ dispatched
559
+ ? `queued_for_reconciler: active lane assigned to ${featureId}`
560
+ : `queued_for_reconciler: waiting on ${claim.activeFeatureId ?? 'unknown'}`,
561
+ );
562
+ return ok<ReconciliationResult>({
563
+ status: 'conflict_detected',
564
+ divergence_detected: true,
565
+ files_changed_mainline: filesChangedMainline,
566
+ files_changed_worktree: filesChangedWorktree,
567
+ conflicting_files: materialized.conflictingFiles,
568
+ merge_base: mergeBase,
569
+ mainline_head: mainlineHead,
570
+ worktree_head: worktreeHead,
571
+ reconciler_active_feature_id: claim.activeFeatureId,
572
+ reconciler_queue_position: queuePosition,
573
+ dispatched_to_reconciler: dispatched,
574
+ conflict_worktree_prepared: true,
575
+ });
576
+ }
577
+
578
+ await this.updateReconciliationState(featureId, mainlineHead);
579
+ await this.resolveMainlineConflictState(featureId, mainlineHead);
580
+ await this.releaseAndDispatchNext(featureId, baseRef);
581
+ await this.appendReconciliationLog(
582
+ featureId,
583
+ `merged: ${mainlineHead} (materialized merge commit)`,
584
+ );
585
+ return ok<ReconciliationResult>({
586
+ status: 'merged',
587
+ divergence_detected: true,
588
+ files_changed_mainline: filesChangedMainline,
589
+ files_changed_worktree: filesChangedWorktree,
590
+ conflicting_files: [],
591
+ merge_base: mergeBase,
592
+ mainline_head: mainlineHead,
593
+ worktree_head: worktreeHead,
594
+ });
595
+ }
596
+
597
+ const mergeResult = await this.performAutoMerge(featureId, worktreePath, baseRef, mainlineHead);
598
+ if (!mergeResult.ok) {
599
+ return mergeResult as ToolResponse<ReconciliationResult>;
600
+ }
601
+
602
+ await this.updateReconciliationState(featureId, mainlineHead);
603
+ await this.resolveMainlineConflictState(featureId, mainlineHead);
604
+ await this.releaseAndDispatchNext(featureId, baseRef);
605
+ await this.appendReconciliationLog(featureId, `merged: ${mainlineHead}`);
606
+
607
+ return ok<ReconciliationResult>({
608
+ status: 'merged',
609
+ divergence_detected: true,
610
+ files_changed_mainline: filesChangedMainline,
611
+ files_changed_worktree: filesChangedWorktree,
612
+ conflicting_files: [],
613
+ merge_base: mergeBase,
614
+ mainline_head: mainlineHead,
615
+ worktree_head: worktreeHead,
616
+ });
617
+ }
618
+
619
+ async detectDivergence(
620
+ featureId: string,
621
+ baseRef: string,
622
+ ): Promise<ToolResponse<ReconciliationResult>> {
623
+ const worktreePath = this.port.worktreePath(featureId);
624
+
625
+ await runGit(worktreePath, ['fetch', 'origin', baseRef]);
626
+
627
+ const localBase = await runGit(worktreePath, ['merge-base', baseRef, 'HEAD']);
628
+ if (localBase.code !== 0) {
629
+ return fail(
630
+ ERROR_CODES.MERGE_BASE_NOT_FOUND,
631
+ `Cannot find merge base between ${baseRef} and HEAD`,
632
+ { featureId, baseRef, retryable: false, requires_human: true },
633
+ );
634
+ }
635
+ const mergeBase = localBase.stdout.trim();
636
+
637
+ const localMainlineResult = await runGit(this.port.getRepoRoot(), ['rev-parse', baseRef]);
638
+ const originMainlineResult = await runGit(worktreePath, ['rev-parse', `origin/${baseRef}`]);
639
+ const mainlineHead =
640
+ localMainlineResult.code === 0
641
+ ? localMainlineResult.stdout.trim()
642
+ : originMainlineResult.stdout.trim();
643
+
644
+ const worktreeHead = (await runGit(worktreePath, ['rev-parse', 'HEAD'])).stdout.trim();
645
+
646
+ if (mainlineHead === mergeBase || mainlineHead === worktreeHead) {
647
+ return ok<ReconciliationResult>({
648
+ status: 'up_to_date',
649
+ divergence_detected: false,
650
+ files_changed_mainline: [],
651
+ files_changed_worktree: [],
652
+ conflicting_files: [],
653
+ merge_base: mergeBase,
654
+ mainline_head: mainlineHead,
655
+ worktree_head: worktreeHead,
656
+ });
657
+ }
658
+
659
+ const mainlineDiff = await runGit(worktreePath, [
660
+ 'diff',
661
+ '--name-only',
662
+ mergeBase,
663
+ mainlineHead,
664
+ ]);
665
+ const worktreeDiff = await runGit(worktreePath, [
666
+ 'diff',
667
+ '--name-only',
668
+ mergeBase,
669
+ worktreeHead,
670
+ ]);
671
+
672
+ const filesChangedMainline = parseDiffNameOnly(mainlineDiff.stdout);
673
+ const filesChangedWorktree = parseDiffNameOnly(worktreeDiff.stdout);
674
+ const conflictingFiles = intersect(filesChangedMainline, filesChangedWorktree);
675
+
676
+ const status = conflictingFiles.length > 0 ? 'conflict_detected' : 'merged';
677
+
678
+ return ok<ReconciliationResult>({
679
+ status,
680
+ divergence_detected: true,
681
+ files_changed_mainline: filesChangedMainline,
682
+ files_changed_worktree: filesChangedWorktree,
683
+ conflicting_files: conflictingFiles,
684
+ merge_base: mergeBase,
685
+ mainline_head: mainlineHead,
686
+ worktree_head: worktreeHead,
687
+ });
688
+ }
689
+
690
+ async redrivePreparedReconcilerConflict(featureId: string): Promise<boolean> {
691
+ const state = await this.port.readState(featureId).catch(() => null);
692
+ if (!state) {
693
+ return false;
694
+ }
695
+
696
+ const unresolved = findLatestUnresolvedMainlineConflict(state.frontMatter);
697
+ if (!unresolved.entry) {
698
+ return false;
699
+ }
700
+
701
+ const conflictInfo = readConflictInfo(unresolved.entry);
702
+ const conflictingFiles = await this.listUnmergedFiles(this.port.worktreePath(featureId));
703
+ const normalizedFiles =
704
+ conflictingFiles.length > 0 ? conflictingFiles : conflictInfo.conflictingFiles;
705
+ if (normalizedFiles.length === 0) {
706
+ return false;
707
+ }
708
+
709
+ let runtimeSessions: RuntimeSessionsSnapshot;
710
+ try {
711
+ runtimeSessions = this.port.normalizeRuntimeSessions(await this.port.readRunLease());
712
+ } catch {
713
+ return false;
714
+ }
715
+
716
+ let sessionId =
717
+ typeof runtimeSessions.reconciler_session_id === 'string' &&
718
+ runtimeSessions.reconciler_session_id.length > 0
719
+ ? runtimeSessions.reconciler_session_id
720
+ : null;
721
+ const activeFeatureId =
722
+ typeof runtimeSessions.reconciler_active_feature_id === 'string' &&
723
+ runtimeSessions.reconciler_active_feature_id.length > 0
724
+ ? runtimeSessions.reconciler_active_feature_id
725
+ : null;
726
+
727
+ if (activeFeatureId !== featureId || sessionId === null) {
728
+ const claim = await this.claimReconcilerLane(featureId);
729
+ if (claim.activeFeatureId !== featureId) {
730
+ return false;
731
+ }
732
+ sessionId = claim.sessionId;
733
+ }
734
+
735
+ const baseRef =
736
+ typeof unresolved.entry.base_ref === 'string' && unresolved.entry.base_ref.length > 0
737
+ ? unresolved.entry.base_ref
738
+ : 'main';
739
+
740
+ await this.upsertMainlineConflictState(
741
+ featureId,
742
+ {
743
+ mergeBase: conflictInfo.mergeBase,
744
+ mainlineHead: conflictInfo.mainlineHead,
745
+ worktreeHead: conflictInfo.worktreeHead,
746
+ conflictingFiles: normalizedFiles,
747
+ },
748
+ {
749
+ baseRef,
750
+ resolutionStatus: 'in_progress',
751
+ queuePosition: 0,
752
+ statusReason: buildQueueStatusReason(featureId, 0),
753
+ },
754
+ );
755
+
756
+ return await this.dispatchReconcilerTurn(featureId, baseRef, sessionId, {
757
+ mergeBase: conflictInfo.mergeBase,
758
+ mainlineHead: conflictInfo.mainlineHead,
759
+ worktreeHead: conflictInfo.worktreeHead,
760
+ conflictingFiles: normalizedFiles,
761
+ });
762
+ }
763
+
764
+ private async performAutoMerge(
765
+ featureId: string,
766
+ worktreePath: string,
767
+ baseRef: string,
768
+ mainlineHead: string,
769
+ ): Promise<ToolResponse> {
770
+ const statusResult = await runGit(worktreePath, [
771
+ 'status',
772
+ '--porcelain',
773
+ '--untracked-files=all',
774
+ ]);
775
+ if (statusResult.code !== 0) {
776
+ return fail(ERROR_CODES.GIT_FAILURE, 'Failed to inspect feature worktree status', {
777
+ featureId,
778
+ baseRef,
779
+ worktree_path: worktreePath,
780
+ stderr: statusResult.stderr,
781
+ retryable: false,
782
+ requires_human: true,
783
+ });
784
+ }
785
+
786
+ const dirtyFiles = parseStatusPaths(statusResult.stdout).filter(
787
+ (filePath) => !isIgnoredReconciliationStatusPath(filePath),
788
+ );
789
+ if (dirtyFiles.length > 0) {
790
+ return fail(
791
+ ERROR_CODES.UNCOMMITTED_CHANGES,
792
+ 'Feature worktree has uncommitted changes - commit or stash them before reconciliation',
793
+ {
794
+ featureId,
795
+ baseRef,
796
+ worktree_path: worktreePath,
797
+ files: dirtyFiles,
798
+ retryable: false,
799
+ requires_human: true,
800
+ },
801
+ );
802
+ }
803
+
804
+ const ffResult = await runGit(worktreePath, ['merge', '--ff-only', `origin/${baseRef}`]);
805
+ if (ffResult.code === 0) {
806
+ return ok({ merged: true, strategy: 'fast_forward', mainline_head: mainlineHead });
807
+ }
808
+
809
+ const rebaseResult = await runGit(worktreePath, ['rebase', `origin/${baseRef}`]);
810
+ if (rebaseResult.code === 0) {
811
+ return ok({ merged: true, strategy: 'rebase', mainline_head: mainlineHead });
812
+ }
813
+
814
+ await runGit(worktreePath, ['rebase', '--abort']);
815
+ return fail(
816
+ ERROR_CODES.RECONCILIATION_FAILED,
817
+ 'Auto-merge failed: fast-forward and rebase both failed',
818
+ {
819
+ retryable: false,
820
+ requires_human: true,
821
+ ff_stderr: ffResult.stderr,
822
+ rebase_stderr: rebaseResult.stderr,
823
+ },
824
+ );
825
+ }
826
+
827
+ private async checkpointFeatureWorktreeIfRequested(
828
+ featureId: string,
829
+ worktreePath: string,
830
+ commitMessage: string | null,
831
+ ): Promise<ToolResponse<{ committed: boolean; files: string[] }>> {
832
+ const normalizedMessage = typeof commitMessage === 'string' ? commitMessage.trim() : '';
833
+ if (normalizedMessage.length === 0) {
834
+ return ok({ committed: false, files: [] });
835
+ }
836
+
837
+ const statusResult = await runGit(worktreePath, [
838
+ 'status',
839
+ '--porcelain',
840
+ '--untracked-files=all',
841
+ ]);
842
+ if (statusResult.code !== 0) {
843
+ return fail(
844
+ ERROR_CODES.GIT_FAILURE,
845
+ 'Failed to inspect feature worktree before reconciliation',
846
+ {
847
+ featureId,
848
+ worktree_path: worktreePath,
849
+ stderr: statusResult.stderr,
850
+ retryable: false,
851
+ requires_human: true,
852
+ },
853
+ );
854
+ }
855
+
856
+ const dirtyFiles = parseStatusPaths(statusResult.stdout).filter(
857
+ (filePath) => !isIgnoredReconciliationStatusPath(filePath),
858
+ );
859
+ if (dirtyFiles.length === 0) {
860
+ return ok({ committed: false, files: [] });
861
+ }
862
+
863
+ const stageResult = await runGit(worktreePath, ['add', '-A']);
864
+ if (stageResult.code !== 0) {
865
+ return fail(
866
+ ERROR_CODES.GIT_FAILURE,
867
+ 'Failed to stage dirty feature worktree before reconciliation',
868
+ {
869
+ featureId,
870
+ worktree_path: worktreePath,
871
+ files: dirtyFiles,
872
+ stderr: stageResult.stderr,
873
+ retryable: false,
874
+ requires_human: true,
875
+ },
876
+ );
877
+ }
878
+
879
+ const commitResult = await runGit(worktreePath, ['commit', '-m', normalizedMessage]);
880
+ if (
881
+ commitResult.code !== 0 &&
882
+ !commitResult.stderr.includes('nothing to commit') &&
883
+ !commitResult.stdout.includes('nothing to commit')
884
+ ) {
885
+ return fail(
886
+ ERROR_CODES.GIT_FAILURE,
887
+ 'Failed to checkpoint feature worktree before reconciliation',
888
+ {
889
+ featureId,
890
+ worktree_path: worktreePath,
891
+ files: dirtyFiles,
892
+ stderr: commitResult.stderr || commitResult.stdout,
893
+ retryable: false,
894
+ requires_human: true,
895
+ },
896
+ );
897
+ }
898
+
899
+ return ok({ committed: true, files: dirtyFiles });
900
+ }
901
+
902
+ private async reuseExistingConflictIfPresent(
903
+ featureId: string,
904
+ worktreePath: string,
905
+ baseRef: string,
906
+ ): Promise<ToolResponse<ReconciliationResult> | null> {
907
+ const state = await this.port.readState(featureId).catch(() => null);
908
+ if (!state) {
909
+ return null;
910
+ }
911
+ const existing = findLatestUnresolvedMainlineConflict(state.frontMatter);
912
+ if (!existing.entry) {
913
+ return null;
914
+ }
915
+
916
+ const unresolvedFiles = await this.listUnmergedFiles(worktreePath);
917
+ if (unresolvedFiles.length === 0) {
918
+ return null;
919
+ }
920
+
921
+ // Verify the recorded conflict is still valid: if mainline has moved
922
+ // beyond the recorded head, the stale merge is no longer meaningful.
923
+ // Abort it so a fresh reconciliation attempt can proceed cleanly.
924
+ const recordedMainlineHead = readConflictInfo(existing.entry).mainlineHead;
925
+ if (recordedMainlineHead && recordedMainlineHead !== 'unknown') {
926
+ const repoRoot = this.port.getRepoRoot();
927
+ const currentMainlineResult = await runGit(repoRoot, ['rev-parse', baseRef]);
928
+ const currentMainlineHead =
929
+ currentMainlineResult.code === 0 ? currentMainlineResult.stdout.trim() : null;
930
+
931
+ if (currentMainlineHead && currentMainlineHead !== recordedMainlineHead) {
932
+ await runGit(worktreePath, ['merge', '--abort']);
933
+ return null;
934
+ }
935
+ }
936
+
937
+ const conflictInfo = readConflictInfo(existing.entry);
938
+ const claim = await this.claimReconcilerLane(featureId);
939
+ const queuePosition = claim.activeFeatureId === featureId ? 0 : (claim.queuePosition ?? 1);
940
+ await this.upsertMainlineConflictState(
941
+ featureId,
942
+ {
943
+ mergeBase: conflictInfo.mergeBase,
944
+ mainlineHead: conflictInfo.mainlineHead,
945
+ worktreeHead: conflictInfo.worktreeHead,
946
+ conflictingFiles: unresolvedFiles,
947
+ },
948
+ {
949
+ baseRef,
950
+ resolutionStatus: claim.activeFeatureId === featureId ? 'in_progress' : 'pending',
951
+ queuePosition,
952
+ statusReason: buildQueueStatusReason(claim.activeFeatureId, queuePosition),
953
+ },
954
+ );
955
+ const dispatched =
956
+ claim.activeFeatureId === featureId
957
+ ? await this.dispatchReconcilerTurn(featureId, baseRef, claim.sessionId, {
958
+ mergeBase: conflictInfo.mergeBase,
959
+ mainlineHead: conflictInfo.mainlineHead,
960
+ worktreeHead: conflictInfo.worktreeHead,
961
+ conflictingFiles: unresolvedFiles,
962
+ })
963
+ : false;
964
+
965
+ return ok<ReconciliationResult>({
966
+ status: 'conflict_detected',
967
+ divergence_detected: true,
968
+ files_changed_mainline: [],
969
+ files_changed_worktree: [],
970
+ conflicting_files: unresolvedFiles,
971
+ merge_base: conflictInfo.mergeBase,
972
+ mainline_head: conflictInfo.mainlineHead,
973
+ worktree_head: conflictInfo.worktreeHead,
974
+ reconciler_active_feature_id: claim.activeFeatureId,
975
+ reconciler_queue_position: queuePosition,
976
+ dispatched_to_reconciler: dispatched,
977
+ conflict_worktree_prepared: true,
978
+ });
979
+ }
980
+
981
+ private async materializeConflictWorktree(
982
+ featureId: string,
983
+ worktreePath: string,
984
+ baseRef: string,
985
+ mainlineHead: string,
986
+ ): Promise<MaterializedConflictResult> {
987
+ // Abort any stale in-progress merge before attempting a fresh one.
988
+ // A leftover MERGE_HEAD from a previous failed reconciliation would cause
989
+ // the new merge to fail immediately without testing for real conflicts.
990
+ const staleUnmerged = await this.listUnmergedFiles(worktreePath);
991
+ if (staleUnmerged.length > 0) {
992
+ await runGit(worktreePath, ['merge', '--abort']);
993
+ }
994
+
995
+ const mergeResult = await runGit(worktreePath, ['merge', '--no-commit', '--no-ff', baseRef]);
996
+ if (mergeResult.code === 0) {
997
+ const commitMessage = `chore: reconcile ${baseRef} into ${featureId}`;
998
+ const commitResult = await runGit(worktreePath, ['commit', '-m', commitMessage]);
999
+ if (commitResult.code !== 0) {
1000
+ await runGit(worktreePath, ['merge', '--abort']);
1001
+ return {
1002
+ kind: 'error',
1003
+ response: fail(
1004
+ ERROR_CODES.RECONCILIATION_FAILED,
1005
+ 'Materialized merge succeeded but commit of reconciled worktree failed',
1006
+ {
1007
+ stderr: commitResult.stderr,
1008
+ retryable: false,
1009
+ requires_human: true,
1010
+ },
1011
+ ),
1012
+ };
1013
+ }
1014
+
1015
+ await this.appendReconciliationLog(featureId, `materialized_merge_commit: ${mainlineHead}`);
1016
+ return { kind: 'merged', strategy: 'merge_commit' };
1017
+ }
1018
+
1019
+ const unmergedFiles = await this.listUnmergedFiles(worktreePath);
1020
+ if (unmergedFiles.length > 0) {
1021
+ return { kind: 'conflict', conflictingFiles: unmergedFiles };
1022
+ }
1023
+
1024
+ await runGit(worktreePath, ['merge', '--abort']);
1025
+ return {
1026
+ kind: 'error',
1027
+ response: fail(
1028
+ ERROR_CODES.RECONCILIATION_FAILED,
1029
+ 'Failed to materialize a merge-conflict worktree for reconciler handoff',
1030
+ {
1031
+ stderr: mergeResult.stderr || mergeResult.stdout,
1032
+ retryable: false,
1033
+ requires_human: true,
1034
+ },
1035
+ ),
1036
+ };
1037
+ }
1038
+
1039
+ private async listUnmergedFiles(worktreePath: string): Promise<string[]> {
1040
+ const status = await runGit(worktreePath, ['diff', '--name-only', '--diff-filter=U']);
1041
+ if (status.code !== 0) {
1042
+ return [];
1043
+ }
1044
+ return parseDiffNameOnly(status.stdout);
1045
+ }
1046
+
1047
+ private async claimReconcilerLane(featureId: string): Promise<ReconcilerLaneClaim> {
1048
+ return await this.port.withIndexLock(async () => {
1049
+ const runtimeSessions = this.port.normalizeRuntimeSessions(await this.port.readRunLease());
1050
+ const activeFeatureId =
1051
+ typeof runtimeSessions.reconciler_active_feature_id === 'string' &&
1052
+ runtimeSessions.reconciler_active_feature_id.length > 0
1053
+ ? runtimeSessions.reconciler_active_feature_id
1054
+ : null;
1055
+ let queue = normalizeQueue(asStringArray(runtimeSessions.reconciler_queue)).filter(
1056
+ (queuedFeatureId) => queuedFeatureId !== activeFeatureId,
1057
+ );
1058
+
1059
+ let nextActiveFeatureId = activeFeatureId;
1060
+ let dispatchNow = false;
1061
+
1062
+ if (!nextActiveFeatureId) {
1063
+ nextActiveFeatureId = featureId;
1064
+ queue = queue.filter((queuedFeatureId) => queuedFeatureId !== featureId);
1065
+ dispatchNow = true;
1066
+ } else if (nextActiveFeatureId !== featureId && !queue.includes(featureId)) {
1067
+ queue = [...queue, featureId];
1068
+ }
1069
+
1070
+ const updated: RuntimeSessionsSnapshot = {
1071
+ ...runtimeSessions,
1072
+ reconciler_active_feature_id: nextActiveFeatureId,
1073
+ reconciler_queue: queue,
1074
+ };
1075
+ await this.port.writeRunLease(updated);
1076
+
1077
+ return {
1078
+ activeFeatureId: nextActiveFeatureId,
1079
+ sessionId:
1080
+ typeof updated.reconciler_session_id === 'string' &&
1081
+ updated.reconciler_session_id.length > 0
1082
+ ? updated.reconciler_session_id
1083
+ : null,
1084
+ queuePosition:
1085
+ nextActiveFeatureId === featureId ? 0 : Math.max(1, queue.indexOf(featureId) + 1),
1086
+ dispatchNow,
1087
+ };
1088
+ });
1089
+ }
1090
+
1091
+ private async releaseReconcilerLane(featureId: string): Promise<ReconcilerLaneRelease> {
1092
+ return await this.port.withIndexLock(async () => {
1093
+ const runtimeSessions = this.port.normalizeRuntimeSessions(await this.port.readRunLease());
1094
+ const queue = normalizeQueue(asStringArray(runtimeSessions.reconciler_queue)).filter(
1095
+ (queuedFeatureId) => queuedFeatureId !== featureId,
1096
+ );
1097
+ const activeFeatureId =
1098
+ typeof runtimeSessions.reconciler_active_feature_id === 'string' &&
1099
+ runtimeSessions.reconciler_active_feature_id.length > 0
1100
+ ? runtimeSessions.reconciler_active_feature_id
1101
+ : null;
1102
+
1103
+ let nextFeatureId = activeFeatureId;
1104
+ let nextQueue = queue;
1105
+ if (activeFeatureId === featureId) {
1106
+ nextFeatureId = queue[0] ?? null;
1107
+ nextQueue = nextFeatureId ? queue.slice(1) : [];
1108
+ }
1109
+
1110
+ const updated: RuntimeSessionsSnapshot = {
1111
+ ...runtimeSessions,
1112
+ reconciler_active_feature_id: nextFeatureId,
1113
+ reconciler_queue: nextQueue,
1114
+ };
1115
+ await this.port.writeRunLease(updated);
1116
+
1117
+ return {
1118
+ nextFeatureId,
1119
+ sessionId:
1120
+ typeof updated.reconciler_session_id === 'string' &&
1121
+ updated.reconciler_session_id.length > 0
1122
+ ? updated.reconciler_session_id
1123
+ : null,
1124
+ };
1125
+ });
1126
+ }
1127
+
1128
+ private async releaseAndDispatchNext(featureId: string, baseRef: string): Promise<void> {
1129
+ let release = await this.releaseReconcilerLane(featureId);
1130
+ while (release.nextFeatureId) {
1131
+ const activated = await this.activateQueuedFeature(
1132
+ release.nextFeatureId,
1133
+ baseRef,
1134
+ release.sessionId,
1135
+ );
1136
+ if (activated) {
1137
+ return;
1138
+ }
1139
+ release = await this.releaseReconcilerLane(release.nextFeatureId);
1140
+ }
1141
+ }
1142
+
1143
+ private async activateQueuedFeature(
1144
+ featureId: string,
1145
+ baseRef: string,
1146
+ sessionId: string | null,
1147
+ ): Promise<boolean> {
1148
+ const state = await this.port.readState(featureId).catch(() => null);
1149
+ if (!state) {
1150
+ return false;
1151
+ }
1152
+
1153
+ const unresolved = findLatestUnresolvedMainlineConflict(state.frontMatter);
1154
+ if (!unresolved.entry) {
1155
+ return false;
1156
+ }
1157
+
1158
+ const conflictInfo = readConflictInfo(unresolved.entry);
1159
+ const unmergedFiles = await this.listUnmergedFiles(this.port.worktreePath(featureId));
1160
+ const conflictingFiles =
1161
+ unmergedFiles.length > 0 ? unmergedFiles : conflictInfo.conflictingFiles;
1162
+ if (conflictingFiles.length === 0) {
1163
+ return false;
1164
+ }
1165
+
1166
+ await this.upsertMainlineConflictState(
1167
+ featureId,
1168
+ {
1169
+ mergeBase: conflictInfo.mergeBase,
1170
+ mainlineHead: conflictInfo.mainlineHead,
1171
+ worktreeHead: conflictInfo.worktreeHead,
1172
+ conflictingFiles,
1173
+ },
1174
+ {
1175
+ baseRef,
1176
+ resolutionStatus: 'in_progress',
1177
+ queuePosition: 0,
1178
+ statusReason: buildQueueStatusReason(featureId, 0),
1179
+ },
1180
+ );
1181
+
1182
+ await this.dispatchReconcilerTurn(featureId, baseRef, sessionId, {
1183
+ mergeBase: conflictInfo.mergeBase,
1184
+ mainlineHead: conflictInfo.mainlineHead,
1185
+ worktreeHead: conflictInfo.worktreeHead,
1186
+ conflictingFiles,
1187
+ });
1188
+ return true;
1189
+ }
1190
+
1191
+ private async dispatchReconcilerTurn(
1192
+ featureId: string,
1193
+ baseRef: string,
1194
+ sessionId: string | null,
1195
+ info: {
1196
+ mergeBase: string;
1197
+ mainlineHead: string;
1198
+ worktreeHead: string;
1199
+ conflictingFiles: string[];
1200
+ },
1201
+ ): Promise<boolean> {
1202
+ if (!sessionId || sessionId === 'unknown' || sessionId === 'unassigned') {
1203
+ return false;
1204
+ }
1205
+
1206
+ const provider = this.port.getProvider();
1207
+ if (!provider?.sendMessage) {
1208
+ return false;
1209
+ }
1210
+
1211
+ await this.waitForSessionToBecomeActive(provider, sessionId);
1212
+ const approvedCollisionContexts = await this.approvedCollisionContextsForFeature(featureId);
1213
+
1214
+ const message = [
1215
+ `Reconciler lane assignment: feature_id=${featureId}`,
1216
+ `base_ref=${baseRef}`,
1217
+ `merge_base=${info.mergeBase}`,
1218
+ `mainline_head=${info.mainlineHead}`,
1219
+ `worktree_head=${info.worktreeHead}`,
1220
+ 'The feature worktree is already in a real conflicted merge state created by `git merge --no-commit --no-ff`.',
1221
+ 'Operate only on this feature until its mainline_divergence conflict is resolved.',
1222
+ 'Use feature-scoped tool calls with this exact feature_id.',
1223
+ `Conflicting files: ${info.conflictingFiles.join(', ') || 'unknown'}`,
1224
+ `approved_collision_context: ${JSON.stringify(approvedCollisionContexts)}`,
1225
+ ].join('\n');
1226
+
1227
+ try {
1228
+ await provider.sendMessage(sessionId, message);
1229
+ } catch (error) {
1230
+ if (isMissingReconcilerConversation(error) && this.port.refreshReconcilerSession) {
1231
+ const refreshedSessionId = await this.port.refreshReconcilerSession().catch(() => null);
1232
+ if (
1233
+ refreshedSessionId &&
1234
+ refreshedSessionId !== 'unknown' &&
1235
+ refreshedSessionId !== 'unassigned' &&
1236
+ refreshedSessionId !== sessionId
1237
+ ) {
1238
+ try {
1239
+ await this.waitForSessionToBecomeActive(provider, refreshedSessionId);
1240
+ await provider.sendMessage(refreshedSessionId, message);
1241
+ await this.appendReconciliationLog(
1242
+ featureId,
1243
+ `reconciler_session_refreshed old_session_id=${sessionId} new_session_id=${refreshedSessionId}`,
1244
+ );
1245
+ return true;
1246
+ } catch (retryError) {
1247
+ await this.appendReconciliationLog(
1248
+ featureId,
1249
+ `dispatch_failed: ${errorMessage(retryError)} session_id=${refreshedSessionId}`,
1250
+ );
1251
+ return false;
1252
+ }
1253
+ }
1254
+ }
1255
+ await this.appendReconciliationLog(
1256
+ featureId,
1257
+ `dispatch_failed: ${errorMessage(error)} session_id=${sessionId}`,
1258
+ );
1259
+ return false;
1260
+ }
1261
+ await this.appendReconciliationLog(
1262
+ featureId,
1263
+ approvedCollisionContexts.length > 0
1264
+ ? `dispatched_with_approved_collision_context override_ids=${approvedCollisionContexts
1265
+ .map((entry) => entry.override_id)
1266
+ .join(',')}`
1267
+ : 'dispatched_without_approved_collision_context',
1268
+ );
1269
+ return true;
1270
+ }
1271
+
1272
+ private async waitForSessionToBecomeActive(
1273
+ provider: ReconciliationControlProvider,
1274
+ sessionId: string,
1275
+ ): Promise<void> {
1276
+ if (!provider.getSessionInfo) {
1277
+ return;
1278
+ }
1279
+
1280
+ const timeoutMs = 5000;
1281
+ const pollIntervalMs = 250;
1282
+ const deadline = Date.now() + timeoutMs;
1283
+
1284
+ while (Date.now() <= deadline) {
1285
+ const sessionInfo = await provider.getSessionInfo(sessionId).catch(() => null);
1286
+ if (sessionInfo?.active) {
1287
+ return;
1288
+ }
1289
+ await new Promise<void>((resolve) => {
1290
+ setTimeout(resolve, pollIntervalMs);
1291
+ });
1292
+ }
1293
+ }
1294
+
1295
+ private async updateReconciliationState(featureId: string, mainlineHead: string): Promise<void> {
1296
+ try {
1297
+ await this.port.withFeatureLock(featureId, async () => {
1298
+ const { frontMatter, body } = await this.port.readState(featureId);
1299
+ await this.port.writeState(
1300
+ featureId,
1301
+ {
1302
+ ...frontMatter,
1303
+ version: readFrontMatterVersion(frontMatter) + 1,
1304
+ last_updated: nowIso(),
1305
+ last_reconciliation_at: nowIso(),
1306
+ last_reconciliation_sha: mainlineHead,
1307
+ },
1308
+ body,
1309
+ );
1310
+ });
1311
+ } catch {
1312
+ // Non-fatal
1313
+ }
1314
+ }
1315
+
1316
+ private async upsertMainlineConflictState(
1317
+ featureId: string,
1318
+ info: {
1319
+ mergeBase: string;
1320
+ mainlineHead: string;
1321
+ worktreeHead: string;
1322
+ conflictingFiles: string[];
1323
+ },
1324
+ options: {
1325
+ baseRef: string;
1326
+ resolutionStatus: 'pending' | 'in_progress';
1327
+ queuePosition: number;
1328
+ statusReason: string;
1329
+ },
1330
+ ): Promise<void> {
1331
+ try {
1332
+ await this.port.withFeatureLock(featureId, async () => {
1333
+ const { frontMatter, body } = await this.port.readState(featureId);
1334
+ const conflicts = Array.isArray(frontMatter.conflicts) ? [...frontMatter.conflicts] : [];
1335
+ const unresolved = findLatestUnresolvedMainlineConflict(frontMatter);
1336
+ const detectedAt =
1337
+ typeof unresolved.entry?.detected_at === 'string'
1338
+ ? unresolved.entry.detected_at
1339
+ : nowIso();
1340
+ const resumeStatus =
1341
+ typeof unresolved.entry?.resume_status === 'string'
1342
+ ? unresolved.entry.resume_status
1343
+ : frontMatter.status === STATUS.BLOCKED
1344
+ ? STATUS.BUILDING
1345
+ : frontMatter.status;
1346
+ const queueMetadata = {
1347
+ queue_position: options.queuePosition,
1348
+ queued_at:
1349
+ typeof unresolved.entry?.queued_at === 'string' ? unresolved.entry.queued_at : nowIso(),
1350
+ status_reason: options.statusReason,
1351
+ };
1352
+ const nextConflict: MainlineConflictEntry = {
1353
+ ...(unresolved.entry ?? {}),
1354
+ type: 'mainline_divergence',
1355
+ detected_at: detectedAt,
1356
+ base_ref: options.baseRef,
1357
+ merge_base: info.mergeBase,
1358
+ mainline_head: info.mainlineHead,
1359
+ worktree_head: info.worktreeHead,
1360
+ conflicting_files: info.conflictingFiles,
1361
+ resolution_status: options.resolutionStatus,
1362
+ resume_status: resumeStatus,
1363
+ ...queueMetadata,
1364
+ ...(options.resolutionStatus === 'in_progress' ? { dispatched_at: nowIso() } : {}),
1365
+ };
1366
+
1367
+ if (unresolved.index >= 0) {
1368
+ conflicts[unresolved.index] = nextConflict;
1369
+ } else {
1370
+ conflicts.push(nextConflict);
1371
+ }
1372
+
1373
+ // When the reconciler is assigned (in_progress or queued), set recovery.state
1374
+ // so the dashboard distinguishes automated recovery from hard operator blocks.
1375
+ const reconcilerAssigned =
1376
+ options.resolutionStatus === 'in_progress' || options.resolutionStatus === 'pending';
1377
+ const recoveryState = reconcilerAssigned
1378
+ ? {
1379
+ state: 'retrying' as const,
1380
+ cause: 'mainline_divergence',
1381
+ role: 'reconciler',
1382
+ resume_phase: resumeStatus,
1383
+ attempt: null,
1384
+ run_id: null,
1385
+ started_at: nowIso(),
1386
+ last_attempt_at: nowIso(),
1387
+ }
1388
+ : null;
1389
+
1390
+ await this.port.writeState(
1391
+ featureId,
1392
+ {
1393
+ ...frontMatter,
1394
+ version: readFrontMatterVersion(frontMatter) + 1,
1395
+ last_updated: nowIso(),
1396
+ status: STATUS.BLOCKED,
1397
+ status_reason: options.statusReason,
1398
+ last_reconciliation_at: nowIso(),
1399
+ recovery: recoveryState,
1400
+ conflicts,
1401
+ },
1402
+ body,
1403
+ );
1404
+ });
1405
+ } catch {
1406
+ // Non-fatal
1407
+ }
1408
+ }
1409
+
1410
+ private async resolveMainlineConflictState(
1411
+ featureId: string,
1412
+ mainlineHead: string,
1413
+ ): Promise<void> {
1414
+ try {
1415
+ await this.port.withFeatureLock(featureId, async () => {
1416
+ const { frontMatter, body } = await this.port.readState(featureId);
1417
+ const conflicts = Array.isArray(frontMatter.conflicts) ? [...frontMatter.conflicts] : [];
1418
+ let resumeStatus: string | null = null;
1419
+ let changed = false;
1420
+
1421
+ for (let index = 0; index < conflicts.length; index += 1) {
1422
+ const entry = conflicts[index];
1423
+ if (!entry || typeof entry !== 'object' || entry.type !== 'mainline_divergence') {
1424
+ continue;
1425
+ }
1426
+ if (entry.resolution_status === 'resolved') {
1427
+ continue;
1428
+ }
1429
+ if (typeof entry.resume_status === 'string' && entry.resume_status.length > 0) {
1430
+ resumeStatus = entry.resume_status;
1431
+ }
1432
+ conflicts[index] = {
1433
+ ...entry,
1434
+ resolution_status: 'resolved',
1435
+ resolved_at: nowIso(),
1436
+ resolved_mainline_head: mainlineHead,
1437
+ };
1438
+ changed = true;
1439
+ }
1440
+
1441
+ if (!changed) {
1442
+ return;
1443
+ }
1444
+
1445
+ await this.port.writeState(
1446
+ featureId,
1447
+ {
1448
+ ...frontMatter,
1449
+ version: readFrontMatterVersion(frontMatter) + 1,
1450
+ last_updated: nowIso(),
1451
+ status:
1452
+ frontMatter.status === STATUS.BLOCKED && resumeStatus
1453
+ ? resumeStatus
1454
+ : frontMatter.status,
1455
+ status_reason: `Reconciled against ${mainlineHead}`,
1456
+ last_reconciliation_at: nowIso(),
1457
+ last_reconciliation_sha: mainlineHead,
1458
+ recovery: null,
1459
+ conflicts,
1460
+ },
1461
+ body,
1462
+ );
1463
+ await this.appendReconciliationCompletionLog(featureId);
1464
+ });
1465
+ } catch {
1466
+ // Non-fatal
1467
+ }
1468
+ }
1469
+
1470
+ private async appendReconciliationCompletionLog(featureId: string): Promise<void> {
1471
+ const approvedCollisionContexts = await this.approvedCollisionContextsForFeature(featureId);
1472
+ const usedApprovedCollisionContext = approvedCollisionContexts.length > 0;
1473
+ const completionSummary = {
1474
+ approved_collision_context_used: usedApprovedCollisionContext,
1475
+ override_ids: approvedCollisionContexts.map((entry) => entry.override_id),
1476
+ counterpart_feature_ids: approvedCollisionContexts.map(
1477
+ (entry) => entry.counterpart_feature_id,
1478
+ ),
1479
+ source_collision_ids: approvedCollisionContexts.map((entry) => entry.source_collision_id),
1480
+ counterpart_behavior_preserved: usedApprovedCollisionContext
1481
+ ? 'not_reported_by_reconciler'
1482
+ : 'not_applicable',
1483
+ residual_compromise_or_follow_up: usedApprovedCollisionContext
1484
+ ? 'not_reported_by_reconciler'
1485
+ : 'none',
1486
+ summary_source: 'service_resolution',
1487
+ };
1488
+
1489
+ await this.appendReconciliationLog(
1490
+ featureId,
1491
+ `reconciliation_completion_summary ${JSON.stringify(completionSummary)}`,
1492
+ );
1493
+ }
1494
+
1495
+ private async writeEvidenceArtifacts(
1496
+ featureId: string,
1497
+ worktreePath: string,
1498
+ mergeBase: string,
1499
+ mainlineHead: string,
1500
+ worktreeHead: string,
1501
+ conflictingFiles: string[],
1502
+ ): Promise<void> {
1503
+ try {
1504
+ const timestamp = nowIso().replaceAll(':', '-').replaceAll('.', '-');
1505
+ const evidenceDir = path.join(this.port.evidencePath(featureId), 'reconciliation');
1506
+ await ensureDir(evidenceDir);
1507
+
1508
+ const mainlineDiff = await runGit(worktreePath, ['diff', `${mergeBase}..${mainlineHead}`]);
1509
+ if (mainlineDiff.code === 0 && mainlineDiff.stdout) {
1510
+ await fs.writeFile(
1511
+ path.join(evidenceDir, `${timestamp}-mainline-diff.patch`),
1512
+ mainlineDiff.stdout,
1513
+ 'utf8',
1514
+ );
1515
+ }
1516
+
1517
+ const worktreeDiff = await runGit(worktreePath, ['diff', `${mergeBase}..${worktreeHead}`]);
1518
+ if (worktreeDiff.code === 0 && worktreeDiff.stdout) {
1519
+ await fs.writeFile(
1520
+ path.join(evidenceDir, `${timestamp}-worktree-diff.patch`),
1521
+ worktreeDiff.stdout,
1522
+ 'utf8',
1523
+ );
1524
+ }
1525
+
1526
+ if (conflictingFiles.length > 0) {
1527
+ await fs.writeFile(
1528
+ path.join(evidenceDir, `${timestamp}-conflict-markers.txt`),
1529
+ conflictingFiles.join('\n'),
1530
+ 'utf8',
1531
+ );
1532
+ }
1533
+ } catch {
1534
+ // Non-fatal
1535
+ }
1536
+ }
1537
+
1538
+ private async appendReconciliationLog(featureId: string, message: string): Promise<void> {
1539
+ try {
1540
+ const logDir = this.port.logsPath(featureId);
1541
+ await ensureDir(logDir);
1542
+ const logFile = path.join(logDir, 'reconciliation.log');
1543
+ const line = `${nowIso()} ${message}\n`;
1544
+ await fs.appendFile(logFile, line, 'utf8');
1545
+ } catch {
1546
+ // Non-fatal
1547
+ }
1548
+ }
1549
+
1550
+ private async approvedCollisionContextsForFeature(
1551
+ featureId: string,
1552
+ ): Promise<ApprovedCollisionContext[]> {
1553
+ if (!this.port.readIndex) {
1554
+ return [];
1555
+ }
1556
+ const index = await this.port.readIndex().catch(() => null);
1557
+ if (!index) {
1558
+ return [];
1559
+ }
1560
+
1561
+ const overrides = asCollisionOverrides(index.collision_overrides).filter(
1562
+ (entry) =>
1563
+ entry.feature_ids.includes(featureId) &&
1564
+ (entry.status === 'active' || entry.status === 'applied'),
1565
+ );
1566
+ const records = asCollisionRecords(index.collision_records);
1567
+
1568
+ return overrides
1569
+ .map((override) => {
1570
+ const record = records.find((entry) => entry.collision_id === override.source_collision_id);
1571
+ if (!record) {
1572
+ return null;
1573
+ }
1574
+ return {
1575
+ override_id: override.override_id,
1576
+ counterpart_feature_id:
1577
+ override.feature_ids[0] === featureId
1578
+ ? override.feature_ids[1]
1579
+ : override.feature_ids[0],
1580
+ rationale: override.rationale,
1581
+ approved_paths: override.approved_paths,
1582
+ approved_collision_fingerprint: override.approved_collision_fingerprint,
1583
+ source_collision_id: override.source_collision_id,
1584
+ detected_at: record.detected_at,
1585
+ applied_at: override.applied_at,
1586
+ source_collision_record: record,
1587
+ };
1588
+ })
1589
+ .filter(defined);
1590
+ }
1591
+ }