@opengsd/gsd-pi 1.2.0-dev.844675c9 → 1.2.0-dev.b1abb545

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 (375) 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/resources/.managed-resources-content-hash +1 -1
  5. package/dist/resources/extensions/ask-user-questions.js +78 -23
  6. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +84 -228
  7. package/dist/resources/extensions/claude-code-cli/turn-assembler.js +224 -0
  8. package/dist/resources/extensions/github-sync/templates.js +3 -3
  9. package/dist/resources/extensions/gsd/artifact-projection.js +14 -0
  10. package/dist/resources/extensions/gsd/auto/loop.js +74 -56
  11. package/dist/resources/extensions/gsd/auto/orchestrator.js +109 -11
  12. package/dist/resources/extensions/gsd/auto/phases.js +28 -3
  13. package/dist/resources/extensions/gsd/auto/run-unit.js +2 -1
  14. package/dist/resources/extensions/gsd/auto/session.js +3 -0
  15. package/dist/resources/extensions/gsd/auto-dashboard.js +16 -4
  16. package/dist/resources/extensions/gsd/auto-dispatch.js +6 -5
  17. package/dist/resources/extensions/gsd/auto-model-selection.js +8 -0
  18. package/dist/resources/extensions/gsd/auto-post-unit.js +4 -3
  19. package/dist/resources/extensions/gsd/auto-prompts.js +81 -8
  20. package/dist/resources/extensions/gsd/auto-recovery.js +48 -49
  21. package/dist/resources/extensions/gsd/auto-runtime-state.js +14 -0
  22. package/dist/resources/extensions/gsd/auto-start.js +12 -23
  23. package/dist/resources/extensions/gsd/auto-timers.js +16 -2
  24. package/dist/resources/extensions/gsd/auto-tool-tracking.js +32 -0
  25. package/dist/resources/extensions/gsd/auto-unit-tool-scope.js +4 -29
  26. package/dist/resources/extensions/gsd/auto-verification.js +7 -7
  27. package/dist/resources/extensions/gsd/auto-worktree.js +21 -19
  28. package/dist/resources/extensions/gsd/auto.js +11 -7
  29. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +28 -37
  30. package/dist/resources/extensions/gsd/bootstrap/dynamic-tools.js +11 -37
  31. package/dist/resources/extensions/gsd/bootstrap/query-tools.js +2 -2
  32. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +100 -138
  33. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +63 -4
  34. package/dist/resources/extensions/gsd/closeout-consistency-gate.js +21 -4
  35. package/dist/resources/extensions/gsd/codebase-generator.js +8 -4
  36. package/dist/resources/extensions/gsd/commands/handlers/auto.js +3 -0
  37. package/dist/resources/extensions/gsd/commands-handlers.js +20 -0
  38. package/dist/resources/extensions/gsd/commands-inspect.js +4 -8
  39. package/dist/resources/extensions/gsd/commands-maintenance.js +61 -41
  40. package/dist/resources/extensions/gsd/commands-ship.js +2 -2
  41. package/dist/resources/extensions/gsd/commands-verdict.js +12 -2
  42. package/dist/resources/extensions/gsd/db-workspace.js +103 -0
  43. package/dist/resources/extensions/gsd/delegation-policy.js +2 -10
  44. package/dist/resources/extensions/gsd/discussion-handoff.js +218 -0
  45. package/dist/resources/extensions/gsd/docs/preferences-reference.md +9 -0
  46. package/dist/resources/extensions/gsd/doctor.js +16 -9
  47. package/dist/resources/extensions/gsd/error-classifier.js +1 -1
  48. package/dist/resources/extensions/gsd/git-conflict-state.js +16 -1
  49. package/dist/resources/extensions/gsd/gsd-db.js +12 -0
  50. package/dist/resources/extensions/gsd/guided-flow.js +34 -468
  51. package/dist/resources/extensions/gsd/guided-unit-completion.js +225 -0
  52. package/dist/resources/extensions/gsd/markdown-renderer.js +2 -1
  53. package/dist/resources/extensions/gsd/mcp-filter.js +2 -1
  54. package/dist/resources/extensions/gsd/mcp-tool-name.js +26 -0
  55. package/dist/resources/extensions/gsd/md-importer.js +4 -3
  56. package/dist/resources/extensions/gsd/migrate/safety.js +2 -2
  57. package/dist/resources/extensions/gsd/migration-auto-check.js +3 -2
  58. package/dist/resources/extensions/gsd/milestone-closeout-proof.js +72 -0
  59. package/dist/resources/extensions/gsd/milestone-closeout.js +12 -4
  60. package/dist/resources/extensions/gsd/milestone-merge-transaction.js +10 -0
  61. package/dist/resources/extensions/gsd/milestone-planning-persistence.js +156 -0
  62. package/dist/resources/extensions/gsd/milestone-readiness.js +77 -0
  63. package/dist/resources/extensions/gsd/milestone-settlement.js +50 -0
  64. package/dist/resources/extensions/gsd/milestone-validation-evidence.js +73 -0
  65. package/dist/resources/extensions/gsd/milestone-validation-verdict.js +57 -0
  66. package/dist/resources/extensions/gsd/parallel-eligibility.js +3 -6
  67. package/dist/resources/extensions/gsd/parallel-orchestrator.js +3 -2
  68. package/dist/resources/extensions/gsd/preferences-diagnostics.js +67 -0
  69. package/dist/resources/extensions/gsd/preferences.js +147 -29
  70. package/dist/resources/extensions/gsd/prompts/complete-slice.md +1 -0
  71. package/dist/resources/extensions/gsd/prompts/execute-task.md +2 -0
  72. package/dist/resources/extensions/gsd/prompts/guided-discuss-project.md +3 -1
  73. package/dist/resources/extensions/gsd/prompts/plan-milestone.md +2 -0
  74. package/dist/resources/extensions/gsd/prompts/plan-slice.md +1 -0
  75. package/dist/resources/extensions/gsd/prompts/refine-slice.md +1 -0
  76. package/dist/resources/extensions/gsd/prompts/system.md +1 -1
  77. package/dist/resources/extensions/gsd/provider-payload-policy.js +83 -0
  78. package/dist/resources/extensions/gsd/pull-request-process.js +13 -0
  79. package/dist/resources/extensions/gsd/quality-gate-closure.js +109 -0
  80. package/dist/resources/extensions/gsd/question-transport.js +86 -0
  81. package/dist/resources/extensions/gsd/roadmap-slices.js +8 -2
  82. package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +3 -2
  83. package/dist/resources/extensions/gsd/state.js +13 -5
  84. package/dist/resources/extensions/gsd/templates/plan.md +7 -0
  85. package/dist/resources/extensions/gsd/templates/project.md +1 -0
  86. package/dist/resources/extensions/gsd/templates/roadmap.md +1 -1
  87. package/dist/resources/extensions/gsd/templates/uat.md +5 -1
  88. package/dist/resources/extensions/gsd/tool-contract.js +52 -8
  89. package/dist/resources/extensions/gsd/tool-presentation-plan.js +15 -34
  90. package/dist/resources/extensions/gsd/tool-surface-snapshot.js +17 -0
  91. package/dist/resources/extensions/gsd/tools/plan-milestone.js +15 -143
  92. package/dist/resources/extensions/gsd/tools/reassess-roadmap.js +39 -0
  93. package/dist/resources/extensions/gsd/tools/validate-milestone.js +15 -78
  94. package/dist/resources/extensions/gsd/uat-policy.js +16 -10
  95. package/dist/resources/extensions/gsd/uat-run.js +9 -14
  96. package/dist/resources/extensions/gsd/unit-context-composer.js +40 -20
  97. package/dist/resources/extensions/gsd/unit-runtime.js +3 -2
  98. package/dist/resources/extensions/gsd/unit-tool-contracts.js +2 -1
  99. package/dist/resources/extensions/gsd/user-input-boundary.js +23 -0
  100. package/dist/resources/extensions/gsd/validation-block-guard.js +2 -0
  101. package/dist/resources/extensions/gsd/web-app-uat.js +80 -0
  102. package/dist/resources/extensions/gsd/workflow-mcp.js +15 -102
  103. package/dist/resources/extensions/gsd/workflow-reconcile.js +4 -3
  104. package/dist/resources/extensions/gsd/workflow-tool-surface.js +46 -0
  105. package/dist/resources/extensions/gsd/workspace-git-guard.js +2 -0
  106. package/dist/resources/extensions/gsd/worktree-state-projection.js +33 -4
  107. package/dist/resources/extensions/gsd/worktree-telemetry.js +12 -0
  108. package/dist/resources/extensions/shared/interview-ui.js +2 -2
  109. package/dist/resources/shared/claude-runtime-floor.js +182 -0
  110. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  111. package/dist/update-cmd.js +20 -0
  112. package/dist/web/standalone/.next/BUILD_ID +1 -1
  113. package/dist/web/standalone/.next/app-path-routes-manifest.json +7 -7
  114. package/dist/web/standalone/.next/build-manifest.json +3 -3
  115. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  116. package/dist/web/standalone/.next/react-loadable-manifest.json +8 -8
  117. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  118. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  119. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  120. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  121. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  122. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  123. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  124. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  125. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  126. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  127. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  128. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  129. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  130. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  131. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  132. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  133. package/dist/web/standalone/.next/server/app/index.html +1 -1
  134. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  135. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  136. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  137. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  138. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  139. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  140. package/dist/web/standalone/.next/server/app-paths-manifest.json +7 -7
  141. package/dist/web/standalone/.next/server/chunks/8357.js +1 -1
  142. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  143. package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
  144. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  145. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  146. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  147. package/dist/web/standalone/.next/static/chunks/2659.b7b129ee6a769448.js +1 -0
  148. package/dist/web/standalone/.next/static/chunks/2772.bfa657f49f955239.js +1 -0
  149. package/dist/web/standalone/.next/static/chunks/{3616.4113d484a994e411.js → 3616.3c60753b8ffcbd2e.js} +1 -1
  150. package/dist/web/standalone/.next/static/chunks/4283.e4873b058df143a1.js +2 -0
  151. package/dist/web/standalone/.next/static/chunks/5826.a46ecdd1cfe8dabc.js +1 -0
  152. package/dist/web/standalone/.next/static/chunks/796.cf859a427a2cb2ac.js +10 -0
  153. package/dist/web/standalone/.next/static/chunks/8785.2e5a118797fb2dd2.js +1 -0
  154. package/dist/web/standalone/.next/static/chunks/{webpack-dda80a1ef5587410.js → webpack-fbea77b5f9953368.js} +1 -1
  155. package/dist/web/standalone/node_modules/node-pty/build/Makefile +1 -1
  156. package/dist/web-mode.d.ts +2 -0
  157. package/dist/web-mode.js +20 -8
  158. package/package.json +2 -1
  159. package/packages/cloud-mcp-gateway/package.json +2 -2
  160. package/packages/contracts/package.json +1 -1
  161. package/packages/daemon/package.json +4 -4
  162. package/packages/gsd-agent-core/dist/session/agent-session-extensions.d.ts +2 -0
  163. package/packages/gsd-agent-core/dist/session/agent-session-extensions.d.ts.map +1 -1
  164. package/packages/gsd-agent-core/dist/session/agent-session-extensions.js +14 -0
  165. package/packages/gsd-agent-core/dist/session/agent-session-extensions.js.map +1 -1
  166. package/packages/gsd-agent-core/package.json +5 -5
  167. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  168. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js +106 -40
  169. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  170. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-extension-widgets.d.ts.map +1 -1
  171. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-extension-widgets.js +6 -0
  172. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-extension-widgets.js.map +1 -1
  173. package/packages/gsd-agent-modes/package.json +7 -7
  174. package/packages/mcp-server/dist/server.d.ts +10 -0
  175. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  176. package/packages/mcp-server/dist/server.js +8 -0
  177. package/packages/mcp-server/dist/server.js.map +1 -1
  178. package/packages/mcp-server/dist/workflow-tools.d.ts +41 -0
  179. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  180. package/packages/mcp-server/dist/workflow-tools.js +2 -1
  181. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  182. package/packages/mcp-server/package.json +3 -3
  183. package/packages/native/package.json +1 -1
  184. package/packages/pi-agent-core/package.json +1 -1
  185. package/packages/pi-ai/dist/models.generated.d.ts +8 -93
  186. package/packages/pi-ai/dist/models.generated.d.ts.map +1 -1
  187. package/packages/pi-ai/dist/models.generated.js +35 -120
  188. package/packages/pi-ai/dist/models.generated.js.map +1 -1
  189. package/packages/pi-ai/package.json +1 -1
  190. package/packages/pi-coding-agent/package.json +7 -7
  191. package/packages/pi-tui/dist/components/input.js +1 -1
  192. package/packages/pi-tui/dist/components/input.js.map +1 -1
  193. package/packages/pi-tui/dist/keys.d.ts.map +1 -1
  194. package/packages/pi-tui/dist/keys.js +39 -30
  195. package/packages/pi-tui/dist/keys.js.map +1 -1
  196. package/packages/pi-tui/dist/stdin-buffer.d.ts.map +1 -1
  197. package/packages/pi-tui/dist/stdin-buffer.js +22 -0
  198. package/packages/pi-tui/dist/stdin-buffer.js.map +1 -1
  199. package/packages/pi-tui/package.json +2 -2
  200. package/packages/rpc-client/package.json +2 -2
  201. package/pkg/package.json +1 -1
  202. package/src/resources/extensions/ask-user-questions.ts +87 -24
  203. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +108 -281
  204. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +240 -0
  205. package/src/resources/extensions/claude-code-cli/turn-assembler.ts +287 -0
  206. package/src/resources/extensions/github-sync/templates.ts +3 -3
  207. package/src/resources/extensions/github-sync/tests/templates.test.ts +2 -2
  208. package/src/resources/extensions/gsd/artifact-projection.ts +31 -0
  209. package/src/resources/extensions/gsd/auto/contracts.ts +32 -2
  210. package/src/resources/extensions/gsd/auto/loop-deps.ts +2 -0
  211. package/src/resources/extensions/gsd/auto/loop.ts +83 -61
  212. package/src/resources/extensions/gsd/auto/orchestrator.ts +125 -12
  213. package/src/resources/extensions/gsd/auto/phases.ts +35 -3
  214. package/src/resources/extensions/gsd/auto/run-unit.ts +2 -1
  215. package/src/resources/extensions/gsd/auto/session.ts +4 -0
  216. package/src/resources/extensions/gsd/auto-dashboard.ts +18 -4
  217. package/src/resources/extensions/gsd/auto-dispatch.ts +20 -7
  218. package/src/resources/extensions/gsd/auto-model-selection.ts +8 -0
  219. package/src/resources/extensions/gsd/auto-post-unit.ts +4 -3
  220. package/src/resources/extensions/gsd/auto-prompts.ts +107 -9
  221. package/src/resources/extensions/gsd/auto-recovery.ts +50 -50
  222. package/src/resources/extensions/gsd/auto-runtime-state.ts +26 -0
  223. package/src/resources/extensions/gsd/auto-start.ts +17 -20
  224. package/src/resources/extensions/gsd/auto-timers.ts +16 -2
  225. package/src/resources/extensions/gsd/auto-tool-tracking.ts +35 -0
  226. package/src/resources/extensions/gsd/auto-unit-tool-scope.ts +9 -30
  227. package/src/resources/extensions/gsd/auto-verification.ts +7 -8
  228. package/src/resources/extensions/gsd/auto-worktree.ts +33 -26
  229. package/src/resources/extensions/gsd/auto.ts +15 -8
  230. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +29 -37
  231. package/src/resources/extensions/gsd/bootstrap/dynamic-tools.ts +10 -37
  232. package/src/resources/extensions/gsd/bootstrap/query-tools.ts +2 -2
  233. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +116 -151
  234. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +107 -3
  235. package/src/resources/extensions/gsd/closeout-consistency-gate.ts +27 -5
  236. package/src/resources/extensions/gsd/codebase-generator.ts +9 -5
  237. package/src/resources/extensions/gsd/commands/handlers/auto.ts +3 -0
  238. package/src/resources/extensions/gsd/commands-handlers.ts +18 -0
  239. package/src/resources/extensions/gsd/commands-inspect.ts +7 -8
  240. package/src/resources/extensions/gsd/commands-maintenance.ts +74 -40
  241. package/src/resources/extensions/gsd/commands-ship.ts +2 -2
  242. package/src/resources/extensions/gsd/commands-verdict.ts +19 -2
  243. package/src/resources/extensions/gsd/db-workspace.ts +170 -0
  244. package/src/resources/extensions/gsd/delegation-policy.ts +3 -11
  245. package/src/resources/extensions/gsd/discussion-handoff.ts +276 -0
  246. package/src/resources/extensions/gsd/docs/preferences-reference.md +9 -0
  247. package/src/resources/extensions/gsd/doctor.ts +15 -5
  248. package/src/resources/extensions/gsd/error-classifier.ts +1 -1
  249. package/src/resources/extensions/gsd/git-conflict-state.ts +17 -1
  250. package/src/resources/extensions/gsd/gsd-db.ts +12 -0
  251. package/src/resources/extensions/gsd/guided-flow.ts +47 -558
  252. package/src/resources/extensions/gsd/guided-unit-completion.ts +275 -0
  253. package/src/resources/extensions/gsd/markdown-renderer.ts +2 -1
  254. package/src/resources/extensions/gsd/mcp-filter.ts +2 -1
  255. package/src/resources/extensions/gsd/mcp-tool-name.ts +35 -0
  256. package/src/resources/extensions/gsd/md-importer.ts +3 -3
  257. package/src/resources/extensions/gsd/migrate/safety.ts +2 -2
  258. package/src/resources/extensions/gsd/migration-auto-check.ts +2 -2
  259. package/src/resources/extensions/gsd/milestone-closeout-proof.ts +131 -0
  260. package/src/resources/extensions/gsd/milestone-closeout.ts +12 -4
  261. package/src/resources/extensions/gsd/milestone-merge-transaction.ts +47 -0
  262. package/src/resources/extensions/gsd/milestone-planning-persistence.ts +224 -0
  263. package/src/resources/extensions/gsd/milestone-readiness.ts +125 -0
  264. package/src/resources/extensions/gsd/milestone-settlement.ts +81 -0
  265. package/src/resources/extensions/gsd/milestone-validation-evidence.ts +95 -0
  266. package/src/resources/extensions/gsd/milestone-validation-verdict.ts +80 -0
  267. package/src/resources/extensions/gsd/parallel-eligibility.ts +4 -5
  268. package/src/resources/extensions/gsd/parallel-orchestrator.ts +6 -2
  269. package/src/resources/extensions/gsd/preferences-diagnostics.ts +98 -0
  270. package/src/resources/extensions/gsd/preferences-types.ts +16 -0
  271. package/src/resources/extensions/gsd/preferences.ts +173 -28
  272. package/src/resources/extensions/gsd/prompts/complete-slice.md +1 -0
  273. package/src/resources/extensions/gsd/prompts/execute-task.md +2 -0
  274. package/src/resources/extensions/gsd/prompts/guided-discuss-project.md +3 -1
  275. package/src/resources/extensions/gsd/prompts/plan-milestone.md +2 -0
  276. package/src/resources/extensions/gsd/prompts/plan-slice.md +1 -0
  277. package/src/resources/extensions/gsd/prompts/refine-slice.md +1 -0
  278. package/src/resources/extensions/gsd/prompts/system.md +1 -1
  279. package/src/resources/extensions/gsd/provider-payload-policy.ts +140 -0
  280. package/src/resources/extensions/gsd/pull-request-process.ts +41 -0
  281. package/src/resources/extensions/gsd/quality-gate-closure.ts +140 -0
  282. package/src/resources/extensions/gsd/question-transport.ts +138 -0
  283. package/src/resources/extensions/gsd/roadmap-slices.ts +8 -2
  284. package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +6 -2
  285. package/src/resources/extensions/gsd/state.ts +15 -5
  286. package/src/resources/extensions/gsd/templates/plan.md +7 -0
  287. package/src/resources/extensions/gsd/templates/project.md +1 -0
  288. package/src/resources/extensions/gsd/templates/roadmap.md +1 -1
  289. package/src/resources/extensions/gsd/templates/uat.md +5 -1
  290. package/src/resources/extensions/gsd/tests/ask-user-questions-render.test.ts +92 -0
  291. package/src/resources/extensions/gsd/tests/auto-dashboard.test.ts +29 -1
  292. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +321 -5
  293. package/src/resources/extensions/gsd/tests/auto-milestone-target.test.ts +23 -0
  294. package/src/resources/extensions/gsd/tests/auto-model-selection-tool-poisoning.test.ts +18 -0
  295. package/src/resources/extensions/gsd/tests/auto-orchestrator.test.ts +133 -4
  296. package/src/resources/extensions/gsd/tests/auto-runtime-state.test.ts +34 -0
  297. package/src/resources/extensions/gsd/tests/canonical-milestone-root.test.ts +20 -0
  298. package/src/resources/extensions/gsd/tests/codebase-generator.test.ts +22 -0
  299. package/src/resources/extensions/gsd/tests/commands-dispatcher-workspace-git.test.ts +11 -0
  300. package/src/resources/extensions/gsd/tests/commands-verdict.test.ts +38 -1
  301. package/src/resources/extensions/gsd/tests/dispatch-complete-milestone-guard.test.ts +34 -3
  302. package/src/resources/extensions/gsd/tests/dispatch-run-uat-browser-tools.test.ts +88 -0
  303. package/src/resources/extensions/gsd/tests/doctor-scope-db-unavailable.test.ts +18 -0
  304. package/src/resources/extensions/gsd/tests/execute-task-rendering.test.ts +1 -0
  305. package/src/resources/extensions/gsd/tests/fixtures/pr-body/swarm-lane-no-blockers.md +1 -5
  306. package/src/resources/extensions/gsd/tests/fixtures/pr-body/swarm-lane-with-blockers.md +1 -5
  307. package/src/resources/extensions/gsd/tests/gate-state-canonicalization.test.ts +48 -1
  308. package/src/resources/extensions/gsd/tests/mcp-tool-name.test.ts +34 -0
  309. package/src/resources/extensions/gsd/tests/migration-auto-check.test.ts +58 -0
  310. package/src/resources/extensions/gsd/tests/milestone-closeout-proof.test.ts +99 -0
  311. package/src/resources/extensions/gsd/tests/milestone-closeout.test.ts +25 -0
  312. package/src/resources/extensions/gsd/tests/milestone-merge-transaction.test.ts +46 -0
  313. package/src/resources/extensions/gsd/tests/milestone-readiness.test.ts +65 -0
  314. package/src/resources/extensions/gsd/tests/milestone-validation-evidence.test.ts +41 -0
  315. package/src/resources/extensions/gsd/tests/milestone-validation-verdict.test.ts +55 -0
  316. package/src/resources/extensions/gsd/tests/plan-milestone.test.ts +45 -0
  317. package/src/resources/extensions/gsd/tests/plan-slice-prompt.test.ts +2 -0
  318. package/src/resources/extensions/gsd/tests/planning-crossval.test.ts +45 -0
  319. package/src/resources/extensions/gsd/tests/preferences-diagnostics.test.ts +67 -0
  320. package/src/resources/extensions/gsd/tests/preferences.test.ts +183 -0
  321. package/src/resources/extensions/gsd/tests/prompt-contracts.test.ts +46 -0
  322. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +9 -0
  323. package/src/resources/extensions/gsd/tests/provider-payload-policy.test.ts +165 -0
  324. package/src/resources/extensions/gsd/tests/pull-request-process.test.ts +47 -0
  325. package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +94 -0
  326. package/src/resources/extensions/gsd/tests/roadmap-parse-regression.test.ts +40 -0
  327. package/src/resources/extensions/gsd/tests/runtime-invariant-modules.test.ts +25 -1
  328. package/src/resources/extensions/gsd/tests/session-start-footer.test.ts +80 -0
  329. package/src/resources/extensions/gsd/tests/single-writer-invariant.test.ts +101 -1
  330. package/src/resources/extensions/gsd/tests/stale-queued-milestone.test.ts +27 -0
  331. package/src/resources/extensions/gsd/tests/start-auto-detached.test.ts +2 -1
  332. package/src/resources/extensions/gsd/tests/tool-availability-audit.test.ts +35 -0
  333. package/src/resources/extensions/gsd/tests/tool-naming.test.ts +35 -42
  334. package/src/resources/extensions/gsd/tests/uat-policy.test.ts +23 -0
  335. package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +47 -0
  336. package/src/resources/extensions/gsd/tests/user-input-boundary.test.ts +86 -1
  337. package/src/resources/extensions/gsd/tests/validate-milestone-stuck-guard.test.ts +39 -0
  338. package/src/resources/extensions/gsd/tests/web-app-uat.test.ts +150 -0
  339. package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +126 -9
  340. package/src/resources/extensions/gsd/tests/workspace-git-preflight.test.ts +15 -0
  341. package/src/resources/extensions/gsd/tests/worktree-manager.test.ts +21 -0
  342. package/src/resources/extensions/gsd/tests/worktree-projection-writers.test.ts +1 -1
  343. package/src/resources/extensions/gsd/tests/worktree-telemetry.test.ts +22 -0
  344. package/src/resources/extensions/gsd/tests/write-gate.test.ts +79 -0
  345. package/src/resources/extensions/gsd/tool-contract.ts +86 -8
  346. package/src/resources/extensions/gsd/tool-presentation-plan.ts +16 -33
  347. package/src/resources/extensions/gsd/tool-surface-snapshot.ts +47 -0
  348. package/src/resources/extensions/gsd/tools/plan-milestone.ts +19 -160
  349. package/src/resources/extensions/gsd/tools/reassess-roadmap.ts +43 -0
  350. package/src/resources/extensions/gsd/tools/validate-milestone.ts +25 -84
  351. package/src/resources/extensions/gsd/uat-policy.ts +19 -10
  352. package/src/resources/extensions/gsd/uat-run.ts +10 -14
  353. package/src/resources/extensions/gsd/unit-context-composer.ts +85 -20
  354. package/src/resources/extensions/gsd/unit-runtime.ts +3 -2
  355. package/src/resources/extensions/gsd/unit-tool-contracts.ts +2 -1
  356. package/src/resources/extensions/gsd/user-input-boundary.ts +18 -0
  357. package/src/resources/extensions/gsd/validation-block-guard.ts +2 -0
  358. package/src/resources/extensions/gsd/web-app-uat.ts +101 -0
  359. package/src/resources/extensions/gsd/workflow-mcp.ts +22 -110
  360. package/src/resources/extensions/gsd/workflow-reconcile.ts +3 -3
  361. package/src/resources/extensions/gsd/workflow-tool-surface.ts +73 -0
  362. package/src/resources/extensions/gsd/workspace-git-guard.ts +1 -0
  363. package/src/resources/extensions/gsd/worktree-lifecycle.ts +7 -16
  364. package/src/resources/extensions/gsd/worktree-state-projection.ts +55 -7
  365. package/src/resources/extensions/gsd/worktree-telemetry.ts +16 -0
  366. package/src/resources/extensions/shared/interview-ui.ts +15 -2
  367. package/src/resources/shared/claude-runtime-floor.ts +248 -0
  368. package/dist/web/standalone/.next/static/chunks/2659.feb6499ca863ebfc.js +0 -1
  369. package/dist/web/standalone/.next/static/chunks/2772.151789db0edea835.js +0 -1
  370. package/dist/web/standalone/.next/static/chunks/4283.10a065467b5340d8.js +0 -2
  371. package/dist/web/standalone/.next/static/chunks/5826.960dc4634cc9b0d3.js +0 -1
  372. package/dist/web/standalone/.next/static/chunks/796.46f811c0fac23aab.js +0 -10
  373. package/dist/web/standalone/.next/static/chunks/8785.d32f7a61f55c1600.js +0 -1
  374. /package/dist/web/standalone/.next/static/{Qbr81pQ-pbQXP4bq2VXLv → 3PtrU9qGPEXwNLWkIyiqk}/_buildManifest.js +0 -0
  375. /package/dist/web/standalone/.next/static/{Qbr81pQ-pbQXP4bq2VXLv → 3PtrU9qGPEXwNLWkIyiqk}/_ssgManifest.js +0 -0
