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

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 (530) hide show
  1. package/dist/cli-web-branch.d.ts +2 -0
  2. package/dist/cli-web-branch.js +9 -2
  3. package/dist/help-text.js +5 -0
  4. package/dist/resources/.managed-resources-content-hash +1 -1
  5. package/dist/resources/extensions/ask-user-questions.js +78 -23
  6. package/dist/resources/extensions/bg-shell/utilities.js +2 -2
  7. package/dist/resources/extensions/claude-code-cli/models.js +9 -0
  8. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +92 -230
  9. package/dist/resources/extensions/claude-code-cli/turn-assembler.js +224 -0
  10. package/dist/resources/extensions/github-sync/templates.js +3 -3
  11. package/dist/resources/extensions/gsd/artifact-projection.js +14 -0
  12. package/dist/resources/extensions/gsd/auto/loop.js +74 -56
  13. package/dist/resources/extensions/gsd/auto/orchestrator.js +142 -15
  14. package/dist/resources/extensions/gsd/auto/phases.js +34 -4
  15. package/dist/resources/extensions/gsd/auto/run-unit.js +2 -1
  16. package/dist/resources/extensions/gsd/auto/session.js +3 -0
  17. package/dist/resources/extensions/gsd/auto-dashboard.js +16 -4
  18. package/dist/resources/extensions/gsd/auto-dispatch.js +6 -5
  19. package/dist/resources/extensions/gsd/auto-model-selection.js +8 -0
  20. package/dist/resources/extensions/gsd/auto-post-unit.js +12 -9
  21. package/dist/resources/extensions/gsd/auto-prompts.js +81 -8
  22. package/dist/resources/extensions/gsd/auto-recovery.js +48 -49
  23. package/dist/resources/extensions/gsd/auto-runtime-state.js +14 -0
  24. package/dist/resources/extensions/gsd/auto-start.js +20 -36
  25. package/dist/resources/extensions/gsd/auto-timers.js +16 -2
  26. package/dist/resources/extensions/gsd/auto-tool-tracking.js +32 -0
  27. package/dist/resources/extensions/gsd/auto-unit-tool-scope.js +4 -29
  28. package/dist/resources/extensions/gsd/auto-verification.js +7 -7
  29. package/dist/resources/extensions/gsd/auto-worktree-repair.js +10 -2
  30. package/dist/resources/extensions/gsd/auto-worktree.js +34 -289
  31. package/dist/resources/extensions/gsd/auto.js +15 -14
  32. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +28 -37
  33. package/dist/resources/extensions/gsd/bootstrap/dynamic-tools.js +20 -43
  34. package/dist/resources/extensions/gsd/bootstrap/query-tools.js +2 -2
  35. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +131 -140
  36. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +89 -8
  37. package/dist/resources/extensions/gsd/captures.js +5 -13
  38. package/dist/resources/extensions/gsd/closeout-consistency-gate.js +21 -4
  39. package/dist/resources/extensions/gsd/closeout-recovery.js +3 -2
  40. package/dist/resources/extensions/gsd/codebase-generator.js +8 -4
  41. package/dist/resources/extensions/gsd/commands/catalog.js +6 -62
  42. package/dist/resources/extensions/gsd/commands/handlers/auto.js +3 -0
  43. package/dist/resources/extensions/gsd/commands-handlers.js +20 -0
  44. package/dist/resources/extensions/gsd/commands-inspect.js +4 -8
  45. package/dist/resources/extensions/gsd/commands-maintenance.js +61 -41
  46. package/dist/resources/extensions/gsd/commands-ship.js +2 -2
  47. package/dist/resources/extensions/gsd/commands-verdict.js +12 -2
  48. package/dist/resources/extensions/gsd/db/engine.js +755 -0
  49. package/dist/resources/extensions/gsd/db/queries.js +372 -0
  50. package/dist/resources/extensions/gsd/db/sql-constants.js +11 -0
  51. package/dist/resources/extensions/gsd/db/writers/cascades.js +194 -0
  52. package/dist/resources/extensions/gsd/db/writers/import-restore.js +182 -0
  53. package/dist/resources/extensions/gsd/db/writers/memory.js +149 -0
  54. package/dist/resources/extensions/gsd/db/writers/reconcile.js +458 -0
  55. package/dist/resources/extensions/gsd/db/writers/status.js +70 -0
  56. package/dist/resources/extensions/gsd/db-workspace.js +103 -0
  57. package/dist/resources/extensions/gsd/delegation-policy.js +2 -10
  58. package/dist/resources/extensions/gsd/discussion-handoff.js +218 -0
  59. package/dist/resources/extensions/gsd/docs/preferences-reference.md +9 -0
  60. package/dist/resources/extensions/gsd/doctor-environment.js +8 -10
  61. package/dist/resources/extensions/gsd/doctor-git-checks.js +4 -3
  62. package/dist/resources/extensions/gsd/doctor-runtime-checks.js +9 -2
  63. package/dist/resources/extensions/gsd/doctor.js +16 -9
  64. package/dist/resources/extensions/gsd/error-classifier.js +1 -1
  65. package/dist/resources/extensions/gsd/git-conflict-state.js +16 -1
  66. package/dist/resources/extensions/gsd/git-service.js +1 -0
  67. package/dist/resources/extensions/gsd/gitignore.js +3 -0
  68. package/dist/resources/extensions/gsd/gsd-db.js +183 -2048
  69. package/dist/resources/extensions/gsd/guided-flow.js +68 -471
  70. package/dist/resources/extensions/gsd/guided-unit-completion.js +225 -0
  71. package/dist/resources/extensions/gsd/markdown-renderer.js +2 -1
  72. package/dist/resources/extensions/gsd/mcp-filter.js +2 -1
  73. package/dist/resources/extensions/gsd/mcp-tool-name.js +26 -0
  74. package/dist/resources/extensions/gsd/md-importer.js +4 -3
  75. package/dist/resources/extensions/gsd/migrate/safety.js +19 -11
  76. package/dist/resources/extensions/gsd/migration-auto-check.js +27 -5
  77. package/dist/resources/extensions/gsd/milestone-closeout-proof.js +72 -0
  78. package/dist/resources/extensions/gsd/milestone-closeout.js +12 -4
  79. package/dist/resources/extensions/gsd/milestone-merge-transaction.js +10 -0
  80. package/dist/resources/extensions/gsd/milestone-planning-persistence.js +156 -0
  81. package/dist/resources/extensions/gsd/milestone-readiness.js +77 -0
  82. package/dist/resources/extensions/gsd/milestone-settlement.js +50 -0
  83. package/dist/resources/extensions/gsd/milestone-validation-evidence.js +73 -0
  84. package/dist/resources/extensions/gsd/milestone-validation-verdict.js +57 -0
  85. package/dist/resources/extensions/gsd/model-cost-table.js +1 -0
  86. package/dist/resources/extensions/gsd/model-router.js +3 -0
  87. package/dist/resources/extensions/gsd/parallel-eligibility.js +3 -6
  88. package/dist/resources/extensions/gsd/parallel-merge.js +14 -11
  89. package/dist/resources/extensions/gsd/parallel-monitor-overlay.js +7 -5
  90. package/dist/resources/extensions/gsd/parallel-orchestrator.js +3 -2
  91. package/dist/resources/extensions/gsd/paths.js +10 -24
  92. package/dist/resources/extensions/gsd/preferences-diagnostics.js +67 -0
  93. package/dist/resources/extensions/gsd/preferences.js +161 -29
  94. package/dist/resources/extensions/gsd/prompts/complete-slice.md +1 -0
  95. package/dist/resources/extensions/gsd/prompts/execute-task.md +2 -0
  96. package/dist/resources/extensions/gsd/prompts/guided-discuss-project.md +3 -1
  97. package/dist/resources/extensions/gsd/prompts/plan-milestone.md +2 -0
  98. package/dist/resources/extensions/gsd/prompts/plan-slice.md +1 -0
  99. package/dist/resources/extensions/gsd/prompts/refine-slice.md +1 -0
  100. package/dist/resources/extensions/gsd/prompts/system.md +1 -1
  101. package/dist/resources/extensions/gsd/provider-payload-policy.js +83 -0
  102. package/dist/resources/extensions/gsd/pull-request-process.js +13 -0
  103. package/dist/resources/extensions/gsd/quality-gate-closure.js +109 -0
  104. package/dist/resources/extensions/gsd/question-transport.js +86 -0
  105. package/dist/resources/extensions/gsd/recovery-classification.js +12 -1
  106. package/dist/resources/extensions/gsd/roadmap-slices.js +8 -2
  107. package/dist/resources/extensions/gsd/safety/evidence-collector.js +37 -4
  108. package/dist/resources/extensions/gsd/safety/evidence-cross-ref.js +7 -2
  109. package/dist/resources/extensions/gsd/safety/file-change-validator.js +10 -0
  110. package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +3 -2
  111. package/dist/resources/extensions/gsd/state-transition-matrix.js +38 -0
  112. package/dist/resources/extensions/gsd/state.js +13 -5
  113. package/dist/resources/extensions/gsd/status-guards.js +56 -8
  114. package/dist/resources/extensions/gsd/templates/plan.md +7 -0
  115. package/dist/resources/extensions/gsd/templates/project.md +1 -0
  116. package/dist/resources/extensions/gsd/templates/roadmap.md +1 -1
  117. package/dist/resources/extensions/gsd/templates/uat.md +5 -1
  118. package/dist/resources/extensions/gsd/tool-contract.js +52 -8
  119. package/dist/resources/extensions/gsd/tool-presentation-plan.js +15 -34
  120. package/dist/resources/extensions/gsd/tool-surface-snapshot.js +17 -0
  121. package/dist/resources/extensions/gsd/tools/complete-slice.js +24 -43
  122. package/dist/resources/extensions/gsd/tools/exec-tool.js +5 -5
  123. package/dist/resources/extensions/gsd/tools/plan-milestone.js +15 -143
  124. package/dist/resources/extensions/gsd/tools/reassess-roadmap.js +39 -0
  125. package/dist/resources/extensions/gsd/tools/reopen-milestone.js +11 -29
  126. package/dist/resources/extensions/gsd/tools/reopen-slice.js +14 -33
  127. package/dist/resources/extensions/gsd/tools/skip-slice.js +18 -36
  128. package/dist/resources/extensions/gsd/tools/validate-milestone.js +15 -78
  129. package/dist/resources/extensions/gsd/uat-policy.js +16 -10
  130. package/dist/resources/extensions/gsd/uat-run.js +9 -14
  131. package/dist/resources/extensions/gsd/undo.js +8 -7
  132. package/dist/resources/extensions/gsd/unit-context-composer.js +40 -20
  133. package/dist/resources/extensions/gsd/unit-runtime.js +3 -2
  134. package/dist/resources/extensions/gsd/unit-tool-contracts.js +2 -1
  135. package/dist/resources/extensions/gsd/user-input-boundary.js +23 -0
  136. package/dist/resources/extensions/gsd/validation-block-guard.js +2 -0
  137. package/dist/resources/extensions/gsd/web-app-uat.js +80 -0
  138. package/dist/resources/extensions/gsd/workflow-mcp.js +15 -102
  139. package/dist/resources/extensions/gsd/workflow-reconcile.js +4 -3
  140. package/dist/resources/extensions/gsd/workflow-tool-surface.js +46 -0
  141. package/dist/resources/extensions/gsd/workspace-git-guard.js +2 -0
  142. package/dist/resources/extensions/gsd/worktree-git-recovery.js +287 -0
  143. package/dist/resources/extensions/gsd/worktree-lifecycle.js +9 -1
  144. package/dist/resources/extensions/gsd/worktree-manager.js +45 -28
  145. package/dist/resources/extensions/gsd/worktree-placement.js +59 -0
  146. package/dist/resources/extensions/gsd/worktree-reentry.js +12 -8
  147. package/dist/resources/extensions/gsd/worktree-root.js +17 -6
  148. package/dist/resources/extensions/gsd/worktree-safety.js +8 -5
  149. package/dist/resources/extensions/gsd/worktree-session-state.js +12 -10
  150. package/dist/resources/extensions/gsd/worktree-state-projection.js +33 -4
  151. package/dist/resources/extensions/gsd/worktree-telemetry.js +12 -0
  152. package/dist/resources/extensions/shared/interview-ui.js +2 -2
  153. package/dist/resources/shared/claude-runtime-floor.js +182 -0
  154. package/dist/resources/skills/gsd-browser/SKILL.md +1 -1
  155. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  156. package/dist/update-cmd.js +20 -0
  157. package/dist/web/standalone/.next/BUILD_ID +1 -1
  158. package/dist/web/standalone/.next/app-path-routes-manifest.json +12 -12
  159. package/dist/web/standalone/.next/build-manifest.json +3 -3
  160. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  161. package/dist/web/standalone/.next/react-loadable-manifest.json +8 -8
  162. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  163. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  164. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  165. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  166. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  167. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  168. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  169. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  170. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  171. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  172. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  173. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  174. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  175. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  176. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  177. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  178. package/dist/web/standalone/.next/server/app/api/boot/route.js.nft.json +1 -1
  179. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js.nft.json +1 -1
  180. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js.nft.json +1 -1
  181. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js.nft.json +1 -1
  182. package/dist/web/standalone/.next/server/app/api/captures/route.js.nft.json +1 -1
  183. package/dist/web/standalone/.next/server/app/api/cleanup/route.js.nft.json +1 -1
  184. package/dist/web/standalone/.next/server/app/api/doctor/route.js.nft.json +1 -1
  185. package/dist/web/standalone/.next/server/app/api/export-data/route.js.nft.json +1 -1
  186. package/dist/web/standalone/.next/server/app/api/files/route.js.nft.json +1 -1
  187. package/dist/web/standalone/.next/server/app/api/forensics/route.js.nft.json +1 -1
  188. package/dist/web/standalone/.next/server/app/api/git/route.js.nft.json +1 -1
  189. package/dist/web/standalone/.next/server/app/api/history/route.js.nft.json +1 -1
  190. package/dist/web/standalone/.next/server/app/api/hooks/route.js.nft.json +1 -1
  191. package/dist/web/standalone/.next/server/app/api/inspect/route.js.nft.json +1 -1
  192. package/dist/web/standalone/.next/server/app/api/knowledge/route.js.nft.json +1 -1
  193. package/dist/web/standalone/.next/server/app/api/live-state/route.js.nft.json +1 -1
  194. package/dist/web/standalone/.next/server/app/api/mcp-connections/route.js.nft.json +1 -1
  195. package/dist/web/standalone/.next/server/app/api/notifications/route.js.nft.json +1 -1
  196. package/dist/web/standalone/.next/server/app/api/onboarding/route.js.nft.json +1 -1
  197. package/dist/web/standalone/.next/server/app/api/projects/route.js.nft.json +1 -1
  198. package/dist/web/standalone/.next/server/app/api/recovery/route.js.nft.json +1 -1
  199. package/dist/web/standalone/.next/server/app/api/session/browser/route.js.nft.json +1 -1
  200. package/dist/web/standalone/.next/server/app/api/session/command/route.js.nft.json +1 -1
  201. package/dist/web/standalone/.next/server/app/api/session/events/route.js.nft.json +1 -1
  202. package/dist/web/standalone/.next/server/app/api/session/manage/route.js.nft.json +1 -1
  203. package/dist/web/standalone/.next/server/app/api/settings-data/route.js.nft.json +1 -1
  204. package/dist/web/standalone/.next/server/app/api/shutdown/route.js.nft.json +1 -1
  205. package/dist/web/standalone/.next/server/app/api/skill-health/route.js.nft.json +1 -1
  206. package/dist/web/standalone/.next/server/app/api/steer/route.js.nft.json +1 -1
  207. package/dist/web/standalone/.next/server/app/api/switch-root/route.js.nft.json +1 -1
  208. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js.nft.json +1 -1
  209. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js.nft.json +1 -1
  210. package/dist/web/standalone/.next/server/app/api/undo/route.js.nft.json +1 -1
  211. package/dist/web/standalone/.next/server/app/api/visualizer/route.js.nft.json +1 -1
  212. package/dist/web/standalone/.next/server/app/index.html +1 -1
  213. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  214. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  215. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  216. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  217. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  218. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  219. package/dist/web/standalone/.next/server/app-paths-manifest.json +12 -12
  220. package/dist/web/standalone/.next/server/chunks/{5047.js → 5942.js} +2 -2
  221. package/dist/web/standalone/.next/server/chunks/8357.js +1 -1
  222. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  223. package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
  224. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  225. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  226. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  227. package/dist/web/standalone/.next/static/chunks/2659.b7b129ee6a769448.js +1 -0
  228. package/dist/web/standalone/.next/static/chunks/2772.bfa657f49f955239.js +1 -0
  229. package/dist/web/standalone/.next/static/chunks/{3616.4113d484a994e411.js → 3616.3c60753b8ffcbd2e.js} +1 -1
  230. package/dist/web/standalone/.next/static/chunks/4283.e4873b058df143a1.js +2 -0
  231. package/dist/web/standalone/.next/static/chunks/5826.a46ecdd1cfe8dabc.js +1 -0
  232. package/dist/web/standalone/.next/static/chunks/796.cf859a427a2cb2ac.js +10 -0
  233. package/dist/web/standalone/.next/static/chunks/8785.2e5a118797fb2dd2.js +1 -0
  234. package/dist/web/standalone/.next/static/chunks/{webpack-dda80a1ef5587410.js → webpack-fbea77b5f9953368.js} +1 -1
  235. package/dist/web/standalone/node_modules/node-pty/build/Makefile +1 -1
  236. package/dist/web-mode.d.ts +2 -0
  237. package/dist/web-mode.js +20 -8
  238. package/dist/worktree-status-banner.js +7 -3
  239. package/package.json +2 -1
  240. package/packages/cloud-mcp-gateway/package.json +2 -2
  241. package/packages/contracts/package.json +1 -1
  242. package/packages/daemon/package.json +4 -4
  243. package/packages/gsd-agent-core/dist/session/agent-session-extensions.d.ts +2 -0
  244. package/packages/gsd-agent-core/dist/session/agent-session-extensions.d.ts.map +1 -1
  245. package/packages/gsd-agent-core/dist/session/agent-session-extensions.js +14 -0
  246. package/packages/gsd-agent-core/dist/session/agent-session-extensions.js.map +1 -1
  247. package/packages/gsd-agent-core/package.json +5 -5
  248. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  249. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.js +3 -0
  250. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.js.map +1 -1
  251. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  252. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js +106 -40
  253. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  254. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-extension-widgets.d.ts.map +1 -1
  255. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-extension-widgets.js +6 -0
  256. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-extension-widgets.js.map +1 -1
  257. package/packages/gsd-agent-modes/package.json +7 -7
  258. package/packages/mcp-server/dist/server.d.ts +10 -0
  259. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  260. package/packages/mcp-server/dist/server.js +8 -0
  261. package/packages/mcp-server/dist/server.js.map +1 -1
  262. package/packages/mcp-server/dist/workflow-tools.d.ts +41 -0
  263. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  264. package/packages/mcp-server/dist/workflow-tools.js +32 -22
  265. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  266. package/packages/mcp-server/package.json +3 -3
  267. package/packages/native/package.json +1 -1
  268. package/packages/pi-agent-core/package.json +1 -1
  269. package/packages/pi-ai/dist/image-models.generated.d.ts +2 -2
  270. package/packages/pi-ai/dist/image-models.generated.js +6 -6
  271. package/packages/pi-ai/dist/image-models.generated.js.map +1 -1
  272. package/packages/pi-ai/dist/models.generated.d.ts +295 -98
  273. package/packages/pi-ai/dist/models.generated.d.ts.map +1 -1
  274. package/packages/pi-ai/dist/models.generated.js +309 -154
  275. package/packages/pi-ai/dist/models.generated.js.map +1 -1
  276. package/packages/pi-ai/package.json +1 -1
  277. package/packages/pi-coding-agent/dist/core/capability-patches.d.ts.map +1 -1
  278. package/packages/pi-coding-agent/dist/core/capability-patches.js +3 -1
  279. package/packages/pi-coding-agent/dist/core/capability-patches.js.map +1 -1
  280. package/packages/pi-coding-agent/package.json +7 -7
  281. package/packages/pi-tui/dist/components/input.js +1 -1
  282. package/packages/pi-tui/dist/components/input.js.map +1 -1
  283. package/packages/pi-tui/dist/keys.d.ts.map +1 -1
  284. package/packages/pi-tui/dist/keys.js +39 -30
  285. package/packages/pi-tui/dist/keys.js.map +1 -1
  286. package/packages/pi-tui/dist/stdin-buffer.d.ts.map +1 -1
  287. package/packages/pi-tui/dist/stdin-buffer.js +22 -0
  288. package/packages/pi-tui/dist/stdin-buffer.js.map +1 -1
  289. package/packages/pi-tui/package.json +2 -2
  290. package/packages/rpc-client/package.json +2 -2
  291. package/pkg/package.json +1 -1
  292. package/src/resources/extensions/ask-user-questions.ts +87 -24
  293. package/src/resources/extensions/bg-shell/utilities.ts +2 -2
  294. package/src/resources/extensions/claude-code-cli/models.ts +9 -0
  295. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +114 -281
  296. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +268 -0
  297. package/src/resources/extensions/claude-code-cli/turn-assembler.ts +287 -0
  298. package/src/resources/extensions/github-sync/templates.ts +3 -3
  299. package/src/resources/extensions/github-sync/tests/templates.test.ts +2 -2
  300. package/src/resources/extensions/gsd/artifact-projection.ts +31 -0
  301. package/src/resources/extensions/gsd/auto/contracts.ts +32 -2
  302. package/src/resources/extensions/gsd/auto/loop-deps.ts +3 -1
  303. package/src/resources/extensions/gsd/auto/loop.ts +83 -61
  304. package/src/resources/extensions/gsd/auto/orchestrator.ts +164 -17
  305. package/src/resources/extensions/gsd/auto/phases.ts +45 -4
  306. package/src/resources/extensions/gsd/auto/run-unit.ts +2 -1
  307. package/src/resources/extensions/gsd/auto/session.ts +4 -0
  308. package/src/resources/extensions/gsd/auto-dashboard.ts +18 -4
  309. package/src/resources/extensions/gsd/auto-dispatch.ts +20 -7
  310. package/src/resources/extensions/gsd/auto-model-selection.ts +8 -0
  311. package/src/resources/extensions/gsd/auto-post-unit.ts +16 -8
  312. package/src/resources/extensions/gsd/auto-prompts.ts +107 -9
  313. package/src/resources/extensions/gsd/auto-recovery.ts +50 -50
  314. package/src/resources/extensions/gsd/auto-runtime-state.ts +26 -0
  315. package/src/resources/extensions/gsd/auto-start.ts +25 -34
  316. package/src/resources/extensions/gsd/auto-timers.ts +16 -2
  317. package/src/resources/extensions/gsd/auto-tool-tracking.ts +35 -0
  318. package/src/resources/extensions/gsd/auto-unit-tool-scope.ts +9 -30
  319. package/src/resources/extensions/gsd/auto-verification.ts +7 -8
  320. package/src/resources/extensions/gsd/auto-worktree-repair.ts +13 -2
  321. package/src/resources/extensions/gsd/auto-worktree.ts +53 -306
  322. package/src/resources/extensions/gsd/auto.ts +27 -17
  323. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +29 -37
  324. package/src/resources/extensions/gsd/bootstrap/dynamic-tools.ts +20 -43
  325. package/src/resources/extensions/gsd/bootstrap/query-tools.ts +2 -2
  326. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +147 -153
  327. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +132 -6
  328. package/src/resources/extensions/gsd/captures.ts +5 -14
  329. package/src/resources/extensions/gsd/closeout-consistency-gate.ts +27 -5
  330. package/src/resources/extensions/gsd/closeout-recovery.ts +2 -1
  331. package/src/resources/extensions/gsd/codebase-generator.ts +9 -5
  332. package/src/resources/extensions/gsd/commands/catalog.ts +6 -68
  333. package/src/resources/extensions/gsd/commands/handlers/auto.ts +3 -0
  334. package/src/resources/extensions/gsd/commands-handlers.ts +18 -0
  335. package/src/resources/extensions/gsd/commands-inspect.ts +7 -8
  336. package/src/resources/extensions/gsd/commands-maintenance.ts +74 -40
  337. package/src/resources/extensions/gsd/commands-ship.ts +2 -2
  338. package/src/resources/extensions/gsd/commands-verdict.ts +19 -2
  339. package/src/resources/extensions/gsd/db/engine.ts +809 -0
  340. package/src/resources/extensions/gsd/db/queries.ts +453 -0
  341. package/src/resources/extensions/gsd/db/sql-constants.ts +12 -0
  342. package/src/resources/extensions/gsd/db/writers/cascades.ts +237 -0
  343. package/src/resources/extensions/gsd/db/writers/import-restore.ts +310 -0
  344. package/src/resources/extensions/gsd/db/writers/memory.ts +220 -0
  345. package/src/resources/extensions/gsd/db/writers/reconcile.ts +500 -0
  346. package/src/resources/extensions/gsd/db/writers/status.ts +88 -0
  347. package/src/resources/extensions/gsd/db-workspace.ts +170 -0
  348. package/src/resources/extensions/gsd/delegation-policy.ts +3 -11
  349. package/src/resources/extensions/gsd/discussion-handoff.ts +276 -0
  350. package/src/resources/extensions/gsd/docs/preferences-reference.md +9 -0
  351. package/src/resources/extensions/gsd/doctor-environment.ts +8 -11
  352. package/src/resources/extensions/gsd/doctor-git-checks.ts +3 -3
  353. package/src/resources/extensions/gsd/doctor-runtime-checks.ts +10 -3
  354. package/src/resources/extensions/gsd/doctor.ts +15 -5
  355. package/src/resources/extensions/gsd/error-classifier.ts +1 -1
  356. package/src/resources/extensions/gsd/git-conflict-state.ts +17 -1
  357. package/src/resources/extensions/gsd/git-service.ts +1 -0
  358. package/src/resources/extensions/gsd/gitignore.ts +3 -0
  359. package/src/resources/extensions/gsd/gsd-db.ts +185 -2373
  360. package/src/resources/extensions/gsd/guided-flow.ts +81 -561
  361. package/src/resources/extensions/gsd/guided-unit-completion.ts +275 -0
  362. package/src/resources/extensions/gsd/markdown-renderer.ts +2 -1
  363. package/src/resources/extensions/gsd/mcp-filter.ts +2 -1
  364. package/src/resources/extensions/gsd/mcp-tool-name.ts +35 -0
  365. package/src/resources/extensions/gsd/md-importer.ts +3 -3
  366. package/src/resources/extensions/gsd/migrate/safety.ts +17 -9
  367. package/src/resources/extensions/gsd/migration-auto-check.ts +30 -5
  368. package/src/resources/extensions/gsd/milestone-closeout-proof.ts +131 -0
  369. package/src/resources/extensions/gsd/milestone-closeout.ts +12 -4
  370. package/src/resources/extensions/gsd/milestone-merge-transaction.ts +47 -0
  371. package/src/resources/extensions/gsd/milestone-planning-persistence.ts +224 -0
  372. package/src/resources/extensions/gsd/milestone-readiness.ts +125 -0
  373. package/src/resources/extensions/gsd/milestone-settlement.ts +81 -0
  374. package/src/resources/extensions/gsd/milestone-validation-evidence.ts +95 -0
  375. package/src/resources/extensions/gsd/milestone-validation-verdict.ts +80 -0
  376. package/src/resources/extensions/gsd/model-cost-table.ts +1 -0
  377. package/src/resources/extensions/gsd/model-router.ts +3 -0
  378. package/src/resources/extensions/gsd/parallel-eligibility.ts +4 -5
  379. package/src/resources/extensions/gsd/parallel-merge.ts +12 -9
  380. package/src/resources/extensions/gsd/parallel-monitor-overlay.ts +6 -5
  381. package/src/resources/extensions/gsd/parallel-orchestrator.ts +6 -2
  382. package/src/resources/extensions/gsd/paths.ts +9 -22
  383. package/src/resources/extensions/gsd/preferences-diagnostics.ts +98 -0
  384. package/src/resources/extensions/gsd/preferences-types.ts +16 -0
  385. package/src/resources/extensions/gsd/preferences.ts +191 -28
  386. package/src/resources/extensions/gsd/prompts/complete-slice.md +1 -0
  387. package/src/resources/extensions/gsd/prompts/execute-task.md +2 -0
  388. package/src/resources/extensions/gsd/prompts/guided-discuss-project.md +3 -1
  389. package/src/resources/extensions/gsd/prompts/plan-milestone.md +2 -0
  390. package/src/resources/extensions/gsd/prompts/plan-slice.md +1 -0
  391. package/src/resources/extensions/gsd/prompts/refine-slice.md +1 -0
  392. package/src/resources/extensions/gsd/prompts/system.md +1 -1
  393. package/src/resources/extensions/gsd/provider-payload-policy.ts +140 -0
  394. package/src/resources/extensions/gsd/pull-request-process.ts +41 -0
  395. package/src/resources/extensions/gsd/quality-gate-closure.ts +140 -0
  396. package/src/resources/extensions/gsd/question-transport.ts +138 -0
  397. package/src/resources/extensions/gsd/recovery-classification.ts +14 -1
  398. package/src/resources/extensions/gsd/roadmap-slices.ts +8 -2
  399. package/src/resources/extensions/gsd/safety/evidence-collector.ts +36 -4
  400. package/src/resources/extensions/gsd/safety/evidence-cross-ref.ts +7 -2
  401. package/src/resources/extensions/gsd/safety/file-change-validator.ts +14 -0
  402. package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +6 -2
  403. package/src/resources/extensions/gsd/state-transition-matrix.ts +42 -0
  404. package/src/resources/extensions/gsd/state.ts +15 -5
  405. package/src/resources/extensions/gsd/status-guards.ts +59 -8
  406. package/src/resources/extensions/gsd/templates/plan.md +7 -0
  407. package/src/resources/extensions/gsd/templates/project.md +1 -0
  408. package/src/resources/extensions/gsd/templates/roadmap.md +1 -1
  409. package/src/resources/extensions/gsd/templates/uat.md +5 -1
  410. package/src/resources/extensions/gsd/tests/ask-user-questions-render.test.ts +92 -0
  411. package/src/resources/extensions/gsd/tests/auto-dashboard.test.ts +29 -1
  412. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +444 -5
  413. package/src/resources/extensions/gsd/tests/auto-milestone-target.test.ts +23 -0
  414. package/src/resources/extensions/gsd/tests/auto-model-selection-tool-poisoning.test.ts +18 -0
  415. package/src/resources/extensions/gsd/tests/auto-orchestrator.test.ts +133 -4
  416. package/src/resources/extensions/gsd/tests/auto-paused-ui-cleanup.test.ts +3 -1
  417. package/src/resources/extensions/gsd/tests/auto-post-unit-evidence-crossref-4909.test.ts +46 -0
  418. package/src/resources/extensions/gsd/tests/auto-runtime-state.test.ts +34 -0
  419. package/src/resources/extensions/gsd/tests/auto-worktree-registry.test.ts +2 -2
  420. package/src/resources/extensions/gsd/tests/auto-worktree-repair.test.ts +4 -2
  421. package/src/resources/extensions/gsd/tests/canonical-milestone-root.test.ts +20 -0
  422. package/src/resources/extensions/gsd/tests/clear-stale-autostart.test.ts +22 -0
  423. package/src/resources/extensions/gsd/tests/codebase-generator.test.ts +22 -0
  424. package/src/resources/extensions/gsd/tests/commands-dispatcher-workspace-git.test.ts +11 -0
  425. package/src/resources/extensions/gsd/tests/commands-verdict.test.ts +38 -1
  426. package/src/resources/extensions/gsd/tests/dispatch-complete-milestone-guard.test.ts +34 -3
  427. package/src/resources/extensions/gsd/tests/dispatch-run-uat-browser-tools.test.ts +88 -0
  428. package/src/resources/extensions/gsd/tests/doctor-scope-db-unavailable.test.ts +18 -0
  429. package/src/resources/extensions/gsd/tests/evidence-xref-gsd-exec.test.ts +157 -0
  430. package/src/resources/extensions/gsd/tests/execute-task-rendering.test.ts +1 -0
  431. package/src/resources/extensions/gsd/tests/file-change-validator.test.ts +33 -1
  432. package/src/resources/extensions/gsd/tests/fixtures/pr-body/swarm-lane-no-blockers.md +1 -5
  433. package/src/resources/extensions/gsd/tests/fixtures/pr-body/swarm-lane-with-blockers.md +1 -5
  434. package/src/resources/extensions/gsd/tests/gate-state-canonicalization.test.ts +48 -1
  435. package/src/resources/extensions/gsd/tests/integration/auto-worktree-milestone-merge.test.ts +5 -4
  436. package/src/resources/extensions/gsd/tests/integration/auto-worktree.test.ts +1 -1
  437. package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +3 -2
  438. package/src/resources/extensions/gsd/tests/mcp-tool-name.test.ts +34 -0
  439. package/src/resources/extensions/gsd/tests/migration-auto-check.test.ts +143 -1
  440. package/src/resources/extensions/gsd/tests/milestone-closeout-proof.test.ts +99 -0
  441. package/src/resources/extensions/gsd/tests/milestone-closeout.test.ts +25 -0
  442. package/src/resources/extensions/gsd/tests/milestone-merge-transaction.test.ts +46 -0
  443. package/src/resources/extensions/gsd/tests/milestone-readiness.test.ts +65 -0
  444. package/src/resources/extensions/gsd/tests/milestone-validation-evidence.test.ts +41 -0
  445. package/src/resources/extensions/gsd/tests/milestone-validation-verdict.test.ts +55 -0
  446. package/src/resources/extensions/gsd/tests/plan-milestone.test.ts +45 -0
  447. package/src/resources/extensions/gsd/tests/plan-slice-prompt.test.ts +2 -0
  448. package/src/resources/extensions/gsd/tests/planning-crossval.test.ts +45 -0
  449. package/src/resources/extensions/gsd/tests/preferences-diagnostics.test.ts +67 -0
  450. package/src/resources/extensions/gsd/tests/preferences.test.ts +183 -0
  451. package/src/resources/extensions/gsd/tests/prompt-contracts.test.ts +46 -0
  452. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +9 -0
  453. package/src/resources/extensions/gsd/tests/provider-payload-policy.test.ts +165 -0
  454. package/src/resources/extensions/gsd/tests/pull-request-process.test.ts +47 -0
  455. package/src/resources/extensions/gsd/tests/recovery-classification-illegal-transition.test.ts +30 -0
  456. package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +185 -1
  457. package/src/resources/extensions/gsd/tests/roadmap-parse-regression.test.ts +40 -0
  458. package/src/resources/extensions/gsd/tests/runtime-invariant-modules.test.ts +25 -1
  459. package/src/resources/extensions/gsd/tests/safety-harness-false-positives.test.ts +38 -0
  460. package/src/resources/extensions/gsd/tests/session-start-footer.test.ts +80 -0
  461. package/src/resources/extensions/gsd/tests/session-switch-clears-pending-autostart.test.ts +108 -0
  462. package/src/resources/extensions/gsd/tests/single-writer-invariant.test.ts +144 -7
  463. package/src/resources/extensions/gsd/tests/stale-queued-milestone.test.ts +27 -0
  464. package/src/resources/extensions/gsd/tests/start-auto-detached.test.ts +2 -1
  465. package/src/resources/extensions/gsd/tests/state-transition-matrix.test.ts +36 -0
  466. package/src/resources/extensions/gsd/tests/status-guards.test.ts +38 -0
  467. package/src/resources/extensions/gsd/tests/tool-availability-audit.test.ts +35 -0
  468. package/src/resources/extensions/gsd/tests/tool-naming.test.ts +35 -42
  469. package/src/resources/extensions/gsd/tests/uat-policy.test.ts +23 -0
  470. package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +47 -0
  471. package/src/resources/extensions/gsd/tests/user-input-boundary.test.ts +86 -1
  472. package/src/resources/extensions/gsd/tests/validate-milestone-stuck-guard.test.ts +39 -0
  473. package/src/resources/extensions/gsd/tests/web-app-uat.test.ts +150 -0
  474. package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +126 -9
  475. package/src/resources/extensions/gsd/tests/workspace-git-preflight.test.ts +15 -0
  476. package/src/resources/extensions/gsd/tests/worktree-lifecycle.test.ts +41 -4
  477. package/src/resources/extensions/gsd/tests/worktree-manager.test.ts +43 -1
  478. package/src/resources/extensions/gsd/tests/worktree-placement.test.ts +113 -0
  479. package/src/resources/extensions/gsd/tests/worktree-projection-writers.test.ts +1 -1
  480. package/src/resources/extensions/gsd/tests/worktree-reentry.test.ts +1 -1
  481. package/src/resources/extensions/gsd/tests/worktree-safety.test.ts +3 -1
  482. package/src/resources/extensions/gsd/tests/worktree-symlink-removal.test.ts +12 -6
  483. package/src/resources/extensions/gsd/tests/worktree-teardown-safety.test.ts +2 -2
  484. package/src/resources/extensions/gsd/tests/worktree-telemetry.test.ts +22 -0
  485. package/src/resources/extensions/gsd/tests/write-gate.test.ts +121 -0
  486. package/src/resources/extensions/gsd/tool-contract.ts +86 -8
  487. package/src/resources/extensions/gsd/tool-presentation-plan.ts +16 -33
  488. package/src/resources/extensions/gsd/tool-surface-snapshot.ts +47 -0
  489. package/src/resources/extensions/gsd/tools/complete-slice.ts +23 -58
  490. package/src/resources/extensions/gsd/tools/exec-tool.ts +5 -5
  491. package/src/resources/extensions/gsd/tools/plan-milestone.ts +19 -160
  492. package/src/resources/extensions/gsd/tools/reassess-roadmap.ts +43 -0
  493. package/src/resources/extensions/gsd/tools/reopen-milestone.ts +11 -38
  494. package/src/resources/extensions/gsd/tools/reopen-slice.ts +14 -42
  495. package/src/resources/extensions/gsd/tools/skip-slice.ts +18 -44
  496. package/src/resources/extensions/gsd/tools/validate-milestone.ts +25 -84
  497. package/src/resources/extensions/gsd/uat-policy.ts +19 -10
  498. package/src/resources/extensions/gsd/uat-run.ts +10 -14
  499. package/src/resources/extensions/gsd/undo.ts +9 -8
  500. package/src/resources/extensions/gsd/unit-context-composer.ts +85 -20
  501. package/src/resources/extensions/gsd/unit-runtime.ts +3 -2
  502. package/src/resources/extensions/gsd/unit-tool-contracts.ts +2 -1
  503. package/src/resources/extensions/gsd/user-input-boundary.ts +18 -0
  504. package/src/resources/extensions/gsd/validation-block-guard.ts +2 -0
  505. package/src/resources/extensions/gsd/web-app-uat.ts +101 -0
  506. package/src/resources/extensions/gsd/workflow-mcp.ts +22 -110
  507. package/src/resources/extensions/gsd/workflow-reconcile.ts +3 -3
  508. package/src/resources/extensions/gsd/workflow-tool-surface.ts +73 -0
  509. package/src/resources/extensions/gsd/workspace-git-guard.ts +1 -0
  510. package/src/resources/extensions/gsd/worktree-git-recovery.ts +308 -0
  511. package/src/resources/extensions/gsd/worktree-lifecycle.ts +17 -17
  512. package/src/resources/extensions/gsd/worktree-manager.ts +47 -28
  513. package/src/resources/extensions/gsd/worktree-placement.ts +63 -0
  514. package/src/resources/extensions/gsd/worktree-reentry.ts +10 -7
  515. package/src/resources/extensions/gsd/worktree-root.ts +17 -6
  516. package/src/resources/extensions/gsd/worktree-safety.ts +8 -5
  517. package/src/resources/extensions/gsd/worktree-session-state.ts +12 -10
  518. package/src/resources/extensions/gsd/worktree-state-projection.ts +55 -7
  519. package/src/resources/extensions/gsd/worktree-telemetry.ts +16 -0
  520. package/src/resources/extensions/shared/interview-ui.ts +15 -2
  521. package/src/resources/shared/claude-runtime-floor.ts +248 -0
  522. package/src/resources/skills/gsd-browser/SKILL.md +1 -1
  523. package/dist/web/standalone/.next/static/chunks/2659.feb6499ca863ebfc.js +0 -1
  524. package/dist/web/standalone/.next/static/chunks/2772.151789db0edea835.js +0 -1
  525. package/dist/web/standalone/.next/static/chunks/4283.10a065467b5340d8.js +0 -2
  526. package/dist/web/standalone/.next/static/chunks/5826.960dc4634cc9b0d3.js +0 -1
  527. package/dist/web/standalone/.next/static/chunks/796.46f811c0fac23aab.js +0 -10
  528. package/dist/web/standalone/.next/static/chunks/8785.d32f7a61f55c1600.js +0 -1
  529. /package/dist/web/standalone/.next/static/{Qbr81pQ-pbQXP4bq2VXLv → C24pqUd-aru-l0Dp0gLZP}/_buildManifest.js +0 -0
  530. /package/dist/web/standalone/.next/static/{Qbr81pQ-pbQXP4bq2VXLv → C24pqUd-aru-l0Dp0gLZP}/_ssgManifest.js +0 -0
