agentic-orchestrator 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (439) hide show
  1. package/.dockerignore +24 -0
  2. package/.github/workflows/mcp-contract-validation.yml +38 -0
  3. package/Agentic-Orchestrator.iml +9 -0
  4. package/LICENSE +21 -0
  5. package/README.md +679 -0
  6. package/agentic/orchestrator/agents.yaml +14 -0
  7. package/agentic/orchestrator/gates.yaml +31 -0
  8. package/agentic/orchestrator/policy.yaml +145 -0
  9. package/agentic/orchestrator/prompts/builder.system.md +1 -0
  10. package/agentic/orchestrator/prompts/planner.system.md +15 -0
  11. package/agentic/orchestrator/prompts/qa.system.md +1 -0
  12. package/agentic/orchestrator/schemas/agents.schema.json +49 -0
  13. package/agentic/orchestrator/schemas/gates.schema.json +65 -0
  14. package/agentic/orchestrator/schemas/index.schema.json +108 -0
  15. package/agentic/orchestrator/schemas/plan.schema.json +127 -0
  16. package/agentic/orchestrator/schemas/policy.schema.json +227 -0
  17. package/agentic/orchestrator/schemas/qa_test_index.schema.json +53 -0
  18. package/agentic/orchestrator/schemas/state.schema.json +92 -0
  19. package/agentic/orchestrator/tools/catalog.json +399 -0
  20. package/agentic/orchestrator/tools/errors.schema.json +21 -0
  21. package/agentic/orchestrator/tools/protocol.json +8 -0
  22. package/agentic/orchestrator/tools/schemas/input/collisions.scan.input.schema.json +7 -0
  23. package/agentic/orchestrator/tools/schemas/input/evidence.latest.input.schema.json +15 -0
  24. package/agentic/orchestrator/tools/schemas/input/feature.delete.input.schema.json +42 -0
  25. package/agentic/orchestrator/tools/schemas/input/feature.discover_specs.input.schema.json +7 -0
  26. package/agentic/orchestrator/tools/schemas/input/feature.get_context.input.schema.json +15 -0
  27. package/agentic/orchestrator/tools/schemas/input/feature.init.input.schema.json +21 -0
  28. package/agentic/orchestrator/tools/schemas/input/feature.log_append.input.schema.json +26 -0
  29. package/agentic/orchestrator/tools/schemas/input/feature.ready_to_merge.input.schema.json +34 -0
  30. package/agentic/orchestrator/tools/schemas/input/feature.state_get.input.schema.json +15 -0
  31. package/agentic/orchestrator/tools/schemas/input/feature.state_patch.input.schema.json +28 -0
  32. package/agentic/orchestrator/tools/schemas/input/gates.list.input.schema.json +11 -0
  33. package/agentic/orchestrator/tools/schemas/input/gates.run.input.schema.json +29 -0
  34. package/agentic/orchestrator/tools/schemas/input/locks.acquire.input.schema.json +29 -0
  35. package/agentic/orchestrator/tools/schemas/input/locks.release.input.schema.json +26 -0
  36. package/agentic/orchestrator/tools/schemas/input/mutating.schema.json +14 -0
  37. package/agentic/orchestrator/tools/schemas/input/plan.get.input.schema.json +15 -0
  38. package/agentic/orchestrator/tools/schemas/input/plan.submit.input.schema.json +28 -0
  39. package/agentic/orchestrator/tools/schemas/input/plan.update.input.schema.json +29 -0
  40. package/agentic/orchestrator/tools/schemas/input/qa.test_index_get.input.schema.json +15 -0
  41. package/agentic/orchestrator/tools/schemas/input/qa.test_index_update.input.schema.json +38 -0
  42. package/agentic/orchestrator/tools/schemas/input/read.schema.json +6 -0
  43. package/agentic/orchestrator/tools/schemas/input/repo.apply_patch.input.schema.json +25 -0
  44. package/agentic/orchestrator/tools/schemas/input/repo.diff.input.schema.json +21 -0
  45. package/agentic/orchestrator/tools/schemas/input/repo.diff_bundle.input.schema.json +15 -0
  46. package/agentic/orchestrator/tools/schemas/input/repo.ensure_worktree.input.schema.json +21 -0
  47. package/agentic/orchestrator/tools/schemas/input/repo.read_file.input.schema.json +20 -0
  48. package/agentic/orchestrator/tools/schemas/input/repo.search.input.schema.json +20 -0
  49. package/agentic/orchestrator/tools/schemas/input/repo.status.input.schema.json +15 -0
  50. package/agentic/orchestrator/tools/schemas/input/report.dashboard.input.schema.json +7 -0
  51. package/agentic/orchestrator/tools/schemas/input/report.feature_summary.input.schema.json +15 -0
  52. package/agentic/orchestrator/tools/schemas/output/collisions.scan.output.schema.json +17 -0
  53. package/agentic/orchestrator/tools/schemas/output/evidence.latest.output.schema.json +20 -0
  54. package/agentic/orchestrator/tools/schemas/output/feature.delete.output.schema.json +224 -0
  55. package/agentic/orchestrator/tools/schemas/output/feature.discover_specs.output.schema.json +32 -0
  56. package/agentic/orchestrator/tools/schemas/output/feature.get_context.output.schema.json +40 -0
  57. package/agentic/orchestrator/tools/schemas/output/feature.init.output.schema.json +24 -0
  58. package/agentic/orchestrator/tools/schemas/output/feature.log_append.output.schema.json +24 -0
  59. package/agentic/orchestrator/tools/schemas/output/feature.ready_to_merge.output.schema.json +30 -0
  60. package/agentic/orchestrator/tools/schemas/output/feature.state_get.output.schema.json +18 -0
  61. package/agentic/orchestrator/tools/schemas/output/feature.state_patch.output.schema.json +24 -0
  62. package/agentic/orchestrator/tools/schemas/output/gates.list.output.schema.json +42 -0
  63. package/agentic/orchestrator/tools/schemas/output/gates.run.output.schema.json +37 -0
  64. package/agentic/orchestrator/tools/schemas/output/locks.acquire.output.schema.json +34 -0
  65. package/agentic/orchestrator/tools/schemas/output/locks.release.output.schema.json +24 -0
  66. package/agentic/orchestrator/tools/schemas/output/plan.get.output.schema.json +26 -0
  67. package/agentic/orchestrator/tools/schemas/output/plan.submit.output.schema.json +23 -0
  68. package/agentic/orchestrator/tools/schemas/output/plan.update.output.schema.json +23 -0
  69. package/agentic/orchestrator/tools/schemas/output/qa.test_index_get.output.schema.json +22 -0
  70. package/agentic/orchestrator/tools/schemas/output/qa.test_index_update.output.schema.json +19 -0
  71. package/agentic/orchestrator/tools/schemas/output/repo.apply_patch.output.schema.json +33 -0
  72. package/agentic/orchestrator/tools/schemas/output/repo.diff.output.schema.json +19 -0
  73. package/agentic/orchestrator/tools/schemas/output/repo.diff_bundle.output.schema.json +32 -0
  74. package/agentic/orchestrator/tools/schemas/output/repo.ensure_worktree.output.schema.json +29 -0
  75. package/agentic/orchestrator/tools/schemas/output/repo.read_file.output.schema.json +24 -0
  76. package/agentic/orchestrator/tools/schemas/output/repo.search.output.schema.json +26 -0
  77. package/agentic/orchestrator/tools/schemas/output/repo.status.output.schema.json +27 -0
  78. package/agentic/orchestrator/tools/schemas/output/report.dashboard.output.schema.json +21 -0
  79. package/agentic/orchestrator/tools/schemas/output/report.feature_summary.output.schema.json +36 -0
  80. package/agentic/orchestrator/tools/schemas/output/standard_success.schema.json +6 -0
  81. package/agentic/orchestrator/tools.md +32 -0
  82. package/apps/control-plane/project.json +39 -0
  83. package/apps/control-plane/scripts/validate-architecture-rules.mjs +170 -0
  84. package/apps/control-plane/scripts/validate-docker-mcp-contract.mjs +84 -0
  85. package/apps/control-plane/scripts/validate-mcp-contracts.ts +61 -0
  86. package/apps/control-plane/src/application/services/collision-queue-service.ts +227 -0
  87. package/apps/control-plane/src/application/services/feature-deletion-service.ts +459 -0
  88. package/apps/control-plane/src/application/services/feature-lifecycle-service.ts +177 -0
  89. package/apps/control-plane/src/application/services/feature-state-service.ts +125 -0
  90. package/apps/control-plane/src/application/services/gate-service.ts +232 -0
  91. package/apps/control-plane/src/application/services/lock-service.ts +298 -0
  92. package/apps/control-plane/src/application/services/merge-service.ts +246 -0
  93. package/apps/control-plane/src/application/services/patch-service.ts +259 -0
  94. package/apps/control-plane/src/application/services/plan-service.ts +302 -0
  95. package/apps/control-plane/src/application/services/qa-index-service.ts +98 -0
  96. package/apps/control-plane/src/application/services/reporting-service.ts +120 -0
  97. package/apps/control-plane/src/application/services/run-lease-service.ts +340 -0
  98. package/apps/control-plane/src/application/tools/tool-metadata.ts +56 -0
  99. package/apps/control-plane/src/application/tools/tool-router.ts +43 -0
  100. package/apps/control-plane/src/cli/aop.ts +31 -0
  101. package/apps/control-plane/src/cli/cli-argument-parser.ts +116 -0
  102. package/apps/control-plane/src/cli/delete-command-handler.ts +90 -0
  103. package/apps/control-plane/src/cli/io.ts +14 -0
  104. package/apps/control-plane/src/cli/resume-command-handler.ts +228 -0
  105. package/apps/control-plane/src/cli/run-command-handler.ts +57 -0
  106. package/apps/control-plane/src/cli/spec-ingestion-service.ts +88 -0
  107. package/apps/control-plane/src/cli/spec-input-resolver.ts +95 -0
  108. package/apps/control-plane/src/cli/spec-utils.ts +40 -0
  109. package/apps/control-plane/src/cli/status-command-handler.ts +17 -0
  110. package/apps/control-plane/src/cli/stop-command-handler.ts +5 -0
  111. package/apps/control-plane/src/cli/tooling.ts +36 -0
  112. package/apps/control-plane/src/cli/types.ts +34 -0
  113. package/apps/control-plane/src/core/collisions.ts +121 -0
  114. package/apps/control-plane/src/core/constants.ts +72 -0
  115. package/apps/control-plane/src/core/error-codes.ts +54 -0
  116. package/apps/control-plane/src/core/frontmatter.ts +42 -0
  117. package/apps/control-plane/src/core/fs.ts +173 -0
  118. package/apps/control-plane/src/core/gates.ts +361 -0
  119. package/apps/control-plane/src/core/git.ts +115 -0
  120. package/apps/control-plane/src/core/kernel.ts +1077 -0
  121. package/apps/control-plane/src/core/patch.ts +152 -0
  122. package/apps/control-plane/src/core/path-layout.ts +113 -0
  123. package/apps/control-plane/src/core/path-rules.ts +71 -0
  124. package/apps/control-plane/src/core/qa-index.ts +179 -0
  125. package/apps/control-plane/src/core/response.ts +62 -0
  126. package/apps/control-plane/src/core/runtime-sessions.ts +20 -0
  127. package/apps/control-plane/src/core/schemas.ts +125 -0
  128. package/apps/control-plane/src/index.ts +21 -0
  129. package/apps/control-plane/src/interfaces/cli/bootstrap.ts +100 -0
  130. package/apps/control-plane/src/mcp/kernel-tool-executor.ts +39 -0
  131. package/apps/control-plane/src/mcp/mcp-server-adapter.ts +74 -0
  132. package/apps/control-plane/src/mcp/operation-ledger.ts +108 -0
  133. package/apps/control-plane/src/mcp/protocol-contract.ts +9 -0
  134. package/apps/control-plane/src/mcp/runtime-factory.ts +105 -0
  135. package/apps/control-plane/src/mcp/runtime-types.ts +44 -0
  136. package/apps/control-plane/src/mcp/token-auth-verifier.ts +63 -0
  137. package/apps/control-plane/src/mcp/token-claims-validator.ts +72 -0
  138. package/apps/control-plane/src/mcp/token-codec.ts +62 -0
  139. package/apps/control-plane/src/mcp/tool-authorizer.ts +43 -0
  140. package/apps/control-plane/src/mcp/tool-client.ts +78 -0
  141. package/apps/control-plane/src/mcp/tool-contract-validator.ts +83 -0
  142. package/apps/control-plane/src/mcp/tool-registry-loader.ts +135 -0
  143. package/apps/control-plane/src/mcp/tool-runtime.ts +336 -0
  144. package/apps/control-plane/src/mcp/tools-markdown-generator.ts +26 -0
  145. package/apps/control-plane/src/mcp/transport-types.ts +16 -0
  146. package/apps/control-plane/src/mcp/types.ts +2 -0
  147. package/apps/control-plane/src/providers/providers.ts +177 -0
  148. package/apps/control-plane/src/supervisor/build-wave-executor.ts +55 -0
  149. package/apps/control-plane/src/supervisor/lease-heartbeat-service.ts +22 -0
  150. package/apps/control-plane/src/supervisor/planning-wave-executor.ts +316 -0
  151. package/apps/control-plane/src/supervisor/prompt-bundle-loader.ts +62 -0
  152. package/apps/control-plane/src/supervisor/qa-wave-executor.ts +99 -0
  153. package/apps/control-plane/src/supervisor/run-coordinator.ts +224 -0
  154. package/apps/control-plane/src/supervisor/runtime.ts +347 -0
  155. package/apps/control-plane/src/supervisor/session-orchestrator.ts +268 -0
  156. package/apps/control-plane/src/supervisor/types.ts +149 -0
  157. package/apps/control-plane/src/supervisor/worker-decision-loop.ts +299 -0
  158. package/apps/control-plane/test/aop.spec.ts +101 -0
  159. package/apps/control-plane/test/cli-helpers.spec.ts +102 -0
  160. package/apps/control-plane/test/cli.spec.ts +12 -0
  161. package/apps/control-plane/test/cli.unit.spec.ts +609 -0
  162. package/apps/control-plane/test/collision-queue.spec.ts +158 -0
  163. package/apps/control-plane/test/collisions.spec.ts +138 -0
  164. package/apps/control-plane/test/core-utils.spec.ts +102 -0
  165. package/apps/control-plane/test/delete-command-handler.spec.ts +202 -0
  166. package/apps/control-plane/test/epoch-tracking.spec.ts +121 -0
  167. package/apps/control-plane/test/gates.spec.ts +452 -0
  168. package/apps/control-plane/test/helpers.ts +68 -0
  169. package/apps/control-plane/test/index.spec.ts +18 -0
  170. package/apps/control-plane/test/kernel-collision-replay.spec.ts +222 -0
  171. package/apps/control-plane/test/kernel.branches.spec.ts +321 -0
  172. package/apps/control-plane/test/kernel.coverage.spec.ts +408 -0
  173. package/apps/control-plane/test/kernel.spec.ts +369 -0
  174. package/apps/control-plane/test/mcp-helpers.spec.ts +195 -0
  175. package/apps/control-plane/test/mcp.spec.ts +776 -0
  176. package/apps/control-plane/test/merge-service.spec.ts +357 -0
  177. package/apps/control-plane/test/plan-service.spec.ts +195 -0
  178. package/apps/control-plane/test/planning-wave-executor.spec.ts +229 -0
  179. package/apps/control-plane/test/providers.spec.ts +168 -0
  180. package/apps/control-plane/test/qa-index-service.spec.ts +187 -0
  181. package/apps/control-plane/test/qa-index.spec.ts +317 -0
  182. package/apps/control-plane/test/response.spec.ts +55 -0
  183. package/apps/control-plane/test/run-coordinator.spec.ts +334 -0
  184. package/apps/control-plane/test/schema-date-time.spec.ts +170 -0
  185. package/apps/control-plane/test/service-retry-paths.spec.ts +305 -0
  186. package/apps/control-plane/test/services.spec.ts +693 -0
  187. package/apps/control-plane/test/spec-input-resolver.spec.ts +76 -0
  188. package/apps/control-plane/test/supervisor-collaborators.spec.ts +201 -0
  189. package/apps/control-plane/test/supervisor.calltool.spec.ts +120 -0
  190. package/apps/control-plane/test/supervisor.spec.ts +415 -0
  191. package/apps/control-plane/test/supervisor.unit.spec.ts +522 -0
  192. package/apps/control-plane/test/token-auth-verifier.spec.ts +111 -0
  193. package/apps/control-plane/test/tool-registry-loader.spec.ts +268 -0
  194. package/apps/control-plane/test/tool-runtime.spec.ts +294 -0
  195. package/apps/control-plane/test/worker-decision-loop.spec.ts +587 -0
  196. package/apps/control-plane/tsconfig.build.json +17 -0
  197. package/apps/control-plane/tsconfig.json +11 -0
  198. package/apps/control-plane/vitest.config.ts +28 -0
  199. package/dist/apps/control-plane/application/services/collision-queue-service.d.ts +69 -0
  200. package/dist/apps/control-plane/application/services/collision-queue-service.js +158 -0
  201. package/dist/apps/control-plane/application/services/collision-queue-service.js.map +1 -0
  202. package/dist/apps/control-plane/application/services/feature-deletion-service.d.ts +79 -0
  203. package/dist/apps/control-plane/application/services/feature-deletion-service.js +336 -0
  204. package/dist/apps/control-plane/application/services/feature-deletion-service.js.map +1 -0
  205. package/dist/apps/control-plane/application/services/feature-lifecycle-service.d.ts +81 -0
  206. package/dist/apps/control-plane/application/services/feature-lifecycle-service.js +117 -0
  207. package/dist/apps/control-plane/application/services/feature-lifecycle-service.js.map +1 -0
  208. package/dist/apps/control-plane/application/services/feature-state-service.d.ts +34 -0
  209. package/dist/apps/control-plane/application/services/feature-state-service.js +90 -0
  210. package/dist/apps/control-plane/application/services/feature-state-service.js.map +1 -0
  211. package/dist/apps/control-plane/application/services/gate-service.d.ts +46 -0
  212. package/dist/apps/control-plane/application/services/gate-service.js +160 -0
  213. package/dist/apps/control-plane/application/services/gate-service.js.map +1 -0
  214. package/dist/apps/control-plane/application/services/lock-service.d.ts +56 -0
  215. package/dist/apps/control-plane/application/services/lock-service.js +242 -0
  216. package/dist/apps/control-plane/application/services/lock-service.js.map +1 -0
  217. package/dist/apps/control-plane/application/services/merge-service.d.ts +33 -0
  218. package/dist/apps/control-plane/application/services/merge-service.js +194 -0
  219. package/dist/apps/control-plane/application/services/merge-service.js.map +1 -0
  220. package/dist/apps/control-plane/application/services/patch-service.d.ts +39 -0
  221. package/dist/apps/control-plane/application/services/patch-service.js +189 -0
  222. package/dist/apps/control-plane/application/services/patch-service.js.map +1 -0
  223. package/dist/apps/control-plane/application/services/plan-service.d.ts +60 -0
  224. package/dist/apps/control-plane/application/services/plan-service.js +234 -0
  225. package/dist/apps/control-plane/application/services/plan-service.js.map +1 -0
  226. package/dist/apps/control-plane/application/services/qa-index-service.d.ts +26 -0
  227. package/dist/apps/control-plane/application/services/qa-index-service.js +66 -0
  228. package/dist/apps/control-plane/application/services/qa-index-service.js.map +1 -0
  229. package/dist/apps/control-plane/application/services/reporting-service.d.ts +47 -0
  230. package/dist/apps/control-plane/application/services/reporting-service.js +90 -0
  231. package/dist/apps/control-plane/application/services/reporting-service.js.map +1 -0
  232. package/dist/apps/control-plane/application/services/run-lease-service.d.ts +74 -0
  233. package/dist/apps/control-plane/application/services/run-lease-service.js +263 -0
  234. package/dist/apps/control-plane/application/services/run-lease-service.js.map +1 -0
  235. package/dist/apps/control-plane/application/tools/tool-metadata.d.ts +8 -0
  236. package/dist/apps/control-plane/application/tools/tool-metadata.js +37 -0
  237. package/dist/apps/control-plane/application/tools/tool-metadata.js.map +1 -0
  238. package/dist/apps/control-plane/application/tools/tool-router.d.ts +16 -0
  239. package/dist/apps/control-plane/application/tools/tool-router.js +25 -0
  240. package/dist/apps/control-plane/application/tools/tool-router.js.map +1 -0
  241. package/dist/apps/control-plane/cli/aop.d.ts +5 -0
  242. package/dist/apps/control-plane/cli/aop.js +19 -0
  243. package/dist/apps/control-plane/cli/aop.js.map +1 -0
  244. package/dist/apps/control-plane/cli/cli-argument-parser.d.ts +5 -0
  245. package/dist/apps/control-plane/cli/cli-argument-parser.js +109 -0
  246. package/dist/apps/control-plane/cli/cli-argument-parser.js.map +1 -0
  247. package/dist/apps/control-plane/cli/delete-command-handler.d.ts +8 -0
  248. package/dist/apps/control-plane/cli/delete-command-handler.js +77 -0
  249. package/dist/apps/control-plane/cli/delete-command-handler.js.map +1 -0
  250. package/dist/apps/control-plane/cli/io.d.ts +2 -0
  251. package/dist/apps/control-plane/cli/io.js +14 -0
  252. package/dist/apps/control-plane/cli/io.js.map +1 -0
  253. package/dist/apps/control-plane/cli/resume-command-handler.d.ts +17 -0
  254. package/dist/apps/control-plane/cli/resume-command-handler.js +178 -0
  255. package/dist/apps/control-plane/cli/resume-command-handler.js.map +1 -0
  256. package/dist/apps/control-plane/cli/run-command-handler.d.ts +15 -0
  257. package/dist/apps/control-plane/cli/run-command-handler.js +39 -0
  258. package/dist/apps/control-plane/cli/run-command-handler.js.map +1 -0
  259. package/dist/apps/control-plane/cli/spec-ingestion-service.d.ts +8 -0
  260. package/dist/apps/control-plane/cli/spec-ingestion-service.js +77 -0
  261. package/dist/apps/control-plane/cli/spec-ingestion-service.js.map +1 -0
  262. package/dist/apps/control-plane/cli/spec-input-resolver.d.ts +9 -0
  263. package/dist/apps/control-plane/cli/spec-input-resolver.js +81 -0
  264. package/dist/apps/control-plane/cli/spec-input-resolver.js.map +1 -0
  265. package/dist/apps/control-plane/cli/spec-utils.d.ts +3 -0
  266. package/dist/apps/control-plane/cli/spec-utils.js +36 -0
  267. package/dist/apps/control-plane/cli/spec-utils.js.map +1 -0
  268. package/dist/apps/control-plane/cli/status-command-handler.d.ts +7 -0
  269. package/dist/apps/control-plane/cli/status-command-handler.js +14 -0
  270. package/dist/apps/control-plane/cli/status-command-handler.js.map +1 -0
  271. package/dist/apps/control-plane/cli/stop-command-handler.d.ts +3 -0
  272. package/dist/apps/control-plane/cli/stop-command-handler.js +6 -0
  273. package/dist/apps/control-plane/cli/stop-command-handler.js.map +1 -0
  274. package/dist/apps/control-plane/cli/tooling.d.ts +4 -0
  275. package/dist/apps/control-plane/cli/tooling.js +24 -0
  276. package/dist/apps/control-plane/cli/tooling.js.map +1 -0
  277. package/dist/apps/control-plane/cli/types.d.ts +31 -0
  278. package/dist/apps/control-plane/cli/types.js +2 -0
  279. package/dist/apps/control-plane/cli/types.js.map +1 -0
  280. package/dist/apps/control-plane/core/collisions.d.ts +39 -0
  281. package/dist/apps/control-plane/core/collisions.js +78 -0
  282. package/dist/apps/control-plane/core/collisions.js.map +1 -0
  283. package/dist/apps/control-plane/core/constants.d.ts +64 -0
  284. package/dist/apps/control-plane/core/constants.js +64 -0
  285. package/dist/apps/control-plane/core/constants.js.map +1 -0
  286. package/dist/apps/control-plane/core/error-codes.d.ts +50 -0
  287. package/dist/apps/control-plane/core/error-codes.js +52 -0
  288. package/dist/apps/control-plane/core/error-codes.js.map +1 -0
  289. package/dist/apps/control-plane/core/frontmatter.d.ts +11 -0
  290. package/dist/apps/control-plane/core/frontmatter.js +30 -0
  291. package/dist/apps/control-plane/core/frontmatter.js.map +1 -0
  292. package/dist/apps/control-plane/core/fs.d.ts +33 -0
  293. package/dist/apps/control-plane/core/fs.js +134 -0
  294. package/dist/apps/control-plane/core/fs.js.map +1 -0
  295. package/dist/apps/control-plane/core/gates.d.ts +88 -0
  296. package/dist/apps/control-plane/core/gates.js +229 -0
  297. package/dist/apps/control-plane/core/gates.js.map +1 -0
  298. package/dist/apps/control-plane/core/git.d.ts +31 -0
  299. package/dist/apps/control-plane/core/git.js +79 -0
  300. package/dist/apps/control-plane/core/git.js.map +1 -0
  301. package/dist/apps/control-plane/core/kernel.d.ts +445 -0
  302. package/dist/apps/control-plane/core/kernel.js +805 -0
  303. package/dist/apps/control-plane/core/kernel.js.map +1 -0
  304. package/dist/apps/control-plane/core/patch.d.ts +23 -0
  305. package/dist/apps/control-plane/core/patch.js +118 -0
  306. package/dist/apps/control-plane/core/patch.js.map +1 -0
  307. package/dist/apps/control-plane/core/path-layout.d.ts +23 -0
  308. package/dist/apps/control-plane/core/path-layout.js +90 -0
  309. package/dist/apps/control-plane/core/path-layout.js.map +1 -0
  310. package/dist/apps/control-plane/core/path-rules.d.ts +13 -0
  311. package/dist/apps/control-plane/core/path-rules.js +52 -0
  312. package/dist/apps/control-plane/core/path-rules.js.map +1 -0
  313. package/dist/apps/control-plane/core/qa-index.d.ts +53 -0
  314. package/dist/apps/control-plane/core/qa-index.js +112 -0
  315. package/dist/apps/control-plane/core/qa-index.js.map +1 -0
  316. package/dist/apps/control-plane/core/response.d.ts +19 -0
  317. package/dist/apps/control-plane/core/response.js +34 -0
  318. package/dist/apps/control-plane/core/response.js.map +1 -0
  319. package/dist/apps/control-plane/core/runtime-sessions.d.ts +19 -0
  320. package/dist/apps/control-plane/core/runtime-sessions.js +2 -0
  321. package/dist/apps/control-plane/core/runtime-sessions.js.map +1 -0
  322. package/dist/apps/control-plane/core/schemas.d.ts +23 -0
  323. package/dist/apps/control-plane/core/schemas.js +80 -0
  324. package/dist/apps/control-plane/core/schemas.js.map +1 -0
  325. package/dist/apps/control-plane/index.d.ts +11 -0
  326. package/dist/apps/control-plane/index.js +9 -0
  327. package/dist/apps/control-plane/index.js.map +1 -0
  328. package/dist/apps/control-plane/interfaces/cli/bootstrap.d.ts +2 -0
  329. package/dist/apps/control-plane/interfaces/cli/bootstrap.js +86 -0
  330. package/dist/apps/control-plane/interfaces/cli/bootstrap.js.map +1 -0
  331. package/dist/apps/control-plane/mcp/kernel-tool-executor.d.ts +14 -0
  332. package/dist/apps/control-plane/mcp/kernel-tool-executor.js +26 -0
  333. package/dist/apps/control-plane/mcp/kernel-tool-executor.js.map +1 -0
  334. package/dist/apps/control-plane/mcp/mcp-server-adapter.d.ts +19 -0
  335. package/dist/apps/control-plane/mcp/mcp-server-adapter.js +55 -0
  336. package/dist/apps/control-plane/mcp/mcp-server-adapter.js.map +1 -0
  337. package/dist/apps/control-plane/mcp/operation-ledger.d.ts +21 -0
  338. package/dist/apps/control-plane/mcp/operation-ledger.js +75 -0
  339. package/dist/apps/control-plane/mcp/operation-ledger.js.map +1 -0
  340. package/dist/apps/control-plane/mcp/protocol-contract.d.ts +8 -0
  341. package/dist/apps/control-plane/mcp/protocol-contract.js +9 -0
  342. package/dist/apps/control-plane/mcp/protocol-contract.js.map +1 -0
  343. package/dist/apps/control-plane/mcp/runtime-factory.d.ts +38 -0
  344. package/dist/apps/control-plane/mcp/runtime-factory.js +71 -0
  345. package/dist/apps/control-plane/mcp/runtime-factory.js.map +1 -0
  346. package/dist/apps/control-plane/mcp/runtime-types.d.ts +40 -0
  347. package/dist/apps/control-plane/mcp/runtime-types.js +2 -0
  348. package/dist/apps/control-plane/mcp/runtime-types.js.map +1 -0
  349. package/dist/apps/control-plane/mcp/token-auth-verifier.d.ts +24 -0
  350. package/dist/apps/control-plane/mcp/token-auth-verifier.js +45 -0
  351. package/dist/apps/control-plane/mcp/token-auth-verifier.js.map +1 -0
  352. package/dist/apps/control-plane/mcp/token-claims-validator.d.ts +9 -0
  353. package/dist/apps/control-plane/mcp/token-claims-validator.js +62 -0
  354. package/dist/apps/control-plane/mcp/token-claims-validator.js.map +1 -0
  355. package/dist/apps/control-plane/mcp/token-codec.d.ts +11 -0
  356. package/dist/apps/control-plane/mcp/token-codec.js +46 -0
  357. package/dist/apps/control-plane/mcp/token-codec.js.map +1 -0
  358. package/dist/apps/control-plane/mcp/tool-authorizer.d.ts +8 -0
  359. package/dist/apps/control-plane/mcp/tool-authorizer.js +36 -0
  360. package/dist/apps/control-plane/mcp/tool-authorizer.js.map +1 -0
  361. package/dist/apps/control-plane/mcp/tool-client.d.ts +30 -0
  362. package/dist/apps/control-plane/mcp/tool-client.js +50 -0
  363. package/dist/apps/control-plane/mcp/tool-client.js.map +1 -0
  364. package/dist/apps/control-plane/mcp/tool-contract-validator.d.ts +29 -0
  365. package/dist/apps/control-plane/mcp/tool-contract-validator.js +61 -0
  366. package/dist/apps/control-plane/mcp/tool-contract-validator.js.map +1 -0
  367. package/dist/apps/control-plane/mcp/tool-registry-loader.d.ts +15 -0
  368. package/dist/apps/control-plane/mcp/tool-registry-loader.js +109 -0
  369. package/dist/apps/control-plane/mcp/tool-registry-loader.js.map +1 -0
  370. package/dist/apps/control-plane/mcp/tool-runtime.d.ts +34 -0
  371. package/dist/apps/control-plane/mcp/tool-runtime.js +252 -0
  372. package/dist/apps/control-plane/mcp/tool-runtime.js.map +1 -0
  373. package/dist/apps/control-plane/mcp/tools-markdown-generator.d.ts +7 -0
  374. package/dist/apps/control-plane/mcp/tools-markdown-generator.js +22 -0
  375. package/dist/apps/control-plane/mcp/tools-markdown-generator.js.map +1 -0
  376. package/dist/apps/control-plane/mcp/transport-types.d.ts +14 -0
  377. package/dist/apps/control-plane/mcp/transport-types.js +2 -0
  378. package/dist/apps/control-plane/mcp/transport-types.js.map +1 -0
  379. package/dist/apps/control-plane/mcp/types.d.ts +2 -0
  380. package/dist/apps/control-plane/mcp/types.js +3 -0
  381. package/dist/apps/control-plane/mcp/types.js.map +1 -0
  382. package/dist/apps/control-plane/providers/providers.d.ts +72 -0
  383. package/dist/apps/control-plane/providers/providers.js +94 -0
  384. package/dist/apps/control-plane/providers/providers.js.map +1 -0
  385. package/dist/apps/control-plane/supervisor/build-wave-executor.d.ts +13 -0
  386. package/dist/apps/control-plane/supervisor/build-wave-executor.js +40 -0
  387. package/dist/apps/control-plane/supervisor/build-wave-executor.js.map +1 -0
  388. package/dist/apps/control-plane/supervisor/lease-heartbeat-service.d.ts +12 -0
  389. package/dist/apps/control-plane/supervisor/lease-heartbeat-service.js +14 -0
  390. package/dist/apps/control-plane/supervisor/lease-heartbeat-service.js.map +1 -0
  391. package/dist/apps/control-plane/supervisor/planning-wave-executor.d.ts +19 -0
  392. package/dist/apps/control-plane/supervisor/planning-wave-executor.js +249 -0
  393. package/dist/apps/control-plane/supervisor/planning-wave-executor.js.map +1 -0
  394. package/dist/apps/control-plane/supervisor/prompt-bundle-loader.d.ts +9 -0
  395. package/dist/apps/control-plane/supervisor/prompt-bundle-loader.js +53 -0
  396. package/dist/apps/control-plane/supervisor/prompt-bundle-loader.js.map +1 -0
  397. package/dist/apps/control-plane/supervisor/qa-wave-executor.d.ts +24 -0
  398. package/dist/apps/control-plane/supervisor/qa-wave-executor.js +70 -0
  399. package/dist/apps/control-plane/supervisor/qa-wave-executor.js.map +1 -0
  400. package/dist/apps/control-plane/supervisor/run-coordinator.d.ts +49 -0
  401. package/dist/apps/control-plane/supervisor/run-coordinator.js +162 -0
  402. package/dist/apps/control-plane/supervisor/run-coordinator.js.map +1 -0
  403. package/dist/apps/control-plane/supervisor/runtime.d.ts +58 -0
  404. package/dist/apps/control-plane/supervisor/runtime.js +270 -0
  405. package/dist/apps/control-plane/supervisor/runtime.js.map +1 -0
  406. package/dist/apps/control-plane/supervisor/session-orchestrator.d.ts +29 -0
  407. package/dist/apps/control-plane/supervisor/session-orchestrator.js +211 -0
  408. package/dist/apps/control-plane/supervisor/session-orchestrator.js.map +1 -0
  409. package/dist/apps/control-plane/supervisor/types.d.ts +148 -0
  410. package/dist/apps/control-plane/supervisor/types.js +2 -0
  411. package/dist/apps/control-plane/supervisor/types.js.map +1 -0
  412. package/dist/apps/control-plane/supervisor/worker-decision-loop.d.ts +37 -0
  413. package/dist/apps/control-plane/supervisor/worker-decision-loop.js +236 -0
  414. package/dist/apps/control-plane/supervisor/worker-decision-loop.js.map +1 -0
  415. package/docker/mcp.Dockerfile +14 -0
  416. package/docker/mcp.compose.yaml +15 -0
  417. package/docker/mcp.entrypoint.sh +17 -0
  418. package/eslint.config.mjs +93 -0
  419. package/example-configurations/README.md +26 -0
  420. package/example-configurations/java/agents.yaml +14 -0
  421. package/example-configurations/java/gates.yaml +29 -0
  422. package/example-configurations/java/policy.yaml +148 -0
  423. package/example-configurations/node/agents.yaml +14 -0
  424. package/example-configurations/node/gates.yaml +32 -0
  425. package/example-configurations/node/policy.yaml +143 -0
  426. package/nx.json +16 -0
  427. package/package.json +39 -0
  428. package/prompts/vitest-testing-standards.instructions.md +204 -0
  429. package/scripts/dev-shell-env.sh +7 -0
  430. package/scripts/nx-safe.mjs +33 -0
  431. package/spec-files/agentic_orchestrator_cli_delete_command_spec.md +310 -0
  432. package/spec-files/agentic_orchestrator_dot_aop_generated_artifacts_spec.md +211 -0
  433. package/spec-files/agentic_orchestrator_mcp_formalization_spec.md +379 -0
  434. package/spec-files/agentic_orchestrator_oop_refactor_spec.md +415 -0
  435. package/spec-files/agentic_orchestrator_single_global_orchestrator_spec.md +265 -0
  436. package/spec-files/agentic_orchestrator_spec.md +1334 -0
  437. package/spec-files/progress.md +452 -0
  438. package/tsconfig.base.json +15 -0
  439. package/tsconfig.json +11 -0
