gsd-pi 2.82.0-dev.725028083 → 2.82.0-dev.98ea09b1e

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 (355) hide show
  1. package/README.md +4 -3
  2. package/dist/resources/.managed-resources-content-hash +1 -1
  3. package/dist/resources/GSD-WORKFLOW.md +10 -1
  4. package/dist/resources/extensions/claude-code-cli/partial-builder.js +2 -1
  5. package/dist/resources/extensions/cmux/index.js +5 -0
  6. package/dist/resources/extensions/gsd/auto/infra-errors.js +9 -3
  7. package/dist/resources/extensions/gsd/auto/loop.js +5 -5
  8. package/dist/resources/extensions/gsd/auto/orchestrator.js +124 -6
  9. package/dist/resources/extensions/gsd/auto/phases.js +8 -1
  10. package/dist/resources/extensions/gsd/auto/workflow-memory-pressure.js +12 -0
  11. package/dist/resources/extensions/gsd/auto-dispatch.js +13 -6
  12. package/dist/resources/extensions/gsd/auto-model-selection.js +2 -0
  13. package/dist/resources/extensions/gsd/auto-post-unit.js +233 -127
  14. package/dist/resources/extensions/gsd/auto-prompts.js +2 -2
  15. package/dist/resources/extensions/gsd/auto-recovery.js +31 -1
  16. package/dist/resources/extensions/gsd/auto-start.js +85 -12
  17. package/dist/resources/extensions/gsd/auto-verification.js +28 -22
  18. package/dist/resources/extensions/gsd/auto-worktree.js +111 -1
  19. package/dist/resources/extensions/gsd/auto.js +158 -55
  20. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +4 -1
  21. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +9 -8
  22. package/dist/resources/extensions/gsd/bootstrap/subagent-input.js +21 -9
  23. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +16 -2
  24. package/dist/resources/extensions/gsd/clean-root-preflight.js +170 -8
  25. package/dist/resources/extensions/gsd/commands/catalog.js +4 -1
  26. package/dist/resources/extensions/gsd/commands/handlers/core.js +37 -0
  27. package/dist/resources/extensions/gsd/commands-bootstrap.js +5 -0
  28. package/dist/resources/extensions/gsd/crash-recovery.js +31 -5
  29. package/dist/resources/extensions/gsd/db/unit-dispatches.js +3 -2
  30. package/dist/resources/extensions/gsd/dispatch-guard.js +2 -2
  31. package/dist/resources/extensions/gsd/doctor-runtime-checks.js +28 -11
  32. package/dist/resources/extensions/gsd/doctor.js +2 -28
  33. package/dist/resources/extensions/gsd/export-html.js +27 -425
  34. package/dist/resources/extensions/gsd/git-service.js +39 -1
  35. package/dist/resources/extensions/gsd/gsd-db.js +1 -0
  36. package/dist/resources/extensions/gsd/guided-flow.js +13 -6
  37. package/dist/resources/extensions/gsd/md-importer.js +1 -1
  38. package/dist/resources/extensions/gsd/migrate/command.js +5 -0
  39. package/dist/resources/extensions/gsd/migrate/parsers.js +10 -0
  40. package/dist/resources/extensions/gsd/migrate/preview.js +9 -0
  41. package/dist/resources/extensions/gsd/migrate/transformer.js +51 -4
  42. package/dist/resources/extensions/gsd/migrate/writer.js +11 -1
  43. package/dist/resources/extensions/gsd/migration-auto-check.js +12 -17
  44. package/dist/resources/extensions/gsd/milestone-actions.js +11 -4
  45. package/dist/resources/extensions/gsd/native-git-bridge.js +48 -12
  46. package/dist/resources/extensions/gsd/post-execution-checks.js +73 -2
  47. package/dist/resources/extensions/gsd/pre-execution-checks.js +28 -1
  48. package/dist/resources/extensions/gsd/prompt-loader.js +1 -1
  49. package/dist/resources/extensions/gsd/prompts/complete-milestone.md +1 -1
  50. package/dist/resources/extensions/gsd/prompts/complete-slice.md +1 -1
  51. package/dist/resources/extensions/gsd/prompts/discuss-headless.md +8 -8
  52. package/dist/resources/extensions/gsd/prompts/discuss.md +9 -9
  53. package/dist/resources/extensions/gsd/prompts/guided-discuss-project.md +4 -4
  54. package/dist/resources/extensions/gsd/prompts/guided-discuss-requirements.md +3 -3
  55. package/dist/resources/extensions/gsd/prompts/plan-slice.md +4 -4
  56. package/dist/resources/extensions/gsd/prompts/queue.md +4 -4
  57. package/dist/resources/extensions/gsd/prompts/refine-slice.md +2 -2
  58. package/dist/resources/extensions/gsd/prompts/rewrite-docs.md +1 -1
  59. package/dist/resources/extensions/gsd/state-reconciliation/drift/merge-state.js +6 -1
  60. package/dist/resources/extensions/gsd/state-reconciliation/drift/project-md.js +9 -14
  61. package/dist/resources/extensions/gsd/state-reconciliation/drift/roadmap.js +19 -24
  62. package/dist/resources/extensions/gsd/status-guards.js +4 -0
  63. package/dist/resources/extensions/gsd/templates/plan.md +8 -5
  64. package/dist/resources/extensions/gsd/templates/task-plan.md +4 -2
  65. package/dist/resources/extensions/gsd/tools/complete-milestone.js +6 -8
  66. package/dist/resources/extensions/gsd/tools/complete-slice.js +6 -8
  67. package/dist/resources/extensions/gsd/tools/plan-milestone.js +7 -1
  68. package/dist/resources/extensions/gsd/tools/plan-slice.js +89 -14
  69. package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +119 -0
  70. package/dist/resources/extensions/gsd/unit-context-manifest.js +32 -10
  71. package/dist/resources/extensions/gsd/validation.js +23 -1
  72. package/dist/resources/extensions/gsd/verification-gate.js +68 -7
  73. package/dist/resources/extensions/gsd/verification-verdict.js +26 -0
  74. package/dist/resources/extensions/gsd/workflow-projections.js +6 -8
  75. package/dist/resources/extensions/gsd/worktree-lifecycle.js +54 -10
  76. package/dist/resources/extensions/shared/html-shell.js +388 -0
  77. package/dist/resources/extensions/subagent/index.js +448 -78
  78. package/dist/resources/extensions/subagent/launch.js +77 -0
  79. package/dist/resources/extensions/subagent/run-store.js +148 -0
  80. package/dist/resources/extensions/visual-brief/artifact-policy.js +29 -0
  81. package/dist/resources/extensions/visual-brief/extension-manifest.json +8 -0
  82. package/dist/resources/extensions/visual-brief/index.js +5 -0
  83. package/dist/resources/extensions/visual-brief/page-contract.js +124 -0
  84. package/dist/resources/extensions/visual-brief/prompts.js +140 -0
  85. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  86. package/dist/web/standalone/.next/BUILD_ID +1 -1
  87. package/dist/web/standalone/.next/app-path-routes-manifest.json +14 -14
  88. package/dist/web/standalone/.next/build-manifest.json +3 -3
  89. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  90. package/dist/web/standalone/.next/react-loadable-manifest.json +3 -3
  91. package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  92. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  93. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  94. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  95. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  96. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  97. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  98. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  99. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  100. package/dist/web/standalone/.next/server/app/_not-found/page.js +2 -2
  101. package/dist/web/standalone/.next/server/app/_not-found/page.js.nft.json +1 -1
  102. package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  103. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  104. package/dist/web/standalone/.next/server/app/_not-found.rsc +4 -7
  105. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +4 -7
  106. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  107. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +4 -5
  108. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  109. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  110. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +2 -5
  111. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  112. package/dist/web/standalone/.next/server/app/index.html +1 -1
  113. package/dist/web/standalone/.next/server/app/index.rsc +4 -7
  114. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  115. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +4 -7
  116. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  117. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +4 -5
  118. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +2 -5
  119. package/dist/web/standalone/.next/server/app/page.js +2 -2
  120. package/dist/web/standalone/.next/server/app/page.js.nft.json +1 -1
  121. package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
  122. package/dist/web/standalone/.next/server/app-paths-manifest.json +14 -14
  123. package/dist/web/standalone/.next/server/chunks/4266.js +2 -0
  124. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  125. package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
  126. package/dist/web/standalone/.next/server/next-font-manifest.js +1 -1
  127. package/dist/web/standalone/.next/server/next-font-manifest.json +1 -1
  128. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  129. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  130. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  131. package/dist/web/standalone/.next/static/chunks/2973.33f26573894b6153.js +2 -0
  132. package/dist/web/standalone/.next/static/chunks/{8359.e059d86b255fce1c.js → 8359.7eb3bb8f8ecf4c01.js} +2 -2
  133. package/dist/web/standalone/.next/static/chunks/app/layout-8c10ec293ae0f1d5.js +1 -0
  134. package/dist/web/standalone/.next/static/chunks/{webpack-de742b64187e13fe.js → webpack-9a4db269f9ed63ad.js} +1 -1
  135. package/dist/web/standalone/.next/static/css/746ee28c929d1880.css +1 -0
  136. package/package.json +4 -4
  137. package/packages/contracts/dist/rpc.test.js +7 -0
  138. package/packages/contracts/dist/rpc.test.js.map +1 -1
  139. package/packages/contracts/dist/workflow.d.ts +21 -0
  140. package/packages/contracts/dist/workflow.d.ts.map +1 -1
  141. package/packages/contracts/dist/workflow.js +24 -0
  142. package/packages/contracts/dist/workflow.js.map +1 -1
  143. package/packages/contracts/src/rpc.test.ts +8 -0
  144. package/packages/contracts/src/workflow.ts +24 -0
  145. package/packages/mcp-server/README.md +13 -4
  146. package/packages/mcp-server/dist/workflow-tools.d.ts +0 -3
  147. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  148. package/packages/mcp-server/dist/workflow-tools.js +80 -0
  149. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  150. package/packages/mcp-server/src/workflow-tools.test.ts +23 -1
  151. package/packages/mcp-server/src/workflow-tools.ts +168 -0
  152. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
  153. package/packages/native/tsconfig.json +2 -1
  154. package/packages/native/tsconfig.tsbuildinfo +1 -1
  155. package/packages/pi-ai/dist/providers/openai-codex-responses.d.ts.map +1 -1
  156. package/packages/pi-ai/dist/providers/openai-codex-responses.js +82 -1
  157. package/packages/pi-ai/dist/providers/openai-codex-responses.js.map +1 -1
  158. package/packages/pi-ai/dist/providers/openai-codex-responses.test.d.ts +2 -0
  159. package/packages/pi-ai/dist/providers/openai-codex-responses.test.d.ts.map +1 -0
  160. package/packages/pi-ai/dist/providers/openai-codex-responses.test.js +52 -0
  161. package/packages/pi-ai/dist/providers/openai-codex-responses.test.js.map +1 -0
  162. package/packages/pi-ai/dist/providers/simple-options.d.ts +2 -4
  163. package/packages/pi-ai/dist/providers/simple-options.d.ts.map +1 -1
  164. package/packages/pi-ai/dist/providers/simple-options.js +5 -6
  165. package/packages/pi-ai/dist/providers/simple-options.js.map +1 -1
  166. package/packages/pi-ai/dist/providers/simple-options.test.d.ts +2 -0
  167. package/packages/pi-ai/dist/providers/simple-options.test.d.ts.map +1 -0
  168. package/packages/pi-ai/dist/providers/simple-options.test.js +50 -0
  169. package/packages/pi-ai/dist/providers/simple-options.test.js.map +1 -0
  170. package/packages/pi-ai/src/providers/openai-codex-responses.test.ts +63 -0
  171. package/packages/pi-ai/src/providers/openai-codex-responses.ts +91 -1
  172. package/packages/pi-ai/src/providers/simple-options.test.ts +60 -0
  173. package/packages/pi-ai/src/providers/simple-options.ts +5 -6
  174. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
  175. package/packages/pi-coding-agent/dist/core/agent-session-thinking-level.test.d.ts +2 -0
  176. package/packages/pi-coding-agent/dist/core/agent-session-thinking-level.test.d.ts.map +1 -0
  177. package/packages/pi-coding-agent/dist/core/agent-session-thinking-level.test.js +66 -0
  178. package/packages/pi-coding-agent/dist/core/agent-session-thinking-level.test.js.map +1 -0
  179. package/packages/pi-coding-agent/dist/core/agent-session.js +1 -1
  180. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  181. package/packages/pi-coding-agent/src/core/agent-session-thinking-level.test.ts +79 -0
  182. package/packages/pi-coding-agent/src/core/agent-session.ts +1 -1
  183. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
  184. package/packages/pi-tui/dist/tui.d.ts.map +1 -1
  185. package/packages/pi-tui/dist/tui.js +5 -0
  186. package/packages/pi-tui/dist/tui.js.map +1 -1
  187. package/packages/pi-tui/src/tui.ts +6 -0
  188. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
  189. package/packages/rpc-client/tsconfig.tsbuildinfo +1 -1
  190. package/src/resources/GSD-WORKFLOW.md +10 -1
  191. package/src/resources/extensions/claude-code-cli/partial-builder.ts +2 -1
  192. package/src/resources/extensions/claude-code-cli/tests/partial-builder.test.ts +19 -2
  193. package/src/resources/extensions/cmux/index.ts +6 -0
  194. package/src/resources/extensions/gsd/auto/contracts.ts +59 -16
  195. package/src/resources/extensions/gsd/auto/infra-errors.ts +9 -3
  196. package/src/resources/extensions/gsd/auto/loop.ts +8 -5
  197. package/src/resources/extensions/gsd/auto/orchestrator.ts +129 -6
  198. package/src/resources/extensions/gsd/auto/phases.ts +7 -1
  199. package/src/resources/extensions/gsd/auto/workflow-memory-pressure.ts +13 -0
  200. package/src/resources/extensions/gsd/auto-dispatch.ts +14 -6
  201. package/src/resources/extensions/gsd/auto-model-selection.ts +2 -1
  202. package/src/resources/extensions/gsd/auto-post-unit.ts +266 -139
  203. package/src/resources/extensions/gsd/auto-prompts.ts +2 -2
  204. package/src/resources/extensions/gsd/auto-recovery.ts +29 -0
  205. package/src/resources/extensions/gsd/auto-start.ts +92 -9
  206. package/src/resources/extensions/gsd/auto-verification.ts +36 -34
  207. package/src/resources/extensions/gsd/auto-worktree.ts +119 -1
  208. package/src/resources/extensions/gsd/auto.ts +167 -53
  209. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +6 -1
  210. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +9 -8
  211. package/src/resources/extensions/gsd/bootstrap/subagent-input.ts +19 -7
  212. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +19 -3
  213. package/src/resources/extensions/gsd/clean-root-preflight.ts +174 -8
  214. package/src/resources/extensions/gsd/commands/catalog.ts +4 -1
  215. package/src/resources/extensions/gsd/commands/handlers/core.ts +40 -0
  216. package/src/resources/extensions/gsd/commands-bootstrap.ts +10 -0
  217. package/src/resources/extensions/gsd/crash-recovery.ts +30 -4
  218. package/src/resources/extensions/gsd/db/unit-dispatches.ts +4 -3
  219. package/src/resources/extensions/gsd/dispatch-guard.ts +2 -2
  220. package/src/resources/extensions/gsd/doctor-runtime-checks.ts +25 -13
  221. package/src/resources/extensions/gsd/doctor.ts +2 -27
  222. package/src/resources/extensions/gsd/export-html.ts +27 -427
  223. package/src/resources/extensions/gsd/git-service.ts +45 -1
  224. package/src/resources/extensions/gsd/gsd-db.ts +3 -0
  225. package/src/resources/extensions/gsd/guided-flow.ts +14 -7
  226. package/src/resources/extensions/gsd/md-importer.ts +1 -1
  227. package/src/resources/extensions/gsd/migrate/command.ts +5 -0
  228. package/src/resources/extensions/gsd/migrate/parsers.ts +11 -0
  229. package/src/resources/extensions/gsd/migrate/preview.ts +10 -0
  230. package/src/resources/extensions/gsd/migrate/transformer.ts +58 -4
  231. package/src/resources/extensions/gsd/migrate/writer.ts +14 -1
  232. package/src/resources/extensions/gsd/migration-auto-check.ts +15 -23
  233. package/src/resources/extensions/gsd/milestone-actions.ts +10 -4
  234. package/src/resources/extensions/gsd/native-git-bridge.ts +54 -12
  235. package/src/resources/extensions/gsd/post-execution-checks.ts +87 -2
  236. package/src/resources/extensions/gsd/pre-execution-checks.ts +32 -1
  237. package/src/resources/extensions/gsd/prompt-loader.ts +1 -1
  238. package/src/resources/extensions/gsd/prompts/complete-milestone.md +1 -1
  239. package/src/resources/extensions/gsd/prompts/complete-slice.md +1 -1
  240. package/src/resources/extensions/gsd/prompts/discuss-headless.md +8 -8
  241. package/src/resources/extensions/gsd/prompts/discuss.md +9 -9
  242. package/src/resources/extensions/gsd/prompts/guided-discuss-project.md +4 -4
  243. package/src/resources/extensions/gsd/prompts/guided-discuss-requirements.md +3 -3
  244. package/src/resources/extensions/gsd/prompts/plan-slice.md +4 -4
  245. package/src/resources/extensions/gsd/prompts/queue.md +4 -4
  246. package/src/resources/extensions/gsd/prompts/refine-slice.md +2 -2
  247. package/src/resources/extensions/gsd/prompts/rewrite-docs.md +1 -1
  248. package/src/resources/extensions/gsd/state-reconciliation/drift/merge-state.ts +8 -1
  249. package/src/resources/extensions/gsd/state-reconciliation/drift/project-md.ts +12 -15
  250. package/src/resources/extensions/gsd/state-reconciliation/drift/roadmap.ts +17 -25
  251. package/src/resources/extensions/gsd/status-guards.ts +5 -0
  252. package/src/resources/extensions/gsd/templates/plan.md +8 -5
  253. package/src/resources/extensions/gsd/templates/task-plan.md +4 -2
  254. package/src/resources/extensions/gsd/tests/auto-deterministic-error-classification-4973.test.ts +116 -0
  255. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +54 -0
  256. package/src/resources/extensions/gsd/tests/auto-orchestrator.test.ts +487 -4
  257. package/src/resources/extensions/gsd/tests/auto-paused-ui-cleanup.test.ts +12 -11
  258. package/src/resources/extensions/gsd/tests/auto-post-unit-step-message.test.ts +12 -1
  259. package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +15 -1
  260. package/src/resources/extensions/gsd/tests/auto-runtime-state.test.ts +4 -4
  261. package/src/resources/extensions/gsd/tests/auto-start-orphan-bootstrap.test.ts +1 -0
  262. package/src/resources/extensions/gsd/tests/auto-worktree-registry.test.ts +69 -1
  263. package/src/resources/extensions/gsd/tests/brief-command.test.ts +89 -0
  264. package/src/resources/extensions/gsd/tests/clean-root-preflight.test.ts +107 -2
  265. package/src/resources/extensions/gsd/tests/closeout-git-deferral.test.ts +16 -0
  266. package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +4 -1
  267. package/src/resources/extensions/gsd/tests/complete-slice.test.ts +5 -9
  268. package/src/resources/extensions/gsd/tests/complete-task.test.ts +3 -1
  269. package/src/resources/extensions/gsd/tests/crash-recovery-via-db.test.ts +43 -2
  270. package/src/resources/extensions/gsd/tests/db-authority-regression.test.ts +208 -0
  271. package/src/resources/extensions/gsd/tests/deep-project-auto-loop.test.ts +59 -2
  272. package/src/resources/extensions/gsd/tests/dispatch-complete-milestone-guard.test.ts +39 -0
  273. package/src/resources/extensions/gsd/tests/dispatch-guard.test.ts +27 -0
  274. package/src/resources/extensions/gsd/tests/evidence-cross-ref.test.ts +38 -0
  275. package/src/resources/extensions/gsd/tests/export-html-enhancements.test.ts +8 -0
  276. package/src/resources/extensions/gsd/tests/guided-discuss-project-prompt-rendering.test.ts +2 -0
  277. package/src/resources/extensions/gsd/tests/guided-flow.test.ts +21 -0
  278. package/src/resources/extensions/gsd/tests/headless-milestone-parity.test.ts +6 -6
  279. package/src/resources/extensions/gsd/tests/hook-model-resolution.test.ts +5 -0
  280. package/src/resources/extensions/gsd/tests/infra-error.test.ts +2 -2
  281. package/src/resources/extensions/gsd/tests/infra-errors-cooldown.test.ts +9 -0
  282. package/src/resources/extensions/gsd/tests/integration/doctor-runtime.test.ts +20 -0
  283. package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +103 -1
  284. package/src/resources/extensions/gsd/tests/integration/migrate-command.test.ts +48 -3
  285. package/src/resources/extensions/gsd/tests/integration/state-machine-runtime-failures.test.ts +6 -1
  286. package/src/resources/extensions/gsd/tests/migrate-transformer.test.ts +5 -1
  287. package/src/resources/extensions/gsd/tests/migrate-validator-parsers.test.ts +24 -1
  288. package/src/resources/extensions/gsd/tests/migrate-writer-integration.test.ts +6 -1
  289. package/src/resources/extensions/gsd/tests/migration-auto-check.test.ts +26 -18
  290. package/src/resources/extensions/gsd/tests/native-git-bridge-exec-fallback.test.ts +63 -2
  291. package/src/resources/extensions/gsd/tests/orphaned-worktree-audit.test.ts +121 -1
  292. package/src/resources/extensions/gsd/tests/park-db-sync.test.ts +55 -1
  293. package/src/resources/extensions/gsd/tests/plan-milestone.test.ts +26 -0
  294. package/src/resources/extensions/gsd/tests/plan-slice-prompt.test.ts +2 -0
  295. package/src/resources/extensions/gsd/tests/plan-slice.test.ts +225 -1
  296. package/src/resources/extensions/gsd/tests/plan-task.test.ts +17 -0
  297. package/src/resources/extensions/gsd/tests/post-exec-retry-bypass.test.ts +79 -1
  298. package/src/resources/extensions/gsd/tests/post-execution-checks.test.ts +86 -0
  299. package/src/resources/extensions/gsd/tests/post-unit-git-failure.test.ts +1 -1
  300. package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +53 -0
  301. package/src/resources/extensions/gsd/tests/prompt-loader.test.ts +23 -0
  302. package/src/resources/extensions/gsd/tests/remediation-completion-guard.test.ts +46 -2
  303. package/src/resources/extensions/gsd/tests/session-switch-abort-misclassification.test.ts +10 -0
  304. package/src/resources/extensions/gsd/tests/start-auto-detached.test.ts +31 -1
  305. package/src/resources/extensions/gsd/tests/state-corruption-2945.test.ts +6 -0
  306. package/src/resources/extensions/gsd/tests/state-reconciliation-drift.test.ts +119 -23
  307. package/src/resources/extensions/gsd/tests/stuck-state-via-db.test.ts +64 -1
  308. package/src/resources/extensions/gsd/tests/summary-render-parity.test.ts +7 -3
  309. package/src/resources/extensions/gsd/tests/unit-context-manifest.test.ts +86 -7
  310. package/src/resources/extensions/gsd/tests/verification-gate.test.ts +110 -1
  311. package/src/resources/extensions/gsd/tests/verification-verdict.test.ts +78 -0
  312. package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +1 -1
  313. package/src/resources/extensions/gsd/tests/workflow-memory-pressure.test.ts +21 -1
  314. package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +1 -1
  315. package/src/resources/extensions/gsd/tests/worktree-git-pathspec.test.ts +39 -0
  316. package/src/resources/extensions/gsd/tests/worktree-journal-events.test.ts +64 -12
  317. package/src/resources/extensions/gsd/tests/worktree-lifecycle.test.ts +25 -0
  318. package/src/resources/extensions/gsd/tests/write-gate-planning-unit.test.ts +54 -0
  319. package/src/resources/extensions/gsd/tools/complete-milestone.ts +8 -10
  320. package/src/resources/extensions/gsd/tools/complete-slice.ts +6 -8
  321. package/src/resources/extensions/gsd/tools/plan-milestone.ts +5 -1
  322. package/src/resources/extensions/gsd/tools/plan-slice.ts +98 -12
  323. package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +135 -0
  324. package/src/resources/extensions/gsd/types.ts +1 -1
  325. package/src/resources/extensions/gsd/unit-context-manifest.ts +47 -11
  326. package/src/resources/extensions/gsd/validation.ts +23 -1
  327. package/src/resources/extensions/gsd/verification-gate.ts +78 -6
  328. package/src/resources/extensions/gsd/verification-verdict.ts +47 -0
  329. package/src/resources/extensions/gsd/workflow-projections.ts +6 -8
  330. package/src/resources/extensions/gsd/worktree-lifecycle.ts +61 -10
  331. package/src/resources/extensions/shared/html-shell.ts +412 -0
  332. package/src/resources/extensions/subagent/index.ts +567 -103
  333. package/src/resources/extensions/subagent/launch.ts +131 -0
  334. package/src/resources/extensions/subagent/run-store.ts +218 -0
  335. package/src/resources/extensions/subagent/tests/launch.test.ts +115 -0
  336. package/src/resources/extensions/subagent/tests/run-store.test.ts +111 -0
  337. package/src/resources/extensions/visual-brief/artifact-policy.ts +41 -0
  338. package/src/resources/extensions/visual-brief/extension-manifest.json +8 -0
  339. package/src/resources/extensions/visual-brief/index.ts +8 -0
  340. package/src/resources/extensions/visual-brief/page-contract.ts +136 -0
  341. package/src/resources/extensions/visual-brief/prompts.ts +183 -0
  342. package/src/resources/extensions/visual-brief/tests/visual-brief.test.ts +212 -0
  343. package/dist/web/standalone/.next/server/chunks/5822.js +0 -2
  344. package/dist/web/standalone/.next/static/chunks/2556.0527fea66e123b7f.js +0 -1
  345. package/dist/web/standalone/.next/static/chunks/app/layout-a16c7a7ecdf0c2cf.js +0 -1
  346. package/dist/web/standalone/.next/static/css/54ec2745c1da488b.css +0 -1
  347. package/dist/web/standalone/.next/static/css/de70bee13400563f.css +0 -1
  348. package/dist/web/standalone/.next/static/media/4cf2300e9c8272f7-s.p.woff2 +0 -0
  349. package/dist/web/standalone/.next/static/media/747892c23ea88013-s.woff2 +0 -0
  350. package/dist/web/standalone/.next/static/media/8d697b304b401681-s.woff2 +0 -0
  351. package/dist/web/standalone/.next/static/media/93f479601ee12b01-s.p.woff2 +0 -0
  352. package/dist/web/standalone/.next/static/media/9610d9e46709d722-s.woff2 +0 -0
  353. package/dist/web/standalone/.next/static/media/ba015fad6dcf6784-s.woff2 +0 -0
  354. /package/dist/web/standalone/.next/static/{KDRTXR-22LPCsa80X9dey → euQ0CLP_v8V4e76Tu3odJ}/_buildManifest.js +0 -0
  355. /package/dist/web/standalone/.next/static/{KDRTXR-22LPCsa80X9dey → euQ0CLP_v8V4e76Tu3odJ}/_ssgManifest.js +0 -0
