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

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 (378) 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 +7 -29
  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/extensions/async-jobs/async-bash-tool.js +30 -64
  11. package/dist/resources/extensions/async-jobs/await-tool.js +80 -12
  12. package/dist/resources/extensions/async-jobs/index.js +65 -0
  13. package/dist/resources/extensions/async-jobs/job-manager.js +12 -1
  14. package/dist/resources/extensions/bg-shell/bg-shell-command.js +6 -6
  15. package/dist/resources/extensions/bg-shell/bg-shell-tool.js +10 -7
  16. package/dist/resources/extensions/bg-shell/overlay.js +9 -6
  17. package/dist/resources/extensions/bg-shell/process-manager.js +54 -25
  18. package/dist/resources/extensions/bg-shell/readiness-detector.js +11 -0
  19. package/dist/resources/extensions/bg-shell/utilities.js +3 -0
  20. package/dist/resources/extensions/browser-tools/engine/managed-gsd-browser.js +209 -88
  21. package/dist/resources/extensions/browser-tools/engine/selection.js +73 -5
  22. package/dist/resources/extensions/browser-tools/index.js +69 -12
  23. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +30 -4
  24. package/dist/resources/extensions/gsd/auto/orchestrator.js +7 -5
  25. package/dist/resources/extensions/gsd/auto-dispatch.js +12 -1
  26. package/dist/resources/extensions/gsd/auto-model-selection.js +25 -6
  27. package/dist/resources/extensions/gsd/auto-post-unit.js +11 -2
  28. package/dist/resources/extensions/gsd/auto-prompts.js +15 -10
  29. package/dist/resources/extensions/gsd/auto-start.js +15 -10
  30. package/dist/resources/extensions/gsd/auto-tool-tracking.js +18 -0
  31. package/dist/resources/extensions/gsd/auto-unit-tool-scope.js +7 -16
  32. package/dist/resources/extensions/gsd/auto-worktree.js +30 -90
  33. package/dist/resources/extensions/gsd/auto.js +4 -13
  34. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +3 -2
  35. package/dist/resources/extensions/gsd/bootstrap/dynamic-tools.js +23 -6
  36. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +19 -0
  37. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +122 -20
  38. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +6 -2
  39. package/dist/resources/extensions/gsd/branch-patterns.js +2 -0
  40. package/dist/resources/extensions/gsd/browser-daemon-auto-prep.js +83 -0
  41. package/dist/resources/extensions/gsd/browser-evidence.js +8 -2
  42. package/dist/resources/extensions/gsd/captures.js +4 -6
  43. package/dist/resources/extensions/gsd/constants.js +0 -2
  44. package/dist/resources/extensions/gsd/crash-recovery.js +4 -12
  45. package/dist/resources/extensions/gsd/doctor-environment.js +2 -6
  46. package/dist/resources/extensions/gsd/doctor-format.js +9 -6
  47. package/dist/resources/extensions/gsd/doctor-runtime-checks.js +13 -15
  48. package/dist/resources/extensions/gsd/error-classifier.js +9 -0
  49. package/dist/resources/extensions/gsd/exec-sandbox.js +30 -10
  50. package/dist/resources/extensions/gsd/guidance.js +98 -0
  51. package/dist/resources/extensions/gsd/guided-flow.js +17 -2
  52. package/dist/resources/extensions/gsd/mcp-filter.js +2 -19
  53. package/dist/resources/extensions/gsd/mcp-tool-name.js +5 -13
  54. package/dist/resources/extensions/gsd/memory-consolidation-scanner.js +1 -1
  55. package/dist/resources/extensions/gsd/migrate/safety.js +4 -1
  56. package/dist/resources/extensions/gsd/notification-store.js +11 -4
  57. package/dist/resources/extensions/gsd/parallel-monitor-overlay.js +6 -4
  58. package/dist/resources/extensions/gsd/paths.js +27 -0
  59. package/dist/resources/extensions/gsd/pre-execution-checks.js +91 -3
  60. package/dist/resources/extensions/gsd/preferences-models.js +14 -48
  61. package/dist/resources/extensions/gsd/prompts/complete-slice.md +1 -1
  62. package/dist/resources/extensions/gsd/prompts/plan-slice.md +1 -1
  63. package/dist/resources/extensions/gsd/prompts/refine-slice.md +1 -1
  64. package/dist/resources/extensions/gsd/prompts/replan-slice.md +1 -1
  65. package/dist/resources/extensions/gsd/prompts/run-uat.md +1 -1
  66. package/dist/resources/extensions/gsd/prompts/system.md +5 -2
  67. package/dist/resources/extensions/gsd/provider-error-guidance.js +1 -5
  68. package/dist/resources/extensions/gsd/provider-switch-observer.js +1 -1
  69. package/dist/resources/extensions/gsd/publication.js +87 -0
  70. package/dist/resources/extensions/gsd/recovery-classification.js +37 -94
  71. package/dist/resources/extensions/gsd/safety/destructive-confirmation.js +108 -0
  72. package/dist/resources/extensions/gsd/state.js +1 -20
  73. package/dist/resources/extensions/gsd/stop-notice.js +57 -0
  74. package/dist/resources/extensions/gsd/tool-surface-readiness.js +56 -0
  75. package/dist/resources/extensions/gsd/tools/exec-tool.js +9 -7
  76. package/dist/resources/extensions/gsd/tools/plan-slice.js +12 -6
  77. package/dist/resources/extensions/gsd/uat-policy.js +2 -1
  78. package/dist/resources/extensions/gsd/unit-closeout.js +138 -0
  79. package/dist/resources/extensions/gsd/unit-context-composer.js +74 -1
  80. package/dist/resources/extensions/gsd/unit-context-manifest.js +4 -27
  81. package/dist/resources/extensions/gsd/unit-registry.js +337 -0
  82. package/dist/resources/extensions/gsd/unit-tool-contracts.js +9 -182
  83. package/dist/resources/extensions/gsd/web-app-uat.js +45 -8
  84. package/dist/resources/extensions/gsd/workflow-tool-surface.js +1 -1
  85. package/dist/resources/extensions/gsd/worktree-git-recovery.js +15 -9
  86. package/dist/resources/extensions/gsd/worktree-root.js +11 -0
  87. package/dist/resources/extensions/gsd/worktree-session-state.js +4 -5
  88. package/dist/resources/extensions/search-the-web/native-search.js +5 -3
  89. package/dist/resources/extensions/shared/browser-contract.js +59 -0
  90. package/dist/resources/extensions/shared/gsd-browser-cli.js +96 -5
  91. package/dist/resources/shared/package.json +3 -0
  92. package/dist/resources/skills/create-skill/references/executable-code.md +1 -1
  93. package/dist/resources/skills/create-skill/workflows/add-reference.md +8 -3
  94. package/dist/resources/skills/create-skill/workflows/add-script.md +4 -2
  95. package/dist/resources/skills/create-skill/workflows/add-template.md +3 -1
  96. package/dist/resources/skills/create-skill/workflows/add-workflow.md +8 -3
  97. package/dist/resources/skills/create-skill/workflows/upgrade-to-router.md +10 -5
  98. package/dist/resources/skills/create-skill/workflows/verify-skill.md +9 -4
  99. package/dist/resources/skills/spike-wrap-up/SKILL.md +9 -9
  100. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  101. package/dist/web/standalone/.next/BUILD_ID +1 -1
  102. package/dist/web/standalone/.next/app-path-routes-manifest.json +13 -13
  103. package/dist/web/standalone/.next/build-manifest.json +3 -3
  104. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  105. package/dist/web/standalone/.next/react-loadable-manifest.json +1 -1
  106. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  107. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  108. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  109. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  110. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  111. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  112. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  113. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  114. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  115. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  116. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  117. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  118. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  119. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  120. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  121. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  122. package/dist/web/standalone/.next/server/app/api/update/route.js +1 -1
  123. package/dist/web/standalone/.next/server/app/api/update/route.js.nft.json +1 -1
  124. package/dist/web/standalone/.next/server/app/index.html +1 -1
  125. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  126. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  127. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  128. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  129. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  130. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  131. package/dist/web/standalone/.next/server/app-paths-manifest.json +13 -13
  132. package/dist/web/standalone/.next/server/chunks/5124.js +1 -1
  133. package/dist/web/standalone/.next/server/chunks/8357.js +1 -1
  134. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  135. package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
  136. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  137. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  138. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  139. package/dist/web/standalone/.next/static/chunks/{796.cf859a427a2cb2ac.js → 796.e0bdc932325d7e03.js} +1 -1
  140. package/dist/web/standalone/.next/static/chunks/{webpack-fbea77b5f9953368.js → webpack-f0285ce91d4ec9ef.js} +1 -1
  141. package/dist/web/standalone/node_modules/node-pty/build/Makefile +1 -1
  142. package/dist/web/standalone/node_modules/postcss/lib/container.js +18 -26
  143. package/dist/web/standalone/node_modules/postcss/lib/css-syntax-error.js +14 -47
  144. package/dist/web/standalone/node_modules/postcss/lib/declaration.js +4 -4
  145. package/dist/web/standalone/node_modules/postcss/lib/fromJSON.js +3 -3
  146. package/dist/web/standalone/node_modules/postcss/lib/input.js +29 -54
  147. package/dist/web/standalone/node_modules/postcss/lib/lazy-result.js +37 -47
  148. package/dist/web/standalone/node_modules/postcss/lib/map-generator.js +9 -26
  149. package/dist/web/standalone/node_modules/postcss/lib/no-work-result.js +55 -57
  150. package/dist/web/standalone/node_modules/postcss/lib/node.js +31 -99
  151. package/dist/web/standalone/node_modules/postcss/lib/parse.js +1 -1
  152. package/dist/web/standalone/node_modules/postcss/lib/parser.js +9 -10
  153. package/dist/web/standalone/node_modules/postcss/lib/postcss.js +12 -12
  154. package/dist/web/standalone/node_modules/postcss/lib/previous-map.js +11 -30
  155. package/dist/web/standalone/node_modules/postcss/lib/processor.js +7 -7
  156. package/dist/web/standalone/node_modules/postcss/lib/result.js +5 -5
  157. package/dist/web/standalone/node_modules/postcss/lib/rule.js +6 -6
  158. package/dist/web/standalone/node_modules/postcss/lib/stringifier.js +28 -69
  159. package/dist/web/standalone/node_modules/postcss/lib/tokenize.js +2 -6
  160. package/dist/web/standalone/node_modules/postcss/package.json +48 -48
  161. package/dist/web/standalone/package.json +1 -1
  162. package/dist/worktree-cli.js +3 -6
  163. package/dist/worktree-status-banner.js +7 -15
  164. package/package.json +1 -1
  165. package/packages/cloud-mcp-gateway/package.json +2 -2
  166. package/packages/contracts/dist/rpc.d.ts +1 -0
  167. package/packages/contracts/dist/rpc.d.ts.map +1 -1
  168. package/packages/contracts/dist/rpc.js.map +1 -1
  169. package/packages/contracts/dist/workflow.d.ts +4 -0
  170. package/packages/contracts/dist/workflow.d.ts.map +1 -1
  171. package/packages/contracts/dist/workflow.js.map +1 -1
  172. package/packages/contracts/package.json +1 -1
  173. package/packages/daemon/package.json +4 -4
  174. package/packages/gsd-agent-core/package.json +5 -5
  175. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.d.ts +5 -0
  176. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  177. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.js +5 -0
  178. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.js.map +1 -1
  179. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  180. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js +7 -0
  181. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  182. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/input-controller.d.ts.map +1 -1
  183. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/input-controller.js +8 -1
  184. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/input-controller.js.map +1 -1
  185. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-chat-render.d.ts.map +1 -1
  186. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-chat-render.js +11 -1
  187. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-chat-render.js.map +1 -1
  188. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-auth.d.ts.map +1 -1
  189. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-auth.js +4 -4
  190. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-auth.js.map +1 -1
  191. package/packages/gsd-agent-modes/dist/modes/rpc/rpc-mode.d.ts.map +1 -1
  192. package/packages/gsd-agent-modes/dist/modes/rpc/rpc-mode.js +3 -1
  193. package/packages/gsd-agent-modes/dist/modes/rpc/rpc-mode.js.map +1 -1
  194. package/packages/gsd-agent-modes/package.json +7 -7
  195. package/packages/mcp-server/dist/cli.js +6 -3
  196. package/packages/mcp-server/dist/cli.js.map +1 -1
  197. package/packages/mcp-server/dist/workflow-tools.d.ts +8 -0
  198. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  199. package/packages/mcp-server/dist/workflow-tools.js +17 -1
  200. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  201. package/packages/mcp-server/package.json +3 -3
  202. package/packages/native/package.json +1 -1
  203. package/packages/pi-agent-core/dist/harness/env/nodejs.d.ts +1 -0
  204. package/packages/pi-agent-core/dist/harness/env/nodejs.d.ts.map +1 -1
  205. package/packages/pi-agent-core/dist/harness/env/nodejs.js +34 -3
  206. package/packages/pi-agent-core/dist/harness/env/nodejs.js.map +1 -1
  207. package/packages/pi-agent-core/dist/index.d.ts +1 -0
  208. package/packages/pi-agent-core/dist/index.d.ts.map +1 -1
  209. package/packages/pi-agent-core/dist/index.js +3 -0
  210. package/packages/pi-agent-core/dist/index.js.map +1 -1
  211. package/packages/pi-agent-core/package.json +1 -1
  212. package/packages/pi-ai/dist/models.generated.d.ts +94 -382
  213. package/packages/pi-ai/dist/models.generated.d.ts.map +1 -1
  214. package/packages/pi-ai/dist/models.generated.js +149 -422
  215. package/packages/pi-ai/dist/models.generated.js.map +1 -1
  216. package/packages/pi-ai/package.json +1 -1
  217. package/packages/pi-coding-agent/dist/core/auth-storage.d.ts +2 -2
  218. package/packages/pi-coding-agent/dist/core/auth-storage.d.ts.map +1 -1
  219. package/packages/pi-coding-agent/dist/core/auth-storage.js +19 -13
  220. package/packages/pi-coding-agent/dist/core/auth-storage.js.map +1 -1
  221. package/packages/pi-coding-agent/dist/core/provider-readiness.d.ts.map +1 -1
  222. package/packages/pi-coding-agent/dist/core/provider-readiness.js +13 -6
  223. package/packages/pi-coding-agent/dist/core/provider-readiness.js.map +1 -1
  224. package/packages/pi-coding-agent/dist/core/tools/bash.d.ts +11 -0
  225. package/packages/pi-coding-agent/dist/core/tools/bash.d.ts.map +1 -1
  226. package/packages/pi-coding-agent/dist/core/tools/bash.js +53 -11
  227. package/packages/pi-coding-agent/dist/core/tools/bash.js.map +1 -1
  228. package/packages/pi-coding-agent/dist/index.d.ts +1 -1
  229. package/packages/pi-coding-agent/dist/index.d.ts.map +1 -1
  230. package/packages/pi-coding-agent/dist/index.js +1 -1
  231. package/packages/pi-coding-agent/dist/index.js.map +1 -1
  232. package/packages/pi-coding-agent/dist/utils/shell.d.ts +28 -2
  233. package/packages/pi-coding-agent/dist/utils/shell.d.ts.map +1 -1
  234. package/packages/pi-coding-agent/dist/utils/shell.js +56 -10
  235. package/packages/pi-coding-agent/dist/utils/shell.js.map +1 -1
  236. package/packages/pi-coding-agent/package.json +7 -7
  237. package/packages/pi-tui/dist/tui.d.ts.map +1 -1
  238. package/packages/pi-tui/dist/tui.js +9 -0
  239. package/packages/pi-tui/dist/tui.js.map +1 -1
  240. package/packages/pi-tui/package.json +2 -2
  241. package/packages/rpc-client/package.json +2 -2
  242. package/pkg/package.json +1 -1
  243. package/src/resources/extensions/async-jobs/async-bash-cancel.test.ts +360 -0
  244. package/src/resources/extensions/async-jobs/async-bash-tool.ts +33 -56
  245. package/src/resources/extensions/async-jobs/await-tool.test.ts +139 -0
  246. package/src/resources/extensions/async-jobs/await-tool.ts +82 -12
  247. package/src/resources/extensions/async-jobs/index.ts +79 -0
  248. package/src/resources/extensions/async-jobs/job-manager.ts +21 -1
  249. package/src/resources/extensions/bg-shell/bg-shell-command.ts +6 -6
  250. package/src/resources/extensions/bg-shell/bg-shell-tool.ts +10 -6
  251. package/src/resources/extensions/bg-shell/overlay.ts +9 -5
  252. package/src/resources/extensions/bg-shell/process-manager.ts +50 -25
  253. package/src/resources/extensions/bg-shell/readiness-detector.ts +12 -0
  254. package/src/resources/extensions/bg-shell/tests/lifecycle-and-utilities.test.ts +48 -1
  255. package/src/resources/extensions/bg-shell/utilities.ts +3 -0
  256. package/src/resources/extensions/browser-tools/engine/managed-gsd-browser.ts +265 -98
  257. package/src/resources/extensions/browser-tools/engine/selection.ts +90 -4
  258. package/src/resources/extensions/browser-tools/index.ts +71 -13
  259. package/src/resources/extensions/browser-tools/tests/browser-engine-selection.test.mjs +83 -13
  260. package/src/resources/extensions/browser-tools/tests/gsd-browser-launch-config.test.mjs +29 -1
  261. package/src/resources/extensions/browser-tools/tests/managed-gsd-browser-tools.test.mjs +136 -0
  262. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +34 -4
  263. package/src/resources/extensions/gsd/auto/orchestrator.ts +7 -5
  264. package/src/resources/extensions/gsd/auto-dispatch.ts +12 -0
  265. package/src/resources/extensions/gsd/auto-model-selection.ts +25 -5
  266. package/src/resources/extensions/gsd/auto-post-unit.ts +13 -2
  267. package/src/resources/extensions/gsd/auto-prompts.ts +40 -26
  268. package/src/resources/extensions/gsd/auto-start.ts +15 -10
  269. package/src/resources/extensions/gsd/auto-tool-tracking.ts +19 -0
  270. package/src/resources/extensions/gsd/auto-unit-tool-scope.ts +10 -17
  271. package/src/resources/extensions/gsd/auto-worktree.ts +30 -93
  272. package/src/resources/extensions/gsd/auto.ts +8 -15
  273. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +3 -5
  274. package/src/resources/extensions/gsd/bootstrap/dynamic-tools.ts +23 -6
  275. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +24 -0
  276. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +151 -15
  277. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +6 -2
  278. package/src/resources/extensions/gsd/branch-patterns.ts +3 -0
  279. package/src/resources/extensions/gsd/browser-daemon-auto-prep.ts +108 -0
  280. package/src/resources/extensions/gsd/browser-evidence.ts +18 -2
  281. package/src/resources/extensions/gsd/captures.ts +4 -6
  282. package/src/resources/extensions/gsd/constants.ts +0 -3
  283. package/src/resources/extensions/gsd/crash-recovery.ts +3 -9
  284. package/src/resources/extensions/gsd/doctor-environment.ts +2 -7
  285. package/src/resources/extensions/gsd/doctor-format.ts +12 -7
  286. package/src/resources/extensions/gsd/doctor-runtime-checks.ts +13 -15
  287. package/src/resources/extensions/gsd/error-classifier.ts +11 -0
  288. package/src/resources/extensions/gsd/exec-sandbox.ts +49 -9
  289. package/src/resources/extensions/gsd/guidance.ts +139 -0
  290. package/src/resources/extensions/gsd/guided-flow.ts +16 -2
  291. package/src/resources/extensions/gsd/mcp-filter.ts +2 -23
  292. package/src/resources/extensions/gsd/mcp-tool-name.ts +6 -11
  293. package/src/resources/extensions/gsd/memory-consolidation-scanner.ts +1 -1
  294. package/src/resources/extensions/gsd/migrate/safety.ts +4 -1
  295. package/src/resources/extensions/gsd/notification-store.ts +26 -3
  296. package/src/resources/extensions/gsd/parallel-monitor-overlay.ts +6 -4
  297. package/src/resources/extensions/gsd/paths.ts +33 -0
  298. package/src/resources/extensions/gsd/pre-execution-checks.ts +109 -3
  299. package/src/resources/extensions/gsd/preferences-models.ts +12 -47
  300. package/src/resources/extensions/gsd/prompts/complete-slice.md +1 -1
  301. package/src/resources/extensions/gsd/prompts/plan-slice.md +1 -1
  302. package/src/resources/extensions/gsd/prompts/refine-slice.md +1 -1
  303. package/src/resources/extensions/gsd/prompts/replan-slice.md +1 -1
  304. package/src/resources/extensions/gsd/prompts/run-uat.md +1 -1
  305. package/src/resources/extensions/gsd/prompts/system.md +5 -2
  306. package/src/resources/extensions/gsd/provider-error-guidance.ts +4 -9
  307. package/src/resources/extensions/gsd/provider-switch-observer.ts +1 -1
  308. package/src/resources/extensions/gsd/publication.ts +122 -0
  309. package/src/resources/extensions/gsd/recovery-classification.ts +42 -96
  310. package/src/resources/extensions/gsd/safety/destructive-confirmation.ts +134 -0
  311. package/src/resources/extensions/gsd/state.ts +4 -21
  312. package/src/resources/extensions/gsd/stop-notice.ts +75 -0
  313. package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +22 -0
  314. package/src/resources/extensions/gsd/tests/auto-orchestrator.test.ts +16 -19
  315. package/src/resources/extensions/gsd/tests/browser-automation-contract-fixture.ts +39 -0
  316. package/src/resources/extensions/gsd/tests/browser-contract.test.ts +44 -0
  317. package/src/resources/extensions/gsd/tests/browser-daemon-auto-prep.test.ts +144 -0
  318. package/src/resources/extensions/gsd/tests/checkout-branch-stash-guard.test.ts +66 -1
  319. package/src/resources/extensions/gsd/tests/clear-stale-autostart.test.ts +22 -0
  320. package/src/resources/extensions/gsd/tests/commands-verdict.test.ts +8 -7
  321. package/src/resources/extensions/gsd/tests/destructive-confirmation.test.ts +303 -0
  322. package/src/resources/extensions/gsd/tests/dispatch-run-uat-browser-tools.test.ts +2 -1
  323. package/src/resources/extensions/gsd/tests/dynamic-bash-no-cap.test.ts +132 -0
  324. package/src/resources/extensions/gsd/tests/exec-graceful-kill.test.ts +193 -0
  325. package/src/resources/extensions/gsd/tests/exec-tool.test.ts +29 -1
  326. package/src/resources/extensions/gsd/tests/extension-bootstrap-isolation.test.ts +35 -1
  327. package/src/resources/extensions/gsd/tests/guidance.test.ts +125 -0
  328. package/src/resources/extensions/gsd/tests/integration/auto-worktree-milestone-merge.test.ts +53 -11
  329. package/src/resources/extensions/gsd/tests/integration/auto-worktree.test.ts +73 -58
  330. package/src/resources/extensions/gsd/tests/integration/gsd-integration-fixture.ts +80 -0
  331. package/src/resources/extensions/gsd/tests/mcp-project-config.test.ts +3 -1
  332. package/src/resources/extensions/gsd/tests/model-unittype-mapping.test.ts +32 -1
  333. package/src/resources/extensions/gsd/tests/notification-store.test.ts +32 -0
  334. package/src/resources/extensions/gsd/tests/oauth-api-model-routing.test.ts +167 -0
  335. package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +193 -1
  336. package/src/resources/extensions/gsd/tests/provider-error-guidance.test.ts +3 -3
  337. package/src/resources/extensions/gsd/tests/publication.test.ts +120 -0
  338. package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +157 -0
  339. package/src/resources/extensions/gsd/tests/runtime-invariant-modules.test.ts +1 -0
  340. package/src/resources/extensions/gsd/tests/stop-notice.test.ts +70 -0
  341. package/src/resources/extensions/gsd/tests/token-tool-gating.test.ts +76 -0
  342. package/src/resources/extensions/gsd/tests/tool-invocation-error-loop-break.test.ts +8 -0
  343. package/src/resources/extensions/gsd/tests/tool-surface-readiness.test.ts +155 -0
  344. package/src/resources/extensions/gsd/tests/uat-policy.test.ts +24 -29
  345. package/src/resources/extensions/gsd/tests/unit-closeout.test.ts +209 -0
  346. package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +67 -2
  347. package/src/resources/extensions/gsd/tests/unit-registry.test.ts +163 -0
  348. package/src/resources/extensions/gsd/tests/web-app-uat.test.ts +44 -1
  349. package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +2 -2
  350. package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +2 -2
  351. package/src/resources/extensions/gsd/tool-surface-readiness.ts +76 -0
  352. package/src/resources/extensions/gsd/tools/exec-tool.ts +8 -7
  353. package/src/resources/extensions/gsd/tools/plan-slice.ts +12 -6
  354. package/src/resources/extensions/gsd/uat-policy.ts +2 -1
  355. package/src/resources/extensions/gsd/unit-closeout.ts +201 -0
  356. package/src/resources/extensions/gsd/unit-context-composer.ts +111 -1
  357. package/src/resources/extensions/gsd/unit-context-manifest.ts +4 -28
  358. package/src/resources/extensions/gsd/unit-registry.ts +412 -0
  359. package/src/resources/extensions/gsd/unit-tool-contracts.ts +27 -192
  360. package/src/resources/extensions/gsd/web-app-uat.ts +51 -8
  361. package/src/resources/extensions/gsd/workflow-tool-surface.ts +4 -1
  362. package/src/resources/extensions/gsd/worktree-git-recovery.ts +15 -9
  363. package/src/resources/extensions/gsd/worktree-root.ts +12 -0
  364. package/src/resources/extensions/gsd/worktree-session-state.ts +3 -5
  365. package/src/resources/extensions/search-the-web/native-search.ts +5 -3
  366. package/src/resources/extensions/shared/browser-contract.ts +66 -0
  367. package/src/resources/extensions/shared/gsd-browser-cli.ts +119 -5
  368. package/src/resources/shared/package.json +3 -0
  369. package/src/resources/skills/create-skill/references/executable-code.md +1 -1
  370. package/src/resources/skills/create-skill/workflows/add-reference.md +8 -3
  371. package/src/resources/skills/create-skill/workflows/add-script.md +4 -2
  372. package/src/resources/skills/create-skill/workflows/add-template.md +3 -1
  373. package/src/resources/skills/create-skill/workflows/add-workflow.md +8 -3
  374. package/src/resources/skills/create-skill/workflows/upgrade-to-router.md +10 -5
  375. package/src/resources/skills/create-skill/workflows/verify-skill.md +9 -4
  376. package/src/resources/skills/spike-wrap-up/SKILL.md +9 -9
  377. /package/dist/web/standalone/.next/static/{C24pqUd-aru-l0Dp0gLZP → mU4QIDlpVHDdjDpeEKh5W}/_buildManifest.js +0 -0
  378. /package/dist/web/standalone/.next/static/{C24pqUd-aru-l0Dp0gLZP → mU4QIDlpVHDdjDpeEKh5W}/_ssgManifest.js +0 -0
