@opengsd/gsd-pi 1.2.0-dev.955e4da0 → 1.2.0-dev.d6c5343c

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 (424) hide show
  1. package/dist/cli-style.d.ts +17 -0
  2. package/dist/cli-style.js +28 -0
  3. package/dist/cli.js +1 -1
  4. package/dist/headless-events.d.ts +4 -2
  5. package/dist/headless-events.js +14 -34
  6. package/dist/models-resolver.d.ts +3 -13
  7. package/dist/models-resolver.js +3 -22
  8. package/dist/resource-loader.js +2 -14
  9. package/dist/resources/.managed-resources-content-hash +1 -1
  10. package/dist/resources/GSD-WORKFLOW.md +5 -4
  11. package/dist/resources/extensions/async-jobs/async-bash-tool.js +30 -64
  12. package/dist/resources/extensions/async-jobs/await-tool.js +80 -12
  13. package/dist/resources/extensions/async-jobs/index.js +65 -0
  14. package/dist/resources/extensions/async-jobs/job-manager.js +12 -1
  15. package/dist/resources/extensions/bg-shell/bg-shell-command.js +6 -6
  16. package/dist/resources/extensions/bg-shell/bg-shell-tool.js +10 -7
  17. package/dist/resources/extensions/bg-shell/overlay.js +9 -6
  18. package/dist/resources/extensions/bg-shell/process-manager.js +54 -25
  19. package/dist/resources/extensions/bg-shell/readiness-detector.js +11 -0
  20. package/dist/resources/extensions/bg-shell/utilities.js +3 -0
  21. package/dist/resources/extensions/browser-tools/engine/managed-gsd-browser.js +209 -88
  22. package/dist/resources/extensions/browser-tools/engine/selection.js +73 -5
  23. package/dist/resources/extensions/browser-tools/index.js +69 -12
  24. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +30 -4
  25. package/dist/resources/extensions/gsd/auto/custom-verify-retry-store.js +17 -2
  26. package/dist/resources/extensions/gsd/auto/detect-stuck.js +33 -13
  27. package/dist/resources/extensions/gsd/auto/dispatch-history.js +105 -0
  28. package/dist/resources/extensions/gsd/auto/dispatch-key.js +37 -0
  29. package/dist/resources/extensions/gsd/auto/loop.js +4 -1
  30. package/dist/resources/extensions/gsd/auto/orchestrator.js +61 -44
  31. package/dist/resources/extensions/gsd/auto/phases.js +2 -2
  32. package/dist/resources/extensions/gsd/auto-direct-dispatch.js +8 -32
  33. package/dist/resources/extensions/gsd/auto-dispatch.js +40 -57
  34. package/dist/resources/extensions/gsd/auto-model-selection.js +25 -6
  35. package/dist/resources/extensions/gsd/auto-post-unit.js +23 -8
  36. package/dist/resources/extensions/gsd/auto-prompts.js +81 -19
  37. package/dist/resources/extensions/gsd/auto-start.js +18 -15
  38. package/dist/resources/extensions/gsd/auto-tool-tracking.js +18 -0
  39. package/dist/resources/extensions/gsd/auto-unit-tool-scope.js +12 -20
  40. package/dist/resources/extensions/gsd/auto-verification.js +9 -28
  41. package/dist/resources/extensions/gsd/auto-worktree.js +30 -90
  42. package/dist/resources/extensions/gsd/auto.js +4 -13
  43. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +3 -2
  44. package/dist/resources/extensions/gsd/bootstrap/dynamic-tools.js +23 -6
  45. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +19 -0
  46. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +212 -48
  47. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +303 -77
  48. package/dist/resources/extensions/gsd/branch-patterns.js +2 -0
  49. package/dist/resources/extensions/gsd/browser-daemon-auto-prep.js +83 -0
  50. package/dist/resources/extensions/gsd/browser-evidence.js +8 -2
  51. package/dist/resources/extensions/gsd/captures.js +4 -6
  52. package/dist/resources/extensions/gsd/consent-question.js +337 -0
  53. package/dist/resources/extensions/gsd/consent-verdict.js +63 -0
  54. package/dist/resources/extensions/gsd/constants.js +0 -2
  55. package/dist/resources/extensions/gsd/crash-recovery.js +4 -12
  56. package/dist/resources/extensions/gsd/db/queries.js +26 -0
  57. package/dist/resources/extensions/gsd/dispatch-guard.js +10 -35
  58. package/dist/resources/extensions/gsd/doctor-environment.js +2 -6
  59. package/dist/resources/extensions/gsd/doctor-format.js +9 -6
  60. package/dist/resources/extensions/gsd/doctor-runtime-checks.js +13 -15
  61. package/dist/resources/extensions/gsd/engine-hook-contract.js +70 -0
  62. package/dist/resources/extensions/gsd/error-classifier.js +9 -0
  63. package/dist/resources/extensions/gsd/exec-sandbox.js +30 -10
  64. package/dist/resources/extensions/gsd/files.js +33 -19
  65. package/dist/resources/extensions/gsd/guidance.js +158 -0
  66. package/dist/resources/extensions/gsd/guided-flow.js +17 -2
  67. package/dist/resources/extensions/gsd/markdown-renderer.js +10 -0
  68. package/dist/resources/extensions/gsd/mcp-filter.js +2 -19
  69. package/dist/resources/extensions/gsd/mcp-tool-name.js +5 -13
  70. package/dist/resources/extensions/gsd/memory-consolidation-scanner.js +1 -1
  71. package/dist/resources/extensions/gsd/migrate/safety.js +4 -1
  72. package/dist/resources/extensions/gsd/milestone-closeout.js +13 -23
  73. package/dist/resources/extensions/gsd/notification-store.js +11 -4
  74. package/dist/resources/extensions/gsd/parallel-monitor-overlay.js +6 -4
  75. package/dist/resources/extensions/gsd/parsers-legacy.js +16 -4
  76. package/dist/resources/extensions/gsd/paths.js +27 -0
  77. package/dist/resources/extensions/gsd/pre-execution-checks.js +91 -3
  78. package/dist/resources/extensions/gsd/preferences-models.js +14 -48
  79. package/dist/resources/extensions/gsd/prompts/complete-slice.md +2 -2
  80. package/dist/resources/extensions/gsd/prompts/plan-slice.md +1 -1
  81. package/dist/resources/extensions/gsd/prompts/refine-slice.md +1 -1
  82. package/dist/resources/extensions/gsd/prompts/replan-slice.md +1 -1
  83. package/dist/resources/extensions/gsd/prompts/run-uat.md +6 -4
  84. package/dist/resources/extensions/gsd/prompts/system.md +5 -2
  85. package/dist/resources/extensions/gsd/provider-error-guidance.js +1 -5
  86. package/dist/resources/extensions/gsd/provider-switch-observer.js +1 -1
  87. package/dist/resources/extensions/gsd/publication.js +87 -0
  88. package/dist/resources/extensions/gsd/reactive-graph.js +8 -1
  89. package/dist/resources/extensions/gsd/recovery-classification.js +37 -94
  90. package/dist/resources/extensions/gsd/safety/destructive-confirmation.js +108 -0
  91. package/dist/resources/extensions/gsd/state.js +6 -20
  92. package/dist/resources/extensions/gsd/stop-notice.js +57 -0
  93. package/dist/resources/extensions/gsd/tool-presentation-plan.js +4 -4
  94. package/dist/resources/extensions/gsd/tool-surface-readiness.js +56 -0
  95. package/dist/resources/extensions/gsd/tools/complete-slice.js +20 -10
  96. package/dist/resources/extensions/gsd/tools/exec-tool.js +9 -7
  97. package/dist/resources/extensions/gsd/tools/plan-slice.js +12 -6
  98. package/dist/resources/extensions/gsd/uat-policy.js +42 -16
  99. package/dist/resources/extensions/gsd/unit-closeout.js +138 -0
  100. package/dist/resources/extensions/gsd/unit-context-composer.js +74 -1
  101. package/dist/resources/extensions/gsd/unit-context-manifest.js +4 -27
  102. package/dist/resources/extensions/gsd/unit-registry.js +337 -0
  103. package/dist/resources/extensions/gsd/unit-tool-contracts.js +9 -182
  104. package/dist/resources/extensions/gsd/verdict-parser.js +1 -1
  105. package/dist/resources/extensions/gsd/web-app-uat.js +45 -8
  106. package/dist/resources/extensions/gsd/workflow-tool-surface.js +1 -1
  107. package/dist/resources/extensions/gsd/worktree-git-recovery.js +15 -9
  108. package/dist/resources/extensions/gsd/worktree-lifecycle.js +3 -2
  109. package/dist/resources/extensions/gsd/worktree-root.js +11 -0
  110. package/dist/resources/extensions/gsd/worktree-session-state.js +4 -5
  111. package/dist/resources/extensions/search-the-web/native-search.js +5 -3
  112. package/dist/resources/extensions/shared/browser-contract.js +59 -0
  113. package/dist/resources/extensions/shared/gsd-browser-cli.js +96 -5
  114. package/dist/resources/shared/package.json +3 -0
  115. package/dist/resources/skills/create-skill/references/executable-code.md +1 -1
  116. package/dist/resources/skills/create-skill/workflows/add-reference.md +8 -3
  117. package/dist/resources/skills/create-skill/workflows/add-script.md +4 -2
  118. package/dist/resources/skills/create-skill/workflows/add-template.md +3 -1
  119. package/dist/resources/skills/create-skill/workflows/add-workflow.md +8 -3
  120. package/dist/resources/skills/create-skill/workflows/upgrade-to-router.md +10 -5
  121. package/dist/resources/skills/create-skill/workflows/verify-skill.md +9 -4
  122. package/dist/resources/skills/spike-wrap-up/SKILL.md +9 -9
  123. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  124. package/dist/web/standalone/.next/BUILD_ID +1 -1
  125. package/dist/web/standalone/.next/app-path-routes-manifest.json +12 -12
  126. package/dist/web/standalone/.next/build-manifest.json +3 -3
  127. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  128. package/dist/web/standalone/.next/react-loadable-manifest.json +1 -1
  129. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  130. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  131. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  132. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  133. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  134. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  135. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  136. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  137. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  138. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  139. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  140. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  141. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  142. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  143. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  144. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  145. package/dist/web/standalone/.next/server/app/api/update/route.js +1 -1
  146. package/dist/web/standalone/.next/server/app/api/update/route.js.nft.json +1 -1
  147. package/dist/web/standalone/.next/server/app/index.html +1 -1
  148. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  149. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  150. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  151. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  152. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  153. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  154. package/dist/web/standalone/.next/server/app-paths-manifest.json +12 -12
  155. package/dist/web/standalone/.next/server/chunks/5124.js +1 -1
  156. package/dist/web/standalone/.next/server/chunks/8357.js +1 -1
  157. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  158. package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
  159. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  160. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  161. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  162. package/dist/web/standalone/.next/static/chunks/{796.cf859a427a2cb2ac.js → 796.e0bdc932325d7e03.js} +1 -1
  163. package/dist/web/standalone/.next/static/chunks/{webpack-fbea77b5f9953368.js → webpack-f0285ce91d4ec9ef.js} +1 -1
  164. package/dist/web/standalone/package.json +1 -1
  165. package/dist/worktree-cli.js +3 -6
  166. package/dist/worktree-status-banner.js +7 -15
  167. package/package.json +1 -1
  168. package/packages/cloud-mcp-gateway/package.json +2 -2
  169. package/packages/contracts/dist/rpc.d.ts +1 -0
  170. package/packages/contracts/dist/rpc.d.ts.map +1 -1
  171. package/packages/contracts/dist/rpc.js.map +1 -1
  172. package/packages/contracts/dist/workflow.d.ts +4 -0
  173. package/packages/contracts/dist/workflow.d.ts.map +1 -1
  174. package/packages/contracts/dist/workflow.js.map +1 -1
  175. package/packages/contracts/package.json +1 -1
  176. package/packages/daemon/package.json +4 -4
  177. package/packages/gsd-agent-core/package.json +5 -5
  178. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.d.ts +5 -0
  179. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  180. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.js +5 -0
  181. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.js.map +1 -1
  182. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  183. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js +7 -0
  184. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  185. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/input-controller.d.ts.map +1 -1
  186. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/input-controller.js +8 -1
  187. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/input-controller.js.map +1 -1
  188. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-chat-render.d.ts.map +1 -1
  189. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-chat-render.js +11 -1
  190. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-chat-render.js.map +1 -1
  191. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-auth.d.ts.map +1 -1
  192. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-auth.js +4 -4
  193. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-auth.js.map +1 -1
  194. package/packages/gsd-agent-modes/dist/modes/rpc/rpc-mode.d.ts.map +1 -1
  195. package/packages/gsd-agent-modes/dist/modes/rpc/rpc-mode.js +3 -1
  196. package/packages/gsd-agent-modes/dist/modes/rpc/rpc-mode.js.map +1 -1
  197. package/packages/gsd-agent-modes/package.json +7 -7
  198. package/packages/mcp-server/dist/cli.js +6 -3
  199. package/packages/mcp-server/dist/cli.js.map +1 -1
  200. package/packages/mcp-server/dist/workflow-tools.d.ts +8 -0
  201. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  202. package/packages/mcp-server/dist/workflow-tools.js +17 -1
  203. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  204. package/packages/mcp-server/package.json +3 -3
  205. package/packages/native/package.json +1 -1
  206. package/packages/pi-agent-core/dist/harness/env/nodejs.d.ts +1 -0
  207. package/packages/pi-agent-core/dist/harness/env/nodejs.d.ts.map +1 -1
  208. package/packages/pi-agent-core/dist/harness/env/nodejs.js +34 -3
  209. package/packages/pi-agent-core/dist/harness/env/nodejs.js.map +1 -1
  210. package/packages/pi-agent-core/dist/index.d.ts +1 -0
  211. package/packages/pi-agent-core/dist/index.d.ts.map +1 -1
  212. package/packages/pi-agent-core/dist/index.js +3 -0
  213. package/packages/pi-agent-core/dist/index.js.map +1 -1
  214. package/packages/pi-agent-core/package.json +1 -1
  215. package/packages/pi-ai/README.md +1 -0
  216. package/packages/pi-ai/dist/image-models.generated.d.ts +2 -2
  217. package/packages/pi-ai/dist/image-models.generated.js +6 -6
  218. package/packages/pi-ai/dist/image-models.generated.js.map +1 -1
  219. package/packages/pi-ai/dist/models.generated.d.ts +35 -125
  220. package/packages/pi-ai/dist/models.generated.d.ts.map +1 -1
  221. package/packages/pi-ai/dist/models.generated.js +46 -120
  222. package/packages/pi-ai/dist/models.generated.js.map +1 -1
  223. package/packages/pi-ai/package.json +3 -2
  224. package/packages/pi-coding-agent/dist/core/auth-storage.d.ts +2 -2
  225. package/packages/pi-coding-agent/dist/core/auth-storage.d.ts.map +1 -1
  226. package/packages/pi-coding-agent/dist/core/auth-storage.js +19 -13
  227. package/packages/pi-coding-agent/dist/core/auth-storage.js.map +1 -1
  228. package/packages/pi-coding-agent/dist/core/provider-readiness.d.ts.map +1 -1
  229. package/packages/pi-coding-agent/dist/core/provider-readiness.js +13 -6
  230. package/packages/pi-coding-agent/dist/core/provider-readiness.js.map +1 -1
  231. package/packages/pi-coding-agent/dist/core/tools/bash.d.ts +11 -0
  232. package/packages/pi-coding-agent/dist/core/tools/bash.d.ts.map +1 -1
  233. package/packages/pi-coding-agent/dist/core/tools/bash.js +53 -11
  234. package/packages/pi-coding-agent/dist/core/tools/bash.js.map +1 -1
  235. package/packages/pi-coding-agent/dist/index.d.ts +1 -1
  236. package/packages/pi-coding-agent/dist/index.d.ts.map +1 -1
  237. package/packages/pi-coding-agent/dist/index.js +1 -1
  238. package/packages/pi-coding-agent/dist/index.js.map +1 -1
  239. package/packages/pi-coding-agent/dist/utils/shell.d.ts +28 -2
  240. package/packages/pi-coding-agent/dist/utils/shell.d.ts.map +1 -1
  241. package/packages/pi-coding-agent/dist/utils/shell.js +56 -10
  242. package/packages/pi-coding-agent/dist/utils/shell.js.map +1 -1
  243. package/packages/pi-coding-agent/package.json +7 -7
  244. package/packages/pi-tui/dist/tui.d.ts.map +1 -1
  245. package/packages/pi-tui/dist/tui.js +9 -0
  246. package/packages/pi-tui/dist/tui.js.map +1 -1
  247. package/packages/pi-tui/package.json +2 -2
  248. package/packages/rpc-client/package.json +2 -2
  249. package/pkg/package.json +1 -1
  250. package/src/resources/GSD-WORKFLOW.md +5 -4
  251. package/src/resources/extensions/async-jobs/async-bash-cancel.test.ts +360 -0
  252. package/src/resources/extensions/async-jobs/async-bash-tool.ts +33 -56
  253. package/src/resources/extensions/async-jobs/await-tool.test.ts +139 -0
  254. package/src/resources/extensions/async-jobs/await-tool.ts +82 -12
  255. package/src/resources/extensions/async-jobs/index.ts +79 -0
  256. package/src/resources/extensions/async-jobs/job-manager.ts +21 -1
  257. package/src/resources/extensions/bg-shell/bg-shell-command.ts +6 -6
  258. package/src/resources/extensions/bg-shell/bg-shell-tool.ts +10 -6
  259. package/src/resources/extensions/bg-shell/overlay.ts +9 -5
  260. package/src/resources/extensions/bg-shell/process-manager.ts +50 -25
  261. package/src/resources/extensions/bg-shell/readiness-detector.ts +12 -0
  262. package/src/resources/extensions/bg-shell/tests/lifecycle-and-utilities.test.ts +48 -1
  263. package/src/resources/extensions/bg-shell/utilities.ts +3 -0
  264. package/src/resources/extensions/browser-tools/engine/managed-gsd-browser.ts +265 -98
  265. package/src/resources/extensions/browser-tools/engine/selection.ts +90 -4
  266. package/src/resources/extensions/browser-tools/index.ts +71 -13
  267. package/src/resources/extensions/browser-tools/tests/browser-engine-selection.test.mjs +83 -13
  268. package/src/resources/extensions/browser-tools/tests/gsd-browser-launch-config.test.mjs +29 -1
  269. package/src/resources/extensions/browser-tools/tests/managed-gsd-browser-tools.test.mjs +136 -0
  270. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +34 -4
  271. package/src/resources/extensions/gsd/auto/custom-verify-retry-store.ts +21 -3
  272. package/src/resources/extensions/gsd/auto/detect-stuck.ts +32 -9
  273. package/src/resources/extensions/gsd/auto/dispatch-history.ts +152 -0
  274. package/src/resources/extensions/gsd/auto/dispatch-key.ts +39 -0
  275. package/src/resources/extensions/gsd/auto/loop.ts +4 -1
  276. package/src/resources/extensions/gsd/auto/orchestrator.ts +70 -46
  277. package/src/resources/extensions/gsd/auto/phases.ts +2 -2
  278. package/src/resources/extensions/gsd/auto-direct-dispatch.ts +8 -32
  279. package/src/resources/extensions/gsd/auto-dispatch.ts +38 -52
  280. package/src/resources/extensions/gsd/auto-model-selection.ts +25 -5
  281. package/src/resources/extensions/gsd/auto-post-unit.ts +25 -8
  282. package/src/resources/extensions/gsd/auto-prompts.ts +118 -35
  283. package/src/resources/extensions/gsd/auto-start.ts +18 -17
  284. package/src/resources/extensions/gsd/auto-tool-tracking.ts +19 -0
  285. package/src/resources/extensions/gsd/auto-unit-tool-scope.ts +14 -21
  286. package/src/resources/extensions/gsd/auto-verification.ts +8 -26
  287. package/src/resources/extensions/gsd/auto-worktree.ts +30 -93
  288. package/src/resources/extensions/gsd/auto.ts +8 -15
  289. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +3 -5
  290. package/src/resources/extensions/gsd/bootstrap/dynamic-tools.ts +23 -6
  291. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +24 -0
  292. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +251 -47
  293. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +352 -84
  294. package/src/resources/extensions/gsd/branch-patterns.ts +3 -0
  295. package/src/resources/extensions/gsd/browser-daemon-auto-prep.ts +108 -0
  296. package/src/resources/extensions/gsd/browser-evidence.ts +18 -2
  297. package/src/resources/extensions/gsd/captures.ts +4 -6
  298. package/src/resources/extensions/gsd/consent-question.ts +416 -0
  299. package/src/resources/extensions/gsd/consent-verdict.ts +86 -0
  300. package/src/resources/extensions/gsd/constants.ts +0 -3
  301. package/src/resources/extensions/gsd/crash-recovery.ts +3 -9
  302. package/src/resources/extensions/gsd/db/queries.ts +37 -0
  303. package/src/resources/extensions/gsd/dispatch-guard.ts +8 -31
  304. package/src/resources/extensions/gsd/doctor-environment.ts +2 -7
  305. package/src/resources/extensions/gsd/doctor-format.ts +12 -7
  306. package/src/resources/extensions/gsd/doctor-runtime-checks.ts +13 -15
  307. package/src/resources/extensions/gsd/engine-hook-contract.ts +79 -0
  308. package/src/resources/extensions/gsd/error-classifier.ts +11 -0
  309. package/src/resources/extensions/gsd/exec-sandbox.ts +49 -9
  310. package/src/resources/extensions/gsd/files.ts +33 -12
  311. package/src/resources/extensions/gsd/guidance.ts +217 -0
  312. package/src/resources/extensions/gsd/guided-flow.ts +16 -2
  313. package/src/resources/extensions/gsd/markdown-renderer.ts +11 -0
  314. package/src/resources/extensions/gsd/mcp-filter.ts +2 -23
  315. package/src/resources/extensions/gsd/mcp-tool-name.ts +6 -11
  316. package/src/resources/extensions/gsd/memory-consolidation-scanner.ts +1 -1
  317. package/src/resources/extensions/gsd/migrate/safety.ts +4 -1
  318. package/src/resources/extensions/gsd/milestone-closeout.ts +13 -23
  319. package/src/resources/extensions/gsd/notification-store.ts +26 -3
  320. package/src/resources/extensions/gsd/parallel-monitor-overlay.ts +6 -4
  321. package/src/resources/extensions/gsd/parsers-legacy.ts +16 -4
  322. package/src/resources/extensions/gsd/paths.ts +33 -0
  323. package/src/resources/extensions/gsd/pre-execution-checks.ts +109 -3
  324. package/src/resources/extensions/gsd/preferences-models.ts +12 -47
  325. package/src/resources/extensions/gsd/prompts/complete-slice.md +2 -2
  326. package/src/resources/extensions/gsd/prompts/plan-slice.md +1 -1
  327. package/src/resources/extensions/gsd/prompts/refine-slice.md +1 -1
  328. package/src/resources/extensions/gsd/prompts/replan-slice.md +1 -1
  329. package/src/resources/extensions/gsd/prompts/run-uat.md +6 -4
  330. package/src/resources/extensions/gsd/prompts/system.md +5 -2
  331. package/src/resources/extensions/gsd/provider-error-guidance.ts +4 -9
  332. package/src/resources/extensions/gsd/provider-switch-observer.ts +1 -1
  333. package/src/resources/extensions/gsd/publication.ts +122 -0
  334. package/src/resources/extensions/gsd/reactive-graph.ts +11 -1
  335. package/src/resources/extensions/gsd/recovery-classification.ts +42 -96
  336. package/src/resources/extensions/gsd/safety/destructive-confirmation.ts +134 -0
  337. package/src/resources/extensions/gsd/state.ts +9 -21
  338. package/src/resources/extensions/gsd/stop-notice.ts +75 -0
  339. package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +22 -0
  340. package/src/resources/extensions/gsd/tests/auto-orchestrator.test.ts +101 -26
  341. package/src/resources/extensions/gsd/tests/browser-automation-contract-fixture.ts +39 -0
  342. package/src/resources/extensions/gsd/tests/browser-contract.test.ts +44 -0
  343. package/src/resources/extensions/gsd/tests/browser-daemon-auto-prep.test.ts +144 -0
  344. package/src/resources/extensions/gsd/tests/checkout-branch-stash-guard.test.ts +66 -1
  345. package/src/resources/extensions/gsd/tests/clear-stale-autostart.test.ts +22 -0
  346. package/src/resources/extensions/gsd/tests/commands-verdict.test.ts +8 -7
  347. package/src/resources/extensions/gsd/tests/complete-slice-verification-gate.test.ts +42 -0
  348. package/src/resources/extensions/gsd/tests/consent-question.test.ts +336 -0
  349. package/src/resources/extensions/gsd/tests/custom-verify-retry-store.test.ts +67 -0
  350. package/src/resources/extensions/gsd/tests/deep-project-auto-loop.test.ts +10 -10
  351. package/src/resources/extensions/gsd/tests/destructive-confirmation.test.ts +303 -0
  352. package/src/resources/extensions/gsd/tests/dispatch-history.test.ts +273 -0
  353. package/src/resources/extensions/gsd/tests/dispatch-run-uat-browser-tools.test.ts +2 -1
  354. package/src/resources/extensions/gsd/tests/dynamic-bash-no-cap.test.ts +132 -0
  355. package/src/resources/extensions/gsd/tests/engine-hook-contract.test.ts +148 -0
  356. package/src/resources/extensions/gsd/tests/exec-graceful-kill.test.ts +193 -0
  357. package/src/resources/extensions/gsd/tests/exec-tool.test.ts +29 -1
  358. package/src/resources/extensions/gsd/tests/extension-bootstrap-isolation.test.ts +35 -1
  359. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +27 -0
  360. package/src/resources/extensions/gsd/tests/guidance.test.ts +148 -0
  361. package/src/resources/extensions/gsd/tests/integration/auto-worktree-milestone-merge.test.ts +53 -11
  362. package/src/resources/extensions/gsd/tests/integration/auto-worktree.test.ts +73 -58
  363. package/src/resources/extensions/gsd/tests/integration/gsd-integration-fixture.ts +80 -0
  364. package/src/resources/extensions/gsd/tests/integration/run-uat.test.ts +199 -0
  365. package/src/resources/extensions/gsd/tests/mcp-project-config.test.ts +3 -1
  366. package/src/resources/extensions/gsd/tests/model-unittype-mapping.test.ts +32 -1
  367. package/src/resources/extensions/gsd/tests/notification-store.test.ts +32 -0
  368. package/src/resources/extensions/gsd/tests/oauth-api-model-routing.test.ts +167 -0
  369. package/src/resources/extensions/gsd/tests/parallel-research-dispatch.test.ts +18 -0
  370. package/src/resources/extensions/gsd/tests/parsers-legacy-importers.test.ts +139 -0
  371. package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +193 -1
  372. package/src/resources/extensions/gsd/tests/prompt-db.test.ts +124 -6
  373. package/src/resources/extensions/gsd/tests/provider-error-guidance.test.ts +3 -3
  374. package/src/resources/extensions/gsd/tests/publication.test.ts +120 -0
  375. package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +157 -0
  376. package/src/resources/extensions/gsd/tests/runtime-invariant-modules.test.ts +1 -0
  377. package/src/resources/extensions/gsd/tests/stop-notice.test.ts +70 -0
  378. package/src/resources/extensions/gsd/tests/token-tool-gating.test.ts +76 -0
  379. package/src/resources/extensions/gsd/tests/tool-invocation-error-loop-break.test.ts +8 -0
  380. package/src/resources/extensions/gsd/tests/tool-surface-readiness.test.ts +155 -0
  381. package/src/resources/extensions/gsd/tests/uat-policy.test.ts +112 -29
  382. package/src/resources/extensions/gsd/tests/unit-closeout.test.ts +209 -0
  383. package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +67 -2
  384. package/src/resources/extensions/gsd/tests/unit-registry.test.ts +163 -0
  385. package/src/resources/extensions/gsd/tests/web-app-uat.test.ts +44 -1
  386. package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +2 -2
  387. package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +2 -2
  388. package/src/resources/extensions/gsd/tests/write-gate-seam.test.ts +358 -0
  389. package/src/resources/extensions/gsd/tests/write-gate.test.ts +67 -1
  390. package/src/resources/extensions/gsd/tool-presentation-plan.ts +4 -4
  391. package/src/resources/extensions/gsd/tool-surface-readiness.ts +76 -0
  392. package/src/resources/extensions/gsd/tools/complete-slice.ts +20 -10
  393. package/src/resources/extensions/gsd/tools/exec-tool.ts +8 -7
  394. package/src/resources/extensions/gsd/tools/plan-slice.ts +12 -6
  395. package/src/resources/extensions/gsd/uat-policy.ts +62 -16
  396. package/src/resources/extensions/gsd/unit-closeout.ts +201 -0
  397. package/src/resources/extensions/gsd/unit-context-composer.ts +111 -1
  398. package/src/resources/extensions/gsd/unit-context-manifest.ts +4 -28
  399. package/src/resources/extensions/gsd/unit-registry.ts +412 -0
  400. package/src/resources/extensions/gsd/unit-tool-contracts.ts +27 -192
  401. package/src/resources/extensions/gsd/verdict-parser.ts +1 -1
  402. package/src/resources/extensions/gsd/web-app-uat.ts +51 -8
  403. package/src/resources/extensions/gsd/workflow-tool-surface.ts +4 -1
  404. package/src/resources/extensions/gsd/worktree-git-recovery.ts +15 -9
  405. package/src/resources/extensions/gsd/worktree-lifecycle.ts +3 -8
  406. package/src/resources/extensions/gsd/worktree-root.ts +12 -0
  407. package/src/resources/extensions/gsd/worktree-session-state.ts +3 -5
  408. package/src/resources/extensions/search-the-web/native-search.ts +5 -3
  409. package/src/resources/extensions/shared/browser-contract.ts +66 -0
  410. package/src/resources/extensions/shared/gsd-browser-cli.ts +119 -5
  411. package/src/resources/shared/package.json +3 -0
  412. package/src/resources/skills/create-skill/references/executable-code.md +1 -1
  413. package/src/resources/skills/create-skill/workflows/add-reference.md +8 -3
  414. package/src/resources/skills/create-skill/workflows/add-script.md +4 -2
  415. package/src/resources/skills/create-skill/workflows/add-template.md +3 -1
  416. package/src/resources/skills/create-skill/workflows/add-workflow.md +8 -3
  417. package/src/resources/skills/create-skill/workflows/upgrade-to-router.md +10 -5
  418. package/src/resources/skills/create-skill/workflows/verify-skill.md +9 -4
  419. package/src/resources/skills/spike-wrap-up/SKILL.md +9 -9
  420. package/dist/resources/extensions/gsd/user-input-boundary.js +0 -218
  421. package/src/resources/extensions/gsd/tests/user-input-boundary.test.ts +0 -173
  422. package/src/resources/extensions/gsd/user-input-boundary.ts +0 -216
  423. /package/dist/web/standalone/.next/static/{C24pqUd-aru-l0Dp0gLZP → jmTLg6xZmAuq_LIqKOxrH}/_buildManifest.js +0 -0
  424. /package/dist/web/standalone/.next/static/{C24pqUd-aru-l0Dp0gLZP → jmTLg6xZmAuq_LIqKOxrH}/_ssgManifest.js +0 -0