@@ -36,6 +36,8 @@ After the opening answer, classify project shape as **`simple`** or **`complex`*
36
36
 
37
37
  Persist the verdict through `gsd_summary_save` with `artifact_type: "PROJECT"` into `## Project Shape`; downstream `discuss-requirements`, `discuss-milestone`, and `discuss-slice` read the rendered projection.
38
38
 
39
+ When the project is browser-facing (web app, SPA, static site with UI, Next/React/Vue, etc.), also record **`Web stack:`** under `## Project Shape` (framework + whether Playwright or browser UAT is expected). Downstream planning uses this to choose `browser-executable` / `runtime-executable` UAT instead of deferring UI proof to humans.
40
+
39
41
  ### Before deeper rounds
40
42
 
41
43
  Ground your questions in the **Preparation Context snapshot above** (stack, structure, greenfield/brownfield and framework signals) plus any prior `.planning/` or `.gsd/` artifacts — those are authoritative. **Do not survey the codebase** with `rg`/`find`/`scout` before asking; read a specific file only when a question's answer genuinely hinges on it.
@@ -103,7 +105,7 @@ Once the user confirms depth:
103
105
 
104
106
  1. Use the **Project** output template (inlined above).
105
107
  2. Call `gsd_summary_save` with `artifact_type: "PROJECT"` and full project markdown as `content`; omit `milestone_id`. The tool persists the DB-backed PROJECT artifact and renders `.gsd/PROJECT.md`. Preserve the user's terms and framing.
