sdd-agent-platform 0.4.1 → 0.5.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 (698) hide show
  1. package/README.md +24 -28
  2. package/node_modules/@sdd-agent-platform/core/dist/ai-tools.js +84 -103
  3. package/node_modules/@sdd-agent-platform/core/dist/ai-tools.js.map +1 -1
  4. package/node_modules/@sdd-agent-platform/core/dist/config/init-project.d.ts +10 -6
  5. package/node_modules/@sdd-agent-platform/core/dist/config/init-project.js +7 -8
  6. package/node_modules/@sdd-agent-platform/core/dist/config/init-project.js.map +1 -1
  7. package/node_modules/@sdd-agent-platform/core/dist/config/project-config.d.ts +3 -1
  8. package/node_modules/@sdd-agent-platform/core/dist/config/project-config.js +7 -3
  9. package/node_modules/@sdd-agent-platform/core/dist/config/project-config.js.map +1 -1
  10. package/node_modules/@sdd-agent-platform/core/dist/config/starter-documents.d.ts +0 -1
  11. package/node_modules/@sdd-agent-platform/core/dist/config/starter-documents.js +374 -421
  12. package/node_modules/@sdd-agent-platform/core/dist/config/starter-documents.js.map +1 -1
  13. package/node_modules/@sdd-agent-platform/core/dist/context/build-package.d.ts +1 -1
  14. package/node_modules/@sdd-agent-platform/core/dist/context/build-package.js +7 -19
  15. package/node_modules/@sdd-agent-platform/core/dist/context/build-package.js.map +1 -1
  16. package/node_modules/@sdd-agent-platform/core/dist/contracts.d.ts +7 -1
  17. package/node_modules/@sdd-agent-platform/core/dist/contracts.js +6 -0
  18. package/node_modules/@sdd-agent-platform/core/dist/contracts.js.map +1 -1
  19. package/node_modules/@sdd-agent-platform/core/dist/doctor/checks/document-chain.js +2 -12
  20. package/node_modules/@sdd-agent-platform/core/dist/doctor/checks/document-chain.js.map +1 -1
  21. package/node_modules/@sdd-agent-platform/core/dist/doctor/doctor.js +1 -18
  22. package/node_modules/@sdd-agent-platform/core/dist/doctor/doctor.js.map +1 -1
  23. package/node_modules/@sdd-agent-platform/core/dist/evidence/lookup.d.ts +1 -1
  24. package/node_modules/@sdd-agent-platform/core/dist/evidence/lookup.js +1 -1
  25. package/node_modules/@sdd-agent-platform/core/dist/evidence/lookup.js.map +1 -1
  26. package/node_modules/@sdd-agent-platform/core/dist/evidence-runtime/contracts.d.ts +0 -1
  27. package/node_modules/@sdd-agent-platform/core/dist/evidence-runtime/coordination.js +110 -0
  28. package/node_modules/@sdd-agent-platform/core/dist/evidence-runtime/coordination.js.map +1 -0
  29. package/node_modules/@sdd-agent-platform/core/dist/execution/host-invocation.js +83 -83
  30. package/node_modules/@sdd-agent-platform/core/dist/instructions.d.ts +1 -1
  31. package/node_modules/@sdd-agent-platform/core/dist/instructions.js +37 -80
  32. package/node_modules/@sdd-agent-platform/core/dist/instructions.js.map +1 -1
  33. package/node_modules/@sdd-agent-platform/core/dist/lifecycle/ship.js +58 -68
  34. package/node_modules/@sdd-agent-platform/core/dist/lifecycle/ship.js.map +1 -1
  35. package/node_modules/@sdd-agent-platform/core/dist/lifecycle-graph/contracts.d.ts +159 -0
  36. package/node_modules/@sdd-agent-platform/core/dist/lifecycle-graph/contracts.js +7 -0
  37. package/node_modules/@sdd-agent-platform/core/dist/lifecycle-graph/contracts.js.map +1 -0
  38. package/node_modules/@sdd-agent-platform/core/dist/lifecycle-graph/kernel.d.ts +16 -0
  39. package/node_modules/@sdd-agent-platform/core/dist/lifecycle-graph/kernel.js +461 -0
  40. package/node_modules/@sdd-agent-platform/core/dist/lifecycle-graph/kernel.js.map +1 -0
  41. package/node_modules/@sdd-agent-platform/core/dist/lifecycle-graph.d.ts +2 -0
  42. package/node_modules/@sdd-agent-platform/core/dist/lifecycle-graph.js +3 -0
  43. package/node_modules/@sdd-agent-platform/core/dist/lifecycle-graph.js.map +1 -0
  44. package/node_modules/@sdd-agent-platform/core/dist/orchestration/contracts.d.ts +1 -1
  45. package/node_modules/@sdd-agent-platform/core/dist/orchestration/runtime.js +21 -28
  46. package/node_modules/@sdd-agent-platform/core/dist/orchestration/runtime.js.map +1 -1
  47. package/node_modules/@sdd-agent-platform/core/dist/registries/agent-registry.js +124 -40
  48. package/node_modules/@sdd-agent-platform/core/dist/registries/agent-registry.js.map +1 -1
  49. package/node_modules/@sdd-agent-platform/core/dist/registries/command-team-runtime.d.ts +1 -1
  50. package/node_modules/@sdd-agent-platform/core/dist/registries/command-team-runtime.js +6 -13
  51. package/node_modules/@sdd-agent-platform/core/dist/registries/command-team-runtime.js.map +1 -1
  52. package/node_modules/@sdd-agent-platform/core/dist/registries/plan-scout-domains.d.ts +13 -0
  53. package/node_modules/@sdd-agent-platform/core/dist/registries/plan-scout-domains.js +76 -0
  54. package/node_modules/@sdd-agent-platform/core/dist/registries/plan-scout-domains.js.map +1 -0
  55. package/node_modules/@sdd-agent-platform/core/dist/registries/skill-capabilities.js +7 -7
  56. package/node_modules/@sdd-agent-platform/core/dist/registries/skill-capabilities.js.map +1 -1
  57. package/node_modules/@sdd-agent-platform/core/dist/registries/tool-capabilities.js +6 -6
  58. package/node_modules/@sdd-agent-platform/core/dist/registries/tool-capabilities.js.map +1 -1
  59. package/node_modules/@sdd-agent-platform/core/dist/registries/workflow-gates.d.ts +1 -1
  60. package/node_modules/@sdd-agent-platform/core/dist/registries/workflow-gates.js +18 -18
  61. package/node_modules/@sdd-agent-platform/core/dist/registries/workflow-gates.js.map +1 -1
  62. package/node_modules/@sdd-agent-platform/core/dist/risk/consumer-diagnostics.js +2 -1
  63. package/node_modules/@sdd-agent-platform/core/dist/risk/consumer-diagnostics.js.map +1 -1
  64. package/node_modules/@sdd-agent-platform/core/dist/risk/contracts.d.ts +2 -2
  65. package/node_modules/@sdd-agent-platform/core/dist/risk/kernel.js +7 -7
  66. package/node_modules/@sdd-agent-platform/core/dist/risk/kernel.js.map +1 -1
  67. package/node_modules/@sdd-agent-platform/core/dist/risk/legacy-adapters.js +12 -27
  68. package/node_modules/@sdd-agent-platform/core/dist/risk/legacy-adapters.js.map +1 -1
  69. package/node_modules/@sdd-agent-platform/core/dist/risk/workflow-gates.js +6 -6
  70. package/node_modules/@sdd-agent-platform/core/dist/risk/workflow-gates.js.map +1 -1
  71. package/node_modules/@sdd-agent-platform/core/dist/router/agent-runtime-config.js +1 -1
  72. package/node_modules/@sdd-agent-platform/core/dist/router/agent-runtime-config.js.map +1 -1
  73. package/node_modules/@sdd-agent-platform/core/dist/router/routing.js +2 -4
  74. package/node_modules/@sdd-agent-platform/core/dist/router/routing.js.map +1 -1
  75. package/node_modules/@sdd-agent-platform/core/dist/router/runtime-import.d.ts +28 -0
  76. package/node_modules/@sdd-agent-platform/core/dist/router/runtime-import.js +383 -0
  77. package/node_modules/@sdd-agent-platform/core/dist/router/runtime-import.js.map +1 -0
  78. package/node_modules/@sdd-agent-platform/core/dist/router/stage-route-binding.d.ts +37 -0
  79. package/node_modules/@sdd-agent-platform/core/dist/router/stage-route-binding.js +227 -0
  80. package/node_modules/@sdd-agent-platform/core/dist/router/stage-route-binding.js.map +1 -0
  81. package/node_modules/@sdd-agent-platform/core/dist/router.d.ts +1 -0
  82. package/node_modules/@sdd-agent-platform/core/dist/router.js +1 -0
  83. package/node_modules/@sdd-agent-platform/core/dist/router.js.map +1 -1
  84. package/node_modules/@sdd-agent-platform/core/dist/run-state/artifacts.d.ts +16 -0
  85. package/node_modules/@sdd-agent-platform/core/dist/run-state/artifacts.js +6 -0
  86. package/node_modules/@sdd-agent-platform/core/dist/run-state/artifacts.js.map +1 -1
  87. package/node_modules/@sdd-agent-platform/core/dist/run-state/model.d.ts +20 -0
  88. package/node_modules/@sdd-agent-platform/core/dist/run-state/run-state.js +7 -7
  89. package/node_modules/@sdd-agent-platform/core/dist/run-state/run-state.js.map +1 -1
  90. package/node_modules/@sdd-agent-platform/core/dist/run-state/task-evidence.d.ts +1 -2
  91. package/node_modules/@sdd-agent-platform/core/dist/run-state/task-evidence.js +2 -9
  92. package/node_modules/@sdd-agent-platform/core/dist/run-state/task-evidence.js.map +1 -1
  93. package/node_modules/@sdd-agent-platform/core/dist/run-state/timing.d.ts +8 -0
  94. package/node_modules/@sdd-agent-platform/core/dist/run-state/timing.js +131 -0
  95. package/node_modules/@sdd-agent-platform/core/dist/run-state/timing.js.map +1 -0
  96. package/node_modules/@sdd-agent-platform/core/dist/runtime-analysis/build.js +1 -4
  97. package/node_modules/@sdd-agent-platform/core/dist/runtime-analysis/build.js.map +1 -1
  98. package/node_modules/@sdd-agent-platform/core/dist/runtime-analysis/findings.js +0 -39
  99. package/node_modules/@sdd-agent-platform/core/dist/runtime-analysis/findings.js.map +1 -1
  100. package/node_modules/@sdd-agent-platform/core/dist/runtime-analysis/model.d.ts +1 -17
  101. package/node_modules/@sdd-agent-platform/core/dist/runtime-paths.d.ts +10 -0
  102. package/node_modules/@sdd-agent-platform/core/dist/runtime-paths.js +65 -0
  103. package/node_modules/@sdd-agent-platform/core/dist/runtime-paths.js.map +1 -1
  104. package/node_modules/@sdd-agent-platform/core/dist/runtime-projection-p0.d.ts +64 -0
  105. package/node_modules/@sdd-agent-platform/core/dist/runtime-projection-p0.js +211 -0
  106. package/node_modules/@sdd-agent-platform/core/dist/runtime-projection-p0.js.map +1 -0
  107. package/node_modules/@sdd-agent-platform/core/dist/sdd-docs/artifact-depth.d.ts +14 -0
  108. package/node_modules/@sdd-agent-platform/core/dist/sdd-docs/artifact-depth.js +179 -0
  109. package/node_modules/@sdd-agent-platform/core/dist/sdd-docs/artifact-depth.js.map +1 -0
  110. package/node_modules/@sdd-agent-platform/core/dist/sdd-docs/task-parser.d.ts +5 -1
  111. package/node_modules/@sdd-agent-platform/core/dist/sdd-docs/task-parser.js +60 -22
  112. package/node_modules/@sdd-agent-platform/core/dist/sdd-docs/task-parser.js.map +1 -1
  113. package/node_modules/@sdd-agent-platform/core/dist/sdd-docs/task-rendering.js +2 -2
  114. package/node_modules/@sdd-agent-platform/core/dist/sdd-docs/task-rendering.js.map +1 -1
  115. package/node_modules/@sdd-agent-platform/core/dist/spec-entry.js +40 -0
  116. package/node_modules/@sdd-agent-platform/core/dist/spec-entry.js.map +1 -0
  117. package/node_modules/@sdd-agent-platform/core/dist/spec-manager-contracts.d.ts +12 -0
  118. package/node_modules/@sdd-agent-platform/core/dist/spec-manager-contracts.js +2 -0
  119. package/node_modules/@sdd-agent-platform/core/dist/spec-manager-contracts.js.map +1 -0
  120. package/node_modules/@sdd-agent-platform/core/dist/stage-artifacts.d.ts +55 -0
  121. package/node_modules/@sdd-agent-platform/core/dist/stage-artifacts.js +315 -0
  122. package/node_modules/@sdd-agent-platform/core/dist/stage-artifacts.js.map +1 -0
  123. package/node_modules/@sdd-agent-platform/core/dist/stage-collaboration-contracts.d.ts +55 -0
  124. package/node_modules/@sdd-agent-platform/core/dist/stage-collaboration-contracts.js +238 -0
  125. package/node_modules/@sdd-agent-platform/core/dist/stage-collaboration-contracts.js.map +1 -0
  126. package/node_modules/@sdd-agent-platform/core/dist/stage-collaboration.d.ts +736 -0
  127. package/node_modules/@sdd-agent-platform/core/dist/stage-collaboration.js +4018 -0
  128. package/node_modules/@sdd-agent-platform/core/dist/stage-collaboration.js.map +1 -0
  129. package/node_modules/@sdd-agent-platform/core/dist/stage-runtime/runtime.js +8 -1
  130. package/node_modules/@sdd-agent-platform/core/dist/stage-runtime/runtime.js.map +1 -1
  131. package/node_modules/@sdd-agent-platform/core/dist/status/project-status.js +25 -1
  132. package/node_modules/@sdd-agent-platform/core/dist/status/project-status.js.map +1 -1
  133. package/node_modules/@sdd-agent-platform/core/dist/storage/runtime-store.d.ts +170 -18
  134. package/node_modules/@sdd-agent-platform/core/dist/storage/runtime-store.js +597 -85
  135. package/node_modules/@sdd-agent-platform/core/dist/storage/runtime-store.js.map +1 -1
  136. package/node_modules/@sdd-agent-platform/core/dist/sync-back/apply.d.ts +1 -17
  137. package/node_modules/@sdd-agent-platform/core/dist/sync-back/apply.js +1 -242
  138. package/node_modules/@sdd-agent-platform/core/dist/sync-back/apply.js.map +1 -1
  139. package/node_modules/@sdd-agent-platform/core/dist/sync-back/inspect.d.ts +1 -110
  140. package/node_modules/@sdd-agent-platform/core/dist/sync-back/inspect.js +1 -496
  141. package/node_modules/@sdd-agent-platform/core/dist/sync-back/inspect.js.map +1 -1
  142. package/node_modules/@sdd-agent-platform/core/dist/sync-back.d.ts +1 -2
  143. package/node_modules/@sdd-agent-platform/core/dist/sync-back.js +1 -2
  144. package/node_modules/@sdd-agent-platform/core/dist/sync-back.js.map +1 -1
  145. package/node_modules/@sdd-agent-platform/core/dist/task-execution-contract.d.ts +167 -0
  146. package/node_modules/@sdd-agent-platform/core/dist/task-execution-contract.js +377 -0
  147. package/node_modules/@sdd-agent-platform/core/dist/task-execution-contract.js.map +1 -0
  148. package/node_modules/@sdd-agent-platform/core/dist/test-support/fixtures.js +329 -314
  149. package/node_modules/@sdd-agent-platform/core/dist/test-support/fixtures.js.map +1 -1
  150. package/node_modules/@sdd-agent-platform/core/dist/test-support/run-state.d.ts +1 -0
  151. package/node_modules/@sdd-agent-platform/core/dist/test-support/run-state.js +31 -0
  152. package/node_modules/@sdd-agent-platform/core/dist/test-support/run-state.js.map +1 -1
  153. package/node_modules/@sdd-agent-platform/core/dist/truth-reconciliation.d.ts +44 -0
  154. package/node_modules/@sdd-agent-platform/core/dist/truth-reconciliation.js +135 -0
  155. package/node_modules/@sdd-agent-platform/core/dist/truth-reconciliation.js.map +1 -0
  156. package/node_modules/@sdd-agent-platform/core/dist/tsconfig.tsbuildinfo +1 -1
  157. package/node_modules/@sdd-agent-platform/core/dist/verification/goal-verify.d.ts +0 -49
  158. package/node_modules/@sdd-agent-platform/core/dist/verification/goal-verify.js +1 -545
  159. package/node_modules/@sdd-agent-platform/core/dist/verification/goal-verify.js.map +1 -1
  160. package/node_modules/@sdd-agent-platform/core/dist/verification/rendering.d.ts +5 -7
  161. package/node_modules/@sdd-agent-platform/core/dist/verification/rendering.js +15 -55
  162. package/node_modules/@sdd-agent-platform/core/dist/verification/rendering.js.map +1 -1
  163. package/node_modules/@sdd-agent-platform/core/dist/verification/single-task-loop.js +1 -40
  164. package/node_modules/@sdd-agent-platform/core/dist/verification/single-task-loop.js.map +1 -1
  165. package/node_modules/@sdd-agent-platform/core/dist/verification/task-evidence-judgment.d.ts +49 -0
  166. package/node_modules/@sdd-agent-platform/core/dist/verification/task-evidence-judgment.js +521 -0
  167. package/node_modules/@sdd-agent-platform/core/dist/verification/task-evidence-judgment.js.map +1 -0
  168. package/node_modules/@sdd-agent-platform/core/dist/verification/test-runtime.d.ts +12 -2
  169. package/node_modules/@sdd-agent-platform/core/dist/verification/test-runtime.js +247 -112
  170. package/node_modules/@sdd-agent-platform/core/dist/verification/test-runtime.js.map +1 -1
  171. package/node_modules/@sdd-agent-platform/core/dist/verification/validation-cache.d.ts +26 -0
  172. package/node_modules/@sdd-agent-platform/core/dist/verification/validation-cache.js +73 -0
  173. package/node_modules/@sdd-agent-platform/core/dist/verification/validation-cache.js.map +1 -0
  174. package/node_modules/@sdd-agent-platform/core/dist/verification/verify-contract.d.ts +1 -1
  175. package/node_modules/@sdd-agent-platform/core/dist/verification/verify-contract.js +49 -72
  176. package/node_modules/@sdd-agent-platform/core/dist/verification/verify-contract.js.map +1 -1
  177. package/node_modules/@sdd-agent-platform/core/dist/verification.d.ts +3 -3
  178. package/node_modules/@sdd-agent-platform/core/dist/verification.js +2 -2
  179. package/node_modules/@sdd-agent-platform/core/dist/verification.js.map +1 -1
  180. package/node_modules/@sdd-agent-platform/core/dist/workflow-gate/evidence-packet.js +2 -7
  181. package/node_modules/@sdd-agent-platform/core/dist/workflow-gate/evidence-packet.js.map +1 -1
  182. package/node_modules/@sdd-agent-platform/core/dist/workflow-gate/hard-checks.js +0 -7
  183. package/node_modules/@sdd-agent-platform/core/dist/workflow-gate/hard-checks.js.map +1 -1
  184. package/node_modules/@sdd-agent-platform/core/dist/workflow-gate/policy.js +2 -4
  185. package/node_modules/@sdd-agent-platform/core/dist/workflow-gate/policy.js.map +1 -1
  186. package/node_modules/@sdd-agent-platform/core/dist/workflow-gate/types.d.ts +3 -5
  187. package/node_modules/@sdd-agent-platform/core/dist/workflow-state/latest-eligible-run.js +30 -4
  188. package/node_modules/@sdd-agent-platform/core/dist/workflow-state/latest-eligible-run.js.map +1 -1
  189. package/node_modules/@sdd-agent-platform/core/dist/workflow-state/migration-recovery.d.ts +40 -0
  190. package/node_modules/@sdd-agent-platform/core/dist/workflow-state/migration-recovery.js +110 -0
  191. package/node_modules/@sdd-agent-platform/core/dist/workflow-state/migration-recovery.js.map +1 -0
  192. package/node_modules/@sdd-agent-platform/core/dist/workflow-state/repair-contract.d.ts +12 -0
  193. package/node_modules/@sdd-agent-platform/core/dist/workflow-state/repair-contract.js +63 -0
  194. package/node_modules/@sdd-agent-platform/core/dist/workflow-state/repair-contract.js.map +1 -0
  195. package/node_modules/@sdd-agent-platform/core/dist/workflow-state/resolve-task-run.d.ts +21 -0
  196. package/node_modules/@sdd-agent-platform/core/dist/workflow-state/resolve-task-run.js +95 -0
  197. package/node_modules/@sdd-agent-platform/core/dist/workflow-state/resolve-task-run.js.map +1 -0
  198. package/node_modules/@sdd-agent-platform/core/dist/workflow-state/resolve.d.ts +55 -5
  199. package/node_modules/@sdd-agent-platform/core/dist/workflow-state/resolve.js +518 -36
  200. package/node_modules/@sdd-agent-platform/core/dist/workflow-state/resolve.js.map +1 -1
  201. package/node_modules/@sdd-agent-platform/core/dist/workflow-state/runtime-projections.d.ts +228 -0
  202. package/node_modules/@sdd-agent-platform/core/dist/workflow-state/runtime-projections.js +452 -0
  203. package/node_modules/@sdd-agent-platform/core/dist/workflow-state/runtime-projections.js.map +1 -0
  204. package/node_modules/@sdd-agent-platform/core/package.json +6 -3
  205. package/node_modules/@sdd-agent-platform/core/src/ai-tools.test.ts +238 -137
  206. package/node_modules/@sdd-agent-platform/core/src/ai-tools.ts +84 -103
  207. package/node_modules/@sdd-agent-platform/core/src/artifacts/ingestion.test.ts +189 -189
  208. package/node_modules/@sdd-agent-platform/core/src/artifacts/ingestion.ts +222 -222
  209. package/node_modules/@sdd-agent-platform/core/src/artifacts/sdd-evidence.test.ts +28 -28
  210. package/node_modules/@sdd-agent-platform/core/src/artifacts/sdd-evidence.ts +302 -302
  211. package/node_modules/@sdd-agent-platform/core/src/artifacts/sdd-result.test.ts +181 -181
  212. package/node_modules/@sdd-agent-platform/core/src/artifacts/sdd-result.ts +231 -231
  213. package/node_modules/@sdd-agent-platform/core/src/artifacts/templates.ts +99 -99
  214. package/node_modules/@sdd-agent-platform/core/src/artifacts.ts +4 -4
  215. package/node_modules/@sdd-agent-platform/core/src/coding-facts/contracts.ts +79 -79
  216. package/node_modules/@sdd-agent-platform/core/src/coding-facts.ts +1 -1
  217. package/node_modules/@sdd-agent-platform/core/src/config/init-project.test.ts +314 -306
  218. package/node_modules/@sdd-agent-platform/core/src/config/init-project.ts +128 -120
  219. package/node_modules/@sdd-agent-platform/core/src/config/project-config.ts +265 -259
  220. package/node_modules/@sdd-agent-platform/core/src/config/project-detection.ts +147 -147
  221. package/node_modules/@sdd-agent-platform/core/src/config/starter-documents.ts +400 -445
  222. package/node_modules/@sdd-agent-platform/core/src/context/budget.ts +30 -30
  223. package/node_modules/@sdd-agent-platform/core/src/context/build-package.ts +305 -317
  224. package/node_modules/@sdd-agent-platform/core/src/context/command-summary.ts +45 -45
  225. package/node_modules/@sdd-agent-platform/core/src/context/context-build.test.ts +188 -188
  226. package/node_modules/@sdd-agent-platform/core/src/context/evidence-summary.ts +144 -144
  227. package/node_modules/@sdd-agent-platform/core/src/context/log-worker.ts +48 -48
  228. package/node_modules/@sdd-agent-platform/core/src/context/source-refs.ts +41 -41
  229. package/node_modules/@sdd-agent-platform/core/src/context-offload/contracts.ts +47 -47
  230. package/node_modules/@sdd-agent-platform/core/src/context-offload/runtime.test.ts +71 -71
  231. package/node_modules/@sdd-agent-platform/core/src/context-offload/runtime.ts +178 -178
  232. package/node_modules/@sdd-agent-platform/core/src/context-offload.ts +2 -2
  233. package/node_modules/@sdd-agent-platform/core/src/context.ts +6 -6
  234. package/node_modules/@sdd-agent-platform/core/src/contracts/issues.ts +13 -13
  235. package/node_modules/@sdd-agent-platform/core/src/contracts.test.ts +9 -9
  236. package/node_modules/@sdd-agent-platform/core/src/contracts.ts +121 -115
  237. package/node_modules/@sdd-agent-platform/core/src/delegation/delegation.test.ts +183 -183
  238. package/node_modules/@sdd-agent-platform/core/src/delegation/model.ts +23 -23
  239. package/node_modules/@sdd-agent-platform/core/src/delegation/queue.ts +58 -58
  240. package/node_modules/@sdd-agent-platform/core/src/delegation/run-state.ts +14 -14
  241. package/node_modules/@sdd-agent-platform/core/src/delegation/state-machine.ts +90 -90
  242. package/node_modules/@sdd-agent-platform/core/src/delegation/validation.ts +124 -124
  243. package/node_modules/@sdd-agent-platform/core/src/delegation.ts +26 -26
  244. package/node_modules/@sdd-agent-platform/core/src/doctor/checks/ai-entries.ts +28 -28
  245. package/node_modules/@sdd-agent-platform/core/src/doctor/checks/document-chain.ts +104 -112
  246. package/node_modules/@sdd-agent-platform/core/src/doctor/checks/local-run-index.ts +27 -27
  247. package/node_modules/@sdd-agent-platform/core/src/doctor/checks/project.ts +84 -84
  248. package/node_modules/@sdd-agent-platform/core/src/doctor/checks/registries.ts +252 -252
  249. package/node_modules/@sdd-agent-platform/core/src/doctor/checks/run-evidence.ts +330 -330
  250. package/node_modules/@sdd-agent-platform/core/src/doctor/checks/run-records.ts +79 -79
  251. package/node_modules/@sdd-agent-platform/core/src/doctor/checks/run-trust.ts +128 -128
  252. package/node_modules/@sdd-agent-platform/core/src/doctor/checks/runtime-contracts.ts +300 -300
  253. package/node_modules/@sdd-agent-platform/core/src/doctor/doctor.test.ts +627 -657
  254. package/node_modules/@sdd-agent-platform/core/src/doctor/doctor.ts +301 -318
  255. package/node_modules/@sdd-agent-platform/core/src/doctor/model.ts +13 -13
  256. package/node_modules/@sdd-agent-platform/core/src/doctor/summary.ts +11 -11
  257. package/node_modules/@sdd-agent-platform/core/src/doctor.ts +2 -2
  258. package/node_modules/@sdd-agent-platform/core/src/evidence/lookup.ts +80 -80
  259. package/node_modules/@sdd-agent-platform/core/src/evidence-runtime/contracts.ts +48 -49
  260. package/node_modules/@sdd-agent-platform/core/src/evidence-runtime.ts +1 -1
  261. package/node_modules/@sdd-agent-platform/core/src/execution/agent-execution-records.ts +195 -195
  262. package/node_modules/@sdd-agent-platform/core/src/execution/background-executor.test.ts +187 -187
  263. package/node_modules/@sdd-agent-platform/core/src/execution/background-executor.ts +305 -305
  264. package/node_modules/@sdd-agent-platform/core/src/execution/foreground-subagents.test.ts +97 -97
  265. package/node_modules/@sdd-agent-platform/core/src/execution/foreground-subagents.ts +453 -453
  266. package/node_modules/@sdd-agent-platform/core/src/execution/host-invocation.ts +225 -225
  267. package/node_modules/@sdd-agent-platform/core/src/execution/resident-worker.test.ts +132 -132
  268. package/node_modules/@sdd-agent-platform/core/src/execution/resident-worker.ts +436 -436
  269. package/node_modules/@sdd-agent-platform/core/src/execution/stage-team-runtime.test.ts +102 -102
  270. package/node_modules/@sdd-agent-platform/core/src/execution/stage-team-runtime.ts +271 -271
  271. package/node_modules/@sdd-agent-platform/core/src/execution/wave-executor.test.ts +111 -111
  272. package/node_modules/@sdd-agent-platform/core/src/execution/wave-executor.ts +231 -231
  273. package/node_modules/@sdd-agent-platform/core/src/execution.ts +5 -5
  274. package/node_modules/@sdd-agent-platform/core/src/governance/policy.test.ts +57 -57
  275. package/node_modules/@sdd-agent-platform/core/src/governance/policy.ts +175 -175
  276. package/node_modules/@sdd-agent-platform/core/src/governance.ts +1 -1
  277. package/node_modules/@sdd-agent-platform/core/src/instructions.test.ts +80 -49
  278. package/node_modules/@sdd-agent-platform/core/src/instructions.ts +38 -81
  279. package/node_modules/@sdd-agent-platform/core/src/lifecycle/decision-gate.test.ts +174 -174
  280. package/node_modules/@sdd-agent-platform/core/src/lifecycle/decision-gate.ts +373 -373
  281. package/node_modules/@sdd-agent-platform/core/src/lifecycle/rendering.ts +29 -29
  282. package/node_modules/@sdd-agent-platform/core/src/lifecycle/risk-signals.ts +146 -146
  283. package/node_modules/@sdd-agent-platform/core/src/lifecycle/ship.test.ts +47 -0
  284. package/node_modules/@sdd-agent-platform/core/src/lifecycle/ship.ts +255 -263
  285. package/node_modules/@sdd-agent-platform/core/src/lifecycle-graph/contracts.ts +179 -0
  286. package/node_modules/@sdd-agent-platform/core/src/lifecycle-graph/kernel.ts +522 -0
  287. package/node_modules/@sdd-agent-platform/core/src/lifecycle-graph.ts +2 -0
  288. package/node_modules/@sdd-agent-platform/core/src/lifecycle.ts +4 -4
  289. package/node_modules/@sdd-agent-platform/core/src/orchestration/contracts.ts +50 -50
  290. package/node_modules/@sdd-agent-platform/core/src/orchestration/index.ts +2 -2
  291. package/node_modules/@sdd-agent-platform/core/src/orchestration/runtime.ts +331 -342
  292. package/node_modules/@sdd-agent-platform/core/src/path-safety.test.ts +22 -22
  293. package/node_modules/@sdd-agent-platform/core/src/phase8-contracts.test.ts +243 -243
  294. package/node_modules/@sdd-agent-platform/core/src/phase8-projection-compat.test.ts +152 -153
  295. package/node_modules/@sdd-agent-platform/core/src/phase8-risk-kernel.test.ts +277 -277
  296. package/node_modules/@sdd-agent-platform/core/src/phase9-lifecycle-graph.test.ts +103 -0
  297. package/node_modules/@sdd-agent-platform/core/src/planning/task-graph.test.ts +88 -88
  298. package/node_modules/@sdd-agent-platform/core/src/planning/task-graph.ts +222 -222
  299. package/node_modules/@sdd-agent-platform/core/src/planning/wave-plan.test.ts +79 -79
  300. package/node_modules/@sdd-agent-platform/core/src/planning/wave-plan.ts +160 -160
  301. package/node_modules/@sdd-agent-platform/core/src/planning.ts +2 -2
  302. package/node_modules/@sdd-agent-platform/core/src/registries/agent-capability-catalog.ts +426 -426
  303. package/node_modules/@sdd-agent-platform/core/src/registries/agent-registry.ts +230 -146
  304. package/node_modules/@sdd-agent-platform/core/src/registries/agent-runtime-static.ts +142 -142
  305. package/node_modules/@sdd-agent-platform/core/src/registries/capability-sources.ts +253 -253
  306. package/node_modules/@sdd-agent-platform/core/src/registries/command-team-runtime.ts +302 -309
  307. package/node_modules/@sdd-agent-platform/core/src/registries/eval-learning-context.ts +246 -246
  308. package/node_modules/@sdd-agent-platform/core/src/registries/plan-scout-domains.ts +89 -0
  309. package/node_modules/@sdd-agent-platform/core/src/registries/query-status.ts +119 -119
  310. package/node_modules/@sdd-agent-platform/core/src/registries/registries.test.ts +454 -429
  311. package/node_modules/@sdd-agent-platform/core/src/registries/skill-capabilities.ts +37 -37
  312. package/node_modules/@sdd-agent-platform/core/src/registries/tool-capabilities.ts +135 -135
  313. package/node_modules/@sdd-agent-platform/core/src/registries/tool-plugins.ts +132 -132
  314. package/node_modules/@sdd-agent-platform/core/src/registries/worker-adapters.ts +144 -144
  315. package/node_modules/@sdd-agent-platform/core/src/registries/workflow-gates.ts +111 -111
  316. package/node_modules/@sdd-agent-platform/core/src/registries.ts +42 -42
  317. package/node_modules/@sdd-agent-platform/core/src/risk/consumer-diagnostics.ts +98 -97
  318. package/node_modules/@sdd-agent-platform/core/src/risk/contracts.ts +63 -63
  319. package/node_modules/@sdd-agent-platform/core/src/risk/kernel.ts +233 -233
  320. package/node_modules/@sdd-agent-platform/core/src/risk/legacy-adapters.ts +251 -266
  321. package/node_modules/@sdd-agent-platform/core/src/risk/workflow-gates.ts +203 -203
  322. package/node_modules/@sdd-agent-platform/core/src/risk.ts +5 -5
  323. package/node_modules/@sdd-agent-platform/core/src/router/agent-runtime-config.ts +327 -327
  324. package/node_modules/@sdd-agent-platform/core/src/router/agent-runtime.ts +388 -388
  325. package/node_modules/@sdd-agent-platform/core/src/router/profile-resolution.ts +154 -154
  326. package/node_modules/@sdd-agent-platform/core/src/router/risk-policy.ts +33 -33
  327. package/node_modules/@sdd-agent-platform/core/src/router/route-cache.ts +100 -100
  328. package/node_modules/@sdd-agent-platform/core/src/router/route-projection.ts +356 -356
  329. package/node_modules/@sdd-agent-platform/core/src/router/route-sdd-task.test.ts +428 -428
  330. package/node_modules/@sdd-agent-platform/core/src/router/route-sdd-task.ts +2 -2
  331. package/node_modules/@sdd-agent-platform/core/src/router/routing-rules.ts +73 -73
  332. package/node_modules/@sdd-agent-platform/core/src/router/routing.ts +189 -191
  333. package/node_modules/@sdd-agent-platform/core/src/router/runtime-import.ts +464 -0
  334. package/node_modules/@sdd-agent-platform/core/src/router/runtime-inspection.ts +124 -124
  335. package/node_modules/@sdd-agent-platform/core/src/router/runtime-registry.ts +123 -123
  336. package/node_modules/@sdd-agent-platform/core/src/router/runtime-validation.ts +277 -277
  337. package/node_modules/@sdd-agent-platform/core/src/router/stage-route-binding.ts +273 -0
  338. package/node_modules/@sdd-agent-platform/core/src/router/team-mode.ts +170 -170
  339. package/node_modules/@sdd-agent-platform/core/src/router.ts +5 -4
  340. package/node_modules/@sdd-agent-platform/core/src/run-state/artifacts.ts +126 -118
  341. package/node_modules/@sdd-agent-platform/core/src/run-state/events.ts +27 -27
  342. package/node_modules/@sdd-agent-platform/core/src/run-state/inspect-run.ts +172 -172
  343. package/node_modules/@sdd-agent-platform/core/src/run-state/invocation-ledger.ts +109 -109
  344. package/node_modules/@sdd-agent-platform/core/src/run-state/model.ts +252 -230
  345. package/node_modules/@sdd-agent-platform/core/src/run-state/run-index.test.ts +52 -52
  346. package/node_modules/@sdd-agent-platform/core/src/run-state/run-index.ts +356 -356
  347. package/node_modules/@sdd-agent-platform/core/src/run-state/run-state.test.ts +70 -70
  348. package/node_modules/@sdd-agent-platform/core/src/run-state/run-state.ts +406 -406
  349. package/node_modules/@sdd-agent-platform/core/src/run-state/task-evidence.ts +198 -206
  350. package/node_modules/@sdd-agent-platform/core/src/run-state/timing.ts +146 -0
  351. package/node_modules/@sdd-agent-platform/core/src/run-state.ts +8 -8
  352. package/node_modules/@sdd-agent-platform/core/src/runtime-analysis/build.ts +60 -63
  353. package/node_modules/@sdd-agent-platform/core/src/runtime-analysis/findings.ts +257 -296
  354. package/node_modules/@sdd-agent-platform/core/src/runtime-analysis/model.ts +140 -152
  355. package/node_modules/@sdd-agent-platform/core/src/runtime-analysis.test.ts +66 -68
  356. package/node_modules/@sdd-agent-platform/core/src/runtime-analysis.ts +2 -2
  357. package/node_modules/@sdd-agent-platform/core/src/runtime-paths.ts +253 -176
  358. package/node_modules/@sdd-agent-platform/core/src/runtime-projection-p0.test.ts +101 -0
  359. package/node_modules/@sdd-agent-platform/core/src/runtime-projection-p0.ts +314 -0
  360. package/node_modules/@sdd-agent-platform/core/src/sdd-docs/artifact-depth.test.ts +380 -0
  361. package/node_modules/@sdd-agent-platform/core/src/sdd-docs/artifact-depth.ts +207 -0
  362. package/node_modules/@sdd-agent-platform/core/src/sdd-docs/context.ts +111 -111
  363. package/node_modules/@sdd-agent-platform/core/src/sdd-docs/document-hashes.ts +207 -207
  364. package/node_modules/@sdd-agent-platform/core/src/sdd-docs/run-binding.ts +95 -95
  365. package/node_modules/@sdd-agent-platform/core/src/sdd-docs/task-inspection.ts +39 -39
  366. package/node_modules/@sdd-agent-platform/core/src/sdd-docs/task-parser.test.ts +467 -401
  367. package/node_modules/@sdd-agent-platform/core/src/sdd-docs/task-parser.ts +738 -694
  368. package/node_modules/@sdd-agent-platform/core/src/sdd-docs/task-rendering.ts +81 -81
  369. package/node_modules/@sdd-agent-platform/core/src/sdd-docs.ts +5 -5
  370. package/node_modules/@sdd-agent-platform/core/src/spec-manager-contracts.ts +13 -0
  371. package/node_modules/@sdd-agent-platform/core/src/stage-artifacts.ts +435 -0
  372. package/node_modules/@sdd-agent-platform/core/src/stage-collaboration-contracts.ts +316 -0
  373. package/node_modules/@sdd-agent-platform/core/src/stage-collaboration.test.ts +2964 -0
  374. package/node_modules/@sdd-agent-platform/core/src/stage-collaboration.ts +5856 -0
  375. package/node_modules/@sdd-agent-platform/core/src/stage-runtime/contracts.ts +40 -40
  376. package/node_modules/@sdd-agent-platform/core/src/stage-runtime/runtime.test.ts +209 -209
  377. package/node_modules/@sdd-agent-platform/core/src/stage-runtime/runtime.ts +360 -352
  378. package/node_modules/@sdd-agent-platform/core/src/stage-runtime.ts +2 -2
  379. package/node_modules/@sdd-agent-platform/core/src/status/project-status.test.ts +288 -288
  380. package/node_modules/@sdd-agent-platform/core/src/status/project-status.ts +651 -625
  381. package/node_modules/@sdd-agent-platform/core/src/status.ts +2 -2
  382. package/node_modules/@sdd-agent-platform/core/src/storage/json-io.ts +10 -10
  383. package/node_modules/@sdd-agent-platform/core/src/storage/runtime-store.test.ts +489 -489
  384. package/node_modules/@sdd-agent-platform/core/src/storage/runtime-store.ts +1981 -1175
  385. package/node_modules/@sdd-agent-platform/core/src/subagents/contracts.ts +45 -45
  386. package/node_modules/@sdd-agent-platform/core/src/subagents/runtime.test.ts +232 -232
  387. package/node_modules/@sdd-agent-platform/core/src/subagents/runtime.ts +307 -307
  388. package/node_modules/@sdd-agent-platform/core/src/subagents.ts +2 -2
  389. package/node_modules/@sdd-agent-platform/core/src/task-execution-contract.test.ts +141 -0
  390. package/node_modules/@sdd-agent-platform/core/src/task-execution-contract.ts +566 -0
  391. package/node_modules/@sdd-agent-platform/core/src/task-risk-profile.ts +193 -193
  392. package/node_modules/@sdd-agent-platform/core/src/test-support/fixtures.ts +413 -398
  393. package/node_modules/@sdd-agent-platform/core/src/test-support/run-state.ts +102 -70
  394. package/node_modules/@sdd-agent-platform/core/src/test-support.ts +2 -2
  395. package/node_modules/@sdd-agent-platform/core/src/truth-reconciliation.test.ts +72 -0
  396. package/node_modules/@sdd-agent-platform/core/src/truth-reconciliation.ts +174 -0
  397. package/node_modules/@sdd-agent-platform/core/src/verification/rendering.ts +137 -181
  398. package/node_modules/@sdd-agent-platform/core/src/verification/review-gate.test.ts +77 -77
  399. package/node_modules/@sdd-agent-platform/core/src/verification/review-gate.ts +77 -77
  400. package/node_modules/@sdd-agent-platform/core/src/verification/single-task-loop.ts +455 -494
  401. package/node_modules/@sdd-agent-platform/core/src/verification/{goal-verify.test.ts → task-evidence-judgment.test.ts} +261 -335
  402. package/node_modules/@sdd-agent-platform/core/src/verification/{goal-verify.ts → task-evidence-judgment.ts} +619 -648
  403. package/node_modules/@sdd-agent-platform/core/src/verification/test-runtime.ts +1190 -1032
  404. package/node_modules/@sdd-agent-platform/core/src/verification/validation-cache.ts +106 -0
  405. package/node_modules/@sdd-agent-platform/core/src/verification/validation-wave.ts +513 -513
  406. package/node_modules/@sdd-agent-platform/core/src/verification/verify-contract.ts +334 -358
  407. package/node_modules/@sdd-agent-platform/core/src/verification.ts +8 -8
  408. package/node_modules/@sdd-agent-platform/core/src/work-units/contracts.ts +26 -26
  409. package/node_modules/@sdd-agent-platform/core/src/work-units/runtime.test.ts +88 -88
  410. package/node_modules/@sdd-agent-platform/core/src/work-units/runtime.ts +112 -112
  411. package/node_modules/@sdd-agent-platform/core/src/work-units.ts +2 -2
  412. package/node_modules/@sdd-agent-platform/core/src/workflow-gate/evidence-packet.ts +190 -196
  413. package/node_modules/@sdd-agent-platform/core/src/workflow-gate/hard-checks.test.ts +169 -171
  414. package/node_modules/@sdd-agent-platform/core/src/workflow-gate/hard-checks.ts +136 -143
  415. package/node_modules/@sdd-agent-platform/core/src/workflow-gate/policy.test.ts +135 -137
  416. package/node_modules/@sdd-agent-platform/core/src/workflow-gate/policy.ts +153 -155
  417. package/node_modules/@sdd-agent-platform/core/src/workflow-gate/types.ts +111 -114
  418. package/node_modules/@sdd-agent-platform/core/src/workflow-state/affected-file-conflicts.ts +95 -95
  419. package/node_modules/@sdd-agent-platform/core/src/workflow-state/dependencies.test.ts +32 -32
  420. package/node_modules/@sdd-agent-platform/core/src/workflow-state/dependencies.ts +114 -114
  421. package/node_modules/@sdd-agent-platform/core/src/workflow-state/latest-eligible-run.ts +184 -156
  422. package/node_modules/@sdd-agent-platform/core/src/workflow-state/migration-recovery.ts +158 -0
  423. package/node_modules/@sdd-agent-platform/core/src/workflow-state/repair-contract.ts +77 -0
  424. package/node_modules/@sdd-agent-platform/core/src/workflow-state/resolve-task-run.ts +114 -0
  425. package/node_modules/@sdd-agent-platform/core/src/workflow-state/resolve.test.ts +970 -464
  426. package/node_modules/@sdd-agent-platform/core/src/workflow-state/resolve.ts +967 -363
  427. package/node_modules/@sdd-agent-platform/core/src/workflow-state/runtime-projections.ts +712 -0
  428. package/node_modules/@sdd-agent-platform/core/src/workflow-state.ts +2 -2
  429. package/node_modules/@sdd-agent-platform/core/src/worktree/isolation.ts +130 -130
  430. package/node_modules/@sdd-agent-platform/core/src/worktree/lifecycle.ts +269 -269
  431. package/node_modules/@sdd-agent-platform/core/src/worktree/worktree.test.ts +150 -150
  432. package/node_modules/@sdd-agent-platform/core/src/worktree.ts +2 -2
  433. package/node_modules/@sdd-agent-platform/core/tsconfig.json +15 -15
  434. package/package.json +2 -2
  435. package/packages/cli/dist/args.js +1 -1
  436. package/packages/cli/dist/args.js.map +1 -1
  437. package/packages/cli/dist/commands/context.js +1 -1
  438. package/packages/cli/dist/commands/context.js.map +1 -1
  439. package/packages/cli/dist/commands/evidence.js.map +1 -0
  440. package/packages/cli/dist/commands/execution.js +126 -0
  441. package/packages/cli/dist/commands/execution.js.map +1 -1
  442. package/packages/cli/dist/commands/instructions.d.ts +1 -1
  443. package/packages/cli/dist/commands/instructions.js +15 -1
  444. package/packages/cli/dist/commands/instructions.js.map +1 -1
  445. package/packages/cli/dist/commands/registry/runtime.js +70 -1
  446. package/packages/cli/dist/commands/registry/runtime.js.map +1 -1
  447. package/packages/cli/dist/commands/run.js +12 -1
  448. package/packages/cli/dist/commands/run.js.map +1 -1
  449. package/packages/cli/dist/commands/stage-close.d.ts +66 -0
  450. package/packages/cli/dist/commands/stage-close.js +524 -0
  451. package/packages/cli/dist/commands/stage-close.js.map +1 -0
  452. package/packages/cli/dist/commands/status.js +8 -1
  453. package/packages/cli/dist/commands/status.js.map +1 -1
  454. package/packages/cli/dist/commands/tasks.js.map +1 -1
  455. package/packages/cli/dist/dispatch.js +6 -31
  456. package/packages/cli/dist/dispatch.js.map +1 -1
  457. package/packages/cli/dist/help.js +153 -158
  458. package/packages/cli/dist/help.js.map +1 -1
  459. package/packages/cli/dist/renderers/workflow.d.ts +51 -2
  460. package/packages/cli/dist/renderers/workflow.js.map +1 -1
  461. package/packages/cli/dist/skill-import-args.d.ts +10 -0
  462. package/packages/cli/dist/skill-import-args.js +47 -0
  463. package/packages/cli/dist/skill-import-args.js.map +1 -0
  464. package/packages/cli/dist/tsconfig.tsbuildinfo +1 -1
  465. package/packages/cli/package.json +2 -2
  466. package/packages/core/dist/ai-tools.js +84 -103
  467. package/packages/core/dist/ai-tools.js.map +1 -1
  468. package/packages/core/dist/config/init-project.d.ts +10 -6
  469. package/packages/core/dist/config/init-project.js +7 -8
  470. package/packages/core/dist/config/init-project.js.map +1 -1
  471. package/packages/core/dist/config/project-config.d.ts +3 -1
  472. package/packages/core/dist/config/project-config.js +7 -3
  473. package/packages/core/dist/config/project-config.js.map +1 -1
  474. package/packages/core/dist/config/starter-documents.d.ts +0 -1
  475. package/packages/core/dist/config/starter-documents.js +374 -421
  476. package/packages/core/dist/config/starter-documents.js.map +1 -1
  477. package/packages/core/dist/context/build-package.d.ts +1 -1
  478. package/packages/core/dist/context/build-package.js +7 -19
  479. package/packages/core/dist/context/build-package.js.map +1 -1
  480. package/packages/core/dist/contracts.d.ts +7 -1
  481. package/packages/core/dist/contracts.js +6 -0
  482. package/packages/core/dist/contracts.js.map +1 -1
  483. package/packages/core/dist/doctor/checks/document-chain.js +2 -12
  484. package/packages/core/dist/doctor/checks/document-chain.js.map +1 -1
  485. package/packages/core/dist/doctor/doctor.js +1 -18
  486. package/packages/core/dist/doctor/doctor.js.map +1 -1
  487. package/packages/core/dist/evidence/lookup.d.ts +1 -1
  488. package/packages/core/dist/evidence/lookup.js +1 -1
  489. package/packages/core/dist/evidence/lookup.js.map +1 -1
  490. package/packages/core/dist/evidence-runtime/contracts.d.ts +0 -1
  491. package/packages/core/dist/evidence-runtime/coordination.js +110 -0
  492. package/packages/core/dist/evidence-runtime/coordination.js.map +1 -0
  493. package/packages/core/dist/execution/host-invocation.js +83 -83
  494. package/packages/core/dist/instructions.d.ts +1 -1
  495. package/packages/core/dist/instructions.js +37 -80
  496. package/packages/core/dist/instructions.js.map +1 -1
  497. package/packages/core/dist/lifecycle/ship.js +58 -68
  498. package/packages/core/dist/lifecycle/ship.js.map +1 -1
  499. package/packages/core/dist/lifecycle-graph/contracts.d.ts +159 -0
  500. package/packages/core/dist/lifecycle-graph/contracts.js +7 -0
  501. package/packages/core/dist/lifecycle-graph/contracts.js.map +1 -0
  502. package/packages/core/dist/lifecycle-graph/kernel.d.ts +16 -0
  503. package/packages/core/dist/lifecycle-graph/kernel.js +461 -0
  504. package/packages/core/dist/lifecycle-graph/kernel.js.map +1 -0
  505. package/packages/core/dist/lifecycle-graph.d.ts +2 -0
  506. package/packages/core/dist/lifecycle-graph.js +3 -0
  507. package/packages/core/dist/lifecycle-graph.js.map +1 -0
  508. package/packages/core/dist/orchestration/contracts.d.ts +1 -1
  509. package/packages/core/dist/orchestration/runtime.js +21 -28
  510. package/packages/core/dist/orchestration/runtime.js.map +1 -1
  511. package/packages/core/dist/registries/agent-registry.js +124 -40
  512. package/packages/core/dist/registries/agent-registry.js.map +1 -1
  513. package/packages/core/dist/registries/command-team-runtime.d.ts +1 -1
  514. package/packages/core/dist/registries/command-team-runtime.js +6 -13
  515. package/packages/core/dist/registries/command-team-runtime.js.map +1 -1
  516. package/packages/core/dist/registries/plan-scout-domains.d.ts +13 -0
  517. package/packages/core/dist/registries/plan-scout-domains.js +76 -0
  518. package/packages/core/dist/registries/plan-scout-domains.js.map +1 -0
  519. package/packages/core/dist/registries/skill-capabilities.js +7 -7
  520. package/packages/core/dist/registries/skill-capabilities.js.map +1 -1
  521. package/packages/core/dist/registries/tool-capabilities.js +6 -6
  522. package/packages/core/dist/registries/tool-capabilities.js.map +1 -1
  523. package/packages/core/dist/registries/workflow-gates.d.ts +1 -1
  524. package/packages/core/dist/registries/workflow-gates.js +18 -18
  525. package/packages/core/dist/registries/workflow-gates.js.map +1 -1
  526. package/packages/core/dist/risk/consumer-diagnostics.js +2 -1
  527. package/packages/core/dist/risk/consumer-diagnostics.js.map +1 -1
  528. package/packages/core/dist/risk/contracts.d.ts +2 -2
  529. package/packages/core/dist/risk/kernel.js +7 -7
  530. package/packages/core/dist/risk/kernel.js.map +1 -1
  531. package/packages/core/dist/risk/legacy-adapters.js +12 -27
  532. package/packages/core/dist/risk/legacy-adapters.js.map +1 -1
  533. package/packages/core/dist/risk/workflow-gates.js +6 -6
  534. package/packages/core/dist/risk/workflow-gates.js.map +1 -1
  535. package/packages/core/dist/router/agent-runtime-config.js +1 -1
  536. package/packages/core/dist/router/agent-runtime-config.js.map +1 -1
  537. package/packages/core/dist/router/routing.js +2 -4
  538. package/packages/core/dist/router/routing.js.map +1 -1
  539. package/packages/core/dist/router/runtime-import.d.ts +28 -0
  540. package/packages/core/dist/router/runtime-import.js +383 -0
  541. package/packages/core/dist/router/runtime-import.js.map +1 -0
  542. package/packages/core/dist/router/stage-route-binding.d.ts +37 -0
  543. package/packages/core/dist/router/stage-route-binding.js +227 -0
  544. package/packages/core/dist/router/stage-route-binding.js.map +1 -0
  545. package/packages/core/dist/router.d.ts +1 -0
  546. package/packages/core/dist/router.js +1 -0
  547. package/packages/core/dist/router.js.map +1 -1
  548. package/packages/core/dist/run-state/artifacts.d.ts +16 -0
  549. package/packages/core/dist/run-state/artifacts.js +6 -0
  550. package/packages/core/dist/run-state/artifacts.js.map +1 -1
  551. package/packages/core/dist/run-state/model.d.ts +20 -0
  552. package/packages/core/dist/run-state/run-state.js +7 -7
  553. package/packages/core/dist/run-state/run-state.js.map +1 -1
  554. package/packages/core/dist/run-state/task-evidence.d.ts +1 -2
  555. package/packages/core/dist/run-state/task-evidence.js +2 -9
  556. package/packages/core/dist/run-state/task-evidence.js.map +1 -1
  557. package/packages/core/dist/run-state/timing.d.ts +8 -0
  558. package/packages/core/dist/run-state/timing.js +131 -0
  559. package/packages/core/dist/run-state/timing.js.map +1 -0
  560. package/packages/core/dist/runtime-analysis/build.js +1 -4
  561. package/packages/core/dist/runtime-analysis/build.js.map +1 -1
  562. package/packages/core/dist/runtime-analysis/findings.js +0 -39
  563. package/packages/core/dist/runtime-analysis/findings.js.map +1 -1
  564. package/packages/core/dist/runtime-analysis/model.d.ts +1 -17
  565. package/packages/core/dist/runtime-paths.d.ts +10 -0
  566. package/packages/core/dist/runtime-paths.js +65 -0
  567. package/packages/core/dist/runtime-paths.js.map +1 -1
  568. package/packages/core/dist/runtime-projection-p0.d.ts +64 -0
  569. package/packages/core/dist/runtime-projection-p0.js +211 -0
  570. package/packages/core/dist/runtime-projection-p0.js.map +1 -0
  571. package/packages/core/dist/sdd-docs/artifact-depth.d.ts +14 -0
  572. package/packages/core/dist/sdd-docs/artifact-depth.js +179 -0
  573. package/packages/core/dist/sdd-docs/artifact-depth.js.map +1 -0
  574. package/packages/core/dist/sdd-docs/task-parser.d.ts +5 -1
  575. package/packages/core/dist/sdd-docs/task-parser.js +60 -22
  576. package/packages/core/dist/sdd-docs/task-parser.js.map +1 -1
  577. package/packages/core/dist/sdd-docs/task-rendering.js +2 -2
  578. package/packages/core/dist/sdd-docs/task-rendering.js.map +1 -1
  579. package/packages/core/dist/spec-entry.js +40 -0
  580. package/packages/core/dist/spec-entry.js.map +1 -0
  581. package/packages/core/dist/spec-manager-contracts.d.ts +12 -0
  582. package/packages/core/dist/spec-manager-contracts.js +2 -0
  583. package/packages/core/dist/spec-manager-contracts.js.map +1 -0
  584. package/packages/core/dist/stage-artifacts.d.ts +55 -0
  585. package/packages/core/dist/stage-artifacts.js +315 -0
  586. package/packages/core/dist/stage-artifacts.js.map +1 -0
  587. package/packages/core/dist/stage-collaboration-contracts.d.ts +55 -0
  588. package/packages/core/dist/stage-collaboration-contracts.js +238 -0
  589. package/packages/core/dist/stage-collaboration-contracts.js.map +1 -0
  590. package/packages/core/dist/stage-collaboration.d.ts +736 -0
  591. package/packages/core/dist/stage-collaboration.js +4018 -0
  592. package/packages/core/dist/stage-collaboration.js.map +1 -0
  593. package/packages/core/dist/stage-runtime/runtime.js +8 -1
  594. package/packages/core/dist/stage-runtime/runtime.js.map +1 -1
  595. package/packages/core/dist/status/project-status.js +25 -1
  596. package/packages/core/dist/status/project-status.js.map +1 -1
  597. package/packages/core/dist/storage/runtime-store.d.ts +170 -18
  598. package/packages/core/dist/storage/runtime-store.js +597 -85
  599. package/packages/core/dist/storage/runtime-store.js.map +1 -1
  600. package/packages/core/dist/sync-back/apply.d.ts +1 -17
  601. package/packages/core/dist/sync-back/apply.js +1 -242
  602. package/packages/core/dist/sync-back/apply.js.map +1 -1
  603. package/packages/core/dist/sync-back/inspect.d.ts +1 -110
  604. package/packages/core/dist/sync-back/inspect.js +1 -496
  605. package/packages/core/dist/sync-back/inspect.js.map +1 -1
  606. package/packages/core/dist/sync-back.d.ts +1 -2
  607. package/packages/core/dist/sync-back.js +1 -2
  608. package/packages/core/dist/sync-back.js.map +1 -1
  609. package/packages/core/dist/task-execution-contract.d.ts +167 -0
  610. package/packages/core/dist/task-execution-contract.js +377 -0
  611. package/packages/core/dist/task-execution-contract.js.map +1 -0
  612. package/packages/core/dist/test-support/fixtures.js +329 -314
  613. package/packages/core/dist/test-support/fixtures.js.map +1 -1
  614. package/packages/core/dist/test-support/run-state.d.ts +1 -0
  615. package/packages/core/dist/test-support/run-state.js +31 -0
  616. package/packages/core/dist/test-support/run-state.js.map +1 -1
  617. package/packages/core/dist/truth-reconciliation.d.ts +44 -0
  618. package/packages/core/dist/truth-reconciliation.js +135 -0
  619. package/packages/core/dist/truth-reconciliation.js.map +1 -0
  620. package/packages/core/dist/tsconfig.tsbuildinfo +1 -1
  621. package/packages/core/dist/verification/goal-verify.d.ts +0 -49
  622. package/packages/core/dist/verification/goal-verify.js +1 -545
  623. package/packages/core/dist/verification/goal-verify.js.map +1 -1
  624. package/packages/core/dist/verification/rendering.d.ts +5 -7
  625. package/packages/core/dist/verification/rendering.js +15 -55
  626. package/packages/core/dist/verification/rendering.js.map +1 -1
  627. package/packages/core/dist/verification/single-task-loop.js +1 -40
  628. package/packages/core/dist/verification/single-task-loop.js.map +1 -1
  629. package/packages/core/dist/verification/task-evidence-judgment.d.ts +49 -0
  630. package/packages/core/dist/verification/task-evidence-judgment.js +521 -0
  631. package/packages/core/dist/verification/task-evidence-judgment.js.map +1 -0
  632. package/packages/core/dist/verification/test-runtime.d.ts +12 -2
  633. package/packages/core/dist/verification/test-runtime.js +247 -112
  634. package/packages/core/dist/verification/test-runtime.js.map +1 -1
  635. package/packages/core/dist/verification/validation-cache.d.ts +26 -0
  636. package/packages/core/dist/verification/validation-cache.js +73 -0
  637. package/packages/core/dist/verification/validation-cache.js.map +1 -0
  638. package/packages/core/dist/verification/verify-contract.d.ts +1 -1
  639. package/packages/core/dist/verification/verify-contract.js +49 -72
  640. package/packages/core/dist/verification/verify-contract.js.map +1 -1
  641. package/packages/core/dist/verification.d.ts +3 -3
  642. package/packages/core/dist/verification.js +2 -2
  643. package/packages/core/dist/verification.js.map +1 -1
  644. package/packages/core/dist/workflow-gate/evidence-packet.js +2 -7
  645. package/packages/core/dist/workflow-gate/evidence-packet.js.map +1 -1
  646. package/packages/core/dist/workflow-gate/hard-checks.js +0 -7
  647. package/packages/core/dist/workflow-gate/hard-checks.js.map +1 -1
  648. package/packages/core/dist/workflow-gate/policy.js +2 -4
  649. package/packages/core/dist/workflow-gate/policy.js.map +1 -1
  650. package/packages/core/dist/workflow-gate/types.d.ts +3 -5
  651. package/packages/core/dist/workflow-state/latest-eligible-run.js +30 -4
  652. package/packages/core/dist/workflow-state/latest-eligible-run.js.map +1 -1
  653. package/packages/core/dist/workflow-state/migration-recovery.d.ts +40 -0
  654. package/packages/core/dist/workflow-state/migration-recovery.js +110 -0
  655. package/packages/core/dist/workflow-state/migration-recovery.js.map +1 -0
  656. package/packages/core/dist/workflow-state/repair-contract.d.ts +12 -0
  657. package/packages/core/dist/workflow-state/repair-contract.js +63 -0
  658. package/packages/core/dist/workflow-state/repair-contract.js.map +1 -0
  659. package/packages/core/dist/workflow-state/resolve-task-run.d.ts +21 -0
  660. package/packages/core/dist/workflow-state/resolve-task-run.js +95 -0
  661. package/packages/core/dist/workflow-state/resolve-task-run.js.map +1 -0
  662. package/packages/core/dist/workflow-state/resolve.d.ts +55 -5
  663. package/packages/core/dist/workflow-state/resolve.js +518 -36
  664. package/packages/core/dist/workflow-state/resolve.js.map +1 -1
  665. package/packages/core/dist/workflow-state/runtime-projections.d.ts +228 -0
  666. package/packages/core/dist/workflow-state/runtime-projections.js +452 -0
  667. package/packages/core/dist/workflow-state/runtime-projections.js.map +1 -0
  668. package/packages/core/package.json +6 -3
  669. package/tsconfig.build.json +6 -7
  670. package/node_modules/@sdd-agent-platform/core/dist/doctor/render.d.ts +0 -2
  671. package/node_modules/@sdd-agent-platform/core/dist/doctor/render.js +0 -44
  672. package/node_modules/@sdd-agent-platform/core/dist/doctor/render.js.map +0 -1
  673. package/node_modules/@sdd-agent-platform/core/src/sync-back/apply.ts +0 -270
  674. package/node_modules/@sdd-agent-platform/core/src/sync-back/inspect.ts +0 -655
  675. package/node_modules/@sdd-agent-platform/core/src/sync-back/sync-back.test.ts +0 -569
  676. package/node_modules/@sdd-agent-platform/core/src/sync-back.ts +0 -2
  677. package/node_modules/@sdd-agent-platform/core/src/verification/single-task-loop.test.ts +0 -255
  678. package/node_modules/@sdd-agent-platform/core/src/verification/test-runtime.test.ts +0 -439
  679. package/node_modules/@sdd-agent-platform/core/src/verification/validation-wave.test.ts +0 -341
  680. package/node_modules/@sdd-agent-platform/core/src/verification/verify-contract.test.ts +0 -204
  681. package/packages/cli/dist/commands/lifecycle.d.ts +0 -6
  682. package/packages/cli/dist/commands/lifecycle.js +0 -112
  683. package/packages/cli/dist/commands/lifecycle.js.map +0 -1
  684. package/packages/cli/dist/commands/sync-back.d.ts +0 -6
  685. package/packages/cli/dist/commands/sync-back.js +0 -82
  686. package/packages/cli/dist/commands/sync-back.js.map +0 -1
  687. package/packages/cli/dist/commands/test.d.ts +0 -6
  688. package/packages/cli/dist/commands/test.js +0 -195
  689. package/packages/cli/dist/commands/test.js.map +0 -1
  690. package/packages/cli/dist/commands/verifies.d.ts +0 -6
  691. package/packages/cli/dist/commands/verifies.js +0 -85
  692. package/packages/cli/dist/commands/verifies.js.map +0 -1
  693. package/packages/cli/dist/commands/verify.d.ts +0 -6
  694. package/packages/cli/dist/commands/verify.js +0 -134
  695. package/packages/cli/dist/commands/verify.js.map +0 -1
  696. package/packages/core/dist/doctor/render.d.ts +0 -2
  697. package/packages/core/dist/doctor/render.js +0 -44
  698. package/packages/core/dist/doctor/render.js.map +0 -1