@@ -0,0 +1,108 @@
1
+ import { execFileSync } from "node:child_process";
2
+ import { resolve } from "node:path";
3
+
4
+ import {
5
+ resolveAmbientBrowserEngineResolution,
6
+ resolveBrowserEngineResolution,
7
+ type BrowserEngineMode,
8
+ } from "../browser-tools/engine/selection.js";
9
+ import {
10
+ resolveGsdBrowserCliAvailability,
11
+ resolveGsdBrowserDaemonStartInvocation,
12
+ } from "../shared/gsd-browser-cli.js";
13
+ import { uatTypeIncludesBrowser, type UatType } from "./uat-policy.js";
14
+
15
+ const DEFAULT_DAEMON_START_TIMEOUT_MS = 30_000;
16
+
17
+ function isEnvDisabled(value: string | undefined): boolean {
18
+ if (!value) return false;
19
+ const normalized = value.trim().toLowerCase();
20
+ return normalized === "0" || normalized === "false" || normalized === "off";
21
+ }
22
+
23
+ function isWarmUpDisabled(env: NodeJS.ProcessEnv = process.env): boolean {
24
+ const value = env.GSD_BROWSER_WARMUP?.trim().toLowerCase();
25
+ return value === "0" || value === "false" || value === "off";
26
+ }
27
+
28
+ export interface BrowserDaemonWarmContext {
29
+ uatType: UatType;
30
+ sessionProvider?: string;
31
+ sessionAuthMode?: "apiKey" | "oauth" | "externalCli" | "none";
32
+ sessionBaseUrl?: string;
33
+ projectRoot: string;
34
+ env?: NodeJS.ProcessEnv;
35
+ }
36
+
37
+ /** Active engine for warm-up: explicit env override, else session-committed ambient resolution. */
38
+ function resolveActiveBrowserEngine(projectRoot: string, env: NodeJS.ProcessEnv): BrowserEngineMode {
39
+ if (env.GSD_BROWSER_ENGINE?.trim()) {
40
+ return resolveBrowserEngineResolution(env, projectRoot).engine;
41
+ }
42
+ return resolveAmbientBrowserEngineResolution(projectRoot).engine;
43
+ }
44
+
45
+ export function shouldWarmBrowserDaemonForUat(ctx: BrowserDaemonWarmContext): boolean {
46
+ if (!uatTypeIncludesBrowser(ctx.uatType)) return false;
47
+
48
+ const env = ctx.env ?? process.env;
49
+ if (isWarmUpDisabled(env)) return false;
50
+ if (isEnvDisabled(env.GSD_BROWSER_MCP_ENABLED)) return false;
51
+
52
+ const availability = resolveGsdBrowserCliAvailability(env);
53
+ if (!availability.available) return false;
54
+
55
+ const projectRoot = resolve(ctx.projectRoot);
56
+ return resolveActiveBrowserEngine(projectRoot, env) === "gsd-browser";
57
+ }
58
+
59
+ export function ensureBrowserDaemonStarted(
60
+ projectRoot: string,
61
+ options: { env?: NodeJS.ProcessEnv; timeoutMs?: number } = {},
62
+ ): { ok: true } | { ok: false; error: string } {
63
+ const env = options.env ?? process.env;
64
+ const availability = resolveGsdBrowserCliAvailability(env);
65
+ if (!availability.available) {
66
+ return { ok: false, error: availability.detail };
67
+ }
68
+
69
+ let invocation: ReturnType<typeof resolveGsdBrowserDaemonStartInvocation>;
70
+ try {
71
+ invocation = resolveGsdBrowserDaemonStartInvocation(projectRoot, env);
72
+ } catch (error) {
73
+ return {
74
+ ok: false,
75
+ error: error instanceof Error ? error.message : String(error),
76
+ };
77
+ }
78
+
79
+ try {
80
+ execFileSync(invocation.command, invocation.args, {
81
+ cwd: invocation.cwd,
82
+ env: { ...process.env, ...env, ...(invocation.env ?? {}) },
83
+ stdio: ["ignore", "pipe", "pipe"],
84
+ timeout: options.timeoutMs ?? DEFAULT_DAEMON_START_TIMEOUT_MS,
85
+ encoding: "utf-8",
86
+ });
87
+ return { ok: true };
88
+ } catch (error) {
89
+ return {
90
+ ok: false,
91
+ error: error instanceof Error ? error.message : String(error),
92
+ };
93
+ }
94
+ }
95
+
96
+ /**
97
+ * Best-effort pre-warm of the gsd-browser session daemon before browser-backed
98
+ * run-uat dispatch. Returns an actionable stop reason when warm-up is required
99
+ * but fails; returns null when warm-up is skipped or succeeds.
100
+ */
101
+ export function prepareBrowserDaemonForUat(ctx: BrowserDaemonWarmContext): string | null {
102
+ if (!shouldWarmBrowserDaemonForUat(ctx)) return null;
103
+
104
+ const result = ensureBrowserDaemonStarted(ctx.projectRoot, { env: ctx.env });
105
+ if (result.ok) return null;
106
+
107
+ return `Cannot dispatch browser-backed run-uat: gsd-browser daemon failed to start (${result.error}). Ensure Chrome/Chromium is installed, run \`gsd-browser daemon health\` with the project session flags from .mcp.json, or set GSD_BROWSER_PATH to a Chromium binary.`;
108
+ }
@@ -1,9 +1,25 @@
1
1
  // Project/App: gsd-pi
