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

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 (575) 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/mcp-server.js +2 -1
  7. package/dist/models-resolver.d.ts +3 -13
  8. package/dist/models-resolver.js +3 -22
  9. package/dist/resource-loader.d.ts +10 -5
  10. package/dist/resource-loader.js +123 -20
  11. package/dist/resources/.managed-resources-content-hash +1 -1
  12. package/dist/resources/GSD-WORKFLOW.md +5 -4
  13. package/dist/resources/extensions/async-jobs/async-bash-tool.js +30 -64
  14. package/dist/resources/extensions/async-jobs/await-tool.js +80 -12
  15. package/dist/resources/extensions/async-jobs/index.js +65 -0
  16. package/dist/resources/extensions/async-jobs/job-manager.js +12 -1
  17. package/dist/resources/extensions/bg-shell/bg-shell-command.js +6 -6
  18. package/dist/resources/extensions/bg-shell/bg-shell-tool.js +10 -7
  19. package/dist/resources/extensions/bg-shell/overlay.js +9 -6
  20. package/dist/resources/extensions/bg-shell/process-manager.js +54 -25
  21. package/dist/resources/extensions/bg-shell/readiness-detector.js +11 -0
  22. package/dist/resources/extensions/bg-shell/utilities.js +3 -0
  23. package/dist/resources/extensions/browser-tools/engine/managed-gsd-browser.js +209 -88
  24. package/dist/resources/extensions/browser-tools/engine/selection.js +73 -5
  25. package/dist/resources/extensions/browser-tools/index.js +69 -12
  26. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +30 -4
  27. package/dist/resources/extensions/gsd/auto/custom-verify-retry-store.js +17 -2
  28. package/dist/resources/extensions/gsd/auto/detect-stuck.js +33 -13
  29. package/dist/resources/extensions/gsd/auto/dispatch-history.js +105 -0
  30. package/dist/resources/extensions/gsd/auto/dispatch-key.js +37 -0
  31. package/dist/resources/extensions/gsd/auto/loop.js +4 -1
  32. package/dist/resources/extensions/gsd/auto/orchestrator.js +89 -54
  33. package/dist/resources/extensions/gsd/auto/phases.js +49 -6
  34. package/dist/resources/extensions/gsd/auto/session.js +3 -0
  35. package/dist/resources/extensions/gsd/auto-direct-dispatch.js +11 -34
  36. package/dist/resources/extensions/gsd/auto-dispatch.js +50 -58
  37. package/dist/resources/extensions/gsd/auto-model-selection.js +36 -13
  38. package/dist/resources/extensions/gsd/auto-post-unit.js +35 -8
  39. package/dist/resources/extensions/gsd/auto-prompts.js +81 -19
  40. package/dist/resources/extensions/gsd/auto-start.js +41 -18
  41. package/dist/resources/extensions/gsd/auto-tool-tracking.js +18 -0
  42. package/dist/resources/extensions/gsd/auto-unit-closeout.js +45 -21
  43. package/dist/resources/extensions/gsd/auto-unit-tool-scope.js +12 -20
  44. package/dist/resources/extensions/gsd/auto-verification.js +23 -30
  45. package/dist/resources/extensions/gsd/auto-worktree.js +44 -91
  46. package/dist/resources/extensions/gsd/auto.js +41 -14
  47. package/dist/resources/extensions/gsd/blocked-models.js +28 -0
  48. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +29 -8
  49. package/dist/resources/extensions/gsd/bootstrap/dynamic-tools.js +23 -6
  50. package/dist/resources/extensions/gsd/bootstrap/exec-tools.js +2 -2
  51. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +19 -0
  52. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +212 -48
  53. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +303 -77
  54. package/dist/resources/extensions/gsd/branch-patterns.js +2 -0
  55. package/dist/resources/extensions/gsd/browser-daemon-auto-prep.js +83 -0
  56. package/dist/resources/extensions/gsd/browser-evidence.js +8 -2
  57. package/dist/resources/extensions/gsd/captures.js +4 -6
  58. package/dist/resources/extensions/gsd/closeout-wizard.js +92 -0
  59. package/dist/resources/extensions/gsd/commands/context.js +16 -2
  60. package/dist/resources/extensions/gsd/commands-handlers.js +46 -3
  61. package/dist/resources/extensions/gsd/consent-question.js +353 -0
  62. package/dist/resources/extensions/gsd/consent-verdict.js +63 -0
  63. package/dist/resources/extensions/gsd/constants.js +0 -2
  64. package/dist/resources/extensions/gsd/crash-recovery.js +12 -15
  65. package/dist/resources/extensions/gsd/db/queries.js +26 -0
  66. package/dist/resources/extensions/gsd/db-writer.js +8 -17
  67. package/dist/resources/extensions/gsd/dispatch-guard.js +10 -35
  68. package/dist/resources/extensions/gsd/doctor-engine-checks.js +5 -5
  69. package/dist/resources/extensions/gsd/doctor-environment.js +2 -6
  70. package/dist/resources/extensions/gsd/doctor-format.js +9 -6
  71. package/dist/resources/extensions/gsd/doctor-git-checks.js +2 -18
  72. package/dist/resources/extensions/gsd/doctor-runtime-checks.js +13 -15
  73. package/dist/resources/extensions/gsd/engine-hook-contract.js +70 -0
  74. package/dist/resources/extensions/gsd/error-classifier.js +9 -0
  75. package/dist/resources/extensions/gsd/exec-sandbox.js +30 -10
  76. package/dist/resources/extensions/gsd/files.js +33 -19
  77. package/dist/resources/extensions/gsd/gsd-command-home.js +22 -12
  78. package/dist/resources/extensions/gsd/gsd-db.js +2 -1
  79. package/dist/resources/extensions/gsd/guidance.js +158 -0
  80. package/dist/resources/extensions/gsd/guided-flow.js +23 -5
  81. package/dist/resources/extensions/gsd/markdown-renderer.js +10 -0
  82. package/dist/resources/extensions/gsd/mcp-filter.js +2 -19
  83. package/dist/resources/extensions/gsd/mcp-tool-name.js +5 -13
  84. package/dist/resources/extensions/gsd/memory-consolidation-scanner.js +1 -1
  85. package/dist/resources/extensions/gsd/migrate/safety.js +4 -1
  86. package/dist/resources/extensions/gsd/milestone-closeout.js +85 -24
  87. package/dist/resources/extensions/gsd/milestone-planning-persistence.js +2 -2
  88. package/dist/resources/extensions/gsd/milestone-reopen-events.js +3 -5
  89. package/dist/resources/extensions/gsd/notification-store.js +11 -4
  90. package/dist/resources/extensions/gsd/parallel-monitor-overlay.js +6 -4
  91. package/dist/resources/extensions/gsd/parsers-legacy.js +16 -4
  92. package/dist/resources/extensions/gsd/paths.js +27 -0
  93. package/dist/resources/extensions/gsd/pre-execution-checks.js +91 -3
  94. package/dist/resources/extensions/gsd/preferences-models.js +14 -48
  95. package/dist/resources/extensions/gsd/projection-flush.js +7 -0
  96. package/dist/resources/extensions/gsd/prompts/complete-slice.md +3 -3
  97. package/dist/resources/extensions/gsd/prompts/execute-task.md +1 -1
  98. package/dist/resources/extensions/gsd/prompts/plan-milestone.md +1 -1
  99. package/dist/resources/extensions/gsd/prompts/plan-slice.md +2 -2
  100. package/dist/resources/extensions/gsd/prompts/quick-task.md +1 -1
  101. package/dist/resources/extensions/gsd/prompts/reassess-roadmap.md +1 -1
  102. package/dist/resources/extensions/gsd/prompts/refine-slice.md +2 -2
  103. package/dist/resources/extensions/gsd/prompts/replan-slice.md +2 -2
  104. package/dist/resources/extensions/gsd/prompts/research-milestone.md +1 -1
  105. package/dist/resources/extensions/gsd/prompts/research-slice.md +1 -1
  106. package/dist/resources/extensions/gsd/prompts/rewrite-docs.md +1 -1
  107. package/dist/resources/extensions/gsd/prompts/run-uat.md +7 -5
  108. package/dist/resources/extensions/gsd/prompts/system.md +5 -2
  109. package/dist/resources/extensions/gsd/prompts/triage-captures.md +1 -1
  110. package/dist/resources/extensions/gsd/prompts/validate-milestone.md +1 -1
  111. package/dist/resources/extensions/gsd/provider-error-guidance.js +1 -5
  112. package/dist/resources/extensions/gsd/provider-switch-observer.js +1 -1
  113. package/dist/resources/extensions/gsd/publication.js +87 -0
  114. package/dist/resources/extensions/gsd/reactive-graph.js +8 -1
  115. package/dist/resources/extensions/gsd/recovery-classification.js +37 -94
  116. package/dist/resources/extensions/gsd/roadmap-slices.js +25 -3
  117. package/dist/resources/extensions/gsd/safety/destructive-confirmation.js +108 -0
  118. package/dist/resources/extensions/gsd/session-lock.js +1 -1
  119. package/dist/resources/extensions/gsd/state.js +6 -20
  120. package/dist/resources/extensions/gsd/stop-notice.js +57 -0
  121. package/dist/resources/extensions/gsd/tool-contract.js +14 -3
  122. package/dist/resources/extensions/gsd/tool-presentation-plan.js +4 -4
  123. package/dist/resources/extensions/gsd/tool-surface-readiness.js +56 -0
  124. package/dist/resources/extensions/gsd/tools/complete-milestone.js +3 -2
  125. package/dist/resources/extensions/gsd/tools/complete-slice.js +22 -12
  126. package/dist/resources/extensions/gsd/tools/complete-task.js +3 -2
  127. package/dist/resources/extensions/gsd/tools/exec-tool.js +9 -7
  128. package/dist/resources/extensions/gsd/tools/plan-slice.js +14 -8
  129. package/dist/resources/extensions/gsd/tools/plan-task.js +2 -2
  130. package/dist/resources/extensions/gsd/tools/reassess-roadmap.js +2 -2
  131. package/dist/resources/extensions/gsd/tools/reopen-milestone.js +2 -2
  132. package/dist/resources/extensions/gsd/tools/reopen-slice.js +2 -2
  133. package/dist/resources/extensions/gsd/tools/reopen-task.js +2 -2
  134. package/dist/resources/extensions/gsd/tools/replan-slice.js +2 -2
  135. package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +67 -2
  136. package/dist/resources/extensions/gsd/uat-policy.js +42 -16
  137. package/dist/resources/extensions/gsd/unit-closeout.js +138 -0
  138. package/dist/resources/extensions/gsd/unit-context-composer.js +74 -1
  139. package/dist/resources/extensions/gsd/unit-context-manifest.js +4 -27
  140. package/dist/resources/extensions/gsd/unit-registry.js +337 -0
  141. package/dist/resources/extensions/gsd/unit-tool-contracts.js +9 -182
  142. package/dist/resources/extensions/gsd/verdict-parser.js +1 -1
  143. package/dist/resources/extensions/gsd/verification-verdict.js +2 -1
  144. package/dist/resources/extensions/gsd/web-app-uat.js +45 -8
  145. package/dist/resources/extensions/gsd/workflow-event-ledger.js +91 -0
  146. package/dist/resources/extensions/gsd/workflow-event-vocabulary.js +46 -0
  147. package/dist/resources/extensions/gsd/workflow-events.js +6 -18
  148. package/dist/resources/extensions/gsd/workflow-reconcile.js +21 -56
  149. package/dist/resources/extensions/gsd/workflow-tool-surface.js +1 -1
  150. package/dist/resources/extensions/gsd/worktree-git-recovery.js +15 -9
  151. package/dist/resources/extensions/gsd/worktree-lifecycle.js +3 -2
  152. package/dist/resources/extensions/gsd/worktree-manager.js +7 -1
  153. package/dist/resources/extensions/gsd/worktree-root.js +11 -0
  154. package/dist/resources/extensions/gsd/worktree-session-state.js +4 -5
  155. package/dist/resources/extensions/gsd/worktree.js +8 -1
  156. package/dist/resources/extensions/search-the-web/native-search.js +5 -3
  157. package/dist/resources/extensions/shared/browser-contract.js +59 -0
  158. package/dist/resources/extensions/shared/gsd-browser-cli.js +116 -6
  159. package/dist/resources/shared/gsd-browser-path-sync.js +214 -0
  160. package/dist/resources/shared/package-manager-detection.js +1 -1
  161. package/dist/resources/shared/package.json +3 -0
  162. package/dist/resources/skills/create-skill/SKILL.md +3 -0
  163. package/dist/resources/skills/create-skill/references/executable-code.md +1 -1
  164. package/dist/resources/skills/create-skill/references/skill-structure.md +1 -0
  165. package/dist/resources/skills/create-skill/workflows/add-reference.md +8 -3
  166. package/dist/resources/skills/create-skill/workflows/add-script.md +4 -2
  167. package/dist/resources/skills/create-skill/workflows/add-template.md +3 -1
  168. package/dist/resources/skills/create-skill/workflows/add-workflow.md +8 -3
  169. package/dist/resources/skills/create-skill/workflows/upgrade-to-router.md +10 -5
  170. package/dist/resources/skills/create-skill/workflows/verify-skill.md +9 -4
  171. package/dist/resources/skills/spike-wrap-up/SKILL.md +9 -9
  172. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  173. package/dist/update-check.d.ts +2 -0
  174. package/dist/update-check.js +24 -1
  175. package/dist/update-cmd.js +20 -3
  176. package/dist/web/standalone/.next/BUILD_ID +1 -1
  177. package/dist/web/standalone/.next/app-path-routes-manifest.json +13 -13
  178. package/dist/web/standalone/.next/build-manifest.json +3 -3
  179. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  180. package/dist/web/standalone/.next/react-loadable-manifest.json +1 -1
  181. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  182. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  183. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  184. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  185. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  186. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  187. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  188. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  189. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  190. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  191. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  192. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  193. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  194. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  195. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  196. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  197. package/dist/web/standalone/.next/server/app/api/update/route.js +1 -1
  198. package/dist/web/standalone/.next/server/app/api/update/route.js.nft.json +1 -1
  199. package/dist/web/standalone/.next/server/app/index.html +1 -1
  200. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  201. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  202. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  203. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  204. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  205. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  206. package/dist/web/standalone/.next/server/app-paths-manifest.json +13 -13
  207. package/dist/web/standalone/.next/server/chunks/5124.js +1 -1
  208. package/dist/web/standalone/.next/server/chunks/8357.js +2 -2
  209. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  210. package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
  211. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  212. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  213. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  214. package/dist/web/standalone/.next/static/chunks/{796.cf859a427a2cb2ac.js → 796.e0bdc932325d7e03.js} +1 -1
  215. package/dist/web/standalone/.next/static/chunks/{webpack-fbea77b5f9953368.js → webpack-f0285ce91d4ec9ef.js} +1 -1
  216. package/dist/web/standalone/node_modules/node-pty/build/Makefile +1 -1
  217. package/dist/web/standalone/package.json +1 -1
  218. package/dist/worktree-cli.js +3 -6
  219. package/dist/worktree-status-banner.js +7 -15
  220. package/package.json +2 -2
  221. package/packages/cloud-mcp-gateway/package.json +2 -2
  222. package/packages/contracts/dist/rpc.d.ts +1 -0
  223. package/packages/contracts/dist/rpc.d.ts.map +1 -1
  224. package/packages/contracts/dist/rpc.js.map +1 -1
  225. package/packages/contracts/dist/workflow.d.ts +4 -0
  226. package/packages/contracts/dist/workflow.d.ts.map +1 -1
  227. package/packages/contracts/dist/workflow.js.map +1 -1
  228. package/packages/contracts/package.json +1 -1
  229. package/packages/daemon/package.json +4 -4
  230. package/packages/gsd-agent-core/package.json +5 -5
  231. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.d.ts +5 -0
  232. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  233. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.js +5 -0
  234. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.js.map +1 -1
  235. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  236. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js +7 -0
  237. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  238. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/input-controller.d.ts.map +1 -1
  239. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/input-controller.js +8 -1
  240. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/input-controller.js.map +1 -1
  241. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-chat-render.d.ts.map +1 -1
  242. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-chat-render.js +11 -1
  243. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-chat-render.js.map +1 -1
  244. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-auth.d.ts.map +1 -1
  245. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-auth.js +4 -4
  246. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-selectors-auth.js.map +1 -1
  247. package/packages/gsd-agent-modes/dist/modes/rpc/rpc-mode.d.ts.map +1 -1
  248. package/packages/gsd-agent-modes/dist/modes/rpc/rpc-mode.js +3 -1
  249. package/packages/gsd-agent-modes/dist/modes/rpc/rpc-mode.js.map +1 -1
  250. package/packages/gsd-agent-modes/package.json +7 -7
  251. package/packages/mcp-server/dist/cli.js +9 -1
  252. package/packages/mcp-server/dist/cli.js.map +1 -1
  253. package/packages/mcp-server/dist/moonshot-tool-schema.d.ts +29 -0
  254. package/packages/mcp-server/dist/moonshot-tool-schema.d.ts.map +1 -0
  255. package/packages/mcp-server/dist/moonshot-tool-schema.js +50 -0
  256. package/packages/mcp-server/dist/moonshot-tool-schema.js.map +1 -0
  257. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  258. package/packages/mcp-server/dist/server.js +4 -0
  259. package/packages/mcp-server/dist/server.js.map +1 -1
  260. package/packages/mcp-server/dist/workflow-tools.d.ts +26 -18
  261. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  262. package/packages/mcp-server/dist/workflow-tools.js +116 -39
  263. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  264. package/packages/mcp-server/package.json +5 -4
  265. package/packages/native/package.json +1 -1
  266. package/packages/pi-agent-core/dist/harness/env/nodejs.d.ts +1 -0
  267. package/packages/pi-agent-core/dist/harness/env/nodejs.d.ts.map +1 -1
  268. package/packages/pi-agent-core/dist/harness/env/nodejs.js +34 -3
  269. package/packages/pi-agent-core/dist/harness/env/nodejs.js.map +1 -1
  270. package/packages/pi-agent-core/dist/index.d.ts +1 -0
  271. package/packages/pi-agent-core/dist/index.d.ts.map +1 -1
  272. package/packages/pi-agent-core/dist/index.js +3 -0
  273. package/packages/pi-agent-core/dist/index.js.map +1 -1
  274. package/packages/pi-agent-core/package.json +1 -1
  275. package/packages/pi-ai/README.md +1 -0
  276. package/packages/pi-ai/dist/image-models.generated.d.ts +2 -2
  277. package/packages/pi-ai/dist/image-models.generated.js +6 -6
  278. package/packages/pi-ai/dist/image-models.generated.js.map +1 -1
  279. package/packages/pi-ai/dist/index.d.ts +2 -0
  280. package/packages/pi-ai/dist/index.d.ts.map +1 -1
  281. package/packages/pi-ai/dist/index.js +2 -0
  282. package/packages/pi-ai/dist/index.js.map +1 -1
  283. package/packages/pi-ai/dist/models.generated.d.ts +35 -125
  284. package/packages/pi-ai/dist/models.generated.d.ts.map +1 -1
  285. package/packages/pi-ai/dist/models.generated.js +46 -120
  286. package/packages/pi-ai/dist/models.generated.js.map +1 -1
  287. package/packages/pi-ai/dist/providers/anthropic.d.ts.map +1 -1
  288. package/packages/pi-ai/dist/providers/anthropic.js +12 -7
  289. package/packages/pi-ai/dist/providers/anthropic.js.map +1 -1
  290. package/packages/pi-ai/dist/providers/google-shared.d.ts +5 -0
  291. package/packages/pi-ai/dist/providers/google-shared.d.ts.map +1 -1
  292. package/packages/pi-ai/dist/providers/google-shared.js +12 -3
  293. package/packages/pi-ai/dist/providers/google-shared.js.map +1 -1
  294. package/packages/pi-ai/dist/providers/openai-completions.d.ts.map +1 -1
  295. package/packages/pi-ai/dist/providers/openai-completions.js +7 -3
  296. package/packages/pi-ai/dist/providers/openai-completions.js.map +1 -1
  297. package/packages/pi-ai/dist/utils/moonshot-tool-schema.d.ts +9 -0
  298. package/packages/pi-ai/dist/utils/moonshot-tool-schema.d.ts.map +1 -0
  299. package/packages/pi-ai/dist/utils/moonshot-tool-schema.js +34 -0
  300. package/packages/pi-ai/dist/utils/moonshot-tool-schema.js.map +1 -0
  301. package/packages/pi-ai/dist/utils/oauth/github-copilot.d.ts.map +1 -1
  302. package/packages/pi-ai/dist/utils/oauth/github-copilot.js +6 -2
  303. package/packages/pi-ai/dist/utils/oauth/github-copilot.js.map +1 -1
  304. package/packages/pi-ai/package.json +3 -2
  305. package/packages/pi-coding-agent/dist/core/auth-storage.d.ts +2 -2
  306. package/packages/pi-coding-agent/dist/core/auth-storage.d.ts.map +1 -1
  307. package/packages/pi-coding-agent/dist/core/auth-storage.js +19 -13
  308. package/packages/pi-coding-agent/dist/core/auth-storage.js.map +1 -1
  309. package/packages/pi-coding-agent/dist/core/provider-readiness.d.ts.map +1 -1
  310. package/packages/pi-coding-agent/dist/core/provider-readiness.js +13 -6
  311. package/packages/pi-coding-agent/dist/core/provider-readiness.js.map +1 -1
  312. package/packages/pi-coding-agent/dist/core/tools/bash.d.ts +11 -0
  313. package/packages/pi-coding-agent/dist/core/tools/bash.d.ts.map +1 -1
  314. package/packages/pi-coding-agent/dist/core/tools/bash.js +53 -11
  315. package/packages/pi-coding-agent/dist/core/tools/bash.js.map +1 -1
  316. package/packages/pi-coding-agent/dist/index.d.ts +1 -1
  317. package/packages/pi-coding-agent/dist/index.d.ts.map +1 -1
  318. package/packages/pi-coding-agent/dist/index.js +1 -1
  319. package/packages/pi-coding-agent/dist/index.js.map +1 -1
  320. package/packages/pi-coding-agent/dist/utils/shell.d.ts +28 -2
  321. package/packages/pi-coding-agent/dist/utils/shell.d.ts.map +1 -1
  322. package/packages/pi-coding-agent/dist/utils/shell.js +56 -10
  323. package/packages/pi-coding-agent/dist/utils/shell.js.map +1 -1
  324. package/packages/pi-coding-agent/package.json +7 -7
  325. package/packages/pi-tui/dist/tui.d.ts.map +1 -1
  326. package/packages/pi-tui/dist/tui.js +9 -0
  327. package/packages/pi-tui/dist/tui.js.map +1 -1
  328. package/packages/pi-tui/package.json +2 -2
  329. package/packages/rpc-client/package.json +2 -2
  330. package/pkg/package.json +1 -1
  331. package/src/resources/GSD-WORKFLOW.md +5 -4
  332. package/src/resources/extensions/async-jobs/async-bash-cancel.test.ts +360 -0
  333. package/src/resources/extensions/async-jobs/async-bash-tool.ts +33 -56
  334. package/src/resources/extensions/async-jobs/await-tool.test.ts +139 -0
  335. package/src/resources/extensions/async-jobs/await-tool.ts +82 -12
  336. package/src/resources/extensions/async-jobs/index.ts +79 -0
  337. package/src/resources/extensions/async-jobs/job-manager.ts +21 -1
  338. package/src/resources/extensions/bg-shell/bg-shell-command.ts +6 -6
  339. package/src/resources/extensions/bg-shell/bg-shell-tool.ts +10 -6
  340. package/src/resources/extensions/bg-shell/overlay.ts +9 -5
  341. package/src/resources/extensions/bg-shell/process-manager.ts +50 -25
  342. package/src/resources/extensions/bg-shell/readiness-detector.ts +12 -0
  343. package/src/resources/extensions/bg-shell/tests/lifecycle-and-utilities.test.ts +48 -1
  344. package/src/resources/extensions/bg-shell/utilities.ts +3 -0
  345. package/src/resources/extensions/browser-tools/engine/managed-gsd-browser.ts +265 -98
  346. package/src/resources/extensions/browser-tools/engine/selection.ts +90 -4
  347. package/src/resources/extensions/browser-tools/index.ts +71 -13
  348. package/src/resources/extensions/browser-tools/tests/browser-engine-selection.test.mjs +83 -13
  349. package/src/resources/extensions/browser-tools/tests/gsd-browser-launch-config.test.mjs +40 -1
  350. package/src/resources/extensions/browser-tools/tests/managed-gsd-browser-tools.test.mjs +136 -0
  351. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +34 -4
  352. package/src/resources/extensions/gsd/auto/custom-verify-retry-store.ts +21 -3
  353. package/src/resources/extensions/gsd/auto/detect-stuck.ts +32 -9
  354. package/src/resources/extensions/gsd/auto/dispatch-history.ts +152 -0
  355. package/src/resources/extensions/gsd/auto/dispatch-key.ts +39 -0
  356. package/src/resources/extensions/gsd/auto/loop.ts +4 -1
  357. package/src/resources/extensions/gsd/auto/orchestrator.ts +98 -56
  358. package/src/resources/extensions/gsd/auto/phases.ts +65 -26
  359. package/src/resources/extensions/gsd/auto/session.ts +3 -0
  360. package/src/resources/extensions/gsd/auto-direct-dispatch.ts +18 -48
  361. package/src/resources/extensions/gsd/auto-dispatch.ts +48 -61
  362. package/src/resources/extensions/gsd/auto-model-selection.ts +41 -12
  363. package/src/resources/extensions/gsd/auto-post-unit.ts +40 -8
  364. package/src/resources/extensions/gsd/auto-prompts.ts +118 -35
  365. package/src/resources/extensions/gsd/auto-start.ts +42 -21
  366. package/src/resources/extensions/gsd/auto-tool-tracking.ts +19 -0
  367. package/src/resources/extensions/gsd/auto-unit-closeout.ts +83 -28
  368. package/src/resources/extensions/gsd/auto-unit-tool-scope.ts +14 -21
  369. package/src/resources/extensions/gsd/auto-verification.ts +26 -28
  370. package/src/resources/extensions/gsd/auto-worktree.ts +44 -94
  371. package/src/resources/extensions/gsd/auto.ts +52 -16
  372. package/src/resources/extensions/gsd/blocked-models.ts +49 -0
  373. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +37 -10
  374. package/src/resources/extensions/gsd/bootstrap/dynamic-tools.ts +23 -6
  375. package/src/resources/extensions/gsd/bootstrap/exec-tools.ts +2 -2
  376. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +24 -0
  377. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +251 -47
  378. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +352 -84
  379. package/src/resources/extensions/gsd/branch-patterns.ts +3 -0
  380. package/src/resources/extensions/gsd/browser-daemon-auto-prep.ts +108 -0
  381. package/src/resources/extensions/gsd/browser-evidence.ts +18 -2
  382. package/src/resources/extensions/gsd/captures.ts +4 -6
  383. package/src/resources/extensions/gsd/closeout-wizard.ts +102 -0
  384. package/src/resources/extensions/gsd/commands/context.ts +16 -2
  385. package/src/resources/extensions/gsd/commands-handlers.ts +46 -3
  386. package/src/resources/extensions/gsd/consent-question.ts +431 -0
  387. package/src/resources/extensions/gsd/consent-verdict.ts +86 -0
  388. package/src/resources/extensions/gsd/constants.ts +0 -3
  389. package/src/resources/extensions/gsd/crash-recovery.ts +13 -11
  390. package/src/resources/extensions/gsd/db/queries.ts +37 -0
  391. package/src/resources/extensions/gsd/db-writer.ts +11 -19
  392. package/src/resources/extensions/gsd/dispatch-guard.ts +8 -31
  393. package/src/resources/extensions/gsd/doctor-engine-checks.ts +5 -4
  394. package/src/resources/extensions/gsd/doctor-environment.ts +2 -7
  395. package/src/resources/extensions/gsd/doctor-format.ts +12 -7
  396. package/src/resources/extensions/gsd/doctor-git-checks.ts +2 -19
  397. package/src/resources/extensions/gsd/doctor-runtime-checks.ts +13 -15
  398. package/src/resources/extensions/gsd/engine-hook-contract.ts +79 -0
  399. package/src/resources/extensions/gsd/error-classifier.ts +11 -0
  400. package/src/resources/extensions/gsd/exec-sandbox.ts +49 -9
  401. package/src/resources/extensions/gsd/files.ts +33 -12
  402. package/src/resources/extensions/gsd/gsd-command-home.ts +13 -3
  403. package/src/resources/extensions/gsd/gsd-db.ts +4 -3
  404. package/src/resources/extensions/gsd/guidance.ts +217 -0
  405. package/src/resources/extensions/gsd/guided-flow.ts +37 -28
  406. package/src/resources/extensions/gsd/markdown-renderer.ts +11 -0
  407. package/src/resources/extensions/gsd/mcp-filter.ts +2 -23
  408. package/src/resources/extensions/gsd/mcp-tool-name.ts +6 -11
  409. package/src/resources/extensions/gsd/memory-consolidation-scanner.ts +1 -1
  410. package/src/resources/extensions/gsd/migrate/safety.ts +4 -1
  411. package/src/resources/extensions/gsd/milestone-closeout.ts +109 -24
  412. package/src/resources/extensions/gsd/milestone-planning-persistence.ts +2 -2
  413. package/src/resources/extensions/gsd/milestone-reopen-events.ts +3 -6
  414. package/src/resources/extensions/gsd/notification-store.ts +26 -3
  415. package/src/resources/extensions/gsd/parallel-monitor-overlay.ts +6 -4
  416. package/src/resources/extensions/gsd/parsers-legacy.ts +16 -4
  417. package/src/resources/extensions/gsd/paths.ts +33 -0
  418. package/src/resources/extensions/gsd/pre-execution-checks.ts +109 -3
  419. package/src/resources/extensions/gsd/preferences-models.ts +12 -47
  420. package/src/resources/extensions/gsd/projection-flush.ts +20 -0
  421. package/src/resources/extensions/gsd/prompts/complete-slice.md +3 -3
  422. package/src/resources/extensions/gsd/prompts/execute-task.md +1 -1
  423. package/src/resources/extensions/gsd/prompts/plan-milestone.md +1 -1
  424. package/src/resources/extensions/gsd/prompts/plan-slice.md +2 -2
  425. package/src/resources/extensions/gsd/prompts/quick-task.md +1 -1
  426. package/src/resources/extensions/gsd/prompts/reassess-roadmap.md +1 -1
  427. package/src/resources/extensions/gsd/prompts/refine-slice.md +2 -2
  428. package/src/resources/extensions/gsd/prompts/replan-slice.md +2 -2
  429. package/src/resources/extensions/gsd/prompts/research-milestone.md +1 -1
  430. package/src/resources/extensions/gsd/prompts/research-slice.md +1 -1
  431. package/src/resources/extensions/gsd/prompts/rewrite-docs.md +1 -1
  432. package/src/resources/extensions/gsd/prompts/run-uat.md +7 -5
  433. package/src/resources/extensions/gsd/prompts/system.md +5 -2
  434. package/src/resources/extensions/gsd/prompts/triage-captures.md +1 -1
  435. package/src/resources/extensions/gsd/prompts/validate-milestone.md +1 -1
  436. package/src/resources/extensions/gsd/provider-error-guidance.ts +4 -9
  437. package/src/resources/extensions/gsd/provider-switch-observer.ts +1 -1
  438. package/src/resources/extensions/gsd/publication.ts +122 -0
  439. package/src/resources/extensions/gsd/reactive-graph.ts +11 -1
  440. package/src/resources/extensions/gsd/recovery-classification.ts +42 -96
  441. package/src/resources/extensions/gsd/roadmap-slices.ts +28 -3
  442. package/src/resources/extensions/gsd/safety/destructive-confirmation.ts +134 -0
  443. package/src/resources/extensions/gsd/session-lock.ts +1 -1
  444. package/src/resources/extensions/gsd/state.ts +9 -21
  445. package/src/resources/extensions/gsd/stop-notice.ts +75 -0
  446. package/src/resources/extensions/gsd/tests/auto-model-selection.test.ts +97 -1
  447. package/src/resources/extensions/gsd/tests/auto-orchestrator.test.ts +198 -26
  448. package/src/resources/extensions/gsd/tests/auto-remote-session-lock-cleanup.test.ts +65 -3
  449. package/src/resources/extensions/gsd/tests/auto-start-orphan-bootstrap.test.ts +236 -0
  450. package/src/resources/extensions/gsd/tests/blocked-models.test.ts +19 -0
  451. package/src/resources/extensions/gsd/tests/browser-automation-contract-fixture.ts +39 -0
  452. package/src/resources/extensions/gsd/tests/browser-contract.test.ts +44 -0
  453. package/src/resources/extensions/gsd/tests/browser-daemon-auto-prep.test.ts +144 -0
  454. package/src/resources/extensions/gsd/tests/checkout-branch-stash-guard.test.ts +66 -1
  455. package/src/resources/extensions/gsd/tests/clear-stale-autostart.test.ts +22 -0
  456. package/src/resources/extensions/gsd/tests/commands-verdict.test.ts +8 -7
  457. package/src/resources/extensions/gsd/tests/complete-slice-verification-gate.test.ts +42 -0
  458. package/src/resources/extensions/gsd/tests/consent-question.test.ts +351 -0
  459. package/src/resources/extensions/gsd/tests/custom-verify-retry-store.test.ts +67 -0
  460. package/src/resources/extensions/gsd/tests/db-writer.test.ts +15 -4
  461. package/src/resources/extensions/gsd/tests/deep-project-auto-loop.test.ts +10 -10
  462. package/src/resources/extensions/gsd/tests/destructive-confirmation.test.ts +303 -0
  463. package/src/resources/extensions/gsd/tests/discuss-routing-fixes.test.ts +12 -2
  464. package/src/resources/extensions/gsd/tests/dispatch-history.test.ts +273 -0
  465. package/src/resources/extensions/gsd/tests/dispatch-run-uat-browser-tools.test.ts +2 -1
  466. package/src/resources/extensions/gsd/tests/doctor-git-checks-terminal.test.ts +73 -0
  467. package/src/resources/extensions/gsd/tests/dynamic-bash-no-cap.test.ts +132 -0
  468. package/src/resources/extensions/gsd/tests/engine-hook-contract.test.ts +148 -0
  469. package/src/resources/extensions/gsd/tests/exec-graceful-kill.test.ts +193 -0
  470. package/src/resources/extensions/gsd/tests/exec-tool.test.ts +29 -1
  471. package/src/resources/extensions/gsd/tests/extension-bootstrap-isolation.test.ts +35 -1
  472. package/src/resources/extensions/gsd/tests/gsd-command-home.test.ts +120 -0
  473. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +27 -0
  474. package/src/resources/extensions/gsd/tests/guidance.test.ts +148 -0
  475. package/src/resources/extensions/gsd/tests/guided-dispatch-root.test.ts +2 -6
  476. package/src/resources/extensions/gsd/tests/integration/auto-worktree-milestone-merge.test.ts +53 -11
  477. package/src/resources/extensions/gsd/tests/integration/auto-worktree.test.ts +73 -58
  478. package/src/resources/extensions/gsd/tests/integration/gsd-integration-fixture.ts +80 -0
  479. package/src/resources/extensions/gsd/tests/integration/run-uat.test.ts +199 -0
  480. package/src/resources/extensions/gsd/tests/mcp-project-config.test.ts +3 -1
  481. package/src/resources/extensions/gsd/tests/milestone-closeout.test.ts +95 -4
  482. package/src/resources/extensions/gsd/tests/model-unittype-mapping.test.ts +32 -1
  483. package/src/resources/extensions/gsd/tests/notification-store.test.ts +32 -0
  484. package/src/resources/extensions/gsd/tests/oauth-api-model-routing.test.ts +167 -0
  485. package/src/resources/extensions/gsd/tests/parallel-research-dispatch.test.ts +18 -0
  486. package/src/resources/extensions/gsd/tests/parsers-legacy-importers.test.ts +138 -0
  487. package/src/resources/extensions/gsd/tests/phases-terminal-complete-idempotent.test.ts +242 -0
  488. package/src/resources/extensions/gsd/tests/post-exec-retry-bypass.test.ts +63 -2
  489. package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +193 -1
  490. package/src/resources/extensions/gsd/tests/prompt-db.test.ts +124 -6
  491. package/src/resources/extensions/gsd/tests/provider-error-guidance.test.ts +3 -3
  492. package/src/resources/extensions/gsd/tests/publication.test.ts +120 -0
  493. package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +157 -0
  494. package/src/resources/extensions/gsd/tests/roadmap-slices.test.ts +68 -0
  495. package/src/resources/extensions/gsd/tests/runtime-invariant-modules.test.ts +20 -1
  496. package/src/resources/extensions/gsd/tests/stop-notice.test.ts +70 -0
  497. package/src/resources/extensions/gsd/tests/teardown-chdir-failure-clears-registry.test.ts +17 -0
  498. package/src/resources/extensions/gsd/tests/token-tool-gating.test.ts +76 -0
  499. package/src/resources/extensions/gsd/tests/tool-invocation-error-loop-break.test.ts +8 -0
  500. package/src/resources/extensions/gsd/tests/tool-surface-readiness.test.ts +155 -0
  501. package/src/resources/extensions/gsd/tests/tool-unavailable-retry.test.ts +33 -0
  502. package/src/resources/extensions/gsd/tests/transport-gate-double-complete.test.ts +139 -0
  503. package/src/resources/extensions/gsd/tests/uat-policy.test.ts +112 -29
  504. package/src/resources/extensions/gsd/tests/unit-closeout.test.ts +209 -0
  505. package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +67 -2
  506. package/src/resources/extensions/gsd/tests/unit-registry.test.ts +163 -0
  507. package/src/resources/extensions/gsd/tests/uok-audit-unified.test.ts +8 -0
  508. package/src/resources/extensions/gsd/tests/verification-verdict.test.ts +2 -0
  509. package/src/resources/extensions/gsd/tests/web-app-uat.test.ts +44 -1
  510. package/src/resources/extensions/gsd/tests/workflow-events.test.ts +19 -0
  511. package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +2 -2
  512. package/src/resources/extensions/gsd/tests/workflow-reconcile.test.ts +20 -0
  513. package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +275 -40
  514. package/src/resources/extensions/gsd/tests/worktree-teardown-safety.test.ts +22 -0
  515. package/src/resources/extensions/gsd/tests/worktree.test.ts +18 -0
  516. package/src/resources/extensions/gsd/tests/write-gate-seam.test.ts +358 -0
  517. package/src/resources/extensions/gsd/tests/write-gate.test.ts +67 -1
  518. package/src/resources/extensions/gsd/tool-contract.ts +38 -3
  519. package/src/resources/extensions/gsd/tool-presentation-plan.ts +4 -4
  520. package/src/resources/extensions/gsd/tool-surface-readiness.ts +76 -0
  521. package/src/resources/extensions/gsd/tools/complete-milestone.ts +3 -2
  522. package/src/resources/extensions/gsd/tools/complete-slice.ts +22 -12
  523. package/src/resources/extensions/gsd/tools/complete-task.ts +3 -2
  524. package/src/resources/extensions/gsd/tools/exec-tool.ts +8 -7
  525. package/src/resources/extensions/gsd/tools/plan-slice.ts +14 -8
  526. package/src/resources/extensions/gsd/tools/plan-task.ts +2 -2
  527. package/src/resources/extensions/gsd/tools/reassess-roadmap.ts +2 -2
  528. package/src/resources/extensions/gsd/tools/reopen-milestone.ts +2 -2
  529. package/src/resources/extensions/gsd/tools/reopen-slice.ts +2 -2
  530. package/src/resources/extensions/gsd/tools/reopen-task.ts +2 -2
  531. package/src/resources/extensions/gsd/tools/replan-slice.ts +2 -2
  532. package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +81 -2
  533. package/src/resources/extensions/gsd/uat-policy.ts +62 -16
  534. package/src/resources/extensions/gsd/unit-closeout.ts +201 -0
  535. package/src/resources/extensions/gsd/unit-context-composer.ts +111 -1
  536. package/src/resources/extensions/gsd/unit-context-manifest.ts +4 -28
  537. package/src/resources/extensions/gsd/unit-registry.ts +412 -0
  538. package/src/resources/extensions/gsd/unit-tool-contracts.ts +27 -192
  539. package/src/resources/extensions/gsd/verdict-parser.ts +1 -1
  540. package/src/resources/extensions/gsd/verification-verdict.ts +4 -2
  541. package/src/resources/extensions/gsd/web-app-uat.ts +51 -8
  542. package/src/resources/extensions/gsd/workflow-event-ledger.ts +131 -0
  543. package/src/resources/extensions/gsd/workflow-event-vocabulary.ts +59 -0
  544. package/src/resources/extensions/gsd/workflow-events.ts +12 -20
  545. package/src/resources/extensions/gsd/workflow-reconcile.ts +29 -62
  546. package/src/resources/extensions/gsd/workflow-tool-surface.ts +4 -1
  547. package/src/resources/extensions/gsd/worktree-git-recovery.ts +15 -9
  548. package/src/resources/extensions/gsd/worktree-lifecycle.ts +3 -8
  549. package/src/resources/extensions/gsd/worktree-manager.ts +6 -1
  550. package/src/resources/extensions/gsd/worktree-root.ts +12 -0
  551. package/src/resources/extensions/gsd/worktree-session-state.ts +3 -5
  552. package/src/resources/extensions/gsd/worktree.ts +7 -1
  553. package/src/resources/extensions/search-the-web/native-search.ts +5 -3
  554. package/src/resources/extensions/shared/browser-contract.ts +66 -0
  555. package/src/resources/extensions/shared/gsd-browser-cli.ts +141 -6
  556. package/src/resources/shared/gsd-browser-path-sync.ts +273 -0
  557. package/src/resources/shared/package-manager-detection.ts +1 -1
  558. package/src/resources/shared/package.json +3 -0
  559. package/src/resources/skills/create-skill/SKILL.md +3 -0
  560. package/src/resources/skills/create-skill/references/executable-code.md +1 -1
  561. package/src/resources/skills/create-skill/references/skill-structure.md +1 -0
  562. package/src/resources/skills/create-skill/workflows/add-reference.md +8 -3
  563. package/src/resources/skills/create-skill/workflows/add-script.md +4 -2
  564. package/src/resources/skills/create-skill/workflows/add-template.md +3 -1
  565. package/src/resources/skills/create-skill/workflows/add-workflow.md +8 -3
  566. package/src/resources/skills/create-skill/workflows/upgrade-to-router.md +10 -5
  567. package/src/resources/skills/create-skill/workflows/verify-skill.md +9 -4
  568. package/src/resources/skills/spike-wrap-up/SKILL.md +9 -9
  569. package/dist/resources/extensions/gsd/user-input-boundary.js +0 -218
  570. package/dist/resources/skills/gsd-browser/SKILL.md +0 -41
  571. package/src/resources/extensions/gsd/tests/user-input-boundary.test.ts +0 -173
  572. package/src/resources/extensions/gsd/user-input-boundary.ts +0 -216
  573. package/src/resources/skills/gsd-browser/SKILL.md +0 -41
  574. /package/dist/web/standalone/.next/static/{C24pqUd-aru-l0Dp0gLZP → FBNo5cT_chy7YNoAQsU3o}/_buildManifest.js +0 -0
  575. /package/dist/web/standalone/.next/static/{C24pqUd-aru-l0Dp0gLZP → FBNo5cT_chy7YNoAQsU3o}/_ssgManifest.js +0 -0
