gsd-pi 2.82.0-dev.c22380fc3 → 2.82.0-dev.e7a7f1ed5

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 (390) hide show
  1. package/README.md +5 -4
  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/claude-code-cli/stream-adapter.js +1 -1
  6. package/dist/resources/extensions/cmux/index.js +5 -0
  7. package/dist/resources/extensions/gsd/auto/infra-errors.js +9 -3
  8. package/dist/resources/extensions/gsd/auto/loop.js +5 -5
  9. package/dist/resources/extensions/gsd/auto/orchestrator.js +11 -0
  10. package/dist/resources/extensions/gsd/auto/phases.js +81 -31
  11. package/dist/resources/extensions/gsd/auto/workflow-memory-pressure.js +12 -0
  12. package/dist/resources/extensions/gsd/auto-dashboard.js +66 -1
  13. package/dist/resources/extensions/gsd/auto-direct-dispatch.js +1 -0
  14. package/dist/resources/extensions/gsd/auto-dispatch.js +18 -17
  15. package/dist/resources/extensions/gsd/auto-model-selection.js +2 -0
  16. package/dist/resources/extensions/gsd/auto-post-unit.js +233 -127
  17. package/dist/resources/extensions/gsd/auto-prompts.js +2 -2
  18. package/dist/resources/extensions/gsd/auto-recovery.js +71 -14
  19. package/dist/resources/extensions/gsd/auto-start.js +87 -14
  20. package/dist/resources/extensions/gsd/auto-verification.js +45 -26
  21. package/dist/resources/extensions/gsd/auto-worktree.js +176 -10
  22. package/dist/resources/extensions/gsd/auto.js +37 -5
  23. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +31 -7
  24. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +9 -8
  25. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +4 -2
  26. package/dist/resources/extensions/gsd/bootstrap/subagent-input.js +21 -9
  27. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +16 -2
  28. package/dist/resources/extensions/gsd/clean-root-preflight.js +170 -8
  29. package/dist/resources/extensions/gsd/commands/catalog.js +4 -1
  30. package/dist/resources/extensions/gsd/commands/handlers/core.js +37 -0
  31. package/dist/resources/extensions/gsd/commands-bootstrap.js +5 -0
  32. package/dist/resources/extensions/gsd/commands-prefs-wizard.js +7 -2
  33. package/dist/resources/extensions/gsd/crash-recovery.js +43 -5
  34. package/dist/resources/extensions/gsd/db/milestone-leases.js +24 -0
  35. package/dist/resources/extensions/gsd/db/unit-dispatches.js +3 -2
  36. package/dist/resources/extensions/gsd/dispatch-guard.js +2 -2
  37. package/dist/resources/extensions/gsd/doctor-git-checks.js +46 -1
  38. package/dist/resources/extensions/gsd/doctor-runtime-checks.js +28 -11
  39. package/dist/resources/extensions/gsd/doctor.js +2 -28
  40. package/dist/resources/extensions/gsd/export-html.js +27 -425
  41. package/dist/resources/extensions/gsd/git-service.js +45 -3
  42. package/dist/resources/extensions/gsd/gsd-db.js +21 -6
  43. package/dist/resources/extensions/gsd/guided-flow-queue.js +4 -3
  44. package/dist/resources/extensions/gsd/guided-flow.js +101 -116
  45. package/dist/resources/extensions/gsd/guided-unit-context.js +23 -0
  46. package/dist/resources/extensions/gsd/migrate/parsers.js +10 -0
  47. package/dist/resources/extensions/gsd/migration-auto-check.js +12 -17
  48. package/dist/resources/extensions/gsd/milestone-actions.js +11 -4
  49. package/dist/resources/extensions/gsd/native-git-bridge.js +48 -12
  50. package/dist/resources/extensions/gsd/pending-auto-start.js +52 -0
  51. package/dist/resources/extensions/gsd/post-execution-checks.js +73 -2
  52. package/dist/resources/extensions/gsd/pre-execution-checks.js +28 -1
  53. package/dist/resources/extensions/gsd/prompt-loader.js +1 -1
  54. package/dist/resources/extensions/gsd/prompts/complete-milestone.md +1 -1
  55. package/dist/resources/extensions/gsd/prompts/complete-slice.md +1 -1
  56. package/dist/resources/extensions/gsd/prompts/discuss-headless.md +8 -8
  57. package/dist/resources/extensions/gsd/prompts/discuss.md +9 -9
  58. package/dist/resources/extensions/gsd/prompts/guided-discuss-project.md +4 -4
  59. package/dist/resources/extensions/gsd/prompts/guided-discuss-requirements.md +3 -3
  60. package/dist/resources/extensions/gsd/prompts/plan-slice.md +4 -4
  61. package/dist/resources/extensions/gsd/prompts/queue.md +4 -4
  62. package/dist/resources/extensions/gsd/prompts/refine-slice.md +2 -2
  63. package/dist/resources/extensions/gsd/prompts/rewrite-docs.md +1 -1
  64. package/dist/resources/extensions/gsd/queue-reorder-ui.js +30 -13
  65. package/dist/resources/extensions/gsd/smart-entry-routing.js +36 -0
  66. package/dist/resources/extensions/gsd/state-reconciliation/drift/merge-state.js +6 -1
  67. package/dist/resources/extensions/gsd/state-reconciliation/drift/project-md.js +9 -14
  68. package/dist/resources/extensions/gsd/state-reconciliation/drift/roadmap.js +19 -24
  69. package/dist/resources/extensions/gsd/status-guards.js +11 -0
  70. package/dist/resources/extensions/gsd/templates/plan.md +8 -5
  71. package/dist/resources/extensions/gsd/templates/task-plan.md +4 -2
  72. package/dist/resources/extensions/gsd/tools/complete-milestone.js +6 -8
  73. package/dist/resources/extensions/gsd/tools/complete-slice.js +6 -8
  74. package/dist/resources/extensions/gsd/tools/plan-milestone.js +7 -1
  75. package/dist/resources/extensions/gsd/tools/plan-slice.js +89 -14
  76. package/dist/resources/extensions/gsd/unit-context-manifest.js +32 -10
  77. package/dist/resources/extensions/gsd/validation.js +23 -1
  78. package/dist/resources/extensions/gsd/verification-gate.js +68 -7
  79. package/dist/resources/extensions/gsd/verification-verdict.js +26 -0
  80. package/dist/resources/extensions/gsd/workflow-mcp.js +17 -1
  81. package/dist/resources/extensions/gsd/workflow-projections.js +6 -8
  82. package/dist/resources/extensions/gsd/worktree-lifecycle.js +33 -8
  83. package/dist/resources/extensions/shared/html-shell.js +388 -0
  84. package/dist/resources/extensions/subagent/index.js +448 -78
  85. package/dist/resources/extensions/subagent/launch.js +77 -0
  86. package/dist/resources/extensions/subagent/run-store.js +148 -0
  87. package/dist/resources/extensions/visual-brief/artifact-policy.js +29 -0
  88. package/dist/resources/extensions/visual-brief/extension-manifest.json +8 -0
  89. package/dist/resources/extensions/visual-brief/index.js +5 -0
  90. package/dist/resources/extensions/visual-brief/page-contract.js +124 -0
  91. package/dist/resources/extensions/visual-brief/prompts.js +140 -0
  92. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  93. package/dist/web/standalone/.next/BUILD_ID +1 -1
  94. package/dist/web/standalone/.next/app-path-routes-manifest.json +13 -13
  95. package/dist/web/standalone/.next/build-manifest.json +3 -3
  96. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  97. package/dist/web/standalone/.next/react-loadable-manifest.json +3 -3
  98. package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  99. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  100. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  101. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  102. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  103. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  104. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  105. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  106. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  107. package/dist/web/standalone/.next/server/app/_not-found/page.js +2 -2
  108. package/dist/web/standalone/.next/server/app/_not-found/page.js.nft.json +1 -1
  109. package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  110. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  111. package/dist/web/standalone/.next/server/app/_not-found.rsc +4 -7
  112. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +4 -7
  113. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  114. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +4 -5
  115. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  116. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  117. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +2 -5
  118. package/dist/web/standalone/.next/server/app/api/browse-directories/route.js +1 -1
  119. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  120. package/dist/web/standalone/.next/server/app/index.html +1 -1
  121. package/dist/web/standalone/.next/server/app/index.rsc +4 -7
  122. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  123. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +4 -7
  124. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  125. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +4 -5
  126. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +2 -5
  127. package/dist/web/standalone/.next/server/app/page.js +2 -2
  128. package/dist/web/standalone/.next/server/app/page.js.nft.json +1 -1
  129. package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
  130. package/dist/web/standalone/.next/server/app-paths-manifest.json +13 -13
  131. package/dist/web/standalone/.next/server/chunks/4266.js +2 -0
  132. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  133. package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
  134. package/dist/web/standalone/.next/server/next-font-manifest.js +1 -1
  135. package/dist/web/standalone/.next/server/next-font-manifest.json +1 -1
  136. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  137. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  138. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  139. package/dist/web/standalone/.next/static/chunks/2973.33f26573894b6153.js +2 -0
  140. package/dist/web/standalone/.next/static/chunks/{8359.e059d86b255fce1c.js → 8359.7eb3bb8f8ecf4c01.js} +2 -2
  141. package/dist/web/standalone/.next/static/chunks/app/layout-8c10ec293ae0f1d5.js +1 -0
  142. package/dist/web/standalone/.next/static/chunks/{webpack-de742b64187e13fe.js → webpack-9a4db269f9ed63ad.js} +1 -1
  143. package/dist/web/standalone/.next/static/css/746ee28c929d1880.css +1 -0
  144. package/package.json +4 -4
  145. package/packages/mcp-server/src/workflow-tools.test.ts +1 -1
  146. package/packages/native/tsconfig.json +2 -1
  147. package/packages/native/tsconfig.tsbuildinfo +1 -1
  148. package/packages/pi-ai/dist/providers/google-gemini-cli.d.ts.map +1 -1
  149. package/packages/pi-ai/dist/providers/google-gemini-cli.js +5 -0
  150. package/packages/pi-ai/dist/providers/google-gemini-cli.js.map +1 -1
  151. package/packages/pi-ai/dist/providers/google-gemini-cli.test.d.ts +2 -0
  152. package/packages/pi-ai/dist/providers/google-gemini-cli.test.d.ts.map +1 -0
  153. package/packages/pi-ai/dist/providers/google-gemini-cli.test.js +41 -0
  154. package/packages/pi-ai/dist/providers/google-gemini-cli.test.js.map +1 -0
  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/google-gemini-cli.test.ts +49 -0
  171. package/packages/pi-ai/src/providers/google-gemini-cli.ts +7 -0
  172. package/packages/pi-ai/src/providers/openai-codex-responses.test.ts +63 -0
  173. package/packages/pi-ai/src/providers/openai-codex-responses.ts +91 -1
  174. package/packages/pi-ai/src/providers/simple-options.test.ts +60 -0
  175. package/packages/pi-ai/src/providers/simple-options.ts +5 -6
  176. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
  177. package/packages/pi-coding-agent/dist/core/agent-session-thinking-level.test.d.ts +2 -0
  178. package/packages/pi-coding-agent/dist/core/agent-session-thinking-level.test.d.ts.map +1 -0
  179. package/packages/pi-coding-agent/dist/core/agent-session-thinking-level.test.js +66 -0
  180. package/packages/pi-coding-agent/dist/core/agent-session-thinking-level.test.js.map +1 -0
  181. package/packages/pi-coding-agent/dist/core/agent-session.js +1 -1
  182. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  183. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.d.ts.map +1 -1
  184. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js +24 -6
  185. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js.map +1 -1
  186. package/packages/pi-coding-agent/src/core/agent-session-thinking-level.test.ts +79 -0
  187. package/packages/pi-coding-agent/src/core/agent-session.ts +1 -1
  188. package/packages/pi-coding-agent/src/modes/interactive/components/footer.ts +23 -7
  189. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
  190. package/packages/pi-tui/dist/__tests__/terminal.test.d.ts +2 -0
  191. package/packages/pi-tui/dist/__tests__/terminal.test.d.ts.map +1 -0
  192. package/packages/pi-tui/dist/__tests__/terminal.test.js +103 -0
  193. package/packages/pi-tui/dist/__tests__/terminal.test.js.map +1 -0
  194. package/packages/pi-tui/dist/terminal.d.ts +2 -0
  195. package/packages/pi-tui/dist/terminal.d.ts.map +1 -1
  196. package/packages/pi-tui/dist/terminal.js +12 -0
  197. package/packages/pi-tui/dist/terminal.js.map +1 -1
  198. package/packages/pi-tui/src/__tests__/terminal.test.ts +121 -0
  199. package/packages/pi-tui/src/terminal.ts +11 -0
  200. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
  201. package/src/resources/GSD-WORKFLOW.md +10 -1
  202. package/src/resources/extensions/claude-code-cli/partial-builder.ts +2 -1
  203. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +1 -1
  204. package/src/resources/extensions/claude-code-cli/tests/partial-builder.test.ts +19 -2
  205. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +9 -0
  206. package/src/resources/extensions/cmux/index.ts +6 -0
  207. package/src/resources/extensions/gsd/auto/contracts.ts +14 -6
  208. package/src/resources/extensions/gsd/auto/infra-errors.ts +9 -3
  209. package/src/resources/extensions/gsd/auto/loop.ts +8 -5
  210. package/src/resources/extensions/gsd/auto/orchestrator.ts +11 -0
  211. package/src/resources/extensions/gsd/auto/phases.ts +90 -38
  212. package/src/resources/extensions/gsd/auto/workflow-memory-pressure.ts +13 -0
  213. package/src/resources/extensions/gsd/auto-dashboard.ts +72 -1
  214. package/src/resources/extensions/gsd/auto-direct-dispatch.ts +1 -0
  215. package/src/resources/extensions/gsd/auto-dispatch.ts +19 -17
  216. package/src/resources/extensions/gsd/auto-model-selection.ts +2 -1
  217. package/src/resources/extensions/gsd/auto-post-unit.ts +266 -139
  218. package/src/resources/extensions/gsd/auto-prompts.ts +2 -2
  219. package/src/resources/extensions/gsd/auto-recovery.ts +74 -11
  220. package/src/resources/extensions/gsd/auto-start.ts +94 -12
  221. package/src/resources/extensions/gsd/auto-verification.ts +58 -36
  222. package/src/resources/extensions/gsd/auto-worktree.ts +193 -10
  223. package/src/resources/extensions/gsd/auto.ts +40 -5
  224. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +42 -7
  225. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +9 -8
  226. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +4 -2
  227. package/src/resources/extensions/gsd/bootstrap/subagent-input.ts +19 -7
  228. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +19 -3
  229. package/src/resources/extensions/gsd/clean-root-preflight.ts +174 -8
  230. package/src/resources/extensions/gsd/commands/catalog.ts +4 -1
  231. package/src/resources/extensions/gsd/commands/handlers/core.ts +40 -0
  232. package/src/resources/extensions/gsd/commands-bootstrap.ts +10 -0
  233. package/src/resources/extensions/gsd/commands-prefs-wizard.ts +8 -3
  234. package/src/resources/extensions/gsd/crash-recovery.ts +44 -4
  235. package/src/resources/extensions/gsd/db/milestone-leases.ts +26 -0
  236. package/src/resources/extensions/gsd/db/unit-dispatches.ts +4 -3
  237. package/src/resources/extensions/gsd/dispatch-guard.ts +2 -2
  238. package/src/resources/extensions/gsd/doctor-git-checks.ts +45 -1
  239. package/src/resources/extensions/gsd/doctor-runtime-checks.ts +25 -13
  240. package/src/resources/extensions/gsd/doctor-types.ts +1 -0
  241. package/src/resources/extensions/gsd/doctor.ts +2 -27
  242. package/src/resources/extensions/gsd/export-html.ts +27 -427
  243. package/src/resources/extensions/gsd/git-service.ts +51 -4
  244. package/src/resources/extensions/gsd/gsd-db.ts +21 -6
  245. package/src/resources/extensions/gsd/guided-flow-queue.ts +4 -3
  246. package/src/resources/extensions/gsd/guided-flow.ts +134 -133
  247. package/src/resources/extensions/gsd/guided-unit-context.ts +30 -0
  248. package/src/resources/extensions/gsd/migrate/parsers.ts +11 -0
  249. package/src/resources/extensions/gsd/migration-auto-check.ts +15 -23
  250. package/src/resources/extensions/gsd/milestone-actions.ts +10 -4
  251. package/src/resources/extensions/gsd/native-git-bridge.ts +54 -12
  252. package/src/resources/extensions/gsd/pending-auto-start.ts +79 -0
  253. package/src/resources/extensions/gsd/post-execution-checks.ts +87 -2
  254. package/src/resources/extensions/gsd/pre-execution-checks.ts +32 -1
  255. package/src/resources/extensions/gsd/prompt-loader.ts +1 -1
  256. package/src/resources/extensions/gsd/prompts/complete-milestone.md +1 -1
  257. package/src/resources/extensions/gsd/prompts/complete-slice.md +1 -1
  258. package/src/resources/extensions/gsd/prompts/discuss-headless.md +8 -8
  259. package/src/resources/extensions/gsd/prompts/discuss.md +9 -9
  260. package/src/resources/extensions/gsd/prompts/guided-discuss-project.md +4 -4
  261. package/src/resources/extensions/gsd/prompts/guided-discuss-requirements.md +3 -3
  262. package/src/resources/extensions/gsd/prompts/plan-slice.md +4 -4
  263. package/src/resources/extensions/gsd/prompts/queue.md +4 -4
  264. package/src/resources/extensions/gsd/prompts/refine-slice.md +2 -2
  265. package/src/resources/extensions/gsd/prompts/rewrite-docs.md +1 -1
  266. package/src/resources/extensions/gsd/queue-reorder-ui.ts +31 -13
  267. package/src/resources/extensions/gsd/smart-entry-routing.ts +77 -0
  268. package/src/resources/extensions/gsd/state-reconciliation/drift/merge-state.ts +8 -1
  269. package/src/resources/extensions/gsd/state-reconciliation/drift/project-md.ts +12 -15
  270. package/src/resources/extensions/gsd/state-reconciliation/drift/roadmap.ts +17 -25
  271. package/src/resources/extensions/gsd/status-guards.ts +13 -0
  272. package/src/resources/extensions/gsd/templates/plan.md +8 -5
  273. package/src/resources/extensions/gsd/templates/task-plan.md +4 -2
  274. package/src/resources/extensions/gsd/tests/auto-dashboard.test.ts +71 -0
  275. package/src/resources/extensions/gsd/tests/auto-deterministic-error-classification-4973.test.ts +116 -0
  276. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +56 -0
  277. package/src/resources/extensions/gsd/tests/auto-orchestrator.test.ts +80 -1
  278. package/src/resources/extensions/gsd/tests/auto-paused-ui-cleanup.test.ts +35 -7
  279. package/src/resources/extensions/gsd/tests/auto-phases-lifecycle.test.ts +53 -2
  280. package/src/resources/extensions/gsd/tests/auto-post-unit-step-message.test.ts +12 -1
  281. package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +91 -6
  282. package/src/resources/extensions/gsd/tests/auto-start-orphan-bootstrap.test.ts +1 -0
  283. package/src/resources/extensions/gsd/tests/auto-stop-notification.test.ts +20 -0
  284. package/src/resources/extensions/gsd/tests/auto-worktree-registry.test.ts +69 -1
  285. package/src/resources/extensions/gsd/tests/brief-command.test.ts +89 -0
  286. package/src/resources/extensions/gsd/tests/checkout-branch-stash-guard.test.ts +87 -0
  287. package/src/resources/extensions/gsd/tests/clean-root-preflight.test.ts +107 -2
  288. package/src/resources/extensions/gsd/tests/clear-stale-autostart.test.ts +11 -2
  289. package/src/resources/extensions/gsd/tests/closeout-git-deferral.test.ts +16 -0
  290. package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +4 -1
  291. package/src/resources/extensions/gsd/tests/complete-slice.test.ts +5 -9
  292. package/src/resources/extensions/gsd/tests/complete-task.test.ts +3 -1
  293. package/src/resources/extensions/gsd/tests/crash-recovery-via-db.test.ts +86 -2
  294. package/src/resources/extensions/gsd/tests/custom-engine-loop-integration.test.ts +2 -0
  295. package/src/resources/extensions/gsd/tests/db-authority-regression.test.ts +208 -0
  296. package/src/resources/extensions/gsd/tests/deep-project-auto-loop.test.ts +59 -2
  297. package/src/resources/extensions/gsd/tests/dispatch-complete-milestone-guard.test.ts +66 -0
  298. package/src/resources/extensions/gsd/tests/dispatch-guard.test.ts +27 -0
  299. package/src/resources/extensions/gsd/tests/doctor-empty-worktree.test.ts +65 -0
  300. package/src/resources/extensions/gsd/tests/evidence-cross-ref.test.ts +38 -0
  301. package/src/resources/extensions/gsd/tests/export-html-enhancements.test.ts +8 -0
  302. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +11 -0
  303. package/src/resources/extensions/gsd/tests/guided-discuss-project-prompt-rendering.test.ts +2 -0
  304. package/src/resources/extensions/gsd/tests/guided-dispatch-root.test.ts +106 -0
  305. package/src/resources/extensions/gsd/tests/guided-flow-session-isolation.test.ts +59 -11
  306. package/src/resources/extensions/gsd/tests/guided-flow.test.ts +21 -0
  307. package/src/resources/extensions/gsd/tests/guided-tool-contract.test.ts +65 -0
  308. package/src/resources/extensions/gsd/tests/headless-milestone-parity.test.ts +7 -7
  309. package/src/resources/extensions/gsd/tests/hook-model-resolution.test.ts +5 -0
  310. package/src/resources/extensions/gsd/tests/infra-error.test.ts +2 -2
  311. package/src/resources/extensions/gsd/tests/infra-errors-cooldown.test.ts +9 -0
  312. package/src/resources/extensions/gsd/tests/integration/doctor-runtime.test.ts +20 -0
  313. package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +112 -1
  314. package/src/resources/extensions/gsd/tests/integration/state-machine-runtime-failures.test.ts +6 -1
  315. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +46 -0
  316. package/src/resources/extensions/gsd/tests/merge-db-cycle.test.ts +179 -0
  317. package/src/resources/extensions/gsd/tests/migrate-validator-parsers.test.ts +24 -1
  318. package/src/resources/extensions/gsd/tests/migration-auto-check.test.ts +26 -18
  319. package/src/resources/extensions/gsd/tests/native-git-bridge-exec-fallback.test.ts +63 -2
  320. package/src/resources/extensions/gsd/tests/orphaned-worktree-audit.test.ts +121 -1
  321. package/src/resources/extensions/gsd/tests/park-db-sync.test.ts +55 -1
  322. package/src/resources/extensions/gsd/tests/pending-autostart-scope.test.ts +29 -5
  323. package/src/resources/extensions/gsd/tests/plan-milestone.test.ts +26 -0
  324. package/src/resources/extensions/gsd/tests/plan-slice-prompt.test.ts +2 -0
  325. package/src/resources/extensions/gsd/tests/plan-slice.test.ts +225 -1
  326. package/src/resources/extensions/gsd/tests/plan-task.test.ts +17 -0
  327. package/src/resources/extensions/gsd/tests/post-exec-retry-bypass.test.ts +79 -1
  328. package/src/resources/extensions/gsd/tests/post-execution-checks.test.ts +86 -0
  329. package/src/resources/extensions/gsd/tests/post-unit-git-failure.test.ts +1 -1
  330. package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +53 -0
  331. package/src/resources/extensions/gsd/tests/prefs-wizard-coverage.test.ts +59 -0
  332. package/src/resources/extensions/gsd/tests/prompt-loader.test.ts +23 -0
  333. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +37 -1
  334. package/src/resources/extensions/gsd/tests/queue-reorder-ui.test.ts +54 -0
  335. package/src/resources/extensions/gsd/tests/remediation-completion-guard.test.ts +89 -2
  336. package/src/resources/extensions/gsd/tests/run-uat-replay-cap.test.ts +2 -3
  337. package/src/resources/extensions/gsd/tests/session-switch-abort-misclassification.test.ts +10 -0
  338. package/src/resources/extensions/gsd/tests/smart-entry-routing.test.ts +113 -0
  339. package/src/resources/extensions/gsd/tests/start-auto-detached.test.ts +53 -2
  340. package/src/resources/extensions/gsd/tests/state-reconciliation-drift.test.ts +119 -23
  341. package/src/resources/extensions/gsd/tests/status-guards.test.ts +13 -1
  342. package/src/resources/extensions/gsd/tests/stuck-state-via-db.test.ts +64 -1
  343. package/src/resources/extensions/gsd/tests/summary-render-parity.test.ts +7 -3
  344. package/src/resources/extensions/gsd/tests/unit-context-manifest.test.ts +86 -7
  345. package/src/resources/extensions/gsd/tests/validate-milestone-stuck-guard.test.ts +29 -2
  346. package/src/resources/extensions/gsd/tests/verification-gate.test.ts +110 -1
  347. package/src/resources/extensions/gsd/tests/verification-verdict.test.ts +78 -0
  348. package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +19 -1
  349. package/src/resources/extensions/gsd/tests/workflow-memory-pressure.test.ts +21 -1
  350. package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +1 -1
  351. package/src/resources/extensions/gsd/tests/worktree-git-pathspec.test.ts +39 -0
  352. package/src/resources/extensions/gsd/tests/worktree-journal-events.test.ts +64 -12
  353. package/src/resources/extensions/gsd/tests/write-gate-planning-unit.test.ts +54 -0
  354. package/src/resources/extensions/gsd/tools/complete-milestone.ts +8 -10
  355. package/src/resources/extensions/gsd/tools/complete-slice.ts +6 -8
  356. package/src/resources/extensions/gsd/tools/plan-milestone.ts +5 -1
  357. package/src/resources/extensions/gsd/tools/plan-slice.ts +98 -12
  358. package/src/resources/extensions/gsd/types.ts +1 -1
  359. package/src/resources/extensions/gsd/unit-context-manifest.ts +47 -11
  360. package/src/resources/extensions/gsd/validation.ts +23 -1
  361. package/src/resources/extensions/gsd/verification-gate.ts +78 -6
  362. package/src/resources/extensions/gsd/verification-verdict.ts +47 -0
  363. package/src/resources/extensions/gsd/workflow-mcp.ts +18 -1
  364. package/src/resources/extensions/gsd/workflow-projections.ts +6 -8
  365. package/src/resources/extensions/gsd/worktree-lifecycle.ts +41 -8
  366. package/src/resources/extensions/shared/html-shell.ts +412 -0
  367. package/src/resources/extensions/subagent/index.ts +567 -103
  368. package/src/resources/extensions/subagent/launch.ts +131 -0
  369. package/src/resources/extensions/subagent/run-store.ts +218 -0
  370. package/src/resources/extensions/subagent/tests/launch.test.ts +115 -0
  371. package/src/resources/extensions/subagent/tests/run-store.test.ts +111 -0
  372. package/src/resources/extensions/visual-brief/artifact-policy.ts +41 -0
  373. package/src/resources/extensions/visual-brief/extension-manifest.json +8 -0
  374. package/src/resources/extensions/visual-brief/index.ts +8 -0
  375. package/src/resources/extensions/visual-brief/page-contract.ts +136 -0
  376. package/src/resources/extensions/visual-brief/prompts.ts +183 -0
  377. package/src/resources/extensions/visual-brief/tests/visual-brief.test.ts +212 -0
  378. package/dist/web/standalone/.next/server/chunks/5822.js +0 -2
  379. package/dist/web/standalone/.next/static/chunks/2556.0527fea66e123b7f.js +0 -1
  380. package/dist/web/standalone/.next/static/chunks/app/layout-a16c7a7ecdf0c2cf.js +0 -1
  381. package/dist/web/standalone/.next/static/css/54ec2745c1da488b.css +0 -1
  382. package/dist/web/standalone/.next/static/css/de70bee13400563f.css +0 -1
  383. package/dist/web/standalone/.next/static/media/4cf2300e9c8272f7-s.p.woff2 +0 -0
  384. package/dist/web/standalone/.next/static/media/747892c23ea88013-s.woff2 +0 -0
  385. package/dist/web/standalone/.next/static/media/8d697b304b401681-s.woff2 +0 -0
  386. package/dist/web/standalone/.next/static/media/93f479601ee12b01-s.p.woff2 +0 -0
  387. package/dist/web/standalone/.next/static/media/9610d9e46709d722-s.woff2 +0 -0
  388. package/dist/web/standalone/.next/static/media/ba015fad6dcf6784-s.woff2 +0 -0
  389. /package/dist/web/standalone/.next/static/{Wop3A7KRGyR06H3rla_1- → 4dSwdrs__8NwCZggxP9KF}/_buildManifest.js +0 -0
  390. /package/dist/web/standalone/.next/static/{Wop3A7KRGyR06H3rla_1- → 4dSwdrs__8NwCZggxP9KF}/_ssgManifest.js +0 -0