106
- 3. The `## Project Shape` section MUST contain `**Complexity:** simple` or `**Complexity:** complex` (matching the verdict you announced) plus a one-line `**Why:**` rationale. Downstream stages read this line.
108
+ 3. The `## Project Shape` section MUST contain `**Complexity:** simple` or `**Complexity:** complex` (matching the verdict you announced) plus a one-line `**Why:**` rationale. For browser-facing projects, also include `**Web stack:**` (framework and expected browser/Playwright UAT approach). Downstream stages read these lines.
107
109
  4. The `## Capability Contract` section MUST reference `.gsd/REQUIREMENTS.md` — that file does not yet exist; the next stage (`discuss-requirements`) will produce it.
108
110
  5. The `## Milestone Sequence` MUST list at least M001 with title and one-liner. Subsequent milestones may be listed as known intents; they will be elaborated in their own discuss-milestone stages.
109
111
  6. Do NOT use `artifact_type: "CONTEXT"` and do NOT pass `milestone_id: "PROJECT"`; that creates a fake milestone named PROJECT.
@@ -47,6 +47,8 @@ If milestone research is inlined, trust it and skip redundant exploration. If fi
47
47
 
48
48
  Narrate decomposition reasoning in complete sentences: grouping, risk order, verification strategy.
49
49
 
