@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
@@ -4,7 +4,7 @@ import { mkdtempSync, mkdirSync, realpathSync, rmSync, writeFileSync } from "nod
4
4
  import { join } from "node:path";
5
5
  import { tmpdir } from "node:os";
6
6
 
7
- import { resolveExpectedArtifactPath } from "../auto-artifact-paths.ts";
7
+ import { resolveExpectedArtifactPath, resolveSliceResearchLocation, resolveExistingSliceResearchPath } from "../auto-artifact-paths.ts";
8
8
  import { clearPathCache, _clearGsdRootCache, isLegacyMilestonesLayout, milestonesDir } from "../paths.ts";
9
9
 
10
10
  test("worktree artifact resolution falls back to project .gsd artifacts", () => {
@@ -359,3 +359,100 @@ test("canonical worktree + project-root legacy dir produces legacy filename (#bu
359
359
  rmSync(root, { recursive: true, force: true });
360
360
  }
361
361
  });
362
+
363
+ // ── resolveSliceResearchLocation / resolveExistingSliceResearchPath ───────────
364
+ //
365
+ // Added in the code-quality consolidation PR as the shared dual-path resolver
366
+ // for slice RESEARCH files (worktree projection first, then canonical path
367
+ // fallback). These tests guard: missing file → null pair; existing legacy-
368
+ // layout file → correct absolute and relative paths.
369
+
370
+ test("resolveSliceResearchLocation returns null pair when no RESEARCH file exists", () => {
371
+ const root = realpathSync(mkdtempSync(join(tmpdir(), "gsd-research-missing-")));
372
+ try {
373
+ const milestoneDir = join(root, ".gsd", "milestones", "M001");
374
+ mkdirSync(join(milestoneDir, "slices", "S01"), { recursive: true });
375
+ // Content-bearing dir so it is not treated as META-only, but no RESEARCH file.
376
+ writeFileSync(join(milestoneDir, "M001-CONTEXT.md"), "# context\n");
377
+
378
+ _clearGsdRootCache();
379
+ clearPathCache();
380
+
381
+ const result = resolveSliceResearchLocation(root, "M001", "S01");
382
+ assert.strictEqual(result.absolutePath, null, "absolutePath must be null when no RESEARCH exists");
383
+ assert.strictEqual(result.relativePath, null, "relativePath must be null when no RESEARCH exists");
384
+ } finally {
385
+ _clearGsdRootCache();
386
+ clearPathCache();
387
+ rmSync(root, { recursive: true, force: true });
388
+ }
389
+ });
390
+
391
+ test("resolveExistingSliceResearchPath returns null when no RESEARCH file exists", () => {
392
+ const root = realpathSync(mkdtempSync(join(tmpdir(), "gsd-research-null-")));
393
+ try {
394
+ const milestoneDir = join(root, ".gsd", "milestones", "M001");
395
+ mkdirSync(join(milestoneDir, "slices", "S01"), { recursive: true });
396
+ writeFileSync(join(milestoneDir, "M001-CONTEXT.md"), "# context\n");
397
+
398
+ _clearGsdRootCache();
399
+ clearPathCache();
400
+
401
+ assert.strictEqual(
402
+ resolveExistingSliceResearchPath(root, "M001", "S01"),
403
+ null,
404
+ "must return null when RESEARCH file is absent",
405
+ );
406
+ } finally {
407
+ _clearGsdRootCache();
408
+ clearPathCache();
409
+ rmSync(root, { recursive: true, force: true });
410
+ }
411
+ });
412
+
413
+ test("resolveSliceResearchLocation finds existing RESEARCH in legacy layout", () => {
414
+ const root = realpathSync(mkdtempSync(join(tmpdir(), "gsd-research-legacy-")));
415
+ try {
416
+ const milestoneDir = join(root, ".gsd", "milestones", "M001");
417
+ const sliceDir = join(milestoneDir, "slices", "S01");
418
+ mkdirSync(sliceDir, { recursive: true });
419
+ writeFileSync(join(milestoneDir, "M001-CONTEXT.md"), "# context\n");
420
+ const researchFile = join(sliceDir, "S01-RESEARCH.md");
421
+ writeFileSync(researchFile, "# slice research\n");
422
+
423
+ _clearGsdRootCache();
424
+ clearPathCache();
425
+
426
+ const result = resolveSliceResearchLocation(root, "M001", "S01");
427
+ assert.ok(result.absolutePath !== null, "absolutePath must be non-null when RESEARCH exists");
428
+ assert.ok(result.relativePath !== null, "relativePath must be non-null when RESEARCH exists");
429
+ assert.equal(result.absolutePath, researchFile, "absolutePath must point to the RESEARCH file");
430
+ } finally {
431
+ _clearGsdRootCache();
432
+ clearPathCache();
433
+ rmSync(root, { recursive: true, force: true });
434
+ }
435
+ });
436
+
437
+ test("resolveExistingSliceResearchPath returns absolute path when RESEARCH exists (legacy layout)", () => {
438
+ const root = realpathSync(mkdtempSync(join(tmpdir(), "gsd-research-existing-")));
439
+ try {
440
+ const milestoneDir = join(root, ".gsd", "milestones", "M001");
441
+ const sliceDir = join(milestoneDir, "slices", "S01");
442
+ mkdirSync(sliceDir, { recursive: true });
443
+ writeFileSync(join(milestoneDir, "M001-CONTEXT.md"), "# context\n");
444
+ const researchFile = join(sliceDir, "S01-RESEARCH.md");
445
+ writeFileSync(researchFile, "# research\n");
446
+
447
+ _clearGsdRootCache();
448
+ clearPathCache();
449
+
450
+ const result = resolveExistingSliceResearchPath(root, "M001", "S01");
451
+ assert.ok(result !== null, "must return non-null when RESEARCH exists");
452
+ assert.equal(result, researchFile, "must return the absolute path to the RESEARCH file");
453
+ } finally {
454
+ _clearGsdRootCache();
455
+ clearPathCache();
456
+ rmSync(root, { recursive: true, force: true });
457
+ }
458
+ });
@@ -2,7 +2,7 @@
2
2
  // File Purpose: Tests auto-mode artifact verification and recovery behavior.