@@ -2,7 +2,7 @@
2
2
  import { importExtensionModule, type ExtensionAPI, type ExtensionContext } from "@gsd/pi-coding-agent";
3
3
 
4
4
  import { closeManagedGsdBrowser, registerManagedGsdBrowserTools, warmUpManagedGsdBrowser } from "./engine/managed-gsd-browser.js";
5
- import { resolveBrowserEngineMode, type BrowserEngineMode } from "./engine/selection.js";
5
+ import { commitBrowserEngineResolution, resolveAmbientBrowserEngineResolution, type BrowserEngineMode } from "./engine/selection.js";
6
6
  import { setArtifactRootForCwd } from "./state.js";
7
7
  import { detectWebApp } from "./web-app-detect.js";
8
8
 
@@ -167,13 +167,59 @@ function withBrowserArtifactCwdScope(pi: ExtensionAPI): ExtensionAPI {
167
167
  };
168
168
  }
169
169
 
170
- async function registerBrowserTools(pi: ExtensionAPI): Promise<void> {
171
- const engine = resolveBrowserEngineMode();
170
+ /** Daemon-connect budget when the probe-resolved managed engine is verified at session start. */
171
+ const PROBE_WARMUP_TIMEOUT_MS = 10_000;
172
+
173
+ async function registerBrowserTools(pi: ExtensionAPI, ctx: ExtensionContext): Promise<void> {
174
+ const projectRoot = ctx.cwd || process.cwd();
175
+ const resolution = resolveAmbientBrowserEngineResolution(projectRoot);
176
+ let engine = resolution.engine;
172
177
  if (engine === "off") return;
178
+
179
+ // A probe-resolved managed engine is only a prediction that gsd-browser
180
+ // works — prove it by connecting the daemon before committing the session's
181
+ // tool registrations to it. Connect failure falls back to legacy Playwright
182
+ // (the failure mode that made ADR-024 freeze the old default) and commits
183
+ // the outcome so ambient readers see the engine actually in use. When eager
184
+ // warm-up is disabled the daemon-connect proof cannot run, so the probe
185
+ // default treats the managed engine as unprovable and falls back to legacy
186
+ // rather than registering it unverified. An explicit
187
+ // GSD_BROWSER_ENGINE=gsd-browser override skips the gate and is honored
188
+ // verbatim, matching prior behavior.
189
+ if (engine === "gsd-browser" && resolution.source === "probe" && !registeredEngine) {
190
+ if (isWarmUpDisabled()) {
191
+ engine = commitLegacyFallback(projectRoot, "warm-up disabled; managed engine unverifiable; using legacy Playwright");
192
+ } else {
193
+ const warmUp = await warmUpManagedGsdBrowser(ctx, AbortSignal.timeout(PROBE_WARMUP_TIMEOUT_MS));
194
+ if (!warmUp.ok) {
195
+ engine = commitLegacyFallback(projectRoot, `gsd-browser daemon connect failed (${warmUp.error}); using legacy Playwright`);
196
+ if (ctx.hasUI) {
197
+ ctx.ui.notify(
198
+ `gsd-browser engine unavailable (${warmUp.error}); using Playwright browser tools for this session.`,
199
+ "warning",
200
+ );
201
+ }
202
+ } else if (warmUp.coverageWarning && ctx.hasUI) {
203
+ ctx.ui.notify(warmUp.coverageWarning, "warning");
204
+ }
205
+ }
206
+ }
207
+
208
+ // Browser tool registrations are process-global and cannot be swapped once
209
+ // live. When an earlier session in this process already registered an engine
210
+ // and this project resolved a different one (per-project probe resolution can
211
+ // diverge across projects in a multi-session process), adopt the registered
212
+ // engine rather than throwing — a throw surfaces as "browser-tools failed to
213
+ // load" and leaves this session with no browser tools at all. Commit the
214
+ // adoption so ambient readers (UAT guidance, warm-up) describe the engine
215
+ // actually in use.
173
216
  if (registeredEngine && registeredEngine !== engine) {
174
- throw new Error(
175
- `Browser tools already registered with GSD_BROWSER_ENGINE=${registeredEngine}. Restart GSD before switching to ${engine}.`,
176
- );
217
+ engine = registeredEngine;
218
+ commitBrowserEngineResolution(projectRoot, {
219
+ engine,
220
+ source: "probe",
221
+ reason: `browser tools already registered with ${engine} earlier in this process; adopting it`,
222
+ });
177
223
  }
178
224
 
179
225
  let registration: Promise<void>;
@@ -202,29 +248,41 @@ async function registerBrowserTools(pi: ExtensionAPI): Promise<void> {
202
248
  }
203
249
  }
204
250
 
251
+ function commitLegacyFallback(projectRoot: string, reason: string): "legacy" {
252
+ commitBrowserEngineResolution(projectRoot, { engine: "legacy", source: "probe", reason });
253
+ return "legacy";
254
+ }
255
+
205
256
  function isWarmUpDisabled(): boolean {
206
257
  const value = process.env.GSD_BROWSER_WARMUP?.trim().toLowerCase();
207
258
  return value === "0" || value === "false" || value === "off";
208
259
  }
209
260
 
210
261
  /**
211
- * Auto-initialize the managed gsd-browser engine only when explicitly selected
212
- * for a web app. Best-effort and non-blocking: warm-up runs in the background
213
- * and only surfaces a warning if it fails.
262
+ * Auto-initialize the managed gsd-browser engine when it was selected via the
263
+ * explicit GSD_BROWSER_ENGINE override, which registers without the
264
+ * daemon-connect gate. Best-effort and non-blocking: warm-up runs in the
265
+ * background and only surfaces a warning if it fails. Probe-resolved sessions
266
+ * already connected (or fell back) during registration, so they are excluded
267
+ * to avoid re-warming and double-notifying.
214
268
  */
215
269
  function maybeWarmUpManagedEngine(pi: ExtensionAPI, ctx: ExtensionContext): void {
216
270
  if (isWarmUpDisabled()) return;
217
- if (resolveBrowserEngineMode() !== "gsd-browser") return;
218
271
 
219
272
  const projectRoot = ctx.cwd || process.cwd();
273
+ const resolution = resolveAmbientBrowserEngineResolution(projectRoot);
274
+ if (resolution.engine !== "gsd-browser" || resolution.source !== "env") return;
220
275
  if (!detectWebApp(projectRoot)) return;
221
276
 
222
277
  void warmUpManagedGsdBrowser(ctx).then((result) => {
223
- if (!result.ok && ctx.hasUI) {
278
+ if (!ctx.hasUI) return;
279
+ if (!result.ok) {
224
280
  ctx.ui.notify(
225
281
  `gsd-browser auto-init failed: ${result.error}. Browser UAT tools will retry on first use; run /gsd doctor if this persists.`,
226
282
  "warning",
227
283
  );
284
+ } else if (result.coverageWarning) {
285
+ ctx.ui.notify(result.coverageWarning, "warning");
228
286
  }
229
287
  });
230
288
  }
@@ -240,7 +298,7 @@ async function closeActiveBrowserEngines(): Promise<void> {
240
298
  export default function (pi: ExtensionAPI) {
241
299
  pi.on("session_start", async (_event, ctx) => {
242
300
  if (ctx.hasUI) {
243
- void registerBrowserTools(pi)
301
+ void registerBrowserTools(pi, ctx)
244
302
  .then(() => maybeWarmUpManagedEngine(pi, ctx))
245
303
  .catch((error) => {
246
304
  ctx.ui.notify(`browser-tools failed to load: ${error instanceof Error ? error.message : String(error)}`, "warning");
@@ -248,7 +306,7 @@ export default function (pi: ExtensionAPI) {
248
306
  return;
249
307
  }
250
308
 
251
- await registerBrowserTools(pi);
309
+ await registerBrowserTools(pi, ctx);
252
310
  maybeWarmUpManagedEngine(pi, ctx);
253
311
  });
254
312
 
@@ -1,35 +1,105 @@
1
1
  import { describe, it } from "node:test";
2
2
  import assert from "node:assert/strict";
3
+ import { mkdtempSync, writeFileSync } from "node:fs";
3
4
  import { createRequire } from "node:module";
4
- import { dirname } from "node:path";
5
+ import { tmpdir } from "node:os";
6
+ import { dirname, join } from "node:path";
5
7
  import { fileURLToPath } from "node:url";
6
8
 
7
9
  const __dirname = dirname(fileURLToPath(import.meta.url));
8
10
  const require = createRequire(import.meta.url);
9
11
  const jiti = require("jiti")(__dirname, { interopDefault: true, debug: false });
10
12
 
11
- const { resolveBrowserEngineMode } = jiti("../engine/selection.ts");
13
+ const {
14
+ commitBrowserEngineResolution,
15
+ resolveAmbientBrowserEngineResolution,
16
+ resolveBrowserEngineResolution,
17
+ } = jiti("../engine/selection.ts");
12
18
 
13
- describe("resolveBrowserEngineMode", () => {
14
- it("defaults to the Playwright engine", () => {
15
- assert.equal(resolveBrowserEngineMode({}), "legacy");
16
- });
19
+ function makeProject({ webApp }) {
20
+ const dir = mkdtempSync(join(tmpdir(), "gsd-engine-selection-"));
21
+ const pkg = webApp ? { dependencies: { react: "^18.0.0" } } : { name: "cli-tool" };
22
+ writeFileSync(join(dir, "package.json"), JSON.stringify(pkg));
23
+ return dir;
24
+ }
25
+
26
+ function makeFakeCli() {
27
+ const dir = mkdtempSync(join(tmpdir(), "gsd-fake-cli-"));
28
+ const cliPath = join(dir, "gsd-browser");
29
+ writeFileSync(cliPath, "#!/bin/sh\n");
30
+ return cliPath;
31
+ }
17
32
 
18
- it("accepts the explicit engine modes", () => {
19
- assert.equal(resolveBrowserEngineMode({ GSD_BROWSER_ENGINE: "gsd-browser" }), "gsd-browser");
20
- assert.equal(resolveBrowserEngineMode({ GSD_BROWSER_ENGINE: "legacy" }), "legacy");
21
- assert.equal(resolveBrowserEngineMode({ GSD_BROWSER_ENGINE: "off" }), "off");
33
+ describe("resolveBrowserEngineResolution", () => {
34
+ it("honors the explicit engine modes verbatim with env source", () => {
35
+ assert.deepEqual(
36
+ ["gsd-browser", "legacy", "off"].map(
37
+ (mode) => resolveBrowserEngineResolution({ GSD_BROWSER_ENGINE: mode }).engine,
38
+ ),
39
+ ["gsd-browser", "legacy", "off"],
40
+ );
41
+ assert.equal(resolveBrowserEngineResolution({ GSD_BROWSER_ENGINE: "gsd-browser" }).source, "env");
22
42
  });
23
43
 
24
44
  it("accepts compatibility aliases", () => {
25
- assert.equal(resolveBrowserEngineMode({ GSD_BROWSER_ENGINE: "playwright" }), "legacy");
26
- assert.equal(resolveBrowserEngineMode({ GSD_BROWSER_ENGINE: "false" }), "off");
45
+ assert.equal(resolveBrowserEngineResolution({ GSD_BROWSER_ENGINE: "playwright" }).engine, "legacy");
46
+ assert.equal(resolveBrowserEngineResolution({ GSD_BROWSER_ENGINE: "false" }).engine, "off");
27
47
  });
28
48
 
29
49
  it("rejects unknown engine modes", () => {
30
50
  assert.throws(
31
- () => resolveBrowserEngineMode({ GSD_BROWSER_ENGINE: "surprise" }),
51
+ () => resolveBrowserEngineResolution({ GSD_BROWSER_ENGINE: "surprise" }),
32
52
  /Expected "gsd-browser", "legacy", or "off"/,
33
53
  );
34
54
  });
55
+
56
+ it("defaults to legacy Playwright when no project root is known", () => {
57
+ const resolution = resolveBrowserEngineResolution({});
58
+ assert.equal(resolution.engine, "legacy");
59
+ assert.equal(resolution.source, "probe");
60
+ });
61
+
62
+ it("keeps legacy Playwright for non-browser-facing projects", () => {
63
+ const cliPath = makeFakeCli();
64
+ const resolution = resolveBrowserEngineResolution(
65
+ { GSD_BROWSER_CLI_PATH: cliPath },
66
+ makeProject({ webApp: false }),
67
+ );
68
+ assert.equal(resolution.engine, "legacy");
69
+ assert.match(resolution.reason, /not browser-facing/);
70
+ });
71
+
72
+ it("prefers the managed gsd-browser engine for web apps when the CLI is provable", () => {
73
+ const cliPath = makeFakeCli();
74
+ const resolution = resolveBrowserEngineResolution(
75
+ { GSD_BROWSER_CLI_PATH: cliPath },
76
+ makeProject({ webApp: true }),
77
+ );
78
+ assert.equal(resolution.engine, "gsd-browser");
79
+ assert.equal(resolution.source, "probe");
80
+ assert.match(resolution.reason, /web app detected/);
81
+ });
82
+
83
+ it("falls back to legacy Playwright with a recorded reason when the CLI is unavailable", () => {
84
+ const resolution = resolveBrowserEngineResolution(
85
+ { GSD_BROWSER_CLI_PATH: "/nonexistent/gsd-browser" },
86
+ makeProject({ webApp: true }),
87
+ );
88
+ assert.equal(resolution.engine, "legacy");
89
+ assert.equal(resolution.source, "probe");
90
+ assert.match(resolution.reason, /falling back to legacy Playwright/);
91
+ });
92
+ });
93
+
94
+ describe("committed resolution", () => {
95
+ it("ambient readers see a committed verification outcome instead of the prediction", () => {
96
+ const projectRoot = makeProject({ webApp: true });
97
+ const fallback = {
98
+ engine: "legacy",
99
+ source: "probe",
100
+ reason: "gsd-browser daemon connect failed (test); using legacy Playwright",
101
+ };
102
+ commitBrowserEngineResolution(projectRoot, fallback);
103
+ assert.deepEqual(resolveAmbientBrowserEngineResolution(projectRoot), fallback);
104
+ });
35
105
  });
@@ -1,7 +1,10 @@
1
1
  import { describe, it } from "node:test";
2
2
  import assert from "node:assert/strict";
3
3
 
4
- const { resolveGsdBrowserMcpLaunchConfig } = await import("../../shared/gsd-browser-cli.ts");
4
+ const {
5
+ resolveGsdBrowserDaemonStartInvocation,
6
+ resolveGsdBrowserMcpLaunchConfig,
7
+ } = await import("../../shared/gsd-browser-cli.ts");
5
8
 
6
9
  describe("resolveGsdBrowserMcpLaunchConfig identity flags", () => {
7
10
  it("emits a non-empty --identity-key alongside --identity-scope", () => {
@@ -34,4 +37,29 @@ describe("resolveGsdBrowserMcpLaunchConfig identity flags", () => {
34
37
  });
35
38
  assert.equal(args[args.indexOf("--identity-key") + 1], "custom-key");
36
39
  });
40
+
41
+ it("uses a path-safe identity-project identifier", () => {
42
+ const { args } = resolveGsdBrowserMcpLaunchConfig("/tmp/example/project", {});
43
+ const projectId = args[args.indexOf("--identity-project") + 1];
44
+ assert.equal(typeof projectId, "string");
45
+ assert.doesNotMatch(projectId, /[\\/]/);
46
+ });
47
+ });
48
+
49
+ describe("resolveGsdBrowserDaemonStartInvocation", () => {
50
+ it("mirrors MCP session and identity flags with daemon start", () => {
51
+ const launch = resolveGsdBrowserMcpLaunchConfig("/tmp/example-project", {});
52
+ const daemon = resolveGsdBrowserDaemonStartInvocation("/tmp/example-project", {});
53
+
54
+ assert.equal(daemon.command, launch.command);
55
+ assert.equal(daemon.cwd, launch.cwd);
56
+ assert.deepEqual(
57
+ daemon.args.slice(daemon.args.indexOf("--session")),
58
+ launch.args.slice(launch.args.indexOf("--session")),
59
+ );
60
+ assert.deepEqual(
61
+ daemon.args.slice(0, daemon.args.indexOf("--session")),
62
+ [...launch.args.slice(0, launch.args.indexOf("mcp")), "daemon", "start"],
63
+ );
64
+ });
37
65
  });
@@ -2,10 +2,33 @@ import { describe, it } from "node:test";
2
2
  import assert from "node:assert/strict";
3
3
 
4
4
  const {
5
+ MANAGED_BROWSER_TOOL_SPECS,
5
6
  MANAGED_GSD_BROWSER_TOOL_NAMES,
7
+ findMissingContractCoverage,
8
+ normalizeManagedArgs,
6
9
  registerManagedGsdBrowserTools,
7
10
  } = await import("../engine/managed-gsd-browser.ts");
8
11
 
12
+ // The tools @opengsd/gsd-browser actually serves over MCP (subset relevant to
13
+ // the contract). Notably absent: browser_click, browser_type, browser_verify,
14
+ // browser_reload — those are satisfied through translations.
15
+ const GSD_BROWSER_SERVED_TOOLS = [
16
+ "browser_navigate",
17
+ "browser_snapshot",
18
+ "browser_click_ref",
19
+ "browser_fill_ref",
20
+ "browser_fill_form",
21
+ "browser_wait_for",
22
+ "browser_assert",
23
+ "browser_screenshot",
24
+ "browser_find_element",
25
+ "browser_console",
26
+ "browser_network",
27
+ "browser_evaluate",
28
+ "browser_batch",
29
+ "browser_act",
30
+ ];
31
+
9
32
  describe("registerManagedGsdBrowserTools", () => {
10
33
  it("registers the curated Pi browser contract", () => {
11
34
  const tools = [];
@@ -31,3 +54,116 @@ describe("registerManagedGsdBrowserTools", () => {
31
54
  assert.equal(screenshot?.compatibility?.producesImages, true);
32
55
  });
33
56
  });
57
+
58
+ describe("findMissingContractCoverage", () => {
59
+ it("reports nothing for the tool list gsd-browser actually serves", () => {
60
+ assert.deepEqual(findMissingContractCoverage(GSD_BROWSER_SERVED_TOOLS), []);
61
+ });
62
+
63
+ it("reports contract tools none of whose MCP candidates are served", () => {
64
+ const served = GSD_BROWSER_SERVED_TOOLS.filter((name) => name !== "browser_assert");
65
+ // browser_verify also depends on browser_assert through its translation.
66
+ assert.deepEqual(findMissingContractCoverage(served), ["browser_assert", "browser_verify"]);
67
+ });
68
+
69
+ it("reports translated tools when a required MCP tool is missing", () => {
70
+ const served = GSD_BROWSER_SERVED_TOOLS.filter((name) => name !== "browser_batch");
71
+ assert.deepEqual(findMissingContractCoverage(served), ["browser_click", "browser_type", "browser_batch"]);
72
+ });
73
+ });
74
+
75
+ describe("contract tool translations", () => {
76
+ it("translates browser_click into a single-step batch call", () => {
77
+ const calls = MANAGED_BROWSER_TOOL_SPECS.browser_click.translate.build({ selector: "#save" });
78
+ assert.deepEqual(calls, [{
79
+ mcpTool: "browser_batch",
80
+ args: { steps: [{ action: "click", selector: "#save" }] },
81
+ }]);
82
+ });
83
+
84
+ it("translates browser_type into a single-step batch call", () => {
85
+ const calls = MANAGED_BROWSER_TOOL_SPECS.browser_type.translate.build({
86
+ selector: "#name",
87
+ text: "hello",
88
+ clearFirst: true,
89
+ submit: true,
90
+ });
91
+ assert.deepEqual(calls, [{
92
+ mcpTool: "browser_batch",
93
+ args: { steps: [{ action: "type", selector: "#name", text: "hello", clearFirst: true, submit: true }] },
94
+ }]);
95
+ });
96
+
97
+ it("normalizes batch options and step keys to the daemon's snake_case", () => {
98
+ const normalized = normalizeManagedArgs("browser_batch", {
99
+ steps: [{ action: "type", selector: "#name", text: "hi", clearFirst: true }],
100
+ stopOnFailure: false,
101
+ finalSummaryOnly: true,
102
+ });
103
+ assert.deepEqual(normalized, {
104
+ steps: [{ action: "type", selector: "#name", text: "hi", clear_first: true }],
105
+ stop_on_failure: false,
106
+ summary_only: true,
107
+ });
108
+ });
109
+
110
+ it("translates browser_verify into navigate, assert, and screenshot calls", () => {
111
+ const calls = MANAGED_BROWSER_TOOL_SPECS.browser_verify.translate.build({
112
+ url: "http://localhost:3000",
113
+ timeout: 5000,
114
+ checks: [
115
+ { description: "heading shows", selector: "h1", expectedText: "Welcome" },
116
+ { description: "spinner gone", selector: ".spinner", expectedVisible: false },
117
+ { description: "evidence", selector: "main", expectedVisible: true, screenshot: true },
118
+ ],
119
+ });
120
+ assert.deepEqual(calls, [
121
+ { mcpTool: "browser_navigate", args: { url: "http://localhost:3000", timeout: 5000 } },
122
+ {
123
+ mcpTool: "browser_assert",
124
+ args: {
125
+ checks: [
126
+ { kind: "text_visible", text: "Welcome" },
127
+ { kind: "selector_hidden", selector: ".spinner" },
128
+ { kind: "selector_visible", selector: "main" },
129
+ ],
130
+ },
131
+ },
132
+ { mcpTool: "browser_screenshot", args: {}, optional: true },
133
+ ]);
134
+ });
135
+
136
+ it("declares every tool a translation can emit in its coverage requirements", () => {
137
+ for (const [name, spec] of Object.entries(MANAGED_BROWSER_TOOL_SPECS)) {
138
+ if (!spec.translate) continue;
139
+ const maximalArgs = {
140
+ url: "http://localhost:3000",
141
+ timeout: 5000,
142
+ selector: "#el",
143
+ text: "hi",
144
+ clearFirst: true,
145
+ checks: [{ description: "d", selector: "#el", expectedText: "hi", expectedVisible: true, screenshot: true }],
146
+ };
147
+ const emitted = spec.translate.build(maximalArgs).map((call) => call.mcpTool);
148
+ for (const mcpTool of emitted) {
149
+ assert.ok(
150
+ spec.translate.requires.includes(mcpTool),
151
+ `${name} translation emits ${mcpTool} but does not require it for coverage`,
152
+ );
153
+ }
154
+ }
155
+ });
156
+
157
+ it("translates browser_verify without checks into navigation only", () => {
158
+ const calls = MANAGED_BROWSER_TOOL_SPECS.browser_verify.translate.build({ url: "http://localhost:3000", checks: [] });
159
+ assert.deepEqual(calls, [{ mcpTool: "browser_navigate", args: { url: "http://localhost:3000" } }]);
160
+ });
161
+
162
+ it("translates browser_reload into evaluate plus best-effort network-idle wait", () => {
163
+ const calls = MANAGED_BROWSER_TOOL_SPECS.browser_reload.translate.build({});
164
+ assert.deepEqual(calls, [
165
+ { mcpTool: "browser_evaluate", args: { expression: "location.reload()" } },
166
+ { mcpTool: "browser_wait_for", args: { condition: "network_idle", timeout: 3_000 }, optional: true },
167
+ ]);
168
+ });
169
+ });
@@ -59,6 +59,8 @@ import {
59
59
  computeMcpDisallowedTools,
60
60
  } from "../gsd/mcp-filter.js";
61
61
  import { RUN_UAT_CLAUDE_NATIVE_TOOL_NAMES, RUN_UAT_FORBIDDEN_TOOL_NAMES, RUN_UAT_WORKFLOW_TOOL_NAMES, resolveToolPresentationPlan } from "../gsd/tool-presentation-plan.js";
62
+ import { getToolSurfaceReadinessError } from "../gsd/tool-surface-readiness.js";
63
+ import { hasBrowserContractPrefix } from "../shared/browser-contract.js";
62
64
  import { showInterviewRound, type Question, type RoundResult } from "../shared/tui.js";
63
65
  import type {
64
66
  SDKAssistantMessage,
@@ -1590,7 +1592,7 @@ function browserMcpServerNameFromAllowedTools(allowedTools: unknown): string | u
1590
1592
  if (typeof toolName !== "string") continue;
1591
1593
  const parsed = parseAllowedMcpToolName(toolName);
1592
1594
  if (!parsed) continue;
1593
- if (parsed.server === "gsd-browser" || parsed.tool.startsWith("browser_")) {
1595
+ if (parsed.server === "gsd-browser" || hasBrowserContractPrefix(parsed.tool)) {
1594
1596
  return parsed.server;
1595
1597
  }
1596
1598
  }
@@ -1603,7 +1605,7 @@ function workflowMcpServerNameFromAllowedTools(allowedTools: unknown): string |
1603
1605
  for (const toolName of allowedTools) {
1604
1606
  if (typeof toolName !== "string") continue;
1605
1607
  const parsed = parseAllowedMcpToolName(toolName);
1606
- if (!parsed || parsed.server === browserServerName || parsed.tool.startsWith("browser_")) continue;
1608
+ if (!parsed || parsed.server === browserServerName || hasBrowserContractPrefix(parsed.tool)) continue;
1607
1609
  return parsed.server;
1608
1610
  }
1609
1611
  return undefined;
@@ -1998,8 +2000,9 @@ async function pumpSdkMessages(
1998
2000
  : {}),
1999
2001
  },
2000
2002
  );
2003
+ const workflowMcpServerName = workflowMcpServerNameFromAllowedTools(sdkOpts.allowedTools);
2001
2004
  const prompt = buildPromptFromContext(context, {
2002
- workflowMcpServerName: workflowMcpServerNameFromAllowedTools(sdkOpts.allowedTools),
2005
+ workflowMcpServerName,
2003
2006
  browserMcpServerName: browserMcpServerNameFromAllowedTools(sdkOpts.allowedTools),
2004
2007
  });
2005
2008
  const queryPrompt = buildSdkQueryPrompt(context, prompt);
@@ -2041,7 +2044,34 @@ async function pumpSdkMessages(
2041
2044
  switch (msg.type) {
2042
2045
  // -- Init --
2043
2046
  case "system": {
2044
- // Nothing to emit the stream is already started.
2047
+ // Tool Surface Readiness gate: the init message is the first (and
2048
+ // only) point where the session reports its live tool surface and
2049
+ // MCP server statuses. If the workflow server failed or has not
2050
+ // registered this Unit's required tools, abort before the first
2051
+ // model turn with a transient, recovery-classifiable error
2052
+ // (tool-unavailable → retry) instead of letting the model hit
2053
+ // "No such tool available" mid-Unit and improvise around it.
2054
+ const init = msg as unknown as {
2055
+ subtype?: string;
2056
+ tools?: string[];
2057
+ mcp_servers?: { name: string; status: string }[];
2058
+ };
2059
+ if (init.subtype === "init") {
2060
+ const readinessError = getToolSurfaceReadinessError({
2061
+ unitType: gsdPhase,
2062
+ workflowServerName: workflowMcpServerName,
2063
+ observation: { tools: init.tools ?? [], mcpServers: init.mcp_servers ?? [] },
2064
+ });
2065
+ if (readinessError) {
2066
+ controller.abort();
2067
+ stream.push({
2068
+ type: "error",
2069
+ reason: "error",
2070
+ error: makeErrorMessage(modelId, readinessError),
2071
+ });
2072
+ return;
2073
+ }
2074
+ }
2045
2075
  break;
2046
2076
  }
2047
2077
 
@@ -1023,16 +1023,18 @@ export class AutoOrchestrator implements AutoOrchestrationModule {
1023
1023
  // checks coexist: idempotency for the common immediate-repeat case,
1024
1024
  // stuck-loop for the saturated-window case.
1025
1025
  if (this.lastAdvanceKey === nextKey && matchingCount < STUCK_WINDOW_SIZE) {
1026
+ // Unit already active — benign no-op. Return skipped so the loop re-polls
1027
+ // without cancelling the in-flight unit (blocked+pause would force-cancel it).
1026
1028
  this.clearPendingDispatch();
1027
- const blocked: AutoAdvanceResult = { kind: "blocked", reason: "idempotent advance: unit already active", action: "pause" };
1029
+ const skipped: AutoAdvanceResult = { kind: "skipped", reason: "idempotent advance: unit already active" };
1028
1030
  this.journalTransition({
1029
- name: "advance-blocked",
1030
- reason: blocked.reason,
1031
+ name: "advance-skipped",
1032
+ reason: skipped.reason,
1031
1033
  unitType: decision.unitType,
1032
1034
  unitId: decision.unitId,
1033
1035
  });
1034
- this.postAdvanceRecord(blocked);
1035
- return blocked;
1036
+ this.postAdvanceRecord(skipped);
1037
+ return skipped;
1036
1038
  }
1037
1039
 
1038
1040
  // Stuck-loop detection: when the ring is saturated with copies of
@@ -90,7 +90,9 @@ import { MILESTONE_ID_RE } from "./milestone-ids.js";
90
90
  import {
91
91
  getWorkflowTransportSupportError,
92
92
  getRequiredWorkflowToolsForAutoUnit,
93
+ resolveWorkflowMcpProjectRoot,
93
94
  } from "./workflow-mcp.js";
95
+ import { prepareBrowserDaemonForUat } from "./browser-daemon-auto-prep.js";
94
96
  import {
95
97
  PROJECT_RESEARCH_INFLIGHT_MARKER,
96
98
  } from "./project-research-policy.js";
@@ -772,6 +774,16 @@ export const DISPATCH_RULES: DispatchRule[] = [
772
774
  if (browserToolError) {
773
775
  return { action: "stop" as const, reason: browserToolError, level: "warning" as const };
774
776
  }
777
+ const browserDaemonError = prepareBrowserDaemonForUat({
778
+ uatType,
779
+ sessionProvider,
780
+ sessionAuthMode,
781
+ sessionBaseUrl,
782
+ projectRoot: resolveWorkflowMcpProjectRoot(basePath),
783
+ });
784
+ if (browserDaemonError) {
785
+ return { action: "stop" as const, reason: browserDaemonError, level: "warning" as const };
786
+ }
775
787
 
776
788
  // Cap run-uat dispatch attempts to prevent infinite replay (#3624).
777
789
  // Check before incrementing so an exhausted counter cannot create a
@@ -1041,16 +1041,36 @@ export function resolveModelId<T extends { id: string; provider: string }>(
1041
1041
  if (providerMatch) return providerMatch;
1042
1042
  }
1043
1043
 
1044
- // Prefer "anthropic" as the canonical provider for Anthropic models.
1045
- // Transport-specific tiebreaker (ADR-012): intentionally keys on provider,
1046
- // not api we want the plain Anthropic transport when multiple are available.
1047
- const anthropicMatch = candidates.find(m => m.provider === "anthropic");
1048
- if (anthropicMatch) return anthropicMatch;
1044
+ // Subscription/OAuth routes beat pay-per-token API when the same model ID
1045
+ // exists on multiple providers. Order matters — first match wins.
1046
+ for (const provider of BARE_ID_SUBSCRIPTION_PROVIDER_PRECEDENCE) {
1047
+ const match = candidates.find(m => m.provider === provider);
1048
+ if (match) return match;
1049
+ }
1049
1050
 
1050
1051
  // Fall back to first non-extension candidate, or any candidate
1051
1052
  return candidates.find(m => !EXTENSION_PROVIDERS.has(m.provider)) ?? candidates[0];
1052
1053
  }
1053
1054
 
1055
+ /**
1056
+ * When a bare model ID exists on multiple providers, prefer subscription/OAuth
1057
+ * routes over pay-per-token API keys. Matches PROVIDER_ROUTES in doctor-providers
1058
+ * but applies when *both* sides are authenticated.
1059
+ *
1060
+ * Order rationale:
1061
+ * - openai-codex before github-copilot: ChatGPT-native for shared GPT IDs
1062
+ * - google-gemini-cli before github-copilot: first-party Gemini CLI
1063
+ * - anthropic before github-copilot: first-party Claude API/OAuth over Copilot
1064
+ * - github-copilot before openai/google: Copilot OAuth over platform API keys
1065
+ */
1066
+ export const BARE_ID_SUBSCRIPTION_PROVIDER_PRECEDENCE = [
1067
+ "openai-codex",
1068
+ "google-gemini-cli",
1069
+ "anthropic",
1070
+ "github-copilot",
1071
+ "google-antigravity",
1072
+ ] as const;
1073
+
1054
1074
  /**
1055
1075
  * Flat-rate providers charge the same per request regardless of model.
1056
1076
  * Dynamic routing provides no cost benefit — it only degrades quality (#3453).
@@ -88,7 +88,7 @@ import { writeTurnGitTransaction } from "./uok/gitops.js";
88
88
  import { isClosedStatus } from "./status-guards.js";
89
89
  import { detectAbandonMilestone } from "./abandon-detect.js";
90
90
  import { getPendingGate } from "./bootstrap/write-gate.js";
91
- import { isDeterministicPolicyError } from "./auto-tool-tracking.js";
91
+ import { isDeterministicPolicyError, isToolUnavailableError } from "./auto-tool-tracking.js";
92
92
  import { formatConnectedStepStack, formatPostUnitStatusCard } from "./auto-status-message.js";
93
93
  import {
94
94
  clearProjectResearchInflightMarker,
@@ -2022,7 +2022,18 @@ export async function postUnitPreVerification(pctx: PostUnitContext, opts?: PreV
2022
2022
  "error",
2023
2023
  );
2024
2024
  } else if (!triggerArtifactVerified) {
2025
- if (s.lastToolInvocationError) {
2025
+ if (s.lastToolInvocationError && isToolUnavailableError(s.lastToolInvocationError)) {
2026
+ // Tool-unavailable is the one transient invocation error: the
2027
+ // workflow MCP server registers its surface asynchronously, so a
2028
+ // Unit's first call can race the registration. Fall through to the
2029
+ // bounded verification retry instead of pausing.
2030
+ debugLog("postUnit", { phase: "tool-unavailable-retry", unitType: s.currentUnit.type, unitId: s.currentUnit.id, error: s.lastToolInvocationError });
2031
+ ctx.ui.notify(
2032
+ `Tool unavailable for ${s.currentUnit.type}: ${s.lastToolInvocationError}. The tool surface may still be registering — retrying.`,
2033
+ "warning",
2034
+ );
2035
+ s.lastToolInvocationError = null;
2036
+ } else if (s.lastToolInvocationError) {
2026
2037
  const isUserSkip = /queued user message/i.test(s.lastToolInvocationError);
2027
2038
  const errMsg = isUserSkip
2028
2039
  ? `Tool skipped for ${s.currentUnit.type}: ${s.lastToolInvocationError}. Queued user message interrupted the turn — pausing auto-mode.`