@opengsd/gsd-pi 1.1.1-dev.b2556262 → 1.2.0-dev.4813ead6

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 (500) hide show
  1. package/dist/cli-web-branch.d.ts +2 -0
  2. package/dist/cli-web-branch.js +9 -2
  3. package/dist/help-text.js +5 -0
  4. package/dist/project-sessions.js +4 -2
  5. package/dist/resources/.managed-resources-content-hash +1 -1
  6. package/dist/resources/extensions/ask-user-questions.js +78 -23
  7. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +101 -237
  8. package/dist/resources/extensions/claude-code-cli/turn-assembler.js +224 -0
  9. package/dist/resources/extensions/github-sync/templates.js +3 -3
  10. package/dist/resources/extensions/gsd/artifact-projection.js +14 -0
  11. package/dist/resources/extensions/gsd/auto/contracts.js +8 -1
  12. package/dist/resources/extensions/gsd/auto/loop.js +74 -56
  13. package/dist/resources/extensions/gsd/auto/orchestrator.js +763 -63
  14. package/dist/resources/extensions/gsd/auto/phases.js +28 -3
  15. package/dist/resources/extensions/gsd/auto/run-unit.js +2 -1
  16. package/dist/resources/extensions/gsd/auto/session.js +3 -0
  17. package/dist/resources/extensions/gsd/auto-dashboard.js +16 -4
  18. package/dist/resources/extensions/gsd/auto-dispatch.js +6 -5
  19. package/dist/resources/extensions/gsd/auto-model-selection.js +8 -0
  20. package/dist/resources/extensions/gsd/auto-post-unit.js +4 -3
  21. package/dist/resources/extensions/gsd/auto-prompts.js +191 -9
  22. package/dist/resources/extensions/gsd/auto-recovery.js +48 -49
  23. package/dist/resources/extensions/gsd/auto-runtime-state.js +17 -0
  24. package/dist/resources/extensions/gsd/auto-start.js +12 -23
  25. package/dist/resources/extensions/gsd/auto-timers.js +16 -2
  26. package/dist/resources/extensions/gsd/auto-tool-tracking.js +37 -0
  27. package/dist/resources/extensions/gsd/auto-unit-tool-scope.js +33 -29
  28. package/dist/resources/extensions/gsd/auto-verification.js +7 -7
  29. package/dist/resources/extensions/gsd/auto-worktree.js +45 -36
  30. package/dist/resources/extensions/gsd/auto.js +73 -471
  31. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +28 -37
  32. package/dist/resources/extensions/gsd/bootstrap/dynamic-tools.js +11 -37
  33. package/dist/resources/extensions/gsd/bootstrap/query-tools.js +2 -2
  34. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +103 -138
  35. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +63 -4
  36. package/dist/resources/extensions/gsd/closeout-consistency-gate.js +21 -4
  37. package/dist/resources/extensions/gsd/codebase-generator.js +8 -4
  38. package/dist/resources/extensions/gsd/commands/handlers/auto.js +3 -0
  39. package/dist/resources/extensions/gsd/commands-handlers.js +20 -0
  40. package/dist/resources/extensions/gsd/commands-inspect.js +4 -8
  41. package/dist/resources/extensions/gsd/commands-maintenance.js +61 -41
  42. package/dist/resources/extensions/gsd/commands-ship.js +2 -2
  43. package/dist/resources/extensions/gsd/commands-verdict.js +12 -2
  44. package/dist/resources/extensions/gsd/db-workspace.js +103 -0
  45. package/dist/resources/extensions/gsd/debug-logger.js +10 -0
  46. package/dist/resources/extensions/gsd/delegation-policy.js +2 -10
  47. package/dist/resources/extensions/gsd/discussion-handoff.js +218 -0
  48. package/dist/resources/extensions/gsd/docs/preferences-reference.md +9 -0
  49. package/dist/resources/extensions/gsd/doctor-proactive.js +7 -2
  50. package/dist/resources/extensions/gsd/doctor.js +16 -9
  51. package/dist/resources/extensions/gsd/error-classifier.js +1 -1
  52. package/dist/resources/extensions/gsd/git-conflict-state.js +16 -1
  53. package/dist/resources/extensions/gsd/gsd-db.js +12 -0
  54. package/dist/resources/extensions/gsd/guided-flow.js +36 -470
  55. package/dist/resources/extensions/gsd/guided-unit-completion.js +225 -0
  56. package/dist/resources/extensions/gsd/markdown-renderer.js +33 -33
  57. package/dist/resources/extensions/gsd/mcp-filter.js +8 -1
  58. package/dist/resources/extensions/gsd/mcp-tool-name.js +26 -0
  59. package/dist/resources/extensions/gsd/md-importer.js +4 -3
  60. package/dist/resources/extensions/gsd/migrate/safety.js +2 -2
  61. package/dist/resources/extensions/gsd/migration-auto-check.js +3 -2
  62. package/dist/resources/extensions/gsd/milestone-closeout-proof.js +72 -0
  63. package/dist/resources/extensions/gsd/milestone-closeout.js +12 -4
  64. package/dist/resources/extensions/gsd/milestone-merge-transaction.js +10 -0
  65. package/dist/resources/extensions/gsd/milestone-planning-persistence.js +156 -0
  66. package/dist/resources/extensions/gsd/milestone-readiness.js +77 -0
  67. package/dist/resources/extensions/gsd/milestone-settlement.js +50 -0
  68. package/dist/resources/extensions/gsd/milestone-validation-evidence.js +73 -0
  69. package/dist/resources/extensions/gsd/milestone-validation-verdict.js +57 -0
  70. package/dist/resources/extensions/gsd/native-git-bridge.js +45 -0
  71. package/dist/resources/extensions/gsd/parallel-eligibility.js +3 -6
  72. package/dist/resources/extensions/gsd/parallel-orchestrator.js +3 -2
  73. package/dist/resources/extensions/gsd/preferences-diagnostics.js +67 -0
  74. package/dist/resources/extensions/gsd/preferences.js +147 -29
  75. package/dist/resources/extensions/gsd/prompts/complete-slice.md +1 -0
  76. package/dist/resources/extensions/gsd/prompts/discuss.md +6 -7
  77. package/dist/resources/extensions/gsd/prompts/execute-task.md +2 -0
  78. package/dist/resources/extensions/gsd/prompts/guided-discuss-milestone.md +5 -7
  79. package/dist/resources/extensions/gsd/prompts/guided-discuss-project.md +6 -6
  80. package/dist/resources/extensions/gsd/prompts/guided-discuss-requirements.md +1 -2
  81. package/dist/resources/extensions/gsd/prompts/guided-discuss-slice.md +5 -6
  82. package/dist/resources/extensions/gsd/prompts/plan-milestone.md +2 -0
  83. package/dist/resources/extensions/gsd/prompts/plan-slice.md +2 -1
  84. package/dist/resources/extensions/gsd/prompts/refine-slice.md +1 -0
  85. package/dist/resources/extensions/gsd/prompts/research-milestone.md +2 -2
  86. package/dist/resources/extensions/gsd/prompts/system.md +1 -1
  87. package/dist/resources/extensions/gsd/prompts/validate-milestone.md +5 -3
  88. package/dist/resources/extensions/gsd/provider-payload-policy.js +83 -0
  89. package/dist/resources/extensions/gsd/pull-request-process.js +13 -0
  90. package/dist/resources/extensions/gsd/quality-gate-closure.js +109 -0
  91. package/dist/resources/extensions/gsd/question-transport.js +86 -0
  92. package/dist/resources/extensions/gsd/roadmap-slices.js +8 -2
  93. package/dist/resources/extensions/gsd/schemas/parsers.js +6 -1
  94. package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +3 -2
  95. package/dist/resources/extensions/gsd/state-reconciliation/drift/artifact-db.js +21 -1
  96. package/dist/resources/extensions/gsd/state.js +13 -5
  97. package/dist/resources/extensions/gsd/templates/plan.md +7 -0
  98. package/dist/resources/extensions/gsd/templates/project.md +1 -0
  99. package/dist/resources/extensions/gsd/templates/roadmap.md +1 -1
  100. package/dist/resources/extensions/gsd/templates/uat.md +5 -1
  101. package/dist/resources/extensions/gsd/tool-contract.js +52 -8
  102. package/dist/resources/extensions/gsd/tool-presentation-plan.js +15 -34
  103. package/dist/resources/extensions/gsd/tool-surface-snapshot.js +17 -0
  104. package/dist/resources/extensions/gsd/tools/plan-milestone.js +15 -143
  105. package/dist/resources/extensions/gsd/tools/reassess-roadmap.js +39 -0
  106. package/dist/resources/extensions/gsd/tools/validate-milestone.js +15 -78
  107. package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +169 -20
  108. package/dist/resources/extensions/gsd/uat-policy.js +16 -10
  109. package/dist/resources/extensions/gsd/uat-run.js +9 -14
  110. package/dist/resources/extensions/gsd/unit-context-composer.js +40 -20
  111. package/dist/resources/extensions/gsd/unit-runtime.js +3 -2
  112. package/dist/resources/extensions/gsd/unit-tool-contracts.js +2 -1
  113. package/dist/resources/extensions/gsd/user-input-boundary.js +65 -4
  114. package/dist/resources/extensions/gsd/validation-block-guard.js +2 -0
  115. package/dist/resources/extensions/gsd/web-app-uat.js +80 -0
  116. package/dist/resources/extensions/gsd/workflow-mcp.js +15 -102
  117. package/dist/resources/extensions/gsd/workflow-reconcile.js +4 -3
  118. package/dist/resources/extensions/gsd/workflow-tool-surface.js +46 -0
  119. package/dist/resources/extensions/gsd/workspace-git-guard.js +2 -0
  120. package/dist/resources/extensions/gsd/worktree-state-projection.js +33 -4
  121. package/dist/resources/extensions/gsd/worktree-telemetry.js +12 -0
  122. package/dist/resources/extensions/shared/interview-ui.js +2 -2
  123. package/dist/resources/shared/claude-runtime-floor.js +182 -0
  124. package/dist/tsconfig.extensions.tsbuildinfo +1 -0
  125. package/dist/update-cmd.js +20 -0
  126. package/dist/web/standalone/.next/BUILD_ID +1 -1
  127. package/dist/web/standalone/.next/app-path-routes-manifest.json +5 -5
  128. package/dist/web/standalone/.next/build-manifest.json +3 -3
  129. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  130. package/dist/web/standalone/.next/react-loadable-manifest.json +8 -8
  131. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  132. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  133. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  134. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  135. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  136. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  137. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  138. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  139. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  140. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  141. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  142. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  143. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  144. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  145. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  146. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  147. package/dist/web/standalone/.next/server/app/api/boot/route.js.nft.json +1 -1
  148. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js.nft.json +1 -1
  149. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js.nft.json +1 -1
  150. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js.nft.json +1 -1
  151. package/dist/web/standalone/.next/server/app/api/captures/route.js.nft.json +1 -1
  152. package/dist/web/standalone/.next/server/app/api/cleanup/route.js.nft.json +1 -1
  153. package/dist/web/standalone/.next/server/app/api/doctor/route.js.nft.json +1 -1
  154. package/dist/web/standalone/.next/server/app/api/export-data/route.js.nft.json +1 -1
  155. package/dist/web/standalone/.next/server/app/api/files/route.js.nft.json +1 -1
  156. package/dist/web/standalone/.next/server/app/api/forensics/route.js.nft.json +1 -1
  157. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  158. package/dist/web/standalone/.next/server/app/api/git/route.js.nft.json +1 -1
  159. package/dist/web/standalone/.next/server/app/api/history/route.js.nft.json +1 -1
  160. package/dist/web/standalone/.next/server/app/api/hooks/route.js.nft.json +1 -1
  161. package/dist/web/standalone/.next/server/app/api/inspect/route.js.nft.json +1 -1
  162. package/dist/web/standalone/.next/server/app/api/knowledge/route.js.nft.json +1 -1
  163. package/dist/web/standalone/.next/server/app/api/live-state/route.js.nft.json +1 -1
  164. package/dist/web/standalone/.next/server/app/api/mcp-connections/route.js.nft.json +1 -1
  165. package/dist/web/standalone/.next/server/app/api/notifications/route.js.nft.json +1 -1
  166. package/dist/web/standalone/.next/server/app/api/onboarding/route.js.nft.json +1 -1
  167. package/dist/web/standalone/.next/server/app/api/projects/route.js.nft.json +1 -1
  168. package/dist/web/standalone/.next/server/app/api/recovery/route.js.nft.json +1 -1
  169. package/dist/web/standalone/.next/server/app/api/session/browser/route.js.nft.json +1 -1
  170. package/dist/web/standalone/.next/server/app/api/session/command/route.js.nft.json +1 -1
  171. package/dist/web/standalone/.next/server/app/api/session/events/route.js.nft.json +1 -1
  172. package/dist/web/standalone/.next/server/app/api/session/manage/route.js.nft.json +1 -1
  173. package/dist/web/standalone/.next/server/app/api/settings-data/route.js.nft.json +1 -1
  174. package/dist/web/standalone/.next/server/app/api/shutdown/route.js.nft.json +1 -1
  175. package/dist/web/standalone/.next/server/app/api/skill-health/route.js.nft.json +1 -1
  176. package/dist/web/standalone/.next/server/app/api/steer/route.js.nft.json +1 -1
  177. package/dist/web/standalone/.next/server/app/api/switch-root/route.js.nft.json +1 -1
  178. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js.nft.json +1 -1
  179. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js.nft.json +1 -1
  180. package/dist/web/standalone/.next/server/app/api/undo/route.js.nft.json +1 -1
  181. package/dist/web/standalone/.next/server/app/api/visualizer/route.js.nft.json +1 -1
  182. package/dist/web/standalone/.next/server/app/index.html +1 -1
  183. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  184. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  185. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  186. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  187. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  188. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  189. package/dist/web/standalone/.next/server/app-paths-manifest.json +5 -5
  190. package/dist/web/standalone/.next/server/chunks/5047.js +2 -0
  191. package/dist/web/standalone/.next/server/chunks/5124.js +1 -0
  192. package/dist/web/standalone/.next/server/chunks/8357.js +2 -2
  193. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  194. package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
  195. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  196. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  197. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  198. package/dist/web/standalone/.next/static/chunks/2659.b7b129ee6a769448.js +1 -0
  199. package/dist/web/standalone/.next/static/chunks/2772.bfa657f49f955239.js +1 -0
  200. package/dist/web/standalone/.next/static/chunks/{3616.4113d484a994e411.js → 3616.3c60753b8ffcbd2e.js} +1 -1
  201. package/dist/web/standalone/.next/static/chunks/4283.e4873b058df143a1.js +2 -0
  202. package/dist/web/standalone/.next/static/chunks/5826.a46ecdd1cfe8dabc.js +1 -0
  203. package/dist/web/standalone/.next/static/chunks/796.cf859a427a2cb2ac.js +10 -0
  204. package/dist/web/standalone/.next/static/chunks/8785.2e5a118797fb2dd2.js +1 -0
  205. package/dist/web/standalone/.next/static/chunks/{webpack-dda80a1ef5587410.js → webpack-fbea77b5f9953368.js} +1 -1
  206. package/dist/web/standalone/node_modules/@gsd/native/package.json +1 -1
  207. package/dist/web/standalone/node_modules/node-pty/build/Makefile +1 -1
  208. package/dist/web/standalone/node_modules/postcss/lib/container.js +26 -18
  209. package/dist/web/standalone/node_modules/postcss/lib/css-syntax-error.js +47 -14
  210. package/dist/web/standalone/node_modules/postcss/lib/declaration.js +4 -4
  211. package/dist/web/standalone/node_modules/postcss/lib/fromJSON.js +3 -3
  212. package/dist/web/standalone/node_modules/postcss/lib/input.js +54 -29
  213. package/dist/web/standalone/node_modules/postcss/lib/lazy-result.js +47 -37
  214. package/dist/web/standalone/node_modules/postcss/lib/map-generator.js +26 -9
  215. package/dist/web/standalone/node_modules/postcss/lib/no-work-result.js +57 -55
  216. package/dist/web/standalone/node_modules/postcss/lib/node.js +99 -31
  217. package/dist/web/standalone/node_modules/postcss/lib/parse.js +1 -1
  218. package/dist/web/standalone/node_modules/postcss/lib/parser.js +10 -9
  219. package/dist/web/standalone/node_modules/postcss/lib/postcss.js +12 -12
  220. package/dist/web/standalone/node_modules/postcss/lib/previous-map.js +30 -11
  221. package/dist/web/standalone/node_modules/postcss/lib/processor.js +7 -7
  222. package/dist/web/standalone/node_modules/postcss/lib/result.js +5 -5
  223. package/dist/web/standalone/node_modules/postcss/lib/rule.js +6 -6
  224. package/dist/web/standalone/node_modules/postcss/lib/stringifier.js +69 -28
  225. package/dist/web/standalone/node_modules/postcss/lib/tokenize.js +6 -2
  226. package/dist/web/standalone/node_modules/postcss/package.json +48 -48
  227. package/dist/web-mode.d.ts +2 -0
  228. package/dist/web-mode.js +20 -8
  229. package/package.json +17 -11
  230. package/packages/cloud-mcp-gateway/package.json +2 -2
  231. package/packages/contracts/package.json +1 -1
  232. package/packages/daemon/package.json +4 -4
  233. package/packages/gsd-agent-core/dist/session/agent-session-extensions.d.ts +2 -0
  234. package/packages/gsd-agent-core/dist/session/agent-session-extensions.d.ts.map +1 -1
  235. package/packages/gsd-agent-core/dist/session/agent-session-extensions.js +14 -0
  236. package/packages/gsd-agent-core/dist/session/agent-session-extensions.js.map +1 -1
  237. package/packages/gsd-agent-core/package.json +5 -5
  238. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  239. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.js +3 -0
  240. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.js.map +1 -1
  241. package/packages/gsd-agent-modes/dist/modes/interactive/components/transcript-design.js +1 -1
  242. package/packages/gsd-agent-modes/dist/modes/interactive/components/transcript-design.js.map +1 -1
  243. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  244. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js +106 -40
  245. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  246. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-extension-widgets.d.ts.map +1 -1
  247. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-extension-widgets.js +6 -0
  248. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-extension-widgets.js.map +1 -1
  249. package/packages/gsd-agent-modes/package.json +7 -7
  250. package/packages/mcp-server/dist/server.d.ts +10 -0
  251. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  252. package/packages/mcp-server/dist/server.js +8 -0
  253. package/packages/mcp-server/dist/server.js.map +1 -1
  254. package/packages/mcp-server/dist/workflow-tools.d.ts +41 -0
  255. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  256. package/packages/mcp-server/dist/workflow-tools.js +2 -1
  257. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  258. package/packages/mcp-server/package.json +3 -3
  259. package/packages/native/package.json +1 -1
  260. package/packages/pi-agent-core/package.json +1 -1
  261. package/packages/pi-ai/dist/image-models.generated.d.ts +30 -0
  262. package/packages/pi-ai/dist/image-models.generated.d.ts.map +1 -1
  263. package/packages/pi-ai/dist/image-models.generated.js +30 -0
  264. package/packages/pi-ai/dist/image-models.generated.js.map +1 -1
  265. package/packages/pi-ai/dist/models.generated.d.ts +8 -127
  266. package/packages/pi-ai/dist/models.generated.d.ts.map +1 -1
  267. package/packages/pi-ai/dist/models.generated.js +47 -166
  268. package/packages/pi-ai/dist/models.generated.js.map +1 -1
  269. package/packages/pi-ai/package.json +1 -1
  270. package/packages/pi-coding-agent/dist/core/auth-storage.d.ts.map +1 -1
  271. package/packages/pi-coding-agent/dist/core/auth-storage.js +11 -3
  272. package/packages/pi-coding-agent/dist/core/auth-storage.js.map +1 -1
  273. package/packages/pi-coding-agent/package.json +7 -7
  274. package/packages/pi-tui/dist/components/input.js +1 -1
  275. package/packages/pi-tui/dist/components/input.js.map +1 -1
  276. package/packages/pi-tui/dist/keys.d.ts.map +1 -1
  277. package/packages/pi-tui/dist/keys.js +39 -30
  278. package/packages/pi-tui/dist/keys.js.map +1 -1
  279. package/packages/pi-tui/dist/stdin-buffer.d.ts.map +1 -1
  280. package/packages/pi-tui/dist/stdin-buffer.js +22 -0
  281. package/packages/pi-tui/dist/stdin-buffer.js.map +1 -1
  282. package/packages/pi-tui/package.json +2 -2
  283. package/packages/rpc-client/package.json +2 -2
  284. package/pkg/package.json +1 -1
  285. package/scripts/install/deps.js +10 -0
  286. package/scripts/link-workspace-packages.cjs +7 -40
  287. package/src/resources/extensions/ask-user-questions.ts +87 -24
  288. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +126 -289
  289. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +242 -2
  290. package/src/resources/extensions/claude-code-cli/turn-assembler.ts +287 -0
  291. package/src/resources/extensions/github-sync/templates.ts +3 -3
  292. package/src/resources/extensions/github-sync/tests/templates.test.ts +2 -2
  293. package/src/resources/extensions/gsd/artifact-projection.ts +31 -0
  294. package/src/resources/extensions/gsd/auto/contracts.ts +40 -121
  295. package/src/resources/extensions/gsd/auto/loop-deps.ts +2 -0
  296. package/src/resources/extensions/gsd/auto/loop.ts +83 -61
  297. package/src/resources/extensions/gsd/auto/orchestrator.ts +913 -64
  298. package/src/resources/extensions/gsd/auto/phases.ts +35 -3
  299. package/src/resources/extensions/gsd/auto/run-unit.ts +2 -1
  300. package/src/resources/extensions/gsd/auto/session.ts +4 -0
  301. package/src/resources/extensions/gsd/auto-dashboard.ts +18 -4
  302. package/src/resources/extensions/gsd/auto-dispatch.ts +20 -7
  303. package/src/resources/extensions/gsd/auto-model-selection.ts +8 -0
  304. package/src/resources/extensions/gsd/auto-post-unit.ts +4 -3
  305. package/src/resources/extensions/gsd/auto-prompts.ts +220 -9
  306. package/src/resources/extensions/gsd/auto-recovery.ts +50 -50
  307. package/src/resources/extensions/gsd/auto-runtime-state.ts +30 -0
  308. package/src/resources/extensions/gsd/auto-start.ts +17 -20
  309. package/src/resources/extensions/gsd/auto-timers.ts +16 -2
  310. package/src/resources/extensions/gsd/auto-tool-tracking.ts +40 -0
  311. package/src/resources/extensions/gsd/auto-unit-tool-scope.ts +42 -30
  312. package/src/resources/extensions/gsd/auto-verification.ts +7 -8
  313. package/src/resources/extensions/gsd/auto-worktree.ts +57 -42
  314. package/src/resources/extensions/gsd/auto.ts +96 -508
  315. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +29 -37
  316. package/src/resources/extensions/gsd/bootstrap/dynamic-tools.ts +10 -37
  317. package/src/resources/extensions/gsd/bootstrap/query-tools.ts +2 -2
  318. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +120 -151
  319. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +107 -3
  320. package/src/resources/extensions/gsd/closeout-consistency-gate.ts +27 -5
  321. package/src/resources/extensions/gsd/codebase-generator.ts +9 -5
  322. package/src/resources/extensions/gsd/commands/handlers/auto.ts +3 -0
  323. package/src/resources/extensions/gsd/commands-handlers.ts +18 -0
  324. package/src/resources/extensions/gsd/commands-inspect.ts +7 -8
  325. package/src/resources/extensions/gsd/commands-maintenance.ts +74 -40
  326. package/src/resources/extensions/gsd/commands-ship.ts +2 -2
  327. package/src/resources/extensions/gsd/commands-verdict.ts +19 -2
  328. package/src/resources/extensions/gsd/db-workspace.ts +170 -0
  329. package/src/resources/extensions/gsd/debug-logger.ts +11 -0
  330. package/src/resources/extensions/gsd/delegation-policy.ts +3 -11
  331. package/src/resources/extensions/gsd/discussion-handoff.ts +276 -0
  332. package/src/resources/extensions/gsd/docs/preferences-reference.md +9 -0
  333. package/src/resources/extensions/gsd/doctor-proactive.ts +8 -2
  334. package/src/resources/extensions/gsd/doctor.ts +15 -5
  335. package/src/resources/extensions/gsd/error-classifier.ts +1 -1
  336. package/src/resources/extensions/gsd/git-conflict-state.ts +17 -1
  337. package/src/resources/extensions/gsd/gsd-db.ts +12 -0
  338. package/src/resources/extensions/gsd/guided-flow.ts +49 -560
  339. package/src/resources/extensions/gsd/guided-unit-completion.ts +275 -0
  340. package/src/resources/extensions/gsd/markdown-renderer.ts +40 -20
  341. package/src/resources/extensions/gsd/mcp-filter.ts +9 -1
  342. package/src/resources/extensions/gsd/mcp-tool-name.ts +35 -0
  343. package/src/resources/extensions/gsd/md-importer.ts +3 -3
  344. package/src/resources/extensions/gsd/migrate/safety.ts +2 -2
  345. package/src/resources/extensions/gsd/migration-auto-check.ts +2 -2
  346. package/src/resources/extensions/gsd/milestone-closeout-proof.ts +131 -0
  347. package/src/resources/extensions/gsd/milestone-closeout.ts +12 -4
  348. package/src/resources/extensions/gsd/milestone-merge-transaction.ts +47 -0
  349. package/src/resources/extensions/gsd/milestone-planning-persistence.ts +224 -0
  350. package/src/resources/extensions/gsd/milestone-readiness.ts +125 -0
  351. package/src/resources/extensions/gsd/milestone-settlement.ts +81 -0
  352. package/src/resources/extensions/gsd/milestone-validation-evidence.ts +95 -0
  353. package/src/resources/extensions/gsd/milestone-validation-verdict.ts +80 -0
  354. package/src/resources/extensions/gsd/native-git-bridge.ts +48 -0
  355. package/src/resources/extensions/gsd/parallel-eligibility.ts +4 -5
  356. package/src/resources/extensions/gsd/parallel-orchestrator.ts +6 -2
  357. package/src/resources/extensions/gsd/preferences-diagnostics.ts +98 -0
  358. package/src/resources/extensions/gsd/preferences-types.ts +16 -0
  359. package/src/resources/extensions/gsd/preferences.ts +173 -28
  360. package/src/resources/extensions/gsd/prompts/complete-slice.md +1 -0
  361. package/src/resources/extensions/gsd/prompts/discuss.md +6 -7
  362. package/src/resources/extensions/gsd/prompts/execute-task.md +2 -0
  363. package/src/resources/extensions/gsd/prompts/guided-discuss-milestone.md +5 -7
  364. package/src/resources/extensions/gsd/prompts/guided-discuss-project.md +6 -6
  365. package/src/resources/extensions/gsd/prompts/guided-discuss-requirements.md +1 -2
  366. package/src/resources/extensions/gsd/prompts/guided-discuss-slice.md +5 -6
  367. package/src/resources/extensions/gsd/prompts/plan-milestone.md +2 -0
  368. package/src/resources/extensions/gsd/prompts/plan-slice.md +2 -1
  369. package/src/resources/extensions/gsd/prompts/refine-slice.md +1 -0
  370. package/src/resources/extensions/gsd/prompts/research-milestone.md +2 -2
  371. package/src/resources/extensions/gsd/prompts/system.md +1 -1
  372. package/src/resources/extensions/gsd/prompts/validate-milestone.md +5 -3
  373. package/src/resources/extensions/gsd/provider-payload-policy.ts +140 -0
  374. package/src/resources/extensions/gsd/pull-request-process.ts +41 -0
  375. package/src/resources/extensions/gsd/quality-gate-closure.ts +140 -0
  376. package/src/resources/extensions/gsd/question-transport.ts +138 -0
  377. package/src/resources/extensions/gsd/roadmap-slices.ts +8 -2
  378. package/src/resources/extensions/gsd/schemas/parsers.ts +6 -1
  379. package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +6 -2
  380. package/src/resources/extensions/gsd/state-reconciliation/drift/artifact-db.ts +31 -10
  381. package/src/resources/extensions/gsd/state.ts +15 -5
  382. package/src/resources/extensions/gsd/templates/plan.md +7 -0
  383. package/src/resources/extensions/gsd/templates/project.md +1 -0
  384. package/src/resources/extensions/gsd/templates/roadmap.md +1 -1
  385. package/src/resources/extensions/gsd/templates/uat.md +5 -1
  386. package/src/resources/extensions/gsd/tests/artifact-db-drift-memo.test.ts +66 -0
  387. package/src/resources/extensions/gsd/tests/ask-user-questions-render.test.ts +92 -0
  388. package/src/resources/extensions/gsd/tests/auto-dashboard.test.ts +29 -1
  389. package/src/resources/extensions/gsd/tests/auto-dispatch-baseline-harness.test.ts +53 -0
  390. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +321 -5
  391. package/src/resources/extensions/gsd/tests/auto-milestone-target.test.ts +23 -0
  392. package/src/resources/extensions/gsd/tests/auto-model-selection-tool-poisoning.test.ts +18 -0
  393. package/src/resources/extensions/gsd/tests/auto-orchestrator.test.ts +709 -845
  394. package/src/resources/extensions/gsd/tests/auto-paused-ui-cleanup.test.ts +38 -10
  395. package/src/resources/extensions/gsd/tests/auto-runtime-state.test.ts +34 -0
  396. package/src/resources/extensions/gsd/tests/canonical-milestone-root.test.ts +20 -0
  397. package/src/resources/extensions/gsd/tests/codebase-generator.test.ts +22 -0
  398. package/src/resources/extensions/gsd/tests/commands-dispatcher-workspace-git.test.ts +11 -0
  399. package/src/resources/extensions/gsd/tests/commands-verdict.test.ts +38 -1
  400. package/src/resources/extensions/gsd/tests/debug-logger.test.ts +15 -0
  401. package/src/resources/extensions/gsd/tests/dispatch-complete-milestone-guard.test.ts +34 -3
  402. package/src/resources/extensions/gsd/tests/dispatch-run-uat-browser-tools.test.ts +88 -0
  403. package/src/resources/extensions/gsd/tests/doctor-scope-db-unavailable.test.ts +18 -0
  404. package/src/resources/extensions/gsd/tests/execute-summary-save-empty-project.test.ts +64 -1
  405. package/src/resources/extensions/gsd/tests/execute-task-rendering.test.ts +1 -0
  406. package/src/resources/extensions/gsd/tests/fixtures/pr-body/swarm-lane-no-blockers.md +1 -5
  407. package/src/resources/extensions/gsd/tests/fixtures/pr-body/swarm-lane-with-blockers.md +1 -5
  408. package/src/resources/extensions/gsd/tests/gate-state-canonicalization.test.ts +48 -1
  409. package/src/resources/extensions/gsd/tests/integration/merge-strategy-regular.test.ts +157 -0
  410. package/src/resources/extensions/gsd/tests/markdown-renderer-parse-cache.test.ts +75 -0
  411. package/src/resources/extensions/gsd/tests/mcp-tool-name.test.ts +34 -0
  412. package/src/resources/extensions/gsd/tests/migration-auto-check.test.ts +58 -0
  413. package/src/resources/extensions/gsd/tests/milestone-closeout-proof.test.ts +99 -0
  414. package/src/resources/extensions/gsd/tests/milestone-closeout.test.ts +25 -0
  415. package/src/resources/extensions/gsd/tests/milestone-merge-transaction.test.ts +46 -0
  416. package/src/resources/extensions/gsd/tests/milestone-readiness.test.ts +65 -0
  417. package/src/resources/extensions/gsd/tests/milestone-validation-evidence.test.ts +41 -0
  418. package/src/resources/extensions/gsd/tests/milestone-validation-verdict.test.ts +55 -0
  419. package/src/resources/extensions/gsd/tests/native-merge-regular.test.ts +139 -0
  420. package/src/resources/extensions/gsd/tests/orchestrator-legacy-parity.test.ts +127 -0
  421. package/src/resources/extensions/gsd/tests/parse-project-milestone-bridge.test.ts +77 -0
  422. package/src/resources/extensions/gsd/tests/plan-milestone.test.ts +45 -0
  423. package/src/resources/extensions/gsd/tests/plan-slice-prompt.test.ts +6 -2
  424. package/src/resources/extensions/gsd/tests/planning-crossval.test.ts +45 -0
  425. package/src/resources/extensions/gsd/tests/preferences-diagnostics.test.ts +67 -0
  426. package/src/resources/extensions/gsd/tests/preferences.test.ts +183 -0
  427. package/src/resources/extensions/gsd/tests/prompt-contracts.test.ts +75 -2
  428. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +9 -0
  429. package/src/resources/extensions/gsd/tests/provider-payload-policy.test.ts +165 -0
  430. package/src/resources/extensions/gsd/tests/pull-request-process.test.ts +47 -0
  431. package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +94 -0
  432. package/src/resources/extensions/gsd/tests/research-milestone-composer.test.ts +65 -0
  433. package/src/resources/extensions/gsd/tests/roadmap-parse-regression.test.ts +40 -0
  434. package/src/resources/extensions/gsd/tests/runtime-invariant-modules.test.ts +25 -1
  435. package/src/resources/extensions/gsd/tests/session-start-footer.test.ts +80 -0
  436. package/src/resources/extensions/gsd/tests/single-writer-invariant.test.ts +101 -1
  437. package/src/resources/extensions/gsd/tests/stale-queued-milestone.test.ts +27 -0
  438. package/src/resources/extensions/gsd/tests/start-auto-detached.test.ts +21 -6
  439. package/src/resources/extensions/gsd/tests/token-tool-gating.test.ts +38 -0
  440. package/src/resources/extensions/gsd/tests/tool-availability-audit.test.ts +35 -0
  441. package/src/resources/extensions/gsd/tests/tool-naming.test.ts +35 -42
  442. package/src/resources/extensions/gsd/tests/uat-policy.test.ts +23 -0
  443. package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +47 -0
  444. package/src/resources/extensions/gsd/tests/user-input-boundary.test.ts +147 -0
  445. package/src/resources/extensions/gsd/tests/validate-milestone-stuck-guard.test.ts +39 -0
  446. package/src/resources/extensions/gsd/tests/web-app-uat.test.ts +150 -0
  447. package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +126 -9
  448. package/src/resources/extensions/gsd/tests/workspace-git-preflight.test.ts +15 -0
  449. package/src/resources/extensions/gsd/tests/worktree-manager.test.ts +21 -0
  450. package/src/resources/extensions/gsd/tests/worktree-projection-writers.test.ts +1 -1
  451. package/src/resources/extensions/gsd/tests/worktree-safety.test.ts +24 -0
  452. package/src/resources/extensions/gsd/tests/worktree-telemetry.test.ts +22 -0
  453. package/src/resources/extensions/gsd/tests/write-gate-planning-unit.test.ts +15 -3
  454. package/src/resources/extensions/gsd/tests/write-gate.test.ts +79 -0
  455. package/src/resources/extensions/gsd/tool-contract.ts +86 -8
  456. package/src/resources/extensions/gsd/tool-presentation-plan.ts +16 -33
  457. package/src/resources/extensions/gsd/tool-surface-snapshot.ts +47 -0
  458. package/src/resources/extensions/gsd/tools/plan-milestone.ts +19 -160
  459. package/src/resources/extensions/gsd/tools/reassess-roadmap.ts +43 -0
  460. package/src/resources/extensions/gsd/tools/validate-milestone.ts +25 -84
  461. package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +183 -21
  462. package/src/resources/extensions/gsd/uat-policy.ts +19 -10
  463. package/src/resources/extensions/gsd/uat-run.ts +10 -14
  464. package/src/resources/extensions/gsd/unit-context-composer.ts +85 -20
  465. package/src/resources/extensions/gsd/unit-runtime.ts +3 -2
  466. package/src/resources/extensions/gsd/unit-tool-contracts.ts +2 -1
  467. package/src/resources/extensions/gsd/user-input-boundary.ts +55 -5
  468. package/src/resources/extensions/gsd/validation-block-guard.ts +2 -0
  469. package/src/resources/extensions/gsd/web-app-uat.ts +101 -0
  470. package/src/resources/extensions/gsd/workflow-mcp.ts +22 -110
  471. package/src/resources/extensions/gsd/workflow-reconcile.ts +3 -3
  472. package/src/resources/extensions/gsd/workflow-tool-surface.ts +73 -0
  473. package/src/resources/extensions/gsd/workspace-git-guard.ts +1 -0
  474. package/src/resources/extensions/gsd/worktree-lifecycle.ts +7 -16
  475. package/src/resources/extensions/gsd/worktree-state-projection.ts +55 -7
  476. package/src/resources/extensions/gsd/worktree-telemetry.ts +16 -0
  477. package/src/resources/extensions/shared/interview-ui.ts +15 -2
  478. package/src/resources/shared/claude-runtime-floor.ts +248 -0
  479. package/dist/web/standalone/.next/server/chunks/678.js +0 -2
  480. package/dist/web/standalone/.next/static/chunks/2659.feb6499ca863ebfc.js +0 -1
  481. package/dist/web/standalone/.next/static/chunks/2772.151789db0edea835.js +0 -1
  482. package/dist/web/standalone/.next/static/chunks/4283.10a065467b5340d8.js +0 -2
  483. package/dist/web/standalone/.next/static/chunks/5826.960dc4634cc9b0d3.js +0 -1
  484. package/dist/web/standalone/.next/static/chunks/796.46f811c0fac23aab.js +0 -10
  485. package/dist/web/standalone/.next/static/chunks/8785.d32f7a61f55c1600.js +0 -1
  486. package/packages/gsd-agent-modes/dist/modes/interactive/components/__prototype__/gsd-widget-prototype.d.ts +0 -21
  487. package/packages/gsd-agent-modes/dist/modes/interactive/components/__prototype__/gsd-widget-prototype.d.ts.map +0 -1
  488. package/packages/gsd-agent-modes/dist/modes/interactive/components/__prototype__/gsd-widget-prototype.js +0 -213
  489. package/packages/gsd-agent-modes/dist/modes/interactive/components/__prototype__/gsd-widget-prototype.js.map +0 -1
  490. package/packages/gsd-agent-modes/dist/modes/interactive/components/__prototype__/transcript-density-prototype.d.ts +0 -28
  491. package/packages/gsd-agent-modes/dist/modes/interactive/components/__prototype__/transcript-density-prototype.d.ts.map +0 -1
  492. package/packages/gsd-agent-modes/dist/modes/interactive/components/__prototype__/transcript-density-prototype.js +0 -249
  493. package/packages/gsd-agent-modes/dist/modes/interactive/components/__prototype__/transcript-density-prototype.js.map +0 -1
  494. package/packages/gsd-agent-modes/dist/modes/interactive/components/__prototype__/transcript-design-prototype.d.ts +0 -19
  495. package/packages/gsd-agent-modes/dist/modes/interactive/components/__prototype__/transcript-design-prototype.d.ts.map +0 -1
  496. package/packages/gsd-agent-modes/dist/modes/interactive/components/__prototype__/transcript-design-prototype.js +0 -797
  497. package/packages/gsd-agent-modes/dist/modes/interactive/components/__prototype__/transcript-design-prototype.js.map +0 -1
  498. package/scripts/ensure-workspace-builds.cjs +0 -129
  499. /package/dist/web/standalone/.next/static/{tJOKQbQRO-9MiFDO8DIDS → tkLHUSzPA2kMmWz4DmGwI}/_buildManifest.js +0 -0
  500. /package/dist/web/standalone/.next/static/{tJOKQbQRO-9MiFDO8DIDS → tkLHUSzPA2kMmWz4DmGwI}/_ssgManifest.js +0 -0