2
2
  // File Purpose: Shared browser-observable UAT requirement and evidence detection.
3
3
 
4
- export const BROWSER_REQUIREMENT_RE = /\b(?:file:\/\/|localhost|playwright|chrome|screenshot|snapshot|browser_(?:assert|batch|find|verify|snapshot_refs))\b|\b(?:open|launch|navigate|load|visit|serve|start)\b.{0,80}\b(?:browser|page|localhost|file:\/\/)\b|\bbrowser\s+(?:check|session|test|uat|tool|automation|interaction|flow)\b/i;
4
+ import { BROWSER_EVIDENCE_SIGNAL_TOOL_NAMES } from "../shared/browser-contract.js";
5
+
6
+ // Alternation fragment over the contract's evidence-signal names, e.g.
7
+ // `browser_(?:assert|batch|...)`. The names are `browser_`-prefixed
8
+ // identifiers (pinned by tests/browser-contract.test.ts), so no escaping is
9
+ // needed.
10
+ const BROWSER_TOOL_SIGNAL = `browser_(?:${
11
+ BROWSER_EVIDENCE_SIGNAL_TOOL_NAMES.map((name) => name.slice("browser_".length)).join("|")
12
+ })`;
13
+
14
+ export const BROWSER_REQUIREMENT_RE = new RegExp(
15
+ String.raw`\b(?:file://|localhost|playwright|chrome|screenshot|snapshot|${BROWSER_TOOL_SIGNAL})\b|\b(?:open|launch|navigate|load|visit|serve|start)\b.{0,80}\b(?:browser|page|localhost|file://)\b|\bbrowser\s+(?:check|session|test|uat|tool|automation|interaction|flow)\b`,
16
+ "i",
17
+ );
5
18
  export const NO_BROWSER_EVIDENCE_RE = /\b(?:no|without|not|wasn'?t|isn'?t)\s+(?:automated\s+)?(?:live\s+)?browser(?:\s+(?:session|test|uat))?|\bno\s+automated\s+browser\b|\bnot\s+conducted\b/i;
