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

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (424) hide show
  1. package/dist/cli-style.d.ts +17 -0
  2. package/dist/cli-style.js +28 -0
  3. package/dist/cli.js +1 -1
  4. package/dist/headless-events.d.ts +4 -2
  5. package/dist/headless-events.js +14 -34
  6. package/dist/models-resolver.d.ts +3 -13
  7. package/dist/models-resolver.js +3 -22
  8. package/dist/resource-loader.js +2 -14
  9. package/dist/resources/.managed-resources-content-hash +1 -1
  10. package/dist/resources/GSD-WORKFLOW.md +5 -4
  11. package/dist/resources/extensions/async-jobs/async-bash-tool.js +30 -64
  12. package/dist/resources/extensions/async-jobs/await-tool.js +80 -12
  13. package/dist/resources/extensions/async-jobs/index.js +65 -0
  14. package/dist/resources/extensions/async-jobs/job-manager.js +12 -1
  15. package/dist/resources/extensions/bg-shell/bg-shell-command.js +6 -6
  16. package/dist/resources/extensions/bg-shell/bg-shell-tool.js +10 -7
  17. package/dist/resources/extensions/bg-shell/overlay.js +9 -6
  18. package/dist/resources/extensions/bg-shell/process-manager.js +54 -25
  19. package/dist/resources/extensions/bg-shell/readiness-detector.js +11 -0
  20. package/dist/resources/extensions/bg-shell/utilities.js +3 -0
  21. package/dist/resources/extensions/browser-tools/engine/managed-gsd-browser.js +209 -88
  22. package/dist/resources/extensions/browser-tools/engine/selection.js +73 -5
  23. package/dist/resources/extensions/browser-tools/index.js +69 -12
  24. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +30 -4
  25. package/dist/resources/extensions/gsd/auto/custom-verify-retry-store.js +17 -2
  26. package/dist/resources/extensions/gsd/auto/detect-stuck.js +33 -13
  27. package/dist/resources/extensions/gsd/auto/dispatch-history.js +105 -0
  28. package/dist/resources/extensions/gsd/auto/dispatch-key.js +37 -0
  29. package/dist/resources/extensions/gsd/auto/loop.js +4 -1
  30. package/dist/resources/extensions/gsd/auto/orchestrator.js +61 -44
  31. package/dist/resources/extensions/gsd/auto/phases.js +2 -2
  32. package/dist/resources/extensions/gsd/auto-direct-dispatch.js +8 -32
  33. package/dist/resources/extensions/gsd/auto-dispatch.js +40 -57
  34. package/dist/resources/extensions/gsd/auto-model-selection.js +25 -6
  35. package/dist/resources/extensions/gsd/auto-post-unit.js +23 -8
  36. package/dist/resources/extensions/gsd/auto-prompts.js +81 -19
  37. package/dist/resources/extensions/gsd/auto-start.js +18 -15
  38. package/dist/resources/extensions/gsd/auto-tool-tracking.js +18 -0
  39. package/dist/resources/extensions/gsd/auto-unit-tool-scope.js +12 -20
  40. package/dist/resources/extensions/gsd/auto-verification.js +9 -28
  41. package/dist/resources/extensions/gsd/auto-worktree.js +30 -90
  42. package/dist/resources/extensions/gsd/auto.js +4 -13
  43. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +3 -2
  44. package/dist/resources/extensions/gsd/bootstrap/dynamic-tools.js +23 -6
  45. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +19 -0
  46. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +212 -48
  47. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +303 -77
  48. package/dist/resources/extensions/gsd/branch-patterns.js +2 -0
  49. package/dist/resources/extensions/gsd/browser-daemon-auto-prep.js +83 -0
  50. package/dist/resources/extensions/gsd/browser-evidence.js +8 -2
  51. package/dist/resources/extensions/gsd/captures.js +4 -6
  52. package/dist/resources/extensions/gsd/consent-question.js +337 -0
  53. package/dist/resources/extensions/gsd/consent-verdict.js +63 -0
  54. package/dist/resources/extensions/gsd/constants.js +0 -2
  55. package/dist/resources/extensions/gsd/crash-recovery.js +4 -12
  56. package/dist/resources/extensions/gsd/db/queries.js +26 -0
  57. package/dist/resources/extensions/gsd/dispatch-guard.js +10 -35
  58. package/dist/resources/extensions/gsd/doctor-environment.js +2 -6
  59. package/dist/resources/extensions/gsd/doctor-format.js +9 -6
  60. package/dist/resources/extensions/gsd/doctor-runtime-checks.js +13 -15
  61. package/dist/resources/extensions/gsd/engine-hook-contract.js +70 -0
  62. package/dist/resources/extensions/gsd/error-classifier.js +9 -0
  63. package/dist/resources/extensions/gsd/exec-sandbox.js +30 -10
  64. package/dist/resources/extensions/gsd/files.js +33 -19
  65. package/dist/resources/extensions/gsd/guidance.js +158 -0
  66. package/dist/resources/extensions/gsd/guided-flow.js +17 -2
  67. package/dist/resources/extensions/gsd/markdown-renderer.js +10 -0
  68. package/dist/resources/extensions/gsd/mcp-filter.js +2 -19
  69. package/dist/resources/extensions/gsd/mcp-tool-name.js +5 -13
  70. package/dist/resources/extensions/gsd/memory-consolidation-scanner.js +1 -1
  71. package/dist/resources/extensions/gsd/migrate/safety.js +4 -1
  72. package/dist/resources/extensions/gsd/milestone-closeout.js +13 -23
  73. package/dist/resources/extensions/gsd/notification-store.js +11 -4
  74. package/dist/resources/extensions/gsd/parallel-monitor-overlay.js +6 -4
  75. package/dist/resources/extensions/gsd/parsers-legacy.js +16 -4
  76. package/dist/resources/extensions/gsd/paths.js +27 -0
  77. package/dist/resources/extensions/gsd/pre-execution-checks.js +91 -3
  78. package/dist/resources/extensions/gsd/preferences-models.js +14 -48
  79. package/dist/resources/extensions/gsd/prompts/complete-slice.md +2 -2
  80. package/dist/resources/extensions/gsd/prompts/plan-slice.md +1 -1
  81. package/dist/resources/extensions/gsd/prompts/refine-slice.md +1 -1
  82. package/dist/resources/extensions/gsd/prompts/replan-slice.md +1 -1
  83. package/dist/resources/extensions/gsd/prompts/run-uat.md +6 -4
  84. package/dist/resources/extensions/gsd/prompts/system.md +5 -2
  85. package/dist/resources/extensions/gsd/provider-error-guidance.js +1 -5
  86. package/dist/resources/extensions/gsd/provider-switch-observer.js +1 -1
  87. package/dist/resources/extensions/gsd/publication.js +87 -0
  88. package/dist/resources/extensions/gsd/reactive-graph.js +8 -1
  89. package/dist/resources/extensions/gsd/recovery-classification.js +37 -94
  90. package/dist/resources/extensions/gsd/safety/destructive-confirmation.js +108 -0
  91. package/dist/resources/extensions/gsd/state.js +6 -20
  92. package/dist/resources/extensions/gsd/stop-notice.js +57 -0
  93. package/dist/resources/extensions/gsd/tool-presentation-plan.js +4 -4
  94. package/dist/resources/extensions/gsd/tool-surface-readiness.js +56 -0
  95. package/dist/resources/extensions/gsd/tools/complete-slice.js +20 -10
  96. package/dist/resources/extensions/gsd/tools/exec-tool.js +9 -7
  97. package/dist/resources/extensions/gsd/tools/plan-slice.js +12 -6
  98. package/dist/resources/extensions/gsd/uat-policy.js +42 -16
  99. package/dist/resources/extensions/gsd/unit-closeout.js +138 -0
  100. package/dist/resources/extensions/gsd/unit-context-composer.js +74 -1
  101. package/dist/resources/extensions/gsd/unit-context-manifest.js +4 -27
  102. package/dist/resources/extensions/gsd/unit-registry.js +337 -0
  103. package/dist/resources/extensions/gsd/unit-tool-contracts.js +9 -182
  104. package/dist/resources/extensions/gsd/verdict-parser.js +1 -1
  105. package/dist/resources/extensions/gsd/web-app-uat.js +45 -8
  106. package/dist/resources/extensions/gsd/workflow-tool-surface.js +1 -1
  107. package/dist/resources/extensions/gsd/worktree-git-recovery.js +15 -9
  108. package/dist/resources/extensions/gsd/worktree-lifecycle.js +3 -2
  109. package/dist/resources/extensions/gsd/worktree-root.js +11 -0
  110. package/dist/resources/extensions/gsd/worktree-session-state.js +4 -5
  111. package/dist/resources/extensions/search-the-web/native-search.js +5 -3
  112. package/dist/resources/extensions/shared/browser-contract.js +59 -0
  113. package/dist/resources/extensions/shared/gsd-browser-cli.js +96 -5
  114. package/dist/resources/shared/package.json +3 -0
  115. package/dist/resources/skills/create-skill/references/executable-code.md +1 -1
  116. package/dist/resources/skills/create-skill/workflows/add-reference.md +8 -3
  117. package/dist/resources/skills/create-skill/workflows/add-script.md +4 -2
  118. package/dist/resources/skills/create-skill/workflows/add-template.md +3 -1
  119. package/dist/resources/skills/create-skill/workflows/add-workflow.md +8 -3
  120. package/dist/resources/skills/create-skill/workflows/upgrade-to-router.md +10 -5
  121. package/dist/resources/skills/create-skill/workflows/verify-skill.md +9 -4
  122. package/dist/resources/skills/spike-wrap-up/SKILL.md +9 -9
  123. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  124. package/dist/web/standalone/.next/BUILD_ID +1 -1
  125. package/dist/web/standalone/.next/app-path-routes-manifest.json +12 -12
  126. package/dist/web/standalone/.next/build-manifest.json +3 -3
  127. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  128. package/dist/web/standalone/.next/react-loadable-manifest.json +1 -1
  129. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  130. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  131. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  132. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  133. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  134. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  135. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  136. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  137. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  138. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  139. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  140. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  141. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  142. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  143. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  144. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  145. package/dist/web/standalone/.next/server/app/api/update/route.js +1 -1
  146. package/dist/web/standalone/.next/server/app/api/update/route.js.nft.json +1 -1
  147. package/dist/web/standalone/.next/server/app/index.html +1 -1
  148. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  149. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  150. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  151. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  152. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  153. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  154. package/dist/web/standalone/.next/server/app-paths-manifest.json +12 -12
  155. package/dist/web/standalone/.next/server/chunks/5124.js +1 -1
  156. package/dist/web/standalone/.next/server/chunks/8357.js +1 -1
  157. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  158. package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
  159. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  160. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  161. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  162. package/dist/web/standalone/.next/static/chunks/{796.cf859a427a2cb2ac.js → 796.e0bdc932325d7e03.js} +1 -1
  163. package/dist/web/standalone/.next/static/chunks/{webpack-fbea77b5f9953368.js → webpack-f0285ce91d4ec9ef.js} +1 -1
  164. package/dist/web/standalone/package.json +1 -1
  165. package/dist/worktree-cli.js +3 -6
  166. package/dist/worktree-status-banner.js +7 -15
  167. package/package.json +1 -1
  168. package/packages/cloud-mcp-gateway/package.json +2 -2
  169. package/packages/contracts/dist/rpc.d.ts +1 -0
  170. package/packages/contracts/dist/rpc.d.ts.map +1 -1
  171. package/packages/contracts/dist/rpc.js.map +1 -1
  172. package/packages/contracts/dist/workflow.d.ts +4 -0
  173. package/packages/contracts/dist/workflow.d.ts.map +1 -1
  174. package/packages/contracts/dist/workflow.js.map +1 -1
  175. package/packages/contracts/package.json +1 -1
  176. package/packages/daemon/package.json +4 -4
  177. package/packages/gsd-agent-core/package.json +5 -5
  178. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.d.ts +5 -0
  179. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  180. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.js +5 -0
  181. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.js.map +1 -1
  182. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  183. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js +7 -0
  184. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  185. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/input-controller.d.ts.map +1 -1
  186. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/input-controller.js +8 -1
  187. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/input-controller.js.map +1 -1
  188. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-chat-render.d.ts.map +1 -1
  189. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-chat-render.js +11 -1
  190. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-chat-render.js.map +1 -1
  191. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-auth.d.ts.map +1 -1
  192. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-auth.js +4 -4
  193. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-auth.js.map +1 -1
  194. package/packages/gsd-agent-modes/dist/modes/rpc/rpc-mode.d.ts.map +1 -1
  195. package/packages/gsd-agent-modes/dist/modes/rpc/rpc-mode.js +3 -1
  196. package/packages/gsd-agent-modes/dist/modes/rpc/rpc-mode.js.map +1 -1
  197. package/packages/gsd-agent-modes/package.json +7 -7
  198. package/packages/mcp-server/dist/cli.js +6 -3
  199. package/packages/mcp-server/dist/cli.js.map +1 -1
  200. package/packages/mcp-server/dist/workflow-tools.d.ts +8 -0
  201. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  202. package/packages/mcp-server/dist/workflow-tools.js +17 -1
  203. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  204. package/packages/mcp-server/package.json +3 -3
  205. package/packages/native/package.json +1 -1
  206. package/packages/pi-agent-core/dist/harness/env/nodejs.d.ts +1 -0
  207. package/packages/pi-agent-core/dist/harness/env/nodejs.d.ts.map +1 -1
  208. package/packages/pi-agent-core/dist/harness/env/nodejs.js +34 -3
  209. package/packages/pi-agent-core/dist/harness/env/nodejs.js.map +1 -1
  210. package/packages/pi-agent-core/dist/index.d.ts +1 -0
  211. package/packages/pi-agent-core/dist/index.d.ts.map +1 -1
  212. package/packages/pi-agent-core/dist/index.js +3 -0
  213. package/packages/pi-agent-core/dist/index.js.map +1 -1
  214. package/packages/pi-agent-core/package.json +1 -1
  215. package/packages/pi-ai/README.md +1 -0
  216. package/packages/pi-ai/dist/image-models.generated.d.ts +2 -2
  217. package/packages/pi-ai/dist/image-models.generated.js +6 -6
  218. package/packages/pi-ai/dist/image-models.generated.js.map +1 -1
  219. package/packages/pi-ai/dist/models.generated.d.ts +35 -125
  220. package/packages/pi-ai/dist/models.generated.d.ts.map +1 -1
  221. package/packages/pi-ai/dist/models.generated.js +46 -120
  222. package/packages/pi-ai/dist/models.generated.js.map +1 -1
  223. package/packages/pi-ai/package.json +3 -2
  224. package/packages/pi-coding-agent/dist/core/auth-storage.d.ts +2 -2
  225. package/packages/pi-coding-agent/dist/core/auth-storage.d.ts.map +1 -1
  226. package/packages/pi-coding-agent/dist/core/auth-storage.js +19 -13
  227. package/packages/pi-coding-agent/dist/core/auth-storage.js.map +1 -1
  228. package/packages/pi-coding-agent/dist/core/provider-readiness.d.ts.map +1 -1
  229. package/packages/pi-coding-agent/dist/core/provider-readiness.js +13 -6
  230. package/packages/pi-coding-agent/dist/core/provider-readiness.js.map +1 -1
  231. package/packages/pi-coding-agent/dist/core/tools/bash.d.ts +11 -0
  232. package/packages/pi-coding-agent/dist/core/tools/bash.d.ts.map +1 -1
  233. package/packages/pi-coding-agent/dist/core/tools/bash.js +53 -11
  234. package/packages/pi-coding-agent/dist/core/tools/bash.js.map +1 -1
  235. package/packages/pi-coding-agent/dist/index.d.ts +1 -1
  236. package/packages/pi-coding-agent/dist/index.d.ts.map +1 -1
  237. package/packages/pi-coding-agent/dist/index.js +1 -1
  238. package/packages/pi-coding-agent/dist/index.js.map +1 -1
  239. package/packages/pi-coding-agent/dist/utils/shell.d.ts +28 -2
  240. package/packages/pi-coding-agent/dist/utils/shell.d.ts.map +1 -1
  241. package/packages/pi-coding-agent/dist/utils/shell.js +56 -10
  242. package/packages/pi-coding-agent/dist/utils/shell.js.map +1 -1
  243. package/packages/pi-coding-agent/package.json +7 -7
  244. package/packages/pi-tui/dist/tui.d.ts.map +1 -1
  245. package/packages/pi-tui/dist/tui.js +9 -0
  246. package/packages/pi-tui/dist/tui.js.map +1 -1
  247. package/packages/pi-tui/package.json +2 -2
  248. package/packages/rpc-client/package.json +2 -2
  249. package/pkg/package.json +1 -1
  250. package/src/resources/GSD-WORKFLOW.md +5 -4
  251. package/src/resources/extensions/async-jobs/async-bash-cancel.test.ts +360 -0
  252. package/src/resources/extensions/async-jobs/async-bash-tool.ts +33 -56
  253. package/src/resources/extensions/async-jobs/await-tool.test.ts +139 -0
  254. package/src/resources/extensions/async-jobs/await-tool.ts +82 -12
  255. package/src/resources/extensions/async-jobs/index.ts +79 -0
  256. package/src/resources/extensions/async-jobs/job-manager.ts +21 -1
  257. package/src/resources/extensions/bg-shell/bg-shell-command.ts +6 -6
  258. package/src/resources/extensions/bg-shell/bg-shell-tool.ts +10 -6
  259. package/src/resources/extensions/bg-shell/overlay.ts +9 -5
  260. package/src/resources/extensions/bg-shell/process-manager.ts +50 -25
  261. package/src/resources/extensions/bg-shell/readiness-detector.ts +12 -0
  262. package/src/resources/extensions/bg-shell/tests/lifecycle-and-utilities.test.ts +48 -1
  263. package/src/resources/extensions/bg-shell/utilities.ts +3 -0
  264. package/src/resources/extensions/browser-tools/engine/managed-gsd-browser.ts +265 -98
  265. package/src/resources/extensions/browser-tools/engine/selection.ts +90 -4
  266. package/src/resources/extensions/browser-tools/index.ts +71 -13
  267. package/src/resources/extensions/browser-tools/tests/browser-engine-selection.test.mjs +83 -13
  268. package/src/resources/extensions/browser-tools/tests/gsd-browser-launch-config.test.mjs +29 -1
  269. package/src/resources/extensions/browser-tools/tests/managed-gsd-browser-tools.test.mjs +136 -0
  270. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +34 -4
  271. package/src/resources/extensions/gsd/auto/custom-verify-retry-store.ts +21 -3
  272. package/src/resources/extensions/gsd/auto/detect-stuck.ts +32 -9
  273. package/src/resources/extensions/gsd/auto/dispatch-history.ts +152 -0
  274. package/src/resources/extensions/gsd/auto/dispatch-key.ts +39 -0
  275. package/src/resources/extensions/gsd/auto/loop.ts +4 -1
  276. package/src/resources/extensions/gsd/auto/orchestrator.ts +70 -46
  277. package/src/resources/extensions/gsd/auto/phases.ts +2 -2
  278. package/src/resources/extensions/gsd/auto-direct-dispatch.ts +8 -32
  279. package/src/resources/extensions/gsd/auto-dispatch.ts +38 -52
  280. package/src/resources/extensions/gsd/auto-model-selection.ts +25 -5
  281. package/src/resources/extensions/gsd/auto-post-unit.ts +25 -8
  282. package/src/resources/extensions/gsd/auto-prompts.ts +118 -35
  283. package/src/resources/extensions/gsd/auto-start.ts +18 -17
  284. package/src/resources/extensions/gsd/auto-tool-tracking.ts +19 -0
  285. package/src/resources/extensions/gsd/auto-unit-tool-scope.ts +14 -21
  286. package/src/resources/extensions/gsd/auto-verification.ts +8 -26
  287. package/src/resources/extensions/gsd/auto-worktree.ts +30 -93
  288. package/src/resources/extensions/gsd/auto.ts +8 -15
  289. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +3 -5
  290. package/src/resources/extensions/gsd/bootstrap/dynamic-tools.ts +23 -6
  291. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +24 -0
  292. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +251 -47
  293. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +352 -84
  294. package/src/resources/extensions/gsd/branch-patterns.ts +3 -0
  295. package/src/resources/extensions/gsd/browser-daemon-auto-prep.ts +108 -0
  296. package/src/resources/extensions/gsd/browser-evidence.ts +18 -2
  297. package/src/resources/extensions/gsd/captures.ts +4 -6
  298. package/src/resources/extensions/gsd/consent-question.ts +416 -0
  299. package/src/resources/extensions/gsd/consent-verdict.ts +86 -0
  300. package/src/resources/extensions/gsd/constants.ts +0 -3
  301. package/src/resources/extensions/gsd/crash-recovery.ts +3 -9
  302. package/src/resources/extensions/gsd/db/queries.ts +37 -0
  303. package/src/resources/extensions/gsd/dispatch-guard.ts +8 -31
  304. package/src/resources/extensions/gsd/doctor-environment.ts +2 -7
  305. package/src/resources/extensions/gsd/doctor-format.ts +12 -7
  306. package/src/resources/extensions/gsd/doctor-runtime-checks.ts +13 -15
  307. package/src/resources/extensions/gsd/engine-hook-contract.ts +79 -0
  308. package/src/resources/extensions/gsd/error-classifier.ts +11 -0
  309. package/src/resources/extensions/gsd/exec-sandbox.ts +49 -9
  310. package/src/resources/extensions/gsd/files.ts +33 -12
  311. package/src/resources/extensions/gsd/guidance.ts +217 -0
  312. package/src/resources/extensions/gsd/guided-flow.ts +16 -2
  313. package/src/resources/extensions/gsd/markdown-renderer.ts +11 -0
  314. package/src/resources/extensions/gsd/mcp-filter.ts +2 -23
  315. package/src/resources/extensions/gsd/mcp-tool-name.ts +6 -11
  316. package/src/resources/extensions/gsd/memory-consolidation-scanner.ts +1 -1
  317. package/src/resources/extensions/gsd/migrate/safety.ts +4 -1
  318. package/src/resources/extensions/gsd/milestone-closeout.ts +13 -23
  319. package/src/resources/extensions/gsd/notification-store.ts +26 -3
  320. package/src/resources/extensions/gsd/parallel-monitor-overlay.ts +6 -4
  321. package/src/resources/extensions/gsd/parsers-legacy.ts +16 -4
  322. package/src/resources/extensions/gsd/paths.ts +33 -0
  323. package/src/resources/extensions/gsd/pre-execution-checks.ts +109 -3
  324. package/src/resources/extensions/gsd/preferences-models.ts +12 -47
  325. package/src/resources/extensions/gsd/prompts/complete-slice.md +2 -2
  326. package/src/resources/extensions/gsd/prompts/plan-slice.md +1 -1
  327. package/src/resources/extensions/gsd/prompts/refine-slice.md +1 -1
  328. package/src/resources/extensions/gsd/prompts/replan-slice.md +1 -1
  329. package/src/resources/extensions/gsd/prompts/run-uat.md +6 -4
  330. package/src/resources/extensions/gsd/prompts/system.md +5 -2
  331. package/src/resources/extensions/gsd/provider-error-guidance.ts +4 -9
  332. package/src/resources/extensions/gsd/provider-switch-observer.ts +1 -1
  333. package/src/resources/extensions/gsd/publication.ts +122 -0
  334. package/src/resources/extensions/gsd/reactive-graph.ts +11 -1
  335. package/src/resources/extensions/gsd/recovery-classification.ts +42 -96
  336. package/src/resources/extensions/gsd/safety/destructive-confirmation.ts +134 -0
  337. package/src/resources/extensions/gsd/state.ts +9 -21
  338. package/src/resources/extensions/gsd/stop-notice.ts +75 -0
  339. package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +22 -0
  340. package/src/resources/extensions/gsd/tests/auto-orchestrator.test.ts +101 -26
  341. package/src/resources/extensions/gsd/tests/browser-automation-contract-fixture.ts +39 -0
  342. package/src/resources/extensions/gsd/tests/browser-contract.test.ts +44 -0
  343. package/src/resources/extensions/gsd/tests/browser-daemon-auto-prep.test.ts +144 -0
  344. package/src/resources/extensions/gsd/tests/checkout-branch-stash-guard.test.ts +66 -1
  345. package/src/resources/extensions/gsd/tests/clear-stale-autostart.test.ts +22 -0
  346. package/src/resources/extensions/gsd/tests/commands-verdict.test.ts +8 -7
  347. package/src/resources/extensions/gsd/tests/complete-slice-verification-gate.test.ts +42 -0
  348. package/src/resources/extensions/gsd/tests/consent-question.test.ts +336 -0
  349. package/src/resources/extensions/gsd/tests/custom-verify-retry-store.test.ts +67 -0
  350. package/src/resources/extensions/gsd/tests/deep-project-auto-loop.test.ts +10 -10
  351. package/src/resources/extensions/gsd/tests/destructive-confirmation.test.ts +303 -0
  352. package/src/resources/extensions/gsd/tests/dispatch-history.test.ts +273 -0
  353. package/src/resources/extensions/gsd/tests/dispatch-run-uat-browser-tools.test.ts +2 -1
  354. package/src/resources/extensions/gsd/tests/dynamic-bash-no-cap.test.ts +132 -0
  355. package/src/resources/extensions/gsd/tests/engine-hook-contract.test.ts +148 -0
  356. package/src/resources/extensions/gsd/tests/exec-graceful-kill.test.ts +193 -0
  357. package/src/resources/extensions/gsd/tests/exec-tool.test.ts +29 -1
  358. package/src/resources/extensions/gsd/tests/extension-bootstrap-isolation.test.ts +35 -1
  359. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +27 -0
  360. package/src/resources/extensions/gsd/tests/guidance.test.ts +148 -0
  361. package/src/resources/extensions/gsd/tests/integration/auto-worktree-milestone-merge.test.ts +53 -11
  362. package/src/resources/extensions/gsd/tests/integration/auto-worktree.test.ts +73 -58
  363. package/src/resources/extensions/gsd/tests/integration/gsd-integration-fixture.ts +80 -0
  364. package/src/resources/extensions/gsd/tests/integration/run-uat.test.ts +199 -0
  365. package/src/resources/extensions/gsd/tests/mcp-project-config.test.ts +3 -1
  366. package/src/resources/extensions/gsd/tests/model-unittype-mapping.test.ts +32 -1
  367. package/src/resources/extensions/gsd/tests/notification-store.test.ts +32 -0
  368. package/src/resources/extensions/gsd/tests/oauth-api-model-routing.test.ts +167 -0
  369. package/src/resources/extensions/gsd/tests/parallel-research-dispatch.test.ts +18 -0
  370. package/src/resources/extensions/gsd/tests/parsers-legacy-importers.test.ts +139 -0
  371. package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +193 -1
  372. package/src/resources/extensions/gsd/tests/prompt-db.test.ts +124 -6
  373. package/src/resources/extensions/gsd/tests/provider-error-guidance.test.ts +3 -3
  374. package/src/resources/extensions/gsd/tests/publication.test.ts +120 -0
  375. package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +157 -0
  376. package/src/resources/extensions/gsd/tests/runtime-invariant-modules.test.ts +1 -0
  377. package/src/resources/extensions/gsd/tests/stop-notice.test.ts +70 -0
  378. package/src/resources/extensions/gsd/tests/token-tool-gating.test.ts +76 -0
  379. package/src/resources/extensions/gsd/tests/tool-invocation-error-loop-break.test.ts +8 -0
  380. package/src/resources/extensions/gsd/tests/tool-surface-readiness.test.ts +155 -0
  381. package/src/resources/extensions/gsd/tests/uat-policy.test.ts +112 -29
  382. package/src/resources/extensions/gsd/tests/unit-closeout.test.ts +209 -0
  383. package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +67 -2
  384. package/src/resources/extensions/gsd/tests/unit-registry.test.ts +163 -0
  385. package/src/resources/extensions/gsd/tests/web-app-uat.test.ts +44 -1
  386. package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +2 -2
  387. package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +2 -2
  388. package/src/resources/extensions/gsd/tests/write-gate-seam.test.ts +358 -0
  389. package/src/resources/extensions/gsd/tests/write-gate.test.ts +67 -1
  390. package/src/resources/extensions/gsd/tool-presentation-plan.ts +4 -4
  391. package/src/resources/extensions/gsd/tool-surface-readiness.ts +76 -0
  392. package/src/resources/extensions/gsd/tools/complete-slice.ts +20 -10
  393. package/src/resources/extensions/gsd/tools/exec-tool.ts +8 -7
  394. package/src/resources/extensions/gsd/tools/plan-slice.ts +12 -6
  395. package/src/resources/extensions/gsd/uat-policy.ts +62 -16
  396. package/src/resources/extensions/gsd/unit-closeout.ts +201 -0
  397. package/src/resources/extensions/gsd/unit-context-composer.ts +111 -1
  398. package/src/resources/extensions/gsd/unit-context-manifest.ts +4 -28
  399. package/src/resources/extensions/gsd/unit-registry.ts +412 -0
  400. package/src/resources/extensions/gsd/unit-tool-contracts.ts +27 -192
  401. package/src/resources/extensions/gsd/verdict-parser.ts +1 -1
  402. package/src/resources/extensions/gsd/web-app-uat.ts +51 -8
  403. package/src/resources/extensions/gsd/workflow-tool-surface.ts +4 -1
  404. package/src/resources/extensions/gsd/worktree-git-recovery.ts +15 -9
  405. package/src/resources/extensions/gsd/worktree-lifecycle.ts +3 -8
  406. package/src/resources/extensions/gsd/worktree-root.ts +12 -0
  407. package/src/resources/extensions/gsd/worktree-session-state.ts +3 -5
  408. package/src/resources/extensions/search-the-web/native-search.ts +5 -3
  409. package/src/resources/extensions/shared/browser-contract.ts +66 -0
  410. package/src/resources/extensions/shared/gsd-browser-cli.ts +119 -5
  411. package/src/resources/shared/package.json +3 -0
  412. package/src/resources/skills/create-skill/references/executable-code.md +1 -1
  413. package/src/resources/skills/create-skill/workflows/add-reference.md +8 -3
  414. package/src/resources/skills/create-skill/workflows/add-script.md +4 -2
  415. package/src/resources/skills/create-skill/workflows/add-template.md +3 -1
  416. package/src/resources/skills/create-skill/workflows/add-workflow.md +8 -3
  417. package/src/resources/skills/create-skill/workflows/upgrade-to-router.md +10 -5
  418. package/src/resources/skills/create-skill/workflows/verify-skill.md +9 -4
  419. package/src/resources/skills/spike-wrap-up/SKILL.md +9 -9
  420. package/dist/resources/extensions/gsd/user-input-boundary.js +0 -218
  421. package/src/resources/extensions/gsd/tests/user-input-boundary.test.ts +0 -173
  422. package/src/resources/extensions/gsd/user-input-boundary.ts +0 -216
  423. /package/dist/web/standalone/.next/static/{C24pqUd-aru-l0Dp0gLZP → jmTLg6xZmAuq_LIqKOxrH}/_buildManifest.js +0 -0
  424. /package/dist/web/standalone/.next/static/{C24pqUd-aru-l0Dp0gLZP → jmTLg6xZmAuq_LIqKOxrH}/_ssgManifest.js +0 -0