@@ -14,7 +14,8 @@ import { appendEvent } from "./workflow-events.js";
14
14
  import { atomicWriteSync } from "./atomic-write.js";
15
15
  import { clearParseCache } from "./files.js";
16
16
  import { parseRoadmap as parseLegacyRoadmap, parsePlan as parseLegacyPlan } from "./parsers-legacy.js";
17
- import { isDbAvailable, getTask, getSlice, getSliceTasks, getPendingGatesForTurn, updateTaskStatus, updateSliceStatus, insertSlice, getMilestone, getMilestoneSlices, getLatestAssessmentByScope, updateMilestoneStatus, refreshOpenDatabaseFromDisk, transaction, } from "./gsd-db.js";
17
+ import { isDbAvailable, getTask, getSlice, getSliceTasks, getPendingGatesForTurn, updateTaskStatus, updateSliceStatus, insertSlice, getMilestone, updateMilestoneStatus, transaction, } from "./gsd-db.js";
18
+ import { refreshWorkflowDatabaseFromDisk } from "./db-workspace.js";
18
19
  import { isValidationTerminal } from "./state.js";
19
20
  import { getErrorMessage } from "./error-utils.js";
20
21
  import { logWarning, logError } from "./workflow-logger.js";
@@ -32,7 +33,7 @@ import { isGsdWorktreePath } from "./worktree-root.js";
32
33
  import { resolveCanonicalMilestoneRoot } from "./worktree-manager.js";
