@opengsd/gsd-pi 1.3.0-dev.65546769 → 1.3.0-dev.72e3af2a

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 (357) hide show
  1. package/dist/resources/.managed-resources-content-hash +1 -1
  2. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +11 -2
  3. package/dist/resources/extensions/google-cli/stream-adapter.js +82 -15
  4. package/dist/resources/extensions/gsd/artifact-verification.js +427 -0
  5. package/dist/resources/extensions/gsd/auto/orchestrator.js +12 -3
  6. package/dist/resources/extensions/gsd/auto/session.js +3 -0
  7. package/dist/resources/extensions/gsd/auto-artifact-paths.js +28 -1
  8. package/dist/resources/extensions/gsd/auto-dispatch.js +20 -19
  9. package/dist/resources/extensions/gsd/auto-prompts.js +26 -11
  10. package/dist/resources/extensions/gsd/auto-recovery.js +6 -507
  11. package/dist/resources/extensions/gsd/auto-runtime-state.js +4 -5
  12. package/dist/resources/extensions/gsd/auto-timeout-recovery.js +3 -3
  13. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +103 -13
  14. package/dist/resources/extensions/gsd/bootstrap/core-session-tools.js +38 -0
  15. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +6 -1
  16. package/dist/resources/extensions/gsd/bootstrap/exec-tools.js +2 -0
  17. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +10 -19
  18. package/dist/resources/extensions/gsd/bootstrap/system-context.js +46 -19
  19. package/dist/resources/extensions/gsd/bootstrap/tool-call-loop-guard.js +68 -10
  20. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +1 -1
  21. package/dist/resources/extensions/gsd/commands-context.js +19 -1
  22. package/dist/resources/extensions/gsd/commands-prefs-wizard.js +16 -10
  23. package/dist/resources/extensions/gsd/commands-worktree.js +12 -10
  24. package/dist/resources/extensions/gsd/dashboard-overlay.js +32 -3
  25. package/dist/resources/extensions/gsd/db/queries.js +60 -0
  26. package/dist/resources/extensions/gsd/db-workspace.js +55 -3
  27. package/dist/resources/extensions/gsd/doctor-providers.js +92 -8
  28. package/dist/resources/extensions/gsd/exec-sandbox.js +45 -9
  29. package/dist/resources/extensions/gsd/forensics.js +2 -32
  30. package/dist/resources/extensions/gsd/git-service.js +4 -4
  31. package/dist/resources/extensions/gsd/guided-flow-queue.js +66 -5
  32. package/dist/resources/extensions/gsd/health-widget.js +55 -29
  33. package/dist/resources/extensions/gsd/layout-policy.js +3 -1
  34. package/dist/resources/extensions/gsd/markdown-renderer.js +8 -9
  35. package/dist/resources/extensions/gsd/memory-consolidation-scanner.js +44 -21
  36. package/dist/resources/extensions/gsd/migration-auto-check.js +22 -0
  37. package/dist/resources/extensions/gsd/milestone-ids.js +32 -2
  38. package/dist/resources/extensions/gsd/milestone-implementation-evidence.js +26 -20
  39. package/dist/resources/extensions/gsd/prompts/code-review.md +6 -4
  40. package/dist/resources/extensions/gsd/quick.js +45 -2
  41. package/dist/resources/extensions/gsd/session-forensics.js +11 -1
  42. package/dist/resources/extensions/gsd/skills/gsd-headless/references/commands.md +1 -1
  43. package/dist/resources/extensions/gsd/state/derive/cache.js +28 -0
  44. package/dist/resources/extensions/gsd/state/derive/db-open.js +39 -0
  45. package/dist/resources/extensions/gsd/state/derive/from-db.js +452 -0
  46. package/dist/resources/extensions/gsd/state/derive/index.js +75 -0
  47. package/dist/resources/extensions/gsd/state/derive/interrupted-work.js +21 -0
  48. package/dist/resources/extensions/gsd/state-reconciliation/drift/stale-render.js +45 -2
  49. package/dist/resources/extensions/gsd/state-reconciliation/index.js +48 -23
  50. package/dist/resources/extensions/gsd/state-reconciliation/registry.js +32 -28
  51. package/dist/resources/extensions/gsd/state.js +12 -611
  52. package/dist/resources/extensions/gsd/tools/complete-slice.js +2 -2
  53. package/dist/resources/extensions/gsd/tools/complete-task.js +43 -14
  54. package/dist/resources/extensions/gsd/tools/exec-tool.js +7 -2
  55. package/dist/resources/extensions/gsd/unit-context-composer.js +23 -7
  56. package/dist/resources/extensions/gsd/unit-registry.js +32 -4
  57. package/dist/resources/extensions/gsd/unmerged-milestone-guard.js +33 -3
  58. package/dist/resources/extensions/gsd/validation-block-guard.js +9 -4
  59. package/dist/resources/extensions/gsd/workflow-projections.js +19 -14
  60. package/dist/resources/extensions/gsd/workspace-git-preflight.js +30 -1
  61. package/dist/resources/extensions/gsd/worktree-manager.js +44 -2
  62. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  63. package/dist/web/standalone/.next/BUILD_ID +1 -1
  64. package/dist/web/standalone/.next/app-path-routes-manifest.json +10 -10
  65. package/dist/web/standalone/.next/build-manifest.json +3 -3
  66. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  67. package/dist/web/standalone/.next/react-loadable-manifest.json +9 -9
  68. package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  69. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  70. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  71. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  72. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  73. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  74. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  75. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  76. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  77. package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  78. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  79. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  80. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  81. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  82. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  83. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  84. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  85. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  86. package/dist/web/standalone/.next/server/app/api/visualizer/route.js +1 -1
  87. package/dist/web/standalone/.next/server/app/index.html +1 -1
  88. package/dist/web/standalone/.next/server/app/index.rsc +2 -2
  89. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
  90. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +2 -2
  91. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  92. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  93. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  94. package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
  95. package/dist/web/standalone/.next/server/app-paths-manifest.json +10 -10
  96. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  97. package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
  98. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  99. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  100. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  101. package/dist/web/standalone/.next/static/chunks/{2659.b7b129ee6a769448.js → 2659.58e950899a9bb82f.js} +1 -1
  102. package/dist/web/standalone/.next/static/chunks/2772.a7c1fcc69a4685ef.js +1 -0
  103. package/dist/web/standalone/.next/static/chunks/{3616.3c60753b8ffcbd2e.js → 3616.61a2af74bb8833c8.js} +1 -1
  104. package/dist/web/standalone/.next/static/chunks/{4283.8e446784528ed9dc.js → 4283.d0d9e0a955e441cb.js} +1 -1
  105. package/dist/web/standalone/.next/static/chunks/{5826.a46ecdd1cfe8dabc.js → 5826.5421d66c72b9f34e.js} +1 -1
  106. package/dist/web/standalone/.next/static/chunks/{8785.481aa5869991b760.js → 8785.e29b3134cab1d153.js} +1 -1
  107. package/dist/web/standalone/.next/static/chunks/8937.640dc9c2aaa1dfad.js +10 -0
  108. package/dist/web/standalone/.next/static/chunks/app/{page-6644fc6ee8ca1247.js → page-72a856634ad14c10.js} +1 -1
  109. package/dist/web/standalone/.next/static/chunks/webpack-9c401904f87ded16.js +1 -0
  110. package/dist/web/standalone/node_modules/node-pty/build/Makefile +1 -1
  111. package/package.json +1 -1
  112. package/packages/cloud-mcp-gateway/package.json +2 -2
  113. package/packages/contracts/dist/workflow.d.ts +1 -0
  114. package/packages/contracts/dist/workflow.d.ts.map +1 -1
  115. package/packages/contracts/dist/workflow.js +2 -0
  116. package/packages/contracts/dist/workflow.js.map +1 -1
  117. package/packages/contracts/package.json +1 -1
  118. package/packages/daemon/package.json +4 -4
  119. package/packages/gsd-agent-core/dist/agent-session.d.ts +1 -0
  120. package/packages/gsd-agent-core/dist/agent-session.d.ts.map +1 -1
  121. package/packages/gsd-agent-core/dist/agent-session.js +3 -0
  122. package/packages/gsd-agent-core/dist/agent-session.js.map +1 -1
  123. package/packages/gsd-agent-core/dist/extension-ui-snapshot.d.ts +41 -0
  124. package/packages/gsd-agent-core/dist/extension-ui-snapshot.d.ts.map +1 -0
  125. package/packages/gsd-agent-core/dist/extension-ui-snapshot.js +62 -0
  126. package/packages/gsd-agent-core/dist/extension-ui-snapshot.js.map +1 -0
  127. package/packages/gsd-agent-core/dist/index.d.ts +2 -0
  128. package/packages/gsd-agent-core/dist/index.d.ts.map +1 -1
  129. package/packages/gsd-agent-core/dist/index.js +2 -0
  130. package/packages/gsd-agent-core/dist/index.js.map +1 -1
  131. package/packages/gsd-agent-core/dist/session/agent-session-events.js +1 -1
  132. package/packages/gsd-agent-core/dist/session/agent-session-events.js.map +1 -1
  133. package/packages/gsd-agent-core/dist/session/agent-session-host.d.ts +1 -0
  134. package/packages/gsd-agent-core/dist/session/agent-session-host.d.ts.map +1 -1
  135. package/packages/gsd-agent-core/dist/session/agent-session-host.js.map +1 -1
  136. package/packages/gsd-agent-core/dist/session/agent-session-prompt.d.ts +5 -0
  137. package/packages/gsd-agent-core/dist/session/agent-session-prompt.d.ts.map +1 -1
  138. package/packages/gsd-agent-core/dist/session/agent-session-prompt.js +60 -3
  139. package/packages/gsd-agent-core/dist/session/agent-session-prompt.js.map +1 -1
  140. package/packages/gsd-agent-core/dist/transcript-store.d.ts +58 -0
  141. package/packages/gsd-agent-core/dist/transcript-store.d.ts.map +1 -0
  142. package/packages/gsd-agent-core/dist/transcript-store.js +132 -0
  143. package/packages/gsd-agent-core/dist/transcript-store.js.map +1 -0
  144. package/packages/gsd-agent-core/package.json +5 -5
  145. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.d.ts +2 -0
  146. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  147. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.js +25 -11
  148. package/packages/gsd-agent-modes/dist/modes/interactive/components/tool-execution.js.map +1 -1
  149. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller-latency.d.ts +4 -0
  150. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller-latency.d.ts.map +1 -0
  151. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller-latency.js +7 -0
  152. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller-latency.js.map +1 -0
  153. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.d.ts +3 -24
  154. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  155. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js +26 -829
  156. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  157. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-handoff-filter.d.ts +58 -0
  158. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-handoff-filter.d.ts.map +1 -0
  159. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-handoff-filter.js +312 -0
  160. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-handoff-filter.js.map +1 -0
  161. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-pinned-zone.d.ts +31 -0
  162. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-pinned-zone.d.ts.map +1 -0
  163. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-pinned-zone.js +130 -0
  164. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-pinned-zone.js.map +1 -0
  165. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-segment-walker.d.ts +15 -0
  166. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-segment-walker.d.ts.map +1 -0
  167. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-segment-walker.js +258 -0
  168. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-segment-walker.js.map +1 -0
  169. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-tool-rollup.d.ts +13 -0
  170. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-tool-rollup.d.ts.map +1 -0
  171. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-tool-rollup.js +118 -0
  172. package/packages/gsd-agent-modes/dist/modes/interactive/controllers/chat-tool-rollup.js.map +1 -0
  173. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode-state.d.ts +9 -0
  174. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode-state.d.ts.map +1 -1
  175. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode-state.js +1 -1
  176. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode-state.js.map +1 -1
  177. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode-ui-state.d.ts +54 -0
  178. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode-ui-state.d.ts.map +1 -0
  179. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode-ui-state.js +20 -0
  180. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode-ui-state.js.map +1 -0
  181. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode.d.ts +4 -1
  182. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  183. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode.js +9 -0
  184. package/packages/gsd-agent-modes/dist/modes/interactive/interactive-mode.js.map +1 -1
  185. package/packages/gsd-agent-modes/dist/modes/interactive/streaming-render-state.d.ts +56 -0
  186. package/packages/gsd-agent-modes/dist/modes/interactive/streaming-render-state.d.ts.map +1 -0
  187. package/packages/gsd-agent-modes/dist/modes/interactive/streaming-render-state.js +44 -0
  188. package/packages/gsd-agent-modes/dist/modes/interactive/streaming-render-state.js.map +1 -0
  189. package/packages/gsd-agent-modes/dist/modes/interactive/tui-transcript-tracker.d.ts +5 -0
  190. package/packages/gsd-agent-modes/dist/modes/interactive/tui-transcript-tracker.d.ts.map +1 -0
  191. package/packages/gsd-agent-modes/dist/modes/interactive/tui-transcript-tracker.js +77 -0
  192. package/packages/gsd-agent-modes/dist/modes/interactive/tui-transcript-tracker.js.map +1 -0
  193. package/packages/gsd-agent-modes/dist/modes/rpc/rpc-mode.d.ts.map +1 -1
  194. package/packages/gsd-agent-modes/dist/modes/rpc/rpc-mode.js +18 -0
  195. package/packages/gsd-agent-modes/dist/modes/rpc/rpc-mode.js.map +1 -1
  196. package/packages/gsd-agent-modes/package.json +7 -7
  197. package/packages/mcp-server/README.md +1 -1
  198. package/packages/mcp-server/dist/server.d.ts +1 -1
  199. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  200. package/packages/mcp-server/dist/server.js +3 -3
  201. package/packages/mcp-server/dist/server.js.map +1 -1
  202. package/packages/mcp-server/dist/workflow-tools.d.ts +13 -1
  203. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  204. package/packages/mcp-server/dist/workflow-tools.js +34 -20
  205. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  206. package/packages/mcp-server/package.json +4 -4
  207. package/packages/native/package.json +1 -1
  208. package/packages/pi-agent-core/package.json +1 -1
  209. package/packages/pi-ai/package.json +1 -1
  210. package/packages/pi-coding-agent/README.md +3 -2
  211. package/packages/pi-coding-agent/dist/core/session-manager-context.d.ts +9 -0
  212. package/packages/pi-coding-agent/dist/core/session-manager-context.d.ts.map +1 -0
  213. package/packages/pi-coding-agent/dist/core/session-manager-context.js +94 -0
  214. package/packages/pi-coding-agent/dist/core/session-manager-context.js.map +1 -0
  215. package/packages/pi-coding-agent/dist/core/session-manager-list.d.ts +8 -0
  216. package/packages/pi-coding-agent/dist/core/session-manager-list.d.ts.map +1 -0
  217. package/packages/pi-coding-agent/dist/core/session-manager-list.js +244 -0
  218. package/packages/pi-coding-agent/dist/core/session-manager-list.js.map +1 -0
  219. package/packages/pi-coding-agent/dist/core/session-manager-migration.d.ts +12 -0
  220. package/packages/pi-coding-agent/dist/core/session-manager-migration.d.ts.map +1 -0
  221. package/packages/pi-coding-agent/dist/core/session-manager-migration.js +84 -0
  222. package/packages/pi-coding-agent/dist/core/session-manager-migration.js.map +1 -0
  223. package/packages/pi-coding-agent/dist/core/session-manager-types.d.ts +135 -0
  224. package/packages/pi-coding-agent/dist/core/session-manager-types.d.ts.map +1 -0
  225. package/packages/pi-coding-agent/dist/core/session-manager-types.js +2 -0
  226. package/packages/pi-coding-agent/dist/core/session-manager-types.js.map +1 -0
  227. package/packages/pi-coding-agent/dist/core/session-manager.d.ts +6 -154
  228. package/packages/pi-coding-agent/dist/core/session-manager.d.ts.map +1 -1
  229. package/packages/pi-coding-agent/dist/core/session-manager.js +22 -459
  230. package/packages/pi-coding-agent/dist/core/session-manager.js.map +1 -1
  231. package/packages/pi-coding-agent/dist/theme/theme-schema.d.ts +75 -75
  232. package/packages/pi-coding-agent/dist/theme/theme-schema.d.ts.map +1 -1
  233. package/packages/pi-coding-agent/dist/theme/theme-schema.js +1 -1
  234. package/packages/pi-coding-agent/dist/theme/theme-schema.js.map +1 -1
  235. package/packages/pi-coding-agent/dist/theme/theme.d.ts.map +1 -1
  236. package/packages/pi-coding-agent/dist/theme/theme.js +11 -7
  237. package/packages/pi-coding-agent/dist/theme/theme.js.map +1 -1
  238. package/packages/pi-coding-agent/package.json +7 -7
  239. package/packages/pi-tui/package.json +2 -2
  240. package/packages/rpc-client/package.json +2 -2
  241. package/pkg/dist/theme/theme-schema.d.ts +75 -75
  242. package/pkg/dist/theme/theme-schema.d.ts.map +1 -1
  243. package/pkg/dist/theme/theme-schema.js +1 -1
  244. package/pkg/dist/theme/theme-schema.js.map +1 -1
  245. package/pkg/dist/theme/theme.d.ts.map +1 -1
  246. package/pkg/dist/theme/theme.js +11 -7
  247. package/pkg/dist/theme/theme.js.map +1 -1
  248. package/pkg/package.json +1 -1
  249. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +20 -2
  250. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +80 -0
  251. package/src/resources/extensions/google-cli/stream-adapter.ts +106 -19
  252. package/src/resources/extensions/gsd/artifact-verification.ts +464 -0
  253. package/src/resources/extensions/gsd/auto/orchestrator.ts +25 -11
  254. package/src/resources/extensions/gsd/auto/session.ts +5 -0
  255. package/src/resources/extensions/gsd/auto-artifact-paths.ts +47 -1
  256. package/src/resources/extensions/gsd/auto-dispatch.ts +21 -23
  257. package/src/resources/extensions/gsd/auto-prompts.ts +38 -12
  258. package/src/resources/extensions/gsd/auto-recovery.ts +10 -508
  259. package/src/resources/extensions/gsd/auto-runtime-state.ts +4 -5
  260. package/src/resources/extensions/gsd/auto-timeout-recovery.ts +3 -2
  261. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +125 -12
  262. package/src/resources/extensions/gsd/bootstrap/core-session-tools.ts +43 -0
  263. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +6 -1
  264. package/src/resources/extensions/gsd/bootstrap/exec-tools.ts +2 -0
  265. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +11 -19
  266. package/src/resources/extensions/gsd/bootstrap/system-context.ts +52 -18
  267. package/src/resources/extensions/gsd/bootstrap/tool-call-loop-guard.ts +74 -10
  268. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +1 -1
  269. package/src/resources/extensions/gsd/commands-context.ts +18 -1
  270. package/src/resources/extensions/gsd/commands-prefs-wizard.ts +14 -9
  271. package/src/resources/extensions/gsd/commands-worktree.ts +12 -10
  272. package/src/resources/extensions/gsd/dashboard-overlay.ts +32 -3
  273. package/src/resources/extensions/gsd/db/queries.ts +79 -0
  274. package/src/resources/extensions/gsd/db-workspace.ts +61 -3
  275. package/src/resources/extensions/gsd/doctor-providers.ts +103 -9
  276. package/src/resources/extensions/gsd/exec-sandbox.ts +49 -9
  277. package/src/resources/extensions/gsd/forensics.ts +2 -33
  278. package/src/resources/extensions/gsd/git-service.ts +5 -5
  279. package/src/resources/extensions/gsd/guided-flow-queue.ts +89 -4
  280. package/src/resources/extensions/gsd/health-widget.ts +69 -32
  281. package/src/resources/extensions/gsd/layout-policy.ts +2 -1
  282. package/src/resources/extensions/gsd/markdown-renderer.ts +8 -11
  283. package/src/resources/extensions/gsd/memory-consolidation-scanner.ts +51 -19
  284. package/src/resources/extensions/gsd/migration-auto-check.ts +23 -0
  285. package/src/resources/extensions/gsd/milestone-ids.ts +31 -2
  286. package/src/resources/extensions/gsd/milestone-implementation-evidence.ts +35 -21
  287. package/src/resources/extensions/gsd/prompts/code-review.md +6 -4
  288. package/src/resources/extensions/gsd/quick.ts +43 -2
  289. package/src/resources/extensions/gsd/session-forensics.ts +11 -1
  290. package/src/resources/extensions/gsd/skills/gsd-headless/references/commands.md +1 -1
  291. package/src/resources/extensions/gsd/state/derive/cache.ts +46 -0
  292. package/src/resources/extensions/gsd/state/derive/db-open.ts +45 -0
  293. package/src/resources/extensions/gsd/state/derive/from-db.ts +561 -0
  294. package/src/resources/extensions/gsd/state/derive/index.ts +104 -0
  295. package/src/resources/extensions/gsd/state/derive/interrupted-work.ts +31 -0
  296. package/src/resources/extensions/gsd/state-reconciliation/drift/stale-render.ts +81 -7
  297. package/src/resources/extensions/gsd/state-reconciliation/index.ts +50 -24
  298. package/src/resources/extensions/gsd/state-reconciliation/registry.ts +43 -28
  299. package/src/resources/extensions/gsd/state.ts +32 -732
  300. package/src/resources/extensions/gsd/tests/auto-artifact-paths.test.ts +98 -1
  301. package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +111 -1
  302. package/src/resources/extensions/gsd/tests/commands-context.test.ts +26 -0
  303. package/src/resources/extensions/gsd/tests/commands-gsd-core.test.ts +1 -0
  304. package/src/resources/extensions/gsd/tests/commands-worktree-clean.test.ts +80 -0
  305. package/src/resources/extensions/gsd/tests/complete-slice.test.ts +11 -0
  306. package/src/resources/extensions/gsd/tests/complete-task-rollback-evidence.test.ts +48 -8
  307. package/src/resources/extensions/gsd/tests/complete-task.test.ts +75 -0
  308. package/src/resources/extensions/gsd/tests/dashboard-overlay.test.ts +55 -2
  309. package/src/resources/extensions/gsd/tests/dispatch-rule-coverage.test.ts +26 -1
  310. package/src/resources/extensions/gsd/tests/doctor-forensics-db-open-regression.test.ts +70 -2
  311. package/src/resources/extensions/gsd/tests/doctor-providers.test.ts +107 -0
  312. package/src/resources/extensions/gsd/tests/exec-graceful-kill.test.ts +38 -0
  313. package/src/resources/extensions/gsd/tests/exec-tool.test.ts +45 -1
  314. package/src/resources/extensions/gsd/tests/forensics-error-filter.test.ts +88 -0
  315. package/src/resources/extensions/gsd/tests/guided-discuss-milestone-prompt-rendering.test.ts +42 -0
  316. package/src/resources/extensions/gsd/tests/health-widget.test.ts +268 -3
  317. package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +119 -1
  318. package/src/resources/extensions/gsd/tests/integration/queue-active-milestone-context-budget.test.ts +93 -0
  319. package/src/resources/extensions/gsd/tests/integration/quick-branch-lifecycle.test.ts +56 -9
  320. package/src/resources/extensions/gsd/tests/knowledge-cold-start.test.ts +14 -0
  321. package/src/resources/extensions/gsd/tests/memory-consolidation-scanner.test.ts +78 -0
  322. package/src/resources/extensions/gsd/tests/migration-auto-check.test.ts +70 -0
  323. package/src/resources/extensions/gsd/tests/orchestrator-logs.test.ts +43 -1
  324. package/src/resources/extensions/gsd/tests/parallel-research-dispatch.test.ts +26 -0
  325. package/src/resources/extensions/gsd/tests/parsers-legacy-importers.test.ts +2 -1
  326. package/src/resources/extensions/gsd/tests/pipeline-variant-dispatch.test.ts +1 -1
  327. package/src/resources/extensions/gsd/tests/prefs-wizard-coverage.test.ts +54 -1
  328. package/src/resources/extensions/gsd/tests/progressive-planning.test.ts +50 -14
  329. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +195 -1
  330. package/src/resources/extensions/gsd/tests/read-uat-gate-verdict.test.ts +185 -0
  331. package/src/resources/extensions/gsd/tests/register-hooks-depth-verification.test.ts +87 -0
  332. package/src/resources/extensions/gsd/tests/runtime-invariant-modules.test.ts +7 -0
  333. package/src/resources/extensions/gsd/tests/stalled-tool-recovery.test.ts +56 -0
  334. package/src/resources/extensions/gsd/tests/state-reconciliation-drift.test.ts +191 -0
  335. package/src/resources/extensions/gsd/tests/tool-call-loop-guard.test.ts +151 -0
  336. package/src/resources/extensions/gsd/tests/tool-param-optionality.test.ts +26 -0
  337. package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +193 -14
  338. package/src/resources/extensions/gsd/tests/unmerged-milestone-guard.test.ts +25 -0
  339. package/src/resources/extensions/gsd/tests/validation-block-guard.test.ts +79 -0
  340. package/src/resources/extensions/gsd/tests/verify-artifact-tightened.test.ts +66 -0
  341. package/src/resources/extensions/gsd/tests/workspace-git-preflight.test.ts +151 -2
  342. package/src/resources/extensions/gsd/tests/worktree-manager.test.ts +39 -0
  343. package/src/resources/extensions/gsd/tools/complete-slice.ts +2 -2
  344. package/src/resources/extensions/gsd/tools/complete-task.ts +53 -15
  345. package/src/resources/extensions/gsd/tools/exec-tool.ts +7 -3
  346. package/src/resources/extensions/gsd/unit-context-composer.ts +33 -7
  347. package/src/resources/extensions/gsd/unit-registry.ts +32 -4
  348. package/src/resources/extensions/gsd/unmerged-milestone-guard.ts +41 -5
  349. package/src/resources/extensions/gsd/validation-block-guard.ts +13 -7
  350. package/src/resources/extensions/gsd/workflow-projections.ts +20 -14
  351. package/src/resources/extensions/gsd/workspace-git-preflight.ts +31 -0
  352. package/src/resources/extensions/gsd/worktree-manager.ts +41 -1
  353. package/dist/web/standalone/.next/static/chunks/2772.bfa657f49f955239.js +0 -1
  354. package/dist/web/standalone/.next/static/chunks/796.e0bdc932325d7e03.js +0 -10
  355. package/dist/web/standalone/.next/static/chunks/webpack-f46ea08200a0227e.js +0 -1
  356. /package/dist/web/standalone/.next/static/{BTKtGFF1Y-hvVJEGhBRo9 → O7xDYXO0r4zFhIzY1hrWV}/_buildManifest.js +0 -0
  357. /package/dist/web/standalone/.next/static/{BTKtGFF1Y-hvVJEGhBRo9 → O7xDYXO0r4zFhIzY1hrWV}/_ssgManifest.js +0 -0