@@ -21,6 +21,7 @@ import { join } from "node:path";
21
21
  import {
22
22
  extractPackageReferences,
23
23
  checkFilePathConsistency,
24
+ checkPlanningArtifactReferences,
24
25
  checkTaskOrdering,
25
26
  checkInterfaceContracts,
26
27
  checkVerificationCommands,
@@ -973,7 +974,12 @@ describe("runPreExecutionChecks", () => {
973
974
  }
974
975
  });
975
976
 
976
- test("resolves .gsd metadata inputs from canonical project root in worktree mode (#5492)", async () => {
977
+ // Supersedes #5492: .gsd metadata inputs used to be accepted in worktree mode
978
+ // by resolving from the canonical project root. Task inputs are source files
979
+ // only (CONTEXT.md "Task Input") — planning artifacts are now blocked with a
980
+ // removal message regardless of where they resolve. The canonical-root
981
+ // resolution itself remains for merged-but-unsynced source files.
982
+ test("blocks .gsd metadata inputs even when resolvable from canonical project root in worktree mode", async () => {
977
983
  const projectRoot = join(tmpdir(), `pre-exec-project-root-${Date.now()}`);
978
984
  const worktreeRoot = join(tmpdir(), `pre-exec-worktree-root-${Date.now()}`);
979
985
  mkdirSync(join(projectRoot, ".gsd"), { recursive: true });
@@ -990,6 +996,35 @@ describe("runPreExecutionChecks", () => {
990
996
  }),
991
997
  ];
992
998
 
999
+ const result = await runPreExecutionChecks(tasks, worktreeRoot, {
1000
+ canonicalProjectRoot: projectRoot,
1001
+ });
1002
+ assert.equal(result.status, "fail");
1003
+ assert.equal(result.checks.length, 1);
1004
+ assert.ok(result.checks[0].message.includes("preloaded as context"));
1005
+ } finally {
1006
+ rmSync(projectRoot, { recursive: true, force: true });
1007
+ rmSync(worktreeRoot, { recursive: true, force: true });
1008
+ }
1009
+ });
1010
+
1011
+ test("resolves source-file inputs from canonical project root in worktree mode (#5492)", async () => {
1012
+ const projectRoot = join(tmpdir(), `pre-exec-project-root-src-${Date.now()}`);
1013
+ const worktreeRoot = join(tmpdir(), `pre-exec-worktree-root-src-${Date.now()}`);
1014
+ mkdirSync(join(projectRoot, "src"), { recursive: true });
1015
+ mkdirSync(worktreeRoot, { recursive: true });
1016
+ writeFileSync(join(projectRoot, "src", "merged.ts"), "// merged upstream");
1017
+
1018
+ try {
1019
+ const tasks = [
1020
+ createTask({
1021
+ id: "T01",
1022
+ files: [],
1023
+ inputs: ["src/merged.ts"],
1024
+ expected_output: [],
1025
+ }),
1026
+ ];
1027
+
993
1028
  const result = await runPreExecutionChecks(tasks, worktreeRoot, {
994
1029
  canonicalProjectRoot: projectRoot,
995
1030
  });
@@ -2263,3 +2298,160 @@ describe("checkFilePathConsistency quote-wrapped annotation (#3747)", () => {
2263
2298
  );
2264
2299
  });
2265
2300
  });