33
34
  import { hasImplementationArtifacts } from "./milestone-implementation-evidence.js";
34
35
  import { loadAllCaptures, loadPendingCaptures } from "./captures.js";
35
- import { checkCloseoutConsistencyGate } from "./closeout-consistency-gate.js";
36
+ import { proveMilestoneCloseout, } from "./milestone-closeout-proof.js";
36
37
  // Re-export so existing consumers of auto-recovery.ts keep working.
37
38
  export { resolveExpectedArtifactPath, diagnoseExpectedArtifact };
38
39
  export { classifyMilestoneSummaryContent, } from "./milestone-summary-classifier.js";
@@ -66,12 +67,24 @@ function resolveArtifactVerificationBase(unitId, base) {
66
67
  return base;
67
68
  return resolveCanonicalMilestoneRoot(base, milestone);
68
69
  }
70
+ function closeoutProofRecoveryReason(reason) {
71
+ switch (reason) {
72
+ case "slice-missing":
73
+ return "complete-milestone-slices-missing";
74
+ case "summary-artifact-missing":
75
+ return "complete-milestone-summary-missing";
76
+ case "summary-artifact-failed":
77
+ return "complete-milestone-summary-failed";
78
+ default:
79
+ return `complete-milestone-${reason}`;
80
+ }
81
+ }
69
82
  export function refreshRecoveryDbForArtifact(unitType, unitId, basePath) {
70
83
  if (unitType !== "plan-slice" && unitType !== "execute-task" && unitType !== "complete-milestone")
71
84
  return { ok: true };
72
85
  if (!isDbAvailable())
73
86
  return { ok: true };
74
- if (!refreshOpenDatabaseFromDisk()) {
87
+ if (!refreshWorkflowDatabaseFromDisk()) {
75
88
  return {
76
89
  ok: false,
77
90
  fatal: unitType === "execute-task" || unitType === "complete-milestone",
@@ -100,50 +113,29 @@ export function refreshRecoveryDbForArtifact(unitType, unitId, basePath) {
100
113
  }
101
114
  if (isClosedStatus(milestone.status))
102
115
  return { ok: true };
103
- const validation = getLatestAssessmentByScope(mid, "milestone-validation");
104
- if (validation?.status !== "pass") {
105
- return {
106
- ok: false,
107
- fatal: true,
108
- reason: "complete-milestone-validation-not-pass",
109
- message: `Stuck recovery found complete-milestone ${unitId} artifacts, but milestone-validation is "${validation?.status ?? "absent"}" in the DB.`,
110
- };
111
- }
112
- const slices = getMilestoneSlices(mid);
113
- if (slices.length === 0) {
114
- return {
115
- ok: false,
116
- fatal: true,
117
- reason: "complete-milestone-slices-missing",
118
- message: `Stuck recovery found complete-milestone ${unitId} artifacts, but no slices exist in the DB.`,
119
- };
120
- }
121
- const openSlice = slices.find((slice) => !isClosedStatus(slice.status));
122
- if (openSlice) {
123
- return {
124
- ok: false,
125
- fatal: true,
126
- reason: "complete-milestone-slice-open",
127
- message: `Stuck recovery found complete-milestone ${unitId} artifacts, but slice ${openSlice.id} is still "${openSlice.status}" in the DB.`,
128
- };
129
- }
130
- for (const slice of slices) {
131
- const openTask = getSliceTasks(mid, slice.id).find((task) => !isClosedStatus(task.status));
132
- if (openTask) {
116
+ const artifactBasePath = resolveArtifactVerificationBase(unitId, basePath);
117
+ const closeoutProof = proveMilestoneCloseout(mid, {
118
+ allowOpenMilestone: true,
119
+ summaryArtifactBasePath: artifactBasePath,
120
+ implementationEvidence: {
121
+ basePath,
122
+ requirement: "present",
123
+ },
124
+ });
125
+ if (!closeoutProof.ok) {
126
+ if (closeoutProof.reason === "implementation-evidence-missing") {
133
127
  return {
134
128
  ok: false,
135
129
  fatal: true,
136
- reason: "complete-milestone-task-open",
137
- message: `Stuck recovery found complete-milestone ${unitId} artifacts, but task ${slice.id}/${openTask.id} is still "${openTask.status}" in the DB.`,
130
+ reason: "complete-milestone-implementation-missing",
131
+ message: `Stuck recovery found complete-milestone ${unitId} artifacts, but implementation evidence is not present.`,
138
132
  };
139
133
  }
140
- }
141
- if (hasImplementationArtifacts(basePath, mid) !== "present") {
142
134
  return {
143
135
  ok: false,
144
136
  fatal: true,
145
- reason: "complete-milestone-implementation-missing",
146
- message: `Stuck recovery found complete-milestone ${unitId} artifacts, but implementation evidence is not present.`,
137
+ reason: closeoutProofRecoveryReason(closeoutProof.reason),
138
+ message: `Stuck recovery found complete-milestone ${unitId} artifacts, but ${closeoutProof.message}`,
147
139
  };
148
140
  }
149
141
  updateMilestoneStatus(mid, "complete", new Date().toISOString());
@@ -457,7 +449,7 @@ export function verifyExpectedArtifact(unitType, unitId, base) {
457
449
  try {
458
450
  let taskIds = null;
459
451
  if (isDbAvailable()) {
460
- const refreshed = refreshOpenDatabaseFromDisk();
452
+ const refreshed = refreshWorkflowDatabaseFromDisk();
461
453
  if (refreshed) {
462
454
  const tasks = getSliceTasks(mid, sid);
463
455
  if (tasks.length > 0)
@@ -569,17 +561,24 @@ export function verifyExpectedArtifact(unitType, unitId, base) {
569
561
  // A milestone with only .gsd/ plan files and zero implementation code is
570
562
  // not genuinely complete — the LLM wrote plan files but skipped actual work.
571
563
  if (unitType === "complete-milestone") {
572
- const summaryOutcome = classifyMilestoneSummaryContent(readFileSync(absPath, "utf-8"));
573
- if (summaryOutcome === "failure")
574
- return false;
575
564
  const { milestone: mid } = parseUnitId(unitId);
576
- if (mid && isDbAvailable()) {
577
- const closeoutGate = checkCloseoutConsistencyGate(mid, { refreshFromDisk: true });
578
- if (!closeoutGate.ok)
579
- return false;
580
- }
581
- if (hasImplementationArtifacts(base, mid) === "absent")
565
+ if (!mid)
582
566
  return false;
567
+ const closeoutProof = proveMilestoneCloseout(mid, {
568
+ refreshFromDisk: true,
569
+ summaryArtifactBasePath: artifactBase,
570
+ implementationEvidence: {
571
+ basePath: base,
572
+ requirement: "not-absent",
573
+ },
574
+ });
575
+ if (!closeoutProof.ok) {
576
+ if (!isDbAvailable() && closeoutProof.reason === "db-unavailable") {
577
+ const summaryOutcome = classifyMilestoneSummaryContent(readFileSync(absPath, "utf-8"));
578
+ return summaryOutcome !== "failure" && hasImplementationArtifacts(base, mid) !== "absent";
579
+ }
580
+ return false;
581
+ }
583
582
  }
584
583
  return true;
585
584
  }
@@ -1,7 +1,13 @@
1
1
  // GSD auto-mode runtime state
2
2
  import { AutoSession } from "./auto/session.js";
3
3
  import { isDeterministicPolicyError, isQueuedUserMessageSkip, isToolInvocationError, markToolEnd as markTrackedToolEnd, markToolStart as markTrackedToolStart, } from "./auto-tool-tracking.js";
4
+ // Re-exported as a pure pass-through. Must stay UNGATED (no autoSession.active
5
+ // argument, unlike markToolStart at the bottom of this file) so it is true in
6
+ // foreground where the foreground approval-gate pause consults it.
7
+ export { isInteractiveElicitationInFlight } from "./auto-tool-tracking.js";
8
+ import { createToolSurfaceSnapshot, } from "./tool-surface-snapshot.js";
4
9
  export const autoSession = new AutoSession();
10
+ let currentToolSurfaceSnapshot = null;
5
11
  export function getAutoRuntimeSnapshot() {
6
12
  const orchestrationStatus = autoSession.orchestration?.getStatus();
7
13
  return {
@@ -12,8 +18,16 @@ export function getAutoRuntimeSnapshot() {
12
18
  orchestrationPhase: orchestrationStatus?.phase,
13
19
  orchestrationTransitionCount: orchestrationStatus?.transitionCount,
14
20
  orchestrationLastTransitionAt: orchestrationStatus?.lastTransitionAt,
21
+ toolSurface: autoSession.active || autoSession.paused ? currentToolSurfaceSnapshot : null,
15
22
  };
16
23
  }
24
+ export function recordAutoToolSurfaceSnapshot(input) {
25
+ currentToolSurfaceSnapshot = createToolSurfaceSnapshot(input);
26
+ return currentToolSurfaceSnapshot;
27
+ }
28
+ export function clearAutoToolSurfaceSnapshot() {
29
+ currentToolSurfaceSnapshot = null;
30
+ }
17
31
  export function isAutoActive() {
18
32
  return autoSession.active;
19
33
  }
@@ -23,6 +37,9 @@ export function isAutoPaused() {
23
37
  export function isAutoCompletionStopInProgress() {
24
38
  return autoSession.completionStopInProgress;
25
39
  }
40
+ export function clearAutoCompletionStopInProgress() {
41
+ autoSession.completionStopInProgress = false;
42
+ }
26
43
  export function markToolStart(toolCallId, toolName) {
27
44
  markTrackedToolStart(toolCallId, autoSession.active, toolName);
28
45
  }
@@ -34,7 +34,8 @@ import { initRoutingHistory } from "./routing-history.js";
34
34
  import { restoreHookState, resetHookState } from "./post-unit-hooks.js";
35
35
  import { resetProactiveHealing, setLevelChangeCallback } from "./doctor-proactive.js";
36
36
  import { snapshotSkills } from "./skill-discovery.js";
37
- import { isDbAvailable, getMilestone, getAllMilestones, insertMilestone, openDatabase, getDbStatus, updateMilestoneStatus } from "./gsd-db.js";
37
+ import { isDbAvailable, getMilestone, getAllMilestones, insertMilestone, updateMilestoneStatus } from "./gsd-db.js";
38
+ import { getWorkflowDatabaseStatus, openExistingWorkflowDatabase, openWorkflowDatabase, resolveProjectRootDbPath, } from "./db-workspace.js";
38
39
  import { isClosedStatus } from "./status-guards.js";
39
40
  import { classifyMilestoneSummaryContent } from "./milestone-summary-classifier.js";
40
41
  import { extractVerdict } from "./verdict-parser.js";
@@ -45,7 +46,6 @@ import { logWarning, logError } from "./workflow-logger.js";
45
46
  import { existsSync, mkdirSync, readFileSync, readdirSync, rmSync, } from "node:fs";
46
47
  import { join } from "node:path";
47
48
  import { sep as pathSep } from "node:path";
48
- import { resolveProjectRootDbPath } from "./bootstrap/dynamic-tools.js";
49
49
  import { validateDirectory } from "./validate-directory.js";
50
50
  import { isCustomProvider, resolveDefaultSessionModel, resolveDynamicRoutingConfig, } from "./preferences-models.js";
51
51
  import { getSessionModelOverride } from "./session-model-override.js";
@@ -76,11 +76,9 @@ export async function openProjectDbIfPresent(basePath) {
76
76
  const gsdDbPath = resolveProjectRootDbPath(basePath);
77
77
  if (!existsSync(gsdDbPath) || isDbAvailable())
78
78
  return;
79
- try {
80
- openDatabase(gsdDbPath);
81
- }
82
- catch (err) {
83
- logWarning("engine", `gsd-db: failed to open existing database: ${err instanceof Error ? err.message : String(err)}`);
79
+ const result = openExistingWorkflowDatabase(basePath);
80
+ if (!result.ok && result.reason === "open-failed") {
81
+ logWarning("engine", `gsd-db: failed to open existing database: ${result.error?.message ?? "open failed"}`);
84
82
  }
85
83
  }
86
84
  export function reconcileProjectMilestonesFromDisk(basePath) {
@@ -1295,23 +1293,14 @@ export async function bootstrapAutoSession(s, ctx, pi, base, verboseMode, reques
1295
1293
  }
1296
1294
  // ── DB lifecycle ──
1297
1295
  const gsdDbPath = resolveProjectRootDbPath(s.basePath);
1298
- const gsdDirPath = join(s.basePath, ".gsd");
1299
- if (existsSync(gsdDirPath) && !existsSync(gsdDbPath)) {
1300
- try {
1301
- const { openDatabase: openDb } = await import("./gsd-db.js");
1302
- openDb(gsdDbPath);
1303
- }
1304
- catch (err) {
1305
- logError("engine", `failed to initialize project database: ${err.message}`);
1306
- }
1296
+ const initialDbOpen = openWorkflowDatabase(s.basePath);
1297
+ if (!initialDbOpen.ok && initialDbOpen.reason === "open-failed") {
1298
+ logError("engine", `failed to initialize project database: ${initialDbOpen.error?.message ?? "open failed"}`);
1307
1299
  }
1308
1300
  if (_shouldAbortBootstrapForUnavailableDbForTest(gsdDbPath, isDbAvailable())) {
1309
- try {
1310
- const { openDatabase: openDb } = await import("./gsd-db.js");
1311
- openDb(gsdDbPath);
1312
- }
1313
- catch (err) {
1314
- logError("engine", `failed to open existing database: ${err.message}`);
1301
+ const retryDbOpen = openWorkflowDatabase(s.basePath);
1302
+ if (!retryDbOpen.ok && retryDbOpen.reason === "open-failed") {
1303
+ logError("engine", `failed to open existing database: ${retryDbOpen.error?.message ?? "open failed"}`);
1315
1304
  }
1316
1305
  }
1317
1306
  // Gate: abort bootstrap if the DB file exists but the provider is
@@ -1320,7 +1309,7 @@ export async function bootstrapAutoSession(s, ctx, pi, base, verboseMode, reques
1320
1309
  // call returns "db_unavailable", triggering artifact-retry which
1321
1310
  // re-dispatches the same task — producing an infinite loop (#2419).
1322
1311
  if (existsSync(gsdDbPath) && !isDbAvailable()) {
1323
- const dbStatus = getDbStatus();
1312
+ const dbStatus = getWorkflowDatabaseStatus();
1324
1313
  const phaseHint = dbStatus.lastPhase === "open"
1325
1314
  ? "The database file could not be opened"
1326
1315
  : dbStatus.lastPhase === "initSchema"
@@ -233,11 +233,24 @@ export function startUnitSupervision(sctx) {
233
233
  }
234
234
  }, 15000);
235
235
  // ── 3. Hard timeout ──
236
- s.unitTimeoutHandle = setTimeout(async () => {
236
+ const hardTimeoutBody = async () => {
237
237
  try {
238
238
  s.unitTimeoutHandle = null;
239
239
  if (!s.active)
240
240
  return;
241
+ // User-interactive tools (ask_user_questions, secure_env_collect) block
242
+ // waiting for human input by design. Unlike the idle watchdog (above),
243
+ // the hard timeout had no interactive exemption, so a long human
244
+ // deliberation crossing the hard cap would still closeout + recover
245
+ // (triggerTurn) and abort the turn hosting the elicitation, reintroducing
246
+ // the self-cancel loop on slow answers (#2676). Re-arm instead of firing
247
+ // while a human is being asked, mirroring the idle watchdog's
248
+ // refresh-and-return. Conditional on hasInteractiveToolInFlight(), so a
249
+ // genuinely hung non-interactive unit still hits the hard cap.
250
+ if (getInFlightToolCount() > 0 && hasInteractiveToolInFlight()) {
251
+ s.unitTimeoutHandle = setTimeout(hardTimeoutBody, hardTimeoutMs);
252
+ return;
253
+ }
241
254
  const expectedCurrentUnit = s.currentUnit
242
255
  ? { type: s.currentUnit.type, id: s.currentUnit.id, startedAt: s.currentUnit.startedAt }
243
256
  : null;
@@ -269,7 +282,8 @@ export function startUnitSupervision(sctx) {
269
282
  logWarning("timer", `notification failed: ${err instanceof Error ? err.message : String(err)}`);
270
283
  }
271
284
  }
272
- }, hardTimeoutMs);
285
+ };
286
+ s.unitTimeoutHandle = setTimeout(hardTimeoutBody, hardTimeoutMs);
273
287
  // ── 4. Context-pressure continue-here monitor ──
274
288
  if (s.continueHereHandle) {
275
289
  clearInterval(s.continueHereHandle);
@@ -10,6 +10,16 @@ const inFlightTools = new Map();
10
10
  * The idle watchdog must not treat these as stalled.
11
11
  */
12
12
  const INTERACTIVE_TOOLS = new Set(["ask_user_questions", "secure_env_collect"]);
13
+ /**
14
+ * Mode-agnostic refcount of in-flight interactive elicitations that are an
15
+ * active human boundary (the model ASKED via ask_user_questions). Unlike the
16
+ * `inFlightTools` Map, this is NOT gated by auto-session.active, so it is true
17
+ * in FOREGROUND (where s.active is false). Kept SEPARATE from inFlightTools so
18
+ * getInFlightToolCount()/getOldestInFlightToolAgeMs()/hasInteractiveToolInFlight()
19
+ * and the auto-watchdog accounting are byte-for-byte unchanged. A refcount (not
20
+ * a boolean) handles nested/back-to-back elicitations in a single turn.
21
+ */
22
+ let interactiveElicitationDepth = 0;
13
23
  /**
14
24
  * Mark a tool execution as in-flight.
15
25
  * Records start time and tool name so the idle watchdog can detect tools
@@ -26,6 +36,27 @@ export function markToolStart(toolCallId, isActive, toolName) {
26
36
  export function markToolEnd(toolCallId) {
27
37
  inFlightTools.delete(toolCallId);
28
38
  }
39
+ /**
40
+ * Mark an interactive elicitation (the model asking via ask_user_questions) as
41
+ * in flight. Ungated by auto-session.active so it is observable in foreground.
42
+ */
43
+ export function markInteractiveElicitationStart() {
44
+ interactiveElicitationDepth++;
45
+ }
46
+ /**
47
+ * Mark an interactive elicitation as completed. Idempotent below zero.
48
+ */
49
+ export function markInteractiveElicitationEnd() {
50
+ if (interactiveElicitationDepth > 0)
51
+ interactiveElicitationDepth--;
52
+ }
53
+ /**
54
+ * Returns true if any interactive elicitation is currently the active human
55
+ * boundary. True in ALL modes (foreground and auto) while one is in flight.
56
+ */
57
+ export function isInteractiveElicitationInFlight() {
58
+ return interactiveElicitationDepth > 0;
59
+ }
29
60
  /**
30
61
  * Returns the age (ms) of the oldest currently in-flight tool, or 0 if none.
31
62
  */
@@ -75,6 +106,7 @@ export function hasInteractiveToolInFlight() {
75
106
  */
76
107
  export function clearInFlightTools() {
77
108
  inFlightTools.clear();
109
+ interactiveElicitationDepth = 0;
78
110
  }
79
111
  // ─── Tool invocation error classification (#2883) ────────────────────────
80
112
  /**
@@ -123,6 +155,11 @@ export const DETERMINISTIC_POLICY_ERROR_STRINGS = [
123
155
  // "Cannot write to milestone CONTEXT.md without depth verification." for direct
124
156
  // write tool calls to *-CONTEXT.md paths (different code path than gsd_summary_save).
125
157
  "CONTEXT.md without depth verification",
158
+ // Section-close gate units (execute-task, complete-slice, validate-milestone) that
159
+ // reach for gsd_save_gate_result get the calm redirect from softGateToolRedirect
160
+ // (auto-unit-tool-scope.ts) instead of a HARD BLOCK. Still deterministic — those
161
+ // phases never own that tool, so a retry hits the same redirect every time.
162
+ "closes its quality gates by writing summary sections",
126
163
  ];
127
164
  /**
128
165
  * Returns true if the error message indicates a deterministic policy gate
@@ -1,30 +1,22 @@
1
1
  import { parseUnitId } from "./unit-id.js";
2
2
  import { AUTO_UNIT_SCOPED_TOOLS, getForbiddenGsdToolReason, } from "./unit-tool-contracts.js";
3
+ import { WORKFLOW_TOOL_ALIAS_PAIRS, canonicalWorkflowSurfaceToolName, isWorkflowSurfaceAliasTool, stripMcpToolPrefix, } from "./workflow-tool-surface.js";
3
4
  export { AUTO_UNIT_SCOPED_TOOLS, RUN_UAT_BROWSER_TOOL_NAMES, } from "./unit-tool-contracts.js";
4
- const WORKFLOW_TOOL_ALIASES = {
5
- gsd_save_decision: "gsd_decision_save",
6
- gsd_update_requirement: "gsd_requirement_update",
7
- gsd_save_requirement: "gsd_requirement_save",
8
- gsd_save_summary: "gsd_summary_save",
9
- gsd_generate_milestone_id: "gsd_milestone_generate_id",
10
- gsd_milestone_plan: "gsd_plan_milestone",
11
- gsd_slice_plan: "gsd_plan_slice",
12
- gsd_task_plan: "gsd_plan_task",
13
- gsd_slice_replan: "gsd_replan_slice",
14
- gsd_complete_slice: "gsd_slice_complete",
15
- gsd_milestone_complete: "gsd_complete_milestone",
16
- gsd_milestone_validate: "gsd_validate_milestone",
17
- gsd_roadmap_reassess: "gsd_reassess_roadmap",
18
- gsd_complete_task: "gsd_task_complete",
19
- gsd_reopen_task: "gsd_task_reopen",
20
- gsd_reopen_slice: "gsd_slice_reopen",
21
- gsd_reopen_milestone: "gsd_milestone_reopen",
22
- };
23
5
  const EXECUTE_TASK_UNIT_TYPES = new Set([
24
6
  "execute-task",
25
7
  "execute-task-simple",
26
8
  "reactive-execute",
27
9
  ]);
10
+ // These units own quality gates, but their completion handlers persist verdicts
11
+ // from artifact sections. gsd_save_gate_result belongs to gate-evaluate, so keep
12
+ // blocking it here with a calm redirect to the section-write path.
13
+ const SECTION_CLOSE_GATE_UNIT_TYPES = new Set([
14
+ "execute-task",
15
+ "execute-task-simple",
16
+ "reactive-execute",
17
+ "complete-slice",
18
+ "validate-milestone",
19
+ ]);
28
20
  const EXTRA_SCOPED_GSD_LIFECYCLE_TOOLS = [
29
21
  "gsd_skip_slice",
30
22
  "gsd_slice_reopen",
@@ -32,24 +24,18 @@ const EXTRA_SCOPED_GSD_LIFECYCLE_TOOLS = [
32
24
  ];
33
25
  const SCOPED_GSD_LIFECYCLE_TOOLS = new Set([
34
26
  ...Object.values(AUTO_UNIT_SCOPED_TOOLS).flat(),
35
- ...Object.values(WORKFLOW_TOOL_ALIASES),
27
+ ...WORKFLOW_TOOL_ALIAS_PAIRS.map(({ canonical }) => canonical),
36
28
  ...EXTRA_SCOPED_GSD_LIFECYCLE_TOOLS,
37
29
  ]
38
30
  .filter((tool) => tool.startsWith("gsd_"))
39
31
  .map(canonicalWorkflowToolName));
40
32
  export const GSD_PHASE_SCOPE_DISPLAY_REASON = "This GSD phase only allows its scoped workflow tools.";
41
- function stripMcpToolPrefix(toolName) {
42
- if (!toolName.startsWith("mcp__"))
43
- return toolName;
44
- const toolSeparator = toolName.indexOf("__", "mcp__".length);
45
- return toolSeparator >= 0 ? toolName.slice(toolSeparator + 2) : toolName;
46
- }
33
+ export const GSD_SECTION_CLOSE_GATE_DISPLAY_REASON = "Gates here close by writing summary sections — gsd_save_gate_result isn't needed.";
47
34
  export function canonicalWorkflowToolName(toolName) {
48
- const base = stripMcpToolPrefix(toolName);
49
- return WORKFLOW_TOOL_ALIASES[base] ?? base;
35
+ return canonicalWorkflowSurfaceToolName(toolName);
50
36
  }
51
37
  export function isWorkflowAliasTool(toolName) {
52
- return Object.prototype.hasOwnProperty.call(WORKFLOW_TOOL_ALIASES, stripMcpToolPrefix(toolName));
38
+ return isWorkflowSurfaceAliasTool(toolName);
53
39
  }
54
40
  function hardBlockReason(unitType, what) {
55
41
  return [
@@ -65,6 +51,21 @@ function hardBlock(unitType, what) {
65
51
  displayReason: GSD_PHASE_SCOPE_DISPLAY_REASON,
66
52
  };
67
53
  }
54
+ // This stable marker is registered in auto-tool-tracking.ts so auto-mode treats
55
+ // the redirect as deterministic policy, not a retryable execution failure.
56
+ function softGateToolRedirect(unitType) {
57
+ return {
58
+ block: true,
59
+ reason: [
60
+ `Skip this call — the "${unitType}" phase closes its quality gates by writing summary sections,`,
61
+ "not by calling gsd_save_gate_result (that tool belongs to the gate-evaluate phase).",
62
+ "Record each gate by filling its named section in your summary: a populated section records `pass`,",
63
+ "an empty one records `omitted`. Then call your completion tool and the handler persists every verdict.",
64
+ "This is expected, not an error — continue without gsd_save_gate_result.",
65
+ ].join(" "),
66
+ displayReason: GSD_SECTION_CLOSE_GATE_DISPLAY_REASON,
67
+ };
68
+ }
68
69
  function allowedGsdToolsForUnit(unitType) {
69
70
  return [...new Set((AUTO_UNIT_SCOPED_TOOLS[unitType] ?? [])
70
71
  .filter((tool) => tool.startsWith("gsd_"))
@@ -122,5 +123,8 @@ export function shouldBlockAutoUnitToolCall(unitType, toolName, input, unitId) {
122
123
  if (forbiddenReason) {
123
124
  return hardBlock(unitType, `GSD lifecycle tool "${canonicalTool}" is not permitted; ${forbiddenReason} Fix unit-tool-contracts.ts or the ${unitType} prompt.`);
124
125
  }
126
+ if (canonicalTool === "gsd_save_gate_result" && SECTION_CLOSE_GATE_UNIT_TYPES.has(unitType)) {
127
+ return softGateToolRedirect(unitType);
128
+ }
125
129
  return hardBlock(unitType, `GSD lifecycle tool "${canonicalTool}" is not permitted; allowed GSD tools: ${allowedTools.length > 0 ? allowedTools.join(", ") : "(none)"}`);
126
130
  }
@@ -2,11 +2,11 @@
2
2
  // File Purpose: Post-unit verification gate for GSD auto-mode units.
3
3
  import { existsSync, mkdirSync, readFileSync, readdirSync, writeFileSync } from "node:fs";
4
4
  import { gsdProjectionRoot, resolveSlicePath, resolveMilestoneFile } from "./paths.js";
5
+ import { resolveMilestoneValidationVerdict } from "./milestone-validation-verdict.js";
5
6
  import { resolveCanonicalMilestoneRoot } from "./worktree-manager.js";
6
7
  import { parseUnitId } from "./unit-id.js";
7
8
  import { isDbAvailable, getTask, getSliceTasks, getMilestoneSlices } from "./gsd-db.js";
8
9
  import { loadEffectiveGSDPreferences } from "./preferences.js";
9
- import { extractVerdict } from "./verdict-parser.js";
10
10
  import { isClosedStatus } from "./status-guards.js";
11
11
  import { loadFile } from "./files.js";
12
12
  import { parseRoadmap } from "./parsers-legacy.js";
@@ -214,22 +214,22 @@ async function runValidateMilestonePostCheck(vctx, pauseAuto) {
214
214
  };
215
215
  const validationBasePath = resolveCanonicalMilestoneRoot(s.basePath, mid);
216
216
  const validationFile = join(gsdProjectionRoot(validationBasePath), "milestones", mid, `${mid}-VALIDATION.md`);
217
- if (!validationFile) {
217
+ const validationContent = await loadFile(validationFile);
218
+ if (validationContent !== null && validationContent.trim() === "") {
218
219
  if (await reassessmentInvalidatedValidation()) {
219
220
  clearValidationRetry();
220
221
  return "continue";
221
222
  }
222
- return setToolFailureRetry("You must call gsd_validate_milestone to persist the validation results. No VALIDATION.md was created.");
223
+ return setToolFailureRetry("You must call gsd_validate_milestone to persist the validation results. VALIDATION.md exists but is empty.");
223
224
  }
224
- const validationContent = await loadFile(validationFile);
225
- if (!validationContent) {
225
+ const verdict = await resolveMilestoneValidationVerdict(s.basePath, mid);
226
+ if (!verdict) {
226
227
  if (await reassessmentInvalidatedValidation()) {
227
228
  clearValidationRetry();
228
229
  return "continue";
229
230
  }
230
- return setToolFailureRetry("You must call gsd_validate_milestone to persist the validation results. VALIDATION.md exists but is empty.");
231
+ return setToolFailureRetry("You must call gsd_validate_milestone to persist the validation results. No VALIDATION.md was created.");
231
232
  }
232
- const verdict = extractVerdict(validationContent);
233
233
  if (verdict === "needs-attention") {
234
234
  ctx.ui.notify(`Milestone ${mid} validation returned verdict=needs-attention. Pausing for human review.`, "error");
235
235
  process.stderr.write([