@@ -13,6 +13,7 @@ import {
13
13
  diffWorktreeNumstat,
14
14
  getWorktreeGSDDiff,
15
15
  getWorktreeLog,
16
+ mergeWorktreeToMain,
16
17
  worktreeBranchName,
17
18
  worktreePath,
18
19
  pruneEphemeralGhostWorktreeDirectories,
@@ -333,6 +334,44 @@ describe("getWorktreeLog", () => {
333
334
  });
334
335
  });
335
336
 
337
+ // ─── mergeWorktreeToMain ─────────────────────────────────────────────────────
338
+
339
+ describe("mergeWorktreeToMain", () => {
340
+ let base: string;
341
+ let wtPath: string;
342
+
343
+ beforeEach(() => {
344
+ base = makeBaseRepo();
345
+ wtPath = worktreePath(base, "M900");
346
+ mkdirSync(join(base, ".gsd-worktrees"), { recursive: true });
347
+ run(`git worktree add -b milestone/M900 "${wtPath}"`, base);
348
+ });
349
+
350
+ afterEach(() => { rmSync(base, { recursive: true, force: true }); });
351
+
352
+ test("cleans failed milestone squash state and deletes orphan branch", () => {
353
+ writeFileSync(join(wtPath, "README.md"), "# Milestone change\n", "utf-8");
354
+ run("git add README.md", wtPath);
355
+ run('git commit -m "feat: milestone change"', wtPath);
356
+
357
+ writeFileSync(join(base, "README.md"), "# Main change\n", "utf-8");
358
+ run("git add README.md", base);
359
+ run('git commit -m "feat: main change"', base);
360
+
361
+ assert.throws(
362
+ () => mergeWorktreeToMain(base, "M900", "feat: merge M900", "milestone/M900"),
363
+ /Merge conflicts detected/,
364
+ );
365
+
366
+ assert.equal(run("git status --porcelain", base), "", "failed squash cleanup should leave main clean");
367
+ assert.ok(!existsSync(wtPath), "failed milestone worktree should be removed");
368
+ assert.ok(
369
+ !run("git branch", base).includes("milestone/M900"),
370
+ "failed milestone branch should be deleted after worktree removal",
371
+ );
372
+ });
373
+ });
374
+
336
375
  // ─── removeWorktree ───────────────────────────────────────────────────────────