2301
+
2302
+ // ─── Planning Artifact Reference Tests ───────────────────────────────────────
2303
+
2304
+ describe("checkPlanningArtifactReferences", () => {
2305
+ function withTempDir(run: (tempDir: string) => void): void {
2306
+ const tempDir = join(tmpdir(), `pre-exec-artifact-test-${Date.now()}-${Math.random().toString(36).slice(2)}`);
2307
+ mkdirSync(tempDir, { recursive: true });
2308
+ try {
2309
+ run(tempDir);
2310
+ } finally {
2311
+ rmSync(tempDir, { recursive: true, force: true });
2312
+ }
2313
+ }
2314
+
2315
+ test("blocks .gsd/ paths in inputs even when the file exists", () => {
2316
+ withTempDir((tempDir) => {
2317
+ const artifactDir = join(tempDir, ".gsd", "milestones", "M001");
2318
+ mkdirSync(artifactDir, { recursive: true });
2319
+ writeFileSync(join(artifactDir, "M001-CONTEXT.md"), "# context");
2320
+
2321
+ const tasks = [
2322
+ createTask({ id: "T01", inputs: [".gsd/milestones/M001/M001-CONTEXT.md"] }),
2323
+ ];
2324
+
2325
+ const results = checkPlanningArtifactReferences(tasks, tempDir);
2326
+ assert.equal(results.length, 1);
2327
+ assert.equal(results[0].blocking, true);
2328
+ assert.ok(results[0].message.includes("preloaded as context"));
2329
+ });
2330
+ });
2331
+
2332
+ test("blocks bare artifact names that do not resolve to a real file, with the truthful message", () => {
2333
+ withTempDir((tempDir) => {
2334
+ const tasks = [createTask({ id: "T01", inputs: ["M001-CONTEXT.md"] })];
2335
+
2336
+ const results = checkPlanningArtifactReferences(tasks, tempDir);
2337
+ assert.equal(results.length, 1);
2338
+ assert.ok(results[0].message.includes("preloaded as context"));
2339
+ assert.ok(!results[0].message.includes("doesn't exist"));
2340
+
2341
+ // The consistency check must not double-report the same entry with the
2342
+ // misleading "doesn't exist" message.
2343
+ const consistency = checkFilePathConsistency(tasks, tempDir);
2344
+ assert.deepEqual(consistency, []);
2345
+ });
2346
+ });
2347
+
2348
+ test("allows a bare artifact-style name that exists as a real source file", () => {
2349
+ withTempDir((tempDir) => {
2350
+ writeFileSync(join(tempDir, "M001-CONTEXT.md"), "# tracked source file");
2351
+
2352
+ const tasks = [createTask({ id: "T01", inputs: ["M001-CONTEXT.md"] })];
2353
+ assert.deepEqual(checkPlanningArtifactReferences(tasks, tempDir), []);
2354
+ });
2355
+ });
2356
+
2357
+ test("leaves artifact-style names with directory components to the existence checks", () => {
2358
+ withTempDir((tempDir) => {
2359
+ const tasks = [createTask({ id: "T01", inputs: ["tests/fixtures/M001-CONTEXT.md"] })];
2360
+
2361
+ assert.deepEqual(checkPlanningArtifactReferences(tasks, tempDir), []);
2362
+ const consistency = checkFilePathConsistency(tasks, tempDir);
2363
+ assert.equal(consistency.length, 1);
2364
+ assert.ok(consistency[0].message.includes("doesn't exist"));
2365
+ });
2366
+ });
2367
+
2368
+ test("blocks .planning/ and .audits/ paths in files", () => {
2369
+ withTempDir((tempDir) => {
2370
+ const tasks = [
2371
+ createTask({ id: "T01", files: [".planning/codebase/STACK.md", ".audits/security.md"] }),
2372
+ ];
2373
+
2374
+ const results = checkPlanningArtifactReferences(tasks, tempDir);
2375
+ assert.equal(results.length, 2);
2376
+ assert.ok(results.every((r) => r.blocking && r.message.includes("never task files")));
2377
+ });
2378
+ });
2379
+
2380
+ test("blocks .gsd/ paths in expectedOutput with the workflow-tools message and no ordering duplicate", () => {
2381
+ withTempDir((tempDir) => {
2382
+ const tasks = [
2383
+ createTask({
2384
+ id: "T01",
2385
+ sequence: 0,
2386
+ expected_output: [".gsd/milestones/M001/M001-SUMMARY.md"],
2387
+ }),
2388
+ createTask({
2389
+ id: "T02",
2390
+ sequence: 1,
2391
+ inputs: [".gsd/milestones/M001/M001-SUMMARY.md"],
2392
+ }),
2393
+ ];
2394
+
2395
+ const results = checkPlanningArtifactReferences(tasks, tempDir);
2396
+ assert.equal(results.length, 2);
2397
+ const outputFinding = results.find((r) => r.message.includes("expectedOutput"));
2398
+ assert.ok(outputFinding);
2399
+ assert.ok(outputFinding.message.includes("written by workflow tools"));
2400
+
2401
+ assert.deepEqual(checkTaskOrdering(tasks, tempDir), []);
2402
+ });
2403
+ });
2404
+
2405
+ test("skips prose, sentinels, and runtime-only entries", () => {
2406
+ withTempDir((tempDir) => {
2407
+ const tasks = [
2408
+ createTask({
2409
+ id: "T01",
2410
+ inputs: ["none", "Current enum shape", "server logs (runtime)"],
2411
+ }),
2412
+ ];
2413
+
2414
+ assert.deepEqual(checkPlanningArtifactReferences(tasks, tempDir), []);
2415
+ });
2416
+ });
2417
+
2418
+ test("runPreExecutionChecks surfaces artifact findings as blocking failures", async () => {
2419
+ const tempDir = join(tmpdir(), `pre-exec-artifact-run-${Date.now()}`);
2420
+ mkdirSync(tempDir, { recursive: true });
2421
+ try {
2422
+ const tasks = [createTask({ id: "T01", inputs: ["M001-CONTEXT.md"] })];
2423
+ const result: PreExecutionResult = await runPreExecutionChecks(tasks, tempDir);
2424
+ assert.equal(result.status, "fail");
2425
+ assert.ok(
2426
+ result.checks.some((c) => !c.passed && c.blocking && c.message.includes("preloaded as context")),
2427
+ );
2428
+ } finally {
2429
+ rmSync(tempDir, { recursive: true, force: true });
2430
+ }
2431
+ });
2432
+
2433
+ test("allows a bare artifact-style name that exists only at canonicalProjectRoot (not at worktree basePath)", () => {
2434
+ withTempDir((canonicalRoot) => {
2435
+ withTempDir((worktreeRoot) => {
2436
+ writeFileSync(join(canonicalRoot, "M001-CONTEXT.md"), "# tracked source file");
2437
+ const tasks = [createTask({ id: "T01", inputs: ["M001-CONTEXT.md"] })];
2438
+ assert.deepEqual(
2439
+ checkPlanningArtifactReferences(tasks, worktreeRoot, { canonicalProjectRoot: canonicalRoot }),
2440
+ [],
2441
+ );
2442
+ });
2443
+ });
2444
+ });
2445
+
2446
+ test("blocks .GSD/ paths with mixed casing (case-insensitive metadata dir match)", () => {
2447
+ withTempDir((tempDir) => {
2448
+ const tasks = [
2449
+ createTask({ id: "T01", inputs: [".GSD/milestones/M001/M001-CONTEXT.md"] }),
2450
+ ];
2451
+ const results = checkPlanningArtifactReferences(tasks, tempDir);
2452
+ assert.equal(results.length, 1);
2453
+ assert.equal(results[0].blocking, true);
2454
+ assert.ok(results[0].message.includes("preloaded as context"));
2455
+ });
2456
+ });
2457
+ });
@@ -7,6 +7,9 @@
7
7
 