3
3
  import test, { afterEach } from "node:test";
4
4
  import assert from "node:assert/strict";
5
- import { mkdtempSync, mkdirSync, rmSync, writeFileSync, existsSync, readFileSync } from "node:fs";
5
+ import { chmodSync, mkdtempSync, mkdirSync, rmSync, writeFileSync, existsSync, readFileSync } from "node:fs";
6
6
  import { join } from "node:path";
7
7
  import { tmpdir } from "node:os";
8
8
  import { randomUUID } from "node:crypto";
@@ -964,6 +964,45 @@ function makeGitBase(): string {
964
964
  return base;
965
965
  }
966
966
 
967
+ function shellQuote(value: string): string {
968
+ return `'${value.replace(/'/g, "'\\''")}'`;
969
+ }
970
+
971
+ function withLoggedGitCommands<T>(base: string, action: () => T): { result: T; commands: string[] } {
972
+ const realGit = execFileSync("which", ["git"], {
973
+ encoding: "utf-8",
974
+ stdio: ["ignore", "pipe", "pipe"],
975
+ }).trim().split(/\r?\n/)[0];
976
+ if (!realGit) throw new Error("Unable to resolve git executable for invocation logging test");
977
+ const binDir = join(base, ".git-wrapper-bin");
978
+ const logFile = join(base, "git-invocations.log");
979
+ mkdirSync(binDir, { recursive: true });
980
+ const wrapper = join(binDir, "git");
981
+ writeFileSync(
982
+ wrapper,
983
+ [
984
+ "#!/usr/bin/env bash",
985
+ `printf '%s\\n' "$1" >> ${shellQuote(logFile)}`,
986
+ `exec ${shellQuote(realGit)} "$@"`,
987
+ "",
988
+ ].join("\n"),
989
+ );
990
+ chmodSync(wrapper, 0o755);
991
+
992
+ const originalPath = process.env.PATH;
993
+ process.env.PATH = originalPath ? `${binDir}:${originalPath}` : binDir;
994
+ try {
995
+ const result = action();
996
+ const commands = existsSync(logFile)
997
+ ? readFileSync(logFile, "utf-8").split(/\r?\n/).filter(Boolean)
998
+ : [];
999
+ return { result, commands };
1000
+ } finally {
1001
+ if (originalPath === undefined) delete process.env.PATH;
1002
+ else process.env.PATH = originalPath;
1003
+ }
1004
+ }
1005
+
967
1006
  test("hasImplementationArtifacts returns false when only .gsd/ files committed (#1703)", () => {
968
1007
  const base = makeGitBase();
969
1008
  try {
@@ -1344,6 +1383,77 @@ test("hasImplementationArtifacts finds implementation commits when .gsd/ is giti
1344
1383
  }
1345
1384
  });
1346
1385
 
1386
+ test("hasImplementationArtifacts scans GSD-tagged history without per-commit diff-tree forks (#892)", { skip: process.platform === "win32" }, () => {
1387
+ const base = makeGitBase();
1388
+ try {
1389
+ writeFileSync(join(base, ".git", "info", "exclude"), ".gsd/\n");
1390
+ mkdirSync(join(base, ".gsd", "milestones", "M001", "slices", "S01", "tasks"), { recursive: true });
1391
+ writeFileSync(
1392
+ join(base, ".gsd", "milestones", "M001", "slices", "S01", "tasks", "T01-SUMMARY.md"),
1393
+ "# Summary",
1394
+ );
1395
+
1396
+ mkdirSync(join(base, "src"), { recursive: true });
1397
+ for (let i = 0; i < 3; i++) {
1398
+ writeFileSync(join(base, "src", `feature-${i}.ts`), `export const feature${i} = true;\n`);
1399
+ execFileSync("git", ["add", "src"], { cwd: base, stdio: "ignore" });
1400
+ execFileSync(
1401
+ "git",
1402
+ ["commit", "-m", `feat: materialize M001 evidence ${i}\n\nGSD-Task: S01/T01`],
1403
+ { cwd: base, stdio: "ignore" },
1404
+ );
1405
+ }
1406
+
1407
+ const { result, commands } = withLoggedGitCommands(base, () => hasImplementationArtifacts(base, "M001"));
1408
+ assert.equal(result, "present", "milestone-tagged commits should still prove implementation evidence");
1409
+ assert.ok(commands.includes("log"), "milestone evidence fallback should scan history with git log");
1410
+ assert.equal(commands.filter((command) => command === "diff-tree").length, 0);
1411
+ } finally {
1412
+ cleanup(base);
1413
+ }
1414
+ });
1415
+
1416
+ test("hasImplementationArtifacts backfill scans commit records without per-commit diff-tree forks (#892)", { skip: process.platform === "win32" }, () => {
1417
+ const base = makeGitBase();
1418
+ try {
1419
+ mkdirSync(join(base, ".gsd"), { recursive: true });
1420
+ openDatabase(join(base, ".gsd", "gsd.db"));
1421
+ insertMilestone({ id: "M001", title: "Milestone One", status: "active" });
1422
+ insertSlice({
1423
+ id: "S01",
1424
+ milestoneId: "M001",
1425
+ title: "Slice One",
1426
+ status: "complete",
1427
+ risk: "low",
1428
+ depends: [],
1429
+ });
1430
+ insertTask({
1431
+ id: "T01",
1432
+ sliceId: "S01",
1433
+ milestoneId: "M001",
1434
+ title: "Task One",
1435
+ status: "complete",
1436
+ keyFiles: ["src/expected-0.ts", "src/expected-1.ts"],
1437
+ planning: { files: ["src/expected-0.ts", "src/expected-1.ts"] },
1438
+ });
1439
+
1440
+ mkdirSync(join(base, "src"), { recursive: true });
1441
+ for (let i = 0; i < 2; i++) {
1442
+ writeFileSync(join(base, "src", `expected-${i}.ts`), `export const expected${i} = true;\n`);
1443
+ execFileSync("git", ["add", "src"], { cwd: base, stdio: "ignore" });
1444
+ execFileSync("git", ["commit", "-m", `feat: untagged implementation ${i}`], { cwd: base, stdio: "ignore" });
1445
+ }
1446
+
1447
+ const { result, commands } = withLoggedGitCommands(base, () => hasImplementationArtifacts(base, "M001"));
1448
+ assert.equal(result, "present", "completed task file hints should still backfill untagged commits");
1449
+ assert.equal(getMilestoneCommitAttributionShas("M001").length, 2);
1450
+ assert.ok(commands.includes("log"), "backfill should scan commit records with git log");
1451
+ assert.equal(commands.filter((command) => command === "diff-tree").length, 0);
1452
+ } finally {
1453
+ cleanup(base);
1454
+ }
1455
+ });
1456
+
1347
1457
  test("hasImplementationArtifacts binds GSD-Task trailer to milestone via DB state when .gsd/ is gitignored", () => {
1348
1458
  const base = makeGitBase();
1349
1459
  try {
@@ -101,6 +101,32 @@ test("analyzeSessionContext buckets injections, tool results, and loaded skills"
101
101
  assert.equal(result.subagentSpawns, 1);
102
102
  });
103
103
 
104
+ test("analyzeSessionContext bounds large assistant tool-call arguments", () => {
105
+ const result = analyzeSessionContext(sessionEntries({
106
+ role: "assistant",
107
+ content: [
108
+ {
109
+ type: "toolCall",
110
+ id: "tc-save",
111
+ name: "gsd_summary_save",
112
+ arguments: {
113
+ path: ".gsd/milestones/M001/M001-SUMMARY.md",
114
+ artifact_type: "SUMMARY",
115
+ content: "x".repeat(20_000),
116
+ },
117
+ },
118
+ ],
119
+ timestamp: TS,
120
+ }), PROVIDER);
121
+
122
+ const assistant = result.conversationSections.find((section) => section.label === "Assistant responses");
123
+ assert.ok(assistant);
124
+ assert.ok(
125
+ assistant.tokens < 200,
126
+ `assistant tool-call arguments should be redacted before token counting, got ${assistant.tokens} tokens`,
127
+ );
128
+ });
129
+
104
130
  test("formatContextReport lists skills and subagents", () => {
105
131
  const report = buildContextBreakdown({
106
132
  modelLabel: "claude-code/claude-sonnet-4-6",
@@ -460,6 +460,7 @@ describe("Batch 3 prompt templates resolve", () => {
460
460
  const out = loadPrompt("code-review", { scope: "s", depth: "deep", fixMode: "off", reviewId: "001" });
461
461
  assert.match(out, /deep/);
462
462
  assert.match(out, /001/);
463
+ assert.match(out, /diff-first/i, "code-review prompt should require diff-first context gathering");
463
464
  });
464
465
  test("review.md loads with target + reviewers", () => {
465
466
  const out = loadPrompt("review", { target: "t", reviewers: "claude, codex" });
@@ -1,10 +1,22 @@
1
1
  import test from "node:test";
2
2
  import assert from "node:assert/strict";
3
+ import { execFileSync } from "node:child_process";
4
+ import { mkdtempSync, mkdirSync, rmSync, writeFileSync } from "node:fs";
5
+ import { tmpdir } from "node:os";
6
+ import { join } from "node:path";
3
7
 
4
8
  import {
5
9
  formatCleanKeepReason,
10
+ handleWorktree,
6
11
  type WorktreeStatus,
7
12
  } from "../commands-worktree.ts";
13
+ import { withCommandCwd } from "../commands/context.ts";
14
+ import { createWorktree } from "../worktree-manager.ts";
15
+ import {
16
+ disableDebug,
17
+ enableDebug,
18
+ getDebugCounters,
19
+ } from "../debug-logger.ts";
8
20
 
9
21
  function mkStatus(over: Partial<WorktreeStatus>): WorktreeStatus {
10
22
  const name = over.name ?? "feat-x";
@@ -22,6 +34,45 @@ function mkStatus(over: Partial<WorktreeStatus>): WorktreeStatus {
22
34
  };
23
35
  }
24
36
 
37
+ function git(cwd: string, args: string[]): string {
38
+ return execFileSync("git", args, {
39
+ cwd,
40
+ stdio: ["ignore", "pipe", "pipe"],
41
+ encoding: "utf-8",
42
+ }).trim();
43
+ }
44
+
45
+ function makeRepo(): string {
46
+ const base = mkdtempSync(join(tmpdir(), "gsd-worktree-command-"));
47
+ git(base, ["init", "-b", "main"]);
48
+ git(base, ["config", "user.name", "Test User"]);
49
+ git(base, ["config", "user.email", "test@example.com"]);
50
+ mkdirSync(join(base, ".gsd"), { recursive: true });
51
+ writeFileSync(join(base, "README.md"), "# Test\n", "utf-8");
52
+ git(base, ["add", "."]);
53
+ git(base, ["commit", "-m", "chore: init"]);
54
+ return base;
55
+ }
56
+
57
+ function createCommittedWorktree(base: string, name: string): void {
58
+ const wt = createWorktree(base, name);
59
+ writeFileSync(join(wt.path, `${name}.txt`), `${name}\n`, "utf-8");
60
+ git(wt.path, ["add", "."]);
61
+ git(wt.path, ["commit", "-m", `feat: ${name}`]);
62
+ }
63
+
64
+ function createMockCtx() {
65
+ const notifications: { message: string; level: string }[] = [];
66
+ return {
67
+ notifications,
68
+ ui: {
69
+ notify(message: string, level: string) {
70
+ notifications.push({ message, level });
71
+ },
72
+ },
73
+ };
74
+ }
75
+
25
76
  test("clean keep reason shows uncommitted-only worktrees clearly", () => {
26
77
  const reason = formatCleanKeepReason(mkStatus({ uncommitted: true }));
27
78
  assert.equal(reason, "uncommitted changes");
@@ -46,3 +97,32 @@ test("clean keep reason uses singular form for a single changed file", () => {
46
97
  const reason = formatCleanKeepReason(mkStatus({ filesChanged: 1, uncommitted: false }));
47
98
  assert.equal(reason, "1 changed file");
48
99
  });
100
+
101
+ test("worktree list detects main branch once for the command", async (t) => {
102
+ if (process.env.GSD_ENABLE_NATIVE_GSD_GIT === "1") {
103
+ t.skip("git invocation regression is specific to the CLI fallback path");
104
+ return;
105
+ }
106
+
107
+ const base = makeRepo();
108
+ try {
109
+ createCommittedWorktree(base, "feature-a");
110
+ createCommittedWorktree(base, "feature-b");
111
+
112
+ const ctx = createMockCtx();
113
+ enableDebug(base);
114
+ try {
115
+ await withCommandCwd(base, async () => {
116
+ await handleWorktree("list", ctx as any);
117
+ });
118
+
119
+ assert.equal(ctx.notifications.length, 1);
120
+ assert.match(ctx.notifications[0].message, /Worktrees — 2/);
121
+ assert.equal(getDebugCounters().gitInvocations, 12);
122
+ } finally {
123
+ disableDebug();
124
+ }
125
+ } finally {
126
+ rmSync(base, { recursive: true, force: true });
127
+ }
128
+ });
@@ -203,6 +203,10 @@ console.log('\n=== complete-slice: handler happy path ===');
203
203
  insertSlice({ id: 'S02', milestoneId: 'M001', title: 'Second Slice', risk: 'low', depends: ['S01'], demo: 'advanced stuff', sequence: 2 });
204
204
  insertTask({ id: 'T01', sliceId: 'S01', milestoneId: 'M001', status: 'complete', title: 'Task 1' });
205
205
  insertTask({ id: 'T02', sliceId: 'S01', milestoneId: 'M001', status: 'complete', title: 'Task 2' });
206
+ insertTask({ id: 'T99', sliceId: 'S02', milestoneId: 'M001', status: 'complete', title: 'Sibling Task' });
207
+ const siblingSummaryPath = path.join(path.dirname(roadmapPath), 'T99-SUMMARY.md');
208
+ const siblingSummaryBefore = '# Existing sibling task summary\n\nDo not rewrite this file.\n';
209
+ fs.writeFileSync(siblingSummaryPath, siblingSummaryBefore);
206
210
 
207
211
  const params = makeValidSliceParams();
208
212
  const result = await handleCompleteSlice(params, basePath);
@@ -267,6 +271,13 @@ console.log('\n=== complete-slice: handler happy path ===');
267
271
  // (e) Verify slice status is complete in DB
268
272
  assertEq(sliceAfter!.status, 'complete', 'slice status should be complete in DB');
269
273
  assertTrue(sliceAfter!.completed_at !== null, 'completed_at should be set in DB');
274
+
275
+ // (f) Verify unrelated completed task summaries are not re-rendered during slice completion
276
+ assertEq(
277
+ fs.readFileSync(siblingSummaryPath, 'utf-8'),
278
+ siblingSummaryBefore,
279
+ 'complete-slice should not re-render sibling completed task summaries',
280
+ );
270
281
  }
271
282
 
272
283
  cleanupDir(basePath);
@@ -1,6 +1,6 @@
1
1
  import { describe, it, afterEach } from "node:test";
2
2
  import assert from "node:assert/strict";
3
- import { mkdirSync, rmSync, writeFileSync } from "node:fs";
3
+ import { mkdirSync, readFileSync, rmSync, writeFileSync, promises as fsPromises } from "node:fs";
4
4
  import { join } from "node:path";
5
5
  import { tmpdir } from "node:os";
6
6
  import { randomUUID } from "node:crypto";
@@ -41,7 +41,7 @@ const VALID_PARAMS = {
41
41
  ],
42
42
  };
43
43
 
44
- describe("complete-task projection failures keep DB completion committed", () => {
44
+ describe("complete-task projection failures roll back DB completion", () => {
45
45
  let base: string;
46
46
 
47
47
  afterEach(() => {
@@ -75,7 +75,44 @@ describe("complete-task projection failures keep DB completion committed", () =>
75
75
  assert.equal(rows.length, 2, "should have 2 evidence rows after success");
76
76
  });
77
77
 
78
- it("keeps task completion and verification_evidence when disk projection write fails", async () => {
78
+ it("reopens the workflow DB when it disappears between SUMMARY.md write and plan render", async (t) => {
79
+ base = makeTmpBase();
80
+ openDatabase(join(base, ".gsd", "gsd.db"));
81
+ insertMilestone({ id: "M001" });
82
+ insertSlice({ id: "S01", milestoneId: "M001" });
83
+
84
+ const planPath = join(base, ".gsd", "milestones", "M001", "slices", "S01", "S01-PLAN.md");
85
+ writeFileSync(
86
+ planPath,
87
+ "# S01 Plan\n\n## Tasks\n\n- [ ] **T01: Test task**\n",
88
+ );
89
+
90
+ const originalRename = fsPromises.rename.bind(fsPromises);
91
+ let closedAfterSummaryWrite = false;
92
+ t.mock.method(fsPromises, "rename", async (...args: Parameters<typeof fsPromises.rename>) => {
93
+ await originalRename(...args);
94
+ const target = String(args[1]);
95
+ if (!closedAfterSummaryWrite && target.endsWith("T01-SUMMARY.md")) {
96
+ closedAfterSummaryWrite = true;
97
+ closeDatabase();
98
+ }
99
+ });
100
+
101
+ const result = await handleCompleteTask(VALID_PARAMS, base);
102
+ assert.ok(closedAfterSummaryWrite, "test fixture should close DB after SUMMARY.md atomic rename");
103
+ assert.ok(!("error" in result), `unexpected error: ${"error" in result ? result.error : ""}`);
104
+
105
+ const content = readFileSync(planPath, "utf-8");
106
+ assert.match(content, /\[x\][^\n]*\*\*T01\*\*/, "PLAN.md checkbox should be rendered after DB reopen");
107
+
108
+ const adapter = _getAdapter()!;
109
+ const task = adapter.prepare(
110
+ `SELECT status FROM tasks WHERE milestone_id = 'M001' AND slice_id = 'S01' AND id = 'T01'`,
111
+ ).get() as { status: string } | undefined;
112
+ assert.equal(task?.status, "complete", "task should remain complete after successful projection");
113
+ });
114
+
115
+ it("rolls back DB completion and clears verification_evidence when disk projection write fails", async () => {
79
116
  base = makeTmpBase();
80
117
  openDatabase(join(base, ".gsd", "gsd.db"));
81
118
  insertMilestone({ id: "M001" });
@@ -87,19 +124,22 @@ describe("complete-task projection failures keep DB completion committed", () =>
87
124
  writeFileSync(tasksDir, "not-a-directory");
88
125
 
89
126
  const result = await handleCompleteTask(VALID_PARAMS, base);
90
- assert.ok(!("error" in result), `unexpected error: ${"error" in result ? result.error : ""}`);
91
- assert.equal(result.stale, true, "result should report stale projection");
127
+ assert.ok("error" in result, "expected rollback error when projection write fails");
128
+ assert.ok(
129
+ (result as { error: string }).error.includes("rolled completion back to pending"),
130
+ `error should mention rollback; got: ${"error" in result ? result.error : ""}`,
131
+ );
92
132
 
93
133
  const adapter = _getAdapter()!;
94
134
  const task = adapter.prepare(
95
135
  `SELECT status FROM tasks WHERE milestone_id = 'M001' AND slice_id = 'S01' AND id = 'T01'`,
96
136
  ).get() as { status: string } | undefined;
97
- assert.ok(task, "task row should still exist");
98
- assert.equal(task!.status, "complete", "task status should remain complete");
137
+ assert.ok(task, "task row should still exist after rollback");
138
+ assert.equal(task!.status, "pending", "task status should be rolled back to pending");
99
139
 
100
140
  const evidenceRows = adapter.prepare(
101
141
  `SELECT * FROM verification_evidence WHERE task_id = 'T01' AND slice_id = 'S01' AND milestone_id = 'M001'`,
102
142
  ).all();
103
- assert.equal(evidenceRows.length, 2, "verification_evidence should remain committed");
143
+ assert.equal(evidenceRows.length, 0, "verification_evidence should be deleted after rollback");
104
144
  });
105
145
  });
@@ -380,6 +380,81 @@ console.log('\n=== complete-task: handler happy path ===');
380
380
  cleanup(dbPath);
381
381
  }
382
382
 
383
+ // ═══════════════════════════════════════════════════════════════════════════
384
+ // complete-task: Projection failure rolls DB completion back
385
+ // ═══════════════════════════════════════════════════════════════════════════
386
+
387
+ console.log('\n=== complete-task: projection failure rolls DB completion back ===');
388
+ {
389
+ const dbPath = tempDbPath();
390
+ openDatabase(dbPath);
391
+
392
+ const { basePath, planPath } = createTempProject();
393
+
394
+ insertMilestone({ id: 'M001', title: 'Test Milestone' });
395
+ insertSlice({ id: 'S01', milestoneId: 'M001', title: 'Test Slice' });
396
+
397
+ fs.unlinkSync(planPath);
398
+ fs.mkdirSync(planPath, { recursive: true });
399
+
400
+ const result = await handleCompleteTask(makeValidParams(), basePath);
401
+
402
+ assertTrue('error' in result, 'projection failure should return an error');
403
+ if ('error' in result) {
404
+ assertMatch(result.error, /projection write failed/, 'error should mention projection write failure');
405
+ }
406
+
407
+ const task = getTask('M001', 'S01', 'T01');
408
+ assertTrue(task !== null, 'task row should remain for retry');
409
+ assertEq(task!.status, 'pending', 'task status should be rolled back to pending');
410
+ assertEq(task!.completed_at, null, 'rolled back task should not keep completed_at');
411
+
412
+ const adapter = _getAdapter()!;
413
+ const evRows = adapter.prepare(
414
+ "SELECT * FROM verification_evidence WHERE task_id = 'T01' AND slice_id = 'S01' AND milestone_id = 'M001'"
415
+ ).all();
416
+ assertEq(evRows.length, 0, 'verification evidence should be deleted when projection rollback runs');
417
+
418
+ const summaryPath = path.join(path.dirname(planPath), 'T01-SUMMARY.md');
419
+ assertTrue(!fs.existsSync(summaryPath), 'SUMMARY.md should be removed so disk state stays pending');
420
+
421
+ cleanupDir(basePath);
422
+ cleanup(dbPath);
423
+ }
424
+
425
+ // ═══════════════════════════════════════════════════════════════════════════
426
+ // complete-task: Handler does not re-render completed sibling summaries
427
+ // ═══════════════════════════════════════════════════════════════════════════
428
+
429
+ console.log('\n=== complete-task: handler leaves completed sibling summaries untouched ===');
430
+ {
431
+ const dbPath = tempDbPath();
432
+ openDatabase(dbPath);
433
+
434
+ const { basePath, planPath } = createTempProject();
435
+
436
+ insertMilestone({ id: 'M001', title: 'Test Milestone' });
437
+ insertSlice({ id: 'S01', milestoneId: 'M001', title: 'Test Slice', risk: 'high', depends: [], demo: 'basic functionality works', sequence: 1 });
438
+ insertTask({ id: 'T00', sliceId: 'S01', milestoneId: 'M001', status: 'complete', title: 'Already complete task', oneLiner: 'Previously completed' });
439
+ insertTask({ id: 'T02', sliceId: 'S01', milestoneId: 'M001', status: 'pending', title: 'Second task' });
440
+
441
+ const siblingSummaryPath = path.join(path.dirname(planPath), 'T00-SUMMARY.md');
442
+ const siblingSummaryContent = 'existing sibling summary marker\n';
443
+ fs.writeFileSync(siblingSummaryPath, siblingSummaryContent);
444
+
445
+ const result = await handleCompleteTask(makeValidParams(), basePath);
446
+
447
+ assertTrue(!('error' in result), 'handler should succeed without error');
448
+ assertEq(
449
+ fs.readFileSync(siblingSummaryPath, 'utf-8'),
450
+ siblingSummaryContent,
451
+ 'complete-task should not re-render summaries for already-completed sibling tasks',
452
+ );
453
+
454
+ cleanupDir(basePath);
455
+ cleanup(dbPath);
456
+ }
457
+
383
458
  // ═══════════════════════════════════════════════════════════════════════════
384
459
  // complete-task: hard-blocker escalation with mid-execution escalation disabled
385
460
  // ═══════════════════════════════════════════════════════════════════════════
@@ -4,9 +4,10 @@
4
4
 
5
5
  import test from "node:test";
6
6
  import assert from "node:assert/strict";
7
- import { mkdirSync, rmSync } from "node:fs";
8
- import { join } from "node:path";
7
+ import { chmodSync, mkdirSync, rmSync, writeFileSync } from "node:fs";
8
+ import { delimiter, join } from "node:path";
9
9
  import { tmpdir } from "node:os";
10
+ import { performance } from "node:perf_hooks";
10
11
 
11
12
  import { GSDDashboardOverlay } from "../dashboard-overlay.ts";
12
13
  import type { UnitMetrics } from "../metrics.ts";
@@ -85,6 +86,58 @@ test("GSDDashboardOverlay non-identity refresh avoids reparsing preferences", as
85
86
  );
86
87
  });
87
88
 
89
+ test("GSDDashboardOverlay render and scroll do not run environment doctor subprocesses", (t) => {
90
+ const basePath = join(
91
+ tmpdir(),
92
+ `gsd-dashboard-overlay-env-${Date.now()}-${Math.random().toString(36).slice(2, 8)}`,
93
+ );
94
+ const shimDir = join(
95
+ tmpdir(),
96
+ `gsd-dashboard-overlay-shim-${Date.now()}-${Math.random().toString(36).slice(2, 8)}`,
97
+ );
98
+ mkdirSync(basePath, { recursive: true });
99
+ mkdirSync(shimDir, { recursive: true });
100
+ writeFileSync(join(basePath, "package.json"), JSON.stringify({ engines: { node: ">=22.0.0" } }));
101
+
102
+ const posixNodeShim = join(shimDir, "node");
103
+ writeFileSync(posixNodeShim, "#!/bin/sh\nsleep 1\nexit 1\n");
104
+ chmodSync(posixNodeShim, 0o755);
105
+ writeFileSync(join(shimDir, "node.cmd"), "@echo off\r\nping -n 2 127.0.0.1 > nul\r\nexit /b 1\r\n");
106
+
107
+ const originalPath = process.env.PATH;
108
+ const overlay = new GSDDashboardOverlay({ requestRender() {} }, fakeTheme as any, () => {});
109
+ overlay.dispose();
110
+
111
+ t.after(() => {
112
+ if (originalPath === undefined) {
113
+ delete process.env.PATH;
114
+ } else {
115
+ process.env.PATH = originalPath;
116
+ }
117
+ rmSync(basePath, { recursive: true, force: true });
118
+ rmSync(shimDir, { recursive: true, force: true });
119
+ });
120
+
121
+ (overlay as any).loading = false;
122
+ (overlay as any).milestoneData = null;
123
+ (overlay as any).dashData = {
124
+ ...(overlay as any).dashData,
125
+ basePath,
126
+ };
127
+
128
+ process.env.PATH = `${shimDir}${delimiter}${originalPath ?? ""}`;
129
+ const start = performance.now();
130
+ overlay.render(100);
131
+ overlay.handleInput("j");
132
+ overlay.render(100);
133
+ const elapsed = performance.now() - start;
134
+
135
+ assert.ok(
136
+ elapsed < 500,
137
+ `rendering and scrolling should not wait for environment subprocesses, took ${Math.round(elapsed)}ms`,
138
+ );
139
+ });
140
+
88
141
  function makeUnit(id: string, cost: number): UnitMetrics {
89
142
  return {
90
143
  type: "execute-task",