agentic-orchestrator 0.1.28 → 0.2.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (836) hide show
  1. package/.claude/settings.local.json +46 -1
  2. package/.cortexrc +28 -0
  3. package/.github/agents/copilot-instructions.md +29 -0
  4. package/.github/copilot-instructions.md +93 -0
  5. package/.vscode/settings.json +13 -0
  6. package/.vscode/tms.code-snippets +223 -0
  7. package/AGENTS.md +72 -1
  8. package/Agentic-Orchestrator.iml +12 -11
  9. package/CLAUDE.md +72 -1
  10. package/CONSTITUTION.md +504 -0
  11. package/FUTURE-ENHANCEMENTS.md +85 -0
  12. package/NEXT-TASKS.md +25 -0
  13. package/PROMPTS.md +161 -0
  14. package/README.md +126 -29
  15. package/agentic/orchestrator/agents.yaml +4 -3
  16. package/agentic/orchestrator/defaults/policy.defaults.yaml +39 -3
  17. package/agentic/orchestrator/gates.yaml +15 -3
  18. package/agentic/orchestrator/policy.yaml +47 -3
  19. package/agentic/orchestrator/prompts/builder.system.md +69 -20
  20. package/agentic/orchestrator/prompts/planner-intake.system.md +149 -0
  21. package/agentic/orchestrator/prompts/planner.system.md +113 -40
  22. package/agentic/orchestrator/prompts/qa.system.md +73 -18
  23. package/agentic/orchestrator/prompts/reconciler.system.md +119 -0
  24. package/agentic/orchestrator/schemas/agents.schema.json +89 -1
  25. package/agentic/orchestrator/schemas/execution-control.schema.json +242 -0
  26. package/agentic/orchestrator/schemas/index.schema.json +234 -0
  27. package/agentic/orchestrator/schemas/intake.review.schema.json +82 -0
  28. package/agentic/orchestrator/schemas/organizer-ordering-artifact.schema.json +75 -0
  29. package/agentic/orchestrator/schemas/plan.schema.json +44 -0
  30. package/agentic/orchestrator/schemas/policy.schema.json +238 -9
  31. package/agentic/orchestrator/schemas/policy.user.schema.json +129 -1
  32. package/agentic/orchestrator/schemas/spec.manifest.bootstrap.schema.json +101 -0
  33. package/agentic/orchestrator/schemas/spec.manifest.verified.schema.json +80 -0
  34. package/agentic/orchestrator/schemas/state.schema.json +298 -3
  35. package/agentic/orchestrator/tools/catalog.json +145 -15
  36. package/agentic/orchestrator/tools/schemas/input/doctor.run.input.schema.json +18 -0
  37. package/agentic/orchestrator/tools/schemas/input/evidence.latest.input.schema.json +4 -0
  38. package/agentic/orchestrator/tools/schemas/input/evidence.verify_chain.input.schema.json +13 -0
  39. package/agentic/orchestrator/tools/schemas/input/feature.intake_submit.input.schema.json +11 -0
  40. package/agentic/orchestrator/tools/schemas/input/feature.question_answer.input.schema.json +15 -0
  41. package/agentic/orchestrator/tools/schemas/input/feature.question_create.input.schema.json +21 -0
  42. package/agentic/orchestrator/tools/schemas/input/feature.question_list.input.schema.json +13 -0
  43. package/agentic/orchestrator/tools/schemas/input/feature.ready_to_merge.input.schema.json +5 -0
  44. package/agentic/orchestrator/tools/schemas/input/feature.send_message.input.schema.json +1 -1
  45. package/agentic/orchestrator/tools/schemas/input/replay.timeline_get.input.schema.json +32 -0
  46. package/agentic/orchestrator/tools/schemas/input/repo.conflict_abort.input.schema.json +16 -0
  47. package/agentic/orchestrator/tools/schemas/input/repo.conflict_files.input.schema.json +16 -0
  48. package/agentic/orchestrator/tools/schemas/input/repo.reconcile_mainline.input.schema.json +37 -0
  49. package/agentic/orchestrator/tools/schemas/input/repo.resolve_conflict.input.schema.json +40 -0
  50. package/agentic/orchestrator/tools/schemas/input/runtime.execution_request_list.input.schema.json +7 -0
  51. package/agentic/orchestrator/tools/schemas/input/runtime.execution_request_submit.input.schema.json +25 -0
  52. package/agentic/orchestrator/tools/schemas/output/doctor.run.output.schema.json +34 -0
  53. package/agentic/orchestrator/tools/schemas/output/evidence.verify_chain.output.schema.json +23 -0
  54. package/agentic/orchestrator/tools/schemas/output/feature.get_context.output.schema.json +62 -2
  55. package/agentic/orchestrator/tools/schemas/output/feature.intake_submit.output.schema.json +24 -0
  56. package/agentic/orchestrator/tools/schemas/output/feature.question_answer.output.schema.json +21 -0
  57. package/agentic/orchestrator/tools/schemas/output/feature.question_create.output.schema.json +12 -0
  58. package/agentic/orchestrator/tools/schemas/output/feature.question_list.output.schema.json +14 -0
  59. package/agentic/orchestrator/tools/schemas/output/feature.ready_to_merge.output.schema.json +31 -0
  60. package/agentic/orchestrator/tools/schemas/output/feature.send_message.output.schema.json +8 -18
  61. package/agentic/orchestrator/tools/schemas/output/replay.timeline_get.output.schema.json +64 -0
  62. package/agentic/orchestrator/tools/schemas/output/repo.conflict_abort.output.schema.json +16 -0
  63. package/agentic/orchestrator/tools/schemas/output/repo.conflict_files.output.schema.json +22 -0
  64. package/agentic/orchestrator/tools/schemas/output/repo.reconcile_mainline.output.schema.json +61 -0
  65. package/agentic/orchestrator/tools/schemas/output/repo.resolve_conflict.output.schema.json +19 -0
  66. package/agentic/orchestrator/tools/schemas/output/report.dashboard.output.schema.json +26 -0
  67. package/agentic/orchestrator/tools/schemas/output/runtime.execution_request_list.output.schema.json +17 -0
  68. package/agentic/orchestrator/tools/schemas/output/runtime.execution_request_submit.output.schema.json +24 -0
  69. package/agentic/orchestrator/tools.md +13 -0
  70. package/apps/control-plane/scripts/validate-mcp-contracts.ts +1 -1
  71. package/apps/control-plane/src/application/kernel-tool-wiring.ts +140 -2
  72. package/apps/control-plane/src/application/services/activity-monitor-service.ts +44 -1
  73. package/apps/control-plane/src/application/services/bootstrap-manifest-generator-service.ts +251 -0
  74. package/apps/control-plane/src/application/services/checkpoint-service.ts +87 -27
  75. package/apps/control-plane/src/application/services/collision-override-service.ts +906 -0
  76. package/apps/control-plane/src/application/services/collision-queue-service.ts +129 -38
  77. package/apps/control-plane/src/application/services/cost-tracking-service.ts +94 -0
  78. package/apps/control-plane/src/application/services/execution-control-service.ts +599 -0
  79. package/apps/control-plane/src/application/services/feature-deletion-service.ts +37 -1
  80. package/apps/control-plane/src/application/services/feature-lifecycle-service.ts +182 -4
  81. package/apps/control-plane/src/application/services/feature-send-message-service.ts +17 -8
  82. package/apps/control-plane/src/application/services/feature-state-service.ts +191 -6
  83. package/apps/control-plane/src/application/services/gate-service.ts +121 -2
  84. package/apps/control-plane/src/application/services/git-reconciliation-service.ts +1591 -0
  85. package/apps/control-plane/src/application/services/intake-service.ts +1468 -0
  86. package/apps/control-plane/src/application/services/merge-service.ts +308 -17
  87. package/apps/control-plane/src/application/services/notifier-service.ts +3 -1
  88. package/apps/control-plane/src/application/services/performance-analytics-service.ts +75 -0
  89. package/apps/control-plane/src/application/services/plan-service.ts +336 -20
  90. package/apps/control-plane/src/application/services/question-service.ts +693 -0
  91. package/apps/control-plane/src/application/services/reactions-service.ts +73 -17
  92. package/apps/control-plane/src/application/services/replay-timeline-service.ts +295 -0
  93. package/apps/control-plane/src/application/services/reporting-service.ts +194 -10
  94. package/apps/control-plane/src/application/services/run-lease-service.ts +121 -5
  95. package/apps/control-plane/src/application/services/worktree-watchdog-service.ts +95 -8
  96. package/apps/control-plane/src/application/tools/tool-metadata.ts +7 -0
  97. package/apps/control-plane/src/application/usage-types.ts +138 -0
  98. package/apps/control-plane/src/cli/add-command-handler.ts +162 -0
  99. package/apps/control-plane/src/cli/answer-command-handler.ts +113 -0
  100. package/apps/control-plane/src/cli/attach-command-handler.ts +12 -3
  101. package/apps/control-plane/src/cli/cli-argument-parser.ts +133 -11
  102. package/apps/control-plane/src/cli/collision-command-handler.ts +113 -0
  103. package/apps/control-plane/src/cli/command-catalog.ts +479 -0
  104. package/apps/control-plane/src/cli/complete-command-handler.ts +23 -0
  105. package/apps/control-plane/src/cli/completion-command-handler.ts +25 -0
  106. package/apps/control-plane/src/cli/completion-resolver.ts +319 -0
  107. package/apps/control-plane/src/cli/completion-shell-renderer.ts +58 -0
  108. package/apps/control-plane/src/cli/dashboard-command-handler.ts +111 -1
  109. package/apps/control-plane/src/cli/dashboard-runtime-runner.ts +1036 -0
  110. package/apps/control-plane/src/cli/dashboard-runtime.ts +31 -0
  111. package/apps/control-plane/src/cli/help-command-handler.ts +17 -185
  112. package/apps/control-plane/src/cli/init-command-handler.ts +51 -6
  113. package/apps/control-plane/src/cli/merge-command-handler.ts +200 -0
  114. package/apps/control-plane/src/cli/questions-command-handler.ts +70 -0
  115. package/apps/control-plane/src/cli/replay-command-handler.ts +98 -0
  116. package/apps/control-plane/src/cli/resume-command-handler.ts +231 -16
  117. package/apps/control-plane/src/cli/retry-command-handler.ts +229 -17
  118. package/apps/control-plane/src/cli/retry-resume-decision.ts +75 -0
  119. package/apps/control-plane/src/cli/rollback-command-handler.ts +4 -2
  120. package/apps/control-plane/src/cli/run-command-handler.ts +35 -1
  121. package/apps/control-plane/src/cli/spec-ingestion-service.ts +45 -55
  122. package/apps/control-plane/src/cli/spec-preparation.ts +114 -0
  123. package/apps/control-plane/src/cli/spec-utils.ts +90 -11
  124. package/apps/control-plane/src/cli/status-command-handler.ts +122 -0
  125. package/apps/control-plane/src/cli/types.ts +41 -3
  126. package/apps/control-plane/src/core/collisions.ts +150 -31
  127. package/apps/control-plane/src/core/constants.ts +18 -1
  128. package/apps/control-plane/src/core/error-codes.ts +39 -0
  129. package/apps/control-plane/src/core/execution-control.ts +56 -0
  130. package/apps/control-plane/src/core/feature-resume-phase.ts +118 -0
  131. package/apps/control-plane/src/core/gate-freshness.ts +359 -0
  132. package/apps/control-plane/src/core/gate-log-extractor.ts +97 -0
  133. package/apps/control-plane/src/core/gates.ts +90 -1
  134. package/apps/control-plane/src/core/intake-artifacts.ts +295 -0
  135. package/apps/control-plane/src/core/kernel-types.ts +11 -0
  136. package/apps/control-plane/src/core/kernel.ts +604 -16
  137. package/apps/control-plane/src/core/mainline-conflict.ts +22 -0
  138. package/apps/control-plane/src/core/merge-repair.ts +149 -0
  139. package/apps/control-plane/src/core/path-layout.ts +46 -2
  140. package/apps/control-plane/src/core/path-rules.ts +11 -3
  141. package/apps/control-plane/src/core/plan-submit-recovery.ts +130 -0
  142. package/apps/control-plane/src/core/questions.ts +49 -0
  143. package/apps/control-plane/src/core/runtime-sessions.ts +4 -0
  144. package/apps/control-plane/src/core/schemas.ts +40 -1
  145. package/apps/control-plane/src/core/tool-caller.ts +25 -1
  146. package/apps/control-plane/src/core/utils/index-normalizer.ts +25 -4
  147. package/apps/control-plane/src/core/worktree-diff.ts +66 -0
  148. package/apps/control-plane/src/index.ts +29 -1
  149. package/apps/control-plane/src/interfaces/cli/bootstrap.ts +300 -6
  150. package/apps/control-plane/src/mcp/kernel-tool-executor.ts +17 -0
  151. package/apps/control-plane/src/mcp/tool-runtime.ts +63 -4
  152. package/apps/control-plane/src/providers/api-worker-provider.ts +62 -15
  153. package/apps/control-plane/src/providers/cli-worker-provider.ts +1037 -61
  154. package/apps/control-plane/src/providers/output-parsers/generic-output-parser.ts +99 -1
  155. package/apps/control-plane/src/providers/output-parsers/types.ts +2 -0
  156. package/apps/control-plane/src/providers/provider-defaults.ts +116 -7
  157. package/apps/control-plane/src/providers/providers.ts +225 -21
  158. package/apps/control-plane/src/providers/worker-provider-factory.ts +26 -2
  159. package/apps/control-plane/src/supervisor/artifact-stager.ts +52 -0
  160. package/apps/control-plane/src/supervisor/build-wave-executor.ts +477 -166
  161. package/apps/control-plane/src/supervisor/execution-enrollment-service.ts +408 -0
  162. package/apps/control-plane/src/supervisor/organizer-enrollment-scheduler.ts +117 -0
  163. package/apps/control-plane/src/supervisor/organizer-sidecar-service.ts +394 -0
  164. package/apps/control-plane/src/supervisor/plan-conformance-scorer.ts +2 -5
  165. package/apps/control-plane/src/supervisor/planner-phase.ts +85 -0
  166. package/apps/control-plane/src/supervisor/planning-wave-executor.ts +993 -64
  167. package/apps/control-plane/src/supervisor/prompt-bundle-loader.ts +20 -1
  168. package/apps/control-plane/src/supervisor/qa-wave-executor.ts +384 -177
  169. package/apps/control-plane/src/supervisor/run-coordinator.ts +723 -20
  170. package/apps/control-plane/src/supervisor/runtime.ts +485 -9
  171. package/apps/control-plane/src/supervisor/session-orchestrator.ts +220 -1
  172. package/apps/control-plane/src/supervisor/types.ts +152 -1
  173. package/apps/control-plane/src/supervisor/worker-decision-loop.ts +1030 -92
  174. package/apps/control-plane/test/activity-monitor.spec.ts +76 -0
  175. package/apps/control-plane/test/add-command-handler.spec.ts +189 -0
  176. package/apps/control-plane/test/application/services/feature-state-service.spec.ts +208 -0
  177. package/apps/control-plane/test/artifact-stager.spec.ts +93 -0
  178. package/apps/control-plane/test/batch-operations.spec.ts +58 -0
  179. package/apps/control-plane/test/bootstrap-edge-cases.spec.ts +50 -2
  180. package/apps/control-plane/test/bootstrap-manifest-generator-service.spec.ts +99 -0
  181. package/apps/control-plane/test/bootstrap.spec.ts +177 -4
  182. package/apps/control-plane/test/checkpoint-service.spec.ts +977 -29
  183. package/apps/control-plane/test/cli-argument-parser.spec.ts +119 -0
  184. package/apps/control-plane/test/cli-helpers.spec.ts +1202 -12
  185. package/apps/control-plane/test/cli.unit.spec.ts +797 -16
  186. package/apps/control-plane/test/collision-command-handler.spec.ts +182 -0
  187. package/apps/control-plane/test/collision-override-service.spec.ts +878 -0
  188. package/apps/control-plane/test/collision-queue.spec.ts +430 -2
  189. package/apps/control-plane/test/collisions.spec.ts +209 -1
  190. package/apps/control-plane/test/core-utils.spec.ts +61 -0
  191. package/apps/control-plane/test/cost-tracking.spec.ts +224 -0
  192. package/apps/control-plane/test/dashboard-api.integration.spec.ts +185 -5
  193. package/apps/control-plane/test/dashboard-client.spec.ts +948 -0
  194. package/apps/control-plane/test/dashboard-command.spec.ts +138 -6
  195. package/apps/control-plane/test/dashboard-runtime-runner.spec.ts +1550 -0
  196. package/apps/control-plane/test/dashboard-runtime.spec.ts +138 -0
  197. package/apps/control-plane/test/dashboard-ui-utils.spec.ts +56 -12
  198. package/apps/control-plane/test/dependency-scheduler.spec.ts +7 -1
  199. package/apps/control-plane/test/env-file.spec.ts +76 -0
  200. package/apps/control-plane/test/execution-control-service.spec.ts +535 -0
  201. package/apps/control-plane/test/execution-enrollment-service.spec.ts +648 -0
  202. package/apps/control-plane/test/feature-lifecycle.spec.ts +126 -0
  203. package/apps/control-plane/test/feature-resume-phase.spec.ts +164 -0
  204. package/apps/control-plane/test/feature-send-message-service.spec.ts +161 -0
  205. package/apps/control-plane/test/feature-state-service.spec.ts +295 -0
  206. package/apps/control-plane/test/fs.spec.ts +80 -0
  207. package/apps/control-plane/test/gate-freshness.spec.ts +590 -0
  208. package/apps/control-plane/test/gate-log-extractor.spec.ts +170 -0
  209. package/apps/control-plane/test/gates.spec.ts +108 -0
  210. package/apps/control-plane/test/git-reconciliation-service.spec.ts +2307 -0
  211. package/apps/control-plane/test/helpers.ts +65 -0
  212. package/apps/control-plane/test/incremental-gates.spec.ts +271 -0
  213. package/apps/control-plane/test/index-normalizer.spec.ts +98 -0
  214. package/apps/control-plane/test/init-wizard.spec.ts +17 -0
  215. package/apps/control-plane/test/intake-artifacts.spec.ts +203 -0
  216. package/apps/control-plane/test/intake-service.spec.ts +3176 -0
  217. package/apps/control-plane/test/kernel-collision-replay.spec.ts +3 -2
  218. package/apps/control-plane/test/kernel-tool-executor.spec.ts +77 -0
  219. package/apps/control-plane/test/kernel-tool-wiring.spec.ts +279 -0
  220. package/apps/control-plane/test/kernel.branches.spec.ts +15 -2
  221. package/apps/control-plane/test/kernel.coverage.spec.ts +7 -3
  222. package/apps/control-plane/test/kernel.coverage2.spec.ts +731 -2
  223. package/apps/control-plane/test/kernel.spec.ts +464 -2
  224. package/apps/control-plane/test/mainline-conflict.spec.ts +66 -0
  225. package/apps/control-plane/test/mcp-helpers.spec.ts +79 -0
  226. package/apps/control-plane/test/mcp.spec.ts +177 -13
  227. package/apps/control-plane/test/merge-command-handler.spec.ts +531 -0
  228. package/apps/control-plane/test/merge-service.spec.ts +570 -4
  229. package/apps/control-plane/test/notifier-service.spec.ts +26 -0
  230. package/apps/control-plane/test/organizer-enrollment-scheduler.spec.ts +340 -0
  231. package/apps/control-plane/test/organizer-ordering-artifact.spec.ts +95 -0
  232. package/apps/control-plane/test/organizer-sidecar-service.spec.ts +468 -0
  233. package/apps/control-plane/test/output-loop-detector.spec.ts +6 -0
  234. package/apps/control-plane/test/path-layout.spec.ts +70 -0
  235. package/apps/control-plane/test/performance-analytics.spec.ts +124 -0
  236. package/apps/control-plane/test/plan-conformance-scorer.spec.ts +53 -0
  237. package/apps/control-plane/test/plan-service.spec.ts +686 -4
  238. package/apps/control-plane/test/planning-wave-executor.spec.ts +3272 -86
  239. package/apps/control-plane/test/policy-loader-service.spec.ts +5 -0
  240. package/apps/control-plane/test/prompt-overlay.spec.ts +65 -0
  241. package/apps/control-plane/test/provider-command-runner-epipe.spec.ts +64 -0
  242. package/apps/control-plane/test/providers/api-worker-provider.spec.ts +129 -0
  243. package/apps/control-plane/test/providers/cli-worker-provider.spec.ts +148 -0
  244. package/apps/control-plane/test/providers/usage-types.spec.ts +98 -0
  245. package/apps/control-plane/test/providers.spec.ts +293 -16
  246. package/apps/control-plane/test/question-command-handlers.spec.ts +156 -0
  247. package/apps/control-plane/test/question-service.spec.ts +1119 -0
  248. package/apps/control-plane/test/reactions.spec.ts +114 -0
  249. package/apps/control-plane/test/replay-command-handler.spec.ts +144 -0
  250. package/apps/control-plane/test/replay-timeline-service.spec.ts +459 -0
  251. package/apps/control-plane/test/response.spec.ts +31 -0
  252. package/apps/control-plane/test/resume-command.spec.ts +757 -9
  253. package/apps/control-plane/test/retry-resume-decision.spec.ts +133 -0
  254. package/apps/control-plane/test/rollback-command-handler.spec.ts +334 -0
  255. package/apps/control-plane/test/rollback-command.spec.ts +120 -0
  256. package/apps/control-plane/test/run-coordinator.spec.ts +3062 -404
  257. package/apps/control-plane/test/schemas/state.schema.spec.ts +71 -0
  258. package/apps/control-plane/test/service-retry-paths.spec.ts +112 -0
  259. package/apps/control-plane/test/services.spec.ts +472 -2
  260. package/apps/control-plane/test/session-management.spec.ts +346 -1
  261. package/apps/control-plane/test/spec-ingestion.spec.ts +102 -28
  262. package/apps/control-plane/test/spec-preparation.spec.ts +182 -0
  263. package/apps/control-plane/test/supervisor-collaborators.spec.ts +191 -3
  264. package/apps/control-plane/test/supervisor.calltool.spec.ts +198 -0
  265. package/apps/control-plane/test/supervisor.spec.ts +95 -16
  266. package/apps/control-plane/test/supervisor.unit.spec.ts +385 -18
  267. package/apps/control-plane/test/tool-runtime.spec.ts +122 -0
  268. package/apps/control-plane/test/worker-decision-loop.spec.ts +3479 -476
  269. package/apps/control-plane/test/worker-execution-policy.spec.ts +1416 -6
  270. package/apps/control-plane/test/worker-provider-adapters.spec.ts +1894 -37
  271. package/apps/control-plane/test/worker-provider-factory.spec.ts +81 -0
  272. package/apps/control-plane/test/worktree-watchdog-service.spec.ts +125 -0
  273. package/apps/control-plane/vitest.config.ts +5 -0
  274. package/config/agentic/orchestrator/agents.yaml +22 -1
  275. package/config/agentic/orchestrator/gates.yaml +24 -7
  276. package/config/agentic/orchestrator/policy.yaml +23 -1
  277. package/config/agentic/orchestrator/prompts/builder.system.md +69 -20
  278. package/config/agentic/orchestrator/prompts/organizer.system.md +85 -0
  279. package/config/agentic/orchestrator/prompts/overrides/builder.claude.md +28 -0
  280. package/config/agentic/orchestrator/prompts/overrides/builder.codex.md +28 -0
  281. package/config/agentic/orchestrator/prompts/overrides/planner.claude.md +20 -0
  282. package/config/agentic/orchestrator/prompts/overrides/planner.codex.md +20 -0
  283. package/config/agentic/orchestrator/prompts/planner-intake.system.md +149 -0
  284. package/config/agentic/orchestrator/prompts/planner.system.md +113 -40
  285. package/config/agentic/orchestrator/prompts/qa.system.md +75 -18
  286. package/config/agentic/orchestrator/prompts/reconciler.system.md +119 -0
  287. package/dist/apps/control-plane/application/kernel-tool-wiring.d.ts +26 -2
  288. package/dist/apps/control-plane/application/kernel-tool-wiring.js +40 -2
  289. package/dist/apps/control-plane/application/kernel-tool-wiring.js.map +1 -1
  290. package/dist/apps/control-plane/application/services/activity-monitor-service.js +37 -1
  291. package/dist/apps/control-plane/application/services/activity-monitor-service.js.map +1 -1
  292. package/dist/apps/control-plane/application/services/bootstrap-manifest-generator-service.d.ts +4 -0
  293. package/dist/apps/control-plane/application/services/bootstrap-manifest-generator-service.js +188 -0
  294. package/dist/apps/control-plane/application/services/bootstrap-manifest-generator-service.js.map +1 -0
  295. package/dist/apps/control-plane/application/services/checkpoint-service.d.ts +5 -0
  296. package/dist/apps/control-plane/application/services/checkpoint-service.js +69 -24
  297. package/dist/apps/control-plane/application/services/checkpoint-service.js.map +1 -1
  298. package/dist/apps/control-plane/application/services/collision-override-service.d.ts +139 -0
  299. package/dist/apps/control-plane/application/services/collision-override-service.js +568 -0
  300. package/dist/apps/control-plane/application/services/collision-override-service.js.map +1 -0
  301. package/dist/apps/control-plane/application/services/collision-queue-service.d.ts +15 -0
  302. package/dist/apps/control-plane/application/services/collision-queue-service.js +92 -33
  303. package/dist/apps/control-plane/application/services/collision-queue-service.js.map +1 -1
  304. package/dist/apps/control-plane/application/services/cost-tracking-service.d.ts +11 -0
  305. package/dist/apps/control-plane/application/services/cost-tracking-service.js +75 -0
  306. package/dist/apps/control-plane/application/services/cost-tracking-service.js.map +1 -1
  307. package/dist/apps/control-plane/application/services/execution-control-service.d.ts +75 -0
  308. package/dist/apps/control-plane/application/services/execution-control-service.js +421 -0
  309. package/dist/apps/control-plane/application/services/execution-control-service.js.map +1 -0
  310. package/dist/apps/control-plane/application/services/feature-deletion-service.d.ts +1 -0
  311. package/dist/apps/control-plane/application/services/feature-deletion-service.js +23 -1
  312. package/dist/apps/control-plane/application/services/feature-deletion-service.js.map +1 -1
  313. package/dist/apps/control-plane/application/services/feature-lifecycle-service.d.ts +24 -1
  314. package/dist/apps/control-plane/application/services/feature-lifecycle-service.js +132 -3
  315. package/dist/apps/control-plane/application/services/feature-lifecycle-service.js.map +1 -1
  316. package/dist/apps/control-plane/application/services/feature-send-message-service.js +16 -8
  317. package/dist/apps/control-plane/application/services/feature-send-message-service.js.map +1 -1
  318. package/dist/apps/control-plane/application/services/feature-state-service.d.ts +36 -0
  319. package/dist/apps/control-plane/application/services/feature-state-service.js +163 -6
  320. package/dist/apps/control-plane/application/services/feature-state-service.js.map +1 -1
  321. package/dist/apps/control-plane/application/services/gate-service.d.ts +2 -1
  322. package/dist/apps/control-plane/application/services/gate-service.js +95 -5
  323. package/dist/apps/control-plane/application/services/gate-service.js.map +1 -1
  324. package/dist/apps/control-plane/application/services/git-reconciliation-service.d.ts +92 -0
  325. package/dist/apps/control-plane/application/services/git-reconciliation-service.js +1097 -0
  326. package/dist/apps/control-plane/application/services/git-reconciliation-service.js.map +1 -0
  327. package/dist/apps/control-plane/application/services/intake-service.d.ts +63 -0
  328. package/dist/apps/control-plane/application/services/intake-service.js +1050 -0
  329. package/dist/apps/control-plane/application/services/intake-service.js.map +1 -0
  330. package/dist/apps/control-plane/application/services/merge-service.d.ts +5 -1
  331. package/dist/apps/control-plane/application/services/merge-service.js +233 -18
  332. package/dist/apps/control-plane/application/services/merge-service.js.map +1 -1
  333. package/dist/apps/control-plane/application/services/notifier-service.d.ts +1 -1
  334. package/dist/apps/control-plane/application/services/notifier-service.js +1 -0
  335. package/dist/apps/control-plane/application/services/notifier-service.js.map +1 -1
  336. package/dist/apps/control-plane/application/services/performance-analytics-service.d.ts +11 -0
  337. package/dist/apps/control-plane/application/services/performance-analytics-service.js +59 -0
  338. package/dist/apps/control-plane/application/services/performance-analytics-service.js.map +1 -1
  339. package/dist/apps/control-plane/application/services/plan-service.d.ts +5 -0
  340. package/dist/apps/control-plane/application/services/plan-service.js +254 -15
  341. package/dist/apps/control-plane/application/services/plan-service.js.map +1 -1
  342. package/dist/apps/control-plane/application/services/question-service.d.ts +72 -0
  343. package/dist/apps/control-plane/application/services/question-service.js +507 -0
  344. package/dist/apps/control-plane/application/services/question-service.js.map +1 -0
  345. package/dist/apps/control-plane/application/services/reactions-service.d.ts +2 -0
  346. package/dist/apps/control-plane/application/services/reactions-service.js +60 -17
  347. package/dist/apps/control-plane/application/services/reactions-service.js.map +1 -1
  348. package/dist/apps/control-plane/application/services/replay-timeline-service.d.ts +39 -0
  349. package/dist/apps/control-plane/application/services/replay-timeline-service.js +205 -0
  350. package/dist/apps/control-plane/application/services/replay-timeline-service.js.map +1 -0
  351. package/dist/apps/control-plane/application/services/reporting-service.d.ts +59 -0
  352. package/dist/apps/control-plane/application/services/reporting-service.js +121 -9
  353. package/dist/apps/control-plane/application/services/reporting-service.js.map +1 -1
  354. package/dist/apps/control-plane/application/services/run-lease-service.d.ts +20 -0
  355. package/dist/apps/control-plane/application/services/run-lease-service.js +81 -4
  356. package/dist/apps/control-plane/application/services/run-lease-service.js.map +1 -1
  357. package/dist/apps/control-plane/application/services/worktree-watchdog-service.d.ts +10 -0
  358. package/dist/apps/control-plane/application/services/worktree-watchdog-service.js +65 -8
  359. package/dist/apps/control-plane/application/services/worktree-watchdog-service.js.map +1 -1
  360. package/dist/apps/control-plane/application/tools/tool-metadata.js +7 -0
  361. package/dist/apps/control-plane/application/tools/tool-metadata.js.map +1 -1
  362. package/dist/apps/control-plane/application/usage-types.d.ts +65 -0
  363. package/dist/apps/control-plane/application/usage-types.js +75 -0
  364. package/dist/apps/control-plane/application/usage-types.js.map +1 -0
  365. package/dist/apps/control-plane/cli/add-command-handler.d.ts +18 -0
  366. package/dist/apps/control-plane/cli/add-command-handler.js +110 -0
  367. package/dist/apps/control-plane/cli/add-command-handler.js.map +1 -0
  368. package/dist/apps/control-plane/cli/answer-command-handler.d.ts +8 -0
  369. package/dist/apps/control-plane/cli/answer-command-handler.js +96 -0
  370. package/dist/apps/control-plane/cli/answer-command-handler.js.map +1 -0
  371. package/dist/apps/control-plane/cli/attach-command-handler.js +8 -3
  372. package/dist/apps/control-plane/cli/attach-command-handler.js.map +1 -1
  373. package/dist/apps/control-plane/cli/cli-argument-parser.js +131 -11
  374. package/dist/apps/control-plane/cli/cli-argument-parser.js.map +1 -1
  375. package/dist/apps/control-plane/cli/collision-command-handler.d.ts +8 -0
  376. package/dist/apps/control-plane/cli/collision-command-handler.js +90 -0
  377. package/dist/apps/control-plane/cli/collision-command-handler.js.map +1 -0
  378. package/dist/apps/control-plane/cli/command-catalog.d.ts +21 -0
  379. package/dist/apps/control-plane/cli/command-catalog.js +416 -0
  380. package/dist/apps/control-plane/cli/command-catalog.js.map +1 -0
  381. package/dist/apps/control-plane/cli/complete-command-handler.d.ts +15 -0
  382. package/dist/apps/control-plane/cli/complete-command-handler.js +26 -0
  383. package/dist/apps/control-plane/cli/complete-command-handler.js.map +1 -0
  384. package/dist/apps/control-plane/cli/completion-command-handler.d.ts +8 -0
  385. package/dist/apps/control-plane/cli/completion-command-handler.js +20 -0
  386. package/dist/apps/control-plane/cli/completion-command-handler.js.map +1 -0
  387. package/dist/apps/control-plane/cli/completion-resolver.d.ts +1 -0
  388. package/dist/apps/control-plane/cli/completion-resolver.js +250 -0
  389. package/dist/apps/control-plane/cli/completion-resolver.js.map +1 -0
  390. package/dist/apps/control-plane/cli/completion-shell-renderer.d.ts +3 -0
  391. package/dist/apps/control-plane/cli/completion-shell-renderer.js +53 -0
  392. package/dist/apps/control-plane/cli/completion-shell-renderer.js.map +1 -0
  393. package/dist/apps/control-plane/cli/dashboard-command-handler.d.ts +1 -0
  394. package/dist/apps/control-plane/cli/dashboard-command-handler.js +84 -1
  395. package/dist/apps/control-plane/cli/dashboard-command-handler.js.map +1 -1
  396. package/dist/apps/control-plane/cli/dashboard-runtime-runner.d.ts +81 -0
  397. package/dist/apps/control-plane/cli/dashboard-runtime-runner.js +724 -0
  398. package/dist/apps/control-plane/cli/dashboard-runtime-runner.js.map +1 -0
  399. package/dist/apps/control-plane/cli/dashboard-runtime.d.ts +1 -0
  400. package/dist/apps/control-plane/cli/dashboard-runtime.js +26 -0
  401. package/dist/apps/control-plane/cli/dashboard-runtime.js.map +1 -0
  402. package/dist/apps/control-plane/cli/help-command-handler.js +13 -172
  403. package/dist/apps/control-plane/cli/help-command-handler.js.map +1 -1
  404. package/dist/apps/control-plane/cli/init-command-handler.js +51 -6
  405. package/dist/apps/control-plane/cli/init-command-handler.js.map +1 -1
  406. package/dist/apps/control-plane/cli/merge-command-handler.d.ts +8 -0
  407. package/dist/apps/control-plane/cli/merge-command-handler.js +139 -0
  408. package/dist/apps/control-plane/cli/merge-command-handler.js.map +1 -0
  409. package/dist/apps/control-plane/cli/questions-command-handler.d.ts +8 -0
  410. package/dist/apps/control-plane/cli/questions-command-handler.js +59 -0
  411. package/dist/apps/control-plane/cli/questions-command-handler.js.map +1 -0
  412. package/dist/apps/control-plane/cli/replay-command-handler.d.ts +15 -0
  413. package/dist/apps/control-plane/cli/replay-command-handler.js +55 -0
  414. package/dist/apps/control-plane/cli/replay-command-handler.js.map +1 -0
  415. package/dist/apps/control-plane/cli/resume-command-handler.d.ts +2 -0
  416. package/dist/apps/control-plane/cli/resume-command-handler.js +180 -17
  417. package/dist/apps/control-plane/cli/resume-command-handler.js.map +1 -1
  418. package/dist/apps/control-plane/cli/retry-command-handler.js +202 -16
  419. package/dist/apps/control-plane/cli/retry-command-handler.js.map +1 -1
  420. package/dist/apps/control-plane/cli/retry-resume-decision.d.ts +26 -0
  421. package/dist/apps/control-plane/cli/retry-resume-decision.js +61 -0
  422. package/dist/apps/control-plane/cli/retry-resume-decision.js.map +1 -0
  423. package/dist/apps/control-plane/cli/rollback-command-handler.js +3 -2
  424. package/dist/apps/control-plane/cli/rollback-command-handler.js.map +1 -1
  425. package/dist/apps/control-plane/cli/run-command-handler.js +26 -2
  426. package/dist/apps/control-plane/cli/run-command-handler.js.map +1 -1
  427. package/dist/apps/control-plane/cli/spec-ingestion-service.d.ts +2 -0
  428. package/dist/apps/control-plane/cli/spec-ingestion-service.js +37 -48
  429. package/dist/apps/control-plane/cli/spec-ingestion-service.js.map +1 -1
  430. package/dist/apps/control-plane/cli/spec-preparation.d.ts +14 -0
  431. package/dist/apps/control-plane/cli/spec-preparation.js +81 -0
  432. package/dist/apps/control-plane/cli/spec-preparation.js.map +1 -0
  433. package/dist/apps/control-plane/cli/spec-utils.d.ts +4 -0
  434. package/dist/apps/control-plane/cli/spec-utils.js +70 -11
  435. package/dist/apps/control-plane/cli/spec-utils.js.map +1 -1
  436. package/dist/apps/control-plane/cli/status-command-handler.js +69 -0
  437. package/dist/apps/control-plane/cli/status-command-handler.js.map +1 -1
  438. package/dist/apps/control-plane/cli/types.d.ts +41 -4
  439. package/dist/apps/control-plane/cli/types.js +9 -1
  440. package/dist/apps/control-plane/cli/types.js.map +1 -1
  441. package/dist/apps/control-plane/core/collisions.d.ts +37 -19
  442. package/dist/apps/control-plane/core/collisions.js +87 -12
  443. package/dist/apps/control-plane/core/collisions.js.map +1 -1
  444. package/dist/apps/control-plane/core/constants.d.ts +17 -1
  445. package/dist/apps/control-plane/core/constants.js +18 -1
  446. package/dist/apps/control-plane/core/constants.js.map +1 -1
  447. package/dist/apps/control-plane/core/error-codes.d.ts +39 -0
  448. package/dist/apps/control-plane/core/error-codes.js +39 -0
  449. package/dist/apps/control-plane/core/error-codes.js.map +1 -1
  450. package/dist/apps/control-plane/core/execution-control.d.ts +45 -0
  451. package/dist/apps/control-plane/core/execution-control.js +2 -0
  452. package/dist/apps/control-plane/core/execution-control.js.map +1 -0
  453. package/dist/apps/control-plane/core/feature-resume-phase.d.ts +3 -0
  454. package/dist/apps/control-plane/core/feature-resume-phase.js +88 -0
  455. package/dist/apps/control-plane/core/feature-resume-phase.js.map +1 -0
  456. package/dist/apps/control-plane/core/gate-freshness.d.ts +48 -0
  457. package/dist/apps/control-plane/core/gate-freshness.js +267 -0
  458. package/dist/apps/control-plane/core/gate-freshness.js.map +1 -0
  459. package/dist/apps/control-plane/core/gate-log-extractor.d.ts +22 -0
  460. package/dist/apps/control-plane/core/gate-log-extractor.js +66 -0
  461. package/dist/apps/control-plane/core/gate-log-extractor.js.map +1 -0
  462. package/dist/apps/control-plane/core/gates.d.ts +11 -2
  463. package/dist/apps/control-plane/core/gates.js +67 -3
  464. package/dist/apps/control-plane/core/gates.js.map +1 -1
  465. package/dist/apps/control-plane/core/intake-artifacts.d.ts +109 -0
  466. package/dist/apps/control-plane/core/intake-artifacts.js +143 -0
  467. package/dist/apps/control-plane/core/intake-artifacts.js.map +1 -0
  468. package/dist/apps/control-plane/core/kernel-types.d.ts +8 -0
  469. package/dist/apps/control-plane/core/kernel.d.ts +256 -8
  470. package/dist/apps/control-plane/core/kernel.js +400 -14
  471. package/dist/apps/control-plane/core/kernel.js.map +1 -1
  472. package/dist/apps/control-plane/core/mainline-conflict.d.ts +7 -0
  473. package/dist/apps/control-plane/core/mainline-conflict.js +20 -0
  474. package/dist/apps/control-plane/core/mainline-conflict.js.map +1 -0
  475. package/dist/apps/control-plane/core/merge-repair.d.ts +35 -0
  476. package/dist/apps/control-plane/core/merge-repair.js +99 -0
  477. package/dist/apps/control-plane/core/merge-repair.js.map +1 -0
  478. package/dist/apps/control-plane/core/path-layout.d.ts +10 -0
  479. package/dist/apps/control-plane/core/path-layout.js +32 -2
  480. package/dist/apps/control-plane/core/path-layout.js.map +1 -1
  481. package/dist/apps/control-plane/core/path-rules.js +9 -3
  482. package/dist/apps/control-plane/core/path-rules.js.map +1 -1
  483. package/dist/apps/control-plane/core/plan-submit-recovery.d.ts +22 -0
  484. package/dist/apps/control-plane/core/plan-submit-recovery.js +78 -0
  485. package/dist/apps/control-plane/core/plan-submit-recovery.js.map +1 -0
  486. package/dist/apps/control-plane/core/questions.d.ts +40 -0
  487. package/dist/apps/control-plane/core/questions.js +2 -0
  488. package/dist/apps/control-plane/core/questions.js.map +1 -0
  489. package/dist/apps/control-plane/core/runtime-sessions.d.ts +4 -0
  490. package/dist/apps/control-plane/core/schemas.d.ts +2 -0
  491. package/dist/apps/control-plane/core/schemas.js +31 -1
  492. package/dist/apps/control-plane/core/schemas.js.map +1 -1
  493. package/dist/apps/control-plane/core/tool-caller.d.ts +18 -1
  494. package/dist/apps/control-plane/core/utils/index-normalizer.js +17 -4
  495. package/dist/apps/control-plane/core/utils/index-normalizer.js.map +1 -1
  496. package/dist/apps/control-plane/core/worktree-diff.d.ts +4 -0
  497. package/dist/apps/control-plane/core/worktree-diff.js +52 -0
  498. package/dist/apps/control-plane/core/worktree-diff.js.map +1 -0
  499. package/dist/apps/control-plane/index.d.ts +10 -2
  500. package/dist/apps/control-plane/index.js +9 -2
  501. package/dist/apps/control-plane/index.js.map +1 -1
  502. package/dist/apps/control-plane/interfaces/cli/bootstrap.js +236 -6
  503. package/dist/apps/control-plane/interfaces/cli/bootstrap.js.map +1 -1
  504. package/dist/apps/control-plane/mcp/kernel-tool-executor.js +16 -0
  505. package/dist/apps/control-plane/mcp/kernel-tool-executor.js.map +1 -1
  506. package/dist/apps/control-plane/mcp/tool-runtime.d.ts +5 -0
  507. package/dist/apps/control-plane/mcp/tool-runtime.js +40 -5
  508. package/dist/apps/control-plane/mcp/tool-runtime.js.map +1 -1
  509. package/dist/apps/control-plane/providers/api-worker-provider.d.ts +2 -2
  510. package/dist/apps/control-plane/providers/api-worker-provider.js +40 -9
  511. package/dist/apps/control-plane/providers/api-worker-provider.js.map +1 -1
  512. package/dist/apps/control-plane/providers/cli-worker-provider.d.ts +59 -3
  513. package/dist/apps/control-plane/providers/cli-worker-provider.js +758 -46
  514. package/dist/apps/control-plane/providers/cli-worker-provider.js.map +1 -1
  515. package/dist/apps/control-plane/providers/output-parsers/generic-output-parser.js +91 -1
  516. package/dist/apps/control-plane/providers/output-parsers/generic-output-parser.js.map +1 -1
  517. package/dist/apps/control-plane/providers/output-parsers/types.d.ts +2 -0
  518. package/dist/apps/control-plane/providers/provider-defaults.d.ts +12 -0
  519. package/dist/apps/control-plane/providers/provider-defaults.js +103 -7
  520. package/dist/apps/control-plane/providers/provider-defaults.js.map +1 -1
  521. package/dist/apps/control-plane/providers/providers.d.ts +50 -4
  522. package/dist/apps/control-plane/providers/providers.js +145 -14
  523. package/dist/apps/control-plane/providers/providers.js.map +1 -1
  524. package/dist/apps/control-plane/providers/worker-provider-factory.d.ts +2 -0
  525. package/dist/apps/control-plane/providers/worker-provider-factory.js +8 -1
  526. package/dist/apps/control-plane/providers/worker-provider-factory.js.map +1 -1
  527. package/dist/apps/control-plane/supervisor/artifact-stager.d.ts +5 -0
  528. package/dist/apps/control-plane/supervisor/artifact-stager.js +45 -0
  529. package/dist/apps/control-plane/supervisor/artifact-stager.js.map +1 -0
  530. package/dist/apps/control-plane/supervisor/build-wave-executor.d.ts +24 -1
  531. package/dist/apps/control-plane/supervisor/build-wave-executor.js +362 -150
  532. package/dist/apps/control-plane/supervisor/build-wave-executor.js.map +1 -1
  533. package/dist/apps/control-plane/supervisor/execution-enrollment-service.d.ts +41 -0
  534. package/dist/apps/control-plane/supervisor/execution-enrollment-service.js +311 -0
  535. package/dist/apps/control-plane/supervisor/execution-enrollment-service.js.map +1 -0
  536. package/dist/apps/control-plane/supervisor/organizer-enrollment-scheduler.d.ts +15 -0
  537. package/dist/apps/control-plane/supervisor/organizer-enrollment-scheduler.js +93 -0
  538. package/dist/apps/control-plane/supervisor/organizer-enrollment-scheduler.js.map +1 -0
  539. package/dist/apps/control-plane/supervisor/organizer-sidecar-service.d.ts +44 -0
  540. package/dist/apps/control-plane/supervisor/organizer-sidecar-service.js +311 -0
  541. package/dist/apps/control-plane/supervisor/organizer-sidecar-service.js.map +1 -0
  542. package/dist/apps/control-plane/supervisor/plan-conformance-scorer.js +2 -5
  543. package/dist/apps/control-plane/supervisor/plan-conformance-scorer.js.map +1 -1
  544. package/dist/apps/control-plane/supervisor/planner-phase.d.ts +3 -0
  545. package/dist/apps/control-plane/supervisor/planner-phase.js +70 -0
  546. package/dist/apps/control-plane/supervisor/planner-phase.js.map +1 -0
  547. package/dist/apps/control-plane/supervisor/planning-wave-executor.d.ts +42 -0
  548. package/dist/apps/control-plane/supervisor/planning-wave-executor.js +753 -55
  549. package/dist/apps/control-plane/supervisor/planning-wave-executor.js.map +1 -1
  550. package/dist/apps/control-plane/supervisor/prompt-bundle-loader.js +19 -1
  551. package/dist/apps/control-plane/supervisor/prompt-bundle-loader.js.map +1 -1
  552. package/dist/apps/control-plane/supervisor/qa-wave-executor.d.ts +21 -0
  553. package/dist/apps/control-plane/supervisor/qa-wave-executor.js +287 -156
  554. package/dist/apps/control-plane/supervisor/qa-wave-executor.js.map +1 -1
  555. package/dist/apps/control-plane/supervisor/run-coordinator.d.ts +30 -1
  556. package/dist/apps/control-plane/supervisor/run-coordinator.js +561 -17
  557. package/dist/apps/control-plane/supervisor/run-coordinator.js.map +1 -1
  558. package/dist/apps/control-plane/supervisor/runtime.d.ts +84 -0
  559. package/dist/apps/control-plane/supervisor/runtime.js +393 -3
  560. package/dist/apps/control-plane/supervisor/runtime.js.map +1 -1
  561. package/dist/apps/control-plane/supervisor/session-orchestrator.d.ts +54 -0
  562. package/dist/apps/control-plane/supervisor/session-orchestrator.js +176 -1
  563. package/dist/apps/control-plane/supervisor/session-orchestrator.js.map +1 -1
  564. package/dist/apps/control-plane/supervisor/types.d.ts +142 -1
  565. package/dist/apps/control-plane/supervisor/types.js.map +1 -1
  566. package/dist/apps/control-plane/supervisor/worker-decision-loop.d.ts +68 -2
  567. package/dist/apps/control-plane/supervisor/worker-decision-loop.js +723 -89
  568. package/dist/apps/control-plane/supervisor/worker-decision-loop.js.map +1 -1
  569. package/docs/core/ARCHITECTURE.md +227 -0
  570. package/docs/core/DECISIONS.md +94 -0
  571. package/docs/core/DOMAIN-LOGIC.md +60 -0
  572. package/docs/core/PATTERNS.md +201 -0
  573. package/docs/core/TROUBLESHOOTING.md +347 -0
  574. package/docs/core/intentgraph-dependencies.json +39860 -0
  575. package/docs/core/intentgraph.index.json +46580 -0
  576. package/docs/plans/2026-03-10-gate-failure-targeted-repair-design.md +224 -0
  577. package/docs/plans/2026-03-10-gate-failure-targeted-repair.md +1032 -0
  578. package/docs/superpowers/plans/2026-03-16-provider-cli-config.md +743 -0
  579. package/docs/superpowers/plans/2026-03-23-reconcile-divergence-fix.md +777 -0
  580. package/docs/superpowers/plans/2026-03-28-ordering-agent-implementation.md +1754 -0
  581. package/docs/superpowers/plans/2026-03-29-drop-zone-and-provider-optimization.md +1108 -0
  582. package/docs/superpowers/plans/2026-03-29-merge-target-feature-branch.md +685 -0
  583. package/docs/superpowers/plans/2026-03-29-organizer-sidecar-runtime-loop.md +1289 -0
  584. package/docs/superpowers/specs/2026-03-23-reconcile-divergence-fix-design.md +118 -0
  585. package/docs/superpowers/specs/2026-03-28-ordering-agent-spec-audit-design.md +50 -0
  586. package/docs/superpowers/specs/2026-03-29-drop-zone-and-provider-optimization-design.md +254 -0
  587. package/docs/superpowers/specs/2026-03-29-merge-target-feature-branch-design.md +152 -0
  588. package/docs/superpowers/specs/2026-03-29-organizer-sidecar-runtime-loop-design.md +225 -0
  589. package/package.json +3 -2
  590. package/packages/web-dashboard/package.json +2 -1
  591. package/packages/web-dashboard/src/app/analytics/page.tsx +36 -2
  592. package/packages/web-dashboard/src/app/api/actions/route.ts +274 -63
  593. package/packages/web-dashboard/src/app/api/actions/status/route.ts +35 -0
  594. package/packages/web-dashboard/src/app/api/analytics/provider/route.ts +18 -0
  595. package/packages/web-dashboard/src/app/api/collisions/approve/route.ts +58 -0
  596. package/packages/web-dashboard/src/app/api/features/[id]/checkpoint-diff/route.ts +36 -0
  597. package/packages/web-dashboard/src/app/api/features/[id]/checkpoints/route.ts +29 -0
  598. package/packages/web-dashboard/src/app/api/features/[id]/conflicts/abort/route.ts +29 -0
  599. package/packages/web-dashboard/src/app/api/features/[id]/conflicts/files/route.ts +30 -0
  600. package/packages/web-dashboard/src/app/api/features/[id]/conflicts/resolve/route.ts +51 -0
  601. package/packages/web-dashboard/src/app/api/features/[id]/conflicts/route.ts +75 -0
  602. package/packages/web-dashboard/src/app/api/features/[id]/diff/route.ts +16 -2
  603. package/packages/web-dashboard/src/app/api/features/[id]/files/route.ts +26 -0
  604. package/packages/web-dashboard/src/app/api/features/[id]/gate-history/route.ts +27 -0
  605. package/packages/web-dashboard/src/app/api/features/[id]/genealogy/route.ts +26 -0
  606. package/packages/web-dashboard/src/app/api/features/[id]/history/run/[runId]/route.ts +20 -0
  607. package/packages/web-dashboard/src/app/api/features/[id]/history/runs/route.ts +34 -0
  608. package/packages/web-dashboard/src/app/api/features/[id]/intake-workspace/route.ts +20 -0
  609. package/packages/web-dashboard/src/app/api/features/[id]/live-output/route.ts +74 -0
  610. package/packages/web-dashboard/src/app/api/features/[id]/plan/amend/route.ts +21 -0
  611. package/packages/web-dashboard/src/app/api/features/[id]/plan-progress/route.ts +20 -0
  612. package/packages/web-dashboard/src/app/api/features/[id]/planner-artifacts/[artifact]/route.ts +78 -0
  613. package/packages/web-dashboard/src/app/api/features/[id]/planner-lifecycle/route.ts +20 -0
  614. package/packages/web-dashboard/src/app/api/features/[id]/planning-workspace/route.ts +20 -0
  615. package/packages/web-dashboard/src/app/api/features/[id]/questions/[questionId]/answer/route.ts +27 -0
  616. package/packages/web-dashboard/src/app/api/features/[id]/questions/route.ts +18 -0
  617. package/packages/web-dashboard/src/app/api/features/[id]/review/route.ts +14 -7
  618. package/packages/web-dashboard/src/app/api/features/[id]/route.ts +57 -2
  619. package/packages/web-dashboard/src/app/api/features/[id]/spec/route.ts +30 -0
  620. package/packages/web-dashboard/src/app/api/features/[id]/triage/route.ts +83 -0
  621. package/packages/web-dashboard/src/app/api/features/[id]/worker-events/route.ts +40 -0
  622. package/packages/web-dashboard/src/app/api/launch/preview/route.ts +86 -0
  623. package/packages/web-dashboard/src/app/api/launch/submit/route.ts +180 -0
  624. package/packages/web-dashboard/src/app/api/mainline/status/route.ts +74 -0
  625. package/packages/web-dashboard/src/app/api/merge-queue/route.ts +13 -0
  626. package/packages/web-dashboard/src/app/api/policy/budget/route.ts +14 -0
  627. package/packages/web-dashboard/src/app/api/projects/route.ts +11 -7
  628. package/packages/web-dashboard/src/app/api/reconciler/queue/route.ts +47 -0
  629. package/packages/web-dashboard/src/app/api/run/route.ts +26 -2
  630. package/packages/web-dashboard/src/app/api/runtime/events/route.ts +227 -0
  631. package/packages/web-dashboard/src/app/api/runtime/operations/route.ts +269 -0
  632. package/packages/web-dashboard/src/app/api/runtime/questions/route.ts +11 -0
  633. package/packages/web-dashboard/src/app/api/runtime/runs/route.ts +80 -0
  634. package/packages/web-dashboard/src/app/api/status/route.ts +4 -2
  635. package/packages/web-dashboard/src/app/feature/[id]/page.tsx +32 -42
  636. package/packages/web-dashboard/src/app/globals.css +34 -3
  637. package/packages/web-dashboard/src/app/launch/page.tsx +362 -0
  638. package/packages/web-dashboard/src/app/layout.tsx +23 -1
  639. package/packages/web-dashboard/src/app/page.tsx +263 -272
  640. package/packages/web-dashboard/src/components/dashboard/attention-strip.tsx +52 -0
  641. package/packages/web-dashboard/src/components/dashboard/collision-approval-drawer.tsx +185 -0
  642. package/packages/web-dashboard/src/components/dashboard/command-center-header.tsx +102 -0
  643. package/packages/web-dashboard/src/components/dashboard/mainline-status-banner.tsx +84 -0
  644. package/packages/web-dashboard/src/components/dashboard/merged-archive.tsx +36 -0
  645. package/packages/web-dashboard/src/components/dashboard/prioritized-queues.tsx +98 -0
  646. package/packages/web-dashboard/src/components/dashboard/reconciler-queue-card.tsx +115 -0
  647. package/packages/web-dashboard/src/components/dashboard/secondary-diagnostics-rail.tsx +48 -0
  648. package/packages/web-dashboard/src/components/dashboard/task-filter-bar.tsx +74 -0
  649. package/packages/web-dashboard/src/components/dashboard/triage-drawer.tsx +455 -0
  650. package/packages/web-dashboard/src/components/diff-viewer.tsx +19 -3
  651. package/packages/web-dashboard/src/components/evidence-viewer.tsx +65 -51
  652. package/packages/web-dashboard/src/components/feature-card.tsx +90 -7
  653. package/packages/web-dashboard/src/components/feature-cost-panel.tsx +112 -11
  654. package/packages/web-dashboard/src/components/feature-list-view.tsx +25 -4
  655. package/packages/web-dashboard/src/components/features/runtime-inspector/EventsTimelineView.tsx +260 -0
  656. package/packages/web-dashboard/src/components/features/runtime-inspector/OperationsListView.tsx +172 -0
  657. package/packages/web-dashboard/src/components/features/runtime-inspector/RuntimeInspectorPanel.tsx +896 -0
  658. package/packages/web-dashboard/src/components/filter-bar.tsx +7 -39
  659. package/packages/web-dashboard/src/components/focus/ActionableRiskList.tsx +46 -0
  660. package/packages/web-dashboard/src/components/focus/AgentRolePerformanceCard.tsx +200 -0
  661. package/packages/web-dashboard/src/components/focus/BlockedGuidanceBanner.tsx +149 -0
  662. package/packages/web-dashboard/src/components/focus/CheckpointInspector.tsx +123 -0
  663. package/packages/web-dashboard/src/components/focus/CheckpointRail.tsx +118 -0
  664. package/packages/web-dashboard/src/components/focus/CheckpointScrubber.tsx +249 -0
  665. package/packages/web-dashboard/src/components/focus/CollisionApprovalBanner.tsx +192 -0
  666. package/packages/web-dashboard/src/components/focus/CollisionRadar.tsx +136 -0
  667. package/packages/web-dashboard/src/components/focus/ConflictStatusCard.tsx +52 -0
  668. package/packages/web-dashboard/src/components/focus/ContextSidebar.tsx +108 -0
  669. package/packages/web-dashboard/src/components/focus/DiagnosisPanel.tsx +68 -0
  670. package/packages/web-dashboard/src/components/focus/FeatureDecisionBanner.tsx +68 -0
  671. package/packages/web-dashboard/src/components/focus/FeatureQuestionAnswerPanel.tsx +167 -0
  672. package/packages/web-dashboard/src/components/focus/FocusHeader.tsx +54 -0
  673. package/packages/web-dashboard/src/components/focus/FocusLayout.tsx +283 -0
  674. package/packages/web-dashboard/src/components/focus/GateFlakinessSummary.tsx +144 -0
  675. package/packages/web-dashboard/src/components/focus/GenealogyTree.tsx +34 -0
  676. package/packages/web-dashboard/src/components/focus/HeroBlock.tsx +67 -0
  677. package/packages/web-dashboard/src/components/focus/LiveAgentConsole.tsx +277 -0
  678. package/packages/web-dashboard/src/components/focus/MergeQueueCard.tsx +78 -0
  679. package/packages/web-dashboard/src/components/focus/OperationalSummaryCard.tsx +227 -0
  680. package/packages/web-dashboard/src/components/focus/PinnedActions.tsx +96 -0
  681. package/packages/web-dashboard/src/components/focus/PlanAmendmentPanel.tsx +250 -0
  682. package/packages/web-dashboard/src/components/focus/PlanProgressPanel.tsx +133 -0
  683. package/packages/web-dashboard/src/components/focus/PlannerArtifactViewer.tsx +158 -0
  684. package/packages/web-dashboard/src/components/focus/PlannerLifecycleHeader.tsx +141 -0
  685. package/packages/web-dashboard/src/components/focus/ProgressSnapshotCard.tsx +113 -0
  686. package/packages/web-dashboard/src/components/focus/RecentMaterialChanges.tsx +69 -0
  687. package/packages/web-dashboard/src/components/focus/RoleLogViewer.tsx +436 -0
  688. package/packages/web-dashboard/src/components/focus/RunHistoryBrowser.tsx +62 -0
  689. package/packages/web-dashboard/src/components/focus/SpecViewer.tsx +172 -0
  690. package/packages/web-dashboard/src/components/focus/TabBar.tsx +33 -0
  691. package/packages/web-dashboard/src/components/focus/UsageBurnChart.tsx +212 -0
  692. package/packages/web-dashboard/src/components/focus/VerificationSummaryCard.tsx +122 -0
  693. package/packages/web-dashboard/src/components/focus/tabs/ChangesTab.tsx +325 -0
  694. package/packages/web-dashboard/src/components/focus/tabs/ConflictsTab.tsx +395 -0
  695. package/packages/web-dashboard/src/components/focus/tabs/GatesQaTab.tsx +38 -0
  696. package/packages/web-dashboard/src/components/focus/tabs/HistoryTab.tsx +213 -0
  697. package/packages/web-dashboard/src/components/focus/tabs/IntakeTab.tsx +429 -0
  698. package/packages/web-dashboard/src/components/focus/tabs/OverviewTab.tsx +217 -0
  699. package/packages/web-dashboard/src/components/focus/tabs/PlanningTab.tsx +390 -0
  700. package/packages/web-dashboard/src/components/focus/tabs/ReviewTab.tsx +497 -0
  701. package/packages/web-dashboard/src/components/focus/tabs/RuntimeTab.tsx +213 -0
  702. package/packages/web-dashboard/src/components/focus/tabs/TranscriptTab.tsx +315 -0
  703. package/packages/web-dashboard/src/components/gate-results.tsx +2 -2
  704. package/packages/web-dashboard/src/components/human-input-panel.tsx +33 -57
  705. package/packages/web-dashboard/src/components/kanban-board.tsx +4 -0
  706. package/packages/web-dashboard/src/components/launch/launch-draft-card.tsx +154 -0
  707. package/packages/web-dashboard/src/components/plan-viewer.tsx +147 -69
  708. package/packages/web-dashboard/src/components/quick-launch-panel.tsx +20 -47
  709. package/packages/web-dashboard/src/components/summary-bar.tsx +30 -76
  710. package/packages/web-dashboard/src/lib/aop-client.ts +2484 -36
  711. package/packages/web-dashboard/src/lib/blocked-state-guidance.ts +475 -0
  712. package/packages/web-dashboard/src/lib/collision-radar.ts +136 -0
  713. package/packages/web-dashboard/src/lib/dashboard-action-states.ts +204 -0
  714. package/packages/web-dashboard/src/lib/dashboard-runtime-client.ts +439 -0
  715. package/packages/web-dashboard/src/lib/dashboard-utils.ts +179 -18
  716. package/packages/web-dashboard/src/lib/drop-zone-utils.ts +92 -0
  717. package/packages/web-dashboard/src/lib/focus-detail-derivations.ts +958 -0
  718. package/packages/web-dashboard/src/lib/focus-view.ts +300 -0
  719. package/packages/web-dashboard/src/lib/health-diagnosis.ts +356 -0
  720. package/packages/web-dashboard/src/lib/launch-contracts.ts +77 -0
  721. package/packages/web-dashboard/src/lib/launch-markdown.ts +103 -0
  722. package/packages/web-dashboard/src/lib/launch-page-preview.ts +89 -0
  723. package/packages/web-dashboard/src/lib/live-feed.ts +1 -1
  724. package/packages/web-dashboard/src/lib/multi-project-config.ts +33 -0
  725. package/packages/web-dashboard/src/lib/orchestrator-tools.ts +881 -60
  726. package/packages/web-dashboard/src/lib/planner-workspace.ts +1285 -0
  727. package/packages/web-dashboard/src/lib/review-contracts.ts +5 -3
  728. package/packages/web-dashboard/src/lib/runtime-files.ts +285 -0
  729. package/packages/web-dashboard/src/lib/tool-catalog.ts +51 -0
  730. package/packages/web-dashboard/src/lib/types.ts +731 -3
  731. package/packages/web-dashboard/src/lib/usage-burn.ts +175 -0
  732. package/packages/web-dashboard/src/lib/worktree-diff.ts +128 -0
  733. package/packages/web-dashboard/src/styles/dashboard.module.css +1742 -459
  734. package/packages/web-dashboard/test/api/actions/route.spec.ts +675 -0
  735. package/packages/web-dashboard/test/api/features/diff.route.spec.ts +57 -0
  736. package/packages/web-dashboard/test/api/features/feature.route.spec.ts +99 -0
  737. package/packages/web-dashboard/test/api/features/live-output.route.spec.ts +123 -0
  738. package/packages/web-dashboard/test/api/features/plan-amend.route.spec.ts +95 -0
  739. package/packages/web-dashboard/test/api/features/planner-workspaces.route.spec.ts +162 -0
  740. package/packages/web-dashboard/test/api/features/question-answer.route.spec.ts +99 -0
  741. package/packages/web-dashboard/test/api/features/triage.route.spec.ts +195 -0
  742. package/packages/web-dashboard/test/api/launch/preview.route.spec.ts +149 -0
  743. package/packages/web-dashboard/test/api/launch/submit.route.spec.ts +382 -0
  744. package/packages/web-dashboard/test/api/runtime/events/route.spec.ts +164 -0
  745. package/packages/web-dashboard/test/api/runtime/operations/route.spec.ts +156 -0
  746. package/packages/web-dashboard/test/api/runtime/runs/route.spec.ts +112 -0
  747. package/packages/web-dashboard/test/components/changes-tab.spec.tsx +76 -0
  748. package/packages/web-dashboard/test/components/command-center-root.spec.tsx +87 -0
  749. package/packages/web-dashboard/test/components/diagnosis-panel.spec.tsx +59 -0
  750. package/packages/web-dashboard/test/components/feature-card.spec.tsx +45 -0
  751. package/packages/web-dashboard/test/components/focus-layout.spec.tsx +299 -0
  752. package/packages/web-dashboard/test/components/gate-results.spec.tsx +39 -0
  753. package/packages/web-dashboard/test/components/gates-qa-tab.spec.tsx +118 -0
  754. package/packages/web-dashboard/test/components/human-input-panel.spec.tsx +54 -0
  755. package/packages/web-dashboard/test/components/intake-tab.spec.tsx +210 -0
  756. package/packages/web-dashboard/test/components/kanban-board.spec.tsx +35 -0
  757. package/packages/web-dashboard/test/components/launch-draft-card.spec.tsx +114 -0
  758. package/packages/web-dashboard/test/components/launch-page.spec.tsx +79 -0
  759. package/packages/web-dashboard/test/components/overview-tab.spec.tsx +236 -0
  760. package/packages/web-dashboard/test/components/planning-tab.spec.tsx +202 -0
  761. package/packages/web-dashboard/test/components/review-tab.spec.tsx +169 -0
  762. package/packages/web-dashboard/test/components/role-log-viewer.spec.ts +42 -0
  763. package/packages/web-dashboard/test/components/runtime-inspector.spec.tsx +22 -0
  764. package/packages/web-dashboard/test/components/runtime-tab.spec.tsx +133 -0
  765. package/packages/web-dashboard/test/components/transcript-tab.spec.tsx +46 -0
  766. package/packages/web-dashboard/test/components/triage-drawer.spec.tsx +159 -0
  767. package/packages/web-dashboard/test/lib/aop-client.spec.ts +235 -0
  768. package/packages/web-dashboard/test/lib/dashboard-runtime-client.spec.ts +144 -0
  769. package/packages/web-dashboard/test/lib/focus-detail-derivations.spec.ts +314 -0
  770. package/packages/web-dashboard/test/lib/focus-view.spec.ts +248 -0
  771. package/packages/web-dashboard/test/lib/health-diagnosis.spec.ts +277 -0
  772. package/packages/web-dashboard/test/lib/launch-markdown.spec.ts +36 -0
  773. package/packages/web-dashboard/test/lib/multi-project-config.spec.ts +54 -0
  774. package/packages/web-dashboard/test/lib/orchestrator-tools.spec.ts +352 -0
  775. package/packages/web-dashboard/test/lib/planner-workspace.spec.ts +289 -0
  776. package/packages/web-dashboard/test/lib/worktree-diff.spec.ts +119 -0
  777. package/packages/web-dashboard/vitest.config.ts +2 -0
  778. package/spec-files/completed/agentic_orchestrator_add_feature_to_active_execution_spec.md +557 -0
  779. package/spec-files/completed/agentic_orchestrator_dashboard_command_center_redesign_spec.md +1147 -0
  780. package/spec-files/completed/agentic_orchestrator_execution_mode_spec.md +18 -16
  781. package/spec-files/completed/agentic_orchestrator_feature_focus_view_track_a_spec.md +672 -0
  782. package/spec-files/completed/agentic_orchestrator_feature_focus_view_track_b_spec.md +794 -0
  783. package/spec-files/completed/agentic_orchestrator_feature_focus_view_track_c_decision_centric_remediation_spec.md +1037 -0
  784. package/spec-files/completed/agentic_orchestrator_feature_focus_view_ux_redesign_spec.md +1432 -0
  785. package/spec-files/completed/agentic_orchestrator_focus_plan_tab_intake_planning_workspace_spec.md +921 -0
  786. package/spec-files/completed/agentic_orchestrator_intentional_collision_override_spec.md +584 -0
  787. package/spec-files/completed/agentic_orchestrator_interactive_planning_intake_and_requirements_verification_spec.md +1185 -0
  788. package/spec-files/completed/agentic_orchestrator_reactive_execution_enrollment_spec.md +864 -0
  789. package/spec-files/{outstanding → completed}/agentic_orchestrator_runtime_inspection_spec.md +92 -19
  790. package/spec-files/completed/agentic_orchestrator_scope_aware_run_lease_spec.md +408 -0
  791. package/spec-files/completed/git-reconciliation-engine.md +827 -0
  792. package/spec-files/outstanding/agentic_orchestrator_dashboard_quick_launch_and_control_surface_spec.md +331 -0
  793. package/spec-files/outstanding/agentic_orchestrator_enterprise_governance_dashboard_spec.md +16 -6
  794. package/spec-files/outstanding/agentic_orchestrator_evidence_integrity_doctor_spec.md +60 -9
  795. package/spec-files/outstanding/agentic_orchestrator_focus_plan_tab_execution_contract_workspace_spec.md +616 -0
  796. package/spec-files/outstanding/agentic_orchestrator_headless_standby_dashboard_runtime_spec.md +310 -0
  797. package/spec-files/outstanding/agentic_orchestrator_human_input_interaction_protocol_spec.md +175 -72
  798. package/spec-files/outstanding/agentic_orchestrator_interactive_rename_cleanup_spec.md +197 -0
  799. package/spec-files/outstanding/agentic_orchestrator_interactive_resume_and_reconciliation_disposition_spec.md +412 -0
  800. package/spec-files/outstanding/agentic_orchestrator_knowledge_canary_spec.md +166 -137
  801. package/spec-files/outstanding/agentic_orchestrator_observability_replay_spec.md +3 -3
  802. package/spec-files/outstanding/agentic_orchestrator_phase_specific_agent_profiles_and_token_telemetry_spec.md +303 -0
  803. package/spec-files/outstanding/agentic_orchestrator_planning_review_quality_spec.md +18 -5
  804. package/spec-files/outstanding/agentic_orchestrator_policy_stratification_spec.md +225 -0
  805. package/spec-files/outstanding/agentic_orchestrator_quality_adoption_execution_spec.md +77 -50
  806. package/spec-files/outstanding/agentic_orchestrator_ready_to_merge_branch_handoff_spec.md +724 -0
  807. package/spec-files/outstanding/agentic_orchestrator_remove_deterministic_mode_spec.md +263 -0
  808. package/spec-files/outstanding/agentic_orchestrator_request_more_context_and_dashboard_human_input_spec.md +456 -0
  809. package/spec-files/outstanding/agentic_orchestrator_spec_coverage_and_reconciliation_enforcement_spec.md +1411 -0
  810. package/spec-files/outstanding/agentic_orchestrator_spec_ordering_agent_spec.md +370 -0
  811. package/spec-files/outstanding/shadow_workspace_implementation_spec.md +1 -1
  812. package/spec-files/progress.md +2026 -120
  813. package/specs/001-runtime-inspection/checklists/requirements.md +35 -0
  814. package/specs/001-runtime-inspection/design.md +338 -0
  815. package/specs/001-runtime-inspection/spec.md +95 -0
  816. package/specs/002-scope-aware-lease/checklists/requirements.md +35 -0
  817. package/specs/002-scope-aware-lease/contracts/lease-registry.schema.json +101 -0
  818. package/specs/002-scope-aware-lease/data-model.md +236 -0
  819. package/specs/002-scope-aware-lease/plan.md +766 -0
  820. package/specs/002-scope-aware-lease/quickstart.md +150 -0
  821. package/specs/002-scope-aware-lease/research.md +135 -0
  822. package/specs/002-scope-aware-lease/spec.md +128 -0
  823. package/specs/002-scope-aware-lease/tasks.md +767 -0
  824. package/tsconfig.json +1 -1
  825. package/vitest.config.ts +28 -0
  826. package/ARCHITECTURE_ADHERENCE_ANALYSIS.md +0 -871
  827. package/packages/web-dashboard/next-env.d.ts +0 -6
  828. package/packages/web-dashboard/src/components/detail-panel.tsx +0 -1124
  829. package/packages/web-dashboard/src/components/review-workspace.tsx +0 -1162
  830. /package/spec-files/{outstanding → completed}/agentic_orchestrator_artifact_database_publishing_spec.md +0 -0
  831. /package/spec-files/{outstanding → completed}/agentic_orchestrator_cli_shell_tab_completion_spec.md +0 -0
  832. /package/spec-files/{outstanding → completed}/agentic_orchestrator_dashboard_diff_and_agent_console_spec.md +0 -0
  833. /package/spec-files/{outstanding → completed}/agentic_orchestrator_performance_improvements_spec.md +0 -0
  834. /package/spec-files/{outstanding → completed}/agentic_orchestrator_persistent_worker_runtime_spec.md +0 -0
  835. /package/spec-files/{outstanding → completed}/agentic_orchestrator_provider_auth_bootstrap_spec.md +0 -0
  836. /package/spec-files/{outstanding → completed}/agentic_orchestrator_real_worker_provider_execution_spec.md +0 -0