50
+ **Web apps:** when inlined Web App UAT guidance is present, set milestone `Verification Classes` → UAT to browser-observable acceptance (Playwright spec or `browser_*` checks). Order an early slice to add Playwright smoke scaffolding when the dependency is missing.
51
+
50
52
  Then:
51
53
  1. Use the **Roadmap** output template from the inlined context above
52
54
  2. {{skillActivation}}
@@ -41,6 +41,7 @@ If slice research is inlined, trust its architectural findings, but verify every
41
41
  3. Use the inlined Output Template sections already present in this prompt. Do not read template files from disk.
42
42
  4. {{skillActivation}} Record expected executor skills in each task plan's `skills_used` frontmatter.
43
43
  5. Define slice verification before tasks. Non-trivial slices need real tests or executable assertions; boundary contracts need contract-exercising checks. Tests must not read .gitignore/gitignored paths such as `.gsd/`, `.planning/`, or `.audits/`.
44
+ **Web apps:** when inlined Web App UAT guidance is present, follow it — add Playwright smoke scaffolding on the first UI slice if missing, name localhost preconditions, and plan verification commands that match browser-capable UAT modes at slice closeout.
44
45
  6. Include Threat Surface (Q3), Requirement Impact (Q4), proof level, observability, integration closure, Failure Modes (Q5), Load Profile (Q6), and Negative Tests (Q7) only where applicable.