8
8
  import { describe, test } from 'node:test';
9
9
  import assert from 'node:assert/strict';
10
+ import { mkdirSync, mkdtempSync, rmSync, writeFileSync } from 'node:fs';
11
+ import { join } from 'node:path';
12
+ import { tmpdir } from 'node:os';
10
13
  import {
11
14
  openDatabase,
12
15
  closeDatabase,
@@ -22,6 +25,17 @@ import {
22
25
  formatDecisionsForPrompt,
23
26
  formatRequirementsForPrompt,
24
27
  } from '../context-store.ts';
28
+ import { inlineRequirementsFromDb } from '../auto-prompts.ts';
29
+ import { migrateFromMarkdown } from '../md-importer.ts';
30
+
31
+ function createDbProjectWithRequirements(content: string): { tmpDir: string; gsdDir: string } {
32
+ const tmpDir = mkdtempSync(join(tmpdir(), 'prompt-db-requirements-'));
33
+ const gsdDir = join(tmpDir, '.gsd');
34
+ mkdirSync(gsdDir, { recursive: true });
35
+ writeFileSync(join(gsdDir, 'REQUIREMENTS.md'), content);
36
+ openDatabase(join(gsdDir, 'gsd.db'));
37
+ return { tmpDir, gsdDir };
38
+ }
25
39
 
26
40
  // ═══════════════════════════════════════════════════════════════════════════
27
41
  // prompt-db: DB-aware decisions helper returns scoped content
@@ -182,6 +196,116 @@ console.log('\n=== prompt-db: fallback when DB unavailable ===');
182
196
  assert.deepStrictEqual(formattedReqs, '', 'formatRequirementsForPrompt returns empty for empty input');
183
197
  }