@@ -0,0 +1,246 @@
1
+ import fs from 'node:fs/promises';
2
+ import { nowIso } from '../../core/fs.js';
3
+ import { runGit } from '../../core/git.js';
4
+ import { ERROR_CODES } from '../../core/error-codes.js';
5
+ import { fail } from '../../core/response.js';
6
+ import { GATE_RESULT, STATUS } from '../../core/constants.js';
7
+
8
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
9
+ type AnyRecord = Record<string, any>;
10
+
11
+ type StateUpdater = (
12
+ frontMatter: AnyRecord,
13
+ body: string
14
+ ) => Promise<{ frontMatter?: AnyRecord; body?: string }>;
15
+
16
+ function normalizeSet(array: string[]): string[] {
17
+ return [...new Set(array)].sort((a, b) => a.localeCompare(b));
18
+ }
19
+
20
+ function readHeldLocks(frontMatter: AnyRecord): string[] {
21
+ const locks = frontMatter?.locks && typeof frontMatter.locks === 'object' ? frontMatter.locks : {};
22
+ const held = Array.isArray(locks.held) ? locks.held : [];
23
+ return held.filter((item: unknown): item is string => typeof item === 'string');
24
+ }
25
+
26
+ function asRecord(value: unknown): AnyRecord {
27
+ return value && typeof value === 'object' ? (value as AnyRecord) : {};
28
+ }
29
+
30
+ function asArray<T = unknown>(value: unknown): T[] {
31
+ return Array.isArray(value) ? (value as T[]) : [];
32
+ }
33
+
34
+ export interface MergeServicePort {
35
+ getRepoRoot(): string;
36
+ getPolicySnapshot(): AnyRecord;
37
+ featurePath(featureId: string): string;
38
+ readState(featureId: string): Promise<{ frontMatter: AnyRecord; body: string }>;
39
+ worktreePath(featureId: string): string;
40
+ updateState(featureId: string, expectedVersion: number | null, updater: StateUpdater): Promise<AnyRecord>;
41
+ locksRelease(resource: string, featureId: string): Promise<unknown>;
42
+ withIndexLock<T>(operation: () => Promise<T>): Promise<T>;
43
+ readIndex(): Promise<AnyRecord>;
44
+ writeIndex(index: AnyRecord): Promise<void>;
45
+ }
46
+
47
+ export class MergeService {
48
+ private readonly port: MergeServicePort;
49
+
50
+ constructor(port: MergeServicePort) {
51
+ this.port = port;
52
+ }
53
+
54
+ async featureReadyToMerge(
55
+ featureId: string,
56
+ commitMessage: string,
57
+ mergeStrategy: string,
58
+ userApprovalToken: string | null
59
+ ): Promise<{ data: { feature_id: string; merge_sha: string; merge_strategy: string; status: string } }> {
60
+ const policy = this.port.getPolicySnapshot();
61
+ const state = await this.port.readState(featureId);
62
+
63
+ if (!policy.merge_policy.allowed_strategies.includes(mergeStrategy)) {
64
+ throw {
65
+ normalizedResponse: fail(ERROR_CODES.MERGE_DISABLED, 'Merge strategy is not allowed by policy', {
66
+ merge_strategy: mergeStrategy,
67
+ retryable: false,
68
+ requires_human: true
69
+ })
70
+ };
71
+ }
72
+
73
+ if (policy.merge_policy.require_user_approval && !userApprovalToken) {
74
+ throw {
75
+ normalizedResponse: fail(ERROR_CODES.USER_APPROVAL_REQUIRED, 'User approval token is required', {
76
+ retryable: false,
77
+ requires_human: true
78
+ })
79
+ };
80
+ }
81
+
82
+ if (state.frontMatter.status !== STATUS.READY_TO_MERGE) {
83
+ throw {
84
+ normalizedResponse: fail(ERROR_CODES.INVALID_STATUS_TRANSITION, 'Feature must be ready_to_merge before merge', {
85
+ current_status: state.frontMatter.status,
86
+ retryable: false,
87
+ requires_human: true
88
+ })
89
+ };
90
+ }
91
+
92
+ const requiredModes = [...policy.required_modes, policy.required_merge_mode];
93
+ for (const mode of requiredModes) {
94
+ if (state.frontMatter.gates?.[mode] !== GATE_RESULT.PASS) {
95
+ throw {
96
+ normalizedResponse: fail(ERROR_CODES.MERGE_DISABLED, 'Required gate mode has not passed', {
97
+ mode,
98
+ retryable: true,
99
+ requires_human: false
100
+ })
101
+ };
102
+ }
103
+ }
104
+
105
+ const repoRoot = this.port.getRepoRoot();
106
+ const featureWorktree = this.port.worktreePath(featureId);
107
+ const commitResult = await runGit(repoRoot, ['add', '-A'], { cwd: featureWorktree });
108
+ if (commitResult.code !== 0) {
109
+ throw {
110
+ normalizedResponse: fail(ERROR_CODES.GIT_FAILURE, 'Failed to stage changes before merge', {
111
+ stderr: commitResult.stderr,
112
+ retryable: true,
113
+ requires_human: false
114
+ })
115
+ };
116
+ }
117
+
118
+ const commit = await runGit(repoRoot, ['commit', '-m', commitMessage], { cwd: featureWorktree });
119
+ if (commit.code !== 0 && !commit.stderr.includes('nothing to commit')) {
120
+ throw {
121
+ normalizedResponse: fail(ERROR_CODES.GIT_FAILURE, 'Failed to commit feature branch', {
122
+ stderr: commit.stderr,
123
+ retryable: true,
124
+ requires_human: false
125
+ })
126
+ };
127
+ }
128
+
129
+ const baseBranch = policy.worktree.base_branch;
130
+ const checkoutBase = await runGit(repoRoot, ['checkout', baseBranch]);
131
+ if (checkoutBase.code !== 0) {
132
+ throw {
133
+ normalizedResponse: fail(ERROR_CODES.GIT_FAILURE, 'Failed to checkout base branch for merge', {
134
+ stderr: checkoutBase.stderr,
135
+ retryable: true,
136
+ requires_human: true
137
+ })
138
+ };
139
+ }
140
+
141
+ let merge;
142
+ if (mergeStrategy === 'merge_commit') {
143
+ merge = await runGit(repoRoot, ['merge', '--no-ff', featureId, '-m', `Merge feature ${featureId}`]);
144
+ } else if (mergeStrategy === 'squash') {
145
+ merge = await runGit(repoRoot, ['merge', '--squash', featureId]);
146
+ if (merge.code === 0) {
147
+ merge = await runGit(repoRoot, ['commit', '-m', `Squash merge ${featureId}`]);
148
+ }
149
+ } else {
150
+ const rebase = await runGit(repoRoot, ['rebase', baseBranch], { cwd: featureWorktree });
151
+ if (rebase.code !== 0) {
152
+ throw {
153
+ normalizedResponse: fail(ERROR_CODES.GIT_FAILURE, 'Rebase failed before merge', {
154
+ stderr: rebase.stderr,
155
+ retryable: true,
156
+ requires_human: true
157
+ })
158
+ };
159
+ }
160
+ merge = await runGit(repoRoot, ['merge', '--ff-only', featureId]);
161
+ }
162
+
163
+ if (merge.code !== 0) {
164
+ throw {
165
+ normalizedResponse: fail(ERROR_CODES.GIT_FAILURE, 'Merge failed', {
166
+ stderr: merge.stderr,
167
+ retryable: true,
168
+ requires_human: true
169
+ })
170
+ };
171
+ }
172
+
173
+ const mergeShaResult = await runGit(repoRoot, ['rev-parse', 'HEAD']);
174
+ const mergeSha = mergeShaResult.stdout.trim();
175
+
176
+ await this.port.updateState(featureId, state.frontMatter.version, (frontMatter, body) =>
177
+ Promise.resolve({
178
+ frontMatter: {
179
+ status: STATUS.MERGED,
180
+ status_reason: `Merged by strategy ${mergeStrategy}`,
181
+ evidence: {
182
+ ...(frontMatter.evidence ?? {}),
183
+ merge_sha: mergeSha,
184
+ merge_strategy: mergeStrategy,
185
+ merge_at: nowIso()
186
+ }
187
+ },
188
+ body: `${body}\n\n## Merge\n\nMerged at ${nowIso()} as ${mergeSha}.\n`
189
+ })
190
+ );
191
+
192
+ const heldLocks = [...readHeldLocks(state.frontMatter)];
193
+ for (const resource of heldLocks) {
194
+ await this.port.locksRelease(resource, featureId);
195
+ }
196
+
197
+ await this.port.withIndexLock(async () => {
198
+ const index = await this.port.readIndex();
199
+ index.active = normalizeSet(asArray<string>(index.active).filter((id) => id !== featureId));
200
+ index.blocked = normalizeSet(asArray<string>(index.blocked).filter((id) => id !== featureId));
201
+ index.merged = normalizeSet([...asArray<string>(index.merged), featureId]);
202
+
203
+ const blockedQueue = asArray<AnyRecord>(index.blocked_queue);
204
+ index.blocked_queue = blockedQueue.filter((entry) => asRecord(entry).feature_id !== featureId);
205
+
206
+ const locks = asRecord(index.locks);
207
+ for (const [resource, holder] of Object.entries(locks)) {
208
+ if (holder === featureId) {
209
+ locks[resource] = null;
210
+ }
211
+ }
212
+ index.locks = locks;
213
+
214
+ const lockLeases = asRecord(index.lock_leases);
215
+ for (const [resource, lease] of Object.entries(lockLeases)) {
216
+ if (asRecord(lease).holder === featureId) {
217
+ lockLeases[resource] = null;
218
+ }
219
+ }
220
+ index.lock_leases = lockLeases;
221
+
222
+ const runtimeSessions = asRecord(index.runtime_sessions);
223
+ const featureSessions = asRecord(runtimeSessions.feature_sessions);
224
+ if (Object.prototype.hasOwnProperty.call(featureSessions, featureId)) {
225
+ delete featureSessions[featureId];
226
+ }
227
+ runtimeSessions.feature_sessions = featureSessions;
228
+ index.runtime_sessions = runtimeSessions;
229
+
230
+ index.version += 1;
231
+ index.updated_at = nowIso();
232
+ await this.port.writeIndex(index);
233
+ });
234
+
235
+ await fs.rm(this.port.featurePath(featureId), { recursive: true, force: true });
236
+
237
+ return {
238
+ data: {
239
+ feature_id: featureId,
240
+ merge_sha: mergeSha,
241
+ merge_strategy: mergeStrategy,
242
+ status: STATUS.MERGED
243
+ }
244
+ };
245
+ }
246
+ }
@@ -0,0 +1,259 @@
1
+ import path from 'node:path';
2
+ import {atomicWriteJson, readJson} from '../../core/fs.js';
3
+ import {parseUnifiedDiff, touchedPathsFromDiff} from '../../core/patch.js';
4
+ import {anyAreaMatch, normalizeRepoPath, sortResourcesDeterministically} from '../../core/path-rules.js';
5
+ import {runGit} from '../../core/git.js';
6
+ import {buildQaIndex, makeRequiredTests} from '../../core/qa-index.js';
7
+ import {ERROR_CODES} from '../../core/error-codes.js';
8
+ import {fail} from '../../core/response.js';
9
+
10
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
11
+ type AnyRecord = Record<string, any>;
12
+
13
+ type StateUpdater = (
14
+ frontMatter: AnyRecord,
15
+ body: string
16
+ ) => Promise<{ frontMatter?: AnyRecord; body?: string }>;
17
+
18
+ function asArray<T = unknown>(value: unknown): T[] {
19
+ return Array.isArray(value) ? (value as T[]) : [];
20
+ }
21
+
22
+ function readObjectField(record: AnyRecord, key: string): AnyRecord {
23
+ const value = record[key];
24
+ return value && typeof value === 'object' ? (value as AnyRecord) : {};
25
+ }
26
+
27
+ function readHeldLocks(frontMatter: AnyRecord): string[] {
28
+ const locks = readObjectField(frontMatter, 'locks');
29
+ return asArray<string>(locks.held).filter((item) => typeof item === 'string');
30
+ }
31
+
32
+ function normalizeFromWorktree(worktreePath: string, repoRoot: string, repoRelativeFromWorktree: string): string {
33
+ const absolute = path.resolve(repoRoot, repoRelativeFromWorktree);
34
+ const maybeRelativeToWorktree = path.relative(worktreePath, absolute).replaceAll('\\\\', '/');
35
+ if (!maybeRelativeToWorktree.startsWith('../')) {
36
+ return maybeRelativeToWorktree;
37
+ }
38
+ return path.relative(repoRoot, absolute).replaceAll('\\\\', '/');
39
+ }
40
+
41
+ export interface PatchServicePort {
42
+ getRepoRoot(): string;
43
+ getPolicySnapshot(): AnyRecord;
44
+ getGatesConfig(): AnyRecord;
45
+ planPath(featureId: string): string;
46
+ qaIndexPath(featureId: string): string;
47
+ worktreePath(featureId: string): string;
48
+ readState(featureId: string): Promise<{ frontMatter: AnyRecord }>;
49
+ validateSchema(schemaName: string, value: unknown): Promise<{ valid: boolean; errors?: unknown }>;
50
+ updateState(featureId: string, expectedVersion: number | null, updater: StateUpdater): Promise<AnyRecord>;
51
+ }
52
+
53
+ export class PatchService {
54
+ private readonly port: PatchServicePort;
55
+
56
+ constructor(port: PatchServicePort) {
57
+ this.port = port;
58
+ }
59
+
60
+ async loadAcceptedPlan(featureId: string): Promise<AnyRecord | null> {
61
+ const plan = await readJson(this.port.planPath(featureId), null);
62
+ if (!plan && this.port.getPolicySnapshot().patch_policy.enforce_plan) {
63
+ throw {
64
+ normalizedResponse: fail(ERROR_CODES.PLAN_REQUIRED, 'Patch requires accepted plan', {
65
+ feature_id: featureId,
66
+ retryable: false,
67
+ requires_human: true,
68
+ suggested_next_actions: ['Submit plan before applying patches']
69
+ })
70
+ };
71
+ }
72
+ return plan;
73
+ }
74
+
75
+ requiredResourcesForPlan(plan: AnyRecord): string[] {
76
+ const resources: string[] = [];
77
+ const policy = this.port.getPolicySnapshot();
78
+ if (plan.contracts?.openapi === 'modify') {
79
+ resources.push(policy.locks.contract_to_resource.openapi);
80
+ }
81
+ if (plan.contracts?.events === 'modify') {
82
+ resources.push(policy.locks.contract_to_resource.events);
83
+ }
84
+ if (plan.contracts?.db === 'migration') {
85
+ resources.push(policy.locks.contract_to_resource.db);
86
+ }
87
+ return sortResourcesDeterministically(resources);
88
+ }
89
+
90
+ async assertPlanLocksHeld(featureId: string, plan: AnyRecord): Promise<void> {
91
+ const state = await this.port.readState(featureId);
92
+ const held = new Set(readHeldLocks(state.frontMatter));
93
+ const required = await this.requiredResourcesForPlan(plan);
94
+ const missing = required.filter((resource) => !held.has(resource));
95
+ if (missing.length > 0) {
96
+ throw {
97
+ normalizedResponse: fail(ERROR_CODES.LOCK_NOT_HELD, 'Plan requires lock(s) that are not held', {
98
+ feature_id: featureId,
99
+ missing_resources: missing,
100
+ retryable: true,
101
+ requires_human: false,
102
+ suggested_next_actions: ['Acquire required locks before applying patch']
103
+ })
104
+ };
105
+ }
106
+ }
107
+
108
+ async validatePatchPaths(
109
+ featureId: string,
110
+ parsedDiff: ReturnType<typeof parseUnifiedDiff>,
111
+ plan: AnyRecord | null
112
+ ): Promise<string[]> {
113
+ const policy = this.port.getPolicySnapshot();
114
+ const repoRoot = this.port.getRepoRoot();
115
+ const matchingMode = policy.path_rules.matching;
116
+ const allowSymlinkTraversal = policy.path_rules.allow_symlink_traversal;
117
+ const touched = touchedPathsFromDiff(parsedDiff);
118
+
119
+ const normalizedTouched: string[] = [];
120
+ for (const touchedPath of touched) {
121
+ const normalized = await normalizeRepoPath(
122
+ repoRoot,
123
+ path.join(this.port.worktreePath(featureId), touchedPath),
124
+ allowSymlinkTraversal
125
+ )
126
+ .then((relativePath) => normalizeFromWorktree(this.port.worktreePath(featureId), repoRoot, relativePath))
127
+ .catch(async () => {
128
+ return await normalizeRepoPath(repoRoot, touchedPath, allowSymlinkTraversal);
129
+ });
130
+ normalizedTouched.push(normalized);
131
+ }
132
+
133
+ const allowed = asArray<string>(plan?.allowed_areas);
134
+ const forbidden = asArray<string>(plan?.forbidden_areas);
135
+ const protectedAreas = asArray<string>(policy.protected_areas);
136
+
137
+ const plannedFiles = new Set([
138
+ ...asArray<string>(plan?.files?.create),
139
+ ...asArray<string>(plan?.files?.modify),
140
+ ...asArray<string>(plan?.files?.delete)
141
+ ]);
142
+
143
+ const violations: Array<{ path: string; reason: string }> = [];
144
+ for (const filePath of normalizedTouched) {
145
+ if (policy.patch_policy.enforce_allowed_areas && allowed.length > 0 && !anyAreaMatch(filePath, allowed, matchingMode)) {
146
+ violations.push({ path: filePath, reason: 'outside_allowed_areas' });
147
+ }
148
+ if (forbidden.length > 0 && anyAreaMatch(filePath, forbidden, matchingMode)) {
149
+ violations.push({ path: filePath, reason: 'in_forbidden_areas' });
150
+ }
151
+ if (protectedAreas.length > 0 && anyAreaMatch(filePath, protectedAreas, matchingMode)) {
152
+ violations.push({ path: filePath, reason: 'in_protected_areas' });
153
+ }
154
+ if (plan && !plannedFiles.has(filePath)) {
155
+ violations.push({ path: filePath, reason: 'not_in_planned_file_set' });
156
+ }
157
+ }
158
+
159
+ if (violations.length > 0) {
160
+ throw {
161
+ normalizedResponse: fail(ERROR_CODES.PATCH_POLICY_VIOLATION, 'Patch violates plan/policy constraints', {
162
+ feature_id: featureId,
163
+ violations,
164
+ retryable: false,
165
+ requires_human: true
166
+ })
167
+ };
168
+ }
169
+
170
+ return normalizedTouched;
171
+ }
172
+
173
+ async repoApplyPatch(
174
+ featureId: string,
175
+ unifiedDiff: string
176
+ ): Promise<{
177
+ data: {
178
+ feature_id: string;
179
+ changed_files: string[];
180
+ status_porcelain: string[];
181
+ qa_index_version: number;
182
+ };
183
+ }> {
184
+ const plan = await this.loadAcceptedPlan(featureId);
185
+ const parsedDiff = parseUnifiedDiff(unifiedDiff);
186
+
187
+ await this.validatePatchPaths(featureId, parsedDiff, plan);
188
+ await this.assertPlanLocksHeld(featureId, plan ?? { contracts: { openapi: 'none', events: 'none', db: 'none' } });
189
+
190
+ const repoRoot = this.port.getRepoRoot();
191
+ const worktree = this.port.worktreePath(featureId);
192
+ const apply = await runGit(repoRoot, ['apply', '--whitespace=nowarn', '-'], {
193
+ cwd: worktree,
194
+ stdin: unifiedDiff
195
+ });
196
+
197
+ if (apply.code !== 0) {
198
+ throw {
199
+ normalizedResponse: fail(
200
+ ERROR_CODES.GIT_FAILURE,
201
+ 'git apply failed',
202
+ {
203
+ stderr: apply.stderr,
204
+ retryable: true,
205
+ requires_human: false
206
+ },
207
+ {
208
+ command: ['git', 'apply', '--whitespace=nowarn', '-'],
209
+ exit_code: apply.code
210
+ }
211
+ )
212
+ };
213
+ }
214
+
215
+ const status = await runGit(repoRoot, ['status', '--porcelain'], { cwd: worktree });
216
+ const changedFiles = (await runGit(repoRoot, ['diff', '--name-only'], { cwd: worktree })).stdout
217
+ .split('\n')
218
+ .map((line) => line.trim())
219
+ .filter(Boolean);
220
+
221
+ const previousQaIndex = await readJson(this.port.qaIndexPath(featureId), null);
222
+ const profileName = plan?.gate_profile ?? 'default';
223
+ const profile = this.port.getGatesConfig().profiles[profileName];
224
+ const requiredTests = makeRequiredTests(profileName, profile);
225
+ const qaIndex = buildQaIndex(featureId, parsedDiff, requiredTests, previousQaIndex);
226
+
227
+ const qaValidation = await this.port.validateSchema('qa_test_index.schema.json', qaIndex);
228
+ if (!qaValidation.valid) {
229
+ throw {
230
+ normalizedResponse: fail(ERROR_CODES.INTERNAL_ERROR, 'qa_test_index validation failed', {
231
+ errors: qaValidation.errors,
232
+ retryable: false,
233
+ requires_human: true
234
+ })
235
+ };
236
+ }
237
+
238
+ await atomicWriteJson(this.port.qaIndexPath(featureId), qaIndex);
239
+
240
+ await this.port.updateState(featureId, null, (frontMatter, body) =>
241
+ Promise.resolve({
242
+ frontMatter: {
243
+ status: frontMatter.status,
244
+ gates: frontMatter.gates
245
+ },
246
+ body
247
+ })
248
+ );
249
+
250
+ return {
251
+ data: {
252
+ feature_id: featureId,
253
+ changed_files: changedFiles,
254
+ status_porcelain: status.stdout.trim().split('\n').filter(Boolean),
255
+ qa_index_version: qaIndex.version
256
+ }
257
+ };
258
+ }
259
+ }