gsd-pi 2.77.0-dev.eaa4973bc → 2.78.0-dev.aeeb2ca00

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 (545) hide show
  1. package/README.md +53 -17
  2. package/dist/claude-cli-check.js +46 -10
  3. package/dist/headless.js +49 -4
  4. package/dist/resource-loader.d.ts +40 -0
  5. package/dist/resource-loader.js +32 -13
  6. package/dist/resources/extensions/browser-tools/capture.js +9 -0
  7. package/dist/resources/extensions/browser-tools/tests/browser-tools-integration.test.mjs +8 -59
  8. package/dist/resources/extensions/browser-tools/tests/browser-tools-unit.test.cjs +36 -24
  9. package/dist/resources/extensions/browser-tools/tests/capture-sharp-optional.test.cjs +69 -71
  10. package/dist/resources/extensions/browser-tools/tools/forms.js +5 -1
  11. package/dist/resources/extensions/browser-tools/tools/intent.js +5 -1
  12. package/dist/resources/extensions/claude-code-cli/readiness.js +72 -16
  13. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +481 -17
  14. package/dist/resources/extensions/github-sync/templates.js +103 -0
  15. package/dist/resources/extensions/google-search/index.js +3 -2
  16. package/dist/resources/extensions/gsd/auto/loop.js +124 -2
  17. package/dist/resources/extensions/gsd/auto/phases.js +57 -39
  18. package/dist/resources/extensions/gsd/auto/session.js +6 -2
  19. package/dist/resources/extensions/gsd/auto-dispatch.js +142 -29
  20. package/dist/resources/extensions/gsd/auto-model-selection.js +124 -4
  21. package/dist/resources/extensions/gsd/auto-post-unit.js +150 -64
  22. package/dist/resources/extensions/gsd/auto-prompts.js +372 -104
  23. package/dist/resources/extensions/gsd/auto-recovery.js +197 -48
  24. package/dist/resources/extensions/gsd/auto-start.js +107 -29
  25. package/dist/resources/extensions/gsd/auto-tool-tracking.js +47 -7
  26. package/dist/resources/extensions/gsd/auto-worktree.js +122 -26
  27. package/dist/resources/extensions/gsd/auto.js +76 -21
  28. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +19 -1
  29. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +209 -0
  30. package/dist/resources/extensions/gsd/bootstrap/provider-error-resume.js +3 -6
  31. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +7 -3
  32. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +127 -9
  33. package/dist/resources/extensions/gsd/component-loader.js +447 -0
  34. package/dist/resources/extensions/gsd/component-types.js +69 -0
  35. package/dist/resources/extensions/gsd/context-store.js +23 -7
  36. package/dist/resources/extensions/gsd/detection.js +49 -1
  37. package/dist/resources/extensions/gsd/dispatch-guard.js +2 -17
  38. package/dist/resources/extensions/gsd/docs/preferences-reference.md +1 -1
  39. package/dist/resources/extensions/gsd/forensics.js +106 -0
  40. package/dist/resources/extensions/gsd/gate-registry.js +2 -2
  41. package/dist/resources/extensions/gsd/git-constants.js +28 -1
  42. package/dist/resources/extensions/gsd/git-self-heal.js +27 -0
  43. package/dist/resources/extensions/gsd/git-service.js +126 -2
  44. package/dist/resources/extensions/gsd/gsd-db.js +6 -3
  45. package/dist/resources/extensions/gsd/guided-flow.js +39 -13
  46. package/dist/resources/extensions/gsd/memory-extractor.js +7 -1
  47. package/dist/resources/extensions/gsd/milestone-scope-classifier.js +299 -0
  48. package/dist/resources/extensions/gsd/milestone-summary-classifier.js +37 -0
  49. package/dist/resources/extensions/gsd/model-cost-table.js +3 -0
  50. package/dist/resources/extensions/gsd/model-router.js +6 -0
  51. package/dist/resources/extensions/gsd/native-git-bridge.js +34 -4
  52. package/dist/resources/extensions/gsd/preferences-validation.js +23 -0
  53. package/dist/resources/extensions/gsd/prompt-cache-optimizer.js +4 -0
  54. package/dist/resources/extensions/gsd/prompts/complete-milestone.md +6 -2
  55. package/dist/resources/extensions/gsd/prompts/discuss-headless.md +23 -4
  56. package/dist/resources/extensions/gsd/prompts/doctor-heal.md +5 -4
  57. package/dist/resources/extensions/gsd/prompts/plan-slice.md +15 -2
  58. package/dist/resources/extensions/gsd/safety/git-checkpoint.js +11 -0
  59. package/dist/resources/extensions/gsd/service-tier.js +5 -2
  60. package/dist/resources/extensions/gsd/session-lock.js +19 -10
  61. package/dist/resources/extensions/gsd/skill-manifest.js +168 -0
  62. package/dist/resources/extensions/gsd/slice-cadence.js +238 -0
  63. package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +278 -8
  64. package/dist/resources/extensions/gsd/state-transition-matrix.js +118 -0
  65. package/dist/resources/extensions/gsd/state.js +69 -58
  66. package/dist/resources/extensions/gsd/sync-lock.js +98 -42
  67. package/dist/resources/extensions/gsd/tools/validate-milestone.js +7 -2
  68. package/dist/resources/extensions/gsd/unit-context-composer.js +147 -0
  69. package/dist/resources/extensions/gsd/unit-context-manifest.js +370 -0
  70. package/dist/resources/extensions/gsd/uok/dispatch-envelope.js +33 -0
  71. package/dist/resources/extensions/gsd/uok/execution-graph.js +10 -0
  72. package/dist/resources/extensions/gsd/uok/gate-runner.js +53 -5
  73. package/dist/resources/extensions/gsd/uok/gitops.js +2 -1
  74. package/dist/resources/extensions/gsd/uok/loop-adapter.js +37 -10
  75. package/dist/resources/extensions/gsd/uok/parity-report.js +58 -0
  76. package/dist/resources/extensions/gsd/uok/plan-v2.js +10 -4
  77. package/dist/resources/extensions/gsd/uok/writer.js +82 -0
  78. package/dist/resources/extensions/gsd/workflow-mcp.js +6 -0
  79. package/dist/resources/extensions/gsd/worktree-manager.js +85 -8
  80. package/dist/resources/extensions/gsd/worktree-resolver.js +86 -7
  81. package/dist/resources/extensions/gsd/worktree-telemetry.js +198 -0
  82. package/dist/resources/extensions/mcp-client/index.js +3 -1
  83. package/dist/resources/extensions/ollama/index.js +5 -1
  84. package/dist/resources/extensions/remote-questions/manager.js +11 -5
  85. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  86. package/dist/web/standalone/.next/BUILD_ID +1 -1
  87. package/dist/web/standalone/.next/app-path-routes-manifest.json +11 -11
  88. package/dist/web/standalone/.next/build-manifest.json +2 -2
  89. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  90. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  91. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  92. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  93. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  94. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  95. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  96. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  97. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  98. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  99. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  100. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  101. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  102. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  103. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  104. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  105. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  106. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  107. package/dist/web/standalone/.next/server/app/index.html +1 -1
  108. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  109. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  110. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  111. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  112. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  113. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  114. package/dist/web/standalone/.next/server/app-paths-manifest.json +11 -11
  115. package/dist/web/standalone/.next/server/chunks/1926.js +1 -1
  116. package/dist/web/standalone/.next/server/chunks/6897.js +1 -1
  117. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  118. package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
  119. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  120. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  121. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  122. package/package.json +2 -3
  123. package/packages/daemon/package.json +2 -2
  124. package/packages/daemon/src/logger.ts +4 -3
  125. package/packages/mcp-server/dist/server.d.ts +24 -0
  126. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  127. package/packages/mcp-server/dist/server.js +88 -87
  128. package/packages/mcp-server/dist/server.js.map +1 -1
  129. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  130. package/packages/mcp-server/dist/workflow-tools.js +15 -6
  131. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  132. package/packages/mcp-server/package.json +2 -2
  133. package/packages/mcp-server/src/mcp-server.test.ts +25 -3
  134. package/packages/mcp-server/src/readers/graph.test.ts +87 -15
  135. package/packages/mcp-server/src/secure-env-collect.test.ts +232 -237
  136. package/packages/mcp-server/src/server.ts +131 -105
  137. package/packages/mcp-server/src/workflow-tools.test.ts +85 -0
  138. package/packages/mcp-server/src/workflow-tools.ts +19 -6
  139. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
  140. package/packages/native/package.json +2 -2
  141. package/packages/native/src/__tests__/_test-coverage-guard.test.mjs +98 -0
  142. package/packages/native/src/__tests__/module-compat.test.mjs +59 -27
  143. package/packages/native/src/__tests__/ps.test.mjs +14 -8
  144. package/packages/native/src/__tests__/stream-process.test.mjs +23 -2
  145. package/packages/native/src/__tests__/truncate.test.mjs +17 -2
  146. package/packages/pi-agent-core/package.json +1 -1
  147. package/packages/pi-agent-core/src/agent-loop.test.ts +5 -15
  148. package/packages/pi-agent-core/src/agent.test.ts +96 -102
  149. package/packages/pi-agent-core/tsconfig.tsbuildinfo +1 -1
  150. package/packages/pi-ai/dist/models/capability-patches.d.ts.map +1 -1
  151. package/packages/pi-ai/dist/models/capability-patches.js +9 -2
  152. package/packages/pi-ai/dist/models/capability-patches.js.map +1 -1
  153. package/packages/pi-ai/dist/models/generated/index.d.ts +34 -0
  154. package/packages/pi-ai/dist/models/generated/index.d.ts.map +1 -1
  155. package/packages/pi-ai/dist/models/generated/openai-codex.d.ts +17 -0
  156. package/packages/pi-ai/dist/models/generated/openai-codex.d.ts.map +1 -1
  157. package/packages/pi-ai/dist/models/generated/openai-codex.js +17 -0
  158. package/packages/pi-ai/dist/models/generated/openai-codex.js.map +1 -1
  159. package/packages/pi-ai/dist/models/generated/openai.d.ts +17 -0
  160. package/packages/pi-ai/dist/models/generated/openai.d.ts.map +1 -1
  161. package/packages/pi-ai/dist/models/generated/openai.js +17 -0
  162. package/packages/pi-ai/dist/models/generated/openai.js.map +1 -1
  163. package/packages/pi-ai/dist/models.generated.test.js +43 -70
  164. package/packages/pi-ai/dist/models.generated.test.js.map +1 -1
  165. package/packages/pi-ai/dist/models.test.js +36 -11
  166. package/packages/pi-ai/dist/models.test.js.map +1 -1
  167. package/packages/pi-ai/package.json +1 -1
  168. package/packages/pi-ai/scripts/generate-models.ts +44 -0
  169. package/packages/pi-ai/src/models/capability-patches.ts +10 -2
  170. package/packages/pi-ai/src/models/generated/openai-codex.ts +17 -0
  171. package/packages/pi-ai/src/models/generated/openai.ts +17 -0
  172. package/packages/pi-ai/src/models.generated.test.ts +46 -73
  173. package/packages/pi-ai/src/models.test.ts +48 -11
  174. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
  175. package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js +96 -32
  176. package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js.map +1 -1
  177. package/packages/pi-coding-agent/dist/core/agent-session-model-switch.test.js +75 -12
  178. package/packages/pi-coding-agent/dist/core/agent-session-model-switch.test.js.map +1 -1
  179. package/packages/pi-coding-agent/dist/core/agent-session-tool-refresh.test.js +99 -31
  180. package/packages/pi-coding-agent/dist/core/agent-session-tool-refresh.test.js.map +1 -1
  181. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts +5 -0
  182. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts.map +1 -1
  183. package/packages/pi-coding-agent/dist/core/extensions/loader.js +61 -0
  184. package/packages/pi-coding-agent/dist/core/extensions/loader.js.map +1 -1
  185. package/packages/pi-coding-agent/dist/core/lsp/lsp-integration.test.js +30 -4
  186. package/packages/pi-coding-agent/dist/core/lsp/lsp-integration.test.js.map +1 -1
  187. package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js +17 -0
  188. package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js.map +1 -1
  189. package/packages/pi-coding-agent/dist/core/resource-loader-cache-reset.test.js +76 -18
  190. package/packages/pi-coding-agent/dist/core/resource-loader-cache-reset.test.js.map +1 -1
  191. package/packages/pi-coding-agent/dist/core/retry-handler.d.ts.map +1 -1
  192. package/packages/pi-coding-agent/dist/core/retry-handler.js +2 -6
  193. package/packages/pi-coding-agent/dist/core/retry-handler.js.map +1 -1
  194. package/packages/pi-coding-agent/dist/core/retry-handler.test.js +5 -1
  195. package/packages/pi-coding-agent/dist/core/retry-handler.test.js.map +1 -1
  196. package/packages/pi-coding-agent/dist/core/retryable-error-regex.d.ts +18 -0
  197. package/packages/pi-coding-agent/dist/core/retryable-error-regex.d.ts.map +1 -0
  198. package/packages/pi-coding-agent/dist/core/retryable-error-regex.js +18 -0
  199. package/packages/pi-coding-agent/dist/core/retryable-error-regex.js.map +1 -0
  200. package/packages/pi-coding-agent/dist/core/system-prompt.d.ts +20 -0
  201. package/packages/pi-coding-agent/dist/core/system-prompt.d.ts.map +1 -1
  202. package/packages/pi-coding-agent/dist/core/system-prompt.js +16 -2
  203. package/packages/pi-coding-agent/dist/core/system-prompt.js.map +1 -1
  204. package/packages/pi-coding-agent/dist/index.d.ts +1 -0
  205. package/packages/pi-coding-agent/dist/index.d.ts.map +1 -1
  206. package/packages/pi-coding-agent/dist/index.js +1 -0
  207. package/packages/pi-coding-agent/dist/index.js.map +1 -1
  208. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js +36 -5
  209. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js.map +1 -1
  210. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js +20 -13
  211. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.test.js.map +1 -1
  212. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  213. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +30 -12
  214. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
  215. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.d.ts.map +1 -1
  216. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js +18 -3
  217. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js.map +1 -1
  218. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js +125 -0
  219. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js.map +1 -1
  220. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.d.ts +2 -0
  221. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.d.ts.map +1 -1
  222. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.js.map +1 -1
  223. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +4 -0
  224. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  225. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +105 -13
  226. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  227. package/packages/pi-coding-agent/dist/tests/system-prompt-skill-filter.test.d.ts +2 -0
  228. package/packages/pi-coding-agent/dist/tests/system-prompt-skill-filter.test.d.ts.map +1 -0
  229. package/packages/pi-coding-agent/dist/tests/system-prompt-skill-filter.test.js +130 -0
  230. package/packages/pi-coding-agent/dist/tests/system-prompt-skill-filter.test.js.map +1 -0
  231. package/packages/pi-coding-agent/package.json +1 -1
  232. package/packages/pi-coding-agent/src/core/agent-session-abort-order.test.ts +113 -37
  233. package/packages/pi-coding-agent/src/core/agent-session-model-switch.test.ts +89 -17
  234. package/packages/pi-coding-agent/src/core/agent-session-tool-refresh.test.ts +112 -43
  235. package/packages/pi-coding-agent/src/core/extensions/loader.ts +58 -0
  236. package/packages/pi-coding-agent/src/core/lsp/lsp-integration.test.ts +35 -4
  237. package/packages/pi-coding-agent/src/core/model-registry-auth-mode.test.ts +20 -0
  238. package/packages/pi-coding-agent/src/core/resource-loader-cache-reset.test.ts +93 -28
  239. package/packages/pi-coding-agent/src/core/retry-handler.test.ts +5 -1
  240. package/packages/pi-coding-agent/src/core/retry-handler.ts +2 -8
  241. package/packages/pi-coding-agent/src/core/retryable-error-regex.ts +18 -0
  242. package/packages/pi-coding-agent/src/core/system-prompt.ts +35 -1
  243. package/packages/pi-coding-agent/src/index.ts +1 -0
  244. package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/tool-execution.test.ts +49 -3
  245. package/packages/pi-coding-agent/src/modes/interactive/components/dynamic-border.test.ts +26 -20
  246. package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +48 -9
  247. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.test.ts +146 -1
  248. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.ts +20 -3
  249. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode-state.ts +2 -0
  250. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +119 -13
  251. package/packages/pi-coding-agent/src/tests/system-prompt-skill-filter.test.ts +157 -0
  252. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
  253. package/packages/pi-tui/dist/__tests__/autocomplete.test.js +18 -8
  254. package/packages/pi-tui/dist/__tests__/autocomplete.test.js.map +1 -1
  255. package/packages/pi-tui/dist/__tests__/overlay-layout.test.js +128 -17
  256. package/packages/pi-tui/dist/__tests__/overlay-layout.test.js.map +1 -1
  257. package/packages/pi-tui/dist/__tests__/stdin-buffer.test.js +37 -11
  258. package/packages/pi-tui/dist/__tests__/stdin-buffer.test.js.map +1 -1
  259. package/packages/pi-tui/dist/__tests__/tui.test.js +18 -30
  260. package/packages/pi-tui/dist/__tests__/tui.test.js.map +1 -1
  261. package/packages/pi-tui/dist/components/__tests__/input.test.js +10 -3
  262. package/packages/pi-tui/dist/components/__tests__/input.test.js.map +1 -1
  263. package/packages/pi-tui/dist/components/__tests__/loader.test.js +53 -9
  264. package/packages/pi-tui/dist/components/__tests__/loader.test.js.map +1 -1
  265. package/packages/pi-tui/dist/components/__tests__/markdown-maxlines.test.js +6 -2
  266. package/packages/pi-tui/dist/components/__tests__/markdown-maxlines.test.js.map +1 -1
  267. package/packages/pi-tui/dist/components/editor.d.ts +14 -0
  268. package/packages/pi-tui/dist/components/editor.d.ts.map +1 -1
  269. package/packages/pi-tui/dist/components/editor.js +19 -0
  270. package/packages/pi-tui/dist/components/editor.js.map +1 -1
  271. package/packages/pi-tui/dist/components/image.test.js +6 -5
  272. package/packages/pi-tui/dist/components/image.test.js.map +1 -1
  273. package/packages/pi-tui/dist/editor-component.d.ts +2 -0
  274. package/packages/pi-tui/dist/editor-component.d.ts.map +1 -1
  275. package/packages/pi-tui/dist/editor-component.js.map +1 -1
  276. package/packages/pi-tui/package.json +1 -1
  277. package/packages/pi-tui/src/__tests__/autocomplete.test.ts +24 -8
  278. package/packages/pi-tui/src/__tests__/overlay-layout.test.ts +140 -17
  279. package/packages/pi-tui/src/__tests__/stdin-buffer.test.ts +42 -11
  280. package/packages/pi-tui/src/__tests__/tui.test.ts +18 -37
  281. package/packages/pi-tui/src/components/__tests__/input.test.ts +19 -3
  282. package/packages/pi-tui/src/components/__tests__/loader.test.ts +112 -35
  283. package/packages/pi-tui/src/components/__tests__/markdown-maxlines.test.ts +9 -2
  284. package/packages/pi-tui/src/components/editor.ts +22 -0
  285. package/packages/pi-tui/src/components/image.test.ts +10 -5
  286. package/packages/pi-tui/src/editor-component.ts +3 -0
  287. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
  288. package/packages/rpc-client/dist/rpc-client.test.js +101 -51
  289. package/packages/rpc-client/dist/rpc-client.test.js.map +1 -1
  290. package/packages/rpc-client/package.json +1 -1
  291. package/packages/rpc-client/src/rpc-client.test.ts +109 -52
  292. package/packages/rpc-client/tsconfig.tsbuildinfo +1 -1
  293. package/pkg/package.json +1 -1
  294. package/scripts/install.js +15 -1
  295. package/src/resources/extensions/browser-tools/capture.ts +12 -0
  296. package/src/resources/extensions/browser-tools/tests/browser-tools-integration.test.mjs +8 -59
  297. package/src/resources/extensions/browser-tools/tests/browser-tools-unit.test.cjs +36 -24
  298. package/src/resources/extensions/browser-tools/tests/capture-sharp-optional.test.cjs +69 -71
  299. package/src/resources/extensions/browser-tools/tools/forms.ts +5 -1
  300. package/src/resources/extensions/browser-tools/tools/intent.ts +5 -1
  301. package/src/resources/extensions/claude-code-cli/readiness.ts +75 -16
  302. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +518 -19
  303. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +919 -75
  304. package/src/resources/extensions/github-sync/templates.ts +151 -0
  305. package/src/resources/extensions/github-sync/tests/cli.test.ts +76 -7
  306. package/src/resources/extensions/github-sync/tests/templates.test.ts +92 -1
  307. package/src/resources/extensions/google-search/index.ts +3 -2
  308. package/src/resources/extensions/gsd/auto/loop.ts +142 -2
  309. package/src/resources/extensions/gsd/auto/phases.ts +62 -38
  310. package/src/resources/extensions/gsd/auto/session.ts +7 -2
  311. package/src/resources/extensions/gsd/auto-dispatch.ts +156 -29
  312. package/src/resources/extensions/gsd/auto-model-selection.ts +131 -4
  313. package/src/resources/extensions/gsd/auto-post-unit.ts +163 -73
  314. package/src/resources/extensions/gsd/auto-prompts.ts +385 -93
  315. package/src/resources/extensions/gsd/auto-recovery.ts +230 -51
  316. package/src/resources/extensions/gsd/auto-start.ts +127 -9
  317. package/src/resources/extensions/gsd/auto-tool-tracking.ts +51 -7
  318. package/src/resources/extensions/gsd/auto-worktree.ts +130 -26
  319. package/src/resources/extensions/gsd/auto.ts +90 -23
  320. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +20 -1
  321. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +221 -0
  322. package/src/resources/extensions/gsd/bootstrap/provider-error-resume.ts +3 -7
  323. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +7 -3
  324. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +158 -9
  325. package/src/resources/extensions/gsd/component-loader.ts +598 -0
  326. package/src/resources/extensions/gsd/component-types.ts +362 -0
  327. package/src/resources/extensions/gsd/context-store.ts +25 -8
  328. package/src/resources/extensions/gsd/detection.ts +58 -1
  329. package/src/resources/extensions/gsd/dispatch-guard.ts +2 -20
  330. package/src/resources/extensions/gsd/docs/preferences-reference.md +1 -1
  331. package/src/resources/extensions/gsd/forensics.ts +118 -1
  332. package/src/resources/extensions/gsd/gate-registry.ts +2 -2
  333. package/src/resources/extensions/gsd/git-constants.ts +30 -1
  334. package/src/resources/extensions/gsd/git-self-heal.ts +31 -0
  335. package/src/resources/extensions/gsd/git-service.ts +149 -2
  336. package/src/resources/extensions/gsd/gsd-db.ts +6 -3
  337. package/src/resources/extensions/gsd/guided-flow.ts +57 -14
  338. package/src/resources/extensions/gsd/journal.ts +11 -1
  339. package/src/resources/extensions/gsd/memory-extractor.ts +11 -3
  340. package/src/resources/extensions/gsd/milestone-scope-classifier.ts +366 -0
  341. package/src/resources/extensions/gsd/milestone-summary-classifier.ts +42 -0
  342. package/src/resources/extensions/gsd/model-cost-table.ts +3 -0
  343. package/src/resources/extensions/gsd/model-router.ts +6 -0
  344. package/src/resources/extensions/gsd/native-git-bridge.ts +34 -4
  345. package/src/resources/extensions/gsd/preferences-validation.ts +21 -0
  346. package/src/resources/extensions/gsd/prompt-cache-optimizer.ts +4 -0
  347. package/src/resources/extensions/gsd/prompts/complete-milestone.md +6 -2
  348. package/src/resources/extensions/gsd/prompts/discuss-headless.md +23 -4
  349. package/src/resources/extensions/gsd/prompts/doctor-heal.md +5 -4
  350. package/src/resources/extensions/gsd/prompts/plan-slice.md +15 -2
  351. package/src/resources/extensions/gsd/safety/git-checkpoint.ts +15 -0
  352. package/src/resources/extensions/gsd/service-tier.ts +5 -2
  353. package/src/resources/extensions/gsd/session-lock.ts +20 -10
  354. package/src/resources/extensions/gsd/skill-manifest.ts +175 -0
  355. package/src/resources/extensions/gsd/slice-cadence.ts +299 -0
  356. package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +309 -8
  357. package/src/resources/extensions/gsd/state-transition-matrix.ts +152 -0
  358. package/src/resources/extensions/gsd/state.ts +76 -66
  359. package/src/resources/extensions/gsd/sync-lock.ts +97 -39
  360. package/src/resources/extensions/gsd/tests/artifact-retry-cap.test.ts +270 -0
  361. package/src/resources/extensions/gsd/tests/artifacts-table-preserved-on-cache-invalidate.test.ts +2 -1
  362. package/src/resources/extensions/gsd/tests/auto-deterministic-error-classification-4973.test.ts +341 -0
  363. package/src/resources/extensions/gsd/tests/auto-discuss-milestone-deadlock-4973.test.ts +264 -0
  364. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +133 -292
  365. package/src/resources/extensions/gsd/tests/auto-model-selection-tool-poisoning.test.ts +742 -0
  366. package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +78 -0
  367. package/src/resources/extensions/gsd/tests/auto-phases-lifecycle.test.ts +61 -0
  368. package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +93 -0
  369. package/src/resources/extensions/gsd/tests/auto-remediate-slice-status.test.ts +4 -1
  370. package/src/resources/extensions/gsd/tests/auto-retry-mcp-churn-fixes.test.ts +8 -194
  371. package/src/resources/extensions/gsd/tests/auto-start-clean-runtime-db-gated.test.ts +3 -2
  372. package/src/resources/extensions/gsd/tests/auto-start-cold-db-bootstrap.test.ts +2 -2
  373. package/src/resources/extensions/gsd/tests/auto-start-needs-discussion.test.ts +15 -58
  374. package/src/resources/extensions/gsd/tests/auto-start-worktree-db-path.test.ts +2 -2
  375. package/src/resources/extensions/gsd/tests/auto-thinking-restore.test.ts +3 -2
  376. package/src/resources/extensions/gsd/tests/auto-warning-noise-regression.test.ts +3 -2
  377. package/src/resources/extensions/gsd/tests/bootstrap-derive-state-db-open.test.ts +2 -1
  378. package/src/resources/extensions/gsd/tests/cache-staleness-regression.test.ts +17 -21
  379. package/src/resources/extensions/gsd/tests/canonical-milestone-root.test.ts +108 -0
  380. package/src/resources/extensions/gsd/tests/complete-milestone-excerpt.test.ts +263 -0
  381. package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +25 -0
  382. package/src/resources/extensions/gsd/tests/complete-slice-composer.test.ts +192 -0
  383. package/src/resources/extensions/gsd/tests/complete-slice-verification-gate.test.ts +2 -1
  384. package/src/resources/extensions/gsd/tests/complete-task.test.ts +16 -8
  385. package/src/resources/extensions/gsd/tests/component-loader.test.ts +589 -0
  386. package/src/resources/extensions/gsd/tests/component-types.test.ts +127 -0
  387. package/src/resources/extensions/gsd/tests/context-store.test.ts +79 -0
  388. package/src/resources/extensions/gsd/tests/copy-planning-artifacts-samepath.test.ts +2 -1
  389. package/src/resources/extensions/gsd/tests/crash-recovery.test.ts +50 -1
  390. package/src/resources/extensions/gsd/tests/custom-engine-loop-integration.test.ts +159 -0
  391. package/src/resources/extensions/gsd/tests/db-access-guardrails.test.ts +1 -0
  392. package/src/resources/extensions/gsd/tests/derive-state-crossval.test.ts +3 -3
  393. package/src/resources/extensions/gsd/tests/derive-state-db-disk-reconcile.test.ts +40 -0
  394. package/src/resources/extensions/gsd/tests/derive-state-db.test.ts +91 -3
  395. package/src/resources/extensions/gsd/tests/derive-state.test.ts +4 -4
  396. package/src/resources/extensions/gsd/tests/discuss-slice-structured-questions.test.ts +2 -1
  397. package/src/resources/extensions/gsd/tests/discuss-tool-scope-leak.test.ts +2 -1
  398. package/src/resources/extensions/gsd/tests/dispatch-complete-milestone-guard.test.ts +5 -0
  399. package/src/resources/extensions/gsd/tests/dispatch-guard.test.ts +25 -0
  400. package/src/resources/extensions/gsd/tests/dispatch-missing-task-plans.test.ts +14 -0
  401. package/src/resources/extensions/gsd/tests/dispatcher-stuck-planning.test.ts +3 -2
  402. package/src/resources/extensions/gsd/tests/double-merge-guard.test.ts +4 -3
  403. package/src/resources/extensions/gsd/tests/empty-content-abort-loop.test.ts +4 -3
  404. package/src/resources/extensions/gsd/tests/execution-entry-missing-context-4671.test.ts +173 -0
  405. package/src/resources/extensions/gsd/tests/extension-bootstrap-isolation.test.ts +139 -129
  406. package/src/resources/extensions/gsd/tests/finalize-timeout-guard.test.ts +8 -104
  407. package/src/resources/extensions/gsd/tests/gate-state-canonicalization.test.ts +102 -0
  408. package/src/resources/extensions/gsd/tests/gate-storage.test.ts +1 -1
  409. package/src/resources/extensions/gsd/tests/google-search-stub.test.ts +14 -4
  410. package/src/resources/extensions/gsd/tests/headless-milestone-parity.test.ts +117 -0
  411. package/src/resources/extensions/gsd/tests/hook-key-parsing.test.ts +4 -55
  412. package/src/resources/extensions/gsd/tests/integration/all-milestones-complete-merge.test.ts +7 -56
  413. package/src/resources/extensions/gsd/tests/integration/auto-recovery.test.ts +20 -0
  414. package/src/resources/extensions/gsd/tests/integration/doctor-proactive.test.ts +18 -2
  415. package/src/resources/extensions/gsd/tests/integration/queue-completed-milestone-perf.test.ts +10 -4
  416. package/src/resources/extensions/gsd/tests/integration/state-machine-edge-cases.test.ts +144 -7
  417. package/src/resources/extensions/gsd/tests/integration/state-machine-live-validation.test.ts +4 -0
  418. package/src/resources/extensions/gsd/tests/integration/state-machine-runtime-failures.test.ts +2 -16
  419. package/src/resources/extensions/gsd/tests/interactive-routing-bypass.test.ts +9 -3
  420. package/src/resources/extensions/gsd/tests/interrupted-session-ui.test.ts +6 -9
  421. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +64 -0
  422. package/src/resources/extensions/gsd/tests/knowledge.test.ts +93 -1
  423. package/src/resources/extensions/gsd/tests/mcp-client-security.test.ts +8 -37
  424. package/src/resources/extensions/gsd/tests/memory-extractor.test.ts +5 -15
  425. package/src/resources/extensions/gsd/tests/merge-conflict-stops-loop.test.ts +227 -55
  426. package/src/resources/extensions/gsd/tests/milestone-scope-classifier.test.ts +187 -0
  427. package/src/resources/extensions/gsd/tests/milestone-summary-classifier.test.ts +30 -0
  428. package/src/resources/extensions/gsd/tests/model-cost-table.test.ts +9 -1
  429. package/src/resources/extensions/gsd/tests/model-router.test.ts +1 -1
  430. package/src/resources/extensions/gsd/tests/native-git-bridge-exec-fallback.test.ts +6 -48
  431. package/src/resources/extensions/gsd/tests/notification-widget.test.ts +6 -3
  432. package/src/resources/extensions/gsd/tests/orphaned-worktree-audit.test.ts +59 -2
  433. package/src/resources/extensions/gsd/tests/parallel-research-dispatch.test.ts +273 -130
  434. package/src/resources/extensions/gsd/tests/pipeline-variant-dispatch.test.ts +301 -0
  435. package/src/resources/extensions/gsd/tests/pre-execution-pause-wiring.test.ts +32 -1
  436. package/src/resources/extensions/gsd/tests/preferences-worktree-sync.test.ts +2 -1
  437. package/src/resources/extensions/gsd/tests/prompt-cache-optimizer.test.ts +12 -0
  438. package/src/resources/extensions/gsd/tests/prompt-step-ordering.test.ts +15 -4
  439. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +23 -24
  440. package/src/resources/extensions/gsd/tests/queue-auto-guard.test.ts +32 -0
  441. package/src/resources/extensions/gsd/tests/queue-draft-detection.test.ts +3 -2
  442. package/src/resources/extensions/gsd/tests/queued-discuss-fast-path.test.ts +4 -5
  443. package/src/resources/extensions/gsd/tests/ready-phrase-no-files-4573.test.ts +75 -2
  444. package/src/resources/extensions/gsd/tests/reassess-default-optin.test.ts +132 -0
  445. package/src/resources/extensions/gsd/tests/recovery-attempts-reset.test.ts +8 -40
  446. package/src/resources/extensions/gsd/tests/regex-hardening.test.ts +136 -256
  447. package/src/resources/extensions/gsd/tests/research-milestone-composer.test.ts +114 -0
  448. package/src/resources/extensions/gsd/tests/restore-tools-after-discuss.test.ts +6 -3
  449. package/src/resources/extensions/gsd/tests/run-uat-composer.test.ts +148 -0
  450. package/src/resources/extensions/gsd/tests/service-tier.test.ts +4 -0
  451. package/src/resources/extensions/gsd/tests/session-lock-regression.test.ts +29 -0
  452. package/src/resources/extensions/gsd/tests/sidecar-queue.test.ts +3 -2
  453. package/src/resources/extensions/gsd/tests/silent-catch-diagnostics.test.ts +55 -95
  454. package/src/resources/extensions/gsd/tests/single-writer-v3-tool-surface.test.ts +158 -0
  455. package/src/resources/extensions/gsd/tests/skill-activation.test.ts +120 -1
  456. package/src/resources/extensions/gsd/tests/skill-manifest.test.ts +112 -0
  457. package/src/resources/extensions/gsd/tests/slice-cadence.test.ts +242 -0
  458. package/src/resources/extensions/gsd/tests/slice-context-injection.test.ts +3 -2
  459. package/src/resources/extensions/gsd/tests/slice-parallel-orchestrator.test.ts +164 -1
  460. package/src/resources/extensions/gsd/tests/smart-entry-draft.test.ts +2 -1
  461. package/src/resources/extensions/gsd/tests/stale-dirlistcache-4648.test.ts +112 -0
  462. package/src/resources/extensions/gsd/tests/state-machine-full-walkthrough.test.ts +29 -5
  463. package/src/resources/extensions/gsd/tests/state-transition-matrix.test.ts +44 -0
  464. package/src/resources/extensions/gsd/tests/stop-auto-race-null-unit.test.ts +3 -3
  465. package/src/resources/extensions/gsd/tests/structured-data-formatter.test.ts +11 -92
  466. package/src/resources/extensions/gsd/tests/subagent-model-dispatch.test.ts +7 -6
  467. package/src/resources/extensions/gsd/tests/survivor-branch-complete.test.ts +102 -101
  468. package/src/resources/extensions/gsd/tests/sync-lock.test.ts +31 -0
  469. package/src/resources/extensions/gsd/tests/sync-worktree-skip-current.test.ts +4 -3
  470. package/src/resources/extensions/gsd/tests/test-helpers.test.ts +98 -0
  471. package/src/resources/extensions/gsd/tests/test-helpers.ts +153 -0
  472. package/src/resources/extensions/gsd/tests/token-profile.test.ts +8 -1
  473. package/src/resources/extensions/gsd/tests/tool-invocation-error-loop-break.test.ts +61 -1
  474. package/src/resources/extensions/gsd/tests/tool-naming.test.ts +8 -1
  475. package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +355 -0
  476. package/src/resources/extensions/gsd/tests/unit-context-manifest.test.ts +258 -0
  477. package/src/resources/extensions/gsd/tests/uok-contracts.test.ts +51 -0
  478. package/src/resources/extensions/gsd/tests/uok-execution-graph.test.ts +16 -0
  479. package/src/resources/extensions/gsd/tests/uok-gate-runner.test.ts +75 -0
  480. package/src/resources/extensions/gsd/tests/uok-gitops-wiring.test.ts +49 -26
  481. package/src/resources/extensions/gsd/tests/uok-loop-adapter-writer.test.ts +65 -0
  482. package/src/resources/extensions/gsd/tests/uok-parity-report.test.ts +42 -0
  483. package/src/resources/extensions/gsd/tests/uok-plan-v2-wiring.test.ts +19 -2
  484. package/src/resources/extensions/gsd/tests/uok-writer.test.ts +75 -0
  485. package/src/resources/extensions/gsd/tests/validate-milestone.test.ts +12 -0
  486. package/src/resources/extensions/gsd/tests/verify-artifact-tightened.test.ts +144 -80
  487. package/src/resources/extensions/gsd/tests/visualizer-critical-path.test.ts +20 -54
  488. package/src/resources/extensions/gsd/tests/visualizer-overlay.test.ts +342 -277
  489. package/src/resources/extensions/gsd/tests/worker-model-override.test.ts +37 -29
  490. package/src/resources/extensions/gsd/tests/worktree-db.test.ts +226 -266
  491. package/src/resources/extensions/gsd/tests/worktree-health-monorepo.test.ts +103 -67
  492. package/src/resources/extensions/gsd/tests/worktree-nested-git-safety.test.ts +92 -90
  493. package/src/resources/extensions/gsd/tests/worktree-submodule-safety.test.ts +238 -59
  494. package/src/resources/extensions/gsd/tests/worktree-sync-overwrite-loop.test.ts +113 -161
  495. package/src/resources/extensions/gsd/tests/worktree-telemetry.test.ts +210 -0
  496. package/src/resources/extensions/gsd/tests/write-gate-planning-unit.test.ts +262 -0
  497. package/src/resources/extensions/gsd/tests/write-gate-predicates.test.ts +186 -0
  498. package/src/resources/extensions/gsd/tests/write-gate.test.ts +7 -5
  499. package/src/resources/extensions/gsd/tests/zombie-gsd-state.test.ts +80 -96
  500. package/src/resources/extensions/gsd/tools/validate-milestone.ts +8 -2
  501. package/src/resources/extensions/gsd/types.ts +3 -3
  502. package/src/resources/extensions/gsd/unit-context-composer.ts +218 -0
  503. package/src/resources/extensions/gsd/unit-context-manifest.ts +574 -0
  504. package/src/resources/extensions/gsd/uok/contracts.ts +65 -0
  505. package/src/resources/extensions/gsd/uok/dispatch-envelope.ts +56 -0
  506. package/src/resources/extensions/gsd/uok/execution-graph.ts +22 -0
  507. package/src/resources/extensions/gsd/uok/gate-runner.ts +65 -5
  508. package/src/resources/extensions/gsd/uok/gitops.ts +6 -1
  509. package/src/resources/extensions/gsd/uok/loop-adapter.ts +45 -10
  510. package/src/resources/extensions/gsd/uok/parity-report.ts +84 -0
  511. package/src/resources/extensions/gsd/uok/plan-v2.ts +13 -5
  512. package/src/resources/extensions/gsd/uok/writer.ts +113 -0
  513. package/src/resources/extensions/gsd/workflow-mcp.ts +6 -0
  514. package/src/resources/extensions/gsd/worktree-manager.ts +108 -7
  515. package/src/resources/extensions/gsd/worktree-resolver.ts +96 -9
  516. package/src/resources/extensions/gsd/worktree-telemetry.ts +322 -0
  517. package/src/resources/extensions/mcp-client/index.ts +3 -1
  518. package/src/resources/extensions/mcp-client/tests/server-name-spaces.test.ts +70 -36
  519. package/src/resources/extensions/ollama/index.ts +5 -1
  520. package/src/resources/extensions/ollama/ollama-auth-mode.test.ts +123 -15
  521. package/src/resources/extensions/ollama/ollama-status-indicator.test.ts +206 -19
  522. package/src/resources/extensions/remote-questions/manager.ts +36 -4
  523. package/src/resources/extensions/remote-questions/tests/command-polling.test.ts +200 -190
  524. package/src/resources/extensions/shared/tests/interview-preview.test.ts +11 -3
  525. package/src/resources/extensions/voice/tests/linux-ready.test.ts +129 -113
  526. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.d.ts +0 -2
  527. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.d.ts.map +0 -1
  528. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.js +0 -289
  529. package/packages/pi-ai/dist/utils/oauth/oauth-providers.test.js.map +0 -1
  530. package/packages/pi-ai/src/utils/oauth/oauth-providers.test.ts +0 -363
  531. package/src/resources/extensions/gsd/tests/auto-start-model-capture.test.ts +0 -143
  532. package/src/resources/extensions/gsd/tests/complete-milestone-false-merge.test.ts +0 -157
  533. package/src/resources/extensions/gsd/tests/dashboard-model-label-ordering.test.ts +0 -107
  534. package/src/resources/extensions/gsd/tests/find-missing-summaries-closed.test.ts +0 -48
  535. package/src/resources/extensions/gsd/tests/forensics-context-persist.test.ts +0 -159
  536. package/src/resources/extensions/gsd/tests/forensics-db-completion.test.ts +0 -96
  537. package/src/resources/extensions/gsd/tests/forensics-dedup.test.ts +0 -79
  538. package/src/resources/extensions/gsd/tests/forensics-hook-key-parse.test.ts +0 -74
  539. package/src/resources/extensions/gsd/tests/forensics-journal.test.ts +0 -162
  540. package/src/resources/extensions/gsd/tests/gitignore-bg-shell.test.ts +0 -38
  541. package/src/resources/extensions/gsd/tests/gsd-no-project-error.test.ts +0 -73
  542. package/src/resources/extensions/gsd/tests/idle-watchdog-stall-override.test.ts +0 -125
  543. package/src/resources/extensions/gsd/tests/import-done-milestones.test.ts +0 -42
  544. /package/dist/web/standalone/.next/static/{5wbu35_C2_MQ3Jj1lEVDx → cAJH99yNS1UPbeSEiNRrV}/_buildManifest.js +0 -0
  545. /package/dist/web/standalone/.next/static/{5wbu35_C2_MQ3Jj1lEVDx → cAJH99yNS1UPbeSEiNRrV}/_ssgManifest.js +0 -0