@@ -5,7 +5,7 @@ import { join } from "node:path";
5
5
  import { tmpdir } from "node:os";
6
6
  import { randomUUID } from "node:crypto";
7
7
 
8
- import { verifyExpectedArtifact, hasImplementationArtifacts, resolveExpectedArtifactPath, diagnoseExpectedArtifact, buildLoopRemediationSteps, writeBlockerPlaceholder, refreshRecoveryDbForArtifact } from "../auto-recovery.ts";
8
+ import { verifyExpectedArtifact, hasImplementationArtifacts, resolveExpectedArtifactPath, diagnoseExpectedArtifact, diagnoseWorktreeIntegrityFailure, buildLoopRemediationSteps, writeBlockerPlaceholder, refreshRecoveryDbForArtifact } from "../auto-recovery.ts";
9
9
  import { resolveMilestoneFile } from "../paths.ts";
10
10
  import { openDatabase, closeDatabase, insertMilestone, insertSlice, insertGateRow, insertTask, getMilestoneCommitAttributionShas } from "../gsd-db.ts";
11
11
  import { clearParseCache } from "../files.ts";
@@ -141,6 +141,20 @@ test("resolveExpectedArtifactPath returns null for unknown type", () => {
141
141
  }
142
142
  });
143
143
 
144
+ test("diagnoseWorktreeIntegrityFailure reports missing GSD worktree paths only", () => {
145
+ const missingWorktreePath = join(tmpdir(), `gsd-test-${randomUUID()}`, ".gsd", "worktrees", "M001-S01");
146
+ assert.equal(
147
+ diagnoseWorktreeIntegrityFailure(join(tmpdir(), `gsd-test-${randomUUID()}`)),
148
+ null,
149
+ "non-GSD paths should keep falling through to artifact recovery",
150
+ );
151
+ assert.equal(
152
+ diagnoseWorktreeIntegrityFailure(missingWorktreePath),
153
+ `Worktree integrity failure: ${missingWorktreePath} does not exist. Repair or recreate the worktree before retrying.`,
154
+ "missing GSD worktree paths should fail terminally before artifact retry",
155
+ );
156
+ });
157
+
144
158
  test("resolveExpectedArtifactPath returns correct path for all milestone-level types", () => {
145
159
  const base = makeTmpBase();
146
160
  try {
@@ -793,6 +807,51 @@ test("hasImplementationArtifacts finds integration implementation-only commits w
793
807
  }
794
808
  });
795
809
 
810
+ test("hasImplementationArtifacts ignores corrupted milestone/* integration metadata", () => {
811
+ const base = makeGitBase();
812
+ try {
813
+ mkdirSync(join(base, "src"), { recursive: true });
814
+ writeFileSync(join(base, "src", "feature.ts"), "export function feature() {}\n");
815
+ execFileSync("git", ["add", "src/feature.ts"], { cwd: base, stdio: "ignore" });
816
+ execFileSync("git", ["commit", "-m", "feat: add milestone feature\n\nGSD-Task: S01/T01"], { cwd: base, stdio: "ignore" });
817
+
818
+ mkdirSync(join(base, ".gsd"), { recursive: true });
819
+ openDatabase(join(base, ".gsd", "gsd.db"));
820
+ insertMilestone({ id: "M001", title: "Milestone One", status: "active" });
821
+ insertSlice({
822
+ id: "S01",
823
+ milestoneId: "M001",
824
+ title: "Slice One",
825
+ status: "complete",
826
+ risk: "low",
827
+ depends: [],
828
+ });
829
+ insertTask({
830
+ id: "T01",
831
+ sliceId: "S01",
832
+ milestoneId: "M001",
833
+ title: "Task One",
834
+ status: "complete",
835
+ });
836
+
837
+ execFileSync("git", ["checkout", "-b", "milestone/M001"], { cwd: base, stdio: "ignore" });
838
+ mkdirSync(join(base, ".gsd", "milestones", "M001"), { recursive: true });
839
+ writeFileSync(join(base, ".gsd", "milestones", "M001", "M001-SUMMARY.md"), "# Milestone Summary\nDone.");
840
+ execFileSync("git", ["add", "."], { cwd: base, stdio: "ignore" });
841
+ execFileSync("git", ["commit", "-m", "chore: auto-commit after complete-milestone\n\nGSD-Unit: M001"], { cwd: base, stdio: "ignore" });
842
+
843
+ writeFileSync(
844
+ join(base, ".gsd", "milestones", "M001", "M001-META.json"),
845
+ JSON.stringify({ integrationBranch: "milestone/M001" }, null, 2) + "\n",
846
+ );
847
+
848
+ const result = hasImplementationArtifacts(base, "M001");
849
+ assert.equal(result, "present", "corrupted milestone integration metadata should fall back to main branch for artifact detection");
850
+ } finally {
851
+ cleanup(base);
852
+ }
853
+ });
854
+
796
855
  test("hasImplementationArtifacts backfills untagged main implementation commits from completed task file hints", () => {
797
856
  const base = makeGitBase();
798
857
  try {
@@ -1029,24 +1088,50 @@ test("hasImplementationArtifacts binds GSD-Task trailer to milestone via DB stat
1029
1088
  }
1030
1089
  });
1031
1090
 
1032
- test("hasImplementationArtifacts does not bind GSD-Task trailer without milestone ownership evidence", () => {
1091
+ test("hasImplementationArtifacts does not claim Sxx/Tyy commit trailers across milestones when ownership points elsewhere", () => {
1033
1092
  const base = makeGitBase();
1034
1093
  try {
1035
1094
  writeFileSync(join(base, ".git", "info", "exclude"), ".gsd/\n");
1095
+ mkdirSync(join(base, ".gsd"), { recursive: true });
1096
+ openDatabase(join(base, ".gsd", "gsd.db"));
1097
+ insertMilestone({ id: "M001", title: "Milestone One", status: "active" });
1098
+ insertMilestone({ id: "M002", title: "Milestone Two", status: "active" });
1099
+ insertSlice({
1100
+ id: "S01",
1101
+ milestoneId: "M002",
1102
+ title: "Slice One",
1103
+ status: "complete",
1104
+ risk: "low",
1105
+ depends: [],
1106
+ });
1107
+ insertTask({
1108
+ id: "T01",
1109
+ sliceId: "S01",
1110
+ milestoneId: "M002",
1111
+ title: "Task One",
1112
+ status: "complete",
1113
+ });
1114
+
1036
1115
  mkdirSync(join(base, "src"), { recursive: true });
1037
1116
  writeFileSync(join(base, "src", "feature.ts"), "export function feature() {}\n");
1038
1117
  execFileSync("git", ["add", "."], { cwd: base, stdio: "ignore" });
1039
1118
  execFileSync(
1040
1119
  "git",
1041
- ["commit", "-m", "feat: add feature\n\nGSD-Task: S01/T01"],
1120
+ ["commit", "-m", "feat: add sibling feature\n\nGSD-Task: S01/T01"],
1042
1121
  { cwd: base, stdio: "ignore" },
1043
1122
  );
1044
1123
 
1045
- const result = hasImplementationArtifacts(base, "M001");
1124
+ const m001Result = hasImplementationArtifacts(base, "M001");
1125
+ const m002Result = hasImplementationArtifacts(base, "M002");
1046
1126
  assert.equal(
1047
- result,
1127
+ m001Result,
1048
1128
  "absent",
1049
- "S01/T01 shape alone must not bind an implementation commit to M001",
1129
+ "Sxx/Tyy commit trailers owned by M002 must not be attributed to M001",
1130
+ );
1131
+ assert.equal(
1132
+ m002Result,
1133
+ "present",
1134
+ "the owning milestone should still claim the implementation-bearing commit",
1050
1135
  );
1051
1136
  } finally {
1052
1137
  cleanup(base);
@@ -98,6 +98,7 @@ test("bootstrap aborts before starting next milestone when completed orphan merg
98
98
  {
99
99
  shouldUseWorktreeIsolation: () => true,
100
100
  registerSigtermHandler: () => {},
101
+ registerAutoWorkerForSession: () => {},
101
102
  lockBase: () => base,
102
103
  buildLifecycle: () => ({
103
104
  adoptSessionRoot: (sessionBase: string, originalBase?: string) => {
@@ -0,0 +1,20 @@
1
+ // Project/App: GSD-2
2
+ // File Purpose: Regression tests for auto-mode stop notification formatting.
3
+
4
+ import test from "node:test";
5
+ import assert from "node:assert/strict";
6
+
7
+ import { formatAutoStopNotification } from "../auto.ts";
8
+
9
+ test("auto stop notification keeps session totals on a separate line", () => {
10
+ const message = formatAutoStopNotification(
11
+ "Auto-mode stopped",
12
+ { cost: 0.652, tokens: { total: 87000 } },
13
+ 2,
14
+ );
15
+
16
+ assert.equal(
17
+ message,
18
+ "Auto-mode stopped.\nSession: $0.652 · 87.0k tokens · 2 units",
19
+ );
20
+ });
@@ -2,7 +2,7 @@
2
2
 
3
3
  import { describe, test, beforeEach } from "node:test";
4
4
  import assert from "node:assert/strict";
5
- import { mkdtempSync, mkdirSync, writeFileSync, rmSync, realpathSync } from "node:fs";
5
+ import { existsSync, mkdtempSync, mkdirSync, writeFileSync, rmSync, realpathSync } from "node:fs";
6
6
  import { join } from "node:path";
7
7
  import { tmpdir } from "node:os";
8
8
  import { execFileSync } from "node:child_process";
@@ -13,6 +13,7 @@ import {
13
13
  _resetAutoWorktreeOriginalBaseForTests,
14
14
  createAutoWorktree,
15
15
  enterAutoWorktree,
16
+ mergeMilestoneToMain,
16
17
  teardownAutoWorktree,
17
18
  } from "../auto-worktree.ts";
18
19
 
@@ -173,4 +174,71 @@ describe("auto-worktree workspace registry", () => {
173
174
  teardownAutoWorktree(dir2, "M020");
174
175
  try { process.chdir(savedCwd); } catch { /* ignore */ }
175
176
  });
177
+
178
+ test("mergeMilestoneToMain cleans up when milestone branch was already regular-merged", (t) => {
179
+ const tempDir = createTempRepo(t);
180
+ const msDir = join(tempDir, ".gsd", "milestones", "M003");
181
+ mkdirSync(msDir, { recursive: true });
182
+ writeFileSync(join(msDir, "CONTEXT.md"), "# M003 Context\n");
183
+ git(["add", "."], tempDir);
184
+ git(["commit", "-m", "add milestone"], tempDir);
185
+
186
+ createAutoWorktree(tempDir, "M003");
187
+ const wtDir = join(tempDir, ".gsd", "worktrees", "M003");
188
+ writeFileSync(join(wtDir, "feature.txt"), "implemented\n");
189
+ git(["add", "feature.txt"], wtDir);
190
+ git(["commit", "-m", "feat: implement M003"], wtDir);
191
+
192
+ process.chdir(tempDir);
193
+ git(["merge", "--no-ff", "milestone/M003", "-m", "merge M003"], tempDir);
194
+
195
+ process.chdir(wtDir);
196
+ const result = mergeMilestoneToMain(tempDir, "M003", "# M003\n- [x] **S01: Done**\n");
197
+
198
+ assert.equal(result.codeFilesChanged, true);
199
+ assert.equal(result.pushed, false);
200
+ assert.equal(result.prCreated, false);
201
+ assert.equal(existsSync(wtDir), false, "worktree directory is removed");
202
+ assert.throws(
203
+ () => git(["rev-parse", "--verify", "milestone/M003"], tempDir),
204
+ /Command failed/,
205
+ "already-merged milestone branch is deleted",
206
+ );
207
+ try { process.chdir(savedCwd); } catch { /* ignore */ }
208
+ });
209
+
210
+ test("mergeMilestoneToMain cleans up already-merged milestone after main advances", (t) => {
211
+ const tempDir = createTempRepo(t);
212
+ const msDir = join(tempDir, ".gsd", "milestones", "M004");
213
+ mkdirSync(msDir, { recursive: true });
214
+ writeFileSync(join(msDir, "CONTEXT.md"), "# M004 Context\n");
215
+ git(["add", "."], tempDir);
216
+ git(["commit", "-m", "add milestone"], tempDir);
217
+
218
+ createAutoWorktree(tempDir, "M004");
219
+ const wtDir = join(tempDir, ".gsd", "worktrees", "M004");
220
+ writeFileSync(join(wtDir, "feature.txt"), "implemented\n");
221
+ git(["add", "feature.txt"], wtDir);
222
+ git(["commit", "-m", "feat: implement M004"], wtDir);
223
+
224
+ process.chdir(tempDir);
225
+ git(["merge", "--no-ff", "milestone/M004", "-m", "merge M004"], tempDir);
226
+ writeFileSync(join(tempDir, "hotfix.txt"), "later main work\n");
227
+ git(["add", "hotfix.txt"], tempDir);
228
+ git(["commit", "-m", "fix: advance main"], tempDir);
229
+
230
+ process.chdir(wtDir);
231
+ const result = mergeMilestoneToMain(tempDir, "M004", "# M004\n- [x] **S01: Done**\n");
232
+
233
+ assert.equal(result.codeFilesChanged, true);
234
+ assert.equal(result.pushed, false);
235
+ assert.equal(result.prCreated, false);
236
+ assert.equal(existsSync(wtDir), false, "worktree directory is removed");
237
+ assert.throws(
238
+ () => git(["rev-parse", "--verify", "milestone/M004"], tempDir),
239
+ /Command failed/,
240
+ "already-merged milestone branch is deleted",
241
+ );
242
+ try { process.chdir(savedCwd); } catch { /* ignore */ }
243
+ });
176
244
  });
@@ -0,0 +1,89 @@
1
+ // GSD-2 + /gsd brief command behavior tests
2
+
3
+ import test from "node:test";
4
+ import assert from "node:assert/strict";
5
+
6
+ import { GSD_COMMAND_DESCRIPTION, getGsdArgumentCompletions } from "../commands/catalog.ts";
7
+ import { handleCoreCommand, showHelp } from "../commands/handlers/core.ts";
8
+ import { VISUAL_BRIEF_USAGE } from "../../visual-brief/prompts.ts";
9
+
10
+ function createMockCtx() {
11
+ const notifications: { message: string; level: string }[] = [];
12
+ return {
13
+ notifications,
14
+ ui: {
15
+ notify(message: string, level: string) {
16
+ notifications.push({ message, level });
17
+ },
18
+ custom: async () => undefined,
19
+ },
20
+ };
21
+ }
22
+
23
+ function createMockPi() {
24
+ const sentMessages: string[] = [];
25
+ return {
26
+ sentMessages,
27
+ sendUserMessage(content: string) {
28
+ sentMessages.push(content);
29
+ },
30
+ };
31
+ }
32
+
33
+ test("/gsd brief appears in the command description and top-level completions", () => {
34
+ assert.match(GSD_COMMAND_DESCRIPTION, /brief/);
35
+
36
+ const completions = getGsdArgumentCompletions("br");
37
+ const entry = completions.find((completion) => completion.value === "brief");
38
+
39
+ assert.ok(entry, "brief should appear in top-level completions");
40
+ assert.match(entry.description, /visual HTML brief/i);
41
+ });
42
+
43
+ test("/gsd brief exposes Visual Brief mode completions", () => {
44
+ const completions = getGsdArgumentCompletions("brief d");
45
+
46
+ assert.ok(
47
+ completions.some((completion) => completion.value === "brief diagram"),
48
+ "diagram should be suggested as a /gsd brief mode",
49
+ );
50
+ assert.ok(
51
+ getGsdArgumentCompletions("brief ").some((completion) => completion.value === "brief diff"),
52
+ "diff should be suggested after /gsd brief",
53
+ );
54
+ });
55
+
56
+ test("/gsd brief sends a Visual Brief prompt for valid args", async () => {
57
+ const ctx = createMockCtx();
58
+ const pi = createMockPi();
59
+
60
+ const handled = await handleCoreCommand("brief diagram extension loading lifecycle", ctx as any, pi as any);
61
+
62
+ assert.equal(handled, true);
63
+ assert.equal(ctx.notifications.length, 0);
64
+ assert.equal(pi.sentMessages.length, 1);
65
+ assert.match(pi.sentMessages[0], /Mode: diagram/);
66
+ assert.match(pi.sentMessages[0], /Subject: extension loading lifecycle/);
67
+ assert.match(pi.sentMessages[0], /Output directory:/);
68
+ });
69
+
70
+ test("/gsd brief reports usage for empty args instead of sending a prompt", async () => {
71
+ const ctx = createMockCtx();
72
+ const pi = createMockPi();
73
+
74
+ const handled = await handleCoreCommand("brief", ctx as any, pi as any);
75
+
76
+ assert.equal(handled, true);
77
+ assert.equal(pi.sentMessages.length, 0);
78
+ assert.deepEqual(ctx.notifications, [{ message: VISUAL_BRIEF_USAGE, level: "info" }]);
79
+ });
80
+
81
+ test("/gsd help full lists brief separately from visualize", () => {
82
+ const ctx = createMockCtx();
83
+
84
+ showHelp(ctx as any, "full");
85
+
86
+ const help = ctx.notifications.at(0)?.message ?? "";
87
+ assert.match(help, /\/gsd visualize\s+Interactive 10-tab TUI/);
88
+ assert.match(help, /\/gsd brief <mode>\s+Generate a visual HTML brief/);
89
+ });
@@ -0,0 +1,87 @@
1
+ import { describe, test } from "node:test";
2
+ import assert from "node:assert/strict";
3
+ import { execFileSync } from "node:child_process";
4
+ import { mkdtempSync, readFileSync, realpathSync, rmSync, writeFileSync } from "node:fs";
5
+ import { join } from "node:path";
6
+ import { tmpdir } from "node:os";
7
+
8
+ import { checkoutBranchWithStashGuard } from "../auto-worktree.ts";
9
+
10
+ function git(args: string[], cwd: string): string {
11
+ return execFileSync("git", args, {
12
+ cwd,
13
+ stdio: ["ignore", "pipe", "pipe"],
14
+ encoding: "utf-8",
15
+ });
16
+ }
17
+
18
+ function createRepo(t: { after: (fn: () => void) => void }): string {
19
+ const dir = realpathSync(mkdtempSync(join(tmpdir(), "checkout-stash-guard-")));
20
+ t.after(() => rmSync(dir, { recursive: true, force: true }));
21
+ git(["init"], dir);
22
+ git(["config", "user.email", "test@example.com"], dir);
23
+ git(["config", "user.name", "Test User"], dir);
24
+ writeFileSync(join(dir, "note.txt"), "base\n");
25
+ git(["add", "note.txt"], dir);
26
+ git(["commit", "-m", "init"], dir);
27
+ git(["branch", "-M", "main"], dir);
28
+ return dir;
29
+ }
30
+
31
+ describe("checkoutBranchWithStashGuard", () => {
32
+ test("restores dirty working tree after successful checkout", (t) => {
33
+ const repo = createRepo(t);
34
+ git(["checkout", "-b", "milestone/M001"], repo);
35
+ git(["checkout", "main"], repo);
36
+
37
+ writeFileSync(join(repo, "note.txt"), "dirty\n");
38
+
39
+ checkoutBranchWithStashGuard(repo, "milestone/M001", "test-success");
40
+
41
+ const branch = git(["branch", "--show-current"], repo).trim();
42
+ assert.equal(branch, "milestone/M001");
43
+ const content = git(["show", "HEAD:note.txt"], repo).trim();
44
+ assert.equal(content, "base");
45
+ const wtContent = readFileSync(join(repo, "note.txt"), "utf8");
46
+ assert.equal(wtContent, "dirty\n");
47
+ const status = git(["status", "--porcelain"], repo);
48
+ assert.match(status, /note\.txt/);
49
+ });
50
+
51
+ test("restores dirty working tree when checkout throws", (t) => {
52
+ const repo = createRepo(t);
53
+ writeFileSync(join(repo, "note.txt"), "dirty\n");
54
+
55
+ assert.throws(
56
+ () => checkoutBranchWithStashGuard(repo, "milestone/DOES-NOT-EXIST", "test-failure"),
57
+ );
58
+
59
+ const status = git(["status", "--porcelain"], repo);
60
+ assert.match(status, /note\.txt/);
61
+ const stashList = git(["stash", "list"], repo).trim();
62
+ assert.equal(stashList, "");
63
+ });
64
+
65
+ test("surfaces distinct error when checkout succeeds but stash pop conflicts", (t) => {
66
+ const repo = createRepo(t);
67
+ // Branch B has a divergent version of note.txt so popping a stash made
68
+ // against main will conflict after the checkout to B.
69
+ git(["checkout", "-b", "milestone/B"], repo);
70
+ writeFileSync(join(repo, "note.txt"), "B-version\n");
71
+ git(["add", "note.txt"], repo);
72
+ git(["commit", "-m", "B"], repo);
73
+ git(["checkout", "main"], repo);
74
+
75
+ writeFileSync(join(repo, "note.txt"), "local\n");
76
+
77
+ assert.throws(
78
+ () => checkoutBranchWithStashGuard(repo, "milestone/B", "test-pop-failure"),
79
+ /checkout to 'milestone\/B' succeeded but stash restore failed/,
80
+ );
81
+
82
+ const branch = git(["branch", "--show-current"], repo).trim();
83
+ assert.equal(branch, "milestone/B");
84
+ const stashList = git(["stash", "list"], repo).trim();
85
+ assert.match(stashList, /gsd: checkout stash/);
86
+ });
87
+ });
@@ -8,7 +8,7 @@
8
8
 
9
9
  import test from "node:test";
10
10
  import assert from "node:assert/strict";
11
- import { mkdtempSync, mkdirSync, writeFileSync, rmSync, readFileSync, realpathSync } from "node:fs";
11
+ import { existsSync, mkdtempSync, mkdirSync, writeFileSync, rmSync, readFileSync, realpathSync } from "node:fs";
12
12
  import { join } from "node:path";
13
13
  import { tmpdir } from "node:os";
14
14
  import { execSync } from "node:child_process";
@@ -208,8 +208,8 @@ test("postflightPopStash conflict warning names the exact stash ref", () => {
208
208
  assert.match(postflight.message, /failed after merge of milestone M005C/);
209
209
 
210
210
  const warning = notifications.find((n) => n.level === "warning")?.msg ?? "";
211
- assert.match(warning, /git stash pop stash@\{\d+\}/);
212
211
  assert.match(warning, /git stash apply stash@\{\d+\}/);
212
+ assert.match(warning, /git stash drop stash@\{\d+\}/);
213
213
  } finally {
214
214
  try { rmSync(repo, { recursive: true, force: true, maxRetries: 3, retryDelay: 100 }); } catch { /* ignore */ }
215
215
  }
@@ -279,3 +279,108 @@ test("postflightPopStash falls back to milestone marker prefix when exact marker
279
279
  try { rmSync(repo, { recursive: true, force: true, maxRetries: 3, retryDelay: 100 }); } catch { /* ignore */ }
280
280
  }
281
281
  });
282
+
283
+ test("postflightPopStash preserves stash when untracked collision differs from merged file", () => {
284
+ const repo = createTempRepo();
285
+ try {
286
+ writeFileSync(join(repo, "tests.txt"), "local preflight test\n");
287
+ const preflight = preflightCleanRoot(repo, "M009", () => {});
288
+ assert.equal(preflight.stashPushed, true, "preflight must stash untracked file");
289
+
290
+ writeFileSync(join(repo, "tests.txt"), "merged milestone test\n");
291
+ run("git add tests.txt", repo);
292
+ run('git commit -m "feat: add merged test"', repo);
293
+
294
+ const notifications: Array<{ msg: string; level: string }> = [];
295
+ const postflight = postflightPopStash(repo, "M009", preflight.stashMarker, (msg, level) => {
296
+ notifications.push({ msg, level });
297
+ });
298
+
299
+ assert.equal(postflight.needsManualRecovery, false, "different already-present untracked files must not stop auto-mode");
300
+ assert.equal(postflight.resolution, "already-present-preserved");
301
+ assert.deepEqual(postflight.collidedPaths, ["tests.txt"]);
302
+ assert.equal(readFileSync(join(repo, "tests.txt"), "utf-8"), "merged milestone test\n");
303
+ assert.equal(run("git status --porcelain", repo), "", "merged file must stay clean");
304
+
305
+ const stashList = run("git stash list", repo);
306
+ assert.ok(preflight.stashMarker && stashList.includes(preflight.stashMarker), "stash backup must be preserved");
307
+ assert.ok(
308
+ notifications.some((n) => n.level === "warning" && n.msg.includes("preserving")),
309
+ "user must be warned that the stash was preserved as backup",
310
+ );
311
+ } finally {
312
+ try { rmSync(repo, { recursive: true, force: true, maxRetries: 3, retryDelay: 100 }); } catch { /* ignore */ }
313
+ }
314
+ });
315
+
316
+ test("postflightPopStash drops stash when untracked collision is identical to merged file", () => {
317
+ const repo = createTempRepo();
318
+ try {
319
+ writeFileSync(join(repo, "tests.txt"), "same test content\n");
320
+ const preflight = preflightCleanRoot(repo, "M010", () => {});
321
+ assert.equal(preflight.stashPushed, true, "preflight must stash untracked file");
322
+
323
+ writeFileSync(join(repo, "tests.txt"), "same test content\n");
324
+ run("git add tests.txt", repo);
325
+ run('git commit -m "feat: add same test"', repo);
326
+
327
+ const postflight = postflightPopStash(repo, "M010", preflight.stashMarker, () => {});
328
+
329
+ assert.equal(postflight.needsManualRecovery, false, "identical already-present files must not stop auto-mode");
330
+ assert.equal(postflight.resolution, "already-present-dropped");
331
+ assert.equal(readFileSync(join(repo, "tests.txt"), "utf-8"), "same test content\n");
332
+
333
+ const stashList = run("git stash list", repo);
334
+ assert.ok(!stashList.includes(preflight.stashMarker ?? ""), "identical stash must be dropped");
335
+ } finally {
336
+ try { rmSync(repo, { recursive: true, force: true, maxRetries: 3, retryDelay: 100 }); } catch { /* ignore */ }
337
+ }
338
+ });
339
+
340
+ test("postflightPopStash requires manual recovery for mixed tracked changes and untracked collision", () => {
341
+ const repo = createTempRepo();
342
+ try {
343
+ writeFileSync(join(repo, "README.md"), "# local tracked work\n");
344
+ writeFileSync(join(repo, "tests.txt"), "local untracked work\n");
345
+ const preflight = preflightCleanRoot(repo, "M011", () => {});
346
+ assert.equal(preflight.stashPushed, true, "preflight must stash mixed changes");
347
+
348
+ writeFileSync(join(repo, "tests.txt"), "merged milestone test\n");
349
+ run("git add tests.txt", repo);
350
+ run('git commit -m "feat: add merged test"', repo);
351
+
352
+ const postflight = postflightPopStash(repo, "M011", preflight.stashMarker, () => {});
353
+
354
+ assert.equal(postflight.needsManualRecovery, true, "tracked stash payload must still require manual recovery");
355
+ assert.equal(postflight.resolution, "manual-recovery");
356
+ const stashList = run("git stash list", repo);
357
+ assert.ok(preflight.stashMarker && stashList.includes(preflight.stashMarker), "mixed stash must be preserved");
358
+ } finally {
359
+ try { rmSync(repo, { recursive: true, force: true, maxRetries: 3, retryDelay: 100 }); } catch { /* ignore */ }
360
+ }
361
+ });
362
+
363
+ test("postflightPopStash requires manual recovery when an untracked stash path is missing after collision", () => {
364
+ const repo = createTempRepo();
365
+ try {
366
+ writeFileSync(join(repo, "tests.txt"), "local untracked work\n");
367
+ writeFileSync(join(repo, "other-tests.txt"), "other local untracked work\n");
368
+ const preflight = preflightCleanRoot(repo, "M012", () => {});
369
+ assert.equal(preflight.stashPushed, true, "preflight must stash untracked files");
370
+
371
+ writeFileSync(join(repo, "tests.txt"), "merged milestone test\n");
372
+ run("git add tests.txt", repo);
373
+ run('git commit -m "feat: add one merged test"', repo);
374
+
375
+ const postflight = postflightPopStash(repo, "M012", preflight.stashMarker, () => {});
376
+
377
+ assert.equal(postflight.needsManualRecovery, true, "partial untracked restores must still require manual recovery");
378
+ assert.equal(postflight.resolution, "manual-recovery");
379
+ assert.equal(existsSync(join(repo, "other-tests.txt")), true, "git may partially restore the non-colliding path");
380
+ assert.match(run("git status --porcelain", repo), /\?\? other-tests\.txt/, "partial restore must leave manual recovery visible");
381
+ const stashList = run("git stash list", repo);
382
+ assert.ok(preflight.stashMarker && stashList.includes(preflight.stashMarker), "stash must remain for manual recovery");
383
+ } finally {
384
+ try { rmSync(repo, { recursive: true, force: true, maxRetries: 3, retryDelay: 100 }); } catch { /* ignore */ }
385
+ }
386
+ });
@@ -17,6 +17,15 @@ import {
17
17
  setPendingAutoStart,
18
18
  } from "../guided-flow.ts";
19
19
 
20
+ function pendingInput(basePath: string, milestoneId: string) {
21
+ return {
22
+ basePath,
23
+ milestoneId,
24
+ ctx: { ui: { notify: () => undefined } } as any,
25
+ pi: { sendMessage: () => undefined } as any,
26
+ };
27
+ }
28
+
20
29
  afterEach(() => {
21
30
  clearPendingAutoStart();
22
31
  });
@@ -28,7 +37,7 @@ describe("clear stale pending auto-start (#3667)", () => {
28
37
  mkdirSync(join(base, ".gsd"), { recursive: true });
29
38
  const before = Date.now();
30
39
 
31
- setPendingAutoStart(base, { basePath: base, milestoneId: "M001" });
40
+ setPendingAutoStart(base, pendingInput(base, "M001"));
32
41
 
33
42
  const entry = _getPendingAutoStart(base);
34
43
  assert.ok(entry);
@@ -41,7 +50,7 @@ describe("clear stale pending auto-start (#3667)", () => {
41
50
  t.after(() => rmSync(base, { recursive: true, force: true }));
42
51
  mkdirSync(join(base, ".gsd"), { recursive: true });
43
52
 
44
- setPendingAutoStart(base, { basePath: base, milestoneId: "M001", createdAt: 123 });
53
+ setPendingAutoStart(base, { ...pendingInput(base, "M001"), createdAt: 123 });
45
54
 
46
55
  assert.equal(_getPendingAutoStart(base)?.createdAt, 123);
47
56
  });
@@ -0,0 +1,16 @@
1
+ // Project/App: GSD-2
2
+ // File Purpose: Tests closeout git action deferral policy for auto-mode units.
3
+
4
+ import test from "node:test";
5
+ import assert from "node:assert/strict";
6
+
7
+ import { shouldDeferCloseoutGitAction } from "../auto-post-unit.ts";
8
+
9
+ test("execute-task defers closeout git action until verification passes", () => {
10
+ assert.equal(shouldDeferCloseoutGitAction("execute-task"), true);
11
+ });
12
+
13
+ test("non execute-task units keep pre-verification closeout git action", () => {
14
+ assert.equal(shouldDeferCloseoutGitAction("plan-slice"), false);
15
+ assert.equal(shouldDeferCloseoutGitAction("complete-slice"), false);
16
+ });
@@ -645,7 +645,7 @@ describe("complete-milestone", () => {
645
645
  assert.strictEqual(sanitized.triggerReason, undefined);
646
646
  });
647
647
 
648
- test("rendered SUMMARY.md uses placeholder text for empty enrichment fields", async () => {
648
+ test("rendered SUMMARY.md uses empty frontmatter lists for empty key fields", async () => {
649
649
  const { handleCompleteMilestone } = await import("../tools/complete-milestone.ts");
650
650
  const base = createFixtureBase();
651
651
  const mid = "M001";
@@ -678,6 +678,9 @@ describe("complete-milestone", () => {
678
678
  assert.match(summary, /## Success Criteria Results\n\nNot provided\./);
679
679
  assert.match(summary, /## Definition of Done Results\n\nNot provided\./);
680
680
  assert.match(summary, /## Requirement Outcomes\n\nNot provided\./);
681
+ assert.match(summary, /key_decisions:\s*\[\]/);
682
+ assert.match(summary, /key_files:\s*\[\]/);
683
+ assert.doesNotMatch(summary, /key_(?:decisions|files):\n - \(none\)/);
681
684
  assert.match(summary, /## Deviations\n\nNone\./);
682
685
  assert.match(summary, /## Follow-ups\n\nNone\./);
683
686
  } finally {
@@ -408,10 +408,10 @@ console.log('\n=== complete-slice: handler with missing roadmap ===');
408
408
  }
409
409
 
410
410
  // ═══════════════════════════════════════════════════════════════════════════
411
- // complete-slice: step 13 specifies write tool for PROJECT.md (#2946)
411
+ // complete-slice: PROJECT refresh uses DB-backed artifact tool.
412
412
  // ═══════════════════════════════════════════════════════════════════════════
413
413
 
414
- console.log('\n=== complete-slice: step 13 specifies write tool for PROJECT.md (#2946) ===');
414
+ console.log('\n=== complete-slice: PROJECT refresh uses gsd_summary_save ===');
415
415
  {
416
416
  const promptPath = path.join(
417
417
  path.dirname(new URL(import.meta.url).pathname),
@@ -419,13 +419,9 @@ console.log('\n=== complete-slice: step 13 specifies write tool for PROJECT.md (
419
419
  );
420
420
  const prompt = fs.readFileSync(promptPath, 'utf-8');
421
421
 
422
- // Step 13 must explicitly name the `write` tool so the LLM doesn't
423
- // confuse it with `edit` (which requires path + oldText + newText).
424
- // See: https://github.com/gsd-build/gsd-2/issues/2946
425
- const mentionsWriteTool =
426
- /PROJECT\.md.*\bwrite\b/i.test(prompt) ||
427
- /\bwrite\b.*PROJECT\.md/i.test(prompt);
428
- assertTrue(mentionsWriteTool, 'step 13 must name the `write` tool when updating PROJECT.md');
422
+ assertTrue(prompt.includes('gsd_summary_save'), 'PROJECT refresh must use gsd_summary_save');
423
+ assertTrue(prompt.includes('artifact_type: "PROJECT"'), 'PROJECT refresh must use artifact_type PROJECT');
424
+ assertTrue(!/with a full `write`/i.test(prompt), 'prompt must not instruct direct PROJECT.md writes');
429
425
  }
430
426
 
431
427
  // ═══════════════════════════════════════════════════════════════════════════