@@ -1,13 +1,16 @@
1
1
  // Project/App: gsd-pi
2
2
  // File Purpose: ADR-015 Recovery Classification module for runtime failure taxonomy.
3
3
 
4
+ import { isToolUnavailableError } from "./auto-tool-tracking.js";
4
5
  import { classifyError, isTransient, type ErrorClass } from "./error-classifier.js";
6
+ import { recoveryRemediation } from "./guidance.js";
5
7
  import { ReconciliationFailedError } from "./state-reconciliation.js";
6
8
  import { IllegalPhaseTransitionError } from "./state-transition-matrix.js";
7
9
 
8
10
  export type RecoveryFailureKind =
9
11
  | "tool-schema"
10
12
  | "tool-contract"
13
+ | "tool-unavailable"
11
14
  | "deterministic-policy"
12
15
  | "lifecycle-progression"
13
16
  | "stale-worker"
@@ -49,106 +52,49 @@ export function classifyFailure(input: RecoveryClassificationInput): RecoveryCla
49
52
  ? "illegal-transition"
50
53
  : input.failureKind ?? inferFailureKind(message);
51
54
 
52
- switch (failureKind) {
53
- case "tool-schema":
54
- return {
55
- failureKind,
56
- action: "stop",
57
- reason: `Tool schema failure${unitSuffix(input)}: ${message}`,
58
- exitReason: "tool-schema",
59
- remediation: "Fix the Unit Tool Contract or tool schema before retrying.",
60
- };
61
- case "tool-contract":
62
- return {
63
- failureKind,
64
- action: "stop",
65
- reason: `Tool Contract failure${unitSuffix(input)}: ${message}`,
66
- exitReason: "tool-contract",
67
- remediation: "Fix the Unit Tool Contract or prompt so the Unit is only asked to use tools owned by its phase.",
68
- };
69
- case "deterministic-policy":
70
- return {
71
- failureKind,
72
- action: "stop",
73
- reason: `Deterministic policy failure${unitSuffix(input)}: ${message}`,
74
- exitReason: "deterministic-policy",
75
- remediation: "Resolve the policy blocker; retrying the same Unit will repeat the failure.",
76
- };
77
- case "lifecycle-progression":
78
- return {
79
- failureKind,
80
- action: "stop",
81
- reason: `Lifecycle progression failure${unitSuffix(input)}: ${message}`,
82
- exitReason: "lifecycle-progression",
83
- remediation: "Route to the required owning Unit or restore the missing artifact before advancing lifecycle state.",
84
- };
85
- case "stale-worker":
86
- return {
87
- failureKind,
88
- action: "stop",
89
- reason: `Stale worker failure${unitSuffix(input)}: ${message}`,
90
- exitReason: "stale-worker",
91
- remediation: "Clear or reconcile the stale worker before dispatching another Unit.",
92
- };
93
- case "worktree-invalid":
94
- return {
95
- failureKind,
96
- action: "stop",
97
- reason: `Worktree invalid${unitSuffix(input)}: ${message}`,
98
- exitReason: "worktree-invalid",
99
- remediation: "Repair or recreate the milestone worktree before launching source-writing Units.",
100
- };
101
- case "verification-drift":
102
- return {
103
- failureKind,
104
- action: "escalate",
105
- reason: `Verification drift${unitSuffix(input)}: ${message}`,
106
- exitReason: "verification-drift",
107
- remediation: "Inspect the verification artifact and reconcile the state snapshot before resuming.",
108
- };
109
- case "reconciliation-drift":
110
- return {
111
- failureKind,
112
- action: "escalate",
113
- reason: `Reconciliation drift${unitSuffix(input)}: ${message}`,
114
- exitReason: "reconciliation-drift",
115
- remediation:
116
- "Inspect the persistent or repair-failed drift kinds reported by the State Reconciliation Module before resuming.",
117
- };
118
- case "illegal-transition":
119
- return {
120
- failureKind,
121
- action: "escalate",
122
- reason: `Illegal phase transition${unitSuffix(input)}: ${message}`,
123
- exitReason: "illegal-transition",
124
- remediation:
125
- "A derived Phase edge rejected by the Phase Transition Invariant survived reconciliation; inspect deriveState and the State Reconciliation Module before resuming.",
126
- };
127
- case "provider": {
128
- const providerClass = classifyError(message, input.retryAfterMs);
129
- return {
130
- failureKind,
131
- action: isTransient(providerClass) ? "retry" : "escalate",
132
- reason: message,
133
- exitReason: `provider-${providerClass.kind}`,
134
- remediation: isTransient(providerClass)
135
- ? "Retry after the provider/network condition clears."
136
- : "Inspect provider credentials, model entitlement, or request shape.",
137
- providerClass: providerClass.kind,
138
- };
139
- }
140
- case "runtime-unknown":
141
- return {
142
- failureKind,
143
- action: "escalate",
144
- reason: message,
145
- exitReason: "runtime-unknown",
146
- remediation: "Inspect the runtime error and add a dedicated classification if it is repeatable.",
147
- };
55
+ if (failureKind === "provider") {
56
+ const providerClass = classifyError(message, input.retryAfterMs);
57
+ const transient = isTransient(providerClass);
58
+ return {
59
+ failureKind,
60
+ action: transient ? "retry" : "escalate",
61
+ reason: message,
62
+ exitReason: `provider-${providerClass.kind}`,
63
+ remediation: recoveryRemediation(transient ? "provider-transient" : "provider-permanent"),
64
+ providerClass: providerClass.kind,
65
+ };
148
66
  }
67
+
68
+ const { action, label } = FAILURE_TAXONOMY[failureKind];
69
+ return {
70
+ failureKind,
71
+ action,
72
+ reason: label ? `${label}${unitSuffix(input)}: ${message}` : message,
73
+ exitReason: failureKind,
74
+ remediation: recoveryRemediation(failureKind),
75
+ };
149
76
  }