@@ -21,6 +21,10 @@ import type {
21
21
  import type { ExtensionUIContext } from "@gsd/pi-coding-agent";
22
22
  import { EventStream } from "@gsd/pi-ai";
23
23
  import { execSync } from "node:child_process";
24
+ import { existsSync, readFileSync } from "node:fs";
25
+ import { homedir } from "node:os";
26
+ import { createRequire } from "node:module";
27
+ import { dirname, join } from "node:path";
24
28
  import { PartialMessageBuilder, ZERO_USAGE, mapUsage } from "./partial-builder.js";
25
29
  import { buildWorkflowMcpServers } from "../gsd/workflow-mcp.js";
26
30
  import { showInterviewRound, type Question, type RoundResult } from "../shared/tui.js";
@@ -178,34 +182,88 @@ export function getResultErrorMessage(result: SDKResultMessage): string {
178
182
  // Claude binary resolution
179
183
  // ---------------------------------------------------------------------------
180
184
 
181
- /** Cached result of the `which`/`where claude` lookup so the shell is only spawned once per process. */
185
+ /** Cached result of the Claude executable/script resolution so lookup runs once per process. */
182
186
  let cachedClaudePath: string | null = null;
187
+ const requireFromHere = createRequire(import.meta.url);
183
188
 
184
189
  /** Return the shell command used to locate the `claude` binary on the given platform. */
185
190
  export function getClaudeLookupCommand(platform: NodeJS.Platform = process.platform): string {
186
191
  return platform === "win32" ? "where claude" : "which claude";
187
192
  }
188
193
 
189
- /** Extract the first line of `which`/`where` output as the resolved binary path. */
190
- export function parseClaudeLookupOutput(output: Buffer | string): string {
191
- return output
194
+ /**
195
+ * Pick the most suitable path from `which`/`where` output.
196
+ *
197
+ * On Windows, `where claude` can return shim entries first (for example
198
+ * `...\\npm\\claude` / `...\\npm\\claude.cmd`) that the Claude Agent SDK treats
199
+ * as a native executable path and then fails to spawn. Prefer a native
200
+ * `.exe` candidate when present.
201
+ */
202
+ export function parseClaudeLookupOutput(output: Buffer | string, platform: NodeJS.Platform = process.platform): string {
203
+ const lines = output
192
204
  .toString()
193
- .trim()
194
- .split(/\r?\n/)[0] ?? "";
205
+ .split(/\r?\n/)
206
+ .map((line) => line.trim())
207
+ .filter(Boolean);
208
+
209
+ if (lines.length === 0) return "";
210
+ if (platform !== "win32") return lines[0] ?? "";
211
+
212
+ const exeCandidate = lines.find((line) => /\.exe$/i.test(line));
213
+ if (exeCandidate) return exeCandidate;
214
+
215
+ const cmdCandidate = lines.find((line) => /\.cmd$/i.test(line));
216
+ if (cmdCandidate) return cmdCandidate;
217
+
218
+ return lines[0] ?? "";
219
+ }
220
+
221
+ /** Resolve the SDK-bundled cli.js path if available. */
222
+ export function resolveBundledClaudeCliPath(): string | null {
223
+ try {
224
+ const sdkEntry = requireFromHere.resolve("@anthropic-ai/claude-agent-sdk");
225
+ const cliPath = join(dirname(sdkEntry), "cli.js");
226
+ return existsSync(cliPath) ? cliPath : null;
227
+ } catch {
228
+ return null;
229
+ }
195
230
  }
196
231
 
197
232
  /**
198
- * Resolve the path to the system-installed `claude` binary.
199
- * The SDK defaults to a bundled cli.js which doesn't exist when
200
- * installed as a library we need to point it at the real CLI.
233
+ * Normalize a discovered path for Claude Agent SDK consumption.
234
+ *
235
+ * On Windows, the SDK treats non-`.js` paths as native binaries. NPM shims
236
+ * like `claude`/`claude.cmd` are not native binaries and can fail with
237
+ * `ENOENT`/`EINVAL` in that mode. When no `.exe` is available, prefer the
238
+ * SDK-bundled `cli.js` so the SDK runs via Node.
201
239
  */
240
+ export function normalizeClaudePathForSdk(
241
+ resolvedPath: string,
242
+ platform: NodeJS.Platform = process.platform,
243
+ bundledCliPath: string | null = resolveBundledClaudeCliPath(),
244
+ ): string {
245
+ if (platform !== "win32") return resolvedPath;
246
+ if (/\.exe$/i.test(resolvedPath)) return resolvedPath;
247
+ if (bundledCliPath) return bundledCliPath;
248
+ return resolvedPath;
249
+ }
250
+
251
+ /** Resolve the path passed to `pathToClaudeCodeExecutable`. */
202
252
  function getClaudePath(): string {
203
253
  if (cachedClaudePath) return cachedClaudePath;
254
+
255
+ const fallback = process.platform === "win32"
256
+ ? (resolveBundledClaudeCliPath() ?? "claude.cmd")
257
+ : "claude";
258
+
204
259
  try {
205
- cachedClaudePath = parseClaudeLookupOutput(execSync(getClaudeLookupCommand(), { timeout: 5_000, stdio: "pipe" }));
260
+ const lookupOutput = execSync(getClaudeLookupCommand(), { timeout: 5_000, stdio: "pipe" });
261
+ const parsed = parseClaudeLookupOutput(lookupOutput, process.platform);
262
+ cachedClaudePath = normalizeClaudePathForSdk(parsed || fallback, process.platform);
206
263
  } catch {
207
- cachedClaudePath = "claude"; // fall back to PATH resolution
264
+ cachedClaudePath = fallback;
208
265
  }
266
+
209
267
  return cachedClaudePath;
210
268
  }
211
269
 
@@ -632,6 +690,439 @@ async function promptTextInputElicitation(
632
690
  return { action: "accept", content };
633
691
  }
634
692
 
693
+ // ---------------------------------------------------------------------------
694
+ // canUseTool handler
695
+ // ---------------------------------------------------------------------------
696
+
697
+ /** Options passed by the SDK to the canUseTool callback. */
698
+ interface CanUseToolOptions {
699
+ signal: AbortSignal;
700
+ suggestions?: Array<Record<string, unknown>>;
701
+ blockedPath?: string;
702
+ decisionReason?: string;
703
+ title?: string;
704
+ displayName?: string;
705
+ description?: string;
706
+ toolUseID: string;
707
+ agentID?: string;
708
+ }
709
+
710
+ /** Result returned by the canUseTool callback to the SDK. */
711
+ type CanUseToolPermissionResult =
712
+ | { behavior: "allow"; updatedInput?: Record<string, unknown>; updatedPermissions?: Array<Record<string, unknown>>; toolUseID?: string }
713
+ | { behavior: "deny"; message: string; interrupt?: boolean; toolUseID?: string };
714
+
715
+ /**
716
+ * Known CLI tools where the subcommand verb changes the risk profile.
717
+ * Value = number of subcommand tokens (beyond the executable) to capture
718
+ * in the "Always Allow" permission pattern.
719
+ *
720
+ * `git push` and `git log` are very different → depth 1 → `Bash(git push:*)`
721
+ * `gh pr create` and `gh pr list` differ at depth 2 → `Bash(gh pr create:*)`
722
+ * `ping` is always safe → not listed → `Bash(ping:*)`
723
+ */
724
+ const SUBCOMMAND_DEPTH: Record<string, number> = {
725
+ git: 1,
726
+ gh: 2,
727
+ npm: 1,
728
+ npx: 1,
729
+ yarn: 1,
730
+ pnpm: 1,
731
+ docker: 1,
732
+ kubectl: 1,
733
+ aws: 2,
734
+ az: 2,
735
+ gcloud: 2,
736
+ cargo: 1,
737
+ pip: 1,
738
+ pip3: 1,
739
+ brew: 1,
740
+ terraform: 1,
741
+ helm: 1,
742
+ dotnet: 1,
743
+ };
744
+
745
+ /** Command wrappers to skip when extracting the base executable. */
746
+ const CMD_PASSTHROUGH = new Set(["sudo", "env", "command"]);
747
+
748
+ /**
749
+ * Build a smart permission pattern for Bash "Always Allow".
750
+ *
751
+ * Simple commands → `Bash(ping:*)` (any args are fine)
752
+ * Subcommand-sensitive CLIs → `Bash(git push:*)` (verb is captured, args wildcarded)
753
+ */
754
+ export function buildBashPermissionPattern(command: string): string {
755
+ // When the command is a chain like "cd /foo && gh pr list", extract the
756
+ // last segment — `cd` is just setup, the meaningful operation is what follows.
757
+ const segments = command.split(/\s*(?:&&|\|\||;)\s*/);
758
+ // Skip leading `cd` (directory setup) and trailing error suppressors
759
+ // like `|| true`, `|| :`, `|| echo ...`. The meaningful command is
760
+ // the first segment that is *neither* of those.
761
+ const SETUP_RE = /^\s*cd\s/;
762
+ const SUPPRESSOR_RE = /^\s*(?:true|:|echo\b)/;
763
+ let meaningful: string | undefined;
764
+ if (segments.length > 1) {
765
+ // Strip suppressors, then strip cd prefixes; take the *last* remaining
766
+ // segment — that's the meaningful command.
767
+ const trimmed = segments.filter(s => !SUPPRESSOR_RE.test(s));
768
+ const core = trimmed.filter(s => !SETUP_RE.test(s));
769
+ meaningful = core.length > 0 ? core[core.length - 1] : trimmed[trimmed.length - 1];
770
+ }
771
+ meaningful = meaningful || segments[0] || command;
772
+ const rawTokens = meaningful.trim().split(/\s+/);
773
+
774
+ // Skip sudo/env wrappers and leading VAR=val assignments
775
+ let idx = 0;
776
+ while (idx < rawTokens.length) {
777
+ if (CMD_PASSTHROUGH.has(rawTokens[idx])) { idx++; continue; }
778
+ if (/^[A-Za-z_]\w*=/.test(rawTokens[idx])) { idx++; continue; }
779
+ break;
780
+ }
781
+ const tokens = rawTokens.slice(idx).filter(Boolean);
782
+ if (tokens.length === 0) return "Bash(*)";
783
+
784
+ // Strip path and .exe from executable name
785
+ const base = tokens[0].replace(/^.*[\\/]/, "").replace(/\.exe$/i, "");
786
+ const depth = SUBCOMMAND_DEPTH[base];
787
+
788
+ if (depth !== undefined) {
789
+ // Capture base + N subcommand tokens: "gh pr list" → Bash(gh pr list:*)
790
+ const significant = [base, ...tokens.slice(1, 1 + depth)].join(" ");
791
+ return `Bash(${significant}:*)`;
792
+ }
793
+
794
+ // Simple command — any args are fine: "ping" → Bash(ping:*)
795
+ return `Bash(${base}:*)`;
796
+ }
797
+
798
+ /**
799
+ * Build the list of granularity options presented after a user chooses
800
+ * "Always Allow" for a Bash command.
801
+ *
802
+ * Rather than assuming the user wants the default smart pattern, the UI
803
+ * shows every meaningful prefix so the user explicitly picks the scope:
804
+ *
805
+ * "gh pr list --limit 5" → [
806
+ * "Bash(gh:*)", // allow any gh command
807
+ * "Bash(gh pr:*)", // allow any gh pr subcommand
808
+ * "Bash(gh pr list:*)", // allow just this verb
809
+ * ]
810
+ *
811
+ * Flags (tokens starting with `-`) terminate the subcommand chain — they
812
+ * are call-site arguments, not stable verbs. Subcommand depth is capped
813
+ * at 3 to keep the menu short (max 4 options).
814
+ *
815
+ * Returns a single-entry list when there is no meaningful subcommand to
816
+ * choose from (e.g. `ls -la`). Callers can skip the second dialog in
817
+ * that case.
818
+ */
819
+ export function buildBashPermissionPatternOptions(command: string): string[] {
820
+ const segments = command.split(/\s*(?:&&|\|\||;)\s*/);
821
+ const SETUP_RE = /^\s*cd\s/;
822
+ const SUPPRESSOR_RE = /^\s*(?:true|:|echo\b)/;
823
+ let meaningful: string | undefined;
824
+ if (segments.length > 1) {
825
+ const trimmed = segments.filter(s => !SUPPRESSOR_RE.test(s));
826
+ const core = trimmed.filter(s => !SETUP_RE.test(s));
827
+ meaningful = core.length > 0 ? core[core.length - 1] : trimmed[trimmed.length - 1];
828
+ }
829
+ meaningful = meaningful || segments[0] || command;
830
+ const rawTokens = meaningful.trim().split(/\s+/);
831
+
832
+ let idx = 0;
833
+ while (idx < rawTokens.length) {
834
+ if (CMD_PASSTHROUGH.has(rawTokens[idx])) { idx++; continue; }
835
+ if (/^[A-Za-z_]\w*=/.test(rawTokens[idx])) { idx++; continue; }
836
+ break;
837
+ }
838
+ const tokens = rawTokens.slice(idx).filter(Boolean);
839
+ if (tokens.length === 0) return ["Bash(*)"];
840
+
841
+ const base = tokens[0].replace(/^.*[\\/]/, "").replace(/\.exe$/i, "");
842
+
843
+ // Collect up to 3 subcommand tokens, stopping at the first flag.
844
+ const subTokens: string[] = [];
845
+ for (let i = 1; i < tokens.length; i++) {
846
+ const t = tokens[i];
847
+ if (t.startsWith("-")) break;
848
+ subTokens.push(t);
849
+ if (subTokens.length >= 3) break;
850
+ }
851
+
852
+ const patterns: string[] = [`Bash(${base}:*)`];
853
+ for (let i = 1; i <= subTokens.length; i++) {
854
+ patterns.push(`Bash(${[base, ...subTokens.slice(0, i)].join(" ")}:*)`);
855
+ }
856
+ return patterns;
857
+ }
858
+
859
+ /**
860
+ * Read Bash allow-rule patterns from project and user settings files.
861
+ *
862
+ * Returns the ruleContent portion (e.g. `"gh pr list:*"`) for each
863
+ * `Bash(...)` entry found in `permissions.allow`.
864
+ */
865
+ function readBashAllowRulesFromSettings(): string[] {
866
+ const rules: string[] = [];
867
+ const paths = [
868
+ join(process.cwd(), ".claude", "settings.local.json"),
869
+ join(process.cwd(), ".claude", "settings.json"),
870
+ ];
871
+ try {
872
+ paths.push(join(homedir(), ".claude", "settings.json"));
873
+ } catch {
874
+ // homedir() can throw on some platforms
875
+ }
876
+ for (const settingsPath of paths) {
877
+ try {
878
+ if (!existsSync(settingsPath)) continue;
879
+ const raw = JSON.parse(readFileSync(settingsPath, "utf8"));
880
+ const allow = raw?.permissions?.allow;
881
+ if (!Array.isArray(allow)) continue;
882
+ for (const entry of allow) {
883
+ if (typeof entry !== "string") continue;
884
+ const m = /^Bash\((.+)\)$/.exec(entry);
885
+ if (m) rules.push(m[1]);
886
+ }
887
+ } catch {
888
+ // Ignore malformed settings files
889
+ }
890
+ }
891
+ return rules;
892
+ }
893
+
894
+ /**
895
+ * Check if a Bash compound command matches saved allow rules after
896
+ * extracting the meaningful segment.
897
+ *
898
+ * The SDK's built-in matcher refuses to match prefix rules against
899
+ * compound commands (e.g. `cd /path && gh pr list`). Claude Code
900
+ * routinely prepends `cd <cwd> &&` to commands, causing saved rules
901
+ * to never match on re-invocation. This function strips safe leading
902
+ * segments (only `cd` commands) and checks the remaining operation
903
+ * against saved rules.
904
+ *
905
+ * For compound commands, returns true only when all leading segments
906
+ * are `cd` commands and the final segment matches a saved rule.
907
+ * For simple (single-segment) commands, checks directly against saved
908
+ * rules — this covers the case where a rule was added mid-session and
909
+ * the SDK's in-memory cache is stale.
910
+ */
911
+ export function bashCommandMatchesSavedRules(command: string): boolean {
912
+ const segments = command.split(/\s*(?:&&|\|\||;)\s*/).filter(Boolean);
913
+ if (segments.length === 0) return false;
914
+
915
+ let meaningful: string;
916
+ if (segments.length === 1) {
917
+ meaningful = segments[0].trim();
918
+ } else {
919
+ // Strip trailing error suppressors (|| true, || :, || echo ...)
920
+ // and leading cd segments. The first remaining segment is the
921
+ // meaningful command. All other non-cd, non-suppressor segments
922
+ // must be absent — otherwise we can't safely auto-approve.
923
+ const SETUP_RE = /^cd\s/;
924
+ const SUPPRESSOR_RE = /^\s*(?:true|:|echo\b)/;
925
+ const trimmed = segments.filter(s => !SUPPRESSOR_RE.test(s.trim()));
926
+ const core = trimmed.filter(s => !SETUP_RE.test(s.trim()));
927
+ if (core.length !== 1) return false; // ambiguous — multiple real commands
928
+ meaningful = core[0].trim();
929
+ }
930
+ if (!meaningful) return false;
931
+
932
+ const rules = readBashAllowRulesFromSettings();
933
+ if (rules.length === 0) return false;
934
+
935
+ for (const rule of rules) {
936
+ const prefixMatch = /^(.+):\*$/.exec(rule);
937
+ if (prefixMatch) {
938
+ const prefix = prefixMatch[1];
939
+ if (meaningful === prefix || meaningful.startsWith(prefix + " ")) {
940
+ return true;
941
+ }
942
+ continue;
943
+ }
944
+ // Exact match
945
+ if (meaningful === rule) return true;
946
+ }
947
+
948
+ return false;
949
+ }
950
+
951
+ /** Format the tool input into a human-readable summary for the permission prompt. */
952
+ function formatToolInput(toolName: string, input: Record<string, unknown>): string {
953
+ // Bash — show the command
954
+ if (input.command && typeof input.command === "string") {
955
+ const cmd = input.command.length > 300 ? input.command.slice(0, 300) + "…" : input.command;
956
+ return cmd;
957
+ }
958
+ // File-oriented tools — show path
959
+ if (input.file_path && typeof input.file_path === "string") {
960
+ return `${toolName}: ${input.file_path}`;
961
+ }
962
+ // Generic fallback — compact JSON, truncated
963
+ const json = JSON.stringify(input);
964
+ if (json.length <= 200) return json;
965
+ return json.slice(0, 200) + "…";
966
+ }
967
+
968
+ /**
969
+ * Create a canUseTool handler that routes SDK permission requests through the
970
+ * extension UI's select dialog, or auto-approves when no UI is available.
971
+ *
972
+ * Presents three options:
973
+ * - **Allow** — approve this one invocation
974
+ * - **Always Allow** — approve and pass `suggestions` back as `updatedPermissions`
975
+ * so the SDK remembers the choice for the rest of the session
976
+ * - **Deny** — reject the invocation
977
+ *
978
+ * Follows the same pattern as {@link createClaudeCodeElicitationHandler}:
979
+ * takes an optional UI context and returns the callback or undefined.
980
+ *
981
+ * When UI is unavailable (headless / auto-mode sub-agents), returns a handler
982
+ * that always approves — replacing the old GSD_AUTO_MODE → bypassPermissions
983
+ * workaround.
984
+ */
985
+ export function createClaudeCodeCanUseToolHandler(
986
+ ui: ExtensionUIContext | undefined,
987
+ ): ((toolName: string, input: Record<string, unknown>, options: CanUseToolOptions) => Promise<CanUseToolPermissionResult>) | undefined {
988
+ if (!ui) return undefined;
989
+
990
+ return async (toolName, _input, options) => {
991
+ // Abort early if the signal is already fired
992
+ if (options.signal.aborted) {
993
+ return { behavior: "deny", message: "Aborted", toolUseID: options.toolUseID };
994
+ }
995
+
996
+ // For Bash compound commands (e.g. "cd /path && gh pr list"),
997
+ // check if the meaningful operation matches a saved allow rule.
998
+ // The SDK's built-in matcher rejects prefix rules for compound
999
+ // commands, but cd-prefixed commands are routine and the actual
1000
+ // operation is already approved.
1001
+ if (toolName === "Bash" && typeof _input.command === "string") {
1002
+ if (bashCommandMatchesSavedRules(_input.command)) {
1003
+ return { behavior: "allow", updatedInput: _input, toolUseID: options.toolUseID };
1004
+ }
1005
+ }
1006
+
1007
+ const inputSummary = formatToolInput(toolName, _input);
1008
+ const title = options.title || `Allow Claude Code to use: ${toolName}?`;
1009
+ const body = [
1010
+ options.description,
1011
+ inputSummary,
1012
+ ].filter(Boolean).join("\n");
1013
+
1014
+ // The 2nd menu (level picker) lets the user choose the exact pattern,
1015
+ // so the 1st menu just shows "Always Allow" without a command suffix.
1016
+ const alwaysAllowLabel = "Always Allow";
1017
+
1018
+ try {
1019
+ const choice = await ui.select(
1020
+ `${title}\n${body}`,
1021
+ ["Allow", alwaysAllowLabel, "Deny"],
1022
+ { signal: options.signal },
1023
+ );
1024
+
1025
+ if (options.signal.aborted) {
1026
+ return { behavior: "deny", message: "Aborted", toolUseID: options.toolUseID };
1027
+ }
1028
+
1029
+ if (choice === alwaysAllowLabel) {
1030
+ // Pass the SDK's own suggestions back as updatedPermissions so
1031
+ // it knows how to persist them (PermissionUpdate[] shape).
1032
+ // For Bash, patch the ruleContent with the user-chosen
1033
+ // granularity pattern (e.g. "gh", "gh pr", "gh pr list") so
1034
+ // the saved rule matches the scope the user actually wants.
1035
+ let perms = options.suggestions;
1036
+ let notifyLabel: string | undefined;
1037
+ if (toolName === "Bash" && typeof _input.command === "string") {
1038
+ // Present every meaningful prefix so the user picks the
1039
+ // scope explicitly rather than getting a blanket match.
1040
+ const patternOptions = buildBashPermissionPatternOptions(_input.command);
1041
+ let chosenPattern: string;
1042
+ if (patternOptions.length <= 1) {
1043
+ // No subcommand choice to make (e.g. "ls -la") — use
1044
+ // the single available pattern directly.
1045
+ chosenPattern = patternOptions[0] ?? buildBashPermissionPattern(_input.command);
1046
+ } else {
1047
+ const levelChoiceRaw = await ui.select(
1048
+ "Save permission at which level?",
1049
+ patternOptions,
1050
+ { signal: options.signal },
1051
+ );
1052
+ if (options.signal.aborted) {
1053
+ return { behavior: "deny", message: "Aborted", toolUseID: options.toolUseID };
1054
+ }
1055
+ const levelChoice = Array.isArray(levelChoiceRaw) ? levelChoiceRaw[0] : levelChoiceRaw;
1056
+ if (!levelChoice || !patternOptions.includes(levelChoice)) {
1057
+ // User dismissed the level picker — cancel the
1058
+ // tool use. Falling back to a one-time allow
1059
+ // here would leave the spawned agent running
1060
+ // with no clear signal that the user bailed.
1061
+ return {
1062
+ behavior: "deny",
1063
+ message: "User cancelled permission selection",
1064
+ toolUseID: options.toolUseID,
1065
+ };
1066
+ }
1067
+ chosenPattern = levelChoice;
1068
+ }
1069
+ notifyLabel = chosenPattern;
1070
+ // Extract the ruleContent portion from "Bash(gh pr list:*)" → "gh pr list:*"
1071
+ const ruleContent = chosenPattern.replace(/^Bash\(/, "").replace(/\)$/, "");
1072
+ if (perms && Array.isArray(perms) && perms.length > 0) {
1073
+ // Clone suggestions and patch ruleContent on any Bash addRules entry
1074
+ perms = perms.map((s: any) => {
1075
+ if (s.type === "addRules" && Array.isArray(s.rules)) {
1076
+ return {
1077
+ ...s,
1078
+ rules: s.rules.map((r: any) =>
1079
+ r.toolName === "Bash" ? { ...r, ruleContent } : r,
1080
+ ),
1081
+ };
1082
+ }
1083
+ return s;
1084
+ });
1085
+ } else {
1086
+ // No suggestions from SDK — build a proper PermissionUpdate
1087
+ perms = [{
1088
+ type: "addRules",
1089
+ rules: [{ toolName: "Bash", ruleContent }],
1090
+ behavior: "allow",
1091
+ destination: "localSettings",
1092
+ }];
1093
+ }
1094
+ }
1095
+ // Notify with the resolved pattern (label already previewed it)
1096
+ if (notifyLabel) {
1097
+ ui.notify(`Saved: ${notifyLabel}`, "info");
1098
+ }
1099
+ return {
1100
+ behavior: "allow",
1101
+ updatedInput: _input,
1102
+ toolUseID: options.toolUseID,
1103
+ ...(perms ? { updatedPermissions: perms } : {}),
1104
+ };
1105
+ }
1106
+
1107
+ if (choice === "Allow") {
1108
+ return {
1109
+ behavior: "allow",
1110
+ updatedInput: _input,
1111
+ toolUseID: options.toolUseID,
1112
+ };
1113
+ }
1114
+
1115
+ return { behavior: "deny", message: "User denied", toolUseID: options.toolUseID };
1116
+ } catch {
1117
+ return { behavior: "deny", message: "Aborted", toolUseID: options.toolUseID };
1118
+ }
1119
+ };
1120
+ }
1121
+
1122
+ // ---------------------------------------------------------------------------
1123
+ // Elicitation handler
1124
+ // ---------------------------------------------------------------------------
1125
+
635
1126
  /** Create an SDK elicitation handler that routes requests through the extension UI dialogs, or undefined if no UI is available. */
636
1127
  export function createClaudeCodeElicitationHandler(
637
1128
  ui: ExtensionUIContext | undefined,
@@ -1135,18 +1626,26 @@ async function pumpSdkMessages(
1135
1626
  const prompt = buildPromptFromContext(context);
1136
1627
  const queryPrompt = buildSdkQueryPrompt(context, prompt);
1137
1628
  const permissionMode = await resolveClaudePermissionMode();
1629
+ const uiContext = (options as ClaudeCodeStreamOptions | undefined)?.extensionUIContext;
1630
+ const canUseToolHandler = createClaudeCodeCanUseToolHandler(uiContext);
1631
+ // When no UI is available (headless / auto-mode), auto-approve all
1632
+ // tool requests. This replaces the old bypassPermissions workaround.
1633
+ const canUseToolFallback = canUseToolHandler
1634
+ ?? (async (_toolName: string, _input: Record<string, unknown>, opts: CanUseToolOptions): Promise<CanUseToolPermissionResult> =>
1635
+ ({ behavior: "allow", toolUseID: opts.toolUseID }));
1138
1636
  const sdkOpts = buildSdkOptions(
1139
1637
  modelId,
1140
1638
  prompt,
1141
1639
  { permissionMode },
1142
- typeof (options as ClaudeCodeStreamOptions | undefined)?.extensionUIContext === "object"
1143
- ? {
1144
- reasoning: options?.reasoning,
1145
- onElicitation: createClaudeCodeElicitationHandler(
1146
- (options as ClaudeCodeStreamOptions | undefined)?.extensionUIContext,
1147
- ),
1148
- }
1149
- : { reasoning: options?.reasoning },
1640
+ {
1641
+ reasoning: options?.reasoning,
1642
+ canUseTool: canUseToolFallback,
1643
+ ...(uiContext
1644
+ ? {
1645
+ onElicitation: createClaudeCodeElicitationHandler(uiContext),
1646
+ }
1647
+ : {}),
1648
+ },
1150
1649
  );
1151
1650
 
1152
1651
  const queryResult = sdk.query({