45
46
  7. Right-size tasks. Simple slices can be one task; split only when context, ownership, or verification boundaries justify it.
46
47
  8. Task `verify` commands must be safe, simple commands. Do not use shell pipes, redirects, semicolons, backticks, command substitution, output trimming, or grep regex alternation with `|`. If multiple checks are needed, create a small test file and run it with `node --test` or a package test script, or use separate simple commands joined only with `&&`. For absence checks, verify a pattern does not exist with `! grep -q 'pattern' file` or `! rg -q 'pattern' file`; do not use `grep -c` or `rg -c` to assert zero matches because count commands exit 1 when they find zero matches, and the verification gate treats that as failure.
@@ -63,6 +63,7 @@ Then:
63
63
  - `{{taskPlanTemplatePath}}`
64
64
  2. {{skillActivation}} Record the installed skills you expect executors to use in each task plan's `skills_used` frontmatter.
65
65
  3. Define slice-level verification: the objective stopping condition. Plan real test files with real assertions; for simple slices, executable commands are fine.
66
+ **Web apps:** when inlined Web App UAT guidance is present, follow it for Playwright scaffolding and browser-capable verification commands.
66
67
  4. For non-trivial slices, plan observability / proof level / integration closure, threat surface, and requirement impact. Omit entirely for simple slices.
67
68
  5. Decompose the slice into tasks that fit one context window each. Every task passed to `gsd_plan_slice` must use the exact keys `taskId`, `title`, `description`, `estimate`, `files`, `verify`, `inputs`, `expectedOutput`, and optional `observabilityImpact`. Put Why / Do / Done-when detail in `description`. `files`, `inputs`, and `expectedOutput` must be JSON arrays of strings, even for one path (for example, `"expectedOutput": ["src/index.ts"]`, never `"expectedOutput": "src/index.ts"`). `expectedOutput` is path-only: list only files the task creates or overwrites, and use `[]` for pure verification tasks.