@@ -5,7 +5,7 @@ import assert from 'node:assert/strict';
5
5
  import { mkdtempSync, mkdirSync, writeFileSync, rmSync, existsSync, symlinkSync, readFileSync } from "node:fs";
6
6
  import { join, dirname } from "node:path";
7
7
  import { tmpdir } from "node:os";
8
- import { execSync } from "node:child_process";
8
+ import { execFileSync, execSync } from "node:child_process";
9
9
 
10
10
  import {
11
11
  inferCommitType,
@@ -371,6 +371,18 @@ describe('git-service', async () => {
371
371
  return dir;
372
372
  }
373
373
 
374
+ function gitRun(args: string[], cwd: string): string {
375
+ return execFileSync("git", args, {
376
+ cwd,
377
+ stdio: ["ignore", "pipe", "pipe"],
378
+ encoding: "utf-8",
379
+ env: {
380
+ ...process.env,
381
+ GIT_ALLOW_PROTOCOL: "file",
382
+ },
383
+ }).trim();
384
+ }
385
+
374
386
  // ─── GitServiceImpl: smart staging ─────────────────────────────────────
375
387
 
376
388
  test('GitServiceImpl: smart staging', () => {
@@ -413,6 +425,96 @@ describe('git-service', async () => {
413
425
  rmSync(repo, { recursive: true, force: true });
414
426
  });
415
427
 
428
+ test('GitServiceImpl: task autoCommit skips keyFiles inside submodules', () => {
429
+ const repo = initTempRepo();
430
+ const subSrc = mkdtempSync(join(tmpdir(), "gsd-git-submodule-src-"));
431
+
432
+ try {
433
+ gitRun(["init", "-b", "main"], subSrc);
434
+ gitRun(["config", "user.name", "Pi Test"], subSrc);
435
+ gitRun(["config", "user.email", "pi@example.com"], subSrc);
436
+ createFile(subSrc, "tracked.txt", "initial\n");
437
+ gitRun(["add", "-A"], subSrc);
438
+ gitRun(["commit", "-m", "init submodule"], subSrc);
439
+
440
+ gitRun(["-c", "protocol.file.allow=always", "submodule", "add", `file://${subSrc}`, "sub"], repo);
441
+ gitRun(["commit", "-m", "add submodule"], repo);
442
+
443
+ createFile(repo, "sub/copied.txt", "copied from source\n");
444
+ createFile(repo, "src/feature.ts", "export const feature = true;\n");
445
+ createFile(repo, "src/unrelated.ts", "export const unrelated = true;\n");
446
+
447
+ const svc = new GitServiceImpl(repo);
448
+ const taskContext: TaskCommitContext = {
449
+ taskId: "S01/T01",
450
+ taskDisplayId: "T01",
451
+ taskTitle: "fix submodule staging",
452
+ milestoneId: "M001",
453
+ milestoneTitle: "Submodule auto commit",
454
+ sliceId: "S01",
455
+ sliceTitle: "Commit scoped files",
456
+ oneLiner: "Fixed auto commit when key files include submodule paths",
457
+ keyFiles: ["sub/copied.txt", "src/feature.ts"],
458
+ };
459
+
460
+ const result = svc.autoCommit("execute-task", "M001/S01/T01", [], taskContext);
461
+
462
+ assert.ok(result !== null, "autoCommit should commit non-submodule changes");
463
+ const committed = gitRun(["show", "--name-only", "--format=", "HEAD"], repo);
464
+ assert.ok(committed.includes("src/feature.ts"), "non-submodule keyFile is committed");
465
+ assert.ok(!committed.includes("sub/copied.txt"), "submodule inner keyFile is not pathspec-staged");
466
+ assert.ok(!committed.includes("src/unrelated.ts"), "scoped staging does not fall back to smartStage");
467
+ } finally {
468
+ rmSync(repo, { recursive: true, force: true });
469
+ rmSync(subSrc, { recursive: true, force: true });
470
+ }
471
+ });
472
+
473
+ test('GitServiceImpl: all keyFiles inside submodules falls back to smartStage', () => {
474
+ const repo = initTempRepo();
475
+ const subSrc = mkdtempSync(join(tmpdir(), "gsd-git-all-submodule-src-"));
476
+
477
+ try {
478
+ gitRun(["init", "-b", "main"], subSrc);
479
+ gitRun(["config", "user.name", "Pi Test"], subSrc);
480
+ gitRun(["config", "user.email", "pi@example.com"], subSrc);
481
+ createFile(subSrc, "tracked.txt", "initial\n");
482
+ gitRun(["add", "-A"], subSrc);
483
+ gitRun(["commit", "-m", "init submodule"], subSrc);
484
+
485
+ gitRun(["-c", "protocol.file.allow=always", "submodule", "add", `file://${subSrc}`, "sub"], repo);
486
+ gitRun(["commit", "-m", "add submodule"], repo);
487
+
488
+ createFile(repo, "sub/file1.txt", "inside submodule\n");
489
+ createFile(repo, "sub/file2.txt", "also inside\n");
490
+ createFile(repo, "src/real.ts", "export const real = true;\n");
491
+
492
+ const svc = new GitServiceImpl(repo);
493
+ const taskContext: TaskCommitContext = {
494
+ taskId: "S01/T02",
495
+ taskDisplayId: "T02",
496
+ taskTitle: "all keyFiles inside submodule",
497
+ milestoneId: "M001",
498
+ milestoneTitle: "Submodule auto commit",
499
+ sliceId: "S01",
500
+ sliceTitle: "Commit scoped files",
501
+ oneLiner: "Fell back when all key files are inside submodules",
502
+ keyFiles: ["sub", "sub/file1.txt", "sub/file2.txt"],
503
+ };
504
+
505
+ const result = svc.autoCommit("execute-task", "M001/S01/T02", [], taskContext);
506
+
507
+ assert.ok(result !== null, "autoCommit falls back to smartStage when all keyFiles are filtered");
508
+ const committed = gitRun(["show", "--name-only", "--format=", "HEAD"], repo);
509
+ assert.ok(!committed.includes("sub/file1.txt"), "submodule keyFile is not committed");
510
+ assert.ok(!committed.includes("sub/file2.txt"), "submodule keyFile is not committed");
511
+ assert.ok(committed.includes("src/real.ts"), "smartStage fallback commits other dirty files");
512
+ } finally {
513
+ rmSync(repo, { recursive: true, force: true });
514
+ rmSync(subSrc, { recursive: true, force: true });
515
+ }
516
+ });
517
+
416
518
  // ─── GitServiceImpl: smart staging excludes tracked runtime files ──────
417
519
 
418
520
  test('GitServiceImpl: smart staging excludes tracked runtime files', () => {
@@ -14,7 +14,9 @@ import {
14
14
  generatePreview,
15
15
  writeGSDDirectory,
16
16
  } from '../../migrate/index.ts';
17
+ import { importWrittenMigrationToDb } from '../../migrate/command.ts';
17
18
  import { deriveState } from '../../state.ts';
19
+ import { closeDatabase, getDecisionById, getRequirementCounts } from '../../gsd-db.ts';
18
20
  import { describe, test, beforeEach, afterEach } from 'node:test';
19
21
  import assert from 'node:assert/strict';
20
22
 
@@ -52,6 +54,16 @@ const SAMPLE_REQUIREMENTS = `# Requirements
52
54
  - Description: Output matches GSD format.
53
55
  `;
54
56
 
57
+ const SAMPLE_REQUIREMENTS_LEGACY_IDS = `# Requirements
58
+
59
+ ## Active
60
+
61
+ - [ ] **CORE-PIPELINE**: Pipeline must work end-to-end.
62
+ - [ ] **OUTPUT-FORMAT**: Output matches GSD format.
63
+ - [ ] **IMPORT-DB**: Migration imports requirements into the DB.
64
+ - [ ] **STATUS-WIDGET**: Status can query migrated requirements.
65
+ `;
66
+
55
67
  const SAMPLE_STATE = `# State
56
68
 
57
69
  **Current Phase:** 20-features
@@ -166,14 +178,14 @@ Depends on foundation work.
166
178
  </context>
167
179
  `;
168
180
 
169
- function createCompleteFixture(): string {
181
+ function createCompleteFixture(requirementsContent: string = SAMPLE_REQUIREMENTS): string {
170
182
  const base = mkdtempSync(join(tmpdir(), 'gsd-cmd-test-'));
171
183
  const planning = join(base, '.planning');
172
184
  mkdirSync(planning, { recursive: true });
173
185
 
174
186
  writeFileSync(join(planning, 'PROJECT.md'), SAMPLE_PROJECT);
175
187
  writeFileSync(join(planning, 'ROADMAP.md'), SAMPLE_ROADMAP);
176
- writeFileSync(join(planning, 'REQUIREMENTS.md'), SAMPLE_REQUIREMENTS);
188
+ writeFileSync(join(planning, 'REQUIREMENTS.md'), requirementsContent);
177
189
  writeFileSync(join(planning, 'STATE.md'), SAMPLE_STATE);
178
190
  writeFileSync(join(planning, 'config.json'), SAMPLE_CONFIG);
179
191
 
@@ -303,6 +315,7 @@ test('Full pipeline: parse → transform → preview → write → deriveState',
303
315
  const expectedTaskPct = totalTasks > 0 ? Math.round((doneTasks / totalTasks) * 100) : 0;
304
316
  assert.deepStrictEqual(preview.sliceCompletionPct, expectedSlicePct, 'pipeline: preview sliceCompletionPct');
305
317
  assert.deepStrictEqual(preview.taskCompletionPct, expectedTaskPct, 'pipeline: preview taskCompletionPct');
318
+ assert.deepStrictEqual(preview.decisions.total, 1, 'pipeline: preview decisions total');
306
319
 
307
320
  // Requirements in preview
308
321
  assert.deepStrictEqual(preview.requirements.active, 1, 'pipeline: preview requirements active');
@@ -342,6 +355,39 @@ test('Full pipeline: parse → transform → preview → write → deriveState',
342
355
  }
343
356
  });
344
357
 
358
+ test('Full pipeline: legacy requirement IDs import into DB with canonical IDs', async () => {
359
+ const base = createCompleteFixture(SAMPLE_REQUIREMENTS_LEGACY_IDS);
360
+ const writeTarget = mkdtempSync(join(tmpdir(), 'gsd-cmd-legacy-reqs-'));
361
+ try {
362
+ const parsed = await parsePlanningDirectory(join(base, '.planning'));
363
+ const project = transformToGSD(parsed);
364
+ const preview = generatePreview(project);
365
+
366
+ assert.deepStrictEqual(
367
+ project.requirements.map((req) => req.id),
368
+ ['R001', 'R002', 'R003', 'R004'],
369
+ 'legacy-reqs: transform assigns canonical R IDs',
370
+ );
371
+ assert.ok(
372
+ project.requirements[0]?.description.includes('Legacy ID: CORE-PIPELINE'),
373
+ 'legacy-reqs: original ID survives in migrated requirement content',
374
+ );
375
+
376
+ await writeGSDDirectory(project, writeTarget);
377
+ const imported = await importWrittenMigrationToDb(writeTarget, preview);
378
+ const counts = getRequirementCounts();
379
+
380
+ assert.deepStrictEqual(imported.decisions, 1, 'legacy-reqs: DB import includes migrated decisions');
381
+ assert.deepStrictEqual(imported.requirements, 4, 'legacy-reqs: DB import count matches preview');
382
+ assert.ok(getDecisionById('D001') !== null, 'legacy-reqs: migrated decision is queryable');
383
+ assert.deepStrictEqual(counts.total, 4, 'legacy-reqs: DB stores all migrated requirements');
384
+ } finally {
385
+ closeDatabase();
386
+ rmSync(base, { recursive: true, force: true });
387
+ rmSync(writeTarget, { recursive: true, force: true });
388
+ }
389
+ });
390
+
345
391
  // ─── Test 6: .gsd/ exists detection ────────────────────────────────────
346
392
 
347
393
  test('.gsd/ exists detection', () => {
@@ -357,4 +403,3 @@ test('.gsd/ exists detection', () => {
357
403
  rmSync(base, { recursive: true, force: true });
358
404
  }
359
405
  });
360
-
@@ -195,6 +195,11 @@ describe("infrastructure error detection", () => {
195
195
  assert.equal(isInfrastructureError(err), "EAGAIN");
196
196
  });
197
197
 
198
+ test("ENOBUFS (no buffer space available) is detected", () => {
199
+ const err = Object.assign(new Error("spawnSync git ENOBUFS"), { code: "ENOBUFS" });
200
+ assert.equal(isInfrastructureError(err), "ENOBUFS");
201
+ });
202
+
198
203
  test("SQLite WAL corruption is detected via message scan", () => {
199
204
  const err = new Error("database disk image is malformed");
200
205
  assert.equal(isInfrastructureError(err), "SQLITE_CORRUPT");
@@ -223,7 +228,7 @@ describe("infrastructure error detection", () => {
223
228
  test("all INFRA_ERROR_CODES are covered", () => {
224
229
  const expectedCodes = [
225
230
  "ENOSPC", "ENOMEM", "EROFS", "EDQUOT", "EMFILE",
226
- "ENFILE", "EAGAIN", "ECONNREFUSED", "ENOTFOUND", "ENETUNREACH",
231
+ "ENFILE", "EAGAIN", "ENOBUFS", "ECONNREFUSED", "ENOTFOUND", "ENETUNREACH",
227
232
  ];
228
233
  for (const code of expectedCodes) {
229
234
  assert.ok(INFRA_ERROR_CODES.has(code), `${code} should be in INFRA_ERROR_CODES`);
@@ -497,6 +497,7 @@ test('Scenario 11: Requirements edge cases', () => {
497
497
  makeRequirement('', 'Another No ID', 'validated'),
498
498
  makeRequirement('R005', 'Has ID', 'something-weird'),
499
499
  makeRequirement('R006', 'Deferred One', 'DEFERRED'),
500
+ makeRequirement('AUTH-7', 'Legacy ID', 'active'),
500
501
  ],
501
502
  phases: {
502
503
  '1-req-edge': makePhase('1-req-edge', 1, 'req-edge'),
@@ -510,6 +511,8 @@ test('Scenario 11: Requirements edge cases', () => {
510
511
  assert.deepStrictEqual(result.requirements[2]?.id, 'R005', 'req-edge: existing id preserved');
511
512
  assert.deepStrictEqual(result.requirements[2]?.status, 'active', 'req-edge: unknown status normalized to active');
512
513
  assert.deepStrictEqual(result.requirements[3]?.status, 'deferred', 'req-edge: uppercase DEFERRED normalized');
514
+ assert.deepStrictEqual(result.requirements[4]?.id, 'R003', 'req-edge: non-R legacy id gets next canonical id');
515
+ assert.ok(result.requirements[4]?.description.includes('Legacy ID: AUTH-7'), 'req-edge: original legacy id is preserved in description');
513
516
  });
514
517
 
515
518
  // ─── Scenario 12: Vision derivation ────────────────────────────────────────
@@ -553,6 +556,8 @@ test('Scenario 13: Decisions content', () => {
553
556
  const result = transformToGSD(project);
554
557
 
555
558
  assert.ok(result.decisionsContent.includes('decision-01'), 'decisions: extracts key-decisions from summaries');
559
+ assert.ok(result.decisionsContent.includes('| D001 |'), 'decisions: writes DB-importable decision ID');
560
+ assert.ok(result.decisionsContent.includes('| # | When | Scope | Decision | Choice | Rationale | Revisable? | Made By |'), 'decisions: writes canonical table header');
556
561
  });
557
562
 
558
563
  // ─── Scenario 14: No undefined values in output ───────────────────────────
@@ -616,4 +621,3 @@ test('Scenario 15: Empty research', () => {
616
621
  });
617
622
 
618
623
  // ─── Results ───────────────────────────────────────────────────────────────
619
-
@@ -41,6 +41,15 @@ const SAMPLE_ROADMAP = `# Project Roadmap
41
41
  - [ ] 31 — Notifications
42
42
  `;
43
43
 
44
+ const SAMPLE_VERSION_PREFIX_ROADMAP = `# Project Roadmap
45
+
46
+ ## Phases
47
+
48
+ - ✅ **v1.0 MVP** — Phases 1-6 (shipped 2026-02-24)
49
+ - ✅ **v1.1 Onboarding** — Phases 7-9 (shipped 2026-03-01)
50
+ - 🚧 **v1.8 Production** — Phases 44-53
51
+ `;
52
+
44
53
  const SAMPLE_PROJECT = `# My Project
45
54
 
46
55
  A sample project for testing the migration parser.
@@ -246,6 +255,21 @@ test('parseOldRoadmap: flat format', () => {
246
255
  assert.deepStrictEqual(roadmap.phases[1].done, false, 'flat roadmap: second phase not done');
247
256
  });
248
257
 
258
+ test('parseOldRoadmap: emoji version-prefix phase ranges', () => {
259
+ const roadmap = parseOldRoadmap(SAMPLE_VERSION_PREFIX_ROADMAP);
260
+ assert.deepStrictEqual(roadmap.milestones.length, 0, 'version roadmap: no milestone sections');
261
+ assert.deepStrictEqual(roadmap.phases.length, 3, 'version roadmap: 3 phase ranges');
262
+ assert.deepStrictEqual(roadmap.phases[0].number, 1, 'version roadmap: first range starts at phase 1');
263
+ assert.deepStrictEqual(roadmap.phases[0].title, 'MVP', 'version roadmap: first title');
264
+ assert.deepStrictEqual(roadmap.phases[0].done, true, 'version roadmap: first range done');
265
+ assert.deepStrictEqual(roadmap.phases[1].number, 7, 'version roadmap: second range starts at phase 7');
266
+ assert.deepStrictEqual(roadmap.phases[1].title, 'Onboarding', 'version roadmap: second title');
267
+ assert.deepStrictEqual(roadmap.phases[1].done, true, 'version roadmap: second range done');
268
+ assert.deepStrictEqual(roadmap.phases[2].number, 44, 'version roadmap: third range starts at phase 44');
269
+ assert.deepStrictEqual(roadmap.phases[2].title, 'Production', 'version roadmap: third title');
270
+ assert.deepStrictEqual(roadmap.phases[2].done, false, 'version roadmap: third range in progress');
271
+ });
272
+
249
273
  test('parseOldRoadmap: milestone-sectioned with <details>', () => {
250
274
  const roadmap = parseOldRoadmap(SAMPLE_MILESTONE_SECTIONED_ROADMAP);
251
275
  assert.ok(roadmap.milestones.length >= 2, 'ms roadmap: has milestone sections');
@@ -387,4 +411,3 @@ test('parseOldProject', () => {
387
411
  const project = parseOldProject(SAMPLE_PROJECT);
388
412
  assert.deepStrictEqual(project, SAMPLE_PROJECT, 'project: returns raw content');
389
413
  });
390
-
@@ -14,7 +14,7 @@ import { parseSummary } from '../files.ts';
14
14
  import { deriveState } from '../state.ts';
15
15
  import { invalidateAllCaches } from '../cache.ts';
16
16
  import { ensureDbOpen } from '../bootstrap/dynamic-tools.ts';
17
- import { closeDatabase, getAllMilestones } from '../gsd-db.ts';
17
+ import { closeDatabase, getAllMilestones, getArtifact } from '../gsd-db.ts';
18
18
  import { importWrittenMigrationToDb } from '../migrate/command.ts';
19
19
  import type {
20
20
  GSDProject,
@@ -336,7 +336,12 @@ test('Scenario 2: Fully complete project — deriveState phase', async () => {
336
336
  assert.deepStrictEqual(preview.taskCompletionPct, 100, 'complete: preview taskCompletionPct');
337
337
  assert.deepStrictEqual(preview.requirements.total, 0, 'complete: preview requirements total');
338
338
 
339
+ const imported = await importWrittenMigrationToDb(base, preview);
340
+ assert.ok(imported.artifacts >= 6, 'complete: imports generated milestone artifacts');
341
+ assert.ok(getArtifact('milestones/M001/M001-VALIDATION.md') !== null, 'complete: M001-VALIDATION.md imported as artifact');
342
+ assert.ok(getArtifact('milestones/M001/M001-SUMMARY.md') !== null, 'complete: M001-SUMMARY.md imported as artifact');
339
343
  } finally {
344
+ closeDatabase();
340
345
  rmSync(base, { recursive: true, force: true });
341
346
  }
342
347
  });
@@ -6,13 +6,15 @@ import test from "node:test";
6
6
 
7
7
  import { ensureDbOpen } from "../bootstrap/dynamic-tools.ts";
8
8
  import {
9
- _getAdapter,
10
9
  closeDatabase,
11
10
  getAllMilestones,
11
+ insertMilestone,
12
+ insertSlice,
13
+ insertTask,
12
14
  getSliceTasks,
13
15
  } from "../gsd-db.ts";
14
16
  import {
15
- autoImportMarkdownHierarchyIfDbMismatch,
17
+ checkMarkdownHierarchyAgainstDb,
16
18
  countMarkdownHierarchy,
17
19
  } from "../migration-auto-check.ts";
18
20
  import { writeGSDDirectory } from "../migrate/writer.ts";
@@ -70,7 +72,7 @@ function projectFixture(): GSDProject {
70
72
  };
71
73
  }
72
74
 
73
- test("migration auto-check imports markdown hierarchy when DB is empty", async () => {
75
+ test("migration auto-check preserves empty DB and reports explicit recovery", async () => {
74
76
  const base = makeBase();
75
77
  try {
76
78
  await writeGSDDirectory(projectFixture(), base);
@@ -79,32 +81,35 @@ test("migration auto-check imports markdown hierarchy when DB is empty", async (
79
81
  assert.equal(await ensureDbOpen(base), true);
80
82
  assert.equal(getAllMilestones().length, 0, "fresh authoritative DB starts empty");
81
83
 
82
- const result = await autoImportMarkdownHierarchyIfDbMismatch(base);
83
- assert.equal(result.action, "imported");
84
+ const result = await checkMarkdownHierarchyAgainstDb(base);
85
+ assert.equal(result.action, "recovery-required");
84
86
  assert.equal(result.reason, "db-empty");
85
- assert.deepEqual(result.afterDb, { milestones: 1, slices: 1, tasks: 1 });
86
- assert.equal(getAllMilestones().length, 1);
87
- assert.equal(getSliceTasks("M001", "S01").length, 1);
87
+ assert.deepEqual(result.afterDb, { milestones: 0, slices: 0, tasks: 0 });
88
+ assert.equal(result.recoveryCommand, "gsd recover");
89
+ assert.match(result.message ?? "", /will not import markdown automatically/);
90
+ assert.equal(getAllMilestones().length, 0);
91
+ assert.equal(getSliceTasks("M001", "S01").length, 0);
88
92
  } finally {
89
93
  cleanup(base);
90
94
  }
91
95
  });
92
96
 
93
- test("migration auto-check repairs DB hierarchy count mismatch", async () => {
97
+ test("migration auto-check preserves DB on hierarchy count mismatch", async () => {
94
98
  const base = makeBase();
95
99
  try {
96
100
  await writeGSDDirectory(projectFixture(), base);
97
- await autoImportMarkdownHierarchyIfDbMismatch(base);
98
-
99
- _getAdapter()!.prepare("DELETE FROM tasks WHERE milestone_id = ? AND slice_id = ? AND id = ?").run("M001", "S01", "T01");
101
+ assert.equal(await ensureDbOpen(base), true);
102
+ insertMilestone({ id: "M001", title: "Legacy Milestone", status: "active" });
103
+ insertSlice({ id: "S01", milestoneId: "M001", title: "Legacy Slice", status: "pending", risk: "medium", depends: [], demo: "Legacy slice demo", sequence: 1 });
100
104
  assert.equal(getSliceTasks("M001", "S01").length, 0, "test fixture simulates stale DB task count");
101
105
 
102
- const result = await autoImportMarkdownHierarchyIfDbMismatch(base);
103
- assert.equal(result.action, "imported");
106
+ const result = await checkMarkdownHierarchyAgainstDb(base);
107
+ assert.equal(result.action, "recovery-required");
104
108
  assert.equal(result.reason, "count-mismatch");
105
109
  assert.deepEqual(result.beforeDb, { milestones: 1, slices: 1, tasks: 0 });
106
- assert.deepEqual(result.afterDb, { milestones: 1, slices: 1, tasks: 1 });
107
- assert.equal(getSliceTasks("M001", "S01").length, 1);
110
+ assert.deepEqual(result.afterDb, { milestones: 1, slices: 1, tasks: 0 });
111
+ assert.equal(result.recoveryCommand, "gsd recover");
112
+ assert.equal(getSliceTasks("M001", "S01").length, 0);
108
113
  } finally {
109
114
  cleanup(base);
110
115
  }
@@ -114,9 +119,12 @@ test("migration auto-check leaves matching DB hierarchy alone", async () => {
114
119
  const base = makeBase();
115
120
  try {
116
121
  await writeGSDDirectory(projectFixture(), base);
117
- await autoImportMarkdownHierarchyIfDbMismatch(base);
122
+ assert.equal(await ensureDbOpen(base), true);
123
+ insertMilestone({ id: "M001", title: "Legacy Milestone", status: "active" });
124
+ insertSlice({ id: "S01", milestoneId: "M001", title: "Legacy Slice", status: "pending", risk: "medium", depends: [], demo: "Legacy slice demo", sequence: 1 });
125
+ insertTask({ id: "T01", sliceId: "S01", milestoneId: "M001", title: "Legacy Task", status: "pending" });
118
126
 
119
- const result = await autoImportMarkdownHierarchyIfDbMismatch(base);
127
+ const result = await checkMarkdownHierarchyAgainstDb(base);
120
128
  assert.equal(result.action, "none");
121
129
  assert.equal(result.reason, "in-sync");
122
130
  assert.deepEqual(result.markdown, { milestones: 1, slices: 1, tasks: 1 });
@@ -13,17 +13,19 @@
13
13
  import { describe, test, beforeEach, afterEach } from "node:test";
14
14
  import assert from "node:assert/strict";
15
15
  import { chmodSync, existsSync, mkdtempSync, writeFileSync, readFileSync, rmSync } from "node:fs";
16
- import { join } from "node:path";
16
+ import { delimiter, join } from "node:path";
17
17
  import { tmpdir } from "node:os";
18
18
  import { execFileSync } from "node:child_process";
19
19
  import {
20
20
  assertWorktreeMaterialized,
21
21
  nativeBranchDelete,
22
22
  nativeCommit,
23
+ nativeGetCurrentBranch,
23
24
  nativeIsRepo,
24
25
  nativeResetHard,
25
26
  nativeWorktreeAdd,
26
27
  } from "../native-git-bridge.js";
28
+ import { GIT_NO_PROMPT_ENV } from "../git-constants.js";
27
29
 
28
30
  // Note: prior static-analysis tests that scanned native-git-bridge.ts for
29
31
  // the raw shell-spawn pattern were removed under #4827 — the integration
@@ -78,6 +80,55 @@ describe("native-git-bridge #4180: fallback runtime behaviour", () => {
78
80
  assert.equal(subject, "test: regression commit #4180");
79
81
  });
80
82
 
83
+ test("nativeCommit retries once after transient ENOBUFS from git", (t) => {
84
+ const bin = mkdtempSync(join(tmpdir(), "ngb-enobufs-bin-"));
85
+ t.after(() => rmSync(bin, { recursive: true, force: true }));
86
+
87
+ const realGit = execFileSync("git", ["--exec-path"], { encoding: "utf-8" }).trim();
88
+ const attempts = join(bin, "attempts.txt");
89
+ const fakeGit = join(bin, "fake-git.cjs");
90
+ writeFileSync(fakeGit, `
91
+ const { appendFileSync, readFileSync } = require("node:fs");
92
+ const { spawnSync } = require("node:child_process");
93
+ const attempts = ${JSON.stringify(attempts)};
94
+ const realGit = ${JSON.stringify(join(realGit, process.platform === "win32" ? "git.exe" : "git"))};
95
+ appendFileSync(attempts, "1");
96
+ if (process.argv[2] === "commit" && readFileSync(attempts, "utf-8").length === 1) {
97
+ console.error("spawnSync git ENOBUFS");
98
+ process.exit(1);
99
+ }
100
+ const result = spawnSync(realGit, process.argv.slice(2), { stdio: "inherit" });
101
+ process.exit(result.status ?? 1);
102
+ `, "utf-8");
103
+
104
+ if (process.platform === "win32") {
105
+ writeFileSync(join(bin, "git.cmd"), `@echo off\r\nnode "${fakeGit}" %*\r\n`, "utf-8");
106
+ } else {
107
+ const shim = join(bin, "git");
108
+ writeFileSync(shim, `#!/bin/sh\nexec node "${fakeGit}" "$@"\n`, "utf-8");
109
+ chmodSync(shim, 0o755);
110
+ }
111
+
112
+ writeFileSync(join(repo, "file.txt"), "retry commit\n");
113
+ git(["add", "."], repo);
114
+
115
+ const originalPath = process.env.PATH ?? "";
116
+ const gitEnv = GIT_NO_PROMPT_ENV as NodeJS.ProcessEnv;
117
+ const originalGitEnvPath = gitEnv.PATH;
118
+ try {
119
+ process.env.PATH = `${bin}${delimiter}${originalPath}`;
120
+ gitEnv.PATH = process.env.PATH;
121
+ const result = nativeCommit(repo, "test: retry ENOBUFS commit");
122
+ assert.ok(result !== null, "commit should succeed after retry");
123
+ } finally {
124
+ process.env.PATH = originalPath;
125
+ gitEnv.PATH = originalGitEnvPath;
126
+ }
127
+
128
+ assert.equal(readFileSync(attempts, "utf-8").length, 2);
129
+ assert.equal(git(["log", "-1", "--format=%s"], repo), "test: retry ENOBUFS commit");
130
+ });
131
+
81
132
  test("nativeCommit runs commit hooks", () => {
82
133
  const hookPath = join(repo, ".git", "hooks", "commit-msg");
83
134
  const marker = join(repo, "hook-ran.txt");
@@ -119,7 +170,17 @@ describe("native-git-bridge #4180: fallback runtime behaviour", () => {
119
170
  test("nativeBranchDelete throws when git cannot delete the branch", () => {
120
171
  assert.throws(
121
172
  () => nativeBranchDelete(repo, "does-not-exist"),
122
- /GSD_GIT_ERROR|git branch -D does-not-exist failed/,
173
+ /git branch -D does-not-exist failed[\s\S]*does-not-exist/,
174
+ );
175
+ });
176
+
177
+ test("nativeGetCurrentBranch preserves git stderr in fallback errors", (t) => {
178
+ const dir = mkdtempSync(join(tmpdir(), "ngb-stderr-notrepo-"));
179
+ t.after(() => rmSync(dir, { recursive: true, force: true }));
180
+
181
+ assert.throws(
182
+ () => nativeGetCurrentBranch(dir),
183
+ /git branch --show-current failed[\s\S]*(not a git repository|not a git repo|fatal:)/,
123
184
  );
124
185
  });
125
186
 
@@ -243,7 +243,7 @@ describe("auditOrphanedMilestoneBranches", () => {
243
243
  );
244
244
  });
245
245
 
246
- test("handles milestone in DB but no branch (no-op)", () => {
246
+ test("milestone in DB, no branch, no worktree dir → no-op", () => {
247
247
  insertMilestone({ id: "M001", title: "Test", status: "complete" });
248
248
 
249
249
  const result = auditOrphanedMilestoneBranches(dir, "worktree");
@@ -251,4 +251,124 @@ describe("auditOrphanedMilestoneBranches", () => {
251
251
  assert.deepStrictEqual(result.recovered, []);
252
252
  assert.deepStrictEqual(result.warnings, []);
253
253
  });
254
+
255
+ test("#5879 — cleans orphaned worktree dir for complete milestone whose branch was already deleted", () => {
256
+ // Reproduces the postflight-stash-restore-failed scenario:
257
+ // 1. An earlier audit deleted milestone/M001 (merged + complete).
258
+ // 2. The worktree dir cleanup failed silently (logWarning only).
259
+ // 3. On the next startup the branch is gone, so the existing branch-keyed
260
+ // loop is invisible to the orphan dir. Without the second pass, the
261
+ // directory lives forever.
262
+ insertMilestone({ id: "M001", title: "Test", status: "complete" });
263
+
264
+ const wtDir = join(dir, ".gsd", "worktrees", "M001");
265
+ mkdirSync(wtDir, { recursive: true });
266
+ writeFileSync(join(wtDir, "leftover.txt"), "stranded from a prior session\n");
267
+
268
+ // No milestone/M001 branch — already deleted on a previous run.
269
+ const branches = run("git branch --list milestone/M001", dir);
270
+ assert.equal(branches, "", "test fixture: branch should not exist");
271
+
272
+ const result = auditOrphanedMilestoneBranches(dir, "worktree");
273
+
274
+ assert.ok(
275
+ result.recovered.some((r) => r.includes("M001") && r.includes("branch already deleted")),
276
+ `should report branch-less orphan cleanup; got: ${JSON.stringify(result.recovered)}`,
277
+ );
278
+ assert.ok(!existsSync(wtDir), "branch-less orphan worktree dir should be removed");
279
+ });
280
+
281
+ test("#5879 — branch list failure still cleans complete orphan only after branch absence is verified", () => {
282
+ insertMilestone({ id: "M001", title: "Test", status: "complete" });
283
+
284
+ const wtDir = join(dir, ".gsd", "worktrees", "M001");
285
+ mkdirSync(wtDir, { recursive: true });
286
+ writeFileSync(join(wtDir, "leftover.txt"), "stranded from a prior session\n");
287
+
288
+ const result = auditOrphanedMilestoneBranches(dir, "worktree", {
289
+ branchList: () => {
290
+ throw new Error("branch list failed");
291
+ },
292
+ branchExists: (_basePath, branch) => {
293
+ assert.equal(branch, "milestone/M001");
294
+ return false;
295
+ },
296
+ });
297
+
298
+ assert.ok(
299
+ result.recovered.some((r) => r.includes("M001") && r.includes("branch already deleted")),
300
+ `should report verified branch-less orphan cleanup; got: ${JSON.stringify(result.recovered)}`,
301
+ );
302
+ assert.ok(!existsSync(wtDir), "verified branch-less orphan worktree dir should be removed");
303
+ });
304
+
305
+ test("#5879 — branch list failure preserves complete worktree when branch still exists", () => {
306
+ insertMilestone({ id: "M001", title: "Test", status: "complete" });
307
+
308
+ const wtDir = join(dir, ".gsd", "worktrees", "M001");
309
+ mkdirSync(wtDir, { recursive: true });
310
+ writeFileSync(join(wtDir, "live-work.txt"), "do not delete\n");
311
+
312
+ const result = auditOrphanedMilestoneBranches(dir, "worktree", {
313
+ branchList: () => {
314
+ throw new Error("branch list failed");
315
+ },
316
+ branchExists: (_basePath, branch) => {
317
+ assert.equal(branch, "milestone/M001");
318
+ return true;
319
+ },
320
+ });
321
+
322
+ assert.deepStrictEqual(result.recovered, []);
323
+ assert.ok(existsSync(wtDir), "worktree dir must be preserved when branch existence is verified");
324
+ });
325
+
326
+ test("#5879 — skips branch-less orphan for milestone that is not complete", () => {
327
+ // Defensive: only `complete` milestones get the branch-less cleanup. An
328
+ // `active` milestone with no branch but a worktree dir is a different
329
+ // state (probably mid-recovery) and should not be silently wiped.
330
+ insertMilestone({ id: "M001", title: "Test", status: "active" });
331
+
332
+ const wtDir = join(dir, ".gsd", "worktrees", "M001");
333
+ mkdirSync(wtDir, { recursive: true });
334
+ writeFileSync(join(wtDir, "live-work.txt"), "do not delete\n");
335
+
336
+ const result = auditOrphanedMilestoneBranches(dir, "worktree");
337
+
338
+ assert.deepStrictEqual(result.recovered, []);
339
+ assert.ok(existsSync(wtDir), "active milestone worktree dir must be preserved");
340
+ });
341
+
342
+ test("#5879 — branch list failure does not delete worktree for milestone that is not complete", () => {
343
+ insertMilestone({ id: "M001", title: "Test", status: "active" });
344
+
345
+ const wtDir = join(dir, ".gsd", "worktrees", "M001");
346
+ mkdirSync(wtDir, { recursive: true });
347
+ writeFileSync(join(wtDir, "live-work.txt"), "do not delete\n");
348
+
349
+ const result = auditOrphanedMilestoneBranches(dir, "worktree", {
350
+ branchList: () => {
351
+ throw new Error("branch list failed");
352
+ },
353
+ branchExists: () => {
354
+ throw new Error("branchExists should not be called for active milestones");
355
+ },
356
+ });
357
+
358
+ assert.deepStrictEqual(result.recovered, []);
359
+ assert.ok(existsSync(wtDir), "active milestone worktree dir must be preserved");
360
+ });
361
+
362
+ test("#5879 — skips branch-less orphan in 'none' isolation mode", () => {
363
+ insertMilestone({ id: "M001", title: "Test", status: "complete" });
364
+
365
+ const wtDir = join(dir, ".gsd", "worktrees", "M001");
366
+ mkdirSync(wtDir, { recursive: true });
367
+ writeFileSync(join(wtDir, "leftover.txt"), "stranded\n");
368
+
369
+ const result = auditOrphanedMilestoneBranches(dir, "none");
370
+
371
+ assert.deepStrictEqual(result.recovered, []);
372
+ assert.ok(existsSync(wtDir), "'none' mode must not touch worktree dirs");
373
+ });
254
374
  });