@@ -40,12 +40,12 @@ import { listUnitRuntimeRecords, clearUnitRuntimeRecord, isInFlightRuntimePhase
40
40
  import { resolveExpectedArtifactPath } from "./auto.js";
41
41
  import { gsdHome } from "./gsd-home.js";
42
42
  import {
43
- gsdRoot, milestonesDir, resolveMilestoneFile, resolveMilestonePath,
43
+ gsdRoot, milestonesDir, resolveMilestoneFile,
44
44
  resolveSliceFile, resolveSlicePath, resolveGsdRootFile, relGsdRootFile,
45
- relMilestoneFile, relSliceFile, clearPathCache,
45
+ relMilestoneFile, relSliceFile,
46
46
  } from "./paths.js";
47
47
  import { join } from "node:path";
48
- import { readFileSync, existsSync, mkdirSync, readdirSync, rmSync, unlinkSync } from "node:fs";
48
+ import { readFileSync, existsSync, mkdirSync, readdirSync, rmSync } from "node:fs";
49
49
  import { readSessionLockData, isSessionLockProcessAlive } from "./session-lock.js";
50
50
  import { nativeAddAll, nativeCommit, nativeHasCommittedHead, nativeIsRepo, nativeInit } from "./native-git-bridge.js";
51
51
  import { isInheritedRepo } from "./repo-identity.js";
@@ -90,7 +90,7 @@ import {
90
90
  } from "./preparation.js";
91
91
  import { verifyExpectedArtifact } from "./auto-recovery.js";
92
92
  import type { MilestoneScope } from "./workspace.js";
93
- import { getPendingGate, extractDepthVerificationMilestoneId } from "./bootstrap/write-gate.js";
93
+ import { getPendingGate } from "./bootstrap/write-gate.js";
94
94
  import {
95
95
  _getPendingAutoStart,
96
96
  clearPendingAutoStart,
@@ -98,9 +98,14 @@ import {
98
98
  getDiscussionMilestoneId,
99
99
  hasPendingAutoStart,
100
100
  setPendingAutoStart,
101
- type PendingAutoStartEntry,
102
101
  } from "./pending-auto-start.js";
103
102
  import { clearGuidedUnitContext, setGuidedUnitContext } from "./guided-unit-context.js";
103
+ import { scheduleAutoStartAfterIdle } from "./discussion-handoff.js";
104
+ export {
105
+ maybeHandleEmptyIntentTurn,
106
+ maybeHandleReadyPhraseWithoutFiles,
107
+ resetEmptyTurnCounter,
108
+ } from "./guided-unit-completion.js";
104
109
 
105
110
  export {
106
111
  _getPendingAutoStart,
@@ -108,6 +113,7 @@ export {
108
113
  getDiscussionMilestoneId,
109
114
  setPendingAutoStart,
110
115
  } from "./pending-auto-start.js";
116
+ export { checkAutoStartAfterDiscuss } from "./discussion-handoff.js";
111
117
 
112
118
  export function shouldSkipGitBootstrapAfterInit(result: { gitEnabled?: boolean }): boolean {
113
119
  return result.gitEnabled === false;
@@ -125,7 +131,6 @@ export {
125
131
  buildExistingMilestonesContext,
126
132
  } from "./guided-flow-queue.js";
127
133
  import { logWarning } from "./workflow-logger.js";
128
- import { readManifest } from "./workflow-manifest.js";
129
134
  import { deleteRuntimeKv } from "./db/runtime-kv.js";
130
135
  import { PAUSED_SESSION_KV_KEY } from "./interrupted-session.js";
131
136
  import { buildWorkflowDispatchContent } from "./workflow-protocol.js";
@@ -141,32 +146,6 @@ export interface HeadlessMilestoneCreationOptions {
141
146
  startAutoAfterReady?: boolean;
142
147
  }
143
148
 
144
- type AutoStartOptions = Parameters<typeof startAutoDetached>[4];
145
- type AutoStartLauncher = typeof startAutoDetached;
146
-
147
- function scheduleAutoStartAfterIdle(
148
- ctx: ExtensionCommandContext,
149
- pi: ExtensionAPI,
150
- basePath: string,
151
- verboseMode: boolean,
152
- options?: AutoStartOptions,
153
- launch: AutoStartLauncher = startAutoDetached,
154
- ): void {
155
- const waitForIdle =
156
- typeof (ctx as { waitForIdle?: unknown }).waitForIdle === "function"
157
- ? ctx.waitForIdle.bind(ctx)
158
- : async () => {};
159
- void waitForIdle()
160
- .then(() => {
161
- setTimeout(() => launch(ctx, pi, basePath, verboseMode, options), 0);
162
- })
163
- .catch((err) => {
164
- const message = err instanceof Error ? err.message : String(err);
165
- ctx.ui.notify(`Auto-start failed while waiting for the prior turn to settle: ${message}`, "error");
166
- logWarning("guided", `auto-start idle wait failed: ${message}`);
167
- });
168
- }
169
-
170
149
  export const _scheduleAutoStartAfterIdleForTest = scheduleAutoStartAfterIdle;
171
150
 
172
151
  // ─── Scope-based validator wrappers ──────────────────────────────────────────
@@ -321,116 +300,6 @@ export function _roadmapHasParseableSlicesForTest(
321
300
  return parseRoadmapSlices(roadmapContent).length > 0;
322
301
  }
323
302
 
324
- function hasExecutablePlanForHandoff(milestoneId: string, roadmapFile: string | null): boolean {
325
- if (isDbAvailable()) {
326
- return getMilestoneSlices(milestoneId).length > 0;
327
- }
328
- if (!roadmapFile) return false;
329
- try {
330
- return parseRoadmapSlices(readFileSync(roadmapFile, "utf-8")).length > 0;
331
- } catch (e) {
332
- logWarning(
333
- "guided",
334
- `failed to parse roadmap slices for ${milestoneId}: ${(e as Error).message}`,
335
- );
336
- return false;
337
- }
338
- }
339
-
340
- function formatAcceptedDiscussHandoffMessage(
341
- milestoneId: string,
342
- contextFile: string | null,
343
- hasExecutablePlan: boolean,
344
- ): string {
345
- if (hasExecutablePlan) return `Milestone ${milestoneId} ready.`;
346
- if (contextFile) return `Milestone ${milestoneId} context captured. Continuing the planning pipeline.`;
347
- return `Milestone ${milestoneId} planning artifacts captured. Continuing the planning pipeline.`;
348
- }
349
-
350
- function manifestContainsMilestone(basePath: string, milestoneId: string): boolean {
351
- try {
352
- const manifest = readManifest(basePath);
353
- return (
354
- Array.isArray(manifest?.milestones) &&
355
- manifest.milestones.some(m => m.id === milestoneId)
356
- );
357
- } catch (e) {
358
- logWarning("guided", `R3b: failed to read state manifest: ${(e as Error).message}`);
359
- return false;
360
- }
361
- }
362
-
363
- function notifyDbRowRecoveryFailed(entry: PendingAutoStartEntry): void {
364
- entry.ctx.ui.notify(
365
- `Milestone ${entry.milestoneId}: DB row recovery failed ${entry.r3bRecoveryCount} times. ` +
366
- `Re-run /gsd to reset the recovery counter, or run /gsd-debug to diagnose without resetting.`,
367
- "error",
368
- );
369
- }
370
-
371
- function noteDbRowRecoveryMiss(entry: PendingAutoStartEntry): void {
372
- entry.r3bRecoveryCount += 1;
373
- if (entry.r3bRecoveryCount >= MAX_DB_ROW_RECOVERIES) {
374
- notifyDbRowRecoveryFailed(entry);
375
- }
376
- }
377
-
378
- function ensureMilestoneRowForAcceptedHandoff(
379
- entry: PendingAutoStartEntry,
380
- contextFile: string | null,
381
- ): boolean {
382
- if (!isDbAvailable()) return true;
383
-
384
- const { basePath, milestoneId } = entry;
385
- const milestoneRow = getMilestone(milestoneId);
386
- if (milestoneRow) return true;
387
-
388
- if (manifestContainsMilestone(basePath, milestoneId)) {
389
- logWarning(
390
- "guided",
391
- `R3b: getMilestone(${milestoneId}) returned null but manifest has the row — treating as stale read`,
392
- );
393
- return true;
394
- }
395
-
396
- if (!contextFile) {
397
- entry.ctx.ui.notify(
398
- `Milestone ${milestoneId}: discuss artifacts on disk but no DB row exists. ` +
399
- `PROJECT.md may have failed to register milestones. ` +
400
- `Re-save PROJECT.md with canonical "- [ ] M001: Title — One-liner" lines, ` +
401
- `then re-run /gsd to recover.`,
402
- "error",
403
- );
404
- return false;
405
- }
406
-
407
- if (entry.r3bRecoveryCount >= MAX_DB_ROW_RECOVERIES) {
408
- logWarning(
409
- "guided",
410
- `R3b: milestone ${milestoneId} DB-row recovery limit reached ` +
411
- `(${entry.r3bRecoveryCount}/${MAX_DB_ROW_RECOVERIES}); user already notified`,
412
- );
413
- return false;
414
- }
415
-
416
- logWarning(
417
- "guided",
418
- `R3b: ${milestoneId} has CONTEXT.md but no DB row — inserting placeholder "queued" row ` +
419
- `(attempt ${entry.r3bRecoveryCount + 1}/${MAX_DB_ROW_RECOVERIES})`,
420
- );
421
-
422
- try {
423
- insertMilestone({ id: milestoneId, title: milestoneId, status: "queued" });
424
- } catch (e) {
425
- logWarning("guided", `R3b: insertMilestone failed: ${(e as Error).message}`);
426
- }
427
-
428
- if (getMilestone(milestoneId)) return true;
429
-
430
- noteDbRowRecoveryMiss(entry);
431
- return false;
432
- }
433
-
434
303
  // ─── Commit Instruction Helpers ──────────────────────────────────────────────
435
304
 
436
305
  /** Build commit instruction for planning prompts. .gsd/ is managed externally and always gitignored. */
@@ -451,18 +320,6 @@ interface PendingDeepProjectSetupEntry {
451
320
  currentUnitId?: string;
452
321
  }
453
322
 
454
- // #4573: cap for how many times we nudge the LLM after a premature ready
455
- // phrase before giving up and asking the user to re-run /gsd.
456
- const MAX_READY_REJECTS = 2;
457
-
458
- // Cap failed in-flight DB row repair attempts before escalating to the user.
459
- const MAX_DB_ROW_RECOVERIES = 3;
460
-
461
- // #4573: matches the canonical ready phrase the discuss prompt asks the LLM
462
- // to emit. Accepts any M-prefixed milestone ID (three digits + optional
463
- // suffix) with optional trailing punctuation.
464
- const READY_PHRASE_RE = /\bMilestone\s+M\d{3}[A-Z0-9-]*\s+ready\.?/i;
465
-
466
323
  const pendingDeepProjectSetupMap = new Map<string, PendingDeepProjectSetupEntry>();
467
324
  const USER_DRIVEN_DEEP_SETUP_UNITS = new Set([
468
325
  "discuss-project",
@@ -687,405 +544,6 @@ async function dispatchNextDeepProjectSetupStage(entry: PendingDeepProjectSetupE
687
544
  return true;
688
545
  }
689
546
 
690
- /** Called from agent_end to check if auto-mode should start after discuss */
691
- export function checkAutoStartAfterDiscuss(lookupBasePath?: string): boolean {
692
- const entry = _getPendingAutoStart(lookupBasePath);
693
- if (!entry) return false;
694
-
695
- const { ctx, pi, basePath, milestoneId, step } = entry;
696
-
697
- // Gate 1: Primary milestone must have CONTEXT.md or ROADMAP.md
698
- // The "discuss" path creates CONTEXT.md; the "plan" path creates ROADMAP.md.
699
- // Use pinned scope (immune to cwd-drift) for existence checks.
700
- const contextFilePath = entry.scope.contextFile();
701
- const roadmapFilePath = entry.scope.roadmapFile();
702
- const contextFile = existsSync(contextFilePath) ? contextFilePath : null;
703
- const roadmapFile = existsSync(roadmapFilePath) ? roadmapFilePath : null;
704
- if (!contextFile && !roadmapFile) return false; // neither artifact yet — keep waiting
705
-
706
- // Gate 1a: a depth-verification gate is still pending for THIS milestone — the
707
- // LLM emitted the confirmation question (via ask_user_questions or plain chat)
708
- // but the user has not answered yet. Advancing now would skip the gate and
709
- // race ahead with unverified context.
710
- const basePathForGate = entry.scope.workspace.projectRoot;
711
- const pendingGateId = getPendingGate(basePathForGate);
712
- if (pendingGateId) {
713
- const pendingMilestoneId = extractDepthVerificationMilestoneId(pendingGateId);
714
- // Block advancement if the gate is for THIS milestone, OR if it's a
715
- // project/requirements gate (no milestone id encoded) for the deep setup flow.
716
- const isProjectGate =
717
- pendingGateId === "depth_verification_project_confirm" ||
718
- pendingGateId === "depth_verification_requirements_confirm" ||
719
- pendingGateId === "depth_verification_research_decision_confirm";
720
- if (pendingMilestoneId === milestoneId || isProjectGate) {
721
- return false;
722
- }
723
- }
724
-
725
- // Gate 1b: accept the in-flight discuss handoff. A queued DB row with pinned
726
- // CONTEXT.md is Discussion Complete, Planning Pending, not a plan-blocked
727
- // failure. If the row is missing, only this pending handoff may repair it.
728
- if (!ensureMilestoneRowForAcceptedHandoff(entry, contextFile)) return false;
729
-
730
- // Gate 2: Multi-milestone completeness warning
731
- // Parse PROJECT.md for milestone sequence, warn if any are missing context.
732
- // Don't block — milestones can be intentionally queued without context.
733
- const projectFile = resolveGsdRootFile(basePath, "PROJECT");
734
- let projectIds: string[] = [];
735
- if (projectFile) {
736
- try {
737
- const projectContent = readFileSync(projectFile, "utf-8");
738
- projectIds = parseMilestoneSequenceFromProject(projectContent);
739
- if (projectIds.length > 1) {
740
- const missing = projectIds.filter(id => {
741
- const hasContext = !!resolveMilestoneFile(basePath, id, "CONTEXT");
742
- const hasDraft = !!resolveMilestoneFile(basePath, id, "CONTEXT-DRAFT");
743
- const hasDir = existsSync(join(gsdRoot(basePath), "milestones", id));
744
- return !hasContext && !hasDraft && !hasDir;
745
- });
746
- if (missing.length > 0) {
747
- ctx.ui.notify(
748
- `Multi-milestone validation: ${missing.join(", ")} not found in filesystem. ` +
749
- `Discussion may not have completed all readiness gates.`,
750
- "warning",
751
- );
752
- }
753
- }
754
- } catch (e) { logWarning("guided", `PROJECT.md parsing failed: ${(e as Error).message}`); }
755
- }
756
-
757
- // Gate 3: Discussion manifest process verification (multi-milestone only)
758
- // The LLM writes DISCUSSION-MANIFEST.json after each Phase 3 gate decision.
759
- // When it exists, validate it before auto-starting. Project history alone is
760
- // not a reliable signal for the current discussion mode.
761
- const manifestPath = join(entry.scope.workspace.contract.projectGsd, "DISCUSSION-MANIFEST.json");
762
- if (existsSync(manifestPath)) {
763
- try {
764
- const manifest = JSON.parse(readFileSync(manifestPath, "utf-8"));
765
- const total = typeof manifest.total === "number" ? manifest.total : 0;
766
- const completed = typeof manifest.gates_completed === "number" ? manifest.gates_completed : 0;
767
-
768
- if (total > 1 && completed < total) {
769
- // Discussion not complete — block auto-start until all gates are done
770
- return false;
771
- }
772
-
773
- // Cross-check manifest milestones against PROJECT.md if available
774
- if (projectIds.length > 0) {
775
- const manifestIds = Object.keys(manifest.milestones ?? {});
776
- const untracked = projectIds.filter(id => !manifestIds.includes(id));
777
- if (untracked.length > 0) {
778
- ctx.ui.notify(
779
- `Discussion manifest missing gates for: ${untracked.join(", ")}`,
780
- "warning",
781
- );
782
- }
783
- }
784
- } catch (e) { logWarning("guided", `discussion manifest verification failed: ${(e as Error).message}`); }
785
- }
786
-
787
- // Draft promotion cleanup: if a CONTEXT-DRAFT.md exists alongside the new
788
- // CONTEXT.md, delete the draft — it's been consumed by the discussion.
789
- try {
790
- const draftFile = resolveMilestoneFile(basePath, milestoneId, "CONTEXT-DRAFT");
791
- if (draftFile) unlinkSync(draftFile);
792
- } catch (e) { logWarning("guided", `CONTEXT-DRAFT.md unlink failed: ${(e as Error).message}`); }
793
-
794
- // Cleanup: remove discussion manifest after auto-start (only needed during discussion)
795
- if (existsSync(manifestPath)) {
796
- try { unlinkSync(manifestPath); } catch (e) { logWarning("guided", `manifest unlink failed: ${(e as Error).message}`); }
797
- }
798
-
799
- deletePendingAutoStart(basePath);
800
- const hasExecutablePlan = hasExecutablePlanForHandoff(milestoneId, roadmapFile);
801
- ctx.ui.notify(
802
- formatAcceptedDiscussHandoffMessage(milestoneId, contextFile, hasExecutablePlan),
803
- "success",
804
- );
805
- if (entry.startAuto !== false) {
806
- scheduleAutoStartAfterIdle(ctx, pi, basePath, false, { step: step ?? true });
807
- }
808
- return true;
809
- }
810
-
811
- /**
812
- * Extract the concatenated text content from an assistant message, whether it
813
- * stores content as a string or as an array of text blocks.
814
- */
815
- function extractAssistantText(msg: any): string {
816
- if (!msg) return "";
817
- const content = msg.content;
818
- if (typeof content === "string") return content;
819
- if (!Array.isArray(content)) return "";
820
- const parts: string[] = [];
821
- for (const block of content) {
822
- if (!block || typeof block !== "object") continue;
823
- if (block.type === "text" && typeof block.text === "string") parts.push(block.text);
824
- }
825
- return parts.join("\n");
826
- }
827
-
828
- /**
829
- * Return true if the assistant message contains any tool-use block.
830
- *
831
- * The canonical pi-ai `AssistantMessage.content` (see packages/pi-ai/src/types.ts)
832
- * uses `type: "toolCall"` and `type: "serverToolUse"` for tool invocations —
833
- * every provider (anthropic-direct, claude-code-cli, openai, etc.) normalizes
834
- * incoming tool blocks into these two shapes before they reach guided-flow.
835
- *
836
- * The Anthropic API wire shape `"tool_use"` / `"server_tool_use"` does NOT appear
837
- * in the internal AssistantMessage — those literals are only used when sending
838
- * messages back out to the Anthropic API. Matching them here was a latent bug:
839
- * `hasToolUse` returned `false` for every real tool call, which let the
840
- * empty-turn nudge fire and pre-empt MCP tools that block on the user
841
- * (e.g. `ask_user_questions`). See investigation in PR for #4658.
842
- */
843
- function hasToolUse(msg: any): boolean {
844
- if (!msg) return false;
845
- const content = msg.content;
846
- if (!Array.isArray(content)) return false;
847
- return content.some(
848
- (b: any) =>
849
- b &&
850
- typeof b === "object" &&
851
- (b.type === "toolCall" || b.type === "serverToolUse"),
852
- );
853
- }
854
-
855
- /**
856
- * #4573 — Detect and recover from the "ready phrase without files" failure mode.
857
- *
858
- * When the LLM emits "Milestone {{id}} ready." but has not written the
859
- * milestone CONTEXT/ROADMAP artifacts, `checkAutoStartAfterDiscuss()` silently
860
- * returns false and the next /gsd invocation loops into the "All milestones
861
- * complete" warning.
862
- *
863
- * This function, called from `handleAgentEnd` after `checkAutoStartAfterDiscuss`
864
- * returns false, pattern-matches the ready phrase on the last assistant message.
865
- * If it fired AND neither the canonical M###-CONTEXT.md/M###-ROADMAP.md nor
866
- * legacy CONTEXT.md/ROADMAP.md files exist, it:
867
- * 1. Notifies the user that the signal was rejected.
868
- * 2. Injects a system message via `pi.sendMessage(..., {triggerTurn:true})`
869
- * telling the LLM the signal was premature and to emit the writes now.
870
- * 3. Caps at `MAX_READY_REJECTS` per-entry; beyond that, gives up and asks
871
- * the user to re-run /gsd.
872
- *
873
- * Returns true when a nudge (or give-up) was emitted, signaling the caller to
874
- * skip `resolveAgentEnd`.
875
- */
876
- export function maybeHandleReadyPhraseWithoutFiles(event: { messages: any[] }, lookupBasePath?: string): boolean {
877
- const entry = _getPendingAutoStart(lookupBasePath);
878
- if (!entry) return false;
879
- const { ctx, pi, basePath, milestoneId } = entry;
880
-
881
- // Gate: last assistant message must contain the ready phrase
882
- const lastMsg = event.messages[event.messages.length - 1];
883
- const text = extractAssistantText(lastMsg);
884
- if (!READY_PHRASE_RE.test(text)) return false;
885
-
886
- // Bust paths.ts cached dir listings before checking for fresh writes. The
887
- // LLM's Write tool calls do not invalidate paths.ts caches, so a stale
888
- // listing taken before the milestone dir or its CONTEXT/ROADMAP files
889
- // existed would falsely report the artifacts as missing and trigger the
890
- // 3-strike "ready without files" abort even though the writes succeeded.
891
- clearPathCache();
892
-
893
- // Gate: artifacts must still be missing — if they exist, the happy path
894
- // already fired and we have nothing to do.
895
- const contextFile = resolveMilestoneFile(basePath, milestoneId, "CONTEXT");
896
- const roadmapFile = resolveMilestoneFile(basePath, milestoneId, "ROADMAP");
897
- if (contextFile || roadmapFile) return false;
898
-
899
- // Diagnostic: when the cached resolver reports both files missing, also probe
900
- // the canonical paths with uncached existsSync so we can tell whether the
901
- // recovery is firing on real-missing files or a path-resolution miss
902
- // (basePath/symlink mismatch, stale cache despite agent-end-recovery flush,
903
- // legacy descriptor dir not matching, etc.).
904
- try {
905
- const mDir = resolveMilestonePath(basePath, milestoneId);
906
- const canonicalCtx = mDir ? join(mDir, `${milestoneId}-CONTEXT.md`) : null;
907
- const canonicalRoadmap = mDir ? join(mDir, `${milestoneId}-ROADMAP.md`) : null;
908
- logWarning(
909
- "guided",
910
- `ready-phrase-reject diagnostic mid=${milestoneId} basePath=${basePath} ` +
911
- `mDir=${mDir ?? "null"} ` +
912
- `canonical-ctx=${canonicalCtx ?? "null"} ctx-exists=${canonicalCtx ? existsSync(canonicalCtx) : "n/a"} ` +
913
- `canonical-roadmap=${canonicalRoadmap ?? "null"} roadmap-exists=${canonicalRoadmap ? existsSync(canonicalRoadmap) : "n/a"}`,
914
- );
915
- } catch (e) {
916
- logWarning("guided", `ready-phrase-reject diagnostic failed: ${(e as Error).message}`);
917
- }
918
-
919
- entry.readyRejectCount = (entry.readyRejectCount ?? 0) + 1;
920
-
921
- if (entry.readyRejectCount > MAX_READY_REJECTS) {
922
- // Give up: clear state and tell the user to re-run /gsd. Avoids an
923
- // infinite nudge loop when the LLM never produces the writes.
924
- deletePendingAutoStart(basePath);
925
- ctx.ui.notify(
926
- `Milestone ${milestoneId}: LLM signaled "ready" ${entry.readyRejectCount} times without writing files. ` +
927
- `Stopping auto-nudge. Run /gsd to try again.`,
928
- "error",
929
- );
930
- return true;
931
- }
932
-
933
- const contextRel = relMilestoneFile(basePath, milestoneId, "CONTEXT");
934
- const roadmapRel = relMilestoneFile(basePath, milestoneId, "ROADMAP");
935
- ctx.ui.notify(
936
- `Milestone ${milestoneId}: "ready" signal rejected — ${contextRel} and ${roadmapRel} are missing. Asking the LLM to complete the writes.`,
937
- "warning",
938
- );
939
-
940
- const nudge =
941
- `You emitted "Milestone ${milestoneId} ready." but neither ` +
942
- `${contextRel} nor ${roadmapRel} exists on disk. ` +
943
- `The ready phrase is a POST-WRITE signal and has been rejected. ` +
944
- `In this turn: (1) write PROJECT.md, REQUIREMENTS.md, and the milestone ` +
945
- `CONTEXT.md, (2) call gsd_plan_milestone, then (3) emit the ready phrase. ` +
946
- `Do not describe these steps — execute them as tool calls. ` +
947
- `This is retry ${entry.readyRejectCount}/${MAX_READY_REJECTS}; further ` +
948
- `premature signals will clear the session.`;
949
-
950
- try {
951
- pi.sendMessage(
952
- { customType: "gsd-ready-no-files", content: nudge, display: false },
953
- { triggerTurn: true },
954
- );
955
- } catch (e) {
956
- logWarning("guided", `ready-phrase nudge sendMessage failed: ${(e as Error).message}`);
957
- return false;
958
- }
959
- return true;
960
- }
961
-
962
- /**
963
- * #4573 — Detect and recover from the "announces tool, never calls it" stall.
964
- *
965
- * The LLM emits text like "I'll now write the CONTEXT.md file" but the turn
966
- * ends with zero tool-use blocks. The harness has no post-turn tool-call
967
- * validation, so the unit promise resolves and the user sees a stalled state.
968
- *
969
- * This function, called from `handleAgentEnd`, inspects the last assistant
970
- * message. If ALL of the following are true, it injects a recovery message:
971
- * - Text-only (no tool-use blocks)
972
- * - Contains a commit-intent phrase ("I'll write", "I'll call", etc.)
973
- * - Auto-mode is active OR a discussion autostart is pending
974
- * - `emptyTurnRetryCount` is under the cap
975
- *
976
- * Per-handler state is held on the `PendingAutoStartEntry` when present, and
977
- * on a module-level map otherwise. The counter resets on any successful
978
- * tool-use turn via `resetEmptyTurnCounter`.
979
- */
980
- const emptyTurnCounterByBase = new Map<string, number>();
981
- const MAX_EMPTY_TURN_RETRIES = 2;
982
-
983
- // Phrases that indicate the LLM is about to do something but has not yet.
984
- // Kept tight to avoid flagging legitimate narration like "I'll wait for your answer."
985
- //
986
- // "make" was previously in the verb list but matches conversational meta phrases
987
- // like "Let me make sure I understand…" which are NOT action announcements —
988
- // removed to prevent the empty-turn nudge from auto-replying to user questions
989
- // in discuss flows.
990
- const COMMIT_INTENT_RE =
991
- /\b(?:I['’]ll|I will|Next,? I['’]ll|Now I['’]ll|Let me|I['’]m going to|I am going to)\s+(?:now\s+)?(?:write|create|call|invoke|update|add|run|execute|generate|produce|emit|compose|implement|save|apply|commit)\b/i;
992
-
993
- /**
994
- * Reset the empty-turn counter for a basePath after a successful tool-use turn.
995
- * Called from handleAgentEnd when the last message contains tool_use blocks.
996
- */
997
- export function resetEmptyTurnCounter(basePath?: string): void {
998
- if (basePath) emptyTurnCounterByBase.delete(basePath);
999
- else emptyTurnCounterByBase.clear();
1000
- }
1001
-
1002
- export function maybeHandleEmptyIntentTurn(
1003
- event: { messages: any[] },
1004
- isAuto: boolean,
1005
- lookupBasePath?: string,
1006
- ): boolean {
1007
- // Gate: only fire when there is system-driven work in flight. Interactive
1008
- // /gsd discuss (user-driven) produces legitimate text-only turns.
1009
- if (!isAuto && !hasPendingAutoStart(lookupBasePath)) return false;
1010
-
1011
- const lastMsg = event.messages[event.messages.length - 1];
1012
- if (!lastMsg) return false;
1013
- if (hasToolUse(lastMsg)) return false;
1014
-
1015
- const text = extractAssistantText(lastMsg).trim();
1016
- if (!text) return false;
1017
-
1018
- // Skip if the LLM is emitting the ready phrase — that is the ready-no-files
1019
- // path, handled by maybeHandleReadyPhraseWithoutFiles.
1020
- if (READY_PHRASE_RE.test(text)) return false;
1021
-
1022
- // Skip if the LLM is clearly handing back to the user. Discuss flows
1023
- // often pose a question and follow it with a conditional intent on the
1024
- // same line ("Did I capture that correctly? If so, I'll write the
1025
- // requirements."). A line-trailing `?` check misses these because the
1026
- // line ends in `.`. Match any sentence-terminating `?` (followed by
1027
- // whitespace or end-of-text) — false negatives here auto-reply to the
1028
- // user, which is a much worse failure mode than a missed nudge.
1029
- if (/\?(?:\s|$)/.test(text)) return false;
1030
-
1031
- // Must contain a commit-intent phrase — this is the stall we care about.
1032
- if (!COMMIT_INTENT_RE.test(text)) return false;
1033
-
1034
- // Resolve the target basePath + pi for injection. Prefer the pending
1035
- // autostart entry (discuss flow); otherwise we cannot inject.
1036
- const entry = _getPendingAutoStart(lookupBasePath);
1037
- if (!entry) return false;
1038
- const { ctx, pi, basePath } = entry;
1039
-
1040
- const count = (emptyTurnCounterByBase.get(basePath) ?? 0) + 1;
1041
- emptyTurnCounterByBase.set(basePath, count);
1042
-
1043
- if (count > MAX_EMPTY_TURN_RETRIES) {
1044
- ctx.ui.notify(
1045
- `Empty-turn recovery: LLM announced intent ${count} times without calling any tool. ` +
1046
- `Stopping auto-nudge.`,
1047
- "error",
1048
- );
1049
- return false; // let the normal flow resolve/pause the unit
1050
- }
1051
-
1052
- ctx.ui.notify(
1053
- `Empty-turn detected: LLM announced intent but called no tool. Prompting it to execute.`,
1054
- "info",
1055
- );
1056
-
1057
- const nudge =
1058
- `Your last turn announced an action (e.g. "I'll write…" or "Let me call…") ` +
1059
- `but contained no tool call. The system records zero tool-use blocks for ` +
1060
- `that turn. Execute the announced action NOW as a tool call in this turn. ` +
1061
- `Do not describe it again. Retry ${count}/${MAX_EMPTY_TURN_RETRIES}.`;
1062
-
1063
- try {
1064
- pi.sendMessage(
1065
- { customType: "gsd-empty-turn-recovery", content: nudge, display: false },
1066
- { triggerTurn: true },
1067
- );
1068
- } catch (e) {
1069
- logWarning("guided", `empty-turn nudge sendMessage failed: ${(e as Error).message}`);
1070
- return false;
1071
- }
1072
- return true;
1073
- }
1074
-
1075
- /**
1076
- * Extract milestone IDs from PROJECT.md milestone sequence table.
1077
- * Looks for rows like "| M001 | Name | Status |" and extracts the ID column.
1078
- */
1079
- function parseMilestoneSequenceFromProject(content: string): string[] {
1080
- const ids: string[] = [];
1081
- const lines = content.split(/\r?\n/);
1082
- for (const line of lines) {
1083
- const match = line.match(/^\|\s*(M\d{3}[A-Z0-9-]*)\s*\|/);
1084
- if (match) ids.push(match[1]);
1085
- }
1086
- return ids;
1087
- }
1088
-
1089
547
  // ─── Types ────────────────────────────────────────────────────────────────────
1090
548
 
1091
549
  type UIContext = ExtensionContext;
@@ -2367,11 +1825,42 @@ export async function showSmartEntry(
2367
1825
  const { checkMarkdownHierarchyAgainstDb } = await import("./migration-auto-check.js");
2368
1826
  const result = await checkMarkdownHierarchyAgainstDb(basePath);
2369
1827
  if (result.action === "recovery-required") {
2370
- ctx.ui.notify(
2371
- result.message ??
2372
- `Markdown planning artifacts do not match the authoritative DB. Run \`${result.recoveryCommand ?? "/gsd recover --confirm"}\` to import markdown explicitly.`,
2373
- "warning",
2374
- );
1828
+ if (result.recoveryCommand === "/gsd rebuild markdown") {
1829
+ try {
1830
+ const { rebuildMarkdownProjectionsFromDb } = await import("./commands-maintenance.js");
1831
+ const rebuild = await rebuildMarkdownProjectionsFromDb(basePath);
1832
+ const after = await checkMarkdownHierarchyAgainstDb(basePath);
1833
+ if (after.action === "none") {
1834
+ ctx.ui.notify(
1835
+ `Self-heal: rebuilt markdown projections from the authoritative DB ` +
1836
+ `(${rebuild.rendered} rendered${rebuild.errors.length > 0 ? `, ${rebuild.errors.length} error(s)` : ""}).`,
1837
+ rebuild.errors.length > 0 ? "warning" : "info",
1838
+ );
1839
+ } else {
1840
+ ctx.ui.notify(
1841
+ (result.message ?? "Markdown planning artifacts still diverge from the DB after auto-rebuild.") +
1842
+ (rebuild.errors.length > 0
1843
+ ? `\nAuto-rebuild had ${rebuild.errors.length} projection error(s). Run \`/gsd rebuild markdown\` after review.`
1844
+ : ""),
1845
+ "warning",
1846
+ );
1847
+ }
1848
+ } catch (rebuildErr) {
1849
+ const rebuildMessage = rebuildErr instanceof Error ? rebuildErr.message : String(rebuildErr);
1850
+ logWarning("guided", `markdown auto-rebuild failed: ${rebuildMessage}`, { file: "guided-flow.ts" });
1851
+ ctx.ui.notify(
1852
+ result.message ??
1853
+ `Markdown planning artifacts do not match the authoritative DB. Run \`${result.recoveryCommand ?? "/gsd rebuild markdown"}\` to re-project from the DB.`,
1854
+ "warning",
1855
+ );
1856
+ }
1857
+ } else {
1858
+ ctx.ui.notify(
1859
+ result.message ??
1860
+ `Markdown planning artifacts do not match the authoritative DB. Run \`${result.recoveryCommand ?? "/gsd recover --confirm"}\` to import markdown explicitly.`,
1861
+ "warning",
1862
+ );
1863
+ }
2375
1864
  }
2376
1865
  } catch (err) {
2377
1866
  const message = err instanceof Error ? err.message : String(err);