@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
@@ -20,6 +20,8 @@ export interface CliFlags {
20
20
  webPort?: number;
21
21
  /** Additional allowed origins for CORS: `--allowed-origins http://192.168.1.10:8080` */
22
22
  webAllowedOrigins?: string[];
23
+ /** Disable the web launcher's bearer token gate: `--no-auth` */
24
+ webNoAuth?: boolean;
23
25
  /** Set by `gsd sessions` when the user picks a specific session to resume */
24
26
  _selectedSessionPath?: string;
25
27
  }
@@ -52,6 +52,9 @@ export function parseCliArgs(argv) {
52
52
  const origins = args[++i].split(',').map(o => o.trim()).filter(Boolean);
53
53
  flags.webAllowedOrigins = (flags.webAllowedOrigins ?? []).concat(origins);
54
54
  }
55
+ else if (arg === '--no-auth') {
56
+ flags.webNoAuth = true;
57
+ }
55
58
  else if (arg === '--model' && i + 1 < args.length) {
56
59
  flags.model = args[++i];
57
60
  }
@@ -232,14 +235,18 @@ export async function runWebCliBranch(flags, deps = {}) {
232
235
  const agentDir = deps.agentDir ?? defaultAgentDir;
233
236
  const projectSessionsDir = getProjectSessionsDir(currentCwd, baseSessionsDir);
234
237
  migrateLegacyFlatSessions(baseSessionsDir, projectSessionsDir);
235
- const status = await (deps.runWebMode ?? launchWebMode)({
238
+ const launchOptions = {
236
239
  cwd: currentCwd,
237
240
  projectSessionsDir,
238
241
  agentDir,
239
242
  host: flags.webHost,
240
243
  port: flags.webPort,
241
244
  allowedOrigins: flags.webAllowedOrigins,
242
- });
245
+ };
246
+ if (flags.webNoAuth !== undefined) {
247
+ launchOptions.noAuth = flags.webNoAuth;
248
+ }
249
+ const status = await (deps.runWebMode ?? launchWebMode)(launchOptions);
243
250
  if (!status.ok) {
244
251
  emitWebModeFailure(stderr, status);
245
252
  }
package/dist/help-text.js CHANGED
@@ -189,6 +189,11 @@ export function printHelp(version) {
189
189
  process.stdout.write(' --worktree, -w [name] Start in an isolated worktree (auto-named if omitted)\n');
190
190
  process.stdout.write(' --model <id> Override model (e.g. provider/model-id)\n');
191
191
  process.stdout.write(' --no-session Disable session persistence\n');
192
+ process.stdout.write(' --web [path] Start browser-only web mode\n');
193
+ process.stdout.write(' --host <host> Web mode bind address\n');
194
+ process.stdout.write(' --port <port> Web mode port\n');
195
+ process.stdout.write(' --allowed-origins <csv> Additional allowed web origins\n');
196
+ process.stdout.write(' --no-auth Disable web token auth; requires external access control\n');
192
197
  process.stdout.write(' --extension <path> Load additional extension\n');
193
198
  process.stdout.write(' --tools <a,b,c> Restrict available tools\n');
194
199
  process.stdout.write(' --list-models [search] List available models and exit\n');
@@ -1 +1 @@
1
- 83d1de3e3261d473
1
+ 9a23824a70bd2722
@@ -114,6 +114,61 @@ function errorResult(message, questions = [], options) {
114
114
  details: { questions, response: null, cancelled: true, interrupted: options?.interrupted === true },
115
115
  };
116
116
  }
117
+ /** Parse the LLM-facing JSON payload back into a RoundResult for TUI rendering. */
118
+ function parseLlmAnswersContent(text) {
119
+ try {
120
+ const parsed = JSON.parse(text);
121
+ if (!parsed?.answers || typeof parsed.answers !== "object")
122
+ return null;
123
+ const answers = {};
124
+ for (const [id, entry] of Object.entries(parsed.answers)) {
125
+ const list = Array.isArray(entry?.answers) ? [...entry.answers] : [];
126
+ let notes = "";
127
+ const noteIdx = list.findIndex((item) => typeof item === "string" && item.startsWith("user_note:"));
128
+ if (noteIdx >= 0) {
129
+ notes = list.splice(noteIdx, 1)[0].replace(/^user_note:\s*/, "");
130
+ }
131
+ if (list.length === 0)
132
+ continue;
133
+ answers[id] = {
134
+ selected: list.length === 1 ? list[0] : list,
135
+ notes,
136
+ };
137
+ }
138
+ if (Object.keys(answers).length === 0)
139
+ return null;
140
+ return { endInterview: false, answers };
141
+ }
142
+ catch {
143
+ return null;
144
+ }
145
+ }
146
+ function isCancelledResultContent(text) {
147
+ return /\bwas cancelled before receiving a response\b/i.test(text)
148
+ || /\bwas interrupted before receiving a response\b/i.test(text);
149
+ }
150
+ function renderAnswerLines(questions, response, theme) {
151
+ const lines = [];
152
+ for (const q of questions) {
153
+ const answer = response.answers[q.id];
154
+ if (!answer) {
155
+ lines.push(`${theme.fg("accent", q.header)}: ${theme.fg("dim", "(no answer)")}`);
156
+ continue;
157
+ }
158
+ const selected = answer.selected;
159
+ const notes = answer.notes;
160
+ const multiSel = !!q.allowMultiple;
161
+ const answerText = multiSel && Array.isArray(selected)
162
+ ? selected.join(", ")
163
+ : (Array.isArray(selected) ? selected[0] : selected) ?? "(no answer)";
164
+ let line = `${theme.fg("success", "✓ ")}${theme.fg("accent", q.header)}: ${answerText}`;
165
+ if (notes) {
166
+ line += ` ${theme.fg("muted", `[note: ${notes}]`)}`;
167
+ }
168
+ lines.push(line);
169
+ }
170
+ return new Text(lines.join("\n"), 0, 0);
171
+ }
117
172
  /** Convert the shared RoundResult into the JSON the LLM expects. */
118
173
  function formatForLLM(result) {
119
174
  const answers = {};
@@ -318,11 +373,17 @@ export default function AskUserQuestions(pi) {
318
373
  }
319
374
  return new Text(text, 0, 0);
320
375
  },
321
- renderResult(result, _options, theme) {
376
+ renderResult(result, _options, theme, context) {
322
377
  const details = result.details;
378
+ const contentText = result.content[0]?.type === "text" ? result.content[0].text : "";
379
+ const questionsFromArgs = context?.args?.questions;
380
+ const parsedResponse = contentText ? parseLlmAnswersContent(contentText) : null;
323
381
  if (!details) {
324
- const text = result.content[0];
325
- return new Text(text?.type === "text" ? text.text : "", 0, 0);
382
+ const questions = questionsFromArgs;
383
+ if (questions?.length && parsedResponse) {
384
+ return renderAnswerLines(questions, parsedResponse, theme);
385
+ }
386
+ return new Text(contentText, 0, 0);
326
387
  }
327
388
  // Remote channel result (discriminated on details.remote === true)
328
389
  if (details.remote) {
@@ -352,30 +413,24 @@ export default function AskUserQuestions(pi) {
352
413
  }
353
414
  return new Text(lines.join("\n"), 0, 0);
354
415
  }
355
- if (details.cancelled || !details.response) {
416
+ const questions = details.questions ?? questionsFromArgs;
417
+ const response = details.response ?? parsedResponse;
418
+ const explicitCancel = details.cancelled === true || isCancelledResultContent(contentText);
419
+ if (explicitCancel && !response) {
356
420
  const interrupted = "interrupted" in details && details.interrupted === true;
357
421
  return new Text(theme.fg("warning", interrupted ? "Interrupted" : "Cancelled"), 0, 0);
358
422
  }
359
- const lines = [];
360
- for (const q of details.questions) {
361
- const answer = details.response.answers[q.id];
362
- if (!answer) {
363
- lines.push(`${theme.fg("accent", q.header)}: ${theme.fg("dim", "(no answer)")}`);
364
- continue;
365
- }
366
- const selected = answer.selected;
367
- const notes = answer.notes;
368
- const multiSel = !!q.allowMultiple;
369
- const answerText = multiSel && Array.isArray(selected)
370
- ? selected.join(", ")
371
- : (Array.isArray(selected) ? selected[0] : selected) ?? "(no answer)";
372
- let line = `${theme.fg("success", "✓ ")}${theme.fg("accent", q.header)}: ${answerText}`;
373
- if (notes) {
374
- line += ` ${theme.fg("muted", `[note: ${notes}]`)}`;
375
- }
376
- lines.push(line);
423
+ if (questions?.length && response) {
424
+ return renderAnswerLines(questions, response, theme);
425
+ }
426
+ if (contentText) {
427
+ return new Text(contentText, 0, 0);
428
+ }
429
+ if (details.cancelled || !details.response) {
430
+ const interrupted = "interrupted" in details && details.interrupted === true;
431
+ return new Text(theme.fg("warning", interrupted ? "Interrupted" : "Cancelled"), 0, 0);
377
432
  }
378
- return new Text(lines.join("\n"), 0, 0);
433
+ return new Text("", 0, 0);
379
434
  },
380
435
  });
381
436
  }
@@ -14,13 +14,18 @@ import { existsSync, readFileSync } from "node:fs";
14
14
  import { homedir } from "node:os";
15
15
  import { createRequire } from "node:module";
16
16
  import { dirname, join } from "node:path";
17
- import { PartialMessageBuilder, ZERO_USAGE, mapUsage } from "./partial-builder.js";
17
+ import { ZERO_USAGE, mapUsage } from "./partial-builder.js";
18
+ import { attachExternalResultsToToolBlocks, buildFinalAssistantContent, extractToolResultsFromSdkUserMessage, handleClaudeCodePartialStreamEvent, } from "./turn-assembler.js";
18
19
  import { buildWorkflowMcpServers, getRequiredWorkflowToolsForAutoUnit, resolveWorkflowMcpProjectRoot, } from "../gsd/workflow-mcp.js";
20
+ import { resolveWorkflowQuestionToolSurface } from "../gsd/question-transport.js";
19
21
  import { buildProjectGsdMcpServers, ensureProjectWorkflowMcpConfig } from "../gsd/mcp-project-config.js";
20
22
  import { loadProjectGSDPreferences } from "../gsd/preferences.js";
23
+ import { markToolStart, markToolEnd } from "../gsd/auto.js";
24
+ import { markInteractiveElicitationStart, markInteractiveElicitationEnd, } from "../gsd/auto-tool-tracking.js";
21
25
  import { discoverBrowserMcpServerName, discoverMcpServers, discoverMcpServerNames, discoverUserMcpServerNames, discoverWorkflowMcpServerName, computeMcpDisallowedTools, } from "../gsd/mcp-filter.js";
22
26
  import { RUN_UAT_CLAUDE_NATIVE_TOOL_NAMES, RUN_UAT_FORBIDDEN_TOOL_NAMES, RUN_UAT_WORKFLOW_TOOL_NAMES, resolveToolPresentationPlan } from "../gsd/tool-presentation-plan.js";
23
27
  import { showInterviewRound } from "../shared/tui.js";
28
+ export { buildFinalAssistantContent, extractToolResultsFromSdkUserMessage, handleClaudeCodePartialStreamEvent, mergePendingToolCalls, } from "./turn-assembler.js";
24
29
  export function serverToolUseToToolCallLike(block) {
25
30
  const argumentsValue = block.input && typeof block.input === "object" && !Array.isArray(block.input)
26
31
  ? block.input
@@ -1078,7 +1083,26 @@ export function createClaudeCodeCanUseToolHandler(ui) {
1078
1083
  // ---------------------------------------------------------------------------
1079
1084
  // Elicitation handler
1080
1085
  // ---------------------------------------------------------------------------
1081
- /** Create an SDK elicitation handler that routes requests through the extension UI dialogs, or undefined if no UI is available. */
1086
+ /**
1087
+ * Create an SDK elicitation handler that routes requests through the extension UI dialogs, or undefined if no UI is available.
1088
+ *
1089
+ * For structured (AskUserQuestion) elicitations, the interview round's result
1090
+ * disambiguates two cases that must not be conflated: an `undefined` result
1091
+ * means the custom UI is unavailable, so we fall back to the simpler `select`
1092
+ * dialogs; an empty-answers result means the user dismissed the question, which
1093
+ * is treated as a clean cancel. Falling back to dialogs on dismissal would
1094
+ * re-ask the same questions (the duplicate-question bug).
1095
+ */
1096
+ /**
1097
+ * Monotonic counter so concurrent/sequential elicitations resolved within the
1098
+ * same millisecond get distinct synthetic in-flight-tool ids (the `cc-elicit-*`
1099
+ * namespace never collides with real MCP toolCallIds).
1100
+ */
1101
+ let _elicitationSeq = 0;
1102
+ function nextElicitationSeq() {
1103
+ _elicitationSeq = (_elicitationSeq + 1) % Number.MAX_SAFE_INTEGER;
1104
+ return _elicitationSeq;
1105
+ }
1082
1106
  export function createClaudeCodeElicitationHandler(ui) {
1083
1107
  if (!ui)
1084
1108
  return undefined;
@@ -1091,18 +1115,64 @@ export function createClaudeCodeElicitationHandler(ui) {
1091
1115
  const headlessAnswer = answerElicitationFromHeadlessAnswers(questions, loadHeadlessAnswers());
1092
1116
  if (headlessAnswer)
1093
1117
  return headlessAnswer;
1094
- const interviewResult = await showInterviewRound(questions, { signal }, { ui }).catch(() => undefined);
1095
- if (interviewResult && Object.keys(interviewResult.answers).length > 0) {
1118
+ // The SDK elicitation blocks waiting for human input, but it is not an
1119
+ // MCP tool dispatch, so markToolStart/markToolEnd are never called for
1120
+ // it. Without this the soft/context/idle/hard watchdogs see zero
1121
+ // in-flight tools and re-dispatch (and ultimately abort) the agent
1122
+ // turn hosting this elicitation, tearing the question down (#2676 /
1123
+ // claude-code-cli self-cancel loop). Bracketing the human wait with
1124
+ // the s.active-gated interactive-tool guard makes it visible to
1125
+ // hasInteractiveToolInFlight()/getInFlightToolCount() so those
1126
+ // watchdogs exempt it. No-op outside auto-mode (wrapper self-gates).
1127
+ //
1128
+ // markInteractiveElicitationStart/End is a SEPARATE, ungated signal that
1129
+ // is observable in FOREGROUND (where the s.active-gated markToolStart is a
1130
+ // no-op). The foreground approval-gate pause path (register-hooks
1131
+ // message_update) consults isInteractiveElicitationInFlight() and bails so
1132
+ // it does not tear down the very elicitation that IS the human boundary
1133
+ // (#cc-elicitation-self-cancel). It clears in the finally on every exit.
1134
+ const elicId = "cc-elicit-" + (request.id ?? `${Date.now()}-${nextElicitationSeq()}`);
1135
+ markInteractiveElicitationStart();
1136
+ markToolStart(elicId, "ask_user_questions");
1137
+ try {
1138
+ const interviewResult = await showInterviewRound(questions, { signal, overlay: true }, { ui }).catch(() => undefined);
1139
+ if (interviewResult === undefined) {
1140
+ // `await` so the dialog human-wait stays inside try/finally and the
1141
+ // in-flight guard is held until the dialog resolves. Without it,
1142
+ // `finally` runs the moment the promise is created and the fallback
1143
+ // wait runs with zero in-flight tools — reintroducing the
1144
+ // self-cancel on this path (Bugbot #1c00624d).
1145
+ return await promptElicitationWithDialogs(request, questions, ui, signal);
1146
+ }
1147
+ if (Object.keys(interviewResult.answers).length === 0) {
1148
+ // A system/host teardown (compaction, session_switch, true
1149
+ // interrupt) that aborted the signal mid-wait sets `interrupted`.
1150
+ // Surface that as a non-affirmative `decline` so it is not
1151
+ // laundered into a clean user-declined `cancel` the model re-asks
1152
+ // against. A genuine user dismissal leaves `interrupted` falsy and
1153
+ // keeps the prior `cancel` semantics.
1154
+ return interviewResult.interrupted ? { action: "decline" } : { action: "cancel" };
1155
+ }
1096
1156
  return {
1097
1157
  action: "accept",
1098
1158
  content: roundResultToElicitationContent(questions, interviewResult),
1099
1159
  };
1100
1160
  }
1101
- return promptElicitationWithDialogs(request, questions, ui, signal);
1161
+ finally {
1162
+ markToolEnd(elicId);
1163
+ markInteractiveElicitationEnd();
1164
+ }
1102
1165
  }
1103
1166
  const textFields = parseTextInputElicitation(request);
1104
1167
  if (textFields) {
1105
- return promptTextInputElicitation(request, textFields, ui, signal);
1168
+ const elicId = "cc-elicit-" + (request.id ?? `${Date.now()}-${nextElicitationSeq()}`);
1169
+ markToolStart(elicId, "secure_env_collect");
1170
+ try {
1171
+ return await promptTextInputElicitation(request, textFields, ui, signal);
1172
+ }
1173
+ finally {
1174
+ markToolEnd(elicId);
1175
+ }
1106
1176
  }
1107
1177
  return { action: "decline" };
1108
1178
  };
@@ -1417,6 +1487,13 @@ export function buildSdkOptions(modelId, prompt, overrides, extraOptions = {}) {
1417
1487
  // leave allowed mcp__... tools with no registered backing tool.
1418
1488
  const settingSources = strictMcpConfig ? [] : ["user", "project", "local"];
1419
1489
  const exactWorkflowMcpTools = resolveExactWorkflowMcpToolsForPhase(gsdPhase, workflowServerName, workflowExplicitlyBlocked);
1490
+ const questionToolSurface = resolveWorkflowQuestionToolSurface({
1491
+ workflowServerName,
1492
+ workflowExplicitlyBlocked,
1493
+ workflowMcpTools,
1494
+ exactWorkflowMcpTools,
1495
+ env: process.env,
1496
+ });
1420
1497
  const runUatDisallowedTools = gsdPhase === "run-uat" && workflowServerName
1421
1498
  ? [
1422
1499
  ...RUN_UAT_FORBIDDEN_TOOL_NAMES.filter((toolName) => !toolName.startsWith("mcp__")),
@@ -1430,6 +1507,7 @@ export function buildSdkOptions(modelId, prompt, overrides, extraOptions = {}) {
1430
1507
  const disallowedTools = [...new Set([
1431
1508
  "ToolSearch",
1432
1509
  ...(workflowMcpTools.length > 0 || exactWorkflowMcpTools.length > 0 ? ["AskUserQuestion"] : []),
1510
+ ...questionToolSurface.disallowedTools,
1433
1511
  ...runUatDisallowedTools,
1434
1512
  ...extraDisallowedTools,
1435
1513
  ])];
@@ -1492,228 +1570,6 @@ export function buildSdkOptions(modelId, prompt, overrides, extraOptions = {}) {
1492
1570
  ...sdkExtraOptions,
1493
1571
  };
1494
1572
  }
1495
- /** Normalise heterogeneous SDK tool-result content (string, array, or object) into a uniform `ExternalToolResultContentBlock[]`. */
1496
- function normalizeToolResultContent(content) {
1497
- if (typeof content === "string") {
1498
- return [{ type: "text", text: content }];
1499
- }
1500
- if (!Array.isArray(content)) {
1501
- if (content == null)
1502
- return [{ type: "text", text: "" }];
1503
- return [{ type: "text", text: JSON.stringify(content) }];
1504
- }
1505
- const blocks = [];
1506
- for (const item of content) {
1507
- if (typeof item === "string") {
1508
- blocks.push({ type: "text", text: item });
1509
- continue;
1510
- }
1511
- if (!item || typeof item !== "object") {
1512
- blocks.push({ type: "text", text: String(item) });
1513
- continue;
1514
- }
1515
- const block = item;
1516
- if (block.type === "text") {
1517
- blocks.push({ type: "text", text: typeof block.text === "string" ? block.text : "" });
1518
- continue;
1519
- }
1520
- if (block.type === "image"
1521
- && typeof block.data === "string"
1522
- && typeof block.mimeType === "string") {
1523
- blocks.push({ type: "image", data: block.data, mimeType: block.mimeType });
1524
- continue;
1525
- }
1526
- blocks.push({ type: "text", text: JSON.stringify(block) });
1527
- }
1528
- return blocks.length > 0 ? blocks : [{ type: "text", text: "" }];
1529
- }
1530
- /**
1531
- * Extract a `details` payload from an MCP tool-result block.
1532
- *
1533
- * MCP's `CallToolResult` carries structured data in `structuredContent` — the
1534
- * protocol's supported channel for non-text payloads. Claude Code's synthetic
1535
- * user message may surface that field in one of two shapes depending on SDK
1536
- * version: as a sibling on the `mcp_tool_result` block itself, or as a
1537
- * dedicated content sub-block with `type: "structuredContent"`. Snake-case
1538
- * (`structured_content`) is accepted defensively in case a transport hop
1539
- * rewrites casing. All other shapes fall back to an empty object so callers
1540
- * can rely on `details` being present.
1541
- */
1542
- function extractStructuredDetailsFromBlock(block) {
1543
- const sibling = block.structuredContent ?? block.structured_content;
1544
- if (sibling && typeof sibling === "object" && !Array.isArray(sibling)) {
1545
- return sibling;
1546
- }
1547
- if (Array.isArray(block.content)) {
1548
- for (const item of block.content) {
1549
- if (!item || typeof item !== "object")
1550
- continue;
1551
- const sub = item;
1552
- if (sub.type !== "structuredContent" && sub.type !== "structured_content")
1553
- continue;
1554
- const payload = sub.structuredContent ?? sub.structured_content ?? sub.data ?? sub.value;
1555
- if (payload && typeof payload === "object" && !Array.isArray(payload)) {
1556
- return payload;
1557
- }
1558
- }
1559
- }
1560
- // Return undefined (not {}) when no structured payload is present, matching
1561
- // the pre-#4477 contract where `details` was nullable. An empty-object
1562
- // sentinel is truthy and breaks downstream consumers that gate on
1563
- // `if (details)`. `undefined` matches the type of the field these results
1564
- // flow into (`Record<string, unknown> | undefined`).
1565
- return undefined;
1566
- }
1567
- /**
1568
- * True for items that are MCP `structuredContent` pseudo-blocks living inside
1569
- * a tool-result `content[]` array. These blocks carry the structured payload
1570
- * (extracted separately by `extractStructuredDetailsFromBlock`) and must NOT
1571
- * leak into the visible content rendered to the user — otherwise the renderer
1572
- * stringifies the JSON pseudo-block and shows it next to the actual tool
1573
- * output. See PR #4477 review (post-fix-round).
1574
- */
1575
- function isStructuredContentPseudoBlock(item) {
1576
- if (!item || typeof item !== "object")
1577
- return false;
1578
- const type = item.type;
1579
- return type === "structuredContent" || type === "structured_content";
1580
- }
1581
- /**
1582
- * Strip `structuredContent` pseudo-blocks from a tool-result content array
1583
- * before normalization. The structured payload is extracted via the sibling
1584
- * `structuredContent` field (or a dedicated extractor pass on the raw block);
1585
- * the visible content path must not include the pseudo-block itself.
1586
- */
1587
- function stripStructuredContentPseudoBlocks(content) {
1588
- if (!Array.isArray(content))
1589
- return content;
1590
- return content.filter((item) => !isStructuredContentPseudoBlock(item));
1591
- }
1592
- /** Extract tool result payloads from an SDK synthetic user message, keyed by tool-use ID. */
1593
- export function extractToolResultsFromSdkUserMessage(message) {
1594
- const extracted = [];
1595
- const seen = new Set();
1596
- const rawMessage = message.message;
1597
- const content = Array.isArray(rawMessage?.content) ? rawMessage.content : [];
1598
- for (const item of content) {
1599
- if (!item || typeof item !== "object")
1600
- continue;
1601
- const block = item;
1602
- const type = typeof block.type === "string" ? block.type : "";
1603
- if (type !== "tool_result" && type !== "mcp_tool_result")
1604
- continue;
1605
- const toolUseId = typeof block.tool_use_id === "string" ? block.tool_use_id : "";
1606
- if (!toolUseId || seen.has(toolUseId))
1607
- continue;
1608
- seen.add(toolUseId);
1609
- extracted.push({
1610
- toolUseId,
1611
- result: {
1612
- content: normalizeToolResultContent(stripStructuredContentPseudoBlocks(block.content)),
1613
- details: extractStructuredDetailsFromBlock(block),
1614
- isError: block.is_error === true,
1615
- },
1616
- });
1617
- }
1618
- if (extracted.length === 0) {
1619
- const fallback = message.tool_use_result;
1620
- if (fallback && typeof fallback === "object") {
1621
- const toolResult = fallback;
1622
- const toolUseId = typeof toolResult.tool_use_id === "string" ? toolResult.tool_use_id : "";
1623
- if (toolUseId) {
1624
- extracted.push({
1625
- toolUseId,
1626
- result: {
1627
- content: normalizeToolResultContent(stripStructuredContentPseudoBlocks(toolResult.content)),
1628
- details: extractStructuredDetailsFromBlock(toolResult),
1629
- isError: toolResult.is_error === true,
1630
- },
1631
- });
1632
- }
1633
- }
1634
- }
1635
- return extracted;
1636
- }
1637
- /** Attach external tool results from the SDK synthetic user message to their corresponding tool-call blocks by ID. */
1638
- function attachExternalResultsToToolBlocks(toolBlocks, toolResultsById) {
1639
- for (const block of toolBlocks) {
1640
- if (block.type !== "toolCall" && block.type !== "serverToolUse")
1641
- continue;
1642
- const externalResult = toolResultsById.get(block.id);
1643
- if (!externalResult)
1644
- continue;
1645
- block.externalResult = externalResult;
1646
- }
1647
- }
1648
- /**
1649
- * Build the final assistant content that Agent Core consumes in
1650
- * `externalToolExecution` mode. This preserves tool-call blocks, attaches any
1651
- * SDK-produced external results by tool-call id, and then appends the final
1652
- * text/thinking blocks for the completed turn.
1653
- */
1654
- export function buildFinalAssistantContent(params) {
1655
- const mergedToolBlocks = [...params.intermediateToolBlocks];
1656
- if (params.pendingContent) {
1657
- mergePendingToolCalls(mergedToolBlocks, params.pendingContent);
1658
- }
1659
- attachExternalResultsToToolBlocks(mergedToolBlocks, params.toolResultsById);
1660
- const finalContent = [...mergedToolBlocks];
1661
- if (params.pendingContent && params.pendingContent.length > 0) {
1662
- for (const block of params.pendingContent) {
1663
- if (block.type === "text" || block.type === "thinking") {
1664
- finalContent.push(block);
1665
- }
1666
- }
1667
- }
1668
- else {
1669
- if (params.lastThinkingContent) {
1670
- finalContent.push({ type: "thinking", thinking: params.lastThinkingContent });
1671
- }
1672
- if (params.lastTextContent) {
1673
- finalContent.push({ type: "text", text: params.lastTextContent });
1674
- }
1675
- }
1676
- if (finalContent.length === 0 && params.fallbackResultText) {
1677
- finalContent.push({ type: "text", text: params.fallbackResultText });
1678
- }
1679
- return finalContent;
1680
- }
1681
- /**
1682
- * Merge tool-call blocks from the active partial-message builder into the
1683
- * running list of intermediate tool calls, preserving order and de-duping
1684
- * by tool-call id. Exposed for testing the F3 fix (final-turn tool calls
1685
- * dropped when `result` arrives without a preceding synthetic `user`).
1686
- */
1687
- export function mergePendingToolCalls(intermediate, pending) {
1688
- const alreadyIncluded = new Set();
1689
- for (const block of intermediate) {
1690
- if (block.type === "toolCall")
1691
- alreadyIncluded.add(block.id);
1692
- }
1693
- for (const block of pending) {
1694
- if (block.type !== "toolCall")
1695
- continue;
1696
- if (alreadyIncluded.has(block.id))
1697
- continue;
1698
- alreadyIncluded.add(block.id);
1699
- intermediate.push(block);
1700
- }
1701
- return intermediate;
1702
- }
1703
- export function handleClaudeCodePartialStreamEvent(builder, event, modelId) {
1704
- if (event.type === "message_start") {
1705
- // Claude Code can emit repeated SDK message_start events inside one
1706
- // logical assistant response. Keep appending until a synthetic user
1707
- // tool-result boundary explicitly clears the builder.
1708
- return {
1709
- builder: builder ?? new PartialMessageBuilder(event.message?.model ?? modelId),
1710
- assistantEvent: null,
1711
- };
1712
- }
1713
- if (!builder)
1714
- return { builder, assistantEvent: null };
1715
- return { builder, assistantEvent: builder.handleEvent(event) };
1716
- }
1717
1573
  // ---------------------------------------------------------------------------
1718
1574
  // streamSimple implementation
1719
1575
  // ---------------------------------------------------------------------------