@@ -1,625 +1,651 @@
1
- import { execFile } from 'node:child_process';
2
- import { stat } from 'node:fs/promises';
3
- import { promisify } from 'node:util';
4
- import type { ContextBranchSource, ContextResolverContract } from '../sdd-docs/context.js';
5
- import type { SddTaskGap, SddTaskModel } from '../sdd-docs/task-parser.js';
6
- import type { RunSummary } from '../run-state/model.js';
7
- import { readRunEvents } from '../run-state/events.js';
8
- import { readRunState } from '../run-state/run-state.js';
9
- import { listAgentExecutionRecords, listTeamSessionRecords } from '../execution/agent-execution-records.js';
10
- import { listResidentWorkerRuntimes } from '../execution/resident-worker.js';
11
- import { runDocumentStaleReasons } from '../sync-back/inspect.js';
12
- import { resolveWorkflowState, type WorkflowLatestTaskRun, type WorkflowNextIntent } from '../workflow-state/resolve.js';
13
- import type { WorkflowAffectedFileConflict } from '../workflow-state/affected-file-conflicts.js';
14
- import type { WorkflowDependencyBlocker } from '../workflow-state/dependencies.js';
15
- import type { LatestEligibleRunSelection } from '../workflow-state/latest-eligible-run.js';
16
- import { buildTaskRiskProfile } from '../task-risk-profile.js';
17
- import { listRuntimeProjections } from '../storage/runtime-store.js';
18
- import { inspectLifecycleRiskDecisionForModel, type LifecycleRiskConsumerDiagnostic } from '../risk.js';
19
- import { inspectWorkflowStageHandoff, type WorkflowStageHandoffDiagnostic } from '../stage-runtime.js';
20
- import { inspectContextOffloadRuntime, type ContextRuntimeDiagnostic } from '../context-offload.js';
21
- import { inspectSubagentDispatches, type SubagentDispatchDiagnostic } from '../subagents.js';
22
- import { inspectAgentCapabilityCatalog, REQUIRED_PROFESSIONAL_CAPABILITY_DOMAINS, type AgentCapabilityDomain, type ProfessionalCapabilityDomain } from '../registries/agent-capability-catalog.js';
23
- import { validateAgentSkillTeamRuntime } from '../router/runtime-validation.js';
24
- import { listRuntimeDurableGaps, listRuntimeFanInEvidenceSets, listRuntimeLlmAdvisorAssessments, listRuntimeStageRoleResults, listRuntimeStageTeamSessions, listRuntimeTestRuns, listRuntimeWorkflowGateDecisions, type RuntimeDurableGapRecord } from '../storage/runtime-store.js';
25
- import type { CapabilityEvidenceClassification } from '../evidence-runtime.js';
26
- import type { WorkflowGateDecision, WorkflowGateDecisionKind } from '../workflow-gate/types.js';
27
-
28
- const execFileAsync = promisify(execFile);
29
-
30
- export interface RunEvidenceSummary {
31
- agentExecutions: number;
32
- teamSessions: number;
33
- artifactIngestions: number;
34
- workerRuntimes: number;
35
- staleWorkerRuntimes: number;
36
- routePreflight: boolean;
37
- tasksChangedAfterRun: boolean;
38
- tasksUpdatedAt: string | null;
39
- runUpdatedAt: string | null;
40
- }
41
-
42
- export interface TaskRiskSummary {
43
- highRiskTasks: string[];
44
- mediumRiskTasks: string[];
45
- sourceBoundaryTasks: string[];
46
- contextRiskTasks: string[];
47
- tokenRiskTasks: string[];
48
- performanceRiskTasks: string[];
49
- }
50
-
51
- export interface TokenRuntimeProjection {
52
- health: 'unknown' | 'nominal' | 'pressure';
53
- estimatedTokens: number | null;
54
- contextPackages: number;
55
- teamRuntimeDecisions: number;
56
- pressureReasons: string[];
57
- }
58
-
59
- export type CapabilityHealthStatus = 'absent' | 'pass' | 'warn' | 'blocked';
60
-
61
- export interface CapabilityHealthProjection {
62
- status: CapabilityHealthStatus;
63
- requiredProfessionalDomains: ProfessionalCapabilityDomain[];
64
- baselineDomains: ProfessionalCapabilityDomain[];
65
- missingBaselineDomains: ProfessionalCapabilityDomain[];
66
- materialPacks: number;
67
- activeDomains: AgentCapabilityDomain[];
68
- activePacks: string[];
69
- sources: {
70
- total: number;
71
- quarantined: number;
72
- denied: number;
73
- futureAdapters: number;
74
- };
75
- evidence: {
76
- accepted: number;
77
- candidate: number;
78
- quarantined: number;
79
- diagnostic: number;
80
- blocked: number;
81
- };
82
- warnings: string[];
83
- releaseCriticalGaps: ProfessionalCapabilityDomain[];
84
- reasons: string[];
85
- }
86
-
87
- export interface StageTeamHealthProjection {
88
- sessions: number;
89
- roleResults: number;
90
- advisorAssessments: number;
91
- fanInEvidenceSets: number;
92
- highConcernAdvisors: number;
93
- lifecycleProfiles: string[];
94
- directProfileRequiresNoTeam: boolean;
95
- }
96
-
97
- export interface WorkflowGateHealthProjection {
98
- decisions: number;
99
- blocked: number;
100
- warnings: number;
101
- humanRequired: number;
102
- latestDecision: WorkflowGateDecision | null;
103
- statusesByKind: Partial<Record<WorkflowGateDecisionKind, WorkflowGateDecision['status']>>;
104
- }
105
-
106
- export interface ProjectStatus {
107
- branch: string;
108
- workflowStatus: 'active' | 'not_started';
109
- context: ContextResolverContract;
110
- gitRoot: string | null;
111
- documents: SddTaskModel['documents'];
112
- tasks: {
113
- total: number;
114
- pending: number;
115
- inProgress: number;
116
- completed: number;
117
- blocked: number;
118
- deferred: number;
119
- unknown: number;
120
- gaps: number;
121
- };
122
- taskRisk: TaskRiskSummary;
123
- latestRun: RunSummary | null;
124
- latestRunsByTask: WorkflowLatestTaskRun[];
125
- latestEligibleRunsByTask: LatestEligibleRunSelection[];
126
- latestRunEvidence: RunEvidenceSummary | null;
127
- latestRunStaleReasons: string[];
128
- affectedFileConflicts: WorkflowAffectedFileConflict[];
129
- dependencyBlockers: WorkflowDependencyBlocker[];
130
- recommendedNextCommand: string;
131
- nextIntent: WorkflowNextIntent;
132
- tokenProjection: TokenRuntimeProjection;
133
- contextRuntime: ContextRuntimeDiagnostic;
134
- subagentDispatches: SubagentDispatchDiagnostic;
135
- lifecycleRisk: LifecycleRiskConsumerDiagnostic;
136
- capabilityHealth: CapabilityHealthProjection;
137
- workflowHandoff: WorkflowStageHandoffDiagnostic;
138
- gaps: SddTaskGap[];
139
- durableGaps: RuntimeDurableGapRecord[];
140
- stageTeamHealth: StageTeamHealthProjection;
141
- workflowGateHealth: WorkflowGateHealthProjection;
142
- }
143
-
144
- export interface StatuslineProjection {
145
- contract: 'sdd-statusline-projection-v1';
146
- branch: string;
147
- workflow: ProjectStatus['workflowStatus'];
148
- taskHealth: 'empty' | 'active' | 'blocked' | 'complete';
149
- runtimeHealth: 'none' | 'pass' | 'warn' | 'blocked';
150
- testHealth: 'none' | 'pass' | 'warn' | 'blocked';
151
- teamHealth: 'none' | 'active';
152
- tokenHealth: 'unknown' | 'nominal' | 'pressure';
153
- contextLoad: ProjectStatus['contextRuntime']['level'];
154
- contextAction: ProjectStatus['contextRuntime']['action'];
155
- subagentHealth: ProjectStatus['subagentDispatches']['status'];
156
- evidenceHealth: 'none' | 'pass' | 'warn' | 'blocked';
157
- capabilityHealth: ProjectStatus['capabilityHealth']['status'];
158
- latestRunId: string | null;
159
- counts: {
160
- tasks: ProjectStatus['tasks'];
161
- agentExecutions: number;
162
- teamSessions: number;
163
- workerRuntimes: number;
164
- staleWorkerRuntimes: number;
165
- artifactIngestions: number;
166
- staleReasons: number;
167
- affectedFileConflicts: number;
168
- subagentDispatches: number;
169
- blockingSubagents: number;
170
- capabilityWarnings: number;
171
- durableGaps: number;
172
- stageTeamSessions: number;
173
- stageRoleResults: number;
174
- advisorAssessments: number;
175
- fanInEvidenceSets: number;
176
- workflowGateDecisions: number;
177
- };
178
- taskRisk: TaskRiskSummary;
179
- next: string;
180
- }
181
-
182
- export async function getProjectStatus(projectRoot: string, options: { branch?: string | null; branchSource?: ContextBranchSource } = {}): Promise<ProjectStatus> {
183
- const [workflow, gitRoot] = await Promise.all([resolveWorkflowState(projectRoot, options), getGitRoot(projectRoot)]);
184
- const latestRun = workflow.latestRun;
185
- const latestRunState = workflow.latestRunState;
186
- const latestRunEvidence = latestRun ? await inspectRunEvidenceSummary(projectRoot, latestRun.runId) : null;
187
- const enrichedLatestRunEvidence = latestRun && latestRunEvidence
188
- ? await addTaskDocumentStalenessToRunEvidence(workflow.model.tasksPath, latestRun, latestRunEvidence)
189
- : latestRunEvidence;
190
- const latestRunStaleReasons = latestRunState ? await runDocumentStaleReasons(projectRoot, latestRunState, workflow.model) : [];
191
- const taskRisk = summarizeTaskRisk(workflow.model.tasks);
192
- const tokenProjection = await inspectTokenRuntimeProjection(projectRoot);
193
- const lifecycleRisk = await inspectLifecycleRiskDecisionForModel(projectRoot, workflow.branch, workflow.model);
194
- const workflowHandoff = await inspectWorkflowStageHandoff(projectRoot, workflow.branch);
195
- const contextRuntime = await inspectContextOffloadRuntime(projectRoot, workflow.branch);
196
- const subagentDispatches = await inspectSubagentDispatches(projectRoot, workflow.branch);
197
- const capabilityHealth = await inspectCapabilityHealth(projectRoot, latestRun?.runId ?? null, workflow.model.tasks);
198
- const durableGaps = await listRuntimeDurableGaps(projectRoot, { partition: workflow.branch, status: 'open_terminal' });
199
- const stageTeamHealth = await inspectStageTeamHealth(projectRoot, workflow.branch, latestRun?.runId ?? null);
200
- const workflowGateHealth = await inspectWorkflowGateHealth(projectRoot, workflow.branch, latestRun?.runId ?? null);
201
-
202
- return {
203
- branch: workflow.branch,
204
- workflowStatus: workflow.workflowStatus,
205
- context: workflow.context,
206
- gitRoot,
207
- documents: workflow.documents,
208
- tasks: workflow.taskCounts,
209
- taskRisk,
210
- latestRun,
211
- latestRunsByTask: workflow.latestRunsByTask,
212
- latestEligibleRunsByTask: workflow.latestEligibleRunsByTask,
213
- latestRunEvidence: enrichedLatestRunEvidence,
214
- latestRunStaleReasons,
215
- affectedFileConflicts: workflow.affectedFileConflicts,
216
- dependencyBlockers: workflow.dependencyBlockers,
217
- recommendedNextCommand: workflow.recommendedNextCommand,
218
- nextIntent: workflow.nextIntent,
219
- tokenProjection,
220
- contextRuntime,
221
- subagentDispatches,
222
- lifecycleRisk,
223
- workflowHandoff,
224
- capabilityHealth,
225
- gaps: workflow.visibleGaps,
226
- durableGaps,
227
- stageTeamHealth,
228
- workflowGateHealth
229
- };
230
- }
231
-
232
- export async function getStatuslineProjection(projectRoot: string, options: { branch?: string | null; branchSource?: ContextBranchSource } = {}): Promise<StatuslineProjection> {
233
- return statuslineProjectionFromStatus(await getProjectStatus(projectRoot, options));
234
- }
235
-
236
- export function statuslineProjectionFromStatus(status: ProjectStatus): StatuslineProjection {
237
- const evidence = status.latestRunEvidence;
238
- const runtimeHealth = status.latestRun
239
- ? status.latestRun.status === 'completed' && status.latestRun.validationStatus === 'pass' ? 'pass' : 'blocked'
240
- : 'none';
241
- const staleReasons = status.latestRunStaleReasons.length;
242
- const affectedFileConflicts = status.affectedFileConflicts.length;
243
- return {
244
- contract: 'sdd-statusline-projection-v1',
245
- branch: status.branch,
246
- workflow: status.workflowStatus,
247
- taskHealth: taskHealth(status),
248
- runtimeHealth,
249
- testHealth: testHealth(status),
250
- teamHealth: (evidence?.teamSessions ?? 0) > 0 || status.stageTeamHealth.sessions > 0 || status.stageTeamHealth.roleResults > 0 ? 'active' : 'none',
251
- tokenHealth: status.tokenProjection.health,
252
- contextLoad: status.contextRuntime.level,
253
- contextAction: status.contextRuntime.action,
254
- subagentHealth: status.subagentDispatches.status,
255
- evidenceHealth: evidenceHealth(status, runtimeHealth),
256
- capabilityHealth: status.capabilityHealth.status,
257
- latestRunId: status.latestRun?.runId ?? null,
258
- counts: {
259
- tasks: status.tasks,
260
- agentExecutions: evidence?.agentExecutions ?? 0,
261
- teamSessions: evidence?.teamSessions ?? 0,
262
- workerRuntimes: evidence?.workerRuntimes ?? 0,
263
- staleWorkerRuntimes: evidence?.staleWorkerRuntimes ?? 0,
264
- artifactIngestions: evidence?.artifactIngestions ?? 0,
265
- staleReasons,
266
- affectedFileConflicts,
267
- subagentDispatches: status.subagentDispatches.dispatches,
268
- blockingSubagents: status.subagentDispatches.blockingOpen,
269
- capabilityWarnings: status.capabilityHealth.warnings.length,
270
- durableGaps: status.durableGaps.length,
271
- stageTeamSessions: status.stageTeamHealth.sessions,
272
- stageRoleResults: status.stageTeamHealth.roleResults,
273
- advisorAssessments: status.stageTeamHealth.advisorAssessments,
274
- fanInEvidenceSets: status.stageTeamHealth.fanInEvidenceSets,
275
- workflowGateDecisions: status.workflowGateHealth.decisions
276
- },
277
- taskRisk: status.taskRisk,
278
- next: status.recommendedNextCommand
279
- };
280
- }
281
-
282
- async function inspectStageTeamHealth(projectRoot: string, partition: string, runId: string | null): Promise<StageTeamHealthProjection> {
283
- const query = runId ? { partition, runId } : { partition };
284
- const [sessions, roleResults, advisorAssessments, fanInEvidenceSets] = await Promise.all([
285
- listRuntimeStageTeamSessions(projectRoot, query),
286
- listRuntimeStageRoleResults(projectRoot, query),
287
- listRuntimeLlmAdvisorAssessments(projectRoot, query),
288
- listRuntimeFanInEvidenceSets(projectRoot, query)
289
- ]);
290
- return {
291
- sessions: sessions.length,
292
- roleResults: roleResults.length,
293
- advisorAssessments: advisorAssessments.length,
294
- fanInEvidenceSets: fanInEvidenceSets.length,
295
- highConcernAdvisors: advisorAssessments.filter((assessment) => assessment.concern === 'high').length,
296
- lifecycleProfiles: [...new Set(sessions.map((session) => session.lifecycleProfile))].sort(),
297
- directProfileRequiresNoTeam: !sessions.some((session) => session.lifecycleProfile === 'direct' && session.assignments.some((assignment) => assignment.required))
298
- };
299
- }
300
-
301
- async function inspectWorkflowGateHealth(projectRoot: string, partition: string, runId: string | null): Promise<WorkflowGateHealthProjection> {
302
- const query = runId ? { partition, runId } : { partition };
303
- const decisions = await listRuntimeWorkflowGateDecisions(projectRoot, query);
304
- const statusesByKind = Object.fromEntries(decisions.map((decision) => [decision.decisionKind, decision.status])) as WorkflowGateHealthProjection['statusesByKind'];
305
- return {
306
- decisions: decisions.length,
307
- blocked: decisions.filter((decision) => decision.status === 'BLOCKED').length,
308
- warnings: decisions.filter((decision) => decision.status === 'WARN').length,
309
- humanRequired: decisions.filter((decision) => decision.humanRequired).length,
310
- latestDecision: decisions[0] ?? null,
311
- statusesByKind
312
- };
313
- }
314
-
315
- function summarizeTaskRisk(tasks: Array<Parameters<typeof buildTaskRiskProfile>[0]>): TaskRiskSummary {
316
- const entries = tasks.map((task) => ({ taskId: task?.id ?? null, profile: buildTaskRiskProfile(task) }));
317
- return {
318
- highRiskTasks: entries.filter(({ profile }) => profile.riskLevel === 'high').map(({ taskId }) => taskId).filter((taskId): taskId is string => Boolean(taskId)).sort(),
319
- mediumRiskTasks: entries.filter(({ profile }) => profile.riskLevel === 'medium').map(({ taskId }) => taskId).filter((taskId): taskId is string => Boolean(taskId)).sort(),
320
- sourceBoundaryTasks: entries.filter(({ profile }) => profile.sourceBoundary).map(({ taskId }) => taskId).filter((taskId): taskId is string => Boolean(taskId)).sort(),
321
- contextRiskTasks: entries.filter(({ profile }) => profile.contextRisk).map(({ taskId }) => taskId).filter((taskId): taskId is string => Boolean(taskId)).sort(),
322
- tokenRiskTasks: entries.filter(({ profile }) => profile.tokenRisk).map(({ taskId }) => taskId).filter((taskId): taskId is string => Boolean(taskId)).sort(),
323
- performanceRiskTasks: entries.filter(({ profile }) => profile.performanceRisk).map(({ taskId }) => taskId).filter((taskId): taskId is string => Boolean(taskId)).sort()
324
- };
325
- }
326
-
327
- function taskHealth(status: ProjectStatus): StatuslineProjection['taskHealth'] {
328
- if (status.tasks.total === 0) {
329
- return 'empty';
330
- }
331
- if (status.tasks.blocked > 0 || status.gaps.some((gap) => gap.severity === 'blocking') || status.durableGaps.some((gap) => gap.severity === 'blocking') || status.dependencyBlockers.length > 0) {
332
- return 'blocked';
333
- }
334
- if (status.tasks.completed === status.tasks.total) {
335
- return 'complete';
336
- }
337
- return 'active';
338
- }
339
-
340
- function testHealth(status: ProjectStatus): StatuslineProjection['testHealth'] {
341
- if (!status.latestRun) {
342
- return 'none';
343
- }
344
- if (status.latestRun.validationStatus === 'pass') {
345
- return 'pass';
346
- }
347
- if (status.latestRun.validationStatus === 'pass_with_gaps') {
348
- return 'warn';
349
- }
350
- return 'blocked';
351
- }
352
-
353
- function evidenceHealth(status: ProjectStatus, runtimeHealth: StatuslineProjection['runtimeHealth']): StatuslineProjection['evidenceHealth'] {
354
- if (!status.latestRunEvidence) {
355
- return 'none';
356
- }
357
- if (status.latestRunStaleReasons.length > 0 || status.affectedFileConflicts.length > 0 || status.latestRunEvidence.staleWorkerRuntimes > 0) {
358
- return 'blocked';
359
- }
360
- if (runtimeHealth === 'blocked' || status.latestRunEvidence.tasksChangedAfterRun) {
361
- return 'warn';
362
- }
363
- return 'pass';
364
- }
365
-
366
- async function inspectCapabilityHealth(projectRoot: string, latestRunId: string | null, tasks: SddTaskModel['tasks']): Promise<CapabilityHealthProjection> {
367
- try {
368
- const [catalog, runtimeValidation, testRuns] = await Promise.all([
369
- inspectAgentCapabilityCatalog(projectRoot),
370
- validateAgentSkillTeamRuntime(projectRoot),
371
- latestRunId ? listRuntimeTestRuns(projectRoot, latestRunId) : Promise.resolve([])
372
- ]);
373
- const baselineDomains = uniqueSorted(catalog.capabilities
374
- .filter((capability) => capability.domainGroup === 'professional' && capability.provenance.sourceId === 'sdd_professional_baseline')
375
- .map((capability) => capability.domain)) as ProfessionalCapabilityDomain[];
376
- const missingBaselineDomains = REQUIRED_PROFESSIONAL_CAPABILITY_DOMAINS.filter((domain) => !baselineDomains.includes(domain));
377
- const latestEvidence = capabilityEvidenceFromTestRuns(testRuns.map((run) => run.payload));
378
- const acceptedProfessionalEvidence = latestEvidence.filter((item) => item.source === 'professional_capability' && item.class === 'accepted');
379
- const activeDomains = uniqueSorted(acceptedProfessionalEvidence
380
- .map((item) => item.domainOrSourceId)) as AgentCapabilityDomain[];
381
- const activePacks = uniqueSorted(catalog.materialPacks
382
- .filter((pack) => pack.domains.some((domain) => activeDomains.includes(domain)))
383
- .map((pack) => pack.id));
384
- const sources = runtimeValidation.inspection.capabilitySources;
385
- const blocked = missingBaselineDomains.length > 0 || !runtimeValidation.valid;
386
- const noEvidence = latestRunId !== null && latestEvidence.length === 0;
387
- const releaseCriticalGaps = releaseCriticalCapabilityGaps(tasks, acceptedProfessionalEvidence);
388
- const warnings = [
389
- ...missingBaselineDomains.map((domain) => `missing baseline professional domain ${domain}`),
390
- ...runtimeValidation.issues.map((issue) => issue.message),
391
- ...noCapabilityEvidenceWarnings(noEvidence),
392
- ...releaseCriticalGaps.map((domain) => `release-critical capability gap ${domain}`)
393
- ];
394
- return {
395
- status: blocked ? 'blocked' : warnings.length > 0 ? 'warn' : 'pass',
396
- requiredProfessionalDomains: [...REQUIRED_PROFESSIONAL_CAPABILITY_DOMAINS],
397
- baselineDomains,
398
- missingBaselineDomains,
399
- materialPacks: catalog.materialPacks.length,
400
- activeDomains,
401
- activePacks,
402
- sources: {
403
- total: sources.length,
404
- quarantined: sources.filter((source) => source.quarantineRequired || source.quarantineStatus === 'quarantined').length,
405
- denied: sources.filter((source) => source.quarantineStatus === 'denied').length,
406
- futureAdapters: sources.filter((source) => source.hostCompatibility.some((item) => item.includes('future_adapter'))).length
407
- },
408
- evidence: capabilityEvidenceCounts(latestEvidence),
409
- releaseCriticalGaps,
410
- warnings: uniqueSorted(warnings),
411
- reasons: capabilityHealthReasons(latestRunId, latestEvidence.length, blocked, warnings.length)
412
- };
413
- } catch (error) {
414
- return {
415
- status: 'absent',
416
- requiredProfessionalDomains: [...REQUIRED_PROFESSIONAL_CAPABILITY_DOMAINS],
417
- baselineDomains: [],
418
- missingBaselineDomains: [],
419
- materialPacks: 0,
420
- activeDomains: [],
421
- activePacks: [],
422
- sources: { total: 0, quarantined: 0, denied: 0, futureAdapters: 0 },
423
- evidence: { accepted: 0, candidate: 0, quarantined: 0, diagnostic: 0, blocked: 0 },
424
- warnings: [],
425
- releaseCriticalGaps: [],
426
- reasons: [`capability health unavailable: ${error instanceof Error ? error.message : String(error)}`]
427
- };
428
- }
429
- }
430
-
431
- function capabilityEvidenceFromTestRuns(payloads: unknown[]): CapabilityEvidenceClassification[] {
432
- return payloads.flatMap((payload) => isRecord(payload) && Array.isArray(payload.capabilityEvidence)
433
- ? payload.capabilityEvidence.filter(isCapabilityEvidenceClassification)
434
- : []);
435
- }
436
-
437
- function isCapabilityEvidenceClassification(value: unknown): value is CapabilityEvidenceClassification {
438
- return isRecord(value)
439
- && (value.class === 'accepted' || value.class === 'candidate' || value.class === 'quarantined' || value.class === 'diagnostic' || value.class === 'blocked')
440
- && (value.source === 'professional_capability' || value.source === 'external_source' || value.source === 'runtime_diagnostic')
441
- && typeof value.domainOrSourceId === 'string';
442
- }
443
-
444
- function capabilityEvidenceCounts(evidence: CapabilityEvidenceClassification[]): CapabilityHealthProjection['evidence'] {
445
- return {
446
- accepted: evidence.filter((item) => item.class === 'accepted').length,
447
- candidate: evidence.filter((item) => item.class === 'candidate').length,
448
- quarantined: evidence.filter((item) => item.class === 'quarantined').length,
449
- diagnostic: evidence.filter((item) => item.class === 'diagnostic').length,
450
- blocked: evidence.filter((item) => item.class === 'blocked').length
451
- };
452
- }
453
-
454
- function releaseCriticalCapabilityGaps(tasks: SddTaskModel['tasks'], acceptedProfessionalEvidence: CapabilityEvidenceClassification[]): ProfessionalCapabilityDomain[] {
455
- const required = releaseCriticalDomainsForTasks(tasks);
456
- const acceptedEvidenceDomains = new Set(acceptedProfessionalEvidence.map((item) => item.domainOrSourceId));
457
- return required.filter((domain) => !acceptedEvidenceDomains.has(domain));
458
- }
459
-
460
- function releaseCriticalDomainsForTasks(tasks: SddTaskModel['tasks']): ProfessionalCapabilityDomain[] {
461
- const domains = new Set<ProfessionalCapabilityDomain>();
462
- for (const task of tasks) {
463
- const text = [task.title ?? '', ...task.risk, ...task.affectedFiles, ...task.validation, ...task.acceptance, task.boundary ?? ''].join('\n').toLowerCase();
464
- if (/security|auth|permission|secret|token|安全/.test(text)) {
465
- domains.add('security-engineering');
466
- }
467
- if (/performance|latency|throughput|benchmark|capacity|性能/.test(text)) {
468
- domains.add('performance-engineering');
469
- }
470
- if (/observability|logging|metrics|tracing|diagnostic|可观测/.test(text)) {
471
- domains.add('observability-engineering');
472
- }
473
- if (/release|deploy|rollback|ci|ship|发布/.test(text)) {
474
- domains.add('release-engineering');
475
- }
476
- if (/database|schema|migration|query|data|db|数据/.test(text)) {
477
- domains.add('db-data-engineering');
478
- }
479
- if (/ui|ux|frontend|browser|figma|accessibility|交互|前端/.test(text)) {
480
- domains.add(text.includes('frontend') || text.includes('browser') || text.includes('前端') ? 'frontend-engineering' : 'ui-ux-product-design');
481
- }
482
- }
483
- return [...domains].sort((left, right) => left.localeCompare(right));
484
- }
485
-
486
- function noCapabilityEvidenceWarnings(noEvidence: boolean): string[] {
487
- return noEvidence ? ['latest run has no capability evidence classification'] : [];
488
- }
489
-
490
- function capabilityHealthReasons(latestRunId: string | null, evidenceCount: number, blocked: boolean, warnings: number): string[] {
491
- if (blocked) {
492
- return ['Capability catalog or runtime source validation has blocking issues.'];
493
- }
494
- if (latestRunId !== null && evidenceCount === 0) {
495
- return ['Latest run has no recorded capability evidence classification.'];
496
- }
497
- if (warnings > 0) {
498
- return ['Capability health has warnings; inspect status JSON for exact missing evidence or release-critical gaps.'];
499
- }
500
- return ['Professional capability baseline and source policy are visible.'];
501
- }
502
-
503
- function isRecord(value: unknown): value is Record<string, unknown> {
504
- return Boolean(value) && typeof value === 'object';
505
- }
506
-
507
- function uniqueSorted<T extends string>(values: T[]): T[] {
508
- return [...new Set(values)].sort((left, right) => left.localeCompare(right));
509
- }
510
-
511
- async function inspectTokenRuntimeProjection(projectRoot: string): Promise<TokenRuntimeProjection> {
512
- const projections = await listRuntimeProjections(projectRoot, ['context_build', 'team_runtime_decision']);
513
- const contextPackages = projections.filter((projection) => projection.projectionType === 'context_build');
514
- const teamRuntimeDecisions = projections.filter((projection) => projection.projectionType === 'team_runtime_decision');
515
- const contextTokenEstimates = contextPackages.map((projection) => tokenEstimateFromContextBuildPayload(projection.payload)).filter((value): value is number => value !== null);
516
- const teamTokenEstimates = teamRuntimeDecisions.map((projection) => tokenEstimateFromTeamRuntimePayload(projection.payload)).filter((value): value is number => value !== null);
517
- const estimatedTokens = [...contextTokenEstimates, ...teamTokenEstimates].reduce((total, value) => total + value, 0);
518
- const pressureReasons = [
519
- ...contextPackages.map((projection) => contextPressureReason(projection.payload)).filter((value): value is string => value !== null),
520
- ...teamRuntimeDecisions.map((projection) => teamPressureReason(projection.payload)).filter((value): value is string => value !== null)
521
- ];
522
-
523
- return {
524
- health: projections.length === 0 ? 'unknown' : pressureReasons.length > 0 ? 'pressure' : 'nominal',
525
- estimatedTokens: projections.length === 0 ? null : estimatedTokens,
526
- contextPackages: contextPackages.length,
527
- teamRuntimeDecisions: teamRuntimeDecisions.length,
528
- pressureReasons
529
- };
530
- }
531
-
532
- function tokenEstimateFromContextBuildPayload(payload: unknown): number | null {
533
- if (!payload || typeof payload !== 'object') {
534
- return null;
535
- }
536
- const budget = (payload as { budget?: { estimatedTokens?: unknown } }).budget;
537
- return typeof budget?.estimatedTokens === 'number' ? budget.estimatedTokens : null;
538
- }
539
-
540
- function tokenEstimateFromTeamRuntimePayload(payload: unknown): number | null {
541
- if (!payload || typeof payload !== 'object') {
542
- return null;
543
- }
544
- const decision = payload as { roleIds?: unknown; telemetryPolicy?: { contextBudget?: unknown } | null };
545
- const roleCount = Array.isArray(decision.roleIds) ? decision.roleIds.length : 0;
546
- const contextBudget = decision.telemetryPolicy?.contextBudget;
547
- const perRole = contextBudget === 'medium' ? 3000 : contextBudget === 'small' ? 1200 : 600;
548
- return roleCount * perRole;
549
- }
550
-
551
- function contextPressureReason(payload: unknown): string | null {
552
- if (!payload || typeof payload !== 'object') {
553
- return null;
554
- }
555
- const contextPackage = payload as { taskId?: unknown; profile?: unknown; budget?: { estimatedBytes?: unknown; maxBytes?: unknown } };
556
- const estimatedBytes = contextPackage.budget?.estimatedBytes;
557
- const maxBytes = contextPackage.budget?.maxBytes;
558
- if (contextPackage.profile === 'brief') {
559
- return null;
560
- }
561
- if (typeof estimatedBytes === 'number' && typeof maxBytes === 'number' && estimatedBytes >= maxBytes * 0.85) {
562
- return `context_budget_pressure:${String(contextPackage.taskId ?? 'unknown')}`;
563
- }
564
- return null;
565
- }
566
-
567
- function teamPressureReason(payload: unknown): string | null {
568
- if (!payload || typeof payload !== 'object') {
569
- return null;
570
- }
571
- const decision = payload as { command?: unknown; mode?: unknown; roleIds?: unknown };
572
- const roleCount = Array.isArray(decision.roleIds) ? decision.roleIds.length : 0;
573
- if (decision.mode === 'team-required' || roleCount > 2) {
574
- return `team_runtime_pressure:${String(decision.command ?? 'unknown')}`;
575
- }
576
- return null;
577
- }
578
-
579
- async function inspectRunEvidenceSummary(projectRoot: string, runId: string): Promise<RunEvidenceSummary> {
580
- const [state, events, agentExecutions, teamSessions, workerRuntimeList] = await Promise.all([
581
- readRunState(projectRoot, runId),
582
- readRunEvents(projectRoot, runId),
583
- listAgentExecutionRecords(projectRoot, runId),
584
- listTeamSessionRecords(projectRoot, runId),
585
- listResidentWorkerRuntimes(projectRoot, { runId })
586
- ]);
587
- return {
588
- agentExecutions: agentExecutions.length,
589
- teamSessions: teamSessions.length,
590
- artifactIngestions: Object.keys(state.artifactIngestions ?? {}).length,
591
- workerRuntimes: workerRuntimeList.runtimes.length,
592
- staleWorkerRuntimes: workerRuntimeList.staleRuntimes,
593
- routePreflight: events.some((event) => event.event === 'agent_router_preflight'),
594
- tasksChangedAfterRun: false,
595
- tasksUpdatedAt: null,
596
- runUpdatedAt: state.updatedAt ?? null
597
- };
598
- }
599
-
600
- async function addTaskDocumentStalenessToRunEvidence(tasksPath: string, latestRun: RunSummary, evidence: RunEvidenceSummary): Promise<RunEvidenceSummary> {
601
- try {
602
- const tasksStat = await stat(tasksPath);
603
- const runUpdatedAtMs = Date.parse(latestRun.updatedAt);
604
- if (Number.isNaN(runUpdatedAtMs)) {
605
- return { ...evidence, runUpdatedAt: latestRun.updatedAt };
606
- }
607
- return {
608
- ...evidence,
609
- tasksChangedAfterRun: tasksStat.mtimeMs > runUpdatedAtMs,
610
- tasksUpdatedAt: tasksStat.mtime.toISOString(),
611
- runUpdatedAt: latestRun.updatedAt
612
- };
613
- } catch {
614
- return { ...evidence, runUpdatedAt: latestRun.updatedAt };
615
- }
616
- }
617
-
618
- async function getGitRoot(projectRoot: string): Promise<string | null> {
619
- try {
620
- const result = await execFileAsync('git', ['-C', projectRoot, 'rev-parse', '--show-toplevel']);
621
- return result.stdout.trim();
622
- } catch {
623
- return null;
624
- }
625
- }
1
+ import { execFile } from 'node:child_process';
2
+ import { stat } from 'node:fs/promises';
3
+ import { promisify } from 'node:util';
4
+ import type { ContextBranchSource, ContextResolverContract } from '../sdd-docs/context.js';
5
+ import type { SddTaskGap, SddTaskModel } from '../sdd-docs/task-parser.js';
6
+ import type { RunState, RunSummary } from '../run-state/model.js';
7
+ import { readRunEvents } from '../run-state/events.js';
8
+ import { readRunState } from '../run-state/run-state.js';
9
+ import { listAgentExecutionRecords, listTeamSessionRecords } from '../execution/agent-execution-records.js';
10
+ import { listResidentWorkerRuntimes } from '../execution/resident-worker.js';
11
+ import { resolveWorkflowState, type WorkflowLatestTaskRun, type WorkflowNextIntent } from '../workflow-state/resolve.js';
12
+ import type { WorkflowAffectedFileConflict } from '../workflow-state/affected-file-conflicts.js';
13
+ import type { WorkflowDependencyBlocker } from '../workflow-state/dependencies.js';
14
+ import type { LatestEligibleRunSelection } from '../workflow-state/latest-eligible-run.js';
15
+ import { buildTaskRiskProfile } from '../task-risk-profile.js';
16
+ import { listRuntimeProjections } from '../storage/runtime-store.js';
17
+ import { inspectLifecycleRiskDecisionForModel, type LifecycleRiskConsumerDiagnostic } from '../risk.js';
18
+ import { inspectWorkflowStageHandoff, type WorkflowStageHandoffDiagnostic } from '../stage-runtime.js';
19
+ import { inspectContextOffloadRuntime, type ContextRuntimeDiagnostic } from '../context-offload.js';
20
+ import { inspectSubagentDispatches, type SubagentDispatchDiagnostic } from '../subagents.js';
21
+ import { inspectAgentCapabilityCatalog, REQUIRED_PROFESSIONAL_CAPABILITY_DOMAINS, type AgentCapabilityDomain, type ProfessionalCapabilityDomain } from '../registries/agent-capability-catalog.js';
22
+ import { validateAgentSkillTeamRuntime } from '../router/runtime-validation.js';
23
+ import { listRuntimeDurableGaps, listRuntimeFanInEvidenceSets, listRuntimeLlmAdvisorAssessments, listRuntimeStageRoleResults, listRuntimeStageTeamSessions, listRuntimeTestRuns, listRuntimeWorkflowGateDecisions, type RuntimeDurableGapRecord } from '../storage/runtime-store.js';
24
+ import type { CapabilityEvidenceClassification } from '../evidence-runtime.js';
25
+ import type { WorkflowGateDecision, WorkflowGateDecisionKind } from '../workflow-gate/types.js';
26
+
27
+ const execFileAsync = promisify(execFile);
28
+
29
+ export interface RunEvidenceSummary {
30
+ agentExecutions: number;
31
+ teamSessions: number;
32
+ artifactIngestions: number;
33
+ workerRuntimes: number;
34
+ staleWorkerRuntimes: number;
35
+ routePreflight: boolean;
36
+ tasksChangedAfterRun: boolean;
37
+ tasksUpdatedAt: string | null;
38
+ runUpdatedAt: string | null;
39
+ }
40
+
41
+ export interface TaskRiskSummary {
42
+ highRiskTasks: string[];
43
+ mediumRiskTasks: string[];
44
+ sourceBoundaryTasks: string[];
45
+ contextRiskTasks: string[];
46
+ tokenRiskTasks: string[];
47
+ performanceRiskTasks: string[];
48
+ }
49
+
50
+ export interface TokenRuntimeProjection {
51
+ health: 'unknown' | 'nominal' | 'pressure';
52
+ estimatedTokens: number | null;
53
+ contextPackages: number;
54
+ teamRuntimeDecisions: number;
55
+ pressureReasons: string[];
56
+ }
57
+
58
+ export type CapabilityHealthStatus = 'absent' | 'pass' | 'warn' | 'blocked';
59
+
60
+ export interface CapabilityHealthProjection {
61
+ status: CapabilityHealthStatus;
62
+ requiredProfessionalDomains: ProfessionalCapabilityDomain[];
63
+ baselineDomains: ProfessionalCapabilityDomain[];
64
+ missingBaselineDomains: ProfessionalCapabilityDomain[];
65
+ materialPacks: number;
66
+ activeDomains: AgentCapabilityDomain[];
67
+ activePacks: string[];
68
+ sources: {
69
+ total: number;
70
+ quarantined: number;
71
+ denied: number;
72
+ futureAdapters: number;
73
+ };
74
+ evidence: {
75
+ accepted: number;
76
+ candidate: number;
77
+ quarantined: number;
78
+ diagnostic: number;
79
+ blocked: number;
80
+ };
81
+ warnings: string[];
82
+ releaseCriticalGaps: ProfessionalCapabilityDomain[];
83
+ reasons: string[];
84
+ }
85
+
86
+ export interface StageTeamHealthProjection {
87
+ sessions: number;
88
+ roleResults: number;
89
+ advisorAssessments: number;
90
+ fanInEvidenceSets: number;
91
+ highConcernAdvisors: number;
92
+ lifecycleProfiles: string[];
93
+ directProfileRequiresNoTeam: boolean;
94
+ }
95
+
96
+ export interface WorkflowGateHealthProjection {
97
+ decisions: number;
98
+ blocked: number;
99
+ warnings: number;
100
+ humanRequired: number;
101
+ latestDecision: WorkflowGateDecision | null;
102
+ statusesByKind: Partial<Record<WorkflowGateDecisionKind, WorkflowGateDecision['status']>>;
103
+ }
104
+
105
+ export interface ProjectStatus {
106
+ branch: string;
107
+ workflowStatus: 'active' | 'not_started';
108
+ context: ContextResolverContract;
109
+ gitRoot: string | null;
110
+ documents: SddTaskModel['documents'];
111
+ tasks: {
112
+ total: number;
113
+ pending: number;
114
+ inProgress: number;
115
+ completed: number;
116
+ blocked: number;
117
+ deferred: number;
118
+ unknown: number;
119
+ gaps: number;
120
+ };
121
+ taskRisk: TaskRiskSummary;
122
+ latestRun: RunSummary | null;
123
+ latestRunsByTask: WorkflowLatestTaskRun[];
124
+ latestEligibleRunsByTask: LatestEligibleRunSelection[];
125
+ latestRunEvidence: RunEvidenceSummary | null;
126
+ latestRunStaleReasons: string[];
127
+ affectedFileConflicts: WorkflowAffectedFileConflict[];
128
+ dependencyBlockers: WorkflowDependencyBlocker[];
129
+ recommendedNextCommand: string;
130
+ nextIntent: WorkflowNextIntent;
131
+ tokenProjection: TokenRuntimeProjection;
132
+ contextRuntime: ContextRuntimeDiagnostic;
133
+ subagentDispatches: SubagentDispatchDiagnostic;
134
+ lifecycleRisk: LifecycleRiskConsumerDiagnostic;
135
+ capabilityHealth: CapabilityHealthProjection;
136
+ workflowHandoff: WorkflowStageHandoffDiagnostic;
137
+ gaps: SddTaskGap[];
138
+ durableGaps: RuntimeDurableGapRecord[];
139
+ stageTeamHealth: StageTeamHealthProjection;
140
+ workflowGateHealth: WorkflowGateHealthProjection;
141
+ }
142
+
143
+ export interface StatuslineProjection {
144
+ contract: 'sdd-statusline-projection-v1';
145
+ branch: string;
146
+ workflow: ProjectStatus['workflowStatus'];
147
+ taskHealth: 'empty' | 'active' | 'blocked' | 'complete';
148
+ runtimeHealth: 'none' | 'pass' | 'warn' | 'blocked';
149
+ testHealth: 'none' | 'pass' | 'warn' | 'blocked';
150
+ teamHealth: 'none' | 'active';
151
+ tokenHealth: 'unknown' | 'nominal' | 'pressure';
152
+ contextLoad: ProjectStatus['contextRuntime']['level'];
153
+ contextAction: ProjectStatus['contextRuntime']['action'];
154
+ subagentHealth: ProjectStatus['subagentDispatches']['status'];
155
+ evidenceHealth: 'none' | 'pass' | 'warn' | 'blocked';
156
+ capabilityHealth: ProjectStatus['capabilityHealth']['status'];
157
+ latestRunId: string | null;
158
+ counts: {
159
+ tasks: ProjectStatus['tasks'];
160
+ agentExecutions: number;
161
+ teamSessions: number;
162
+ workerRuntimes: number;
163
+ staleWorkerRuntimes: number;
164
+ artifactIngestions: number;
165
+ staleReasons: number;
166
+ affectedFileConflicts: number;
167
+ subagentDispatches: number;
168
+ blockingSubagents: number;
169
+ capabilityWarnings: number;
170
+ durableGaps: number;
171
+ stageTeamSessions: number;
172
+ stageRoleResults: number;
173
+ advisorAssessments: number;
174
+ fanInEvidenceSets: number;
175
+ workflowGateDecisions: number;
176
+ };
177
+ taskRisk: TaskRiskSummary;
178
+ next: string;
179
+ }
180
+
181
+ export async function getProjectStatus(projectRoot: string, options: { branch?: string | null; branchSource?: ContextBranchSource } = {}): Promise<ProjectStatus> {
182
+ const [workflow, gitRoot] = await Promise.all([resolveWorkflowState(projectRoot, options), getGitRoot(projectRoot)]);
183
+ const latestRun = workflow.latestRun;
184
+ const latestRunState = workflow.latestRunState;
185
+ const latestRunEvidence = latestRun ? await inspectRunEvidenceSummary(projectRoot, latestRun.runId) : null;
186
+ const enrichedLatestRunEvidence = latestRun && latestRunEvidence
187
+ ? await addTaskDocumentStalenessToRunEvidence(workflow.model.tasksPath, latestRun, latestRunEvidence)
188
+ : latestRunEvidence;
189
+ const latestRunStaleReasons = latestRunState ? await runDocumentStaleReasons(projectRoot, latestRunState, workflow.model) : [];
190
+ const taskRisk = summarizeTaskRisk(workflow.model.tasks);
191
+ const tokenProjection = await inspectTokenRuntimeProjection(projectRoot);
192
+ const lifecycleRisk = await inspectLifecycleRiskDecisionForModel(projectRoot, workflow.branch, workflow.model);
193
+ const workflowHandoff = await inspectWorkflowStageHandoff(projectRoot, workflow.branch);
194
+ const contextRuntime = await inspectContextOffloadRuntime(projectRoot, workflow.branch);
195
+ const subagentDispatches = await inspectSubagentDispatches(projectRoot, workflow.branch);
196
+ const capabilityHealth = await inspectCapabilityHealth(projectRoot, latestRun?.runId ?? null, workflow.model.tasks);
197
+ const durableGaps = await listRuntimeDurableGaps(projectRoot, { partition: workflow.branch, status: 'open_terminal' });
198
+ const stageTeamHealth = await inspectStageTeamHealth(projectRoot, workflow.branch, latestRun?.runId ?? null);
199
+ const workflowGateHealth = await inspectWorkflowGateHealth(projectRoot, workflow.branch, latestRun?.runId ?? null);
200
+
201
+ return {
202
+ branch: workflow.branch,
203
+ workflowStatus: workflow.workflowStatus,
204
+ context: workflow.context,
205
+ gitRoot,
206
+ documents: workflow.documents,
207
+ tasks: workflow.taskCounts,
208
+ taskRisk,
209
+ latestRun,
210
+ latestRunsByTask: workflow.latestRunsByTask,
211
+ latestEligibleRunsByTask: workflow.latestEligibleRunsByTask,
212
+ latestRunEvidence: enrichedLatestRunEvidence,
213
+ latestRunStaleReasons,
214
+ affectedFileConflicts: workflow.affectedFileConflicts,
215
+ dependencyBlockers: workflow.dependencyBlockers,
216
+ recommendedNextCommand: workflow.recommendedNextCommand,
217
+ nextIntent: workflow.nextIntent,
218
+ tokenProjection,
219
+ contextRuntime,
220
+ subagentDispatches,
221
+ lifecycleRisk,
222
+ workflowHandoff,
223
+ capabilityHealth,
224
+ gaps: workflow.visibleGaps,
225
+ durableGaps,
226
+ stageTeamHealth,
227
+ workflowGateHealth
228
+ };
229
+ }
230
+
231
+ export async function getStatuslineProjection(projectRoot: string, options: { branch?: string | null; branchSource?: ContextBranchSource } = {}): Promise<StatuslineProjection> {
232
+ return statuslineProjectionFromStatus(await getProjectStatus(projectRoot, options));
233
+ }
234
+
235
+ export function statuslineProjectionFromStatus(status: ProjectStatus): StatuslineProjection {
236
+ const evidence = status.latestRunEvidence;
237
+ const runtimeHealth = status.latestRun
238
+ ? status.latestRun.status === 'completed' && status.latestRun.validationStatus === 'pass' ? 'pass' : 'blocked'
239
+ : 'none';
240
+ const staleReasons = status.latestRunStaleReasons.length;
241
+ const affectedFileConflicts = status.affectedFileConflicts.length;
242
+ return {
243
+ contract: 'sdd-statusline-projection-v1',
244
+ branch: status.branch,
245
+ workflow: status.workflowStatus,
246
+ taskHealth: taskHealth(status),
247
+ runtimeHealth,
248
+ testHealth: testHealth(status),
249
+ teamHealth: (evidence?.teamSessions ?? 0) > 0 || status.stageTeamHealth.sessions > 0 || status.stageTeamHealth.roleResults > 0 ? 'active' : 'none',
250
+ tokenHealth: status.tokenProjection.health,
251
+ contextLoad: status.contextRuntime.level,
252
+ contextAction: status.contextRuntime.action,
253
+ subagentHealth: status.subagentDispatches.status,
254
+ evidenceHealth: evidenceHealth(status, runtimeHealth),
255
+ capabilityHealth: status.capabilityHealth.status,
256
+ latestRunId: status.latestRun?.runId ?? null,
257
+ counts: {
258
+ tasks: status.tasks,
259
+ agentExecutions: evidence?.agentExecutions ?? 0,
260
+ teamSessions: evidence?.teamSessions ?? 0,
261
+ workerRuntimes: evidence?.workerRuntimes ?? 0,
262
+ staleWorkerRuntimes: evidence?.staleWorkerRuntimes ?? 0,
263
+ artifactIngestions: evidence?.artifactIngestions ?? 0,
264
+ staleReasons,
265
+ affectedFileConflicts,
266
+ subagentDispatches: status.subagentDispatches.dispatches,
267
+ blockingSubagents: status.subagentDispatches.blockingOpen,
268
+ capabilityWarnings: status.capabilityHealth.warnings.length,
269
+ durableGaps: status.durableGaps.length,
270
+ stageTeamSessions: status.stageTeamHealth.sessions,
271
+ stageRoleResults: status.stageTeamHealth.roleResults,
272
+ advisorAssessments: status.stageTeamHealth.advisorAssessments,
273
+ fanInEvidenceSets: status.stageTeamHealth.fanInEvidenceSets,
274
+ workflowGateDecisions: status.workflowGateHealth.decisions
275
+ },
276
+ taskRisk: status.taskRisk,
277
+ next: status.recommendedNextCommand
278
+ };
279
+ }
280
+
281
+ async function inspectStageTeamHealth(projectRoot: string, partition: string, runId: string | null): Promise<StageTeamHealthProjection> {
282
+ const query = runId ? { partition, runId } : { partition };
283
+ const [sessions, roleResults, advisorAssessments, fanInEvidenceSets] = await Promise.all([
284
+ listRuntimeStageTeamSessions(projectRoot, query),
285
+ listRuntimeStageRoleResults(projectRoot, query),
286
+ listRuntimeLlmAdvisorAssessments(projectRoot, query),
287
+ listRuntimeFanInEvidenceSets(projectRoot, query)
288
+ ]);
289
+ return {
290
+ sessions: sessions.length,
291
+ roleResults: roleResults.length,
292
+ advisorAssessments: advisorAssessments.length,
293
+ fanInEvidenceSets: fanInEvidenceSets.length,
294
+ highConcernAdvisors: advisorAssessments.filter((assessment) => assessment.concern === 'high').length,
295
+ lifecycleProfiles: [...new Set(sessions.map((session) => session.lifecycleProfile))].sort(),
296
+ directProfileRequiresNoTeam: !sessions.some((session) => session.lifecycleProfile === 'direct' && session.assignments.some((assignment) => assignment.required))
297
+ };
298
+ }
299
+
300
+ async function inspectWorkflowGateHealth(projectRoot: string, partition: string, runId: string | null): Promise<WorkflowGateHealthProjection> {
301
+ const query = runId ? { partition, runId } : { partition };
302
+ const decisions = await listRuntimeWorkflowGateDecisions(projectRoot, query);
303
+ const statusesByKind = Object.fromEntries(decisions.map((decision) => [decision.decisionKind, decision.status])) as WorkflowGateHealthProjection['statusesByKind'];
304
+ return {
305
+ decisions: decisions.length,
306
+ blocked: decisions.filter((decision) => decision.status === 'BLOCKED').length,
307
+ warnings: decisions.filter((decision) => decision.status === 'WARN').length,
308
+ humanRequired: decisions.filter((decision) => decision.humanRequired).length,
309
+ latestDecision: decisions[0] ?? null,
310
+ statusesByKind
311
+ };
312
+ }
313
+
314
+ function summarizeTaskRisk(tasks: Array<Parameters<typeof buildTaskRiskProfile>[0]>): TaskRiskSummary {
315
+ const entries = tasks.map((task) => ({ taskId: task?.id ?? null, profile: buildTaskRiskProfile(task) }));
316
+ return {
317
+ highRiskTasks: entries.filter(({ profile }) => profile.riskLevel === 'high').map(({ taskId }) => taskId).filter((taskId): taskId is string => Boolean(taskId)).sort(),
318
+ mediumRiskTasks: entries.filter(({ profile }) => profile.riskLevel === 'medium').map(({ taskId }) => taskId).filter((taskId): taskId is string => Boolean(taskId)).sort(),
319
+ sourceBoundaryTasks: entries.filter(({ profile }) => profile.sourceBoundary).map(({ taskId }) => taskId).filter((taskId): taskId is string => Boolean(taskId)).sort(),
320
+ contextRiskTasks: entries.filter(({ profile }) => profile.contextRisk).map(({ taskId }) => taskId).filter((taskId): taskId is string => Boolean(taskId)).sort(),
321
+ tokenRiskTasks: entries.filter(({ profile }) => profile.tokenRisk).map(({ taskId }) => taskId).filter((taskId): taskId is string => Boolean(taskId)).sort(),
322
+ performanceRiskTasks: entries.filter(({ profile }) => profile.performanceRisk).map(({ taskId }) => taskId).filter((taskId): taskId is string => Boolean(taskId)).sort()
323
+ };
324
+ }
325
+
326
+ function taskHealth(status: ProjectStatus): StatuslineProjection['taskHealth'] {
327
+ if (status.tasks.total === 0) {
328
+ return 'empty';
329
+ }
330
+ if (status.tasks.blocked > 0 || status.gaps.some((gap) => gap.severity === 'blocking') || status.durableGaps.some((gap) => gap.severity === 'blocking') || status.dependencyBlockers.length > 0) {
331
+ return 'blocked';
332
+ }
333
+ if (status.tasks.completed === status.tasks.total) {
334
+ return 'complete';
335
+ }
336
+ return 'active';
337
+ }
338
+
339
+ function testHealth(status: ProjectStatus): StatuslineProjection['testHealth'] {
340
+ if (!status.latestRun) {
341
+ return 'none';
342
+ }
343
+ if (status.latestRun.validationStatus === 'pass') {
344
+ return 'pass';
345
+ }
346
+ if (status.latestRun.validationStatus === 'pass_with_gaps') {
347
+ return 'warn';
348
+ }
349
+ return 'blocked';
350
+ }
351
+
352
+ function evidenceHealth(status: ProjectStatus, runtimeHealth: StatuslineProjection['runtimeHealth']): StatuslineProjection['evidenceHealth'] {
353
+ if (!status.latestRunEvidence) {
354
+ return 'none';
355
+ }
356
+ if (status.latestRunStaleReasons.length > 0 || status.affectedFileConflicts.length > 0 || status.latestRunEvidence.staleWorkerRuntimes > 0) {
357
+ return 'blocked';
358
+ }
359
+ if (runtimeHealth === 'blocked' || status.latestRunEvidence.tasksChangedAfterRun) {
360
+ return 'warn';
361
+ }
362
+ return 'pass';
363
+ }
364
+
365
+ async function inspectCapabilityHealth(projectRoot: string, latestRunId: string | null, tasks: SddTaskModel['tasks']): Promise<CapabilityHealthProjection> {
366
+ try {
367
+ const [catalog, runtimeValidation, testRuns] = await Promise.all([
368
+ inspectAgentCapabilityCatalog(projectRoot),
369
+ validateAgentSkillTeamRuntime(projectRoot),
370
+ latestRunId ? listRuntimeTestRuns(projectRoot, latestRunId) : Promise.resolve([])
371
+ ]);
372
+ const baselineDomains = uniqueSorted(catalog.capabilities
373
+ .filter((capability) => capability.domainGroup === 'professional' && capability.provenance.sourceId === 'sdd_professional_baseline')
374
+ .map((capability) => capability.domain)) as ProfessionalCapabilityDomain[];
375
+ const missingBaselineDomains = REQUIRED_PROFESSIONAL_CAPABILITY_DOMAINS.filter((domain) => !baselineDomains.includes(domain));
376
+ const latestEvidence = capabilityEvidenceFromTestRuns(testRuns.map((run) => run.payload));
377
+ const acceptedProfessionalEvidence = latestEvidence.filter((item) => item.source === 'professional_capability' && item.class === 'accepted');
378
+ const activeDomains = uniqueSorted(acceptedProfessionalEvidence
379
+ .map((item) => item.domainOrSourceId)) as AgentCapabilityDomain[];
380
+ const activePacks = uniqueSorted(catalog.materialPacks
381
+ .filter((pack) => pack.domains.some((domain) => activeDomains.includes(domain)))
382
+ .map((pack) => pack.id));
383
+ const sources = runtimeValidation.inspection.capabilitySources;
384
+ const blocked = missingBaselineDomains.length > 0 || !runtimeValidation.valid;
385
+ const noEvidence = latestRunId !== null && latestEvidence.length === 0;
386
+ const releaseCriticalGaps = releaseCriticalCapabilityGaps(tasks, acceptedProfessionalEvidence);
387
+ const warnings = [
388
+ ...missingBaselineDomains.map((domain) => `missing baseline professional domain ${domain}`),
389
+ ...runtimeValidation.issues.map((issue) => issue.message),
390
+ ...noCapabilityEvidenceWarnings(noEvidence),
391
+ ...releaseCriticalGaps.map((domain) => `release-critical capability gap ${domain}`)
392
+ ];
393
+ return {
394
+ status: blocked ? 'blocked' : warnings.length > 0 ? 'warn' : 'pass',
395
+ requiredProfessionalDomains: [...REQUIRED_PROFESSIONAL_CAPABILITY_DOMAINS],
396
+ baselineDomains,
397
+ missingBaselineDomains,
398
+ materialPacks: catalog.materialPacks.length,
399
+ activeDomains,
400
+ activePacks,
401
+ sources: {
402
+ total: sources.length,
403
+ quarantined: sources.filter((source) => source.quarantineRequired || source.quarantineStatus === 'quarantined').length,
404
+ denied: sources.filter((source) => source.quarantineStatus === 'denied').length,
405
+ futureAdapters: sources.filter((source) => source.hostCompatibility.some((item) => item.includes('future_adapter'))).length
406
+ },
407
+ evidence: capabilityEvidenceCounts(latestEvidence),
408
+ releaseCriticalGaps,
409
+ warnings: uniqueSorted(warnings),
410
+ reasons: capabilityHealthReasons(latestRunId, latestEvidence.length, blocked, warnings.length)
411
+ };
412
+ } catch (error) {
413
+ return {
414
+ status: 'absent',
415
+ requiredProfessionalDomains: [...REQUIRED_PROFESSIONAL_CAPABILITY_DOMAINS],
416
+ baselineDomains: [],
417
+ missingBaselineDomains: [],
418
+ materialPacks: 0,
419
+ activeDomains: [],
420
+ activePacks: [],
421
+ sources: { total: 0, quarantined: 0, denied: 0, futureAdapters: 0 },
422
+ evidence: { accepted: 0, candidate: 0, quarantined: 0, diagnostic: 0, blocked: 0 },
423
+ warnings: [],
424
+ releaseCriticalGaps: [],
425
+ reasons: [`capability health unavailable: ${error instanceof Error ? error.message : String(error)}`]
426
+ };
427
+ }
428
+ }
429
+
430
+ function capabilityEvidenceFromTestRuns(payloads: unknown[]): CapabilityEvidenceClassification[] {
431
+ return payloads.flatMap((payload) => isRecord(payload) && Array.isArray(payload.capabilityEvidence)
432
+ ? payload.capabilityEvidence.filter(isCapabilityEvidenceClassification)
433
+ : []);
434
+ }
435
+
436
+ function isCapabilityEvidenceClassification(value: unknown): value is CapabilityEvidenceClassification {
437
+ return isRecord(value)
438
+ && (value.class === 'accepted' || value.class === 'candidate' || value.class === 'quarantined' || value.class === 'diagnostic' || value.class === 'blocked')
439
+ && (value.source === 'professional_capability' || value.source === 'external_source' || value.source === 'runtime_diagnostic')
440
+ && typeof value.domainOrSourceId === 'string';
441
+ }
442
+
443
+ function capabilityEvidenceCounts(evidence: CapabilityEvidenceClassification[]): CapabilityHealthProjection['evidence'] {
444
+ return {
445
+ accepted: evidence.filter((item) => item.class === 'accepted').length,
446
+ candidate: evidence.filter((item) => item.class === 'candidate').length,
447
+ quarantined: evidence.filter((item) => item.class === 'quarantined').length,
448
+ diagnostic: evidence.filter((item) => item.class === 'diagnostic').length,
449
+ blocked: evidence.filter((item) => item.class === 'blocked').length
450
+ };
451
+ }
452
+
453
+ function releaseCriticalCapabilityGaps(tasks: SddTaskModel['tasks'], acceptedProfessionalEvidence: CapabilityEvidenceClassification[]): ProfessionalCapabilityDomain[] {
454
+ const required = releaseCriticalDomainsForTasks(tasks);
455
+ const acceptedEvidenceDomains = new Set(acceptedProfessionalEvidence.map((item) => item.domainOrSourceId));
456
+ return required.filter((domain) => !acceptedEvidenceDomains.has(domain));
457
+ }
458
+
459
+ function releaseCriticalDomainsForTasks(tasks: SddTaskModel['tasks']): ProfessionalCapabilityDomain[] {
460
+ const domains = new Set<ProfessionalCapabilityDomain>();
461
+ for (const task of tasks) {
462
+ const text = [task.title ?? '', ...task.risk, ...task.affectedFiles, ...task.validation, ...task.acceptance, task.boundary ?? ''].join('\n').toLowerCase();
463
+ if (/security|auth|permission|secret|token|安全/.test(text)) {
464
+ domains.add('security-engineering');
465
+ }
466
+ if (/performance|latency|throughput|benchmark|capacity|性能/.test(text)) {
467
+ domains.add('performance-engineering');
468
+ }
469
+ if (/observability|logging|metrics|tracing|diagnostic|可观测/.test(text)) {
470
+ domains.add('observability-engineering');
471
+ }
472
+ if (/release|deploy|rollback|ci|ship|发布/.test(text)) {
473
+ domains.add('release-engineering');
474
+ }
475
+ if (/database|schema|migration|query|data|db|数据/.test(text)) {
476
+ domains.add('db-data-engineering');
477
+ }
478
+ if (/ui|ux|frontend|browser|figma|accessibility|交互|前端/.test(text)) {
479
+ domains.add(text.includes('frontend') || text.includes('browser') || text.includes('前端') ? 'frontend-engineering' : 'ui-ux-product-design');
480
+ }
481
+ }
482
+ return [...domains].sort((left, right) => left.localeCompare(right));
483
+ }
484
+
485
+ function noCapabilityEvidenceWarnings(noEvidence: boolean): string[] {
486
+ return noEvidence ? ['latest run has no capability evidence classification'] : [];
487
+ }
488
+
489
+ function capabilityHealthReasons(latestRunId: string | null, evidenceCount: number, blocked: boolean, warnings: number): string[] {
490
+ if (blocked) {
491
+ return ['Capability catalog or runtime source validation has blocking issues.'];
492
+ }
493
+ if (latestRunId !== null && evidenceCount === 0) {
494
+ return ['Latest run has no recorded capability evidence classification.'];
495
+ }
496
+ if (warnings > 0) {
497
+ return ['Capability health has warnings; inspect status JSON for exact missing evidence or release-critical gaps.'];
498
+ }
499
+ return ['Professional capability baseline and source policy are visible.'];
500
+ }
501
+
502
+ function isRecord(value: unknown): value is Record<string, unknown> {
503
+ return Boolean(value) && typeof value === 'object';
504
+ }
505
+
506
+ function uniqueSorted<T extends string>(values: T[]): T[] {
507
+ return [...new Set(values)].sort((left, right) => left.localeCompare(right));
508
+ }
509
+
510
+ async function inspectTokenRuntimeProjection(projectRoot: string): Promise<TokenRuntimeProjection> {
511
+ const projections = await listRuntimeProjections(projectRoot, ['context_build', 'team_runtime_decision']);
512
+ const contextPackages = projections.filter((projection) => projection.projectionType === 'context_build');
513
+ const teamRuntimeDecisions = projections.filter((projection) => projection.projectionType === 'team_runtime_decision');
514
+ const contextTokenEstimates = contextPackages.map((projection) => tokenEstimateFromContextBuildPayload(projection.payload)).filter((value): value is number => value !== null);
515
+ const teamTokenEstimates = teamRuntimeDecisions.map((projection) => tokenEstimateFromTeamRuntimePayload(projection.payload)).filter((value): value is number => value !== null);
516
+ const estimatedTokens = [...contextTokenEstimates, ...teamTokenEstimates].reduce((total, value) => total + value, 0);
517
+ const pressureReasons = [
518
+ ...contextPackages.map((projection) => contextPressureReason(projection.payload)).filter((value): value is string => value !== null),
519
+ ...teamRuntimeDecisions.map((projection) => teamPressureReason(projection.payload)).filter((value): value is string => value !== null)
520
+ ];
521
+
522
+ return {
523
+ health: projections.length === 0 ? 'unknown' : pressureReasons.length > 0 ? 'pressure' : 'nominal',
524
+ estimatedTokens: projections.length === 0 ? null : estimatedTokens,
525
+ contextPackages: contextPackages.length,
526
+ teamRuntimeDecisions: teamRuntimeDecisions.length,
527
+ pressureReasons
528
+ };
529
+ }
530
+
531
+ function tokenEstimateFromContextBuildPayload(payload: unknown): number | null {
532
+ if (!payload || typeof payload !== 'object') {
533
+ return null;
534
+ }
535
+ const budget = (payload as { budget?: { estimatedTokens?: unknown } }).budget;
536
+ return typeof budget?.estimatedTokens === 'number' ? budget.estimatedTokens : null;
537
+ }
538
+
539
+ function tokenEstimateFromTeamRuntimePayload(payload: unknown): number | null {
540
+ if (!payload || typeof payload !== 'object') {
541
+ return null;
542
+ }
543
+ const decision = payload as { roleIds?: unknown; telemetryPolicy?: { contextBudget?: unknown } | null };
544
+ const roleCount = Array.isArray(decision.roleIds) ? decision.roleIds.length : 0;
545
+ const contextBudget = decision.telemetryPolicy?.contextBudget;
546
+ const perRole = contextBudget === 'medium' ? 3000 : contextBudget === 'small' ? 1200 : 600;
547
+ return roleCount * perRole;
548
+ }
549
+
550
+ function contextPressureReason(payload: unknown): string | null {
551
+ if (!payload || typeof payload !== 'object') {
552
+ return null;
553
+ }
554
+ const contextPackage = payload as { taskId?: unknown; profile?: unknown; budget?: { estimatedBytes?: unknown; maxBytes?: unknown } };
555
+ const estimatedBytes = contextPackage.budget?.estimatedBytes;
556
+ const maxBytes = contextPackage.budget?.maxBytes;
557
+ if (contextPackage.profile === 'brief') {
558
+ return null;
559
+ }
560
+ if (typeof estimatedBytes === 'number' && typeof maxBytes === 'number' && estimatedBytes >= maxBytes * 0.85) {
561
+ return `context_budget_pressure:${String(contextPackage.taskId ?? 'unknown')}`;
562
+ }
563
+ return null;
564
+ }
565
+
566
+ function teamPressureReason(payload: unknown): string | null {
567
+ if (!payload || typeof payload !== 'object') {
568
+ return null;
569
+ }
570
+ const decision = payload as { command?: unknown; mode?: unknown; roleIds?: unknown };
571
+ const roleCount = Array.isArray(decision.roleIds) ? decision.roleIds.length : 0;
572
+ if (decision.mode === 'team-required' || roleCount > 2) {
573
+ return `team_runtime_pressure:${String(decision.command ?? 'unknown')}`;
574
+ }
575
+ return null;
576
+ }
577
+
578
+ async function inspectRunEvidenceSummary(projectRoot: string, runId: string): Promise<RunEvidenceSummary> {
579
+ const [state, events, agentExecutions, teamSessions, workerRuntimeList] = await Promise.all([
580
+ readRunState(projectRoot, runId),
581
+ readRunEvents(projectRoot, runId),
582
+ listAgentExecutionRecords(projectRoot, runId),
583
+ listTeamSessionRecords(projectRoot, runId),
584
+ listResidentWorkerRuntimes(projectRoot, { runId })
585
+ ]);
586
+ return {
587
+ agentExecutions: agentExecutions.length,
588
+ teamSessions: teamSessions.length,
589
+ artifactIngestions: Object.keys(state.artifactIngestions ?? {}).length,
590
+ workerRuntimes: workerRuntimeList.runtimes.length,
591
+ staleWorkerRuntimes: workerRuntimeList.staleRuntimes,
592
+ routePreflight: events.some((event) => event.event === 'agent_router_preflight'),
593
+ tasksChangedAfterRun: false,
594
+ tasksUpdatedAt: null,
595
+ runUpdatedAt: state.updatedAt ?? null
596
+ };
597
+ }
598
+
599
+ async function addTaskDocumentStalenessToRunEvidence(tasksPath: string, latestRun: RunSummary, evidence: RunEvidenceSummary): Promise<RunEvidenceSummary> {
600
+ try {
601
+ const tasksStat = await stat(tasksPath);
602
+ const runUpdatedAtMs = Date.parse(latestRun.updatedAt);
603
+ if (Number.isNaN(runUpdatedAtMs)) {
604
+ return { ...evidence, runUpdatedAt: latestRun.updatedAt };
605
+ }
606
+ return {
607
+ ...evidence,
608
+ tasksChangedAfterRun: tasksStat.mtimeMs > runUpdatedAtMs,
609
+ tasksUpdatedAt: tasksStat.mtime.toISOString(),
610
+ runUpdatedAt: latestRun.updatedAt
611
+ };
612
+ } catch {
613
+ return { ...evidence, runUpdatedAt: latestRun.updatedAt };
614
+ }
615
+ }
616
+
617
+ async function getGitRoot(projectRoot: string): Promise<string | null> {
618
+ try {
619
+ const result = await execFileAsync('git', ['-C', projectRoot, 'rev-parse', '--show-toplevel']);
620
+ return result.stdout.trim();
621
+ } catch {
622
+ return null;
623
+ }
624
+ }
625
+
626
+ async function runDocumentStaleReasons(_projectRoot: string, state: RunState, model: SddTaskModel): Promise<string[]> {
627
+ const reasons: string[] = [];
628
+ const snapshot = state.documentSnapshot;
629
+ if (!snapshot.specHash && !snapshot.planHash && !snapshot.tasksHash) {
630
+ reasons.push('Run has no document snapshot hashes; rerun execute before validation.');
631
+ }
632
+ appendDocumentHashMismatch(reasons, 'spec.md', snapshot.specHash, model.documents.specHash ?? null);
633
+ appendDocumentHashMismatch(reasons, 'plan.md', snapshot.planHash, model.documents.planHash ?? null);
634
+ appendDocumentHashMismatch(reasons, 'tasks.md', snapshot.tasksHash, model.documents.tasksHash ?? null);
635
+ if (model.documents.planStale) {
636
+ reasons.push('Current plan.md is stale against spec.md.');
637
+ }
638
+ if (model.documents.tasksStale) {
639
+ reasons.push('Current tasks.md is stale against plan.md or spec.md.');
640
+ }
641
+ return reasons;
642
+ }
643
+
644
+ function appendDocumentHashMismatch(reasons: string[], documentName: string, expected: string | null, actual: string | null): void {
645
+ if (!expected && !actual) {
646
+ return;
647
+ }
648
+ if (!expected || !actual || expected.replace(/^sha256:/, '') !== actual) {
649
+ reasons.push(`Run snapshot for ${documentName} is ${expected ?? 'missing'}, current hash is ${actual ?? 'missing'}.`);
650
+ }
651
+ }