agentic-orchestrator 0.1.2 → 0.1.4

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 (300) hide show
  1. package/.claude/settings.local.json +15 -0
  2. package/CLAUDE.md +126 -0
  3. package/README.md +166 -25
  4. package/agentic/orchestrator/adapters.yaml +3 -0
  5. package/agentic/orchestrator/gates.yaml +47 -0
  6. package/agentic/orchestrator/policy.yaml +89 -0
  7. package/agentic/orchestrator/schemas/adapters.schema.json +12 -0
  8. package/agentic/orchestrator/schemas/gates.schema.json +6 -1
  9. package/agentic/orchestrator/schemas/index.schema.json +14 -0
  10. package/agentic/orchestrator/schemas/multi-project.schema.json +41 -0
  11. package/agentic/orchestrator/schemas/policy.schema.json +449 -52
  12. package/agentic/orchestrator/schemas/state.schema.json +16 -0
  13. package/agentic/orchestrator/tools/catalog.json +68 -0
  14. package/agentic/orchestrator/tools/schemas/input/cost.get.input.schema.json +10 -0
  15. package/agentic/orchestrator/tools/schemas/input/cost.record.input.schema.json +13 -0
  16. package/agentic/orchestrator/tools/schemas/input/feature.send_message.input.schema.json +11 -0
  17. package/agentic/orchestrator/tools/schemas/input/performance.get_analytics.input.schema.json +10 -0
  18. package/agentic/orchestrator/tools/schemas/input/performance.record_outcome.input.schema.json +18 -0
  19. package/agentic/orchestrator/tools/schemas/output/cost.get.output.schema.json +13 -0
  20. package/agentic/orchestrator/tools/schemas/output/cost.record.output.schema.json +13 -0
  21. package/agentic/orchestrator/tools/schemas/output/feature.ready_to_merge.output.schema.json +7 -0
  22. package/agentic/orchestrator/tools/schemas/output/feature.send_message.output.schema.json +23 -0
  23. package/agentic/orchestrator/tools/schemas/output/performance.get_analytics.output.schema.json +46 -0
  24. package/agentic/orchestrator/tools/schemas/output/performance.record_outcome.output.schema.json +10 -0
  25. package/agentic/orchestrator/tools.md +5 -0
  26. package/apps/control-plane/scripts/validate-architecture-rules.mjs +28 -2
  27. package/apps/control-plane/scripts/validate-docker-mcp-contract.mjs +12 -0
  28. package/apps/control-plane/scripts/validate-mcp-contracts.ts +92 -0
  29. package/apps/control-plane/src/application/adapters/adapter-registry.ts +169 -0
  30. package/apps/control-plane/src/application/multi-project-loader.ts +119 -0
  31. package/apps/control-plane/src/application/services/activity-monitor-service.ts +199 -0
  32. package/apps/control-plane/src/application/services/cost-tracking-service.ts +82 -0
  33. package/apps/control-plane/src/application/services/dependency-scheduler-service.ts +86 -0
  34. package/apps/control-plane/src/application/services/feature-deletion-service.ts +8 -7
  35. package/apps/control-plane/src/application/services/gate-interpolation-service.ts +15 -0
  36. package/apps/control-plane/src/application/services/gate-service.ts +38 -2
  37. package/apps/control-plane/src/application/services/instance-isolation-service.ts +18 -0
  38. package/apps/control-plane/src/application/services/issue-tracker-service.ts +469 -0
  39. package/apps/control-plane/src/application/services/merge-service.ts +67 -3
  40. package/apps/control-plane/src/application/services/notifier-service.ts +295 -0
  41. package/apps/control-plane/src/application/services/performance-analytics-service.ts +122 -0
  42. package/apps/control-plane/src/application/services/plan-service.ts +51 -0
  43. package/apps/control-plane/src/application/services/pr-monitor-service.ts +262 -0
  44. package/apps/control-plane/src/application/services/reactions-service.ts +175 -0
  45. package/apps/control-plane/src/application/services/reporting-service.ts +17 -2
  46. package/apps/control-plane/src/application/services/run-lease-service.ts +16 -38
  47. package/apps/control-plane/src/application/tools/tool-metadata.ts +4 -1
  48. package/apps/control-plane/src/cli/aop.ts +1 -1
  49. package/apps/control-plane/src/cli/attach-command-handler.ts +120 -0
  50. package/apps/control-plane/src/cli/cleanup-command-handler.ts +190 -0
  51. package/apps/control-plane/src/cli/cli-argument-parser.ts +69 -3
  52. package/apps/control-plane/src/cli/dashboard-command-handler.ts +57 -0
  53. package/apps/control-plane/src/cli/help-command-handler.ts +163 -0
  54. package/apps/control-plane/src/cli/init-command-handler.ts +609 -0
  55. package/apps/control-plane/src/cli/resume-command-handler.ts +1 -0
  56. package/apps/control-plane/src/cli/retry-command-handler.ts +138 -0
  57. package/apps/control-plane/src/cli/run-command-handler.ts +115 -3
  58. package/apps/control-plane/src/cli/send-command-handler.ts +65 -0
  59. package/apps/control-plane/src/cli/status-command-handler.ts +102 -2
  60. package/apps/control-plane/src/cli/types.ts +26 -1
  61. package/apps/control-plane/src/core/constants.ts +8 -2
  62. package/apps/control-plane/src/core/error-codes.ts +3 -1
  63. package/apps/control-plane/src/core/gates.ts +170 -50
  64. package/apps/control-plane/src/core/kernel.ts +280 -5
  65. package/apps/control-plane/src/core/path-layout.ts +12 -0
  66. package/apps/control-plane/src/core/tool-caller.ts +36 -0
  67. package/apps/control-plane/src/core/workspace-hooks.ts +87 -0
  68. package/apps/control-plane/src/interfaces/cli/bootstrap.ts +258 -9
  69. package/apps/control-plane/src/providers/providers.ts +235 -14
  70. package/apps/control-plane/src/supervisor/build-wave-executor.ts +129 -8
  71. package/apps/control-plane/src/supervisor/qa-wave-executor.ts +123 -5
  72. package/apps/control-plane/src/supervisor/run-coordinator.ts +143 -6
  73. package/apps/control-plane/src/supervisor/runtime.ts +135 -6
  74. package/apps/control-plane/src/supervisor/types.ts +12 -21
  75. package/apps/control-plane/src/supervisor/worker-decision-loop.ts +8 -0
  76. package/apps/control-plane/test/activity-monitor.spec.ts +294 -0
  77. package/apps/control-plane/test/adapter-registry.spec.ts +132 -0
  78. package/apps/control-plane/test/batch-operations.spec.ts +112 -0
  79. package/apps/control-plane/test/bootstrap-attach.spec.ts +102 -0
  80. package/apps/control-plane/test/bootstrap-edge-cases.spec.ts +252 -0
  81. package/apps/control-plane/test/bootstrap.spec.ts +560 -0
  82. package/apps/control-plane/test/cleanup-command.spec.ts +301 -0
  83. package/apps/control-plane/test/cli-helpers.spec.ts +404 -1
  84. package/apps/control-plane/test/cli.unit.spec.ts +182 -1
  85. package/apps/control-plane/test/collision-queue.spec.ts +104 -1
  86. package/apps/control-plane/test/core-utils.spec.ts +175 -2
  87. package/apps/control-plane/test/cost-tracking.spec.ts +143 -0
  88. package/apps/control-plane/test/dashboard-api.integration.spec.ts +247 -0
  89. package/apps/control-plane/test/dashboard-client.spec.ts +116 -0
  90. package/apps/control-plane/test/dashboard-command.spec.ts +103 -0
  91. package/apps/control-plane/test/dependency-scheduler.spec.ts +189 -0
  92. package/apps/control-plane/test/epoch-tracking.spec.ts +4 -4
  93. package/apps/control-plane/test/feature-deletion-service.spec.ts +422 -0
  94. package/apps/control-plane/test/feature-lifecycle.spec.ts +202 -0
  95. package/apps/control-plane/test/git-spawn-error.spec.ts +24 -0
  96. package/apps/control-plane/test/incremental-gates.spec.ts +137 -0
  97. package/apps/control-plane/test/init-wizard.spec.ts +506 -0
  98. package/apps/control-plane/test/instance-isolation.spec.ts +83 -0
  99. package/apps/control-plane/test/issue-tracker.spec.ts +890 -0
  100. package/apps/control-plane/test/kernel.coverage.spec.ts +3 -5
  101. package/apps/control-plane/test/kernel.coverage2.spec.ts +871 -0
  102. package/apps/control-plane/test/kernel.spec.ts +13 -11
  103. package/apps/control-plane/test/lock-service.spec.ts +508 -0
  104. package/apps/control-plane/test/mcp-helpers.spec.ts +176 -0
  105. package/apps/control-plane/test/mcp.spec.ts +50 -15
  106. package/apps/control-plane/test/merge-service.spec.ts +67 -4
  107. package/apps/control-plane/test/multi-project.spec.ts +372 -0
  108. package/apps/control-plane/test/notifier-service.spec.ts +388 -0
  109. package/apps/control-plane/test/parallel-gates.spec.ts +312 -0
  110. package/apps/control-plane/test/patch-service.spec.ts +253 -0
  111. package/apps/control-plane/test/performance-analytics.spec.ts +338 -0
  112. package/apps/control-plane/test/planning-wave-executor.spec.ts +168 -0
  113. package/apps/control-plane/test/pr-monitor.spec.ts +385 -0
  114. package/apps/control-plane/test/providers.spec.ts +344 -1
  115. package/apps/control-plane/test/reactions.spec.ts +392 -0
  116. package/apps/control-plane/test/resume-command.spec.ts +390 -0
  117. package/apps/control-plane/test/run-coordinator.spec.ts +481 -2
  118. package/apps/control-plane/test/schema-date-time.spec.ts +46 -0
  119. package/apps/control-plane/test/service-retry-paths.spec.ts +30 -0
  120. package/apps/control-plane/test/services.spec.ts +95 -2
  121. package/apps/control-plane/test/session-management.spec.ts +450 -0
  122. package/apps/control-plane/test/spec-ingestion.spec.ts +190 -0
  123. package/apps/control-plane/test/supervisor-collaborators.spec.ts +699 -2
  124. package/apps/control-plane/test/supervisor.spec.ts +36 -30
  125. package/apps/control-plane/test/supervisor.unit.spec.ts +405 -0
  126. package/apps/control-plane/test/worker-decision-loop.spec.ts +57 -0
  127. package/apps/control-plane/test/workspace-hooks.spec.ts +177 -0
  128. package/apps/control-plane/vitest.config.ts +21 -5
  129. package/dist/apps/control-plane/application/adapters/adapter-registry.d.ts +44 -0
  130. package/dist/apps/control-plane/application/adapters/adapter-registry.js +76 -0
  131. package/dist/apps/control-plane/application/adapters/adapter-registry.js.map +1 -0
  132. package/dist/apps/control-plane/application/multi-project-loader.d.ts +31 -0
  133. package/dist/apps/control-plane/application/multi-project-loader.js +82 -0
  134. package/dist/apps/control-plane/application/multi-project-loader.js.map +1 -0
  135. package/dist/apps/control-plane/application/services/activity-monitor-service.d.ts +43 -0
  136. package/dist/apps/control-plane/application/services/activity-monitor-service.js +132 -0
  137. package/dist/apps/control-plane/application/services/activity-monitor-service.js.map +1 -0
  138. package/dist/apps/control-plane/application/services/cost-tracking-service.d.ts +28 -0
  139. package/dist/apps/control-plane/application/services/cost-tracking-service.js +48 -0
  140. package/dist/apps/control-plane/application/services/cost-tracking-service.js.map +1 -0
  141. package/dist/apps/control-plane/application/services/dependency-scheduler-service.d.ts +26 -0
  142. package/dist/apps/control-plane/application/services/dependency-scheduler-service.js +75 -0
  143. package/dist/apps/control-plane/application/services/dependency-scheduler-service.js.map +1 -0
  144. package/dist/apps/control-plane/application/services/feature-deletion-service.d.ts +2 -0
  145. package/dist/apps/control-plane/application/services/feature-deletion-service.js +6 -7
  146. package/dist/apps/control-plane/application/services/feature-deletion-service.js.map +1 -1
  147. package/dist/apps/control-plane/application/services/gate-interpolation-service.d.ts +7 -0
  148. package/dist/apps/control-plane/application/services/gate-interpolation-service.js +7 -0
  149. package/dist/apps/control-plane/application/services/gate-interpolation-service.js.map +1 -0
  150. package/dist/apps/control-plane/application/services/gate-service.js +32 -2
  151. package/dist/apps/control-plane/application/services/gate-service.js.map +1 -1
  152. package/dist/apps/control-plane/application/services/instance-isolation-service.d.ts +11 -0
  153. package/dist/apps/control-plane/application/services/instance-isolation-service.js +17 -0
  154. package/dist/apps/control-plane/application/services/instance-isolation-service.js.map +1 -0
  155. package/dist/apps/control-plane/application/services/issue-tracker-service.d.ts +65 -0
  156. package/dist/apps/control-plane/application/services/issue-tracker-service.js +358 -0
  157. package/dist/apps/control-plane/application/services/issue-tracker-service.js.map +1 -0
  158. package/dist/apps/control-plane/application/services/merge-service.d.ts +4 -0
  159. package/dist/apps/control-plane/application/services/merge-service.js +44 -2
  160. package/dist/apps/control-plane/application/services/merge-service.js.map +1 -1
  161. package/dist/apps/control-plane/application/services/notifier-service.d.ts +74 -0
  162. package/dist/apps/control-plane/application/services/notifier-service.js +212 -0
  163. package/dist/apps/control-plane/application/services/notifier-service.js.map +1 -0
  164. package/dist/apps/control-plane/application/services/performance-analytics-service.d.ts +39 -0
  165. package/dist/apps/control-plane/application/services/performance-analytics-service.js +75 -0
  166. package/dist/apps/control-plane/application/services/performance-analytics-service.js.map +1 -0
  167. package/dist/apps/control-plane/application/services/plan-service.d.ts +1 -0
  168. package/dist/apps/control-plane/application/services/plan-service.js +53 -0
  169. package/dist/apps/control-plane/application/services/plan-service.js.map +1 -1
  170. package/dist/apps/control-plane/application/services/pr-monitor-service.d.ts +44 -0
  171. package/dist/apps/control-plane/application/services/pr-monitor-service.js +192 -0
  172. package/dist/apps/control-plane/application/services/pr-monitor-service.js.map +1 -0
  173. package/dist/apps/control-plane/application/services/reactions-service.d.ts +67 -0
  174. package/dist/apps/control-plane/application/services/reactions-service.js +114 -0
  175. package/dist/apps/control-plane/application/services/reactions-service.js.map +1 -0
  176. package/dist/apps/control-plane/application/services/reporting-service.d.ts +1 -0
  177. package/dist/apps/control-plane/application/services/reporting-service.js +13 -2
  178. package/dist/apps/control-plane/application/services/reporting-service.js.map +1 -1
  179. package/dist/apps/control-plane/application/services/run-lease-service.d.ts +2 -0
  180. package/dist/apps/control-plane/application/services/run-lease-service.js +14 -38
  181. package/dist/apps/control-plane/application/services/run-lease-service.js.map +1 -1
  182. package/dist/apps/control-plane/application/tools/tool-metadata.js +3 -1
  183. package/dist/apps/control-plane/application/tools/tool-metadata.js.map +1 -1
  184. package/dist/apps/control-plane/cli/aop.d.ts +1 -1
  185. package/dist/apps/control-plane/cli/aop.js +1 -1
  186. package/dist/apps/control-plane/cli/attach-command-handler.d.ts +12 -0
  187. package/dist/apps/control-plane/cli/attach-command-handler.js +98 -0
  188. package/dist/apps/control-plane/cli/attach-command-handler.js.map +1 -0
  189. package/dist/apps/control-plane/cli/cleanup-command-handler.d.ts +12 -0
  190. package/dist/apps/control-plane/cli/cleanup-command-handler.js +162 -0
  191. package/dist/apps/control-plane/cli/cleanup-command-handler.js.map +1 -0
  192. package/dist/apps/control-plane/cli/cli-argument-parser.js +73 -3
  193. package/dist/apps/control-plane/cli/cli-argument-parser.js.map +1 -1
  194. package/dist/apps/control-plane/cli/dashboard-command-handler.d.ts +7 -0
  195. package/dist/apps/control-plane/cli/dashboard-command-handler.js +45 -0
  196. package/dist/apps/control-plane/cli/dashboard-command-handler.js.map +1 -0
  197. package/dist/apps/control-plane/cli/help-command-handler.d.ts +8 -0
  198. package/dist/apps/control-plane/cli/help-command-handler.js +146 -0
  199. package/dist/apps/control-plane/cli/help-command-handler.js.map +1 -0
  200. package/dist/apps/control-plane/cli/init-command-handler.d.ts +26 -0
  201. package/dist/apps/control-plane/cli/init-command-handler.js +517 -0
  202. package/dist/apps/control-plane/cli/init-command-handler.js.map +1 -0
  203. package/dist/apps/control-plane/cli/resume-command-handler.js +1 -1
  204. package/dist/apps/control-plane/cli/resume-command-handler.js.map +1 -1
  205. package/dist/apps/control-plane/cli/retry-command-handler.d.ts +8 -0
  206. package/dist/apps/control-plane/cli/retry-command-handler.js +111 -0
  207. package/dist/apps/control-plane/cli/retry-command-handler.js.map +1 -0
  208. package/dist/apps/control-plane/cli/run-command-handler.d.ts +5 -0
  209. package/dist/apps/control-plane/cli/run-command-handler.js +82 -3
  210. package/dist/apps/control-plane/cli/run-command-handler.js.map +1 -1
  211. package/dist/apps/control-plane/cli/send-command-handler.d.ts +8 -0
  212. package/dist/apps/control-plane/cli/send-command-handler.js +55 -0
  213. package/dist/apps/control-plane/cli/send-command-handler.js.map +1 -0
  214. package/dist/apps/control-plane/cli/status-command-handler.d.ts +12 -1
  215. package/dist/apps/control-plane/cli/status-command-handler.js +55 -2
  216. package/dist/apps/control-plane/cli/status-command-handler.js.map +1 -1
  217. package/dist/apps/control-plane/cli/types.d.ts +26 -1
  218. package/dist/apps/control-plane/cli/types.js +15 -1
  219. package/dist/apps/control-plane/cli/types.js.map +1 -1
  220. package/dist/apps/control-plane/core/constants.d.ts +6 -0
  221. package/dist/apps/control-plane/core/constants.js +8 -2
  222. package/dist/apps/control-plane/core/constants.js.map +1 -1
  223. package/dist/apps/control-plane/core/error-codes.d.ts +2 -0
  224. package/dist/apps/control-plane/core/error-codes.js +3 -1
  225. package/dist/apps/control-plane/core/error-codes.js.map +1 -1
  226. package/dist/apps/control-plane/core/gates.d.ts +4 -0
  227. package/dist/apps/control-plane/core/gates.js +140 -43
  228. package/dist/apps/control-plane/core/gates.js.map +1 -1
  229. package/dist/apps/control-plane/core/kernel.d.ts +50 -1
  230. package/dist/apps/control-plane/core/kernel.js +220 -7
  231. package/dist/apps/control-plane/core/kernel.js.map +1 -1
  232. package/dist/apps/control-plane/core/path-layout.d.ts +3 -0
  233. package/dist/apps/control-plane/core/path-layout.js +9 -0
  234. package/dist/apps/control-plane/core/path-layout.js.map +1 -1
  235. package/dist/apps/control-plane/core/tool-caller.d.ts +32 -0
  236. package/dist/apps/control-plane/core/tool-caller.js +2 -0
  237. package/dist/apps/control-plane/core/tool-caller.js.map +1 -0
  238. package/dist/apps/control-plane/core/workspace-hooks.d.ts +20 -0
  239. package/dist/apps/control-plane/core/workspace-hooks.js +69 -0
  240. package/dist/apps/control-plane/core/workspace-hooks.js.map +1 -0
  241. package/dist/apps/control-plane/interfaces/cli/bootstrap.js +245 -9
  242. package/dist/apps/control-plane/interfaces/cli/bootstrap.js.map +1 -1
  243. package/dist/apps/control-plane/providers/providers.d.ts +42 -3
  244. package/dist/apps/control-plane/providers/providers.js +216 -5
  245. package/dist/apps/control-plane/providers/providers.js.map +1 -1
  246. package/dist/apps/control-plane/supervisor/build-wave-executor.d.ts +3 -0
  247. package/dist/apps/control-plane/supervisor/build-wave-executor.js +115 -6
  248. package/dist/apps/control-plane/supervisor/build-wave-executor.js.map +1 -1
  249. package/dist/apps/control-plane/supervisor/qa-wave-executor.d.ts +3 -0
  250. package/dist/apps/control-plane/supervisor/qa-wave-executor.js +109 -5
  251. package/dist/apps/control-plane/supervisor/qa-wave-executor.js.map +1 -1
  252. package/dist/apps/control-plane/supervisor/run-coordinator.d.ts +15 -0
  253. package/dist/apps/control-plane/supervisor/run-coordinator.js +132 -6
  254. package/dist/apps/control-plane/supervisor/run-coordinator.js.map +1 -1
  255. package/dist/apps/control-plane/supervisor/runtime.d.ts +3 -0
  256. package/dist/apps/control-plane/supervisor/runtime.js +110 -6
  257. package/dist/apps/control-plane/supervisor/runtime.js.map +1 -1
  258. package/dist/apps/control-plane/supervisor/types.d.ts +9 -16
  259. package/dist/apps/control-plane/supervisor/types.js.map +1 -1
  260. package/dist/apps/control-plane/supervisor/worker-decision-loop.d.ts +3 -0
  261. package/dist/apps/control-plane/supervisor/worker-decision-loop.js +5 -0
  262. package/dist/apps/control-plane/supervisor/worker-decision-loop.js.map +1 -1
  263. package/eslint.config.mjs +2 -1
  264. package/package.json +12 -2
  265. package/packages/web-dashboard/next-env.d.ts +5 -0
  266. package/packages/web-dashboard/next.config.js +7 -0
  267. package/packages/web-dashboard/package.json +26 -0
  268. package/packages/web-dashboard/src/app/api/actions/route.ts +64 -0
  269. package/packages/web-dashboard/src/app/api/events/route.ts +51 -0
  270. package/packages/web-dashboard/src/app/api/features/[id]/checkout/route.ts +256 -0
  271. package/packages/web-dashboard/src/app/api/features/[id]/diff/route.ts +10 -0
  272. package/packages/web-dashboard/src/app/api/features/[id]/evidence/[artifact]/route.ts +25 -0
  273. package/packages/web-dashboard/src/app/api/features/[id]/review/route.ts +63 -0
  274. package/packages/web-dashboard/src/app/api/features/[id]/route.ts +16 -0
  275. package/packages/web-dashboard/src/app/api/projects/route.ts +31 -0
  276. package/packages/web-dashboard/src/app/api/status/route.ts +15 -0
  277. package/packages/web-dashboard/src/app/globals.css +2 -0
  278. package/packages/web-dashboard/src/app/layout.tsx +15 -0
  279. package/packages/web-dashboard/src/app/page.tsx +393 -0
  280. package/packages/web-dashboard/src/lib/aop-client.ts +244 -0
  281. package/packages/web-dashboard/src/lib/multi-project-config.ts +116 -0
  282. package/packages/web-dashboard/src/lib/orchestrator-tools.ts +284 -0
  283. package/packages/web-dashboard/src/lib/types.ts +58 -0
  284. package/packages/web-dashboard/tsconfig.json +40 -0
  285. package/packages/web-dashboard/vitest.config.ts +6 -0
  286. package/spec-files/completed/agentic_orchestrator_feature_gaps_closure_spec.md +1764 -0
  287. package/spec-files/outstanding/agentic_orchestrator_enterprise_governance_dashboard_spec.md +348 -0
  288. package/spec-files/outstanding/agentic_orchestrator_knowledge_canary_spec.md +344 -0
  289. package/spec-files/outstanding/agentic_orchestrator_observability_integrity_diagnostics_spec.md +374 -0
  290. package/spec-files/outstanding/agentic_orchestrator_performance_improvements_spec.md +1059 -0
  291. package/spec-files/outstanding/agentic_orchestrator_planning_review_quality_spec.md +466 -0
  292. package/spec-files/outstanding/agentic_orchestrator_quality_adoption_execution_spec.md +198 -0
  293. package/spec-files/outstanding/agentic_orchestrator_validator_hardening_spec.md +365 -0
  294. package/spec-files/progress.md +481 -52
  295. /package/spec-files/{agentic_orchestrator_cli_delete_command_spec.md → completed/agentic_orchestrator_cli_delete_command_spec.md} +0 -0
  296. /package/spec-files/{agentic_orchestrator_dot_aop_generated_artifacts_spec.md → completed/agentic_orchestrator_dot_aop_generated_artifacts_spec.md} +0 -0
  297. /package/spec-files/{agentic_orchestrator_mcp_formalization_spec.md → completed/agentic_orchestrator_mcp_formalization_spec.md} +0 -0
  298. /package/spec-files/{agentic_orchestrator_oop_refactor_spec.md → completed/agentic_orchestrator_oop_refactor_spec.md} +0 -0
  299. /package/spec-files/{agentic_orchestrator_single_global_orchestrator_spec.md → completed/agentic_orchestrator_single_global_orchestrator_spec.md} +0 -0
  300. /package/spec-files/{agentic_orchestrator_spec.md → completed/agentic_orchestrator_spec.md} +0 -0