6
- export const BROWSER_RUNTIME_RE = /\b(?:browser|playwright|chrome|camoufox|browser_(?:assert|batch|find|verify|snapshot_refs)|screenshot|snapshot|file:\/\/|localhost)\b/i;
19
+ export const BROWSER_RUNTIME_RE = new RegExp(
20
+ String.raw`\b(?:browser|playwright|chrome|camoufox|${BROWSER_TOOL_SIGNAL}|screenshot|snapshot|file://|localhost)\b`,
21
+ "i",
22
+ );
7
23
  export const BROWSER_ACTION_RE = /\b(?:open(?:ed)?|navigate(?:d)?|click(?:ed)?|type(?:d)?|reload(?:ed)?|capture(?:d)?|screenshot|snapshot)\b/i;
8
24
  export const BROWSER_ASSERTION_RE = /\b(?:assert(?:ed|ion)?|observed|confirmed|verified|expected|visible|text|count|label|strikethrough|localstorage|screenshot|snapshot|passed)\b/i;
9
25
  const NON_REQUIREMENT_BROWSER_HEADING_RE = /^(?:not\s+proven|not\s+covered|out\s+of\s+scope|deferred|follow-?ups?|known\s+limitations|notes\s+for\s+tester)\b/i;
@@ -12,7 +12,7 @@ import { existsSync, readFileSync, writeFileSync, mkdirSync } from "node:fs";
12
12
  import { join, resolve } from "node:path";