@@ -0,0 +1,1468 @@
1
+ import fs from 'node:fs/promises';
2
+ import path from 'node:path';
3
+ import { atomicWriteJson, nowIso, readJson } from '../../core/fs.js';
4
+ import { ERROR_CODES } from '../../core/error-codes.js';
5
+ import type {
6
+ BootstrapManifest,
7
+ IntakeClarificationAnswer,
8
+ IntakeReview,
9
+ IntakeSubmission,
10
+ IntakeSummary,
11
+ ManifestConfidence,
12
+ ManifestObligation,
13
+ ManifestObligationKind,
14
+ ManifestSourceRef,
15
+ ManifestVerificationHint,
16
+ VerifiedManifest,
17
+ } from '../../core/intake-artifacts.js';
18
+ import {
19
+ normalizeManifestConfidence,
20
+ obligationsMeetConfidenceThreshold,
21
+ } from '../../core/intake-artifacts.js';
22
+ import { GATE_RESULT, STATUS } from '../../core/constants.js';
23
+ import { fail } from '../../core/response.js';
24
+ import { BootstrapManifestGeneratorService } from './bootstrap-manifest-generator-service.js';
25
+
26
+ type StateRecord = Record<string, unknown>;
27
+ type QuestionEvidenceRecord = {
28
+ question_id: string | null;
29
+ status: string;
30
+ question_type: string | null;
31
+ phase: string | null;
32
+ resume_status: string | null;
33
+ resume_phase: string | null;
34
+ prompt: string | null;
35
+ answer: string | Record<string, unknown> | null;
36
+ answered_at: string | null;
37
+ answered_by: string | null;
38
+ details: Record<string, unknown>;
39
+ };
40
+
41
+ type StateUpdater = (
42
+ frontMatter: StateRecord,
43
+ body: string,
44
+ ) => Promise<{ frontMatter?: StateRecord; body?: string }>;
45
+
46
+ const AMBIGUITY_ID_TOKEN_PATTERN = /\bAMB[\s_-]*(\d{1,})\b/gi;
47
+ const OBLIGATION_ID_TOKEN_PATTERN = /\bOBL(?:[\s_-]*V)?[\s_-]*(\d{1,})\b/gi;
48
+ const MANIFEST_REVIEW_AMBIGUITY_ID = 'AMB-MANIFEST-REVIEW';
49
+
50
+ const OBLIGATION_KIND_ALIASES: Record<string, ManifestObligationKind> = {
51
+ artifact: 'artifact',
52
+ behavior: 'behavior',
53
+ component: 'artifact',
54
+ constraint: 'contract',
55
+ contract: 'contract',
56
+ endpoint: 'endpoint',
57
+ flow: 'workflow',
58
+ integration: 'integration',
59
+ route: 'endpoint',
60
+ schema: 'contract',
61
+ screen: 'artifact',
62
+ test: 'verification',
63
+ ui: 'artifact',
64
+ validation: 'verification',
65
+ verification: 'verification',
66
+ view: 'artifact',
67
+ workflow: 'workflow',
68
+ };
69
+
70
+ const VERIFICATION_HINT_ALIASES: Record<string, ManifestVerificationHint> = {
71
+ must: 'required',
72
+ optional: 'optional',
73
+ required: 'required',
74
+ shall: 'required',
75
+ };
76
+
77
+ function asRecord(value: unknown): Record<string, unknown> | null {
78
+ if (!value || typeof value !== 'object' || Array.isArray(value)) {
79
+ return null;
80
+ }
81
+ return value as Record<string, unknown>;
82
+ }
83
+
84
+ function asString(value: unknown): string | null {
85
+ if (typeof value !== 'string') {
86
+ return null;
87
+ }
88
+ const trimmed = value.trim();
89
+ return trimmed.length > 0 ? trimmed : null;
90
+ }
91
+
92
+ function asAnswerValue(value: unknown): string | Record<string, unknown> | null {
93
+ const stringValue = asString(value);
94
+ if (stringValue !== null) {
95
+ return stringValue;
96
+ }
97
+ return asRecord(value);
98
+ }
99
+
100
+ function normalizeUniqueStrings(values: string[]): string[] {
101
+ return [...new Set(values)].sort((left, right) => left.localeCompare(right));
102
+ }
103
+
104
+ function extractCanonicalIdMatches(
105
+ value: unknown,
106
+ pattern: RegExp,
107
+ prefix: 'AMB' | 'OBL',
108
+ ): string[] {
109
+ if (typeof value === 'string') {
110
+ return Array.from(value.matchAll(new RegExp(pattern.source, 'gi')), (match) => {
111
+ const digits = match[1] ?? '';
112
+ return `${prefix}-${digits.padStart(3, '0')}`;
113
+ });
114
+ }
115
+ if (!Array.isArray(value)) {
116
+ return [];
117
+ }
118
+ return value.flatMap((entry) => extractCanonicalIdMatches(entry, pattern, prefix));
119
+ }
120
+
121
+ function normalizeAmbiguityIds(value: unknown): string[] {
122
+ return normalizeUniqueStrings(
123
+ extractCanonicalIdMatches(value, AMBIGUITY_ID_TOKEN_PATTERN, 'AMB'),
124
+ );
125
+ }
126
+
127
+ function normalizeObligationIds(value: unknown): string[] {
128
+ return normalizeUniqueStrings(
129
+ extractCanonicalIdMatches(value, OBLIGATION_ID_TOKEN_PATTERN, 'OBL'),
130
+ );
131
+ }
132
+
133
+ function normalizeVerificationBasis(value: unknown): VerifiedManifest['verification_basis'] | null {
134
+ return value === 'structured_spec' ||
135
+ value === 'questions_resolved' ||
136
+ value === 'policy_approved_auto_promotion' ||
137
+ value === 'human_review_confirmed'
138
+ ? value
139
+ : null;
140
+ }
141
+
142
+ function normalizeReviewStatus(value: unknown): IntakeReview['status'] {
143
+ return value === 'proposed' || value === 'awaiting_input' || value === 'verified'
144
+ ? value
145
+ : 'proposed';
146
+ }
147
+
148
+ function readPositiveInteger(value: unknown): number | null {
149
+ return typeof value === 'number' && Number.isFinite(value) && value >= 1
150
+ ? Math.floor(value)
151
+ : null;
152
+ }
153
+
154
+ function readNonNegativeInteger(value: unknown): number | null {
155
+ return typeof value === 'number' && Number.isFinite(value) && value >= 0
156
+ ? Math.floor(value)
157
+ : null;
158
+ }
159
+
160
+ function normalizeAmbiguityStatus(
161
+ value: unknown,
162
+ fallback: IntakeReview['ambiguities'][number]['status'] = 'open',
163
+ ): IntakeReview['ambiguities'][number]['status'] {
164
+ if (value === 'open' || value === 'resolved') {
165
+ return value;
166
+ }
167
+ return fallback;
168
+ }
169
+
170
+ function normalizeReviewAmbiguity(
171
+ value: unknown,
172
+ fallback: IntakeReview['ambiguities'][number] | null = null,
173
+ ): IntakeReview['ambiguities'][number] | null {
174
+ const record = asRecord(value);
175
+ const id =
176
+ normalizeAmbiguityIds(record?.id)[0] ??
177
+ normalizeAmbiguityIds(record?.ambiguity_id)[0] ??
178
+ fallback?.id ??
179
+ null;
180
+ const summary = asString(record?.summary) ?? fallback?.summary ?? null;
181
+ if (!id || !summary) {
182
+ return null;
183
+ }
184
+
185
+ const fallbackResolution =
186
+ asString(record?.resolution) ?? asString(asRecord(record?.resolution)?.status);
187
+ const status = normalizeAmbiguityStatus(
188
+ record?.status ?? fallbackResolution,
189
+ fallback?.status ?? 'open',
190
+ );
191
+ const obligationIds = normalizeObligationIds(record?.obligation_ids);
192
+ const normalized: IntakeReview['ambiguities'][number] = {
193
+ id,
194
+ summary,
195
+ status,
196
+ };
197
+ const finalObligationIds =
198
+ obligationIds.length > 0 ? obligationIds : normalizeObligationIds(fallback?.obligation_ids);
199
+ if (finalObligationIds.length > 0) {
200
+ normalized.obligation_ids = finalObligationIds;
201
+ }
202
+ return normalized;
203
+ }
204
+
205
+ function normalizeReviewClarificationAnswer(
206
+ value: unknown,
207
+ ambiguityStatus: Map<string, IntakeReview['ambiguities'][number]['status']>,
208
+ ): IntakeClarificationAnswer | null {
209
+ const record = asRecord(value);
210
+ const questionId = asString(record?.question_id);
211
+ const prompt = asString(record?.prompt);
212
+ const answer = asAnswerValue(record?.answer);
213
+ const answeredAt = asString(record?.answered_at);
214
+ const answeredBy = asString(record?.answered_by);
215
+ const ambiguityIds = normalizeAmbiguityIds(record?.ambiguity_ids);
216
+ if (
217
+ !questionId ||
218
+ !prompt ||
219
+ answer === null ||
220
+ !answeredAt ||
221
+ !answeredBy ||
222
+ ambiguityIds.length === 0
223
+ ) {
224
+ return null;
225
+ }
226
+
227
+ const resolutionStatus =
228
+ record?.resolution_status === 'candidate' || record?.resolution_status === 'applied'
229
+ ? record.resolution_status
230
+ : ambiguityIds.every((ambiguityId) => ambiguityStatus.get(ambiguityId) === 'resolved')
231
+ ? 'applied'
232
+ : 'candidate';
233
+
234
+ return {
235
+ question_id: questionId,
236
+ ambiguity_ids: ambiguityIds,
237
+ obligation_ids: normalizeObligationIds(record?.obligation_ids),
238
+ prompt,
239
+ answer,
240
+ answered_at: answeredAt,
241
+ answered_by: answeredBy,
242
+ resolution_status: resolutionStatus,
243
+ };
244
+ }
245
+
246
+ function mergeReviewAmbiguities(
247
+ bootstrapAmbiguities: BootstrapManifest['ambiguities'],
248
+ reviewAmbiguities: unknown,
249
+ ): IntakeReview['ambiguities'] {
250
+ const normalizedReview = Array.isArray(reviewAmbiguities)
251
+ ? reviewAmbiguities
252
+ .map((item) => normalizeReviewAmbiguity(item))
253
+ .filter((item): item is IntakeReview['ambiguities'][number] => item !== null)
254
+ : [];
255
+ const reviewById = new Map(normalizedReview.map((item) => [item.id, item]));
256
+ const merged = bootstrapAmbiguities
257
+ .map((item) => normalizeReviewAmbiguity(item, item))
258
+ .filter((item): item is IntakeReview['ambiguities'][number] => item !== null)
259
+ .map((item) => {
260
+ const existing = reviewById.get(item.id);
261
+ return existing
262
+ ? {
263
+ ...item,
264
+ status: existing.status,
265
+ ...(existing.obligation_ids && existing.obligation_ids.length > 0
266
+ ? { obligation_ids: existing.obligation_ids }
267
+ : {}),
268
+ }
269
+ : item;
270
+ });
271
+ const bootstrapIds = new Set(merged.map((item) => item.id));
272
+ const extras = normalizedReview.filter((item) => !bootstrapIds.has(item.id));
273
+ return [...merged, ...extras];
274
+ }
275
+
276
+ function normalizeReviewClarificationAnswers(
277
+ value: unknown,
278
+ ambiguities: IntakeReview['ambiguities'],
279
+ ): IntakeClarificationAnswer[] {
280
+ const ambiguityStatus = new Map(ambiguities.map((item) => [item.id, item.status]));
281
+ if (!Array.isArray(value)) {
282
+ return [];
283
+ }
284
+ return value
285
+ .map((item) => normalizeReviewClarificationAnswer(item, ambiguityStatus))
286
+ .filter((item): item is IntakeClarificationAnswer => item !== null)
287
+ .sort((left, right) => {
288
+ if (left.answered_at === right.answered_at) {
289
+ return left.question_id.localeCompare(right.question_id);
290
+ }
291
+ return left.answered_at.localeCompare(right.answered_at);
292
+ });
293
+ }
294
+
295
+ function normalizeSubmissionAmbiguities(
296
+ value: unknown,
297
+ status: IntakeReview['ambiguities'][number]['status'],
298
+ bootstrapManifest: BootstrapManifest,
299
+ ): IntakeReview['ambiguities'] {
300
+ if (!Array.isArray(value)) {
301
+ return [];
302
+ }
303
+
304
+ const bootstrapById = new Map(bootstrapManifest.ambiguities.map((item) => [item.id, item]));
305
+ const normalized = value
306
+ .map((item) => {
307
+ const raw = asRecord(item);
308
+ const candidateId =
309
+ normalizeAmbiguityIds(raw?.id)[0] ?? normalizeAmbiguityIds(raw?.ambiguity_id)[0] ?? null;
310
+ const fallback = candidateId ? (bootstrapById.get(candidateId) ?? null) : null;
311
+ const normalizedItem = normalizeReviewAmbiguity(item, fallback);
312
+ if (!normalizedItem) {
313
+ return null;
314
+ }
315
+ return {
316
+ ...normalizedItem,
317
+ status,
318
+ };
319
+ })
320
+ .filter((item): item is IntakeReview['ambiguities'][number] => item !== null);
321
+
322
+ return normalized
323
+ .filter(
324
+ (item, index, items) => items.findIndex((candidate) => candidate.id === item.id) === index,
325
+ )
326
+ .sort((left, right) => left.id.localeCompare(right.id));
327
+ }
328
+
329
+ function normalizeManifestObligationKind(value: unknown): ManifestObligationKind | null {
330
+ const normalized = asString(value)?.toLowerCase();
331
+ if (!normalized) {
332
+ return null;
333
+ }
334
+ return OBLIGATION_KIND_ALIASES[normalized] ?? null;
335
+ }
336
+
337
+ function normalizeManifestVerificationHint(value: unknown): ManifestVerificationHint | null {
338
+ const normalized = asString(value)?.toLowerCase();
339
+ if (!normalized) {
340
+ return null;
341
+ }
342
+ return VERIFICATION_HINT_ALIASES[normalized] ?? null;
343
+ }
344
+
345
+ function normalizeManifestSourceRef(value: unknown): ManifestSourceRef | null {
346
+ const record = asRecord(value);
347
+ const lineStart = readPositiveInteger(record?.line_start);
348
+ const lineEnd = readPositiveInteger(record?.line_end);
349
+ if (lineStart === null || lineEnd === null) {
350
+ return null;
351
+ }
352
+ return {
353
+ line_start: lineStart,
354
+ line_end: Math.max(lineStart, lineEnd),
355
+ heading: asString(record?.heading),
356
+ };
357
+ }
358
+
359
+ function normalizeManifestSourceRefs(value: unknown): ManifestSourceRef[] {
360
+ if (!Array.isArray(value)) {
361
+ return [];
362
+ }
363
+ return value
364
+ .map((item) => normalizeManifestSourceRef(item))
365
+ .filter((item): item is ManifestSourceRef => item !== null);
366
+ }
367
+
368
+ function normalizeSubmittedManifestObligation(
369
+ value: unknown,
370
+ verificationBasis: VerifiedManifest['verification_basis'] | null,
371
+ ): ManifestObligation | null {
372
+ const record = asRecord(value);
373
+ if (!record) {
374
+ return null;
375
+ }
376
+
377
+ const obligationId = normalizeObligationIds(record.obligation_id)[0] ?? null;
378
+ const kind = normalizeManifestObligationKind(record.kind);
379
+ const name = asString(record.name) ?? asString(record.title);
380
+ const description = asString(record.description) ?? asString(record.summary) ?? name;
381
+ const sourceRefs = normalizeManifestSourceRefs(
382
+ record.source_refs ?? (record.source_ref ? [record.source_ref] : []),
383
+ );
384
+ const verificationHint = normalizeManifestVerificationHint(record.verification_hint);
385
+ const confidence = normalizeManifestConfidence(record.confidence);
386
+
387
+ if (
388
+ !obligationId ||
389
+ !kind ||
390
+ !name ||
391
+ !description ||
392
+ !verificationHint ||
393
+ !confidence ||
394
+ (sourceRefs.length === 0 && verificationBasis !== 'human_review_confirmed')
395
+ ) {
396
+ return null;
397
+ }
398
+
399
+ return {
400
+ obligation_id: obligationId,
401
+ kind,
402
+ name,
403
+ description,
404
+ source_refs: sourceRefs,
405
+ verification_hint: verificationHint,
406
+ confidence,
407
+ };
408
+ }
409
+
410
+ function asBootstrapManifest(value: unknown): BootstrapManifest | null {
411
+ const record = asRecord(value);
412
+ if (!record || record.artifact_type !== 'bootstrap') {
413
+ return null;
414
+ }
415
+ return record as unknown as BootstrapManifest;
416
+ }
417
+
418
+ function asVerifiedManifest(value: unknown): VerifiedManifest | null {
419
+ const record = asRecord(value);
420
+ if (!record || record.artifact_type !== 'verified') {
421
+ return null;
422
+ }
423
+ return record as unknown as VerifiedManifest;
424
+ }
425
+
426
+ function asIntakeReview(value: unknown): IntakeReview | null {
427
+ const record = asRecord(value);
428
+ if (!record || typeof record.feature_id !== 'string') {
429
+ return null;
430
+ }
431
+ const ambiguities = Array.isArray(record.ambiguities)
432
+ ? record.ambiguities
433
+ .map((item) => normalizeReviewAmbiguity(item))
434
+ .filter((item): item is IntakeReview['ambiguities'][number] => item !== null)
435
+ : [];
436
+ const clarificationAnswers = normalizeReviewClarificationAnswers(
437
+ record.clarification_answers,
438
+ ambiguities,
439
+ );
440
+ return {
441
+ feature_id: record.feature_id,
442
+ version: readPositiveInteger(record.version) ?? 1,
443
+ status: normalizeReviewStatus(record.status),
444
+ questions_open: readNonNegativeInteger(record.questions_open) ?? 0,
445
+ questions_resolved:
446
+ readNonNegativeInteger(record.questions_resolved) ?? clarificationAnswers.length,
447
+ ambiguities,
448
+ clarification_answers: clarificationAnswers,
449
+ bootstrap_manifest_version: readPositiveInteger(record.bootstrap_manifest_version),
450
+ verified_manifest_version: readPositiveInteger(record.verified_manifest_version),
451
+ promotion_basis: normalizeVerificationBasis(record.promotion_basis),
452
+ last_updated_at: asString(record.last_updated_at) ?? nowIso(),
453
+ };
454
+ }
455
+
456
+ function readIntakeRecord(frontMatter: StateRecord): Record<string, unknown> {
457
+ const intake = asRecord(frontMatter.intake);
458
+ return intake ?? {};
459
+ }
460
+
461
+ function readPlanningIntakePolicy(policy: Record<string, unknown>): Record<string, unknown> {
462
+ return asRecord(asRecord(policy.planning)?.intake) ?? {};
463
+ }
464
+
465
+ function buildDefaultReview(featureId: string, bootstrapManifest: BootstrapManifest): IntakeReview {
466
+ return {
467
+ feature_id: featureId,
468
+ version: 1,
469
+ status: 'proposed',
470
+ questions_open: 0,
471
+ questions_resolved: 0,
472
+ ambiguities: bootstrapManifest.ambiguities,
473
+ clarification_answers: [],
474
+ bootstrap_manifest_version: bootstrapManifest.manifest_version,
475
+ verified_manifest_version: null,
476
+ promotion_basis: null,
477
+ last_updated_at: nowIso(),
478
+ };
479
+ }
480
+
481
+ function summarizeIntake(
482
+ frontMatter: StateRecord,
483
+ bootstrapManifest: BootstrapManifest | null,
484
+ verifiedManifest: VerifiedManifest | null,
485
+ review: IntakeReview | null,
486
+ ): IntakeSummary {
487
+ const intake = readIntakeRecord(frontMatter);
488
+ const normalizedIntakeStatus =
489
+ intake.status === 'not_started' ||
490
+ intake.status === 'in_progress' ||
491
+ intake.status === 'awaiting_input'
492
+ ? intake.status
493
+ : intake.status === 'verified'
494
+ ? verifiedManifest
495
+ ? 'verified'
496
+ : frontMatter.status === STATUS.AWAITING_INPUT
497
+ ? 'awaiting_input'
498
+ : 'in_progress'
499
+ : null;
500
+ const openAmbiguityCount =
501
+ typeof intake.open_ambiguity_count === 'number'
502
+ ? Math.max(0, Math.floor(intake.open_ambiguity_count))
503
+ : (review?.ambiguities.filter((item) => item.status === 'open').length ??
504
+ bootstrapManifest?.ambiguities.filter((item) => item.status === 'open').length ??
505
+ 0);
506
+ return {
507
+ status:
508
+ normalizedIntakeStatus ??
509
+ (verifiedManifest
510
+ ? 'verified'
511
+ : frontMatter.status === STATUS.AWAITING_INPUT
512
+ ? 'awaiting_input'
513
+ : 'in_progress'),
514
+ bootstrap_manifest_version:
515
+ typeof intake.bootstrap_manifest_version === 'number'
516
+ ? Math.floor(intake.bootstrap_manifest_version)
517
+ : (bootstrapManifest?.manifest_version ?? null),
518
+ verified_manifest_version: verifiedManifest
519
+ ? typeof intake.verified_manifest_version === 'number'
520
+ ? Math.floor(intake.verified_manifest_version)
521
+ : verifiedManifest.manifest_version
522
+ : null,
523
+ open_ambiguity_count: openAmbiguityCount,
524
+ last_verified_at: verifiedManifest
525
+ ? typeof intake.last_verified_at === 'string'
526
+ ? intake.last_verified_at
527
+ : verifiedManifest.verified_at
528
+ : null,
529
+ promotion_basis: verifiedManifest
530
+ ? intake.promotion_basis === 'structured_spec' ||
531
+ intake.promotion_basis === 'questions_resolved' ||
532
+ intake.promotion_basis === 'policy_approved_auto_promotion' ||
533
+ intake.promotion_basis === 'human_review_confirmed'
534
+ ? intake.promotion_basis
535
+ : (review?.promotion_basis ?? null)
536
+ : null,
537
+ };
538
+ }
539
+
540
+ function normalizeQuestionEvidence(value: unknown): QuestionEvidenceRecord | null {
541
+ const record = asRecord(value);
542
+ const status = asString(record?.status);
543
+ if (!record || !status) {
544
+ return null;
545
+ }
546
+ return {
547
+ question_id: asString(record.question_id),
548
+ status,
549
+ question_type: asString(record.question_type),
550
+ phase: asString(record.phase),
551
+ resume_status: asString(record.resume_status),
552
+ resume_phase: asString(record.resume_phase),
553
+ prompt: asString(record.prompt),
554
+ answer: asAnswerValue(record.answer),
555
+ answered_at: asString(record.answered_at),
556
+ answered_by: asString(record.answered_by),
557
+ details: asRecord(record.details) ?? {},
558
+ };
559
+ }
560
+
561
+ function isAnsweredIntakeQuestion(
562
+ question: QuestionEvidenceRecord,
563
+ bootstrapManifest: BootstrapManifest,
564
+ ): boolean {
565
+ if (question.status !== 'answered' || !question.answered_by || !question.answered_at) {
566
+ return false;
567
+ }
568
+ const answeredAt = Date.parse(question.answered_at);
569
+ const generatedAt = Date.parse(bootstrapManifest.generated_at);
570
+ if (Number.isFinite(answeredAt) && Number.isFinite(generatedAt) && answeredAt < generatedAt) {
571
+ return false;
572
+ }
573
+ return (
574
+ question.resume_status === STATUS.INTAKE ||
575
+ question.resume_phase === STATUS.INTAKE ||
576
+ question.phase === STATUS.INTAKE
577
+ );
578
+ }
579
+
580
+ function questionReferencesAmbiguity(
581
+ question: QuestionEvidenceRecord,
582
+ ambiguityId: string,
583
+ ): boolean {
584
+ const singular =
585
+ normalizeAmbiguityIds(question.details.ambiguity_id)[0] ??
586
+ asString(question.details.ambiguity_id);
587
+ if (singular === ambiguityId) {
588
+ return true;
589
+ }
590
+ return normalizeAmbiguityIds(question.details.ambiguity_ids).includes(ambiguityId);
591
+ }
592
+
593
+ function questionConfirmsVerifiedManifest(question: QuestionEvidenceRecord): boolean {
594
+ return (
595
+ asString(question.details.confirmation_kind) === 'manifest_review' ||
596
+ asString(question.details.review_target) === 'verified_manifest' ||
597
+ asString(question.details.ambiguity_id) === MANIFEST_REVIEW_AMBIGUITY_ID
598
+ );
599
+ }
600
+
601
+ function isIntakeQuestion(question: QuestionEvidenceRecord): boolean {
602
+ return (
603
+ question.resume_status === STATUS.INTAKE ||
604
+ question.resume_phase === STATUS.INTAKE ||
605
+ question.phase === STATUS.INTAKE
606
+ );
607
+ }
608
+
609
+ function isOpenIntakeQuestion(question: QuestionEvidenceRecord): boolean {
610
+ return question.status === 'open' && isIntakeQuestion(question);
611
+ }
612
+
613
+ function isAnsweredIntakeClarification(question: QuestionEvidenceRecord): boolean {
614
+ return (
615
+ question.status === 'answered' &&
616
+ question.question_id !== null &&
617
+ question.prompt !== null &&
618
+ question.answer !== null &&
619
+ question.answered_at !== null &&
620
+ question.answered_by !== null &&
621
+ isIntakeQuestion(question) &&
622
+ normalizeAmbiguityIds(question.details.ambiguity_ids).length > 0
623
+ );
624
+ }
625
+
626
+ function buildClarificationAnswers(
627
+ review: IntakeReview,
628
+ questions: QuestionEvidenceRecord[],
629
+ ): IntakeClarificationAnswer[] {
630
+ const ambiguityStatus = new Map(review.ambiguities.map((item) => [item.id, item.status]));
631
+ return questions
632
+ .filter((question) => isAnsweredIntakeClarification(question))
633
+ .map((question) => {
634
+ const ambiguityIds = normalizeAmbiguityIds(question.details.ambiguity_ids);
635
+ const obligationIds = normalizeObligationIds(question.details.obligation_ids);
636
+ const resolutionStatus = ambiguityIds.every(
637
+ (ambiguityId) => ambiguityStatus.get(ambiguityId) === 'resolved',
638
+ )
639
+ ? 'applied'
640
+ : 'candidate';
641
+ return {
642
+ question_id: question.question_id,
643
+ ambiguity_ids: ambiguityIds,
644
+ obligation_ids: obligationIds,
645
+ prompt: question.prompt,
646
+ answer: question.answer,
647
+ answered_at: question.answered_at,
648
+ answered_by: question.answered_by,
649
+ resolution_status: resolutionStatus,
650
+ } satisfies IntakeClarificationAnswer;
651
+ })
652
+ .sort((left, right) => {
653
+ if (left.answered_at === right.answered_at) {
654
+ return left.question_id.localeCompare(right.question_id);
655
+ }
656
+ return left.answered_at.localeCompare(right.answered_at);
657
+ });
658
+ }
659
+
660
+ export interface IntakeServicePort {
661
+ featurePath(featureId: string): string;
662
+ specPath(featureId: string): string;
663
+ withFeatureLock<T>(featureId: string, operation: () => Promise<T>): Promise<T>;
664
+ readState(featureId: string): Promise<{ frontMatter: StateRecord; body: string }>;
665
+ updateState(
666
+ featureId: string,
667
+ expectedVersion: number | null,
668
+ updater: StateUpdater,
669
+ ): Promise<StateRecord>;
670
+ validateSchema(schemaName: string, value: unknown): Promise<{ valid: boolean; errors?: unknown }>;
671
+ getPolicySnapshot(): Record<string, unknown>;
672
+ }
673
+
674
+ export class IntakeService {
675
+ private readonly port: IntakeServicePort;
676
+ private readonly generator: BootstrapManifestGeneratorService;
677
+
678
+ constructor(
679
+ port: IntakeServicePort,
680
+ generator: BootstrapManifestGeneratorService = new BootstrapManifestGeneratorService(),
681
+ ) {
682
+ this.port = port;
683
+ this.generator = generator;
684
+ }
685
+
686
+ bootstrapManifestPath(featureId: string): string {
687
+ return path.join(this.port.featurePath(featureId), 'spec.manifest.bootstrap.json');
688
+ }
689
+
690
+ verifiedManifestPath(featureId: string): string {
691
+ return path.join(this.port.featurePath(featureId), 'spec.manifest.verified.json');
692
+ }
693
+
694
+ intakeReviewPath(featureId: string): string {
695
+ return path.join(this.port.featurePath(featureId), 'intake.review.json');
696
+ }
697
+
698
+ questionsPath(featureId: string): string {
699
+ return path.join(this.port.featurePath(featureId), 'questions.json');
700
+ }
701
+
702
+ planPath(featureId: string): string {
703
+ return path.join(this.port.featurePath(featureId), 'plan.json');
704
+ }
705
+
706
+ planMarkdownPath(featureId: string): string {
707
+ return path.join(this.port.featurePath(featureId), 'plan.md');
708
+ }
709
+
710
+ async readBootstrapManifest(featureId: string): Promise<BootstrapManifest | null> {
711
+ return asBootstrapManifest(await readJson(this.bootstrapManifestPath(featureId), null));
712
+ }
713
+
714
+ async readVerifiedManifest(featureId: string): Promise<VerifiedManifest | null> {
715
+ const rawManifest = await readJson(this.verifiedManifestPath(featureId), null);
716
+ if (!rawManifest) {
717
+ return null;
718
+ }
719
+ const validation = await this.port.validateSchema(
720
+ 'spec.manifest.verified.schema.json',
721
+ rawManifest,
722
+ );
723
+ if (!validation.valid) {
724
+ return null;
725
+ }
726
+ return asVerifiedManifest(rawManifest);
727
+ }
728
+
729
+ async readIntakeReview(featureId: string): Promise<IntakeReview | null> {
730
+ return asIntakeReview(await readJson(this.intakeReviewPath(featureId), null));
731
+ }
732
+
733
+ private normalizeSubmittedVerifiedManifest(
734
+ featureId: string,
735
+ verifiedManifest: unknown,
736
+ ): Record<string, unknown> {
737
+ const record = asRecord(verifiedManifest);
738
+ if (!record) {
739
+ return {};
740
+ }
741
+
742
+ const verificationBasis = normalizeVerificationBasis(record.verification_basis);
743
+ const rawObligations = Array.isArray(record.obligations) ? record.obligations : [];
744
+ const normalizedObligations = rawObligations
745
+ .map((item) => normalizeSubmittedManifestObligation(item, verificationBasis))
746
+ .filter((item): item is ManifestObligation => item !== null);
747
+
748
+ const invalidObligations = rawObligations
749
+ .map((item, index) => {
750
+ const obligation = normalizeSubmittedManifestObligation(item, verificationBasis);
751
+ if (obligation) {
752
+ return null;
753
+ }
754
+ const raw = asRecord(item);
755
+ return {
756
+ index,
757
+ obligation_id: asString(raw?.obligation_id),
758
+ kind: asString(raw?.kind),
759
+ name: asString(raw?.name) ?? asString(raw?.title),
760
+ };
761
+ })
762
+ .filter(
763
+ (
764
+ item,
765
+ ): item is {
766
+ index: number;
767
+ obligation_id: string | null;
768
+ kind: string | null;
769
+ name: string | null;
770
+ } => item !== null,
771
+ );
772
+
773
+ if (invalidObligations.length > 0) {
774
+ throw {
775
+ normalizedResponse: fail(
776
+ ERROR_CODES.INTAKE_SUBMISSION_INVALID,
777
+ 'verified manifest contains malformed obligations that could not be normalized',
778
+ {
779
+ feature_id: featureId,
780
+ invalid_obligations: invalidObligations,
781
+ retryable: false,
782
+ requires_human: true,
783
+ suggested_next_actions: [
784
+ 'Use canonical obligation ids such as OBL-001 or draft ids such as OBL-V01',
785
+ 'Use only supported obligation kinds or constraint aliases that normalize to contract',
786
+ 'Include source_refs for non-human-review verification bases',
787
+ ],
788
+ },
789
+ ),
790
+ };
791
+ }
792
+
793
+ const seenObligationIds = new Set<string>();
794
+ const duplicateObligationIds = normalizedObligations
795
+ .map((obligation) => obligation.obligation_id)
796
+ .filter((obligationId) => {
797
+ if (seenObligationIds.has(obligationId)) {
798
+ return true;
799
+ }
800
+ seenObligationIds.add(obligationId);
801
+ return false;
802
+ });
803
+ if (duplicateObligationIds.length > 0) {
804
+ throw {
805
+ normalizedResponse: fail(
806
+ ERROR_CODES.INTAKE_SUBMISSION_INVALID,
807
+ 'verified manifest contains duplicate obligation ids after normalization',
808
+ {
809
+ feature_id: featureId,
810
+ duplicate_obligation_ids: normalizeUniqueStrings(duplicateObligationIds),
811
+ retryable: false,
812
+ requires_human: true,
813
+ },
814
+ ),
815
+ };
816
+ }
817
+
818
+ return {
819
+ feature_id: asString(record.feature_id) ?? featureId,
820
+ manifest_version: readPositiveInteger(record.manifest_version) ?? record.manifest_version,
821
+ artifact_type:
822
+ record.artifact_type === 'verified' ? 'verified' : (record.artifact_type ?? null),
823
+ verification_basis: verificationBasis ?? record.verification_basis,
824
+ verified_at: asString(record.verified_at) ?? record.verified_at,
825
+ source_bootstrap_version:
826
+ record.source_bootstrap_version === null
827
+ ? null
828
+ : (readPositiveInteger(record.source_bootstrap_version) ??
829
+ record.source_bootstrap_version),
830
+ obligations: normalizedObligations,
831
+ };
832
+ }
833
+
834
+ private async validateArtifact(
835
+ schemaName: string,
836
+ value: unknown,
837
+ message: string,
838
+ ): Promise<void> {
839
+ const validation = await this.port.validateSchema(schemaName, value);
840
+ if (!validation.valid) {
841
+ throw {
842
+ normalizedResponse: fail(ERROR_CODES.STATE_VALIDATION_FAILED, message, {
843
+ errors: validation.errors,
844
+ retryable: false,
845
+ requires_human: true,
846
+ }),
847
+ };
848
+ }
849
+ }
850
+
851
+ private async validateSubmittedVerifiedManifest(
852
+ featureId: string,
853
+ verifiedManifest: unknown,
854
+ ): Promise<void> {
855
+ const validation = await this.port.validateSchema(
856
+ 'spec.manifest.verified.schema.json',
857
+ verifiedManifest,
858
+ );
859
+ if (!validation.valid) {
860
+ throw {
861
+ normalizedResponse: fail(
862
+ ERROR_CODES.INTAKE_SUBMISSION_INVALID,
863
+ 'verified manifest failed schema validation',
864
+ {
865
+ feature_id: featureId,
866
+ errors: validation.errors,
867
+ retryable: false,
868
+ requires_human: true,
869
+ suggested_next_actions: [
870
+ 'Normalize obligation ids to OBL-### before emitting INTAKE_SUBMISSION',
871
+ 'Use supported obligation kinds only',
872
+ 'For human-review additions, keep source_refs empty rather than adding unsupported fields',
873
+ ],
874
+ },
875
+ ),
876
+ };
877
+ }
878
+ }
879
+
880
+ private async syncIntakeState(
881
+ featureId: string,
882
+ patch: (frontMatter: StateRecord) => StateRecord,
883
+ ): Promise<void> {
884
+ await this.port.updateState(featureId, null, (frontMatter, body) =>
885
+ Promise.resolve({
886
+ frontMatter: patch(frontMatter),
887
+ body,
888
+ }),
889
+ );
890
+ }
891
+
892
+ private isAutoPromotionEnabled(): boolean {
893
+ const intake = readPlanningIntakePolicy(this.port.getPolicySnapshot());
894
+ return intake.allow_auto_promotion === true;
895
+ }
896
+
897
+ private readAutoPromotionMinConfidence(): ManifestConfidence {
898
+ const intake = readPlanningIntakePolicy(this.port.getPolicySnapshot());
899
+ return normalizeManifestConfidence(intake.auto_promotion_min_confidence) ?? 'high';
900
+ }
901
+
902
+ private canAutoPromoteBootstrapManifest(bootstrapManifest: BootstrapManifest): boolean {
903
+ return (
904
+ this.isAutoPromotionEnabled() &&
905
+ bootstrapManifest.ambiguities.length === 0 &&
906
+ obligationsMeetConfidenceThreshold(
907
+ bootstrapManifest.obligations,
908
+ this.readAutoPromotionMinConfidence(),
909
+ )
910
+ );
911
+ }
912
+
913
+ private async readQuestionEvidence(featureId: string): Promise<QuestionEvidenceRecord[]> {
914
+ const artifact = await readJson(this.questionsPath(featureId), null);
915
+ const items = Array.isArray(artifact?.items) ? artifact.items : [];
916
+ return items
917
+ .map((item) => normalizeQuestionEvidence(item))
918
+ .filter((item): item is QuestionEvidenceRecord => item !== null);
919
+ }
920
+
921
+ private async buildSyncedReview(
922
+ featureId: string,
923
+ bootstrapManifest: BootstrapManifest,
924
+ verifiedManifest: VerifiedManifest | null,
925
+ ): Promise<IntakeReview> {
926
+ const existingReview =
927
+ (await this.readIntakeReview(featureId)) ?? buildDefaultReview(featureId, bootstrapManifest);
928
+ const ambiguities = mergeReviewAmbiguities(
929
+ bootstrapManifest.ambiguities,
930
+ existingReview.ambiguities,
931
+ );
932
+ const unverifiedReviewStatus =
933
+ existingReview.status === 'verified' && !verifiedManifest
934
+ ? 'proposed'
935
+ : existingReview.status;
936
+ const questions = await this.readQuestionEvidence(featureId);
937
+ if (questions.length === 0) {
938
+ return {
939
+ ...existingReview,
940
+ ambiguities,
941
+ clarification_answers: normalizeReviewClarificationAnswers(
942
+ existingReview.clarification_answers,
943
+ ambiguities,
944
+ ),
945
+ status: verifiedManifest ? 'verified' : unverifiedReviewStatus,
946
+ bootstrap_manifest_version: bootstrapManifest.manifest_version,
947
+ verified_manifest_version: verifiedManifest?.manifest_version ?? null,
948
+ promotion_basis:
949
+ verifiedManifest?.verification_basis ??
950
+ normalizeVerificationBasis(existingReview.promotion_basis) ??
951
+ null,
952
+ };
953
+ }
954
+ const openIntakeQuestions = questions.filter((question) => isOpenIntakeQuestion(question));
955
+ const clarificationAnswers = buildClarificationAnswers(
956
+ {
957
+ ...existingReview,
958
+ ambiguities,
959
+ },
960
+ questions,
961
+ );
962
+ return {
963
+ ...existingReview,
964
+ ambiguities,
965
+ status: verifiedManifest
966
+ ? 'verified'
967
+ : openIntakeQuestions.length > 0
968
+ ? 'awaiting_input'
969
+ : 'proposed',
970
+ questions_open: openIntakeQuestions.length,
971
+ questions_resolved: clarificationAnswers.length,
972
+ clarification_answers: clarificationAnswers,
973
+ bootstrap_manifest_version: bootstrapManifest.manifest_version,
974
+ verified_manifest_version: verifiedManifest?.manifest_version ?? null,
975
+ promotion_basis:
976
+ verifiedManifest?.verification_basis ??
977
+ normalizeVerificationBasis(existingReview.promotion_basis) ??
978
+ null,
979
+ last_updated_at: nowIso(),
980
+ };
981
+ }
982
+
983
+ async syncReviewFromQuestions(featureId: string): Promise<void> {
984
+ const bootstrapManifest = await this.readBootstrapManifest(featureId);
985
+ if (!bootstrapManifest) {
986
+ return;
987
+ }
988
+ const verifiedManifest = await this.readVerifiedManifest(featureId);
989
+ const nextReview = await this.buildSyncedReview(featureId, bootstrapManifest, verifiedManifest);
990
+ await this.validateArtifact(
991
+ 'intake.review.schema.json',
992
+ nextReview,
993
+ 'intake review failed schema validation',
994
+ );
995
+ await atomicWriteJson(this.intakeReviewPath(featureId), nextReview);
996
+ }
997
+
998
+ private async validateVerificationBasisEvidence(
999
+ featureId: string,
1000
+ submission: IntakeSubmission,
1001
+ bootstrapManifest: BootstrapManifest,
1002
+ ): Promise<void> {
1003
+ const verificationBasis = submission.verified_manifest.verification_basis;
1004
+ if (verificationBasis === 'policy_approved_auto_promotion') {
1005
+ return;
1006
+ }
1007
+
1008
+ if (
1009
+ verificationBasis === 'structured_spec' &&
1010
+ (bootstrapManifest.ambiguities.length > 0 || submission.resolved_ambiguities.length > 0)
1011
+ ) {
1012
+ throw {
1013
+ normalizedResponse: fail(
1014
+ ERROR_CODES.INTAKE_SUBMISSION_INVALID,
1015
+ 'structured_spec verification cannot resolve inferred intake ambiguities',
1016
+ {
1017
+ feature_id: featureId,
1018
+ bootstrap_ambiguity_ids: bootstrapManifest.ambiguities.map((item) => item.id),
1019
+ resolved_ambiguity_ids: submission.resolved_ambiguities.map((item) => item.id),
1020
+ retryable: false,
1021
+ requires_human: true,
1022
+ suggested_next_actions: [
1023
+ 'Emit REQUEST.action=ask_user_input for the blocking ambiguity',
1024
+ 'Use verification_basis=questions_resolved after the operator answers',
1025
+ ],
1026
+ },
1027
+ ),
1028
+ };
1029
+ }
1030
+
1031
+ if (
1032
+ verificationBasis !== 'questions_resolved' &&
1033
+ verificationBasis !== 'human_review_confirmed'
1034
+ ) {
1035
+ return;
1036
+ }
1037
+
1038
+ const questions = await this.readQuestionEvidence(featureId);
1039
+ const openIntakeQuestions = questions.filter(
1040
+ (question) =>
1041
+ question.status === 'open' &&
1042
+ (question.resume_status === STATUS.INTAKE ||
1043
+ question.resume_phase === STATUS.INTAKE ||
1044
+ question.phase === STATUS.INTAKE),
1045
+ );
1046
+ if (openIntakeQuestions.length > 0) {
1047
+ throw {
1048
+ normalizedResponse: fail(
1049
+ ERROR_CODES.INTAKE_SUBMISSION_INVALID,
1050
+ 'verified manifest cannot be promoted while intake questions remain open',
1051
+ {
1052
+ feature_id: featureId,
1053
+ open_question_count: openIntakeQuestions.length,
1054
+ retryable: false,
1055
+ requires_human: true,
1056
+ suggested_next_actions: [
1057
+ 'Wait for the operator to answer the open intake question',
1058
+ 'Resubmit INTAKE_SUBMISSION after the answered question is persisted',
1059
+ ],
1060
+ },
1061
+ ),
1062
+ };
1063
+ }
1064
+
1065
+ const answeredIntakeQuestions = questions.filter((question) =>
1066
+ isAnsweredIntakeQuestion(question, bootstrapManifest),
1067
+ );
1068
+ if (answeredIntakeQuestions.length === 0) {
1069
+ throw {
1070
+ normalizedResponse: fail(
1071
+ ERROR_CODES.INTAKE_SUBMISSION_INVALID,
1072
+ 'verified manifest requires persisted answered intake-question evidence',
1073
+ {
1074
+ feature_id: featureId,
1075
+ verification_basis: verificationBasis,
1076
+ retryable: false,
1077
+ requires_human: true,
1078
+ suggested_next_actions: [
1079
+ 'Emit REQUEST.action=ask_user_input instead of resolving the ambiguity yourself',
1080
+ 'Resubmit INTAKE_SUBMISSION only after the operator answer is stored in questions.json',
1081
+ ],
1082
+ },
1083
+ ),
1084
+ };
1085
+ }
1086
+
1087
+ if (verificationBasis === 'questions_resolved') {
1088
+ if (submission.resolved_ambiguities.length === 0) {
1089
+ throw {
1090
+ normalizedResponse: fail(
1091
+ ERROR_CODES.INTAKE_SUBMISSION_INVALID,
1092
+ 'questions_resolved verification requires at least one resolved ambiguity',
1093
+ {
1094
+ feature_id: featureId,
1095
+ retryable: false,
1096
+ requires_human: true,
1097
+ },
1098
+ ),
1099
+ };
1100
+ }
1101
+ const missingEvidence = submission.resolved_ambiguities
1102
+ .map((ambiguity) => ambiguity.id)
1103
+ .filter(
1104
+ (ambiguityId) =>
1105
+ !answeredIntakeQuestions.some((question) =>
1106
+ questionReferencesAmbiguity(question, ambiguityId),
1107
+ ),
1108
+ );
1109
+ if (missingEvidence.length > 0) {
1110
+ throw {
1111
+ normalizedResponse: fail(
1112
+ ERROR_CODES.INTAKE_SUBMISSION_INVALID,
1113
+ 'resolved intake ambiguities must map to answered intake questions',
1114
+ {
1115
+ feature_id: featureId,
1116
+ verification_basis: verificationBasis,
1117
+ missing_ambiguity_evidence_ids: missingEvidence,
1118
+ retryable: false,
1119
+ requires_human: true,
1120
+ suggested_next_actions: [
1121
+ 'Include ambiguity_id or ambiguity_ids on the blocking intake question',
1122
+ 'Only mark ambiguities resolved after their answered question is persisted',
1123
+ ],
1124
+ },
1125
+ ),
1126
+ };
1127
+ }
1128
+ return;
1129
+ }
1130
+
1131
+ if (!answeredIntakeQuestions.some((question) => questionConfirmsVerifiedManifest(question))) {
1132
+ throw {
1133
+ normalizedResponse: fail(
1134
+ ERROR_CODES.INTAKE_SUBMISSION_INVALID,
1135
+ 'human_review_confirmed requires an explicit answered manifest-review confirmation question',
1136
+ {
1137
+ feature_id: featureId,
1138
+ verification_basis: verificationBasis,
1139
+ retryable: false,
1140
+ requires_human: true,
1141
+ suggested_next_actions: [
1142
+ 'Use verification_basis=questions_resolved for answered ambiguity questions',
1143
+ 'Reserve human_review_confirmed for explicit manifest confirmation responses',
1144
+ ],
1145
+ },
1146
+ ),
1147
+ };
1148
+ }
1149
+ }
1150
+
1151
+ async ensureBootstrapManifest(featureId: string): Promise<BootstrapManifest> {
1152
+ const existing = await this.readBootstrapManifest(featureId);
1153
+ if (existing) {
1154
+ return existing;
1155
+ }
1156
+
1157
+ const specText = await fs.readFile(this.port.specPath(featureId), 'utf8').catch(() => '');
1158
+ const manifest = this.generator.generate(featureId, specText);
1159
+ await this.validateArtifact(
1160
+ 'spec.manifest.bootstrap.schema.json',
1161
+ manifest,
1162
+ 'bootstrap manifest failed schema validation',
1163
+ );
1164
+
1165
+ const persistedManifest = await this.port.withFeatureLock(featureId, async () => {
1166
+ const current = await this.readBootstrapManifest(featureId);
1167
+ if (current) {
1168
+ return current;
1169
+ }
1170
+
1171
+ await atomicWriteJson(this.bootstrapManifestPath(featureId), manifest);
1172
+
1173
+ const existingReview = await this.readIntakeReview(featureId);
1174
+ if (!existingReview) {
1175
+ const review = buildDefaultReview(featureId, manifest);
1176
+ await this.validateArtifact(
1177
+ 'intake.review.schema.json',
1178
+ review,
1179
+ 'intake review failed schema validation',
1180
+ );
1181
+ await atomicWriteJson(this.intakeReviewPath(featureId), review);
1182
+ }
1183
+
1184
+ return manifest;
1185
+ });
1186
+
1187
+ await this.syncReviewFromQuestions(featureId);
1188
+
1189
+ await this.syncIntakeState(featureId, (frontMatter) => ({
1190
+ ...frontMatter,
1191
+ intake: {
1192
+ ...readIntakeRecord(frontMatter),
1193
+ status: frontMatter.status === STATUS.AWAITING_INPUT ? 'awaiting_input' : 'in_progress',
1194
+ bootstrap_manifest_version: persistedManifest.manifest_version,
1195
+ verified_manifest_version:
1196
+ typeof readIntakeRecord(frontMatter).verified_manifest_version === 'number'
1197
+ ? readIntakeRecord(frontMatter).verified_manifest_version
1198
+ : null,
1199
+ open_ambiguity_count: persistedManifest.ambiguities.filter((item) => item.status === 'open')
1200
+ .length,
1201
+ last_verified_at:
1202
+ typeof readIntakeRecord(frontMatter).last_verified_at === 'string'
1203
+ ? readIntakeRecord(frontMatter).last_verified_at
1204
+ : null,
1205
+ promotion_basis:
1206
+ typeof readIntakeRecord(frontMatter).promotion_basis === 'string'
1207
+ ? readIntakeRecord(frontMatter).promotion_basis
1208
+ : null,
1209
+ },
1210
+ }));
1211
+
1212
+ return persistedManifest;
1213
+ }
1214
+
1215
+ async ensureIntakeArtifacts(featureId: string): Promise<{
1216
+ bootstrapManifest: BootstrapManifest;
1217
+ verifiedManifest: VerifiedManifest | null;
1218
+ review: IntakeReview;
1219
+ summary: IntakeSummary;
1220
+ }> {
1221
+ const bootstrapManifest = await this.ensureBootstrapManifest(featureId);
1222
+ await this.syncReviewFromQuestions(featureId);
1223
+ const [initialVerifiedManifest, initialReview, initialState] = await Promise.all([
1224
+ this.readVerifiedManifest(featureId),
1225
+ this.readIntakeReview(featureId),
1226
+ this.port.readState(featureId),
1227
+ ]);
1228
+ let verifiedManifest = initialVerifiedManifest;
1229
+ let review = initialReview ?? buildDefaultReview(featureId, bootstrapManifest);
1230
+ let state = initialState;
1231
+
1232
+ if (!verifiedManifest && this.canAutoPromoteBootstrapManifest(bootstrapManifest)) {
1233
+ await this.intakeSubmit(featureId, {
1234
+ verified_manifest: {
1235
+ feature_id: featureId,
1236
+ manifest_version: bootstrapManifest.manifest_version,
1237
+ artifact_type: 'verified',
1238
+ verification_basis: 'policy_approved_auto_promotion',
1239
+ verified_at: nowIso(),
1240
+ source_bootstrap_version: bootstrapManifest.manifest_version,
1241
+ obligations: bootstrapManifest.obligations,
1242
+ },
1243
+ open_ambiguities: [],
1244
+ resolved_ambiguities: [],
1245
+ requires_user_input: false,
1246
+ });
1247
+ [verifiedManifest, review, state] = await Promise.all([
1248
+ this.readVerifiedManifest(featureId),
1249
+ this.readIntakeReview(featureId),
1250
+ this.port.readState(featureId),
1251
+ ]);
1252
+ }
1253
+
1254
+ return {
1255
+ bootstrapManifest,
1256
+ verifiedManifest,
1257
+ review: review ?? buildDefaultReview(featureId, bootstrapManifest),
1258
+ summary: summarizeIntake(state.frontMatter, bootstrapManifest, verifiedManifest, review),
1259
+ };
1260
+ }
1261
+
1262
+ async intakeSubmit(
1263
+ featureId: string,
1264
+ submission: IntakeSubmission,
1265
+ ): Promise<{
1266
+ data: {
1267
+ feature_id: string;
1268
+ accepted: boolean;
1269
+ feature_status: string;
1270
+ verified_manifest_version: number;
1271
+ };
1272
+ }> {
1273
+ if (!submission || !submission.verified_manifest) {
1274
+ throw {
1275
+ normalizedResponse: fail(
1276
+ ERROR_CODES.INVALID_ARGUMENT,
1277
+ 'verified_manifest is required for intake submission',
1278
+ {
1279
+ retryable: false,
1280
+ requires_human: true,
1281
+ },
1282
+ ),
1283
+ };
1284
+ }
1285
+
1286
+ const normalizedVerifiedManifest = this.normalizeSubmittedVerifiedManifest(
1287
+ featureId,
1288
+ submission.verified_manifest,
1289
+ );
1290
+ const verifiedManifest = normalizedVerifiedManifest as unknown as VerifiedManifest;
1291
+ if (verifiedManifest.feature_id !== featureId) {
1292
+ throw {
1293
+ normalizedResponse: fail(
1294
+ ERROR_CODES.INVALID_ARGUMENT,
1295
+ 'verified_manifest.feature_id must match feature_id',
1296
+ {
1297
+ feature_id: featureId,
1298
+ manifest_feature_id: verifiedManifest.feature_id,
1299
+ retryable: false,
1300
+ requires_human: true,
1301
+ },
1302
+ ),
1303
+ };
1304
+ }
1305
+
1306
+ await this.validateSubmittedVerifiedManifest(featureId, normalizedVerifiedManifest);
1307
+ const bootstrapManifest = await this.ensureBootstrapManifest(featureId);
1308
+ const normalizedSubmission: IntakeSubmission = {
1309
+ verified_manifest: verifiedManifest,
1310
+ open_ambiguities: normalizeSubmissionAmbiguities(
1311
+ submission.open_ambiguities,
1312
+ 'open',
1313
+ bootstrapManifest,
1314
+ ),
1315
+ resolved_ambiguities: normalizeSubmissionAmbiguities(
1316
+ submission.resolved_ambiguities,
1317
+ 'resolved',
1318
+ bootstrapManifest,
1319
+ ),
1320
+ requires_user_input: submission.requires_user_input === true,
1321
+ };
1322
+ if (
1323
+ normalizedSubmission.requires_user_input === true ||
1324
+ normalizedSubmission.open_ambiguities.some((item) => item.status === 'open')
1325
+ ) {
1326
+ throw {
1327
+ normalizedResponse: fail(
1328
+ ERROR_CODES.INTAKE_SUBMISSION_INVALID,
1329
+ 'verified manifest cannot be promoted while intake ambiguities remain unresolved',
1330
+ {
1331
+ feature_id: featureId,
1332
+ requires_user_input: normalizedSubmission.requires_user_input,
1333
+ open_ambiguity_ids: normalizedSubmission.open_ambiguities
1334
+ .filter((item) => item.status === 'open')
1335
+ .map((item) => item.id),
1336
+ retryable: false,
1337
+ requires_human: true,
1338
+ suggested_next_actions: [
1339
+ 'Emit REQUEST.action=ask_user_input for the unresolved ambiguity',
1340
+ 'Resubmit INTAKE_SUBMISSION only after the ambiguity is resolved',
1341
+ ],
1342
+ },
1343
+ ),
1344
+ };
1345
+ }
1346
+ if (
1347
+ verifiedManifest.verification_basis === 'policy_approved_auto_promotion' &&
1348
+ !this.canAutoPromoteBootstrapManifest(bootstrapManifest)
1349
+ ) {
1350
+ throw {
1351
+ normalizedResponse: fail(
1352
+ ERROR_CODES.INTAKE_SUBMISSION_INVALID,
1353
+ 'policy-approved auto-promotion is not allowed for this bootstrap manifest',
1354
+ {
1355
+ feature_id: featureId,
1356
+ allow_auto_promotion: this.isAutoPromotionEnabled(),
1357
+ minimum_confidence: this.readAutoPromotionMinConfidence(),
1358
+ retryable: false,
1359
+ requires_human: true,
1360
+ },
1361
+ ),
1362
+ };
1363
+ }
1364
+ if (
1365
+ verifiedManifest.source_bootstrap_version !== null &&
1366
+ verifiedManifest.source_bootstrap_version !== bootstrapManifest.manifest_version
1367
+ ) {
1368
+ throw {
1369
+ normalizedResponse: fail(
1370
+ ERROR_CODES.INTAKE_SUBMISSION_INVALID,
1371
+ 'verified manifest must reference the current bootstrap manifest version',
1372
+ {
1373
+ feature_id: featureId,
1374
+ expected_source_bootstrap_version: bootstrapManifest.manifest_version,
1375
+ received_source_bootstrap_version: verifiedManifest.source_bootstrap_version,
1376
+ retryable: false,
1377
+ requires_human: true,
1378
+ },
1379
+ ),
1380
+ };
1381
+ }
1382
+ await this.validateVerificationBasisEvidence(
1383
+ featureId,
1384
+ normalizedSubmission,
1385
+ bootstrapManifest,
1386
+ );
1387
+
1388
+ const result = await this.port.withFeatureLock(featureId, async () => {
1389
+ const questionEvidence = await this.readQuestionEvidence(featureId);
1390
+ const ambiguities = [
1391
+ ...normalizedSubmission.open_ambiguities,
1392
+ ...normalizedSubmission.resolved_ambiguities,
1393
+ ];
1394
+ const clarificationAnswers = buildClarificationAnswers(
1395
+ {
1396
+ feature_id: featureId,
1397
+ version: ((await this.readIntakeReview(featureId))?.version ?? 0) + 1,
1398
+ status: 'verified',
1399
+ questions_open: 0,
1400
+ questions_resolved: 0,
1401
+ ambiguities,
1402
+ clarification_answers: [],
1403
+ bootstrap_manifest_version: bootstrapManifest.manifest_version,
1404
+ verified_manifest_version: verifiedManifest.manifest_version,
1405
+ promotion_basis: verifiedManifest.verification_basis,
1406
+ last_updated_at: nowIso(),
1407
+ },
1408
+ questionEvidence,
1409
+ );
1410
+ const review: IntakeReview = {
1411
+ feature_id: featureId,
1412
+ version: ((await this.readIntakeReview(featureId))?.version ?? 0) + 1,
1413
+ status: 'verified',
1414
+ questions_open: 0,
1415
+ questions_resolved: clarificationAnswers.length,
1416
+ ambiguities,
1417
+ clarification_answers: clarificationAnswers,
1418
+ bootstrap_manifest_version: bootstrapManifest.manifest_version,
1419
+ verified_manifest_version: verifiedManifest.manifest_version,
1420
+ promotion_basis: verifiedManifest.verification_basis,
1421
+ last_updated_at: nowIso(),
1422
+ };
1423
+ await this.validateArtifact(
1424
+ 'intake.review.schema.json',
1425
+ review,
1426
+ 'intake review failed schema validation',
1427
+ );
1428
+
1429
+ await atomicWriteJson(this.verifiedManifestPath(featureId), verifiedManifest);
1430
+ await atomicWriteJson(this.intakeReviewPath(featureId), review);
1431
+ await Promise.all([
1432
+ fs.rm(this.planPath(featureId), { force: true }),
1433
+ fs.rm(this.planMarkdownPath(featureId), { force: true }),
1434
+ ]);
1435
+
1436
+ return {
1437
+ verified_manifest_version: verifiedManifest.manifest_version,
1438
+ };
1439
+ });
1440
+
1441
+ await this.syncIntakeState(featureId, (frontMatter) => ({
1442
+ ...frontMatter,
1443
+ status: STATUS.PLANNING,
1444
+ gates: {
1445
+ ...(asRecord(frontMatter.gates) ?? {}),
1446
+ plan: GATE_RESULT.NA,
1447
+ },
1448
+ intake: {
1449
+ ...readIntakeRecord(frontMatter),
1450
+ status: 'verified',
1451
+ bootstrap_manifest_version: bootstrapManifest.manifest_version,
1452
+ verified_manifest_version: verifiedManifest.manifest_version,
1453
+ open_ambiguity_count: 0,
1454
+ last_verified_at: verifiedManifest.verified_at,
1455
+ promotion_basis: verifiedManifest.verification_basis,
1456
+ },
1457
+ }));
1458
+
1459
+ return {
1460
+ data: {
1461
+ feature_id: featureId,
1462
+ accepted: true,
1463
+ feature_status: STATUS.PLANNING,
1464
+ verified_manifest_version: result.verified_manifest_version,
1465
+ },
1466
+ };
1467
+ }
1468
+ }