68
69
  6. **Persist planning state through `gsd_plan_slice`.** Call it with the full payload. The tool writes to the DB and renders `{{outputPath}}` and `{{slicePath}}/tasks/T##-PLAN.md` automatically. Do NOT rely on direct `PLAN.md` writes.
@@ -158,7 +158,7 @@ Fix root causes, not symptoms. If applying temporary mitigation, label it and pr
158
158
  - State uncertainty plainly: "Not sure this handles X - testing it." No performed confidence, no hedging paragraphs.
159
159
  - All user-visible narration must be grammatical English. Do not emit compressed planner notes like "Need inspect X". If it fits a commit comment or standup note, it is acceptable.
160
160
  - When debugging, stay curious. Problems are puzzles. Say what's interesting about the failure before reaching for fixes.
161
- - After completing a task, give a brief summary and 2-4 numbered next-step options; last option is always "Other". Omit the list for strict output formats.
161
+ - After completing a task, give a brief summary and 2-4 numbered next-step options; last option is always "Other". Omit the list for strict output formats, or when the active workflow prompt already ends with its own explicit "Next steps:" handoff block — in that case follow the workflow's handoff and do not add a second list.
162
162
 
163
163
  If any next step is destructive/outward-facing, present it via `ask_user_questions` and wait for the user's answer before execution. Do not execute a next-step item from a prior plain-text numbered list without fresh confirmation.
164
164
 