150
77
 
78
+ /** Per-kind action and reason label. Remediation lives in the Guidance module. */
79
+ const FAILURE_TAXONOMY: Record<
80
+ Exclude<RecoveryFailureKind, "provider">,
81
+ { action: RecoveryAction; label: string | null }
82
+ > = {
83
+ "tool-schema": { action: "stop", label: "Tool schema failure" },
84
+ "tool-contract": { action: "stop", label: "Tool Contract failure" },
85
+ "tool-unavailable": { action: "retry", label: "Tool unavailable" },
86
+ "deterministic-policy": { action: "stop", label: "Deterministic policy failure" },
87
+ "lifecycle-progression": { action: "stop", label: "Lifecycle progression failure" },
88
+ "stale-worker": { action: "stop", label: "Stale worker failure" },
89
+ "worktree-invalid": { action: "stop", label: "Worktree invalid" },
90
+ "verification-drift": { action: "escalate", label: "Verification drift" },
91
+ "reconciliation-drift": { action: "escalate", label: "Reconciliation drift" },
92
+ "illegal-transition": { action: "escalate", label: "Illegal phase transition" },
93
+ "runtime-unknown": { action: "escalate", label: null },
94
+ };
95
+
151
96
  function inferFailureKind(message: string): RecoveryFailureKind {
97
+ if (isToolUnavailableError(message)) return "tool-unavailable";
152
98
  if (/tool contract|auto-unit tool scope|phase-boundary gate|not permitted.*own/i.test(message)) return "tool-contract";
153
99
  if (/lifecycle progression|required artifact|missing .*assessment|missing .*closeout|cannot legally (?:advance|progress)/i.test(message)) return "lifecycle-progression";
154
100
  if (/schema|invalid.*tool|tool.*invalid|enum/i.test(message)) return "tool-schema";
@@ -0,0 +1,134 @@
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
+
23
+ import { resolve } from "node:path";
24
+
25
+ /**
26
+ * Question-id substring that marks an ask_user_questions call as a
27
+ * destructive-command confirmation. The tool_result handler promotes the
28
+ * pending command to a confirmed token when an affirmative answer arrives for
29
+ * a question whose id contains this marker.
30
+ */
31
+ export const DESTRUCTIVE_CONFIRM_GATE_MARKER = "destructive_confirm";
32
+
33
+ interface DestructiveConfirmationState {
34
+ /** Command awaiting confirmation, captured when the guard blocked it. */
35
+ pendingCommand: string | null;
36
+ /** Confirmed command cleared on first matching consume. */
37
+ confirmedCommand: string | null;
38
+ }
39
+
40
+ const statesByBasePath = new Map<string, DestructiveConfirmationState>();
41
+
42
+ function stateKey(basePath: string): string {
43
+ return resolve(basePath);
44
+ }
45
+
46
+ function getState(basePath: string): DestructiveConfirmationState {
47
+ const key = stateKey(basePath);
48
+ let state = statesByBasePath.get(key);
49
+ if (!state) {
50
+ state = { pendingCommand: null, confirmedCommand: null };
51
+ statesByBasePath.set(key, state);
52
+ }
53
+ return state;
54
+ }
55
+
56
+ /**
57
+ * Normalize a command for stable matching across block → confirm → retry.
58
+ * Trims surrounding whitespace and collapses internal runs of whitespace so
59
+ * cosmetic reformatting of the same command still matches the token.
60
+ */
61
+ export function normalizeDestructiveCommand(command: string): string {
62
+ return command.replace(/\s+/g, " ").trim();
63
+ }
64
+
65
+ /**
66
+ * Whether an ask_user_questions question id is a destructive-confirm gate.
67
+ */
68
+ export function isDestructiveConfirmGateId(questionId: unknown): boolean {
69
+ return typeof questionId === "string" && questionId.includes(DESTRUCTIVE_CONFIRM_GATE_MARKER);
70
+ }
71
+
72
+ /**
73
+ * Record that a destructive command was blocked and is awaiting confirmation.
74
+ * Called by the guard at block time. Overwrites any prior pending command —
75
+ * only the most recently blocked command can be confirmed.
76
+ */
77
+ export function requestDestructiveConfirmation(
78
+ command: string,
79
+ basePath: string = process.cwd(),
80
+ ): void {
81
+ const state = getState(basePath);
82
+ state.pendingCommand = normalizeDestructiveCommand(command);
83
+ // A fresh request invalidates any stale confirmed token for a different
84
+ // command so confirmation cannot leak across distinct destructive actions.
85
+ state.confirmedCommand = null;
86
+ }
87
+
88
+ /**
89
+ * Promote the pending command to a confirmed, one-shot token. Called by the
90
+ * tool_result handler when the user gives an affirmative answer to a
91
+ * destructive-confirm gate. Returns the confirmed command, or null if there
92
+ * was nothing pending (e.g. confirmation arrived without a preceding block).
93
+ */
94
+ export function confirmDestructiveCommand(
95
+ basePath: string = process.cwd(),
96
+ ): string | null {
97
+ const state = getState(basePath);
98
+ if (!state.pendingCommand) return null;
99
+ state.confirmedCommand = state.pendingCommand;
100
+ state.pendingCommand = null;
101
+ return state.confirmedCommand;
102
+ }
103
+
104
+ /**
105
+ * Check whether the given command has been confirmed, consuming the token if
106
+ * so. Returns true exactly once per confirmation; subsequent calls (or a
107
+ * non-matching command) return false. Called by the guard before blocking.
108
+ */
109
+ export function consumeDestructiveConfirmation(
110
+ command: string,
111
+ basePath: string = process.cwd(),
112
+ ): boolean {
113
+ const state = getState(basePath);
114
+ if (!state.confirmedCommand) return false;
115
+ if (state.confirmedCommand !== normalizeDestructiveCommand(command)) return false;
116
+ state.confirmedCommand = null;
117
+ return true;
118
+ }
119
+
120
+ /**
121
+ * Inspect the pending command without consuming it (diagnostics/tests).
122
+ */
123
+ export function peekPendingDestructiveCommand(
124
+ basePath: string = process.cwd(),
125
+ ): string | null {
126
+ return getState(basePath).pendingCommand;
127
+ }
128
+
129
+ /**
130
+ * Clear all destructive-confirmation state for a basePath (tests / flow reset).
131
+ */
132
+ export function resetDestructiveConfirmation(basePath: string = process.cwd()): void {
133
+ statesByBasePath.delete(stateKey(basePath));
134
+ }
@@ -13,6 +13,11 @@ import type {
13
13
  MilestoneRegistryEntry,
14
14
  } from './types.js';
15
15
 
16
+ // Pre-migration fallback ONLY (ADR-017): deriveState must work on projects
17
+ // whose DB does not exist yet (before md-importer runs), so it parses markdown
18
+ // projections when `isDbAvailable()` is false or the DB has no rows. Once the
19
+ // DB is populated, decision reads go through gsd-db queries — these parsers
20
+ // must never be consulted when DB data is present.
16
21
  import {
17
22
  parseRoadmap,
18
23
  parsePlan,
@@ -71,27 +76,10 @@ import {
71
76
  readinessNeedsDiscussion,
72
77
  } from './milestone-readiness.js';
73
78
 
74
- function formatNeedsAttentionBlocker(milestoneId: string): string {
75
- return [
76
- `Milestone ${milestoneId} is blocked because milestone validation returned needs-attention.`,
77
- `Fix options:`,
78
- `1. Review the validation details: \`/gsd status\``,
79
- `2. If you fixed the missing evidence or issue, re-run milestone validation: \`/gsd validate-milestone\``,
80
- `3. If the finding is acceptable, override it: \`/gsd verdict pass --rationale "why this is okay"\``,
81
- `4. If this should wait, defer it explicitly: \`/gsd park ${milestoneId}\``,
82
- `After validation or override passes, run \`/gsd auto\` to complete and merge the milestone.`,
83
- ].join("\n");
84
- }
85
-
86
- function formatNeedsRemediationBlocker(milestoneId: string): string {
87
- return [
88
- `Milestone ${milestoneId} is blocked because milestone validation returned needs-remediation, but all slices are complete.`,
89
- `Fix options:`,
90
- `1. Run \`/gsd dispatch reassess\` to add remediation slices, then run \`/gsd auto\``,
91
- `2. If the finding is acceptable, override it: \`/gsd verdict pass --rationale "why this is okay"\``,
92
- `3. If this should wait, defer it explicitly: \`/gsd park ${milestoneId}\``,
93
- ].join("\n");
94
- }
79
+ import {
80
+ needsAttentionBlockerGuidance as formatNeedsAttentionBlocker,
81
+ needsRemediationBlockerGuidance as formatNeedsRemediationBlocker,
82
+ } from './guidance.js';
95
83
 
96
84
  /**
97
85
  * A "ghost" milestone directory contains only META.json (and no substantive
@@ -0,0 +1,75 @@
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
+
9
+ export type StopNoticeKind = "stopped" | "blocked";
10
+
11
+ /** A reason string of the form "Blocked: …" marks a blocked stop. */
12
+ export function isBlockedStopReason(reason?: string | null): boolean {
13
+ return /^Blocked:\s*/i.test(reason ?? "");
14
+ }
15
+
16
+ /** Strip the "Blocked: " marker for display. */
17
+ export function stopNoticeDisplayReason(reason?: string | null): string {
18
+ return (reason ?? "").replace(/^Blocked:\s*/i, "").trim();
19
+ }
20
+
21
+ export function stopNoticeKind(reason?: string | null): StopNoticeKind {
22
+ return isBlockedStopReason(reason) ? "blocked" : "stopped";
23
+ }
24
+
25
+ /** Canonical stop-notice prefix: "Auto-mode blocked — reason" / "Auto-mode stopped". */
26
+ export function formatStopNoticePrefix(reason?: string | null): string {
27
+ const displayReason = stopNoticeDisplayReason(reason);
28
+ const prefix = stopNoticeKind(reason) === "blocked" ? "Auto-mode blocked" : "Auto-mode stopped";
29
+ return displayReason ? `${prefix} — ${displayReason}` : prefix;
30
+ }
31
+
32
+ // ─── Classification (headless host side) ────────────────────────────────
33
+ // The canonical lowercase prefixes the headless event loop recognizes in
34
+ // notify messages. Emitters above and ad-hoc emitters elsewhere must start
35
+ // their terminal notices with one of these.
36
+
37
+ export const PAUSED_NOTICE_PREFIXES = ["auto-mode paused", "step-mode paused"] as const;
38
+
39
+ export const TERMINAL_NOTICE_PREFIXES = [
40
+ "auto-mode stopped",
41
+ "step-mode stopped",
42
+ "auto-mode complete",
43
+ "no active milestone",
44
+ "auto-mode idle",
45
+ ] as const;
46
+
47
+ /** Manual-resolution notices emitted before auto-mode can formally pause/stop. */
48
+ export function isManualResolutionNotice(message: string): boolean {
49
+ return (
50
+ message.includes("resolve manually and re-run /gsd auto") ||
51
+ message.includes("resolve conflicts manually and run /gsd auto to resume") ||
52
+ message.includes("resolve and run /gsd auto to resume")
53
+ );
54
+ }
55
+
56
+ export function isPauseNotice(message: string): boolean {
57
+ return PAUSED_NOTICE_PREFIXES.some((prefix) => message.startsWith(prefix));
58
+ }
59
+
60
+ export function isTerminalNotice(message: string): boolean {
61
+ return TERMINAL_NOTICE_PREFIXES.some((prefix) => message.startsWith(prefix));
62
+ }
63
+
64
+ /** Pauses that do not require operator intervention in headless mode. */
65
+ export function isNonBlockingPauseNotice(message: string): boolean {
66
+ return message.includes("idempotent advance: unit already active");
67
+ }
68
+
69
+ export function isBlockedNoticeMessage(message: string): boolean {
70
+ return (
71
+ message.includes("blocked:") ||
72
+ (isPauseNotice(message) && !isNonBlockingPauseNotice(message)) ||
73
+ isManualResolutionNotice(message)
74
+ );
75
+ }
@@ -857,3 +857,25 @@ test("resolveModelId: claude-code wins when session is claude-code regardless of
857
857
  assert.ok(result, "should resolve a model");
858
858
  assert.equal(result.provider, "claude-code", "claude-code must win when it is the session provider");
859
859
  });
860
+
861
+ test("resolveModelId: openai-codex wins over openai for bare GPT IDs when both are available", () => {
862
+ const availableModels = [
863
+ { id: "gpt-5.5", provider: "openai" },
864
+ { id: "gpt-5.5", provider: "openai-codex" },
865
+ ];
866
+
867
+ const result = resolveModelId("gpt-5.5", availableModels, undefined);
868
+ assert.ok(result, "should resolve a model");
869
+ assert.equal(result.provider, "openai-codex", "ChatGPT OAuth must win over platform API for bare IDs");
870
+ });
871
+
872
+ test("resolveModelId: github-copilot wins over openai when both offer the same GPT model", () => {
873
+ const availableModels = [
874
+ { id: "gpt-5.5", provider: "openai" },
875
+ { id: "gpt-5.5", provider: "github-copilot" },
876
+ ];
877
+
878
+ const result = resolveModelId("gpt-5.5", availableModels, undefined);
879
+ assert.ok(result);
880
+ assert.equal(result.provider, "github-copilot");
881
+ });
@@ -22,8 +22,8 @@ import {
22
22
  createAutoOrchestrator,
23
23
  decideOrchestratorDispatch,
24
24
  resolveLiveOrchestratorBasePath,
25
- STUCK_WINDOW_SIZE,
26
25
  } from "../auto/orchestrator.js";
26
+ import { STUCK_WINDOW_SIZE } from "../auto/dispatch-history.js";
27
27
  import type { OrchestratorContext } from "../auto/orchestrator.js";
28
28
  import type { AutoOrchestrationModule, AutoSessionContext } from "../auto/contracts.js";
29
29
  import type { GSDState } from "../types.js";
@@ -41,6 +41,10 @@ import {
41
41
  openDatabase,
42
42
  } from "../gsd-db.js";
43
43
  import { AutoSession } from "../auto/session.js";
44
+ import { registerAutoWorker } from "../db/auto-workers.js";
45
+ import { claimMilestoneLease } from "../db/milestone-leases.js";
46
+ import { recordDispatchClaim, markFailed } from "../db/unit-dispatches.js";
47
+ import { normalizeRealPath } from "../paths.js";
44
48
  import { acquireSessionLock, releaseSessionLock } from "../session-lock.js";
45
49
  import { queryJournal } from "../journal.js";
46
50
  import { invalidateAllCaches } from "../cache.js";
@@ -518,13 +522,12 @@ test("advance() is idempotent for the same active unit", async (t) => {
518
522
  if (first.kind === "advanced") {
519
523
  assert.deepEqual(first.unit, { unitType: "execute-task", unitId: "M001/S01/T01" });
520
524
  }
521
- assert.equal(second.kind, "blocked");
522
- if (second.kind !== "blocked") return;
525
+ assert.equal(second.kind, "skipped");
526
+ if (second.kind !== "skipped") return;
523
527
  assert.equal(second.reason, "idempotent advance: unit already active");
524
- assert.equal(second.action, "pause");
525
528
  });
526
529
 
527
- test("idempotency block fires with its own reason before saturation", async (t) => {
530
+ test("idempotency skip fires with its own reason before saturation", async (t) => {
528
531
  const f = makeFixture();
529
532
  t.after(() => f.cleanup());
530
533
 
@@ -532,10 +535,9 @@ test("idempotency block fires with its own reason before saturation", async (t)
532
535
  const second = await f.orchestrator.advance();
533
536
 
534
537
  assert.equal(first.kind, "advanced");
535
- assert.equal(second.kind, "blocked");
536
- if (second.kind !== "blocked") return;
538
+ assert.equal(second.kind, "skipped");
539
+ if (second.kind !== "skipped") return;
537
540
  assert.equal(second.reason, "idempotent advance: unit already active");
538
- assert.equal(second.action, "pause");
539
541
  });
540
542
 
541
543
  test("completeActiveUnit clears in-flight idempotency and stops stale same-unit advance", async (t) => {
@@ -671,12 +673,12 @@ test("resume() clears idempotent lock and allows re-advance", async (t) => {
671
673
  t.after(() => f.cleanup());
672
674
 
673
675
  const first = await f.orchestrator.advance();
674
- const blocked = await f.orchestrator.advance();
676
+ const idempotent = await f.orchestrator.advance();
675
677
  const resumed = await f.orchestrator.resume();
676
678
  const next = await f.orchestrator.advance();
677
679
 
678
680
  assert.equal(first.kind, "advanced");
679
- assert.equal(blocked.kind, "blocked");
681
+ assert.equal(idempotent.kind, "skipped");
680
682
  assert.equal(resumed.kind, "resumed");
681
683
  assert.equal(next.kind, "advanced");
682
684
  });
@@ -686,11 +688,11 @@ test("start() clears prior idempotent lock", async (t) => {
686
688
  t.after(() => f.cleanup());
687
689
 
688
690
  await f.orchestrator.advance();
689
- const blocked = await f.orchestrator.advance();
691
+ const idempotent = await f.orchestrator.advance();
690
692
  const restarted = await f.orchestrator.start(SESSION_CONTEXT);
691
693
  const next = await f.orchestrator.advance();
692
694
 
693
- assert.equal(blocked.kind, "blocked");
695
+ assert.equal(idempotent.kind, "skipped");
694
696
  assert.equal(restarted.kind, "started");
695
697
  assert.equal(next.kind, "advanced");
696
698
  });
@@ -700,32 +702,28 @@ test("stop() clears idempotent unit lock so advance can run again", async (t) =>
700
702
  t.after(() => f.cleanup());
701
703
 
702
704
  const first = await f.orchestrator.advance();
703
- const blocked = await f.orchestrator.advance();
705
+ const idempotent = await f.orchestrator.advance();
704
706
  const stopped = await f.orchestrator.stop("reset");
705
707
  const second = await f.orchestrator.advance();
706
708
 
707
709
  assert.equal(first.kind, "advanced");
708
- assert.equal(blocked.kind, "blocked");
710
+ assert.equal(idempotent.kind, "skipped");
709
711
  assert.equal(stopped.kind, "stopped");
710
712
  assert.equal(second.kind, "advanced");
711
713
  });
712
714
 
713
- test("blocked path journals advance-blocked and records a health snapshot", async (t) => {
715
+ test("idempotent path journals advance-skipped and records a health snapshot", async (t) => {
714
716
  const f = makeFixture();
715
717
  t.after(() => f.cleanup());
716
718
 
717
719
  await f.orchestrator.advance();
718
720
  await f.orchestrator.advance();
719
721
 
720
- assert.ok(f.journalNames().includes("advance-blocked"));
722
+ assert.ok(f.journalNames().includes("advance-skipped"));
721
723
  });
722
724
 
723
725
  // ─── Stuck-loop ring buffer (issue #5787) ──────────────────────────────────
724
726
 
725
- test("STUCK_WINDOW_SIZE matches the legacy auto/phases.ts constant", () => {
726
- assert.equal(STUCK_WINDOW_SIZE, 6);
727
- });
728
-
729
727
  test("stuck-loop: empty ring on a freshly constructed orchestrator advances normally", async (t) => {
730
728
  const f = makeFixture();
731
729
  t.after(() => f.cleanup());
@@ -761,21 +759,24 @@ test("stuck-loop: ring saturated with same unit blocks with action 'stop' and st
761
759
  // First call advances.
762
760
  assert.equal(results[0].kind, "advanced");
763
761
 
764
- // Intermediate calls are blocked by idempotency (not stuck-loop yet).
762
+ // Intermediate calls are skipped by idempotency (not stuck-loop yet).
765
763
  for (let i = 1; i < STUCK_WINDOW_SIZE - 1; i++) {
766
764
  const r = results[i];
767
- assert.equal(r.kind, "blocked", `round ${i} should be blocked`);
768
- if (r.kind !== "blocked") return;
765
+ assert.equal(r.kind, "skipped", `round ${i} should be skipped`);
766
+ if (r.kind !== "skipped") return;
769
767
  assert.equal(r.reason, "idempotent advance: unit already active");
770
- assert.equal(r.action, "pause");
771
768
  }
772
769
 
773
- // The final call (ring now holds STUCK_WINDOW_SIZE copies) returns stuck-loop.
770
+ // The final call (ring now holds STUCK_WINDOW_SIZE copies) returns stuck-loop
771
+ // with the detect-stuck rule verdict in the reason.
774
772
  const last = results[STUCK_WINDOW_SIZE - 1];
775
773
  assert.equal(last.kind, "blocked");
776
774
  if (last.kind !== "blocked") return;
777
775
  assert.equal(last.action, "stop");
778
- assert.equal(last.reason, `stuck-loop: execute-task:M001/S01/T01 picked ${STUCK_WINDOW_SIZE} times`);
776
+ assert.ok(
777
+ last.reason.startsWith("stuck-loop: execute-task:M001/S01/T01 derived"),
778
+ `expected detect-stuck verdict reason, got: ${last.reason}`,
779
+ );
779
780
  });
780
781
 
781
782
  test("stuck-loop: start() resets the ring so a fresh saturation cycle is required", async (t) => {
@@ -855,6 +856,80 @@ test("stuck-loop: stop('user-request') resets the ring (hard stop)", async (t) =
855
856
  assert.equal(next.kind, "advanced");
856
857
  });
857
858
 
859
+ test("stuck-loop #482 regression: start() rehydrates the window from the dispatch ledger so cross-session re-dispatch loops are detected", async (t) => {
860
+ const f = makeFixture();
861
+ t.after(() => f.cleanup());
862
+
863
+ // Simulate a PRIOR session: the dispatch ledger recorded the same unit
864
+ // being re-dispatched repeatedly without progress. The orchestrator under test is a
865
+ // fresh instance (as it would be after a session restart) — before the
866
+ // Dispatch History module, start() reset the window to [] and the loop
867
+ // would silently re-dispatch the unit forever (#482: 146 re-dispatches).
868
+ const worker = registerAutoWorker({ projectRootRealpath: normalizeRealPath(f.base) });
869
+ const lease = claimMilestoneLease(worker, "M001");
870
+ assert.equal(lease.ok, true);
871
+ if (!lease.ok) return;
872
+ for (let i = 0; i < STUCK_WINDOW_SIZE - 1; i++) {
873
+ const claim = recordDispatchClaim({
874
+ traceId: `prior-session-${i}`,
875
+ workerId: worker,
876
+ milestoneLeaseToken: lease.token,
877
+ milestoneId: "M001",
878
+ unitType: "execute-task",
879
+ unitId: "M001/S01/T01",
880
+ });
881
+ assert.equal(claim.ok, true);
882
+ if (!claim.ok) return;
883
+ markFailed(claim.dispatchId, { errorSummary: "" });
884
+ }
885
+
886
+ const started = await f.orchestrator.start(SESSION_CONTEXT);
887
+ assert.equal(started.kind, "started");
888
+
889
+ // The very next decision for the same unit must trip the stuck verdict
890
+ // instead of advancing.
891
+ const result = await f.orchestrator.advance();
892
+ assert.equal(result.kind, "blocked");
893
+ if (result.kind !== "blocked") return;
894
+ assert.equal(result.action, "stop");
895
+ assert.ok(result.reason.startsWith("stuck-loop:"), `expected stuck-loop reason, got: ${result.reason}`);
896
+ });
897
+
898
+ test("stuck-loop #482: resume() with an empty window rehydrates from the dispatch ledger", async (t) => {
899
+ const f = makeFixture();
900
+ t.after(() => f.cleanup());
901
+
902
+ const worker = registerAutoWorker({ projectRootRealpath: normalizeRealPath(f.base) });
903
+ const lease = claimMilestoneLease(worker, "M001");
904
+ assert.equal(lease.ok, true);
905
+ if (!lease.ok) return;
906
+ for (let i = 0; i < STUCK_WINDOW_SIZE - 1; i++) {
907
+ const claim = recordDispatchClaim({
908
+ traceId: `prior-session-resume-${i}`,
909
+ workerId: worker,
910
+ milestoneLeaseToken: lease.token,
911
+ milestoneId: "M001",
912
+ unitType: "execute-task",
913
+ unitId: "M001/S01/T01",
914
+ });
915
+ assert.equal(claim.ok, true);
916
+ if (!claim.ok) return;
917
+ markFailed(claim.dispatchId, { errorSummary: "" });
918
+ }
919
+
920
+ // Fresh orchestrator resuming a prior session: window starts empty, so
921
+ // resume() must rehydrate (while in-process resume keeps the live window —
922
+ // see the #572 preservation tests above).
923
+ const resumed = await f.orchestrator.resume();
924
+ assert.equal(resumed.kind, "resumed");
925
+
926
+ const result = await f.orchestrator.advance();
927
+ assert.equal(result.kind, "blocked");
928
+ if (result.kind !== "blocked") return;
929
+ assert.equal(result.action, "stop");
930
+ assert.ok(result.reason.startsWith("stuck-loop:"), `expected stuck-loop reason, got: ${result.reason}`);
931
+ });
932
+
858
933
  test("stuck-loop: journal records the stuck-loop reason on advance-blocked", async (t) => {
859
934
  const f = makeFixture();
860
935
  t.after(() => f.cleanup());