337
376
 
338
377
  describe("removeWorktree", () => {
@@ -30,7 +30,7 @@ import { classifyUatContent, escalatesArtifactUatToBrowser } from "../uat-policy
30
30
  import { invalidateStateCache } from "../state.js";
31
31
  import { renderRoadmapFromDb, roadmapRenderMarksSliceDone } from "../markdown-renderer.js";
32
32
  import { isStaleWrite } from "../auto/turn-epoch.js";
33
- import { flushWorkflowProjections } from "../projection-flush.js";
33
+ import { renderMilestoneShellProjections } from "../workflow-projections.js";
34
34
  import { writeManifest } from "../workflow-manifest.js";
35
35
  import { appendEvent } from "../workflow-events.js";
36
36
  import { logWarning, logError } from "../workflow-logger.js";
@@ -540,7 +540,7 @@ export async function handleCompleteSlice(
540
540
  // Separate try/catch per step so a projection failure doesn't prevent
541
541
  // the event log entry (critical for worktree reconciliation).
542
542
  try {
543
- await flushWorkflowProjections(artifactBasePath, { milestoneId: params.milestoneId });
543
+ await renderMilestoneShellProjections(artifactBasePath, params.milestoneId);
544
544
  } catch (projErr) {
545
545
  logWarning("tool", `complete-slice projection warning for ${params.milestoneId}/${params.sliceId}: ${(projErr as Error).message}`);
546
546
  }
@@ -6,11 +6,12 @@
6
6
  *
7
7
  * Validates inputs, writes task row and rendered SUMMARY.md to DB in a
8
8
  * transaction, then renders projections to disk and invalidates caches.
9
- * Projection write failures are reported as stale projections and do not roll
10
- * back committed DB state.
9
+ * If the critical task summary / plan projection write fails, the DB
10
+ * completion is compensated back to pending so DB state does not drift ahead
11
+ * of PLAN.md.
11
12
  */
12
13
 
13
- import { existsSync } from "node:fs";
14
+ import { existsSync, unlinkSync } from "node:fs";
14
15
  import { join } from "node:path";
15
16
 
16
17
  import type { CompleteTaskParams, EscalationArtifact } from "../types.js";
@@ -29,6 +30,7 @@ import {
29
30
  saveGateResult,
30
31
  getPendingGatesForTurn,
31
32
  } from "../gsd-db.js";
33
+ import { getWorkflowDatabasePath, ensureWorkflowDbAtPath } from "../db-workspace.js";
32
34
  import { getGatesForTurn } from "../gate-registry.js";
33
35
  import { gsdProjectionRoot, clearPathCache, resolveMilestonePath, resolveSlicePath } from "../paths.js";
34
36
  import { resolveCanonicalMilestoneRoot } from "../worktree-manager.js";
@@ -36,8 +38,10 @@ import { checkOwnership, taskUnitKey } from "../unit-ownership.js";
36
38
  import { saveFile, clearParseCache } from "../files.js";
37
39
  import { invalidateStateCache } from "../state.js";
38
40
  import { renderPlanCheckboxes } from "../markdown-renderer.js";
39
- import { renderSummaryContent } from "../workflow-projections.js";
40
- import { flushWorkflowProjections } from "../projection-flush.js";
41
+ import {
42
+ renderMilestoneShellProjections,
43
+ renderSummaryContent,
44
+ } from "../workflow-projections.js";
41
45
  import { writeManifest } from "../workflow-manifest.js";
42
46
  import { appendEvent } from "../workflow-events.js";
43
47
  import { logWarning, logError } from "../workflow-logger.js";
@@ -131,7 +135,7 @@ async function repairMissingTaskSummaryProjection(
131
135
  clearParseCache();
132
136
 
133
137
  try {
134
- await flushWorkflowProjections(artifactBasePath, { milestoneId: taskRow.milestone_id });
138
+ await renderMilestoneShellProjections(artifactBasePath, taskRow.milestone_id);
135
139
  } catch (projErr) {
136
140
  logWarning("tool", `complete-task repair projection warning: ${(projErr as Error).message}`);
137
141
  }
@@ -267,6 +271,7 @@ export async function handleCompleteTask(
267
271
  let guardError: string | null = null;
268
272
  let summaryMd = "";
269
273
  let repairTaskSummaryRow: TaskRow | null = null;
274
+ const rollbackDbPath = getWorkflowDatabasePath();
270
275
 
271
276
  // ── ADR-011 Phase 2: validate escalation payload BEFORE any side effects ─
272
277
  // Building the artifact runs the full shape validation (2-4 options, unique
@@ -423,8 +428,6 @@ export async function handleCompleteTask(
423
428
  return { error: guardError };
424
429
  }
425
430
 
426
- let projectionStale = false;
427
-
428
431
  // Resolve and write summary to disk
429
432
  const summaryPath = taskSummaryPath(
430
433
  artifactBasePath,
@@ -438,12 +441,48 @@ export async function handleCompleteTask(
438
441
 
439
442
  // Toggle or regenerate the plan projection from DB. Missing projection
440
443
  // files are rebuilt by the renderer instead of being skipped.
441
- await renderPlanCheckboxes(artifactBasePath, params.milestoneId, params.sliceId);
444
+ if (!ensureWorkflowDbAtPath(rollbackDbPath)) {
445
+ throw new Error(`database unavailable before plan projection render for ${params.milestoneId}/${params.sliceId}`);
446
+ }
447
+ const wrotePlan = await renderPlanCheckboxes(artifactBasePath, params.milestoneId, params.sliceId);
448
+ if (!wrotePlan) {
449
+ throw new Error(`plan projection write returned false for ${params.milestoneId}/${params.sliceId}`);
450
+ }
442
451
  } catch (renderErr) {
443
- projectionStale = true;
444
- logWarning("projection", `complete_task projection write failed for ${params.milestoneId}/${params.sliceId}/${params.taskId}; DB completion remains committed`, {
445
- error: (renderErr as Error).message,
446
- });
452
+ logWarning(
453
+ "projection",
454
+ `complete_task projection write failed for ${params.milestoneId}/${params.sliceId}/${params.taskId}`,
455
+ { error: (renderErr as Error).message },
456
+ );
457
+ let rollbackSucceeded = false;
458
+ try {
459
+ ensureWorkflowDbAtPath(rollbackDbPath);
460
+ deleteVerificationEvidence(params.milestoneId, params.sliceId, params.taskId);
461
+ updateTaskStatus(params.milestoneId, params.sliceId, params.taskId, "pending");
462
+ invalidateStateCache();
463
+ rollbackSucceeded = true;
464
+ } catch (rollbackErr) {
465
+ logWarning(
466
+ "projection",
467
+ `complete_task rollback failed after projection write failure for ${params.milestoneId}/${params.sliceId}/${params.taskId}: ${(rollbackErr as Error).message}`,
468
+ );
469
+ }
470
+ try {
471
+ if (existsSync(summaryPath)) unlinkSync(summaryPath);
472
+ } catch (summaryErr) {
473
+ logWarning(
474
+ "projection",
475
+ `complete_task could not remove SUMMARY.md after projection write failure for ${params.milestoneId}/${params.sliceId}/${params.taskId}: ${(summaryErr as Error).message}`,
476
+ );
477
+ }
478
+ // Clear path/parse caches regardless of rollback outcome so stale
479
+ // entries from the failed write attempt don't leak into subsequent calls.
480
+ clearPathCache();
481
+ clearParseCache();
482
+ const returnMsg = rollbackSucceeded
483
+ ? `complete_task projection write failed for ${params.milestoneId}/${params.sliceId}/${params.taskId}; rolled completion back to pending`
484
+ : `complete_task projection write failed for ${params.milestoneId}/${params.sliceId}/${params.taskId}; rollback also failed — task may remain complete with stale plan`;
485
+ return { error: returnMsg };
447
486
  }
448
487
 
449
488
  // ── Close gates owned by execute-task (Q5/Q6/Q7) for this task ────────
@@ -569,7 +608,7 @@ export async function handleCompleteTask(
569
608
  // Separate try/catch per step so a projection failure doesn't prevent
570
609
  // the event log entry (critical for worktree reconciliation).
571
610
  try {
572
- await flushWorkflowProjections(artifactBasePath, { milestoneId: params.milestoneId });
611
+ await renderMilestoneShellProjections(artifactBasePath, params.milestoneId);
573
612
  } catch (projErr) {
574
613
  logWarning("tool", `complete-task projection warning: ${(projErr as Error).message}`);
575
614
  }
@@ -597,6 +636,5 @@ export async function handleCompleteTask(
597
636
  milestoneId: params.milestoneId,
598
637
  summaryPath,
599
638
  ...(escalationMetadata ? { escalation: escalationMetadata } : {}),
600
- ...(projectionStale ? { stale: true } : {}),
601
639
  };
602
640
  }
@@ -31,6 +31,7 @@ export interface ExecToolDeps {
31
31
  env?: NodeJS.ProcessEnv;
32
32
  now?: () => Date;
33
33
  generateId?: () => string;
34
+ signal?: AbortSignal;
34
35
  }
35
36
 
36
37
  export type UatExecIntent =
@@ -74,7 +75,7 @@ const UAT_EXEC_INTENT_ALIASES: Record<string, UatExecIntent> = {
74
75
  export function buildExecOptions(
75
76
  baseDir: string,
76
77
  cfg: ContextModeConfig | undefined,
77
- extras?: Pick<ExecSandboxOptions, "env" | "now" | "generateId">,
78
+ extras?: Pick<ExecSandboxOptions, "env" | "now" | "generateId" | "signal">,
78
79
  ): ExecSandboxOptions {
79
80
  const allowlist = Array.isArray(cfg?.exec_env_allowlist) ? cfg!.exec_env_allowlist! : EXEC_DEFAULTS.envAllowlist;
80
81
  const stdoutCap = clampNumber(
@@ -209,7 +210,7 @@ export async function executeGsdExec(
209
210
  const opts = buildExecOptions(
210
211
  deps.baseDir,
211
212
  deps.preferences?.context_mode,
212
- { env: deps.env, now: deps.now, generateId: deps.generateId },
213
+ { env: deps.env, now: deps.now, generateId: deps.generateId, signal: deps.signal },
213
214
  );
214
215
  const run = deps.run ?? runExecSandbox;
215
216
 
@@ -308,6 +309,7 @@ function formatResult(result: ExecSandboxResult): ToolExecutionResult {
308
309
  exit_code: result.exit_code,
309
310
  signal: result.signal,
310
311
  timed_out: result.timed_out,
312
+ aborted: result.aborted === true,
311
313
  force_resolved: result.force_resolved,
312
314
  duration_ms: result.duration_ms,
313
315
  stdout_bytes: result.stdout_bytes,
@@ -318,13 +320,15 @@ function formatResult(result: ExecSandboxResult): ToolExecutionResult {
318
320
  stderr_path: result.stderr_path,
319
321
  meta_path: result.meta_path,
320
322
  },
321
- isError: result.timed_out || result.signal !== null || result.exit_code !== 0,
323
+ isError: result.aborted === true || result.timed_out || result.signal !== null || result.exit_code !== 0,
322
324
  };
323
325
  }
324
326
 
325
327
  function formatExit(result: ExecSandboxResult): string {
326
328
  // force_resolved means a non-closing (D-state) child was force-resolved past its
327
329
  // hard deadline rather than observed exiting; distinguish it from a clean timeout.
330
+ if (result.aborted && result.force_resolved) return "aborted(force-killed)";
331
+ if (result.aborted) return "aborted";
328
332
  if (result.force_resolved) return "timeout(force-killed)";
329
333
  if (result.timed_out) return "timeout";
330
334
  if (result.signal) return `signal:${result.signal}`;
@@ -135,14 +135,38 @@ const CONTEXT_MODE_GUIDANCE_BY_LANE: Record<Exclude<ContextModePolicy, "none">,
135
135
  "Use `gsd_resume` for prior context, `gsd_exec_search` for saved evidence, and `gsd_exec` for noisy doc validation commands.",
136
136
  };
137
137
 
138
- // Per-unit overrides win over the lane default. run-uat's tool contract
139
- // forbids `gsd_exec`/`gsd_exec_search` (acceptance evidence must flow through
140
- // `gsd_uat_exec`) and Claude Code dispatch strips the tools entirely, so the
141
- // shared verification-lane guidance would steer the agent into calling an
142
- // unavailable tool.
143
- const CONTEXT_MODE_GUIDANCE_BY_UNIT: Record<string, string> = {
138
+ // Per-unit overrides win over the lane default. Some units intentionally run
139
+ // with narrower tool contracts than their shared Context Mode lane, so their
140
+ // guidance must name only tools the unit can actually call.
141
+ export const CONTEXT_MODE_GUIDANCE_BY_UNIT: Readonly<Record<string, string>> = {
142
+ "discuss-milestone":
143
+ "Use `ask_user_questions` to continue the milestone interview, then persist outcomes with `gsd_summary_save`, `gsd_decision_save`, `gsd_requirement_save`, `gsd_requirement_update`, `gsd_plan_milestone`, or `gsd_milestone_generate_id` as appropriate.",
144
+ "discuss-slice":
145
+ "Use `ask_user_questions` to continue the slice interview, then persist outcomes with `gsd_summary_save` or `gsd_decision_save` as appropriate.",
146
+ "discuss-project":
147
+ "Use `ask_user_questions` to continue the project interview, then persist outcomes with `gsd_summary_save`, `gsd_decision_save`, or `gsd_requirement_save` as appropriate.",
148
+ "discuss-requirements":
149
+ "Use `ask_user_questions` to continue the requirements interview, then persist outcomes with `gsd_requirement_save` or `gsd_summary_save` as appropriate.",
150
+ "replan-slice":
151
+ "Use `gsd_replan_slice` to persist the revised slice plan, and `gsd_decision_save` for planning decisions that need durable rationale.",
152
+ "reassess-roadmap":
153
+ "Use `gsd_milestone_status` to inspect current milestone state, then `gsd_reassess_roadmap` to persist the roadmap reassessment.",
144
154
  "run-uat":
145
155
  "Use `gsd_uat_exec` for acceptance checks so evidence is typed as UAT-owned, and `gsd_resume` after compaction or resume.",
156
+ "research-project":
157
+ "Dispatch parallel scout subagents for stack, features, architecture, and pitfalls research; each writes one file under `.gsd/research/` (`STACK.md`, `FEATURES.md`, `ARCHITECTURE.md`, `PITFALLS.md`).",
158
+ "gate-evaluate":
159
+ "Use `subagent` to dispatch tester agents, then persist each gate with `gsd_save_gate_result`; rely on testers for verification evidence.",
160
+ };
161
+
162
+ // Per-unit guidance for the nested render mode (renderMode: "nested"), used when this
163
+ // unit's Context Mode line is embedded into a subagent prompt — e.g. the tester prompts
164
+ // dispatched by gate-evaluate. Must instruct the subagent on what IT should do, not
165
+ // re-state the parent coordinator's dispatch instructions. Falls back to
166
+ // CONTEXT_MODE_GUIDANCE_BY_UNIT then the lane default when no nested entry exists.
167
+ export const CONTEXT_MODE_NESTED_GUIDANCE_BY_UNIT: Readonly<Record<string, string>> = {
168
+ "gate-evaluate":
169
+ "Run verification checks to answer the gate question, then persist the verdict with `gsd_save_gate_result`.",
146
170
  };
147
171
 
148
172
  /**
@@ -160,7 +184,9 @@ export function composeContextModeInstructions(
160
184
 
161
185
  const lane = CONTEXT_MODE_LANE_LABELS[manifest.contextMode];
162
186
  const guidance =
163
- CONTEXT_MODE_GUIDANCE_BY_UNIT[unitType] ?? CONTEXT_MODE_GUIDANCE_BY_LANE[manifest.contextMode];
187
+ (opts.renderMode === "nested" ? CONTEXT_MODE_NESTED_GUIDANCE_BY_UNIT[unitType] : undefined)
188
+ ?? CONTEXT_MODE_GUIDANCE_BY_UNIT[unitType]
189
+ ?? CONTEXT_MODE_GUIDANCE_BY_LANE[manifest.contextMode];
164
190
  if (opts.renderMode === "nested") {
165
191
  return `Context Mode (${lane} lane): ${guidance}`;
166
192
  }
@@ -98,7 +98,13 @@ export const UNIT_REGISTRY = {
98
98
  scopeClass: "standard",
99
99
  phaseChain: ["research"],
100
100
  toolContract: {
101
- allowedGsdTools: ["gsd_summary_save", "gsd_decision_save"],
101
+ allowedGsdTools: [
102
+ "gsd_summary_save",
103
+ "gsd_decision_save",
104
+ "gsd_exec",
105
+ "gsd_exec_search",
106
+ "gsd_resume",
107
+ ],
102
108
  requiredWorkflowTools: ["gsd_summary_save"],
103
109
  },
104
110
  },
@@ -154,7 +160,15 @@ export const UNIT_REGISTRY = {
154
160
  scopeClass: "section-close",
155
161
  phaseChain: ["validation", "planning"],
156
162
  toolContract: {
157
- allowedGsdTools: ["gsd_milestone_status", "gsd_validate_milestone", "gsd_reassess_roadmap", "subagent"],
163
+ allowedGsdTools: [
164
+ "gsd_milestone_status",
165
+ "gsd_exec",
166
+ "gsd_exec_search",
167
+ "gsd_resume",
168
+ "gsd_validate_milestone",
169
+ "gsd_reassess_roadmap",
170
+ "subagent",
171
+ ],
158
172
  requiredWorkflowTools: ["gsd_milestone_status", "gsd_validate_milestone", "gsd_reassess_roadmap"],
159
173
  },
160
174
  },
@@ -165,6 +179,9 @@ export const UNIT_REGISTRY = {
165
179
  toolContract: {
166
180
  allowedGsdTools: [
167
181
  "gsd_milestone_status",
182
+ "gsd_exec",
183
+ "gsd_exec_search",
184
+ "gsd_resume",
168
185
  "gsd_requirement_update",
169
186
  "gsd_summary_save",
170
187
  "gsd_complete_milestone",
@@ -244,6 +261,8 @@ export const UNIT_REGISTRY = {
244
261
  allowedGsdTools: [
245
262
  "gsd_milestone_status",
246
263
  "gsd_exec",
264
+ "gsd_exec_search",
265
+ "gsd_resume",
247
266
  "gsd_slice_complete",
248
267
  "gsd_task_reopen",
249
268
  "gsd_replan_slice",
@@ -326,7 +345,13 @@ export const UNIT_REGISTRY = {
326
345
  scopeClass: "execute-task",
327
346
  phaseChain: ["execution"],
328
347
  toolContract: {
329
- allowedGsdTools: ["gsd_task_complete", "gsd_summary_save", "gsd_decision_save"],
348
+ allowedGsdTools: [
349
+ "gsd_milestone_status",
350
+ "gsd_task_complete",
351
+ "gsd_exec",
352
+ "gsd_summary_save",
353
+ "gsd_decision_save",
354
+ ],
330
355
  requiredWorkflowTools: ["gsd_task_complete", "gsd_summary_save"],
331
356
  },
332
357
  },
@@ -413,12 +438,15 @@ export const UNIT_REGISTRY = {
413
438
  requiredWorkflowTools: ["ask_user_questions"],
414
439
  },
415
440
  },
441
+ // research-project dispatches 4 parallel scout subagents (Task calls); each scout
442
+ // writes one file under .gsd/research/ directly. The parent coordinator does not
443
+ // call gsd_summary_save or gsd_decision_save — the scouts own their own output.
416
444
  "research-project": {
417
445
  kind: "primary",
418
446
  scopeClass: "standard",
419
447
  phaseChain: ["research"],
420
448
  toolContract: {
421
- allowedGsdTools: ["gsd_summary_save", "gsd_decision_save"],
449
+ allowedGsdTools: [],
422
450
  requiredWorkflowTools: [],
423
451
  },
424
452
  },
@@ -1,6 +1,8 @@
1
1
  // Project/App: gsd-pi
2
2
  // File Purpose: Block new workflow entry when completed milestone branches are still unmerged.
3
3
 
4
+ import { execFileSync } from "node:child_process";
5
+
4
6
  import {
5
7
  nativeBranchExists,
6
8
  nativeDetectMainBranch,
@@ -12,17 +14,23 @@ import { ensureDbOpen } from "./bootstrap/dynamic-tools.js";
12
14
  import { getAllMilestones } from "./gsd-db.js";
13
15
  import { resolveMilestoneIntegrationBranch } from "./git-service.js";
14
16
  import { loadEffectiveGSDPreferences } from "./preferences.js";
15
- import { captureRootDirtySnapshot, type RootDirtyEntry } from "./root-write-leak-guard.js";
16
17
  import { isClosedStatus } from "./status-guards.js";
17
18
 
19
+ export interface UnmergedMilestoneDirtyEntry {
20
+ path: string;
21
+ status: string;
22
+ }
23
+
18
24
  export interface UnmergedMilestoneBlocker {
19
25
  milestoneId: string;
20
26
  branch: string;
21
27
  integrationBranch: string;
22
28
  files: string[];
23
- dirtyOverlap: RootDirtyEntry[];
29
+ dirtyOverlap: UnmergedMilestoneDirtyEntry[];
24
30
  }
25
31
 
32
+ type UnmergedMilestoneDirtySnapshot = Map<string, UnmergedMilestoneDirtyEntry>;
33
+
26
34
  const BLOCKED_COMMANDS = new Set([
27
35
  "auto",
28
36
  "next",
@@ -67,6 +75,32 @@ function isRuntimePath(path: string): boolean {
67
75
  return path === ".gsd" || path.startsWith(".gsd/");
68
76
  }
69
77
 
78
+ function captureDirtyPathStatusSnapshot(rootPath: string): UnmergedMilestoneDirtySnapshot {
79
+ const snapshot: UnmergedMilestoneDirtySnapshot = new Map();
80
+ let status = "";
81
+ try {
82
+ status = execFileSync("git", ["status", "--porcelain", "--untracked-files=all"], {
83
+ cwd: rootPath,
84
+ stdio: ["ignore", "pipe", "pipe"],
85
+ encoding: "utf-8",
86
+ });
87
+ } catch {
88
+ return snapshot;
89
+ }
90
+
91
+ for (const line of status.split("\n")) {
92
+ if (!line.trim()) continue;
93
+ const code = line.slice(0, 2);
94
+ const path = line.slice(3).replace(/^"|"$/g, "");
95
+ if (!path || isRuntimePath(path)) continue;
96
+ snapshot.set(path, {
97
+ path,
98
+ status: code.trim() || code,
99
+ });
100
+ }
101
+ return snapshot;
102
+ }
103
+
70
104
  function formatCommandLabel(attemptedCommand: string): string {
71
105
  const trimmed = attemptedCommand.trim();
72
106
  return trimmed ? `/gsd ${trimmed}` : "/gsd";
@@ -132,7 +166,7 @@ export async function findUnmergedCompletedMilestones(base: string): Promise<Unm
132
166
  await ensureDbOpen(base);
133
167
 
134
168
  const blockers: UnmergedMilestoneBlocker[] = [];
135
- const dirtyByPath = captureRootDirtySnapshot(base);
169
+ let dirtyByPath: UnmergedMilestoneDirtySnapshot | null = null;
136
170
 
137
171
  for (const milestone of getAllMilestones()) {
138
172
  if (!isClosedStatus(milestone.status)) continue;
@@ -158,14 +192,16 @@ export async function findUnmergedCompletedMilestones(base: string): Promise<Unm
158
192
  const uniqueFiles = [...new Set(files)].sort();
159
193
  if (uniqueFiles.length === 0) continue;
160
194
 
195
+ if (!dirtyByPath) dirtyByPath = captureDirtyPathStatusSnapshot(base);
196
+ const dirtySnapshot = dirtyByPath;
161
197
  blockers.push({
162
198
  milestoneId: milestone.id,
163
199
  branch,
164
200
  integrationBranch,
165
201
  files: uniqueFiles,
166
202
  dirtyOverlap: uniqueFiles
167
- .map((path) => dirtyByPath.get(path))
168
- .filter((entry): entry is RootDirtyEntry => Boolean(entry)),
203
+ .map((path) => dirtySnapshot.get(path))
204
+ .filter((entry): entry is UnmergedMilestoneDirtyEntry => Boolean(entry)),
169
205
  });
170
206
  }
171
207
 
@@ -7,7 +7,7 @@ import { getAutoWorktreePath, isInAutoWorktree } from "./auto-worktree.js";
7
7
  import { ensureDbOpen } from "./bootstrap/dynamic-tools.js";
8
8
  import { refreshWorkflowDatabaseFromDisk } from "./db-workspace.js";
9
9
  import { getIsolationMode } from "./preferences.js";
10
- import { deriveState } from "./state.js";
10
+ import { deriveState, invalidateStateCache } from "./state.js";
11
11
  import type { GSDState } from "./types.js";
12
12
  import { detectWorktreeName } from "./worktree.js";
13
13
 
@@ -133,12 +133,7 @@ export function formatValidationBlockedMessage(
133
133
  ].join("\n\n");
134
134
  }
135
135
 
136
- export async function getValidationBlockMessageForBase(
137
- base: string,
138
- attemptedCommand = "",
139
- ): Promise<string | null> {
140
- await ensureDbOpen(base);
141
- refreshWorkflowDatabaseFromDisk();
136
+ async function deriveValidationBlockState(base: string): Promise<GSDState> {
142
137
  let state = await deriveState(base);
143
138
 
144
139
  if (
@@ -153,5 +148,16 @@ export async function getValidationBlockMessageForBase(
153
148
  }
154
149
  }
155
150
 
151
+ return state;
152
+ }
153
+
154
+ export async function getValidationBlockMessageForBase(
155
+ base: string,
156
+ attemptedCommand = "",
157
+ ): Promise<string | null> {
158
+ await ensureDbOpen(base);
159
+ refreshWorkflowDatabaseFromDisk();
160
+ invalidateStateCache();
161
+ const state = await deriveValidationBlockState(base);
156
162
  return formatValidationBlockedMessage(state, attemptedCommand);
157
163
  }
@@ -451,16 +451,14 @@ export async function renderStateProjection(basePath: string): Promise<void> {
451
451
  }
452
452
  }
453
453
 
454
- // ─── renderAllProjections ───────────────────────────────────────────────
454
+ // ─── renderMilestoneShellProjections ─────────────────────────────────────
455
455
 
456
456
  /**
457
- * Regenerate all projection files for a milestone from DB state.
458
- * All calls are wrapped in try/catch projection failure is non-fatal per D-02.
457
+ * Regenerate milestone-level projections (roadmap, top-level views, STATE.md).
458
+ * Does not touch slice PLAN.md or task SUMMARY.md those have authoritative
459
+ * renderers in plan-slice / complete-task.
459
460
  */
460
- export async function renderAllProjections(basePath: string, milestoneId: string): Promise<void> {
461
- // Delegate to the authoritative roadmap renderer — the reduced
462
- // renderRoadmapProjection omits sections like ## Boundary Map and would
463
- // clobber the output written by plan-milestone / reassess-roadmap.
461
+ export async function renderMilestoneShellProjections(basePath: string, milestoneId: string): Promise<void> {
464
462
  try {
465
463
  await renderRoadmapFromDb(basePath, milestoneId);
466
464
  } catch (err) {
@@ -476,6 +474,21 @@ export async function renderAllProjections(basePath: string, milestoneId: string
476
474
  } catch (err) {
477
475
  logWarning("projection", `renderTopLevelQueueFromDb failed: ${(err as Error).message}`);
478
476
  }
477
+ try {
478
+ await renderStateProjection(basePath);
479
+ } catch (err) {
480
+ logWarning("projection", `renderStateProjection failed: ${(err as Error).message}`);
481
+ }
482
+ }
483
+
484
+ // ─── renderAllProjections ───────────────────────────────────────────────
485
+
486
+ /**
487
+ * Regenerate all projection files for a milestone from DB state.
488
+ * All calls are wrapped in try/catch — projection failure is non-fatal per D-02.
489
+ */
490
+ export async function renderAllProjections(basePath: string, milestoneId: string): Promise<void> {
491
+ await renderMilestoneShellProjections(basePath, milestoneId);
479
492
 
480
493
  // Query all slices for this milestone
481
494
  const sliceRows = getMilestoneSlices(milestoneId);
@@ -498,13 +511,6 @@ export async function renderAllProjections(basePath: string, milestoneId: string
498
511
  }
499
512
  }
500
513
  }
501
-
502
- // Render STATE.md
503
- try {
504
- await renderStateProjection(basePath);
505
- } catch (err) {
506
- logWarning("projection", `renderStateProjection failed: ${(err as Error).message}`);
507
- }
508
514
  }
509
515
 
510
516
  // ─── regenerateIfMissing ────────────────────────────────────────────────
@@ -7,6 +7,7 @@ import { getAutoWorktreePath } from "./auto-worktree.js";
7
7
  import { isSafeToAutoResolve } from "./auto-worktree.js";
8
8
  import { ensureDbOpen } from "./bootstrap/dynamic-tools.js";
9
9
  import {
10
+ listMergeStateBlockers,
10
11
  probeGitConflictState,
11
12
  reconcileGitConflictsOnSignal,
12
13
  type GitConflictProbeResult,
@@ -30,6 +31,9 @@ export type WorkspaceGitReadyResult =
30
31
  targets: string[];
31
32
  };
32
33
 
34
+ const CLEAN_TARGET_CACHE_TTL_MS = 10_000;
35
+ const cleanTargetProbeCache = new Map<string, number>();
36
+
33
37
  function normalizeTargetPath(path: string): string {
34
38
  try {
35
39
  return realpathSync(path);
@@ -110,6 +114,25 @@ function formatBlockReason(
110
114
 
111
115
  async function ensureTargetGitReady(target: string): Promise<WorkspaceGitReadyResult> {
112
116
  const fixesApplied: string[] = [];
117
+ const cacheKey = normalizeTargetPath(target);
118
+ const cachedCleanAt = cleanTargetProbeCache.get(cacheKey);
119
+ if (cachedCleanAt !== undefined) {
120
+ if (Date.now() - cachedCleanAt < CLEAN_TARGET_CACHE_TTL_MS) {
121
+ // Merge-state markers (MERGE_HEAD, rebase-apply, rebase-merge) are the most
122
+ // common way a repo transitions from clean to dirty within the TTL window,
123
+ // including when a non-git folder is initialized as a repo mid-TTL and a
124
+ // merge immediately introduces conflicts. Check them here — they are pure
125
+ // existsSync calls with no git subprocess — and fall through to a full probe
126
+ // only when markers are present.
127
+ if (listMergeStateBlockers(cacheKey).length === 0) {
128
+ return { ok: true, fixesApplied };
129
+ }
130
+ cleanTargetProbeCache.delete(cacheKey);
131
+ } else {
132
+ cleanTargetProbeCache.delete(cacheKey);
133
+ }
134
+ }
135
+
113
136
  let probe = probeGitConflictState(target);
114
137
 
115
138
  for (let attempt = 0; attempt < 3 && probe.status === "dirty"; attempt++) {
@@ -129,6 +152,7 @@ async function ensureTargetGitReady(target: string): Promise<WorkspaceGitReadyRe
129
152
  }
130
153
 
131
154
  if (probe.status === "unknown") {
155
+ cleanTargetProbeCache.delete(cacheKey);
132
156
  return {
133
157
  ok: false,
134
158
  reason: formatBlockReason("unrecoverable", [], [target]),
@@ -141,6 +165,7 @@ async function ensureTargetGitReady(target: string): Promise<WorkspaceGitReadyRe
141
165
 
142
166
  const conflictedPaths = productConflictPaths(probe);
143
167
  if (conflictedPaths.length > 0 || probe.checkFailures.length > 0) {
168
+ cleanTargetProbeCache.delete(cacheKey);
144
169
  return {
145
170
  ok: false,
146
171
  reason: formatBlockReason("product-conflicts", conflictedPaths, [target]),
@@ -151,6 +176,12 @@ async function ensureTargetGitReady(target: string): Promise<WorkspaceGitReadyRe
151
176
  };
152
177
  }
153
178
 
179
+ if (probe.status === "clean") {
180
+ cleanTargetProbeCache.set(cacheKey, Date.now());
181
+ } else {
182
+ cleanTargetProbeCache.delete(cacheKey);
183
+ }
184
+
154
185
  return { ok: true, fixesApplied };
155
186
  }
156
187