@@ -0,0 +1,422 @@
1
+ import fs from 'node:fs/promises';
2
+ import os from 'node:os';
3
+ import path from 'node:path';
4
+ import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest';
5
+
6
+ vi.mock('../src/core/git.js', () => ({
7
+ runGit: vi.fn()
8
+ }));
9
+
10
+ import { runGit } from '../src/core/git.js';
11
+ import { FeatureDeletionService } from '../src/application/services/feature-deletion-service.js';
12
+ import { ERROR_CODES } from '../src/core/error-codes.js';
13
+ import type { RuntimeSessionsSnapshot } from '../src/core/runtime-sessions.js';
14
+
15
+ function commandResult(overrides: Partial<{
16
+ code: number;
17
+ signal: NodeJS.Signals | null;
18
+ stdout: string;
19
+ stderr: string;
20
+ timeout: boolean;
21
+ }> = {}) {
22
+ return { code: 0, signal: null, stdout: '', stderr: '', timeout: false, ...overrides };
23
+ }
24
+
25
+ function makeStaleRunLease(): RuntimeSessionsSnapshot {
26
+ const now = new Date().toISOString();
27
+ return {
28
+ run_id: 'none',
29
+ orchestrator_session_id: 'unknown',
30
+ provider: 'unknown',
31
+ model: 'unknown',
32
+ provider_config_ref_hash: 'hash',
33
+ owner_instance_id: 'none',
34
+ lease_id: 'none',
35
+ started_at: now,
36
+ last_heartbeat_at: now,
37
+ lease_expires_at: now,
38
+ feature_sessions: {}
39
+ };
40
+ }
41
+
42
+ describe('FeatureDeletionService git failure paths', () => {
43
+ let repoRoot: string;
44
+ let featureDir: string;
45
+ let worktreeDir: string;
46
+
47
+ beforeEach(async () => {
48
+ vi.clearAllMocks();
49
+ repoRoot = await fs.mkdtemp(path.join(os.tmpdir(), 'aop-del-'));
50
+ featureDir = path.join(repoRoot, '.aop', 'features', 'feature_x');
51
+ worktreeDir = path.join(repoRoot, '.worktrees', 'feature_x');
52
+ await fs.mkdir(featureDir, { recursive: true });
53
+ await fs.mkdir(worktreeDir, { recursive: true });
54
+ });
55
+
56
+ afterEach(async () => {
57
+ await fs.rm(repoRoot, { recursive: true, force: true });
58
+ });
59
+
60
+ function makeMockPort() {
61
+ const staleRunLease = makeStaleRunLease();
62
+ return {
63
+ getRepoRoot: () => repoRoot,
64
+ featurePath: (_featureId: string) => featureDir,
65
+ worktreePath: (_featureId: string) => worktreeDir,
66
+ statePath: (_featureId: string) => path.join(featureDir, 'state.md'),
67
+ readState: vi.fn(async () => ({ frontMatter: {}, body: '' })),
68
+ withIndexLock: async <T>(op: () => Promise<T>) => await op(),
69
+ readIndex: vi.fn(async () => ({
70
+ version: 1,
71
+ active: ['feature_x'],
72
+ blocked: [],
73
+ merged: [],
74
+ locks: {},
75
+ lock_leases: {},
76
+ blocked_queue: [],
77
+ updated_at: new Date().toISOString()
78
+ })),
79
+ writeIndex: vi.fn(async () => {}),
80
+ readRunLease: vi.fn(async () => ({ ...staleRunLease })),
81
+ writeRunLease: vi.fn(async () => {}),
82
+ normalizeRuntimeSessions: vi.fn(() => staleRunLease),
83
+ isRunLeaseFresh: vi.fn(() => false),
84
+ locksRelease: vi.fn(async () => {})
85
+ };
86
+ }
87
+
88
+ it('GIVEN_worktree_exists_WHEN_runGit_worktree_remove_fails_THEN_throws_GIT_FAILURE', async () => {
89
+ vi.mocked(runGit)
90
+ .mockResolvedValueOnce(commandResult({ code: 1 })) // show-ref → no branch
91
+ .mockResolvedValueOnce(commandResult({ code: 1, stderr: 'not a worktree' })); // worktree remove fails
92
+
93
+ const service = new FeatureDeletionService(makeMockPort());
94
+
95
+ await expect(
96
+ service.featureDelete('feature_x', { dryRun: false, confirm: true, removeWorktree: true, removeBranch: 'none' })
97
+ ).rejects.toMatchObject({
98
+ normalizedResponse: expect.objectContaining({
99
+ ok: false,
100
+ error: expect.objectContaining({ code: ERROR_CODES.GIT_FAILURE })
101
+ })
102
+ });
103
+ });
104
+
105
+ it('GIVEN_branch_exists_WHEN_branch_removed_safely_succeeds_THEN_summary_branch_removed_true', async () => {
106
+ vi.mocked(runGit)
107
+ .mockResolvedValueOnce(commandResult({ code: 0 })) // show-ref → branch exists
108
+ .mockResolvedValueOnce(commandResult({ code: 0 })); // branch -d succeeds
109
+
110
+ const service = new FeatureDeletionService(makeMockPort());
111
+
112
+ const result = await service.featureDelete('feature_x', {
113
+ dryRun: false,
114
+ confirm: true,
115
+ removeWorktree: false,
116
+ removeBranch: 'safe'
117
+ });
118
+
119
+ expect(result.data.summary.branch_removed).toBe(true);
120
+ expect(vi.mocked(runGit)).toHaveBeenCalledWith(repoRoot, ['branch', '-d', 'feature_x']);
121
+ });
122
+
123
+ it('GIVEN_branch_exists_WHEN_runGit_branch_remove_fails_THEN_throws_GIT_FAILURE', async () => {
124
+ vi.mocked(runGit)
125
+ .mockResolvedValueOnce(commandResult({ code: 0 })) // show-ref → branch exists
126
+ .mockResolvedValueOnce(commandResult({ code: 1, stderr: 'branch not merged' })); // branch -d fails
127
+
128
+ const service = new FeatureDeletionService(makeMockPort());
129
+
130
+ await expect(
131
+ service.featureDelete('feature_x', { dryRun: false, confirm: true, removeWorktree: false, removeBranch: 'safe' })
132
+ ).rejects.toMatchObject({
133
+ normalizedResponse: expect.objectContaining({
134
+ ok: false,
135
+ error: expect.objectContaining({ code: ERROR_CODES.GIT_FAILURE })
136
+ })
137
+ });
138
+ });
139
+
140
+ it('GIVEN_branch_exists_WHEN_force_remove_mode_THEN_uses_dash_capital_D_arg', async () => {
141
+ vi.mocked(runGit)
142
+ .mockResolvedValueOnce(commandResult({ code: 0 })) // show-ref → branch exists
143
+ .mockResolvedValueOnce(commandResult({ code: 0 })); // branch -D succeeds
144
+
145
+ const service = new FeatureDeletionService(makeMockPort());
146
+
147
+ const result = await service.featureDelete('feature_x', {
148
+ dryRun: false,
149
+ confirm: true,
150
+ removeWorktree: false,
151
+ removeBranch: 'force'
152
+ });
153
+
154
+ expect(result.data.summary.branch_removed).toBe(true);
155
+ expect(vi.mocked(runGit)).toHaveBeenCalledWith(repoRoot, ['branch', '-D', 'feature_x']);
156
+ });
157
+ });
158
+
159
+ describe('FeatureDeletionService input validation paths', () => {
160
+ let repoRoot: string;
161
+ let featureDir: string;
162
+ let worktreeDir: string;
163
+
164
+ beforeEach(async () => {
165
+ vi.clearAllMocks();
166
+ repoRoot = await fs.mkdtemp(path.join(os.tmpdir(), 'aop-del-val-'));
167
+ featureDir = path.join(repoRoot, '.aop', 'features', 'feature_x');
168
+ worktreeDir = path.join(repoRoot, '.worktrees', 'feature_x');
169
+ await fs.mkdir(featureDir, { recursive: true });
170
+ await fs.mkdir(worktreeDir, { recursive: true });
171
+ });
172
+
173
+ afterEach(async () => {
174
+ await fs.rm(repoRoot, { recursive: true, force: true });
175
+ });
176
+
177
+ function makeMockPort() {
178
+ const staleRunLease = makeStaleRunLease();
179
+ return {
180
+ getRepoRoot: () => repoRoot,
181
+ featurePath: (_featureId: string) => featureDir,
182
+ worktreePath: (_featureId: string) => worktreeDir,
183
+ statePath: (_featureId: string) => path.join(featureDir, 'state.md'),
184
+ readState: vi.fn(async () => ({ frontMatter: {}, body: '' })),
185
+ withIndexLock: async <T>(op: () => Promise<T>) => await op(),
186
+ readIndex: vi.fn(async () => ({
187
+ version: 1,
188
+ active: ['feature_x'],
189
+ blocked: [],
190
+ merged: [],
191
+ locks: {},
192
+ lock_leases: {},
193
+ blocked_queue: [],
194
+ updated_at: new Date().toISOString()
195
+ })),
196
+ writeIndex: vi.fn(async () => {}),
197
+ readRunLease: vi.fn(async () => ({ ...staleRunLease })),
198
+ writeRunLease: vi.fn(async () => {}),
199
+ normalizeRuntimeSessions: vi.fn(() => staleRunLease),
200
+ isRunLeaseFresh: vi.fn(() => false),
201
+ locksRelease: vi.fn(async () => {})
202
+ };
203
+ }
204
+
205
+ it('GIVEN_null_feature_id_WHEN_featureDelete_called_THEN_throws_INVALID_ARGUMENT', async () => {
206
+ const service = new FeatureDeletionService(makeMockPort());
207
+
208
+ await expect(
209
+ service.featureDelete(null)
210
+ ).rejects.toMatchObject({
211
+ normalizedResponse: expect.objectContaining({
212
+ ok: false,
213
+ error: expect.objectContaining({ code: ERROR_CODES.INVALID_ARGUMENT })
214
+ })
215
+ });
216
+ });
217
+
218
+ it('GIVEN_invalid_feature_id_format_WHEN_featureDelete_called_THEN_throws_INVALID_ARGUMENT', async () => {
219
+ const service = new FeatureDeletionService(makeMockPort());
220
+
221
+ await expect(
222
+ service.featureDelete('INVALID FEATURE ID!')
223
+ ).rejects.toMatchObject({
224
+ normalizedResponse: expect.objectContaining({
225
+ ok: false,
226
+ error: expect.objectContaining({ code: ERROR_CODES.INVALID_ARGUMENT })
227
+ })
228
+ });
229
+ });
230
+
231
+ it('GIVEN_invalid_remove_branch_value_WHEN_featureDelete_called_THEN_throws_INVALID_ARGUMENT', async () => {
232
+ const service = new FeatureDeletionService(makeMockPort());
233
+
234
+ await expect(
235
+ service.featureDelete('feature_x', { removeBranch: 'invalid-mode' })
236
+ ).rejects.toMatchObject({
237
+ normalizedResponse: expect.objectContaining({
238
+ ok: false,
239
+ error: expect.objectContaining({ code: ERROR_CODES.INVALID_ARGUMENT })
240
+ })
241
+ });
242
+ });
243
+
244
+ it('GIVEN_dry_run_false_and_confirm_false_WHEN_featureDelete_called_THEN_throws_INVALID_ARGUMENT', async () => {
245
+ const service = new FeatureDeletionService(makeMockPort());
246
+
247
+ await expect(
248
+ service.featureDelete('feature_x', { dryRun: false, confirm: false, removeBranch: 'none' })
249
+ ).rejects.toMatchObject({
250
+ normalizedResponse: expect.objectContaining({
251
+ ok: false,
252
+ error: expect.objectContaining({ code: ERROR_CODES.INVALID_ARGUMENT })
253
+ })
254
+ });
255
+ });
256
+
257
+ it('GIVEN_dry_run_true_WHEN_featureDelete_called_THEN_returns_plan_without_applying_changes', async () => {
258
+ vi.mocked(runGit).mockResolvedValue(commandResult({ code: 1 })); // no branch
259
+
260
+ const service = new FeatureDeletionService(makeMockPort());
261
+
262
+ const result = await service.featureDelete('feature_x', { dryRun: true, removeBranch: 'none' });
263
+
264
+ expect(result.data.dry_run).toBe(true);
265
+ expect(result.data.plan).toBeDefined();
266
+ expect(result.data.applied).toBeUndefined();
267
+ });
268
+
269
+ it('GIVEN_fresh_run_lease_WHEN_featureDelete_called_THEN_throws_RUN_ALREADY_ACTIVE', async () => {
270
+ vi.mocked(runGit).mockResolvedValue(commandResult({ code: 1 })); // no branch
271
+
272
+ const now = new Date().toISOString();
273
+ const freshLease: RuntimeSessionsSnapshot = {
274
+ run_id: 'run:active',
275
+ orchestrator_session_id: 'orch-1',
276
+ provider: 'custom',
277
+ model: 'model-x',
278
+ provider_config_ref_hash: 'hash',
279
+ owner_instance_id: 'owner:1',
280
+ lease_id: 'lease:1',
281
+ started_at: now,
282
+ last_heartbeat_at: now,
283
+ lease_expires_at: new Date(Date.now() + 60_000).toISOString(),
284
+ feature_sessions: {}
285
+ };
286
+
287
+ const port = makeMockPort();
288
+ port.readRunLease.mockResolvedValue(freshLease);
289
+ port.isRunLeaseFresh.mockReturnValue(true);
290
+
291
+ const service = new FeatureDeletionService(port);
292
+
293
+ await expect(
294
+ service.featureDelete('feature_x', { dryRun: false, confirm: true, removeBranch: 'none' })
295
+ ).rejects.toMatchObject({
296
+ normalizedResponse: expect.objectContaining({
297
+ ok: false,
298
+ error: expect.objectContaining({ code: ERROR_CODES.RUN_ALREADY_ACTIVE })
299
+ })
300
+ });
301
+ });
302
+
303
+ it('GIVEN_non_dry_run_with_no_branch_and_no_worktree_WHEN_featureDelete_called_THEN_removes_feature_dir', async () => {
304
+ vi.mocked(runGit).mockResolvedValue(commandResult({ code: 1 })); // no branch
305
+
306
+ const service = new FeatureDeletionService(makeMockPort());
307
+
308
+ const result = await service.featureDelete('feature_x', {
309
+ dryRun: false,
310
+ confirm: true,
311
+ removeWorktree: false,
312
+ removeBranch: 'none'
313
+ });
314
+
315
+ expect(result.data.dry_run).toBe(false);
316
+ expect(result.data.summary.feature_dir_removed).toBe(true);
317
+ });
318
+ });
319
+
320
+ describe('FeatureDeletionService readLockResourcesFromIndex and normalizeFeatureSessions branches', () => {
321
+ let repoRoot: string;
322
+ let featureDir: string;
323
+ let worktreeDir: string;
324
+
325
+ beforeEach(async () => {
326
+ vi.clearAllMocks();
327
+ repoRoot = await fs.mkdtemp(path.join(os.tmpdir(), 'aop-del-extra-'));
328
+ featureDir = path.join(repoRoot, '.aop', 'features', 'feature_lock_lease');
329
+ worktreeDir = path.join(repoRoot, '.worktrees', 'feature_lock_lease');
330
+ await fs.mkdir(featureDir, { recursive: true });
331
+ await fs.mkdir(worktreeDir, { recursive: true });
332
+ });
333
+
334
+ afterEach(async () => {
335
+ await fs.rm(repoRoot, { recursive: true, force: true });
336
+ });
337
+
338
+ it('GIVEN_lock_leases_with_holder_matching_featureId_WHEN_featureDelete_THEN_lock_released', async () => {
339
+ vi.mocked(runGit).mockResolvedValue(commandResult({ code: 1 }));
340
+
341
+ const staleRunLease = makeStaleRunLease();
342
+ const port = {
343
+ getRepoRoot: () => repoRoot,
344
+ featurePath: () => featureDir,
345
+ worktreePath: () => worktreeDir,
346
+ statePath: () => path.join(featureDir, 'state.md'),
347
+ readState: vi.fn(async () => ({ frontMatter: {}, body: '' })),
348
+ withIndexLock: async <T>(op: () => Promise<T>) => await op(),
349
+ readIndex: vi.fn(async () => ({
350
+ version: 1,
351
+ active: ['feature_lock_lease'],
352
+ blocked: [],
353
+ merged: [],
354
+ locks: {},
355
+ lock_leases: {
356
+ openapi: { holder: 'feature_lock_lease', expires_at: new Date().toISOString() }
357
+ },
358
+ blocked_queue: [],
359
+ updated_at: new Date().toISOString()
360
+ })),
361
+ writeIndex: vi.fn(async () => {}),
362
+ readRunLease: vi.fn(async () => ({ ...staleRunLease })),
363
+ writeRunLease: vi.fn(async () => {}),
364
+ normalizeRuntimeSessions: vi.fn(() => staleRunLease),
365
+ isRunLeaseFresh: vi.fn(() => false),
366
+ locksRelease: vi.fn(async () => {})
367
+ };
368
+
369
+ const service = new FeatureDeletionService(port);
370
+ const result = await service.featureDelete('feature_lock_lease', {
371
+ dryRun: false,
372
+ confirm: true,
373
+ removeWorktree: false,
374
+ removeBranch: 'none'
375
+ });
376
+
377
+ expect(result.data.summary.locks_released).toContain('openapi');
378
+ });
379
+
380
+ it('GIVEN_feature_sessions_is_undefined_WHEN_normalizeFeatureSessions_called_THEN_returns_empty_object', async () => {
381
+ vi.mocked(runGit).mockResolvedValue(commandResult({ code: 1 }));
382
+
383
+ const staleRunLease = {
384
+ ...makeStaleRunLease(),
385
+ feature_sessions: undefined as unknown as Record<string, unknown>
386
+ };
387
+ const port = {
388
+ getRepoRoot: () => repoRoot,
389
+ featurePath: () => featureDir,
390
+ worktreePath: () => worktreeDir,
391
+ statePath: () => path.join(featureDir, 'state.md'),
392
+ readState: vi.fn(async () => ({ frontMatter: {}, body: '' })),
393
+ withIndexLock: async <T>(op: () => Promise<T>) => await op(),
394
+ readIndex: vi.fn(async () => ({
395
+ version: 1,
396
+ active: ['feature_lock_lease'],
397
+ blocked: [],
398
+ merged: [],
399
+ locks: {},
400
+ lock_leases: {},
401
+ blocked_queue: [],
402
+ updated_at: new Date().toISOString()
403
+ })),
404
+ writeIndex: vi.fn(async () => {}),
405
+ readRunLease: vi.fn(async () => ({ ...staleRunLease })),
406
+ writeRunLease: vi.fn(async () => {}),
407
+ normalizeRuntimeSessions: vi.fn(() => ({ ...staleRunLease })),
408
+ isRunLeaseFresh: vi.fn(() => false),
409
+ locksRelease: vi.fn(async () => {})
410
+ };
411
+
412
+ const service = new FeatureDeletionService(port);
413
+ const result = await service.featureDelete('feature_lock_lease', {
414
+ dryRun: false,
415
+ confirm: true,
416
+ removeWorktree: false,
417
+ removeBranch: 'none'
418
+ });
419
+
420
+ expect(result.data.summary.runtime_session_assignment_removed).toBe(false);
421
+ });
422
+ });
@@ -0,0 +1,202 @@
1
+ import { beforeEach, describe, expect, it, vi } from 'vitest';
2
+ import path from 'node:path';
3
+ import os from 'node:os';
4
+ import fs from 'node:fs/promises';
5
+ import { FeatureLifecycleService } from '../src/application/services/feature-lifecycle-service.js';
6
+ import type { FeatureLifecycleServicePort } from '../src/application/services/feature-lifecycle-service.js';
7
+
8
+ function makeTmpDir(): string {
9
+ return path.join(os.tmpdir(), `aop-lifecycle-test-${process.pid}-${Date.now()}`);
10
+ }
11
+
12
+ function makePort(overrides: Partial<FeatureLifecycleServicePort> = {}): FeatureLifecycleServicePort {
13
+ const repoRoot = '/tmp/repo';
14
+ const featuresDir = '/tmp/repo/.aop/features';
15
+
16
+ return {
17
+ getRepoRoot: vi.fn(() => repoRoot),
18
+ getFeaturesDir: vi.fn(() => featuresDir),
19
+ featurePath: vi.fn((id) => `${featuresDir}/${id}`),
20
+ statePath: vi.fn((id) => `${featuresDir}/${id}/state.md`),
21
+ qaIndexPath: vi.fn((id) => `${featuresDir}/${id}/qa_test_index.json`),
22
+ decisionsPath: vi.fn((id) => `${featuresDir}/${id}/decisions.md`),
23
+ specPath: vi.fn((id) => `${featuresDir}/${id}/spec.md`),
24
+ ensureFeatureLayout: vi.fn(async () => {}),
25
+ repoEnsureWorktree: vi.fn(async () => ({ data: { worktree_path_abs: `/tmp/worktrees/f1` } })),
26
+ makeDefaultState: vi.fn(() => ({ feature_id: 'f1', status: 'planning', version: 0 })),
27
+ writeState: vi.fn(async () => {}),
28
+ withIndexLock: vi.fn(async (op: () => Promise<unknown>) => await op()),
29
+ readIndex: vi.fn(async () => ({ active: [], blocked: [], version: 0, updated_at: '' })),
30
+ writeIndex: vi.fn(async () => {}),
31
+ featureStateGet: vi.fn(async () => ({ data: { front_matter: { version: 1, status: 'planning' }, body: '' } })),
32
+ planGet: vi.fn(async () => ({ data: { plan: { files: {} } } })),
33
+ qaTestIndexGet: vi.fn(async () => ({ data: { items: [] } })),
34
+ evidenceLatest: vi.fn(async () => ({ data: { latest: null } })),
35
+ ...overrides
36
+ };
37
+ }
38
+
39
+ describe('FeatureLifecycleService', () => {
40
+ beforeEach(() => {
41
+ vi.clearAllMocks();
42
+ });
43
+
44
+ describe('featureDiscoverSpecs', () => {
45
+ it('GIVEN_features_dir_does_not_exist_WHEN_featureDiscoverSpecs_called_THEN_returns_empty_specs', async () => {
46
+ const tmpDir = makeTmpDir();
47
+ const port = makePort({
48
+ getRepoRoot: vi.fn(() => tmpDir),
49
+ getFeaturesDir: vi.fn(() => path.join(tmpDir, '.aop', 'features'))
50
+ });
51
+ const service = new FeatureLifecycleService(port);
52
+ const result = await service.featureDiscoverSpecs();
53
+ expect(result.data.specs).toEqual([]);
54
+ });
55
+
56
+ it('GIVEN_features_dir_with_spec_files_WHEN_featureDiscoverSpecs_called_THEN_returns_specs', async () => {
57
+ const tmpDir = await fs.mkdtemp(path.join(os.tmpdir(), 'aop-lc-'));
58
+ try {
59
+ const featuresDir = path.join(tmpDir, '.aop', 'features');
60
+ const f1 = path.join(featuresDir, 'feature_a');
61
+ const f2 = path.join(featuresDir, 'not_a_feature'); // no spec.md
62
+ await fs.mkdir(f1, { recursive: true });
63
+ await fs.mkdir(f2, { recursive: true });
64
+ await fs.writeFile(path.join(f1, 'spec.md'), '# Feature A', 'utf8');
65
+
66
+ const port = makePort({
67
+ getRepoRoot: vi.fn(() => tmpDir),
68
+ getFeaturesDir: vi.fn(() => featuresDir)
69
+ });
70
+ const service = new FeatureLifecycleService(port);
71
+ const result = await service.featureDiscoverSpecs();
72
+ expect(result.data.specs).toHaveLength(1);
73
+ expect(result.data.specs[0].feature_id).toBe('feature_a');
74
+ } finally {
75
+ await fs.rm(tmpDir, { recursive: true, force: true });
76
+ }
77
+ });
78
+ });
79
+
80
+ describe('featureInit', () => {
81
+ it('GIVEN_null_featureId_WHEN_featureInit_called_THEN_throws_INVALID_ARGUMENT', async () => {
82
+ const service = new FeatureLifecycleService(makePort());
83
+ await expect(service.featureInit(null)).rejects.toMatchObject({
84
+ normalizedResponse: { error: { code: 'invalid_argument' } }
85
+ });
86
+ });
87
+
88
+ it('GIVEN_valid_featureId_with_no_existing_state_WHEN_featureInit_called_THEN_creates_state_and_qa_index', async () => {
89
+ const tmpDir = await fs.mkdtemp(path.join(os.tmpdir(), 'aop-lc2-'));
90
+ try {
91
+ const port = makePort({
92
+ getRepoRoot: vi.fn(() => tmpDir),
93
+ statePath: vi.fn(() => path.join(tmpDir, 'nonexistent-state.md')),
94
+ qaIndexPath: vi.fn(() => path.join(tmpDir, 'nonexistent-qa.json'))
95
+ });
96
+ const service = new FeatureLifecycleService(port);
97
+ const result = await service.featureInit('test_feature');
98
+ expect(result.data.initialized).toBe(true);
99
+ expect(port.writeState).toHaveBeenCalledOnce();
100
+ expect(port.writeIndex).toHaveBeenCalledOnce();
101
+ } finally {
102
+ await fs.rm(tmpDir, { recursive: true, force: true });
103
+ }
104
+ });
105
+
106
+ it('GIVEN_featureId_with_existing_state_and_qa_index_WHEN_featureInit_called_THEN_skips_creation', async () => {
107
+ const tmpDir = await fs.mkdtemp(path.join(os.tmpdir(), 'aop-lc3-'));
108
+ try {
109
+ await fs.writeFile(path.join(tmpDir, 'state.md'), '---\nfeature_id: f1\n---', 'utf8');
110
+ await fs.writeFile(path.join(tmpDir, 'qa.json'), '{}', 'utf8');
111
+
112
+ const port = makePort({
113
+ getRepoRoot: vi.fn(() => tmpDir),
114
+ statePath: vi.fn(() => path.join(tmpDir, 'state.md')),
115
+ qaIndexPath: vi.fn(() => path.join(tmpDir, 'qa.json'))
116
+ });
117
+ const service = new FeatureLifecycleService(port);
118
+ await service.featureInit('f1');
119
+ expect(port.writeState).not.toHaveBeenCalled();
120
+ } finally {
121
+ await fs.rm(tmpDir, { recursive: true, force: true });
122
+ }
123
+ });
124
+ });
125
+
126
+ describe('featureGetContext', () => {
127
+ it('GIVEN_featureId_with_no_spec_file_WHEN_featureGetContext_called_THEN_spec_is_empty_string', async () => {
128
+ const port = makePort({
129
+ specPath: vi.fn(() => '/nonexistent/spec.md')
130
+ });
131
+ const service = new FeatureLifecycleService(port);
132
+ const result = await service.featureGetContext('f1');
133
+ expect(result.data.spec).toBe('');
134
+ expect(result.data.feature_id).toBe('f1');
135
+ });
136
+
137
+ it('GIVEN_featureId_with_evidence_WHEN_featureGetContext_called_THEN_latest_evidence_included', async () => {
138
+ const evidence = { gate_profile: 'fast', passed: true };
139
+ const port = makePort({
140
+ evidenceLatest: vi.fn(async () => ({ data: { latest: evidence } })),
141
+ specPath: vi.fn(() => '/nonexistent/spec.md')
142
+ });
143
+ const service = new FeatureLifecycleService(port);
144
+ const result = await service.featureGetContext('f1');
145
+ expect(result.data.latest_evidence).toEqual(evidence);
146
+ });
147
+ });
148
+
149
+ describe('featureLogAppend', () => {
150
+ it('GIVEN_featureId_and_note_WHEN_featureLogAppend_called_THEN_line_appended_to_log', async () => {
151
+ const tmpDir = await fs.mkdtemp(path.join(os.tmpdir(), 'aop-lc4-'));
152
+ try {
153
+ const logPath = path.join(tmpDir, 'decisions.md');
154
+ const port = makePort({
155
+ getRepoRoot: vi.fn(() => tmpDir),
156
+ decisionsPath: vi.fn(() => logPath)
157
+ });
158
+ const service = new FeatureLifecycleService(port);
159
+ const result = await service.featureLogAppend('f1', 'starting build', { actorType: 'orchestrator', actorId: 'orch-1' });
160
+ expect(result.data.appended).toBe(true);
161
+ const content = await fs.readFile(logPath, 'utf8');
162
+ expect(content).toContain('[orchestrator:orch-1] starting build');
163
+ } finally {
164
+ await fs.rm(tmpDir, { recursive: true, force: true });
165
+ }
166
+ });
167
+ });
168
+ });
169
+
170
+ describe('ToolRouter and ToolHandlerRegistry', () => {
171
+ it('GIVEN_registered_handler_WHEN_route_called_THEN_handler_invoked', async () => {
172
+ const { ToolHandlerRegistry, ToolRouter } = await import('../src/application/tools/tool-router.js');
173
+ const registry = new ToolHandlerRegistry();
174
+ const handler = vi.fn(async () => ({ ok: true }));
175
+ registry.register('test.tool', handler);
176
+
177
+ const unknownHandler = vi.fn(async () => ({ error: 'unknown' }));
178
+ const router = new ToolRouter(registry, unknownHandler);
179
+
180
+ const result = await router.route('test.tool', { arg: 1 }, { actorType: 'orchestrator', actorId: 'o1' });
181
+ expect(handler).toHaveBeenCalledWith({ arg: 1 }, { actorType: 'orchestrator', actorId: 'o1' });
182
+ expect(result).toEqual({ ok: true });
183
+ expect(unknownHandler).not.toHaveBeenCalled();
184
+ });
185
+
186
+ it('GIVEN_no_handler_registered_WHEN_route_called_THEN_unknown_handler_invoked', async () => {
187
+ const { ToolHandlerRegistry, ToolRouter } = await import('../src/application/tools/tool-router.js');
188
+ const registry = new ToolHandlerRegistry();
189
+ const unknownHandler = vi.fn(async (name: string) => ({ error: `unknown: ${name}` }));
190
+ const router = new ToolRouter(registry, unknownHandler);
191
+
192
+ const result = await router.route('nonexistent.tool', {}, { actorType: 'builder', actorId: 'b1' });
193
+ expect(unknownHandler).toHaveBeenCalledWith('nonexistent.tool');
194
+ expect(result).toEqual({ error: 'unknown: nonexistent.tool' });
195
+ });
196
+
197
+ it('GIVEN_registry_WHEN_lookup_called_for_unregistered_tool_THEN_returns_undefined', async () => {
198
+ const { ToolHandlerRegistry } = await import('../src/application/tools/tool-router.js');
199
+ const registry = new ToolHandlerRegistry();
200
+ expect(registry.lookup('missing.tool')).toBeUndefined();
201
+ });
202
+ });
@@ -0,0 +1,24 @@
1
+ import { vi, it, expect, describe } from 'vitest';
2
+ import type * as ChildProcess from 'node:child_process';
3
+
4
+ vi.mock('node:child_process', async (importOriginal) => {
5
+ const actual = await importOriginal<typeof ChildProcess>();
6
+ return {
7
+ ...actual,
8
+ spawn: vi.fn().mockImplementationOnce(() => {
9
+ throw new Error('spawn sync error');
10
+ })
11
+ };
12
+ });
13
+
14
+ import { NodeCommandRunner } from '../src/core/git.js';
15
+
16
+ describe('NodeCommandRunner spawn error', () => {
17
+ it('GIVEN_spawn_throws_synchronously_WHEN_run_called_THEN_returns_code_127', async () => {
18
+ const runner = new NodeCommandRunner();
19
+ const result = await runner.run('git', ['status']);
20
+ expect(result.code).toBe(127);
21
+ expect(result.stderr).toBe('spawn sync error');
22
+ expect(result.timeout).toBe(false);
23
+ });
24
+ });