@@ -0,0 +1,140 @@
1
+ /**
2
+ * Provider Payload Policy - ordered shaping of provider request payloads.
3
+ *
4
+ * The order is intentional:
5
+ * 1. observation budgeting masks old tool results in auto-mode,
6
+ * 2. display truncation caps tool-result text for every mode,
7
+ * 3. the protected Source Context Block is appended after truncation,
8
+ * 4. supported models receive the configured service tier.
9
+ */
10
+
11
+ import type { ContextManagementConfig } from "./preferences-types.js";
12
+ import type { ServiceTierSetting } from "./service-tier.js";
13
+
14
+ import {
15
+ createObservationMask,
16
+ createResponsesInputObservationMask,
17
+ truncateContextResultMessages,
18
+ truncateResponsesInputResultItems,
19
+ } from "./context-masker.js";
20
+ import { getSourceObservationStore, isAutoActive } from "./auto-runtime-state.js";
21
+ import { loadEffectiveGSDPreferences } from "./preferences.js";
22
+ import { getEffectiveServiceTier, supportsServiceTier } from "./service-tier.js";
23
+ import { injectSourceContextBlockIntoPayload } from "./source-observations.js";
24
+
25
+ const DEFAULT_OBSERVATION_MASK_TURNS = 8;
26
+ const DEFAULT_TOOL_RESULT_MAX_CHARS = 800;
27
+
28
+ type MessagePayload = Parameters<ReturnType<typeof createObservationMask>>[0];
29
+ type ResponsesInputPayload = Parameters<ReturnType<typeof createResponsesInputObservationMask>>[0];
30
+
31
+ export interface ProviderPayloadPolicyDeps {
32
+ isAutoActive(): boolean;
33
+ loadContextManagementConfig(): ContextManagementConfig | undefined;
34
+ renderSourceContextBlock(): string | null;
35
+ getEffectiveServiceTier(): ServiceTierSetting;
36
+ supportsServiceTier(modelId: string): boolean;
37
+ }
38
+
39
+ export interface ProviderPayloadPolicyInput {
40
+ payload: Record<string, unknown>;
41
+ modelId?: string;
42
+ deps?: Partial<ProviderPayloadPolicyDeps>;
43
+ }
44
+
45
+ export const DEFAULT_PROVIDER_PAYLOAD_POLICY_DEPS: ProviderPayloadPolicyDeps = {
46
+ isAutoActive,
47
+ loadContextManagementConfig: () => loadEffectiveGSDPreferences()?.preferences.context_management,
48
+ renderSourceContextBlock: () => getSourceObservationStore().renderActiveBlock(),
49
+ getEffectiveServiceTier,
50
+ supportsServiceTier,
51
+ };
52
+
53
+ export function applyProviderPayloadPolicy({
54
+ payload,
55
+ modelId,
56
+ deps: overrides,
57
+ }: ProviderPayloadPolicyInput): Record<string, unknown> {
58
+ const deps = { ...DEFAULT_PROVIDER_PAYLOAD_POLICY_DEPS, ...overrides };
59
+
60
+ try {
61
+ applyContextManagement(payload, deps);
62
+ } catch {
63
+ // Provider payload shaping should not block a request when optional
64
+ // context management preferences or adapters fail.
65
+ }
66
+
67
+ try {
68
+ applySourceContextBlock(payload, deps);
69
+ } catch {
70
+ // Source observations are opportunistic; execution can continue without
71
+ // an injected block.
72
+ }
73
+
74
+ applyServiceTier(payload, modelId, deps);
75
+ return payload;
76
+ }
77
+
78
+ function applyContextManagement(
79
+ payload: Record<string, unknown>,
80
+ deps: ProviderPayloadPolicyDeps,
81
+ ): void {
82
+ const config = deps.loadContextManagementConfig();
83
+ applyObservationBudget(payload, config, deps.isAutoActive());
84
+ applyDisplayTruncation(payload, config);
85
+ }
86
+
87
+ function applyObservationBudget(
88
+ payload: Record<string, unknown>,
89
+ config: ContextManagementConfig | undefined,
90
+ autoActive: boolean,
91
+ ): void {
92
+ if (!autoActive || config?.observation_masking === false) return;
93
+
94
+ const keepTurns = config?.observation_mask_turns ?? DEFAULT_OBSERVATION_MASK_TURNS;
95
+ if (Array.isArray(payload.messages)) {
96
+ payload.messages = createObservationMask(keepTurns)(payload.messages as MessagePayload);
97
+ }
98
+ if (Array.isArray(payload.input)) {
99
+ payload.input = createResponsesInputObservationMask(keepTurns)(payload.input as ResponsesInputPayload);
100
+ }
101
+ }
102
+
103
+ function applyDisplayTruncation(
104
+ payload: Record<string, unknown>,
105
+ config: ContextManagementConfig | undefined,
106
+ ): void {
107
+ const maxChars = config?.tool_result_max_chars ?? DEFAULT_TOOL_RESULT_MAX_CHARS;
108
+
109
+ if (Array.isArray(payload.messages)) {
110
+ payload.messages = truncateContextResultMessages(payload.messages as MessagePayload, maxChars);
111
+ }
112
+ if (Array.isArray(payload.input)) {
113
+ payload.input = truncateResponsesInputResultItems(payload.input as ResponsesInputPayload, maxChars);
114
+ }
115
+ }
116
+
117
+ function applySourceContextBlock(
118
+ payload: Record<string, unknown>,
119
+ deps: ProviderPayloadPolicyDeps,
120
+ ): void {
121
+ if (!deps.isAutoActive()) return;
122
+
123
+ const sourceContextBlock = deps.renderSourceContextBlock();
124
+ if (!sourceContextBlock) return;
125
+
126
+ Object.assign(payload, injectSourceContextBlockIntoPayload(payload, sourceContextBlock));
127
+ }
128
+
129
+ function applyServiceTier(
130
+ payload: Record<string, unknown>,
131
+ modelId: string | undefined,
132
+ deps: ProviderPayloadPolicyDeps,
133
+ ): void {
134
+ if (!modelId) return;
135
+
136
+ const tier = deps.getEffectiveServiceTier();
137
+ if (!tier || !deps.supportsServiceTier(modelId)) return;
138
+
139
+ payload.service_tier = tier;
140
+ }
@@ -0,0 +1,41 @@
1
+ // Project/App: gsd-pi
2
+ // File Purpose: Process-level pull request policy for GSD-generated PRs.
3
+
4
+ import { createDraftPR } from "./git-service.js";
5
+ import {
6
+ buildPrEvidence,
7
+ type PrEvidence,
8
+ type PrEvidenceInput,
9
+ } from "./pr-evidence.js";
10
+
11
+ export interface DraftPullRequestOptions {
12
+ head: string;
13
+ base: string;
14
+ }
15
+
16
+ export interface DraftPullRequestDeps {
17
+ createDraftPR: (
18
+ basePath: string,
19
+ milestoneId: string,
20
+ title: string,
21
+ body: string,
22
+ opts: DraftPullRequestOptions,
23
+ ) => string | null;
24
+ }
25
+
26
+ export function buildPullRequestEvidence(input: PrEvidenceInput): PrEvidence {
27
+ return buildPrEvidence({
28
+ ...input,
29
+ aiAssisted: false,
30
+ });
31
+ }
32
+
33
+ export function createDraftPullRequestFromEvidence(
34
+ basePath: string,
35
+ milestoneId: string,
36
+ evidence: PrEvidence,
37
+ options: DraftPullRequestOptions,
38
+ deps: DraftPullRequestDeps = { createDraftPR },
39
+ ): string | null {
40
+ return deps.createDraftPR(basePath, milestoneId, evidence.title, evidence.body, options);
41
+ }
@@ -0,0 +1,140 @@
1
+ // Project/App: gsd-pi
2
+ // File Purpose: Canonical quality-gate closure from durable DB and artifact evidence.
3
+
4
+ import { existsSync, readFileSync } from "node:fs";
5
+
6
+ import { extractSection } from "./files.js";
7
+ import { getGateDefinition } from "./gate-registry.js";
8
+ import { getMilestoneSlices, getPendingGates, saveGateResult } from "./gsd-db.js";
9
+ import { resolveSliceFile, resolveTaskFile } from "./paths.js";
10
+ import type { GateId, GateRow, GateVerdict } from "./types.js";
11
+
12
+ export interface QualityGateClosureOptions {
13
+ artifactBasePath?: string;
14
+ milestoneValidationPassed?: boolean;
15
+ }
16
+
17
+ export interface QualityGateClosureResult {
18
+ repaired: Array<{ gateId: GateId; sliceId: string; taskId?: string; verdict: GateVerdict }>;
19
+ unresolved: GateRow[];
20
+ }
21
+
22
+ interface GateEvidence {
23
+ verdict: GateVerdict;
24
+ rationale: string;
25
+ findings: string;
26
+ }
27
+
28
+ const GATE_SECTION_HEADINGS: Partial<Record<GateId, string[]>> = {
29
+ Q3: ["Threat Surface", "Abuse Surface"],
30
+ Q4: ["Requirement Impact", "Broken Promises"],
31
+ Q5: ["Failure Modes"],
32
+ Q6: ["Load Profile"],
33
+ Q7: ["Negative Tests"],
34
+ Q8: ["Operational Readiness"],
35
+ };
36
+
37
+ function readFile(path: string | null): string | null {
38
+ if (!path || !existsSync(path)) return null;
39
+ return readFileSync(path, "utf-8");
40
+ }
41
+
42
+ function firstSection(content: string | null, gateId: GateId): string | null {
43
+ if (!content) return null;
44
+ for (const heading of GATE_SECTION_HEADINGS[gateId] ?? []) {
45
+ const section = extractSection(content, heading);
46
+ if (section) return section;
47
+ }
48
+ return null;
49
+ }
50
+
51
+ function evidenceArtifactContent(row: GateRow, basePath: string): string | null {
52
+ const def = getGateDefinition(row.gate_id);
53
+ switch (def?.ownerTurn) {
54
+ case "gate-evaluate":
55
+ return readFile(resolveSliceFile(basePath, row.milestone_id, row.slice_id, "PLAN"));
56
+ case "execute-task":
57
+ return readFile(resolveTaskFile(basePath, row.milestone_id, row.slice_id, row.task_id, "SUMMARY"));
58
+ case "complete-slice":
59
+ return readFile(resolveSliceFile(basePath, row.milestone_id, row.slice_id, "SUMMARY"));
60
+ default:
61
+ return null;
62
+ }
63
+ }
64
+
65
+ function closureEvidence(row: GateRow, options: QualityGateClosureOptions): GateEvidence | null {
66
+ const def = getGateDefinition(row.gate_id);
67
+ if (!def) return null;
68
+
69
+ if (def.ownerTurn === "validate-milestone" && options.milestoneValidationPassed) {
70
+ return {
71
+ verdict: "pass",
72
+ rationale: `${def.promptSection} covered by passing milestone validation`,
73
+ findings: "",
74
+ };
75
+ }
76
+
77
+ if (!options.artifactBasePath) return null;
78
+
79
+ const section = firstSection(evidenceArtifactContent(row, options.artifactBasePath), row.gate_id);
80
+ if (section) {
81
+ return {
82
+ verdict: "pass",
83
+ rationale: `${def.promptSection} evidence found in durable artifact`,
84
+ findings: section,
85
+ };
86
+ }
87
+
88
+ if (!options.milestoneValidationPassed) return null;
89
+ return {
90
+ verdict: "omitted",
91
+ rationale: `${def.promptSection} has no durable artifact section; milestone validation passed`,
92
+ findings: "",
93
+ };
94
+ }
95
+
96
+ function closeGate(row: GateRow, evidence: GateEvidence): void {
97
+ saveGateResult({
98
+ milestoneId: row.milestone_id,
99
+ sliceId: row.slice_id,
100
+ gateId: row.gate_id,
101
+ taskId: row.task_id,
102
+ verdict: evidence.verdict,
103
+ rationale: evidence.rationale,
104
+ findings: evidence.findings,
105
+ });
106
+ }
107
+
108
+ export function closeQualityGatesFromEvidence(
109
+ milestoneId: string,
110
+ options: QualityGateClosureOptions = {},
111
+ ): QualityGateClosureResult {
112
+ const repaired: QualityGateClosureResult["repaired"] = [];
113
+ const unresolved: GateRow[] = [];
114
+
115
+ for (const slice of getMilestoneSlices(milestoneId)) {
116
+ const sliceId = slice.id;
117
+ for (const row of getPendingGates(milestoneId, sliceId)) {
118
+ if (!getGateDefinition(row.gate_id)) {
119
+ unresolved.push(row);
120
+ continue;
121
+ }
122
+
123
+ const evidence = closureEvidence(row, options);
124
+ if (!evidence) {
125
+ unresolved.push(row);
126
+ continue;
127
+ }
128
+
129
+ closeGate(row, evidence);
130
+ repaired.push({
131
+ gateId: row.gate_id,
132
+ sliceId: row.slice_id,
133
+ ...(row.task_id ? { taskId: row.task_id } : {}),
134
+ verdict: evidence.verdict,
135
+ });
136
+ }
137
+ }
138
+
139
+ return { repaired, unresolved };
140
+ }
@@ -0,0 +1,138 @@
1
+ // Project/App: gsd-pi
2
+ // File Purpose: Resolve how workflow units should ask the user for input.
3
+
4
+ import { parseMcpToolName, toMcpToolName } from "./mcp-tool-name.js";
5
+
6
+ export type StructuredQuestionsFlag = "true" | "false";
7
+
8
+ export interface QuestionTransportOptions {
9
+ activeTools: string[];
10
+ authMode?: "apiKey" | "oauth" | "externalCli" | "none";
11
+ baseUrl?: string;
12
+ env?: NodeJS.ProcessEnv;
13
+ }
14
+
15
+ export interface QuestionTransportResolution {
16
+ structuredQuestionsAvailable: StructuredQuestionsFlag;
17
+ questionToolAvailable: boolean;
18
+ usesWorkflowMcp: boolean;
19
+ reason: "question-tool-available" | "question-tool-missing" | "workflow-mcp-disabled";
20
+ }
21
+
22
+ export interface WorkflowQuestionToolSurfaceOptions {
23
+ workflowServerName?: string;
24
+ workflowExplicitlyBlocked?: boolean;
25
+ workflowMcpTools: string[];
26
+ exactWorkflowMcpTools: string[];
27
+ env?: NodeJS.ProcessEnv;
28
+ }
29
+
30
+ export interface WorkflowQuestionToolSurfaceResolution extends QuestionTransportResolution {
31
+ questionToolName?: string;
32
+ workflowQuestionsEnabled: boolean;
33
+ disallowedTools: string[];
34
+ }
35
+
36
+ function isWorkflowMcpServerName(serverName: string): boolean {
37
+ const normalized = serverName.toLowerCase();
38
+ return normalized === "gsd" || normalized.includes("workflow");
39
+ }
40
+
41
+ export function usesWorkflowMcpTransport(
42
+ authMode: QuestionTransportOptions["authMode"],
43
+ baseUrl: string | undefined,
44
+ ): boolean {
45
+ return authMode === "externalCli" && typeof baseUrl === "string" && baseUrl.startsWith("local://");
46
+ }
47
+
48
+ export function hasAskUserQuestionsTool(activeTools: string[]): boolean {
49
+ return activeTools.some((toolName) => {
50
+ if (toolName === "ask_user_questions") return true;
51
+ const mcp = parseMcpToolName(toolName);
52
+ if (!mcp) return false;
53
+ if (mcp.toolName === "ask_user_questions") return true;
54
+ return mcp.toolName === "*" && isWorkflowMcpServerName(mcp.serverName);
55
+ });
56
+ }
57
+
58
+ function workflowMcpStructuredQuestionsEnabled(env: NodeJS.ProcessEnv = process.env): boolean {
59
+ const value = env.GSD_WORKFLOW_MCP_STRUCTURED_QUESTIONS?.trim().toLowerCase();
60
+ return value !== "0" && value !== "false" && value !== "off";
61
+ }
62
+
63
+ export function resolveQuestionTransport(
64
+ options: QuestionTransportOptions,
65
+ ): QuestionTransportResolution {
66
+ const questionToolAvailable = hasAskUserQuestionsTool(options.activeTools);
67
+ const usesWorkflowMcp = usesWorkflowMcpTransport(options.authMode, options.baseUrl);
68
+
69
+ if (!questionToolAvailable) {
70
+ return {
71
+ structuredQuestionsAvailable: "false",
72
+ questionToolAvailable,
73
+ usesWorkflowMcp,
74
+ reason: "question-tool-missing",
75
+ };
76
+ }
77
+
78
+ if (options.authMode === "externalCli" && !workflowMcpStructuredQuestionsEnabled(options.env)) {
79
+ return {
80
+ structuredQuestionsAvailable: "false",
81
+ questionToolAvailable,
82
+ usesWorkflowMcp,
83
+ reason: "workflow-mcp-disabled",
84
+ };
85
+ }
86
+
87
+ return {
88
+ structuredQuestionsAvailable: "true",
89
+ questionToolAvailable,
90
+ usesWorkflowMcp,
91
+ reason: "question-tool-available",
92
+ };
93
+ }
94
+
95
+ export function supportsStructuredQuestions(
96
+ activeTools: string[],
97
+ options: Omit<QuestionTransportOptions, "activeTools"> = {},
98
+ ): boolean {
99
+ return resolveQuestionTransport({
100
+ ...options,
101
+ activeTools,
102
+ }).structuredQuestionsAvailable === "true";
103
+ }
104
+
105
+ export function resolveWorkflowQuestionToolSurface(
106
+ options: WorkflowQuestionToolSurfaceOptions,
107
+ ): WorkflowQuestionToolSurfaceResolution {
108
+ const questionToolName = options.workflowServerName && !options.workflowExplicitlyBlocked
109
+ ? toMcpToolName(options.workflowServerName, "ask_user_questions")
110
+ : undefined;
111
+ const activeTools = [
112
+ ...options.exactWorkflowMcpTools,
113
+ ...options.workflowMcpTools,
114
+ ...(questionToolName ? [questionToolName] : []),
115
+ ];
116
+ const transport = resolveQuestionTransport({
117
+ activeTools,
118
+ authMode: "externalCli",
119
+ baseUrl: "local://claude-code",
120
+ env: options.env,
121
+ });
122
+ const exactQuestionToolAllowed =
123
+ !!questionToolName && options.exactWorkflowMcpTools.includes(questionToolName);
124
+ const workflowQuestionsEnabled =
125
+ transport.structuredQuestionsAvailable === "true" &&
126
+ (options.workflowMcpTools.length > 0 || exactQuestionToolAllowed);
127
+ const disallowedTools =
128
+ options.workflowServerName && transport.questionToolAvailable && !workflowQuestionsEnabled
129
+ ? [toMcpToolName(options.workflowServerName, "ask_user_questions")]
130
+ : [];
131
+
132
+ return {
133
+ ...transport,
134
+ questionToolName,
135
+ workflowQuestionsEnabled,
136
+ disallowedTools,
137
+ };
138
+ }
@@ -3,6 +3,7 @@
3
3
 
