@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
@@ -1,6 +1,8 @@
1
1
  // Project/App: gsd-pi
2
2
  // File Purpose: ADR-015 Recovery Classification module for runtime failure taxonomy.
3
+ import { isToolUnavailableError } from "./auto-tool-tracking.js";
3
4
  import { classifyError, isTransient } from "./error-classifier.js";
5
+ import { recoveryRemediation } from "./guidance.js";
4
6
  import { ReconciliationFailedError } from "./state-reconciliation.js";
5
7
  import { IllegalPhaseTransitionError } from "./state-transition-matrix.js";
6
8
  export function classifyFailure(input) {
@@ -13,103 +15,44 @@ export function classifyFailure(input) {
13
15
  : input.error instanceof IllegalPhaseTransitionError
14
16
  ? "illegal-transition"
15
17
  : input.failureKind ?? inferFailureKind(message);
16
- switch (failureKind) {
17
- case "tool-schema":
18
- return {
19
- failureKind,
20
- action: "stop",
21
- reason: `Tool schema failure${unitSuffix(input)}: ${message}`,
22
- exitReason: "tool-schema",
23
- remediation: "Fix the Unit Tool Contract or tool schema before retrying.",
24
- };
25
- case "tool-contract":
26
- return {
27
- failureKind,
28
- action: "stop",
29
- reason: `Tool Contract failure${unitSuffix(input)}: ${message}`,
30
- exitReason: "tool-contract",
31
- remediation: "Fix the Unit Tool Contract or prompt so the Unit is only asked to use tools owned by its phase.",
32
- };
33
- case "deterministic-policy":
34
- return {
35
- failureKind,
36
- action: "stop",
37
- reason: `Deterministic policy failure${unitSuffix(input)}: ${message}`,
38
- exitReason: "deterministic-policy",
39
- remediation: "Resolve the policy blocker; retrying the same Unit will repeat the failure.",
40
- };
41
- case "lifecycle-progression":
42
- return {
43
- failureKind,
44
- action: "stop",
45
- reason: `Lifecycle progression failure${unitSuffix(input)}: ${message}`,
46
- exitReason: "lifecycle-progression",
47
- remediation: "Route to the required owning Unit or restore the missing artifact before advancing lifecycle state.",
48
- };
49
- case "stale-worker":
50
- return {
51
- failureKind,
52
- action: "stop",
53
- reason: `Stale worker failure${unitSuffix(input)}: ${message}`,
54
- exitReason: "stale-worker",
55
- remediation: "Clear or reconcile the stale worker before dispatching another Unit.",
56
- };
57
- case "worktree-invalid":
58
- return {
59
- failureKind,
60
- action: "stop",
61
- reason: `Worktree invalid${unitSuffix(input)}: ${message}`,
62
- exitReason: "worktree-invalid",
63
- remediation: "Repair or recreate the milestone worktree before launching source-writing Units.",
64
- };
65
- case "verification-drift":
66
- return {
67
- failureKind,
68
- action: "escalate",
69
- reason: `Verification drift${unitSuffix(input)}: ${message}`,
70
- exitReason: "verification-drift",
71
- remediation: "Inspect the verification artifact and reconcile the state snapshot before resuming.",
72
- };
73
- case "reconciliation-drift":
74
- return {
75
- failureKind,
76
- action: "escalate",
77
- reason: `Reconciliation drift${unitSuffix(input)}: ${message}`,
78
- exitReason: "reconciliation-drift",
79
- remediation: "Inspect the persistent or repair-failed drift kinds reported by the State Reconciliation Module before resuming.",
80
- };
81
- case "illegal-transition":
82
- return {
83
- failureKind,
84
- action: "escalate",
85
- reason: `Illegal phase transition${unitSuffix(input)}: ${message}`,
86
- exitReason: "illegal-transition",
87
- remediation: "A derived Phase edge rejected by the Phase Transition Invariant survived reconciliation; inspect deriveState and the State Reconciliation Module before resuming.",
88
- };
89
- case "provider": {
90
- const providerClass = classifyError(message, input.retryAfterMs);
91
- return {
92
- failureKind,
93
- action: isTransient(providerClass) ? "retry" : "escalate",
94
- reason: message,
95
- exitReason: `provider-${providerClass.kind}`,
96
- remediation: isTransient(providerClass)
97
- ? "Retry after the provider/network condition clears."
98
- : "Inspect provider credentials, model entitlement, or request shape.",
99
- providerClass: providerClass.kind,
100
- };
101
- }
102
- case "runtime-unknown":
103
- return {
104
- failureKind,
105
- action: "escalate",
106
- reason: message,
107
- exitReason: "runtime-unknown",
108
- remediation: "Inspect the runtime error and add a dedicated classification if it is repeatable.",
109
- };
18
+ if (failureKind === "provider") {
19
+ const providerClass = classifyError(message, input.retryAfterMs);
20
+ const transient = isTransient(providerClass);
21
+ return {
22
+ failureKind,
23
+ action: transient ? "retry" : "escalate",
24
+ reason: message,
25
+ exitReason: `provider-${providerClass.kind}`,
26
+ remediation: recoveryRemediation(transient ? "provider-transient" : "provider-permanent"),
27
+ providerClass: providerClass.kind,
28
+ };
110
29
  }
30
+ const { action, label } = FAILURE_TAXONOMY[failureKind];
31
+ return {
32
+ failureKind,
33
+ action,
34
+ reason: label ? `${label}${unitSuffix(input)}: ${message}` : message,
35
+ exitReason: failureKind,
36
+ remediation: recoveryRemediation(failureKind),
37
+ };
111
38
  }
39
+ /** Per-kind action and reason label. Remediation lives in the Guidance module. */
40
+ const FAILURE_TAXONOMY = {
41
+ "tool-schema": { action: "stop", label: "Tool schema failure" },
42
+ "tool-contract": { action: "stop", label: "Tool Contract failure" },
43
+ "tool-unavailable": { action: "retry", label: "Tool unavailable" },
44
+ "deterministic-policy": { action: "stop", label: "Deterministic policy failure" },
45
+ "lifecycle-progression": { action: "stop", label: "Lifecycle progression failure" },
46
+ "stale-worker": { action: "stop", label: "Stale worker failure" },
47
+ "worktree-invalid": { action: "stop", label: "Worktree invalid" },
48
+ "verification-drift": { action: "escalate", label: "Verification drift" },
49
+ "reconciliation-drift": { action: "escalate", label: "Reconciliation drift" },
50
+ "illegal-transition": { action: "escalate", label: "Illegal phase transition" },
51
+ "runtime-unknown": { action: "escalate", label: null },
52
+ };
112
53
  function inferFailureKind(message) {
54
+ if (isToolUnavailableError(message))
55
+ return "tool-unavailable";
113
56
  if (/tool contract|auto-unit tool scope|phase-boundary gate|not permitted.*own/i.test(message))
114
57
  return "tool-contract";
115
58
  if (/lifecycle progression|required artifact|missing .*assessment|missing .*closeout|cannot legally (?:advance|progress)/i.test(message))
@@ -0,0 +1,108 @@
1
+ /**
2
+ * One-shot confirmation token for destructive bash commands.
3
+ *
4
+ * The destructive-command guard hard-blocks classified commands (force push,
5
+ * rm -rf, SQL drop, etc.) in all modes. The block instructs the model to
6
+ * confirm via ask_user_questions and re-issue the command. This module is the
7
+ * missing escape hatch: it records the user's confirmation and lets the exact
8
+ * confirmed command through exactly once.
9
+ *
10
+ * Design constraints:
11
+ * - In-memory only, never persisted. A confirmation token written to disk
12
+ * could silently auto-approve a destructive command in a later session —
13
+ * confirmation must be re-obtained every process lifetime.
14
+ * - One-shot. Consuming a token clears it, so a second destructive command
15
+ * (even an identical one) re-blocks and re-prompts.
16
+ * - Command-bound. The token only matches the exact (normalized) command
17
+ * string the user confirmed. A reworded command re-blocks, which is safe.
18
+ * - Per basePath, so concurrent workspaces in one process never share tokens.
19
+ *
20
+ * Copyright (c) 2026 Jeremy McSpadden <jeremy@fluxlabs.net>
21
+ */
22
+ import { resolve } from "node:path";
23
+ /**
24
+ * Question-id substring that marks an ask_user_questions call as a
25
+ * destructive-command confirmation. The tool_result handler promotes the
26
+ * pending command to a confirmed token when an affirmative answer arrives for
27
+ * a question whose id contains this marker.
28
+ */
29
+ export const DESTRUCTIVE_CONFIRM_GATE_MARKER = "destructive_confirm";
30
+ const statesByBasePath = new Map();
31
+ function stateKey(basePath) {
32
+ return resolve(basePath);
33
+ }
34
+ function getState(basePath) {
35
+ const key = stateKey(basePath);
36
+ let state = statesByBasePath.get(key);
37
+ if (!state) {
38
+ state = { pendingCommand: null, confirmedCommand: null };
39
+ statesByBasePath.set(key, state);
40
+ }
41
+ return state;
42
+ }
43
+ /**
44
+ * Normalize a command for stable matching across block → confirm → retry.
45
+ * Trims surrounding whitespace and collapses internal runs of whitespace so
46
+ * cosmetic reformatting of the same command still matches the token.
47
+ */
48
+ export function normalizeDestructiveCommand(command) {
49
+ return command.replace(/\s+/g, " ").trim();
50
+ }
51
+ /**
52
+ * Whether an ask_user_questions question id is a destructive-confirm gate.
53
+ */
54
+ export function isDestructiveConfirmGateId(questionId) {
55
+ return typeof questionId === "string" && questionId.includes(DESTRUCTIVE_CONFIRM_GATE_MARKER);
56
+ }
57
+ /**
58
+ * Record that a destructive command was blocked and is awaiting confirmation.
59
+ * Called by the guard at block time. Overwrites any prior pending command —
60
+ * only the most recently blocked command can be confirmed.
61
+ */
62
+ export function requestDestructiveConfirmation(command, basePath = process.cwd()) {
63
+ const state = getState(basePath);
64
+ state.pendingCommand = normalizeDestructiveCommand(command);
65
+ // A fresh request invalidates any stale confirmed token for a different
66
+ // command so confirmation cannot leak across distinct destructive actions.
67
+ state.confirmedCommand = null;
68
+ }
69
+ /**
70
+ * Promote the pending command to a confirmed, one-shot token. Called by the
71
+ * tool_result handler when the user gives an affirmative answer to a
72
+ * destructive-confirm gate. Returns the confirmed command, or null if there
73
+ * was nothing pending (e.g. confirmation arrived without a preceding block).
74
+ */
75
+ export function confirmDestructiveCommand(basePath = process.cwd()) {
76
+ const state = getState(basePath);
77
+ if (!state.pendingCommand)
78
+ return null;
79
+ state.confirmedCommand = state.pendingCommand;
80
+ state.pendingCommand = null;
81
+ return state.confirmedCommand;
82
+ }
83
+ /**
84
+ * Check whether the given command has been confirmed, consuming the token if
85
+ * so. Returns true exactly once per confirmation; subsequent calls (or a
86
+ * non-matching command) return false. Called by the guard before blocking.
87
+ */
88
+ export function consumeDestructiveConfirmation(command, basePath = process.cwd()) {
89
+ const state = getState(basePath);
90
+ if (!state.confirmedCommand)
91
+ return false;
92
+ if (state.confirmedCommand !== normalizeDestructiveCommand(command))
93
+ return false;
94
+ state.confirmedCommand = null;
95
+ return true;
96
+ }
97
+ /**
98
+ * Inspect the pending command without consuming it (diagnostics/tests).
99
+ */
100
+ export function peekPendingDestructiveCommand(basePath = process.cwd()) {
101
+ return getState(basePath).pendingCommand;
102
+ }
103
+ /**
104
+ * Clear all destructive-confirmation state for a basePath (tests / flow reset).
105
+ */
106
+ export function resetDestructiveConfirmation(basePath = process.cwd()) {
107
+ statesByBasePath.delete(stateKey(basePath));
108
+ }
@@ -22,26 +22,7 @@ import { isDbAvailable, getAllMilestones, getMilestone, getMilestoneSlices, getS
22
22
  import { wasWorkflowDatabaseOpenAttempted } from './db-workspace.js';
23
23
  import { formatCompletePhaseNextAction, countUnmappedActiveRequirements } from './requirements-backlog.js';
24
24
  import { classifyMilestoneReadiness, readinessNeedsDiscussion, } from './milestone-readiness.js';
25
- function formatNeedsAttentionBlocker(milestoneId) {
26
- return [
27
- `Milestone ${milestoneId} is blocked because milestone validation returned needs-attention.`,
28
- `Fix options:`,
29
- `1. Review the validation details: \`/gsd status\``,
30
- `2. If you fixed the missing evidence or issue, re-run milestone validation: \`/gsd validate-milestone\``,
31
- `3. If the finding is acceptable, override it: \`/gsd verdict pass --rationale "why this is okay"\``,
32
- `4. If this should wait, defer it explicitly: \`/gsd park ${milestoneId}\``,
33
- `After validation or override passes, run \`/gsd auto\` to complete and merge the milestone.`,
34
- ].join("\n");
35
- }
36
- function formatNeedsRemediationBlocker(milestoneId) {
37
- return [
38
- `Milestone ${milestoneId} is blocked because milestone validation returned needs-remediation, but all slices are complete.`,
39
- `Fix options:`,
40
- `1. Run \`/gsd dispatch reassess\` to add remediation slices, then run \`/gsd auto\``,
41
- `2. If the finding is acceptable, override it: \`/gsd verdict pass --rationale "why this is okay"\``,
42
- `3. If this should wait, defer it explicitly: \`/gsd park ${milestoneId}\``,
43
- ].join("\n");
44
- }
25
+ import { needsAttentionBlockerGuidance as formatNeedsAttentionBlocker, needsRemediationBlockerGuidance as formatNeedsRemediationBlocker, } from './guidance.js';
45
26
  /**
46
27
  * A "ghost" milestone directory contains only META.json (and no substantive
47
28
  * files like CONTEXT, CONTEXT-DRAFT, ROADMAP, or SUMMARY). These appear when
@@ -0,0 +1,57 @@
1
+ // Project/App: gsd-pi
2
+ // File Purpose: Stop Notice module — single owner of the auto/step-mode
3
+ // stop/pause notice vocabulary. Both sides of the wire live here: the
4
+ // formatters that produce the canonical prefixes (used by stopAuto/pauseAuto)
5
+ // and the classifiers that recognize them (used by the headless host to pick
6
+ // exit codes). Wording changes in this file keep emitter and detector in
7
+ // lockstep; round-trip tests enforce it.
8
+ /** A reason string of the form "Blocked: …" marks a blocked stop. */
9
+ export function isBlockedStopReason(reason) {
10
+ return /^Blocked:\s*/i.test(reason ?? "");
11
+ }
12
+ /** Strip the "Blocked: " marker for display. */
13
+ export function stopNoticeDisplayReason(reason) {
14
+ return (reason ?? "").replace(/^Blocked:\s*/i, "").trim();
15
+ }
16
+ export function stopNoticeKind(reason) {
17
+ return isBlockedStopReason(reason) ? "blocked" : "stopped";
18
+ }
19
+ /** Canonical stop-notice prefix: "Auto-mode blocked — reason" / "Auto-mode stopped". */
20
+ export function formatStopNoticePrefix(reason) {
21
+ const displayReason = stopNoticeDisplayReason(reason);
22
+ const prefix = stopNoticeKind(reason) === "blocked" ? "Auto-mode blocked" : "Auto-mode stopped";
23
+ return displayReason ? `${prefix} — ${displayReason}` : prefix;
24
+ }
25
+ // ─── Classification (headless host side) ────────────────────────────────
26
+ // The canonical lowercase prefixes the headless event loop recognizes in
27
+ // notify messages. Emitters above and ad-hoc emitters elsewhere must start
28
+ // their terminal notices with one of these.
29
+ export const PAUSED_NOTICE_PREFIXES = ["auto-mode paused", "step-mode paused"];
30
+ export const TERMINAL_NOTICE_PREFIXES = [
31
+ "auto-mode stopped",
32
+ "step-mode stopped",
33
+ "auto-mode complete",
34
+ "no active milestone",
35
+ "auto-mode idle",
36
+ ];
37
+ /** Manual-resolution notices emitted before auto-mode can formally pause/stop. */
38
+ export function isManualResolutionNotice(message) {
39
+ return (message.includes("resolve manually and re-run /gsd auto") ||
40
+ message.includes("resolve conflicts manually and run /gsd auto to resume") ||
41
+ message.includes("resolve and run /gsd auto to resume"));
42
+ }
43
+ export function isPauseNotice(message) {
44
+ return PAUSED_NOTICE_PREFIXES.some((prefix) => message.startsWith(prefix));
45
+ }
46
+ export function isTerminalNotice(message) {
47
+ return TERMINAL_NOTICE_PREFIXES.some((prefix) => message.startsWith(prefix));
48
+ }
49
+ /** Pauses that do not require operator intervention in headless mode. */
50
+ export function isNonBlockingPauseNotice(message) {
51
+ return message.includes("idempotent advance: unit already active");
52
+ }
53
+ export function isBlockedNoticeMessage(message) {
54
+ return (message.includes("blocked:") ||
55
+ (isPauseNotice(message) && !isNonBlockingPauseNotice(message)) ||
56
+ isManualResolutionNotice(message));
57
+ }
@@ -0,0 +1,56 @@
1
+ // Project/App: gsd-pi
2
+ // File Purpose: Tool Contract module's runtime face — verify the live SDK tool surface covers a Unit's required workflow tools.
3
+ import { mcpToolMatchesBaseName } from "./mcp-tool-name.js";
4
+ import { getRequiredWorkflowToolsForUnit } from "./unit-tool-contracts.js";
5
+ import { isWorkflowToolSurfaceName } from "./workflow-tool-surface.js";
6
+ /**
7
+ * Stable phrase recognized as transient by auto-tool-tracking's
8
+ * isToolUnavailableError and error-classifier's transient buckets,
9
+ * which build their matchers from this constant.
10
+ */
11
+ export const TOOL_SURFACE_NOT_READY = "workflow tool surface not ready";
12
+ /** MCP server statuses that will not self-heal within the session. */
13
+ const TERMINAL_MCP_SERVER_STATUSES = new Set(["failed", "needs-auth", "disabled"]);
14
+ /**
15
+ * Verify the live tool surface observed at SDK session init covers the Unit's
16
+ * required workflow tools. Complements the static pre-dispatch gate
17
+ * (getWorkflowTransportSupportError), which only proves the MCP launch config
18
+ * is discoverable — the workflow server connects asynchronously after session
19
+ * start, so the static gate cannot see whether the tools actually registered.
20
+ *
21
+ * Returns a transient, recovery-classifiable error (kind tool-unavailable →
22
+ * retry) when the workflow server failed or has not yet registered a required
23
+ * tool, so dispatch aborts before the first model turn instead of letting the
24
+ * Unit improvise around "No such tool available". Returns null when no
25
+ * workflow server is part of this session (native tool path), when the Unit
26
+ * requires no workflow tools, or when the surface is ready.
27
+ */
28
+ export function getToolSurfaceReadinessError(input) {
29
+ const { unitType, workflowServerName, observation } = input;
30
+ if (!unitType || !workflowServerName)
31
+ return null;
32
+ const required = getRequiredWorkflowToolsForUnit(unitType).filter(isWorkflowToolSurfaceName);
33
+ if (required.length === 0)
34
+ return null;
35
+ const server = observation.mcpServers.find((entry) => entry.name === workflowServerName);
36
+ if (!server) {
37
+ return `${TOOL_SURFACE_NOT_READY} for ${unitType}: MCP server "${workflowServerName}" is absent from the init surface (not yet connected): ${required.join(", ")}`;
38
+ }
39
+ // The SDK does not wait for MCP servers before init — a still-connecting
40
+ // server reports "pending" there routinely, then registers within seconds,
41
+ // usually well before the Unit's first workflow tool call. Aborting on
42
+ // "pending" would fail the common healthy session, so it passes through;
43
+ // a genuine miss after pass-through still surfaces in-session as
44
+ // "No such tool available" and classifies tool-unavailable → bounded retry.
45
+ // Only statuses that cannot self-heal abort here.
46
+ if (server.status !== "connected" && !TERMINAL_MCP_SERVER_STATUSES.has(server.status)) {
47
+ return null;
48
+ }
49
+ const missing = required.filter((tool) => !observation.tools.some((name) => name === tool || mcpToolMatchesBaseName(name, tool)));
50
+ if (missing.length === 0)
51
+ return null;
52
+ const serverDetail = server.status === "connected"
53
+ ? `MCP server "${workflowServerName}" is connected but has not registered`
54
+ : `MCP server "${workflowServerName}" status is "${server.status}" and it has not registered`;
55
+ return `${TOOL_SURFACE_NOT_READY} for ${unitType}: ${serverDetail}: ${missing.join(", ")}`;
56
+ }
@@ -4,7 +4,7 @@ import { EXEC_DEFAULTS, runExecSandbox, } from "../exec-sandbox.js";
4
4
  import { realpathSync } from "node:fs";
5
5
  import path from "node:path";
6
6
  import { isContextModeEnabled } from "../preferences-types.js";
7
- import { findWorktreeSegment } from "../worktree-root.js";
7
+ import { projectRootFromWorktreePath } from "../worktree-root.js";
8
8
  import { contextModeDisabledResult } from "./context-mode-tool-result.js";
9
9
  const UAT_EXEC_INTENTS = [
10
10
  "uat-artifact-check",
@@ -142,13 +142,10 @@ function normalizeScanPath(value) {
142
142
  }
143
143
  function parseWorktreeBase(baseDir) {
144
144
  const normalizedBase = normalizeScanPath(baseDir);
145
- const segment = findWorktreeSegment(normalizedBase);
146
- if (!segment || segment.gsdIdx <= 0)
145
+ const originalRoot = projectRootFromWorktreePath(normalizedBase);
146
+ if (!originalRoot)
147
147
  return null;
148
- return {
149
- originalRoot: normalizedBase.slice(0, segment.gsdIdx),
150
- worktreeRoot: normalizedBase,
151
- };
148
+ return { originalRoot, worktreeRoot: normalizedBase };
152
149
  }
153
150
  function pathInside(parent, target) {
154
151
  const parentWithSep = parent.endsWith("/") ? parent : `${parent}/`;
@@ -345,6 +342,7 @@ function formatResult(result) {
345
342
  exit_code: result.exit_code,
346
343
  signal: result.signal,
347
344
  timed_out: result.timed_out,
345
+ force_resolved: result.force_resolved,
348
346
  duration_ms: result.duration_ms,
349
347
  stdout_bytes: result.stdout_bytes,
350
348
  stderr_bytes: result.stderr_bytes,
@@ -358,6 +356,10 @@ function formatResult(result) {
358
356
  };
359
357
  }
360
358
  function formatExit(result) {
359
+ // force_resolved means a non-closing (D-state) child was force-resolved past its
360
+ // hard deadline rather than observed exiting; distinguish it from a clean timeout.
361
+ if (result.force_resolved)
362
+ return "timeout(force-killed)";
361
363
  if (result.timed_out)
362
364
  return "timeout";
363
365
  if (result.signal)
@@ -12,7 +12,8 @@ import { writeManifest } from "../workflow-manifest.js";
12
12
  import { appendEvent } from "../workflow-events.js";
13
13
  import { logWarning } from "../workflow-logger.js";
14
14
  import { validatePathOnlyPlanningFields, validatePlanningPathScope } from "../planning-path-scope.js";
15
- import { checkFilePathConsistency, checkTaskOrdering } from "../pre-execution-checks.js";
15
+ import { runTaskPathChecks } from "../pre-execution-checks.js";
16
+ import { resolveWorktreeProjectRoot } from "../worktree-root.js";
16
17
  import { buildTaskFileName, gsdProjectionRoot } from "../paths.js";
17
18
  import { loadEffectiveGSDPreferences } from "../preferences.js";
18
19
  import { createRepositoryRegistryFromPreferences, defaultRepositoryTargets } from "../repository-registry.js";
@@ -193,11 +194,16 @@ function validateTaskPathsBeforePersist(params, basePath, defaultTargets, allowe
193
194
  const additionalRoots = allowedRoots
194
195
  .map((root) => resolve(root))
195
196
  .filter((root) => root !== baseRoot);
196
- const context = additionalRoots.length > 0 ? { additionalRoots } : undefined;
197
- const checks = [
198
- ...checkFilePathConsistency(taskRows, basePath, context),
199
- ...checkTaskOrdering(taskRows, basePath, context),
200
- ];
197
+ const resolvedCanonicalRoot = resolve(resolveWorktreeProjectRoot(basePath));
198
+ const canonicalProjectRoot = resolvedCanonicalRoot !== baseRoot ? resolvedCanonicalRoot : undefined;
199
+ const hasContext = additionalRoots.length > 0 || canonicalProjectRoot !== undefined;
200
+ const context = hasContext
201
+ ? {
202
+ ...(additionalRoots.length > 0 ? { additionalRoots } : {}),
203
+ ...(canonicalProjectRoot !== undefined ? { canonicalProjectRoot } : {}),
204
+ }
205
+ : undefined;
206
+ const checks = runTaskPathChecks(taskRows, basePath, context);
201
207
  const blocking = checks.filter((check) => !check.passed && check.blocking);
202
208
  if (blocking.length === 0)
203
209
  return null;
@@ -1,5 +1,6 @@
1
1
  // Project/App: gsd-pi
2
2
  // File Purpose: Central UAT mode policy for dispatch, tool presentation, and result validation.
3
+ import { hasBrowserContractPrefix } from "../shared/browser-contract.js";
3
4
  import { extractUatType } from "./files.js";
4
5
  import { hasBrowserRequiredText } from "./browser-evidence.js";
5
6
  import { parseMcpToolName } from "./mcp-tool-name.js";
@@ -84,7 +85,7 @@ export function uatTypeIncludesBrowser(uatType) {
84
85
  export function isUatBrowserToolName(toolName) {
85
86
  const parsed = parseMcpToolName(toolName);
86
87
  const canonicalName = parsed?.toolName ?? toolName;
87
- if (canonicalName.startsWith("browser_"))
88
+ if (hasBrowserContractPrefix(canonicalName))
88
89
  return true;
89
90
  return parsed?.toolName === "*" && parsed.serverName.toLowerCase().includes("browser");
90
91
  }
@@ -0,0 +1,138 @@
1
+ // Project/App: gsd-pi
2
+ // File Purpose: Unit Closeout module — durable completion pipeline behind one seam (ADR-032).
3
+ //
4
+ // `closeUnit` owns what makes a Unit's completion durable. This pass ships the
5
+ // **Interactive Closeout adapter** path: commit the work and compute the
6
+ // Closeout Git Verdict, failing closed (loudly) instead of silently when a
7
+ // non-`none` `git.isolation` preference was never honoured by the session.
8
+ // Motivating failure (2026-06-10): an interactive session under
9
+ // `git.isolation: worktree` completed a milestone with every source file
10
+ // untracked on the integration branch — no commit, no merge, no warning.
11
+ //
12
+ // The interactive adapter only fires on milestone boundaries — the durability
13
+ // gap is the milestone close, and committing at every task/slice would sweep a
14
+ // developer's unrelated working-tree changes. `closeUnit` itself stays general
15
+ // over all boundaries for the pending Auto Closeout adapter re-seat.
16
+ //
17
+ // The auto loop keeps its existing closeout pipeline for now (the Auto
18
+ // Closeout adapter re-seat is the documented next step in ADR-032); the
19
+ // interactive trigger in bootstrap/register-hooks.ts is a no-op while
20
+ // `isAutoActive()`, so auto-mode behaviour is untouched.
21
+ //
22
+ // Re-entrancy is naturally safe: a re-fired completion commits an already-clean
23
+ // tree, which yields `nothing-to-commit`, and `appendNotification` carries its
24
+ // own dedup window — so `closeUnit` keeps no result cache of its own.
25
+ import { appendNotification } from "./notification-store.js";
26
+ import { readStringField } from "./auto-unit-tool-scope.js";
27
+ import { MILESTONE_BRANCH_PREFIX } from "./branch-patterns.js";
28
+ import { nativeGetCurrentBranch } from "./native-git-bridge.js";
29
+ import { getIsolationMode } from "./preferences.js";
30
+ import { autoCommitCurrentBranch } from "./worktree.js";
31
+ import { logWarning } from "./workflow-logger.js";
32
+ const defaultDeps = {
33
+ isolationMode: (basePath) => getIsolationMode(basePath),
34
+ currentBranch: (basePath) => {
35
+ try {
36
+ return nativeGetCurrentBranch(basePath);
37
+ }
38
+ catch {
39
+ return null;
40
+ }
41
+ },
42
+ commit: (basePath, unitType, unitId) => autoCommitCurrentBranch(basePath, unitType, unitId),
43
+ notify: (message, severity) => appendNotification(message, severity),
44
+ };
45
+ export function closeUnit(request, deps = defaultDeps) {
46
+ let commitMessage = null;
47
+ let gitVerdict;
48
+ let notice;
49
+ try {
50
+ commitMessage = deps.commit(request.basePath, request.unitType, request.unitId);
51
+ gitVerdict = commitMessage === null ? "nothing-to-commit" : "committed";
52
+ }
53
+ catch (err) {
54
+ gitVerdict = "commit-failed";
55
+ notice = `Unit closeout commit failed for ${request.unitId}: ${err instanceof Error ? err.message : String(err)}`;
56
+ logWarning("engine", notice);
57
+ deps.notify(notice, "error");
58
+ }
59
+ if (request.boundary === "milestone" && gitVerdict !== "commit-failed") {
60
+ const isolation = deps.isolationMode(request.basePath);
61
+ if (isolation !== "none") {
62
+ const branch = deps.currentBranch(request.basePath);
63
+ if (branch?.startsWith(MILESTONE_BRANCH_PREFIX)) {
64
+ gitVerdict = "milestone-branch";
65
+ notice =
66
+ `Milestone ${request.unitId} completed on ${branch}. ` +
67
+ `Merge it to the integration branch with the worktree tooling (/gsd worktree merge).`;
68
+ deps.notify(notice, "info");
69
+ }
70
+ else {
71
+ gitVerdict = "isolation-bypassed";
72
+ notice =
73
+ `Needs attention: milestone ${request.unitId} completed outside a milestone worktree/branch ` +
74
+ `while git.isolation is "${isolation}" — the isolation preference was not honoured this session. ` +
75
+ (commitMessage
76
+ ? `Work was committed directly on "${branch ?? "the current branch"}".`
77
+ : `The working tree had nothing left to commit on "${branch ?? "the current branch"}".`);
78
+ logWarning("engine", notice);
79
+ deps.notify(notice, "warning");
80
+ }
81
+ }
82
+ }
83
+ return { gitVerdict, commitMessage, notice };
84
+ }
85
+ // ─── Interactive Closeout adapter (tool-observation trigger) ─────────────
86
+ // Canonical closeout tool → boundary. Aliases are canonicalized by the hook.
87
+ // Only the milestone boundary is wired interactively: it is the durability gap
88
+ // that motivated ADR-032, and committing at every task/slice would sweep a
89
+ // developer's unrelated working-tree changes. `closeUnit` still handles every
90
+ // boundary for the pending Auto Closeout adapter re-seat.
91
+ const CLOSEOUT_TOOL_BOUNDARIES = {
92
+ gsd_complete_milestone: "milestone",
93
+ };
94
+ // Commit attribution uses the canonical unit types so interactive closeout
95
+ // commits carry the same GSD-Unit evidence trailers the verification prompts
96
+ // look for.
97
+ const BOUNDARY_UNIT_TYPES = {
98
+ task: "execute-task",
99
+ slice: "complete-slice",
100
+ milestone: "complete-milestone",
101
+ };
102
+ export function isUnitCloseoutTool(canonicalToolName) {
103
+ return canonicalToolName in CLOSEOUT_TOOL_BOUNDARIES;
104
+ }
105
+ function readId(input, camel, snake) {
106
+ const value = readStringField(input, camel, snake);
107
+ return value && value.length > 0 ? value : undefined;
108
+ }
109
+ /**
110
+ * Interactive Closeout adapter entry, called from the host's tool-observation
111
+ * hook for successful closeout tool calls when auto-mode is NOT active.
112
+ * Returns null when the tool input doesn't identify a unit.
113
+ */
114
+ export function runInteractiveUnitCloseout(args, deps = defaultDeps) {
115
+ const boundary = CLOSEOUT_TOOL_BOUNDARIES[args.canonicalToolName];
116
+ if (!boundary)
117
+ return null;
118
+ const milestoneId = readId(args.input, "milestoneId", "milestone_id");
119
+ if (!milestoneId)
120
+ return null;
121
+ const sliceId = readId(args.input, "sliceId", "slice_id");
122
+ const taskId = readId(args.input, "taskId", "task_id");
123
+ let unitId = milestoneId;
124
+ if (boundary !== "milestone") {
125
+ if (!sliceId)
126
+ return null;
127
+ unitId = boundary === "slice" ? `${milestoneId}/${sliceId}` : `${milestoneId}/${sliceId}/${taskId ?? ""}`;
128
+ if (boundary === "task" && !taskId)
129
+ return null;
130
+ }
131
+ return closeUnit({
132
+ basePath: args.basePath,
133
+ unitType: BOUNDARY_UNIT_TYPES[boundary],
134
+ unitId,
135
+ boundary,
136
+ outcome: "complete",
137
+ }, deps);
138
+ }