13
13
  import { randomUUID } from "node:crypto";
14
14
  import { gsdRoot } from "./paths.js";
15
- import { findWorktreeSegment } from "./worktree-root.js";
15
+ import { projectRootFromWorktreePath } from "./worktree-root.js";
16
16
 
17
17
  // ─── Types ────────────────────────────────────────────────────────────────────
18
18
 
@@ -60,11 +60,9 @@ const VALID_CLASSIFICATIONS: readonly string[] = [
60
60
  * directory that contains `.gsd/worktrees/` — that's the project root.
61
61
  */
62
62
  export function resolveCapturesPath(basePath: string): string {
63
- const resolved = resolve(basePath);
64
- const segment = findWorktreeSegment(resolved.replaceAll("\\", "/"));
65
- if (segment) {
66
- // basePath is inside a worktree — resolve to project root
67
- const projectRoot = resolved.slice(0, segment.gsdIdx);
63
+ // If basePath is inside a worktree, resolve to the project root.
64
+ const projectRoot = projectRootFromWorktreePath(resolve(basePath));
65
+ if (projectRoot) {
68
66
  return join(projectRoot, ".gsd", CAPTURES_FILENAME);
69
67
  }
70
68
  return join(gsdRoot(basePath), CAPTURES_FILENAME);
@@ -0,0 +1,416 @@
1
+ /**
2
+ * Consent Question module — the single home for the user-question lifecycle:
3
+ * classification → gating → answer validation → cancellation.
4
+ *
5
+ * Every question the assistant puts to a human is classified into a kind, and
6
+ * the kind alone decides the fail policy:
7
+ *
8
+ * - "gate" — mechanical write gates (depth verification, destructive
9
+ * confirm). Fail-closed; structural answer validation is
10
+ * delegated to the write-gate validators.
11
+ * - "consent" — approval/confirmation questions ("ready to write?",
12
+ * "is this correct?"). Fail-closed.
13
+ * - "decision" — explicit user decisions (research vs skip). Fail-closed.
14
+ * - "informational" — anything that is not asking the user to consent or
15
+ * decide. Fail-open.
16
+ *
17
+ * Fail-closed means an empty/missing answer is NEVER treated as an answer —
18
+ * evaluateAnswer returns "waiting" so callers pause instead of proceeding.
19
+ * This fixes #528 (empty `selected` on a non-gate question used to pass
20
+ * through as a real answer) by construction.
21
+ *
22
+ * shouldPauseForQuestion replaces the old unit-type allowlist: a classified
23
+ * consent/decision question pauses regardless of unit type, including
24
+ * interactive mode where no unit is active. This fixes #682 (prose approval
25
+ * questions outside the 4 allowlisted discuss units rendered as un-gated
26
+ * prose menus) by construction.
27
+ */
28
+
29
+ import { isGateQuestionId } from "./bootstrap/write-gate.js";
30
+ import {
31
+ evaluateGateAnswer,
32
+ hasNotesValue,
33
+ hasSelectedValue,
34
+ type VerdictAnswerDetails,
35
+ type VerdictQuestionShape,
36
+ } from "./consent-verdict.js";
37
+ import { isDestructiveConfirmGateId } from "./safety/destructive-confirmation.js";
38
+
39
+ // ── Taxonomy ────────────────────────────────────────────────────────────────
40
+
41
+ export type QuestionKind = "gate" | "consent" | "decision" | "informational";
42
+
43
+ export type GateSubKind = "depth-verification" | "approval" | "destructive-confirm";
44
+
45
+ export type FailPolicy = "closed" | "open";
46
+
47
+ export type AnswerOutcome = "waiting" | "answered" | "verified" | "declined" | "cancelled";
48
+
49
+ export interface ClassifiedQuestion {
50
+ kind: QuestionKind;
51
+ gateSubKind?: GateSubKind;
52
+ }
53
+
54
+ /** Fail policy is derived from the kind — there is no per-question override. */
55
+ export function failPolicyForKind(kind: QuestionKind): FailPolicy {
56
+ return kind === "informational" ? "open" : "closed";
57
+ }
58
+
59
+ // ── Prose detectors (moved from user-input-boundary) ────────────────────────
60
+
61
+ const REMOTE_QUESTION_FAILURE_RE =
62
+ /(?:Remote (?:auth failed|questions failed|channel configured but returned no result|questions timed out|questions timed out or failed)|Failed to send questions via)/i;
63
+
64
+ const APPROVAL_WAIT_RE =
65
+ /\bwait(?:ing)?\s+for\s+(?:your\s+)?(?:confirmation|approval|input|response|answer)\b/i;
66
+
67
+ const APPROVAL_QUESTION_RE =
68
+ /\b(?:confirm|confirmation|approve|approval|approved|captured|correct|correctly|happy\s+with|ready\s+to\s+(?:write|save|proceed|ship)|(?:want|need)\s+to\s+adjust|should\s+I\s+(?:write|save|proceed)|do\s+you\s+want\s+me\s+to\s+(?:write|save|proceed)|ship\s+it)\b/i;
69
+
70
+ const APPROVAL_RIGHT_QUESTION_RE =
71
+ /\b(?:does|do|is|are|was|were|did)\b[^\n?]{0,120}\bright\b/i;
72
+
73
+ const APPROVAL_CHANGE_QUESTION_RE =
74
+ /\b(?:anything\s+else|anything|something)\s+to\s+(?:adjust|add|remove|reclassify)\b/i;
75
+
76
+ const RESEARCH_DECISION_QUESTION_RE =
77
+ /\b(?:research|skip)\b/i;
78
+
79
+ const ASK_USER_QUESTIONS_CANCELLED_RE =
80
+ /ask_user_questions was cancelled before receiving a response/i;
81
+
82
+ /** Scan question-mark-terminated fragments of `text` against `patterns`. */
83
+ function hasQuestionMatching(text: string, patterns: RegExp[]): boolean {
84
+ for (let i = 0; i < text.length; i++) {
85
+ if (text[i] !== "?") continue;
86
+ const previousBreak = Math.max(
87
+ text.lastIndexOf("\n", i),
88
+ text.lastIndexOf(".", i),
89
+ text.lastIndexOf("!", i),
90
+ text.lastIndexOf("?", i - 1),
91
+ );
92
+ const fragment = text.slice(previousBreak + 1, i + 1);
93
+ if (patterns.some((pattern) => pattern.test(fragment))) return true;
94
+ }
95
+ return false;
96
+ }
97
+
98
+ export function hasApprovalQuestion(text: string): boolean {
99
+ return hasQuestionMatching(text, [
100
+ APPROVAL_QUESTION_RE,
101
+ APPROVAL_RIGHT_QUESTION_RE,
102
+ APPROVAL_CHANGE_QUESTION_RE,
103
+ ]);
104
+ }
105
+
106
+ export function hasResearchDecisionQuestion(text: string): boolean {
107
+ return hasQuestionMatching(text, [RESEARCH_DECISION_QUESTION_RE]);
108
+ }
109
+
110
+ // ── Message text extraction (moved from user-input-boundary) ────────────────
111
+
112
+ function extractMessageText(msg: unknown, includeThinking: boolean): string {
113
+ if (!msg || typeof msg !== "object") return "";
114
+ const content = (msg as { content?: unknown }).content;
115
+ if (typeof content === "string") return content;
116
+ if (!Array.isArray(content)) return "";
117
+ const parts: string[] = [];
118
+ for (const block of content) {
119
+ if (!block || typeof block !== "object") continue;
120
+ const typed = block as { type?: unknown; text?: unknown; thinking?: unknown };
121
+ if (typed.type === "text" && typeof typed.text === "string") {
122
+ parts.push(typed.text);
123
+ }
124
+ // Thinking blocks are internal reasoning, not user-visible — included only
125
+ // when the caller asks for the full transcript text.
126
+ if (includeThinking && typed.type === "thinking" && typeof typed.thinking === "string") {
127
+ parts.push(typed.thinking);
128
+ }
129
+ }
130
+ return parts.join("\n");
131
+ }
132
+
133
+ function lastAssistantMessageText(
134
+ messages: unknown[] | null | undefined,
135
+ includeThinking: boolean,
136
+ ): string {
137
+ if (!Array.isArray(messages)) return "";
138
+ for (let i = messages.length - 1; i >= 0; i--) {
139
+ const msg = messages[i];
140
+ if (!msg || typeof msg !== "object") continue;
141
+ if ((msg as { role?: unknown }).role !== "assistant") continue;
142
+ const text = extractMessageText(msg, includeThinking).trim();
143
+ if (text) return text;
144
+ }
145
+ return "";
146
+ }
147
+
148
+ export function lastAssistantText(messages: unknown[] | null | undefined): string {
149
+ return lastAssistantMessageText(messages, true);
150
+ }
151
+
152
+ function lastAssistantVisibleText(messages: unknown[] | null | undefined): string {
153
+ return lastAssistantMessageText(messages, false);
154
+ }
155
+
156
+ function anyMessageMatches(messages: unknown[] | undefined, patterns: RegExp[]): boolean {
157
+ if (!Array.isArray(messages)) return false;
158
+ return messages.some((msg) => {
159
+ if (!msg || typeof msg !== "object") return false;
160
+ if ((msg as { role?: unknown }).role === "user") return false;
161
+ const text = extractMessageText(msg, false);
162
+ return patterns.some((pattern) => pattern.test(text));
163
+ });
164
+ }
165
+
166
+ // ── Classification ──────────────────────────────────────────────────────────
167
+
168
+ const APPROVAL_GATE_ID_RE = /^depth_verification_.+_confirm$/;
169
+
170
+ export interface ClassifyQuestionInput {
171
+ /** ask_user_questions question id, when classifying a structured question. */
172
+ id?: unknown;
173
+ /** Question options, when classifying a structured question. */
174
+ options?: Array<{ label?: string }> | undefined;
175
+ /** Prose text, when classifying a streamed text boundary. */
176
+ text?: string;
177
+ /** Active unit type, used to pick the prose detector for decisions. */
178
+ unitType?: string;
179
+ }
180
+
181
+ /**
182
+ * Classify a question — structured (by id) or prose (by text) — into a kind.
183
+ *
184
+ * Structured questions with a recognized gate id are gates; every other
185
+ * structured question is asking the user something, so it classifies as
186
+ * consent (fail-closed). Prose classifies by the approval/decision detectors;
187
+ * prose that matches neither is informational (fail-open).
188
+ */
189
+ export function classifyQuestion(input: ClassifyQuestionInput): ClassifiedQuestion {
190
+ if (isDestructiveConfirmGateId(input.id)) {
191
+ return { kind: "gate", gateSubKind: "destructive-confirm" };
192
+ }
193
+ if (typeof input.id === "string" && isGateQuestionId(input.id)) {
194
+ return {
195
+ kind: "gate",
196
+ gateSubKind: APPROVAL_GATE_ID_RE.test(input.id) ? "approval" : "depth-verification",
197
+ };
198
+ }
199
+ // Any other structured question is a real elicitation of the user.
200
+ if (typeof input.id === "string" || Array.isArray(input.options)) {
201
+ return { kind: "consent" };
202
+ }
203
+ const text = input.text ?? "";
204
+ if (input.unitType === "research-decision" && hasResearchDecisionQuestion(text)) {
205
+ return { kind: "decision" };
206
+ }
207
+ if (hasApprovalQuestion(text)) {
208
+ return { kind: "consent" };
209
+ }
210
+ return { kind: "informational" };
211
+ }
212
+
213
+ // ── Answer validation ───────────────────────────────────────────────────────
214
+
215
+ // The question/answer shapes are the consent-verdict leaf's shapes — one
216
+ // definition shared with write-gate so the two consumers cannot drift.
217
+ export type ConsentQuestionShape = VerdictQuestionShape;
218
+ export type ConsentAnswerDetails = VerdictAnswerDetails;
219
+
220
+ /**
221
+ * THE single policy point for whether a question's answer counts as answered.
222
+ *
223
+ * - cancelled rounds → "cancelled" for every kind.
224
+ * - gate: delegates to evaluateGateAnswer in the consent-verdict leaf — the
225
+ * same verdict engine write-gate's applyAskUserQuestionsGateResult consumes;
226
+ * confirm option → "verified", any other real selection → "declined",
227
+ * empty/missing → "waiting" (fail-closed).
228
+ * - consent/decision: a non-empty selection or non-empty notes → "answered";
229
+ * empty/missing → "waiting" (fail-closed; fixes #528).
230
+ * - informational: always "answered" (fail-open).
231
+ */
232
+ export function evaluateAnswer(options: {
233
+ question: ConsentQuestionShape;
234
+ details: ConsentAnswerDetails;
235
+ }): AnswerOutcome {
236
+ const { question, details } = options;
237
+ if (details.cancelled) return "cancelled";
238
+
239
+ const { kind } = classifyQuestion({ id: question.id, options: question.options });
240
+ if (failPolicyForKind(kind) === "open") return "answered";
241
+
242
+ if (kind === "gate") {
243
+ // Gates keep strict structural validation: only the confirmation option
244
+ // verifies; notes never satisfy a gate.
245
+ return evaluateGateAnswer(question, details);
246
+ }
247
+
248
+ const questionId = typeof question.id === "string" ? question.id : "";
249
+ const answer = details.response?.answers?.[questionId];
250
+
251
+ if (hasSelectedValue(answer?.selected)) return "answered";
252
+ // Notes-only is a real user utterance for consent/decision questions, but
253
+ // never for gates (handled above).
254
+ if (hasNotesValue(answer?.notes)) return "answered";
255
+ return "waiting";
256
+ }
257
+
258
+ const OUTCOME_PRECEDENCE: AnswerOutcome[] = [
259
+ "cancelled",
260
+ "waiting",
261
+ "declined",
262
+ "verified",
263
+ "answered",
264
+ ];
265
+
266
+ /**
267
+ * Evaluate a whole ask_user_questions round. The round outcome is the most
268
+ * blocking per-question outcome (cancelled > waiting > declined > verified >
269
+ * answered). An empty round with a response is "answered"; an empty round
270
+ * without a response is "waiting" only when cancelled is not set and there is
271
+ * nothing to validate — callers treat a missing response with no questions as
272
+ * a no-op, so it reports "answered" here.
273
+ */
274
+ export function evaluateAskUserQuestionsRound(
275
+ questions: ConsentQuestionShape[],
276
+ details: ConsentAnswerDetails,
277
+ ): AnswerOutcome {
278
+ if (details.cancelled) return "cancelled";
279
+ let worst: AnswerOutcome = "answered";
280
+ for (const question of questions) {
281
+ const outcome = evaluateAnswer({ question, details });
282
+ if (OUTCOME_PRECEDENCE.indexOf(outcome) < OUTCOME_PRECEDENCE.indexOf(worst)) {
283
+ worst = outcome;
284
+ }
285
+ }
286
+ return worst;
287
+ }
288
+
289
+ export function formatUnansweredConsentQuestionMessage(questions: ConsentQuestionShape[]): string {
290
+ const ids = questions
291
+ .map((question) => (typeof question.id === "string" ? question.id : null))
292
+ .filter((id): id is string => Boolean(id));
293
+ return [
294
+ `ask_user_questions returned without a selection${ids.length ? ` for ${ids.join(", ")}` : ""}.`,
295
+ "An empty answer is not consent — do not infer approval or proceed.",
296
+ "Re-call ask_user_questions with the same question(s) and wait for the user's response.",
297
+ ].join(" ");
298
+ }
299
+
300
+ // ── Pause gating (replaces the unit-type allowlist) ─────────────────────────
301
+
302
+ /**
303
+ * Shared preamble for the awaiting-input predicates: cancellation and remote
304
+ * delivery failures always pause (an undelivered question can never be
305
+ * answered, so proceeding would be fail-open), as does an explicit
306
+ * "waiting for your approval/input" phrase in the last assistant text.
307
+ *
308
+ * Returns `forced: true` when the boundary is unconditional, plus the last
309
+ * assistant visible text for the caller's own classification.
310
+ */
311
+ function awaitingBoundary(messages: unknown[] | undefined): { forced: boolean; text: string } {
312
+ if (anyMessageMatches(messages, [ASK_USER_QUESTIONS_CANCELLED_RE, REMOTE_QUESTION_FAILURE_RE])) {
313
+ return { forced: true, text: "" };
314
+ }
315
+ const text = lastAssistantVisibleText(messages);
316
+ if (text && APPROVAL_WAIT_RE.test(text)) return { forced: true, text };
317
+ return { forced: false, text };
318
+ }
319
+
320
+ /**
321
+ * Decide whether the assistant should pause for a prose user question.
322
+ *
323
+ * Unlike the retired USER_APPROVAL_UNIT_TYPES allowlist, this pauses for any
324
+ * classified consent/decision question regardless of unit type — including
325
+ * interactive mode where no unit is active (#682).
326
+ */
327
+ export function shouldPauseForQuestion(
328
+ unitType: string | undefined,
329
+ messages: unknown[] | undefined,
330
+ ): boolean {
331
+ const { forced, text } = awaitingBoundary(messages);
332
+ if (forced) return true;
333
+ // Streaming hot path: this runs on every message_update for every unit type.
334
+ // The classifiers only ever match question-mark-terminated fragments, so
335
+ // text without a "?" can never classify as consent/decision — bail before
336
+ // the multi-regex scan.
337
+ if (!text || !text.includes("?")) return false;
338
+ const { kind } = classifyQuestion({ text, unitType });
339
+ return kind === "consent" || kind === "decision";
340
+ }
341
+
342
+ // ── Awaiting-input boundaries (moved from user-input-boundary) ──────────────
343
+
344
+ export function isAwaitingUserInput(messages: unknown[] | undefined): boolean {
345
+ const { forced, text } = awaitingBoundary(messages);
346
+ if (forced) return true;
347
+ if (!text) return false;
348
+ const lines = text.split(/\r?\n/).map((line) => line.trim()).filter(Boolean);
349
+ if (lines.some((line) => line.endsWith("?"))) return true;
350
+ return hasApprovalQuestion(text);
351
+ }
352
+
353
+ export function isAwaitingApprovalBoundary(messages: unknown[] | undefined): boolean {
354
+ // With no unit type, classification reduces to the approval detectors —
355
+ // exactly the approval boundary.
356
+ return shouldPauseForQuestion(undefined, messages);
357
+ }
358
+
359
+ // ── Approval gate ids + explicit responses (moved from user-input-boundary) ─
360
+
361
+ export function approvalGateIdForUnit(
362
+ unitType: string | undefined,
363
+ unitId?: string | null,
364
+ ): string | null {
365
+ if (!unitType) return null;
366
+ if (unitType === "discuss-project") return "depth_verification_project_confirm";
367
+ if (unitType === "discuss-requirements") return "depth_verification_requirements_confirm";
368
+ if (unitType === "research-decision") return "depth_verification_research_decision_confirm";
369
+ if (unitType === "discuss-milestone") {
370
+ const safeUnitId = typeof unitId === "string" && /^[A-Za-z0-9_-]+$/.test(unitId)
371
+ ? unitId
372
+ : "milestone";
373
+ return `depth_verification_${safeUnitId}_confirm`;
374
+ }
375
+ return null;
376
+ }
377
+
378
+ const CHANGE_REQUEST_RESPONSE_RE =
379
+ /\b(?:no|nope|nah|not\s+yet|don't|do\s+not|change|add|remove|reclassify|adjust|clarify|missing|instead|but|however|wait|hold)\b/i;
380
+
381
+ const APPROVAL_RESPONSE_RE =
382
+ /^(?:y|yes|yeah|yep|approve|approved|confirm|confirmed|correct|right|looks\s+(?:good|right)|sounds\s+good|all\s+good|ok|okay|go\s+ahead|proceed|write\s+it|save\s+it|do\s+it)\b/i;
383
+
384
+ const RESEARCH_DECISION_RESPONSE_RE =
385
+ /^(?:research|run\s+research|do\s+research|skip|skip\s+research|no\s+research)\b/i;
386
+
387
+ export function isExplicitApprovalResponse(
388
+ input: string | undefined,
389
+ pendingGateId?: string | null,
390
+ ): boolean {
391
+ const text = input?.trim() ?? "";
392
+ if (!text) return false;
393
+ if (pendingGateId?.includes("research_decision")) {
394
+ return RESEARCH_DECISION_RESPONSE_RE.test(text);
395
+ }
396
+ if (CHANGE_REQUEST_RESPONSE_RE.test(text)) return false;
397
+ return APPROVAL_RESPONSE_RE.test(text);
398
+ }
399
+
400
+ /** True when an assistant message already has an in-flight ask_user_questions tool call. */
401
+ export function messageHasPendingAskUserQuestionsTool(message: unknown): boolean {
402
+ if (!message || typeof message !== "object") return false;
403
+ const content = (message as { content?: unknown }).content;
404
+ if (!Array.isArray(content)) return false;
405
+ return content.some((block) => {
406
+ if (!block || typeof block !== "object") return false;
407
+ // Claude Code marks completion by attaching externalResult, not by setting state.
408
+ // Streaming blocks often carry no state; serverToolUse is the claude-code-cli MCP path.
409
+ const tool = block as { type?: string; name?: string; state?: string; externalResult?: unknown };
410
+ if (tool.type !== "toolCall" && tool.type !== "serverToolUse") return false;
411
+ const name = String(tool.name ?? "").toLowerCase();
412
+ if (!name.includes("ask_user_questions")) return false;
413
+ if (tool.externalResult !== undefined) return false;
414
+ return tool.state !== "completed" && tool.state !== "done";
415
+ });
416
+ }
@@ -0,0 +1,86 @@
1
+ /**
2
+ * Consent verdict leaf — the single per-question verdict engine shared by the
3
+ * write gate (bootstrap/write-gate.ts) and the Consent Question module
4
+ * (consent-question.ts).
5
+ *
6
+ * This module is a dependency leaf on purpose: write-gate consumes
7
+ * evaluateGateAnswer here while consent-question imports write-gate's id
8
+ * predicates, so putting the verdict anywhere else would create an import
9
+ * cycle. It must not import from either module (ADR-039).
10
+ */
11
+
12
+ export type GateAnswerVerdict = "waiting" | "verified" | "declined" | "cancelled";
13
+
14
+ export interface VerdictQuestionShape {
15
+ id?: unknown;
16
+ options?: Array<{ label?: string }>;
17
+ }
18
+
19
+ export interface VerdictAnswerDetails {
20
+ cancelled?: boolean;
21
+ interrupted?: boolean;
22
+ response?: {
23
+ answers?: Record<string, { selected?: unknown; notes?: unknown } | undefined>;
24
+ } | null;
25
+ }
26
+
27
+ /**
28
+ * Check whether a depth_verification answer confirms the discussion is complete.
29
+ * Uses structural validation: the selected answer must exactly match the first
30
+ * option label from the question definition (the confirmation option by convention).
31
+ * This rejects free-form "Other" text, decline options, and garbage input without
32
+ * coupling to any specific label substring.
33
+ *
34
+ * @param selected The answer's selected value from details.response.answers[id].selected
35
+ * @param options The question's options array from event.input.questions[n].options
36
+ */
37
+ export function isDepthConfirmationAnswer(
38
+ selected: unknown,
39
+ options?: Array<{ label?: string }>,
40
+ ): boolean {
41
+ const value = Array.isArray(selected) ? selected[0] : selected;
42
+ if (typeof value !== "string" || !value) return false;
43
+
44
+ // If options are available, structurally validate: selected must exactly match
45
+ // the first option (confirmation) label. Rejects free-form "Other" and decline options.
46
+ if (Array.isArray(options) && options.length > 0) {
47
+ const confirmLabel = options[0]?.label;
48
+ return typeof confirmLabel === "string" && value === confirmLabel;
49
+ }
50
+
51
+ // Fail-closed: no options means we cannot structurally validate the answer.
52
+ // Returning false prevents any free-form string from unlocking the gate.
53
+ return false;
54
+ }
55
+
56
+ export function hasSelectedValue(selected: unknown): boolean {
57
+ if (Array.isArray(selected)) {
58
+ return selected.some((value) => typeof value === "string" && value.length > 0);
59
+ }
60
+ return typeof selected === "string" && selected.length > 0;
61
+ }
62
+
63
+ export function hasNotesValue(notes: unknown): boolean {
64
+ return typeof notes === "string" && notes.trim().length > 0;
65
+ }
66
+
67
+ /**
68
+ * THE per-question verdict for gate questions (fail-closed):
69
+ *
70
+ * - cancelled rounds → "cancelled".
71
+ * - the confirmation option (structural match) → "verified".
72
+ * - any other real selection → "declined".
73
+ * - empty/missing selection → "waiting" — an empty answer is NEVER an answer,
74
+ * so notes can never satisfy a gate either.
75
+ */
76
+ export function evaluateGateAnswer(
77
+ question: VerdictQuestionShape,
78
+ details: VerdictAnswerDetails,
79
+ ): GateAnswerVerdict {
80
+ if (details.cancelled) return "cancelled";
81
+ const questionId = typeof question.id === "string" ? question.id : "";
82
+ const answer = details.response?.answers?.[questionId];
83
+ if (isDepthConfirmationAnswer(answer?.selected, question.options)) return "verified";
84
+ if (hasSelectedValue(answer?.selected)) return "declined";
85
+ return "waiting";
86
+ }
@@ -9,9 +9,6 @@
9
9
  /** Default timeout for verification-gate commands (ms). */
10
10
  export const DEFAULT_COMMAND_TIMEOUT_MS = 120_000;
11
11
 
12
- /** Default timeout for the dynamic bash tool (seconds). */
13
- export const DEFAULT_BASH_TIMEOUT_SECS = 120;
14
-
15
12
  // ─── Cache Sizes ──────────────────────────────────────────────────────────────
16
13
 
17
14
  /** Max directory-listing cache entries before eviction (#611). */