184
198
 
199
+ // ═══════════════════════════════════════════════════════════════════════════
200
+ // prompt-db: DB-aware requirements helper does not fall back on empty DB rows
201
+ // ═══════════════════════════════════════════════════════════════════════════
202
+
203
+ test('prompt-db: inlineRequirementsFromDb returns null instead of full file when DB query is empty', async () => {
204
+ const { tmpDir } = createDbProjectWithRequirements(
205
+ `# Requirements\n\nFULL FILE SHOULD NOT BE INLINED\n\n${'large requirement body\n'.repeat(500)}`,
206
+ );
207
+ try {
208
+ const inlined = await inlineRequirementsFromDb(tmpDir, 'M999', undefined, 'full');
209
+
210
+ assert.equal(inlined, null);
211
+ } finally {
212
+ closeDatabase();
213
+ rmSync(tmpDir, { recursive: true, force: true });
214
+ }
215
+ });
216
+
217
+ test('prompt-db: inlineRequirementsFromDb cascades from empty milestone query to active requirements', async () => {
218
+ const { tmpDir } = createDbProjectWithRequirements(
219
+ '# Requirements\n\nFULL FILE SHOULD NOT BE INLINED\n',
220
+ );
221
+ try {
222
+ insertRequirement({
223
+ id: 'R001',
224
+ class: 'functional',
225
+ status: 'active',
226
+ description: 'active requirement for another milestone',
227
+ why: 'needed',
228
+ source: 'M001',
229
+ primary_owner: 'M001/S01',
230
+ supporting_slices: '',
231
+ validation: 'test',
232
+ notes: '',
233
+ full_content: '',
234
+ superseded_by: null,
235
+ });
236
+ insertRequirement({
237
+ id: 'R002',
238
+ class: 'functional',
239
+ status: 'validated',
240
+ description: 'validated requirement should not be active fallback',
241
+ why: 'already done',
242
+ source: 'M001',
243
+ primary_owner: 'M001/S02',
244
+ supporting_slices: '',
245
+ validation: 'done',
246
+ notes: '',
247
+ full_content: '',
248
+ superseded_by: null,
249
+ });
250
+
251
+ const inlined = await inlineRequirementsFromDb(tmpDir, 'M999', undefined, 'full');
252
+
253
+ assert.ok(inlined);
254
+ assert.match(inlined, /### Requirements/);
255
+ assert.match(inlined, /# Requirements \(compact\)/);
256
+ assert.match(inlined, /R001/);
257
+ assert.match(inlined, /active requirement for another milestone/);
258
+ assert.doesNotMatch(inlined, /R002/);
259
+ assert.doesNotMatch(inlined, /FULL FILE SHOULD NOT BE INLINED/);
260
+ } finally {
261
+ closeDatabase();
262
+ rmSync(tmpDir, { recursive: true, force: true });
263
+ }
264
+ });
265
+
266
+ test('prompt-db: inlineRequirementsFromDb keeps active fallback compact for slice-scoped full calls', async () => {
267
+ const { tmpDir } = createDbProjectWithRequirements(
268
+ '# Requirements\n\nFULL FILE SHOULD NOT BE INLINED\n',
269
+ );
270
+ try {
271
+ insertRequirement({
272
+ id: 'R001',
273
+ class: 'functional',
274
+ status: 'active',
275
+ description: 'active requirement for another milestone',
276
+ why: 'needed',
277
+ source: 'M001',
278
+ primary_owner: 'M001/S01',
279
+ supporting_slices: '',
280
+ validation: 'test',
281
+ notes: '',
282
+ full_content: '',
283
+ superseded_by: null,
284
+ });
285
+
286
+ // Slice-scoped call at inlineLevel "full" with no matching rows for the
287
+ // milestone+slice or milestone-only queries. The cascade should land on
288
+ // status: "active" rows from another milestone, and those project-wide
289
+ // active rows must be formatted compactly even though sliceId is set —
290
+ // otherwise the full per-requirement markdown would be inlined for every
291
+ // active row across the project.
292
+ const inlined = await inlineRequirementsFromDb(tmpDir, 'M999', 'S99', 'full');
293
+
294
+ assert.ok(inlined);
295
+ assert.match(inlined, /### Requirements/);
296
+ assert.match(inlined, /# Requirements \(compact\)/);
297
+ assert.match(inlined, /R001/);
298
+ // Full per-requirement markdown is forbidden here — its hallmark fields
299
+ // (Class:/Status:/Why: as bolded list items) are emitted only by
300
+ // formatRequirementsForPrompt, never by formatRequirementsCompact.
301
+ assert.doesNotMatch(inlined, /\*\*Class:\*\*/);
302
+ assert.doesNotMatch(inlined, /FULL FILE SHOULD NOT BE INLINED/);
303
+ } finally {
304
+ closeDatabase();
305
+ rmSync(tmpDir, { recursive: true, force: true });
306
+ }
307
+ });
308
+
185
309
  // ═══════════════════════════════════════════════════════════════════════════
186
310
  // prompt-db: scoped filtering reduces content vs unscoped
187
311
  // ═══════════════════════════════════════════════════════════════════════════
@@ -316,12 +440,6 @@ console.log('\n=== prompt-db: DB helpers wrapper format matches expected pattern
316
440
  // prompt-db: re-import updates DB when source markdown changes
317
441
  // ═══════════════════════════════════════════════════════════════════════════
318
442
 
319
- import { mkdtempSync, writeFileSync, mkdirSync } from 'node:fs';
320
- import { join } from 'node:path';
321
- import { tmpdir } from 'node:os';
322
- import { migrateFromMarkdown } from '../md-importer.ts';
323
-
324
-
325
443
  describe('prompt-db', () => {
326
444
  test('prompt-db: re-import updates DB when source markdown changes', () => {
327
445
  // Create a temp dir simulating a project with .gsd/DECISIONS.md
@@ -3,10 +3,10 @@ import assert from "node:assert/strict";
3
3
 
4
4
  import { classifyError, isTransient } from "../error-classifier.ts";
5
5
  import {
6
- formatProviderErrorGuidance,
7
6
  resolveProviderErrorGuidance,
8
7
  unitTypeToPrefsPhaseKey,
9
8
  } from "../provider-error-guidance.ts";
9
+ import { formatGuidance } from "../guidance.ts";
10
10
 
11
11
  test("classifyError: Cloud Code Assist 400 invalid argument is model-error", () => {
12
12
  const result = classifyError(
@@ -75,8 +75,8 @@ test("resolveProviderErrorGuidance suggests gemini-3-flash for antigravity pro-h
75
75
  assert.ok(guidance.steps.some((step) => step.includes("fallbacks")));
76
76
  });
77
77
 
78
- test("formatProviderErrorGuidance numbers steps", () => {
79
- const text = formatProviderErrorGuidance({
78
+ test("formatGuidance numbers steps", () => {
79
+ const text = formatGuidance({
80
80
  summary: "Provider error on test/model.",
81
81
  steps: ["Change model", "Run /gsd next"],
82
82
  });
@@ -0,0 +1,120 @@
1
+ // gsd-pi — ADR-034 Publication module tests.
2
+ //
3
+ // Exercises publishMilestone against real local git fixtures (a bare repo as
4
+ // the remote) — no network, no gh. The PR path is only tested up to its
5
+ // non-fatal failure contract, since createDraftPullRequestFromEvidence shells
6
+ // out to gh.
7
+
8
+ import test from "node:test";
9
+ import assert from "node:assert/strict";
10
+ import { rmSync } from "node:fs";
11
+ import { join } from "node:path";
12
+
13
+ import { gitRemoteExists, publishMilestone } from "../publication.ts";
14
+ import { git, makeTempDir, makeTempRepo } from "./test-utils.ts";
15
+
16
+ function makeRepoWithBareRemote(): { repo: string; bare: string; cleanup: () => void } {
17
+ const repo = makeTempRepo("gsd-publication-test-");
18
+ const bare = join(makeTempDir("gsd-publication-remote-"), "remote.git");
19
+ git(repo, "init", "--bare", bare);
20
+ git(repo, "remote", "add", "origin", bare);
21
+ return {
22
+ repo,
23
+ bare,
24
+ cleanup: () => {
25
+ rmSync(repo, { recursive: true, force: true });
26
+ rmSync(bare, { recursive: true, force: true });
27
+ },
28
+ };
29
+ }
30
+
31
+ const NO_PUBLISH_PREFS = { autoPush: false, autoPr: false };
32
+
33
+ function makeRequest(repo: string, prefs: { autoPush: boolean; autoPr: boolean; remote?: string; prTargetBranch?: string }) {
34
+ return {
35
+ basePath: repo,
36
+ milestoneId: "M001",
37
+ milestoneTitle: "Test milestone",
38
+ integrationBranch: "main",
39
+ milestoneBranch: "milestone/M001",
40
+ sliceSummaries: ["### S01\nSlice one"],
41
+ nothingToCommit: false,
42
+ prefs,
43
+ };
44
+ }
45
+
46
+ test("gitRemoteExists distinguishes configured from missing remotes", () => {
47
+ const { repo, cleanup } = makeRepoWithBareRemote();
48
+ try {
49
+ assert.equal(gitRemoteExists(repo, "origin"), true);
50
+ assert.equal(gitRemoteExists(repo, "upstream"), false);
51
+ } finally {
52
+ cleanup();
53
+ }
54
+ });
55
+
56
+ test("auto-push pushes the integration branch to the remote", () => {
57
+ const { repo, bare, cleanup } = makeRepoWithBareRemote();
58
+ try {
59
+ const result = publishMilestone(makeRequest(repo, { autoPush: true, autoPr: false }));
60
+ assert.equal(result.pushed, true);
61
+ assert.equal(result.prCreated, false);
62
+ const remoteHeads = git(repo, "ls-remote", "--heads", bare);
63
+ assert.match(remoteHeads, /refs\/heads\/main/);
64
+ } finally {
65
+ cleanup();
66
+ }
67
+ });
68
+
69
+ test("auto-push is suppressed when auto-PR owns the remote interaction", () => {
70
+ const { repo, bare, cleanup } = makeRepoWithBareRemote();
71
+ try {
72
+ git(repo, "branch", "milestone/M001");
73
+ const result = publishMilestone(makeRequest(repo, { autoPush: true, autoPr: true }));
74
+ // PR creation pushes the milestone branch, then fails non-fatally at gh.
75
+ assert.equal(result.pushed, false);
76
+ assert.equal(result.prCreated, false);
77
+ assert.equal(result.prUrl, undefined);
78
+ const remoteHeads = git(repo, "ls-remote", "--heads", bare);
79
+ assert.match(remoteHeads, /refs\/heads\/milestone\/M001/);
80
+ assert.doesNotMatch(remoteHeads, /refs\/heads\/main/);
81
+ } finally {
82
+ cleanup();
83
+ }
84
+ });
85
+
86
+ test("nothing-to-commit short-circuits all publication", () => {
87
+ const { repo, bare, cleanup } = makeRepoWithBareRemote();
88
+ try {
89
+ const result = publishMilestone({
90
+ ...makeRequest(repo, { autoPush: true, autoPr: false }),
91
+ nothingToCommit: true,
92
+ });
93
+ assert.deepEqual(result, { pushed: false, prCreated: false });
94
+ const remoteHeads = git(repo, "ls-remote", "--heads", bare);
95
+ assert.equal(remoteHeads.trim(), "");
96
+ } finally {
97
+ cleanup();
98
+ }
99
+ });
100
+
101
+ test("missing remote makes publication a silent no-op", () => {
102
+ const { repo, cleanup } = makeRepoWithBareRemote();
103
+ try {
104
+ git(repo, "remote", "remove", "origin");
105
+ const result = publishMilestone(makeRequest(repo, { autoPush: true, autoPr: true }));
106
+ assert.deepEqual(result, { pushed: false, prCreated: false });
107
+ } finally {
108
+ cleanup();
109
+ }
110
+ });
111
+
112
+ test("publication disabled returns the empty result without touching git", () => {
113
+ const { repo, cleanup } = makeRepoWithBareRemote();
114
+ try {
115
+ const result = publishMilestone(makeRequest(repo, NO_PUBLISH_PREFS));
116
+ assert.deepEqual(result, { pushed: false, prCreated: false });
117
+ } finally {
118
+ cleanup();
119
+ }
120
+ });
@@ -14,6 +14,8 @@ import { deriveState, invalidateStateCache } from "../state.ts";
14
14
  import {
15
15
  getPendingGate,
16
16
  loadWriteGateSnapshot,
17
+ markApprovalGateVerified,
18
+ markDepthVerified,
17
19
  resetWriteGateState,
18
20
  setPendingGate,
19
21
  shouldBlockContextArtifactSave,
@@ -892,3 +894,158 @@ test("register-hooks agent_end does not re-arm deferred gate after workflow MCP
892
894
  pendingGateId: null,
893
895
  });
894
896
  });
897
+
898
+ // ── External-engine post-hoc gate replay (write-gate two-process sync) ──────
899
+ // On claude-code-cli, pi ingests the SDK turn's tool blocks after the workflow
900
+ // MCP child already executed them. The depth gate can therefore arrive at
901
+ // tool_execution_start AFTER the child verified it and allowed the CONTEXT
902
+ // save; re-arming then wipes the verification and permanently blocks the
903
+ // discuss→auto handoff.
904
+
905
+ function makeHookHarness(): {
906
+ handlers: Map<string, Array<(event: any, ctx?: any) => Promise<any> | any>>;
907
+ pi: any;
908
+ } {
909
+ const handlers = new Map<string, Array<(event: any, ctx?: any) => Promise<any> | any>>();
910
+ const pi = {
911
+ on(event: string, handler: (event: any, ctx?: any) => Promise<any> | any) {
912
+ const existing = handlers.get(event) ?? [];
913
+ existing.push(handler);
914
+ handlers.set(event, existing);
915
+ },
916
+ } as any;
917
+ return { handlers, pi };
918
+ }
919
+
920
+ test("tool_execution_start does not re-arm a depth gate the MCP child already verified", async (t) => {
921
+ const dir = makeTempDir("posthoc-no-rearm");
922
+ resetWriteGateState(dir);
923
+ t.after(() => {
924
+ resetWriteGateState(dir);
925
+ rmSync(dir, { recursive: true, force: true });
926
+ });
927
+
928
+ const { handlers, pi } = makeHookHarness();
929
+ registerHooks(pi, []);
930
+ const ctx = { cwd: dir, ui: { notify: () => undefined } } as any;
931
+
932
+ // The child verified the gate and allowed the CONTEXT save before the host
933
+ // ever saw the tool block.
934
+ markApprovalGateVerified("depth_verification_M002_confirm", dir);
935
+ markDepthVerified("M002", dir);
936
+
937
+ for (const handler of handlers.get("tool_execution_start") ?? []) {
938
+ await handler({
939
+ toolCallId: "t-depth",
940
+ toolName: "mcp__gsd-workflow__ask_user_questions",
941
+ args: { questions: [{ id: "depth_verification_M002_confirm" }] },
942
+ }, ctx);
943
+ }
944
+
945
+ assert.equal(getPendingGate(dir), null, "post-hoc replay must not re-arm a verified gate");
946
+ const snapshot = loadWriteGateSnapshot(dir);
947
+ assert.ok(
948
+ snapshot.verifiedDepthMilestones.includes("M002"),
949
+ "re-arm wipes verifiedDepthMilestones — the verification must survive the replay",
950
+ );
951
+ assert.equal(
952
+ shouldBlockContextArtifactSave("CONTEXT", "M002", null, dir).block,
953
+ false,
954
+ "context saves must stay unlocked after the replayed tool_execution_start",
955
+ );
956
+ });
957
+
958
+ test("tool_execution_start still arms an unverified depth gate", async (t) => {
959
+ const dir = makeTempDir("live-arm");
960
+ resetWriteGateState(dir);
961
+ t.after(() => {
962
+ resetWriteGateState(dir);
963
+ rmSync(dir, { recursive: true, force: true });
964
+ });
965
+
966
+ const { handlers, pi } = makeHookHarness();
967
+ registerHooks(pi, []);
968
+ const ctx = { cwd: dir, ui: { notify: () => undefined } } as any;
969
+
970
+ for (const handler of handlers.get("tool_execution_start") ?? []) {
971
+ await handler({
972
+ toolCallId: "t-depth",
973
+ toolName: "ask_user_questions",
974
+ args: { questions: [{ id: "depth_verification_M002_confirm" }] },
975
+ }, ctx);
976
+ }
977
+
978
+ assert.equal(getPendingGate(dir), "depth_verification_M002_confirm");
979
+ });
980
+
981
+ test("tool_result verifies the gate from result.structuredContent when event.details is missing", async (t) => {
982
+ const dir = makeTempDir("structured-fallback");
983
+ resetWriteGateState(dir);
984
+ t.after(() => {
985
+ resetWriteGateState(dir);
986
+ rmSync(dir, { recursive: true, force: true });
987
+ });
988
+
989
+ const { handlers, pi } = makeHookHarness();
990
+ registerHooks(pi, []);
991
+ const ctx = { cwd: dir, ui: { notify: () => undefined } } as any;
992
+
993
+ setPendingGate("depth_verification_M002_confirm", dir);
994
+
995
+ const questions = [{
996
+ id: "depth_verification_M002_confirm",
997
+ options: [
998
+ { label: "Yes, you got it (Recommended)" },
999
+ { label: "Not quite — let me clarify" },
1000
+ ],
1001
+ }];
1002
+ for (const handler of handlers.get("tool_result") ?? []) {
1003
+ await handler({
1004
+ toolCallId: "t-depth",
1005
+ toolName: "mcp__gsd-workflow__ask_user_questions",
1006
+ input: { questions },
1007
+ // External MCP relay: no pi-native details, structured payload on result.
1008
+ result: {
1009
+ content: [{ type: "text", text: "answered" }],
1010
+ structuredContent: {
1011
+ questions,
1012
+ response: {
1013
+ answers: {
1014
+ depth_verification_M002_confirm: { selected: "Yes, you got it (Recommended)", notes: "" },
1015
+ },
1016
+ },
1017
+ cancelled: false,
1018
+ },
1019
+ },
1020
+ }, ctx);
1021
+ }
1022
+
1023
+ assert.equal(getPendingGate(dir), null, "structured fallback must clear the pending gate");
1024
+ assert.ok(loadWriteGateSnapshot(dir).verifiedDepthMilestones.includes("M002"));
1025
+ });
1026
+
1027
+ test("tool_result without details or structured content leaves the gate pending without crashing", async (t) => {
1028
+ const dir = makeTempDir("no-details");
1029
+ resetWriteGateState(dir);
1030
+ t.after(() => {
1031
+ resetWriteGateState(dir);
1032
+ rmSync(dir, { recursive: true, force: true });
1033
+ });
1034
+
1035
+ const { handlers, pi } = makeHookHarness();
1036
+ registerHooks(pi, []);
1037
+ const ctx = { cwd: dir, ui: { notify: () => undefined } } as any;
1038
+
1039
+ setPendingGate("depth_verification_M002_confirm", dir);
1040
+
1041
+ for (const handler of handlers.get("tool_result") ?? []) {
1042
+ await handler({
1043
+ toolCallId: "t-depth",
1044
+ toolName: "ask_user_questions",
1045
+ input: { questions: [{ id: "depth_verification_M002_confirm" }] },
1046
+ result: { content: [{ type: "text", text: "answered" }] },
1047
+ }, ctx);
1048
+ }
1049
+
1050
+ assert.equal(getPendingGate(dir), "depth_verification_M002_confirm");
1051
+ });