4
4
  import { classifyError, isTransient, type ErrorClass } from "./error-classifier.js";
5
5
  import { ReconciliationFailedError } from "./state-reconciliation.js";
6
+ import { IllegalPhaseTransitionError } from "./state-transition-matrix.js";
6
7
 
7
8
  export type RecoveryFailureKind =
8
9
  | "tool-schema"
@@ -13,6 +14,7 @@ export type RecoveryFailureKind =
13
14
  | "worktree-invalid"
14
15
  | "verification-drift"
15
16
  | "reconciliation-drift"
17
+ | "illegal-transition"
16
18
  | "provider"
17
19
  | "runtime-unknown";
18
20
 
@@ -43,7 +45,9 @@ export function classifyFailure(input: RecoveryClassificationInput): RecoveryCla
43
45
  const failureKind =
44
46
  input.error instanceof ReconciliationFailedError
45
47
  ? "reconciliation-drift"
46
- : input.failureKind ?? inferFailureKind(message);
48
+ : input.error instanceof IllegalPhaseTransitionError
49
+ ? "illegal-transition"
50
+ : input.failureKind ?? inferFailureKind(message);
47
51
 
48
52
  switch (failureKind) {
49
53
  case "tool-schema":
@@ -111,6 +115,15 @@ export function classifyFailure(input: RecoveryClassificationInput): RecoveryCla
111
115
  remediation:
112
116
  "Inspect the persistent or repair-failed drift kinds reported by the State Reconciliation Module before resuming.",
113
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
+ };
114
127
  case "provider": {
115
128
  const providerClass = classifyError(message, input.retryAfterMs);
116
129
  return {
@@ -189,9 +189,15 @@ export function parseRoadmapSlices(content: string): RoadmapSliceEntry[] {
189
189
  const risk = (riskMatch ? riskMatch[1] : "low") as RiskLevel;
190
190
 
191
191
  const depsMatch = rest.match(/`depends:\[([^\]]*)\]`/);
192
- const depends = depsMatch && depsMatch[1]!.trim()
193
- ? expandDependencies(depsMatch[1]!.split(",").map(s => s.trim()))
192
+ // Recovery fallback: double-bracket form `[[id]]` from serialized bracket-wrapped IDs
193
+ const fallbackDepsMatch = depsMatch ? null : rest.match(/`depends:\[(\[(?:[^\]]*)\](?:,\[(?:[^\]]*)\])*)\]`/);
194
+ const rawDepContent = (depsMatch ?? fallbackDepsMatch)?.[1] ?? "";
195
+ const SLICE_ID_RE = /^[A-Za-z0-9][A-Za-z0-9-]*$/;
196
+ const RANGE_RE = /^[A-Za-z]+\d+(?:-|\.\.)[A-Za-z]+\d+$/;
197
+ const rawDepParts = rawDepContent.trim()
198
+ ? rawDepContent.replace(/\[|\]/g, "").split(",").map(s => s.trim()).filter(s => SLICE_ID_RE.test(s) || RANGE_RE.test(s))
194
199
  : [];
200
+ const depends = expandDependencies(rawDepParts);
195
201
 
196
202
  // ADR-011: the renderer writes a `[sketch]` badge for sketch slices.
197
203
  // Parse it back so the is_sketch flag survives a markdown → DB re-import
@@ -57,9 +57,10 @@ const EXECUTION_TOOL_NAMES = new Set([
57
57
  "functions.exec_command",
58
58
  "gsd_exec",
59
59
  "gsd_exec_search",
60
+ "gsd_uat_exec",
60
61
  "powershell",
61
62
  ]);
62
- const MCP_EXECUTION_TOOL_RE = /^mcp__.+__gsd_exec(?:_search)?$/;
63
+ const MCP_EXECUTION_TOOL_RE = /^mcp__.+__gsd_(?:uat_)?exec(?:_search)?$/;
63
64
 
64
65
  // ─── Module State ───────────────────────────────────────────────────────────
65
66
 
@@ -206,11 +207,17 @@ export function clearEvidenceFromDisk(
206
207
  * Exit codes and output are filled in by recordToolResult after execution.
207
208
  */
208
209
  export function recordToolCall(toolCallId: string, toolName: string, input: Record<string, unknown>): void {
210
+ // Idempotent by toolCallId: native tools reach this via both
211
+ // tool_execution_start and tool_call; external (pre-executed) tools only
212
+ // via tool_execution_start. First recording wins.
213
+ if (unitEvidence.some(e => e.toolCallId === toolCallId)) return;
209
214
  if (isExecutionToolName(toolName)) {
210
215
  unitEvidence.push({
211
216
  kind: "bash",
212
217
  toolCallId,
213
- command: String(input.command ?? input.cmd ?? input.query ?? ""),
218
+ // gsd_exec / gsd_uat_exec carry the script body in `script` (or `code`);
219
+ // bash-style tools use `command`/`cmd`; gsd_exec_search uses `query`.
220
+ command: String(input.command ?? input.script ?? input.cmd ?? input.code ?? input.query ?? ""),
214
221
  exitCode: -1,
215
222
  outputSnippet: "",
216
223
  timestamp: Date.now(),
@@ -249,11 +256,36 @@ export function recordToolResult(
249
256
  if (entry.kind === "bash") {
250
257
  const text = extractResultText(result);
251
258
  entry.outputSnippet = text.slice(0, 500);
252
- const exitMatch = text.match(/Command exited with code (\d+)/);
253
- entry.exitCode = exitMatch ? Number(exitMatch[1]) : (isError ? 1 : 0);
259
+ entry.exitCode = resolveExitCode(text, isError);
254
260
  }
255
261
  }
256
262
 
263
+ /**
264
+ * Resolve the exit code from a tool result's text. Handles the bash tool's
265
+ * prose marker, the gsd_exec / gsd_uat_exec JSON envelope (`"exit_code": N`),
266
+ * and a last-resort read of the run's persisted `.gsd/exec/<id>.meta.json`
267
+ * (covers truncated result text).
268
+ */
269
+ function resolveExitCode(text: string, isError: boolean): number {
270
+ const proseMatch = text.match(/Command exited with code (\d+)/);
271
+ if (proseMatch) return Number(proseMatch[1]);
272
+
273
+ const jsonMatch = text.match(/"exit_code"\s*:\s*(-?\d+)/);
274
+ if (jsonMatch) return Number(jsonMatch[1]);
275
+
276
+ const metaMatch = text.match(/"meta_path"\s*:\s*"([^"]+)"/);
277
+ if (metaMatch) {
278
+ try {
279
+ const meta = JSON.parse(readFileSync(metaMatch[1], "utf-8")) as Record<string, unknown>;
280
+ if (typeof meta.exit_code === "number") return meta.exit_code;
281
+ } catch {
282
+ // Fall through to the isError heuristic
283
+ }
284
+ }
285
+
286
+ return isError ? 1 : 0;
287
+ }
288
+
257
289
  // ─── Internals ──────────────────────────────────────────────────────────────
258
290
 
259
291
  function extractResultText(result: unknown): string {
@@ -121,9 +121,14 @@ function findMatches(
121
121
  const exact = bashCalls.filter(b => b.command.trim() === normalized);
122
122
  if (exact.length > 0) return exact;
123
123
 
124
- // Substring match: claimed is contained in actual or actual in claimed
124
+ // Substring match: claimed is contained in actual or actual in claimed.
125
+ // A claimed verification command typically appears verbatim inside a
126
+ // larger gsd_exec script body (cd prefix, multi-line scripts), so
127
+ // script-containing-claim is the common direction. Blank-command entries
128
+ // must be excluded — `"x".includes("")` is true, so they'd match anything.
125
129
  const substring = bashCalls.filter(
126
- b => b.command.includes(normalized) || normalized.includes(b.command),
130
+ b => b.command.trim().length > 0 &&
131
+ (b.command.includes(normalized) || normalized.includes(b.command)),
127
132
  );
128
133
  if (substring.length > 0) return substring;
129
134