gsd-pi 2.82.0-dev.2841a1e44 → 2.82.0-dev.3a3c6509d

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 (377) hide show
  1. package/README.md +3 -3
  2. package/dist/resources/.managed-resources-content-hash +1 -1
  3. package/dist/resources/GSD-WORKFLOW.md +7 -0
  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/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 +11 -0
  9. package/dist/resources/extensions/gsd/auto/phases.js +81 -31
  10. package/dist/resources/extensions/gsd/auto/workflow-memory-pressure.js +12 -0
  11. package/dist/resources/extensions/gsd/auto-dashboard.js +66 -1
  12. package/dist/resources/extensions/gsd/auto-direct-dispatch.js +1 -0
  13. package/dist/resources/extensions/gsd/auto-dispatch.js +20 -19
  14. package/dist/resources/extensions/gsd/auto-model-selection.js +2 -0
  15. package/dist/resources/extensions/gsd/auto-post-unit.js +71 -10
  16. package/dist/resources/extensions/gsd/auto-recovery.js +71 -14
  17. package/dist/resources/extensions/gsd/auto-start.js +87 -14
  18. package/dist/resources/extensions/gsd/auto-verification.js +17 -4
  19. package/dist/resources/extensions/gsd/auto-worktree.js +176 -10
  20. package/dist/resources/extensions/gsd/auto.js +37 -5
  21. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +31 -7
  22. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +10 -9
  23. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +4 -2
  24. package/dist/resources/extensions/gsd/bootstrap/subagent-input.js +5 -2
  25. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +14 -2
  26. package/dist/resources/extensions/gsd/commands/handlers/core.js +17 -1
  27. package/dist/resources/extensions/gsd/commands-prefs-wizard.js +7 -2
  28. package/dist/resources/extensions/gsd/crash-recovery.js +43 -5
  29. package/dist/resources/extensions/gsd/db/milestone-leases.js +24 -0
  30. package/dist/resources/extensions/gsd/db/unit-dispatches.js +3 -2
  31. package/dist/resources/extensions/gsd/dispatch-guard.js +2 -2
  32. package/dist/resources/extensions/gsd/doctor-git-checks.js +46 -1
  33. package/dist/resources/extensions/gsd/doctor-runtime-checks.js +28 -11
  34. package/dist/resources/extensions/gsd/doctor.js +2 -28
  35. package/dist/resources/extensions/gsd/export-html.js +27 -425
  36. package/dist/resources/extensions/gsd/forensics.js +3 -3
  37. package/dist/resources/extensions/gsd/git-service.js +45 -3
  38. package/dist/resources/extensions/gsd/gsd-db.js +21 -6
  39. package/dist/resources/extensions/gsd/guided-flow-queue.js +4 -3
  40. package/dist/resources/extensions/gsd/guided-flow.js +101 -116
  41. package/dist/resources/extensions/gsd/guided-unit-context.js +23 -0
  42. package/dist/resources/extensions/gsd/migrate/parsers.js +10 -0
  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/pending-auto-start.js +52 -0
  47. package/dist/resources/extensions/gsd/post-execution-checks.js +73 -2
  48. package/dist/resources/extensions/gsd/pre-execution-checks.js +28 -1
  49. package/dist/resources/extensions/gsd/prompt-loader.js +1 -1
  50. package/dist/resources/extensions/gsd/prompts/complete-milestone.md +1 -1
  51. package/dist/resources/extensions/gsd/prompts/complete-slice.md +1 -1
  52. package/dist/resources/extensions/gsd/prompts/discuss-headless.md +8 -8
  53. package/dist/resources/extensions/gsd/prompts/discuss.md +9 -9
  54. package/dist/resources/extensions/gsd/prompts/guided-discuss-project.md +4 -4
  55. package/dist/resources/extensions/gsd/prompts/guided-discuss-requirements.md +3 -3
  56. package/dist/resources/extensions/gsd/prompts/plan-slice.md +4 -4
  57. package/dist/resources/extensions/gsd/prompts/queue.md +4 -4
  58. package/dist/resources/extensions/gsd/prompts/refine-slice.md +2 -2
  59. package/dist/resources/extensions/gsd/prompts/rewrite-docs.md +1 -1
  60. package/dist/resources/extensions/gsd/queue-reorder-ui.js +30 -13
  61. package/dist/resources/extensions/gsd/smart-entry-routing.js +36 -0
  62. package/dist/resources/extensions/gsd/state-reconciliation/drift/merge-state.js +6 -1
  63. package/dist/resources/extensions/gsd/state-reconciliation/drift/project-md.js +9 -14
  64. package/dist/resources/extensions/gsd/state-reconciliation/drift/roadmap.js +19 -24
  65. package/dist/resources/extensions/gsd/state.js +1 -1
  66. package/dist/resources/extensions/gsd/status-guards.js +11 -0
  67. package/dist/resources/extensions/gsd/templates/plan.md +8 -5
  68. package/dist/resources/extensions/gsd/templates/task-plan.md +4 -2
  69. package/dist/resources/extensions/gsd/tools/complete-milestone.js +6 -8
  70. package/dist/resources/extensions/gsd/tools/complete-slice.js +6 -8
  71. package/dist/resources/extensions/gsd/tools/plan-milestone.js +7 -1
  72. package/dist/resources/extensions/gsd/tools/plan-slice.js +89 -14
  73. package/dist/resources/extensions/gsd/unit-context-manifest.js +7 -8
  74. package/dist/resources/extensions/gsd/validation.js +23 -1
  75. package/dist/resources/extensions/gsd/verification-gate.js +68 -7
  76. package/dist/resources/extensions/gsd/workflow-mcp.js +17 -1
  77. package/dist/resources/extensions/gsd/workflow-projections.js +6 -8
  78. package/dist/resources/extensions/gsd/worktree-lifecycle.js +33 -8
  79. package/dist/resources/extensions/gsd/worktree-manager.js +1 -1
  80. package/dist/resources/extensions/shared/html-shell.js +388 -0
  81. package/dist/resources/extensions/visual-brief/page-contract.js +2 -0
  82. package/dist/resources/extensions/visual-brief/prompts.js +29 -0
  83. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  84. package/dist/web/standalone/.next/BUILD_ID +1 -1
  85. package/dist/web/standalone/.next/app-path-routes-manifest.json +11 -11
  86. package/dist/web/standalone/.next/build-manifest.json +3 -3
  87. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  88. package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  89. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  90. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  91. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  92. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  93. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  94. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  95. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  96. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  97. package/dist/web/standalone/.next/server/app/_not-found/page.js +2 -2
  98. package/dist/web/standalone/.next/server/app/_not-found/page.js.nft.json +1 -1
  99. package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  100. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  101. package/dist/web/standalone/.next/server/app/_not-found.rsc +4 -7
  102. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +4 -7
  103. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  104. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +4 -5
  105. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  106. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  107. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +2 -5
  108. package/dist/web/standalone/.next/server/app/api/browse-directories/route.js +1 -1
  109. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  110. package/dist/web/standalone/.next/server/app/index.html +1 -1
  111. package/dist/web/standalone/.next/server/app/index.rsc +4 -7
  112. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  113. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +4 -7
  114. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  115. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +4 -5
  116. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +2 -5
  117. package/dist/web/standalone/.next/server/app/page.js +2 -2
  118. package/dist/web/standalone/.next/server/app/page.js.nft.json +1 -1
  119. package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
  120. package/dist/web/standalone/.next/server/app-paths-manifest.json +11 -11
  121. package/dist/web/standalone/.next/server/chunks/4266.js +2 -0
  122. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  123. package/dist/web/standalone/.next/server/next-font-manifest.js +1 -1
  124. package/dist/web/standalone/.next/server/next-font-manifest.json +1 -1
  125. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  126. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  127. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  128. package/dist/web/standalone/.next/static/chunks/app/layout-8c10ec293ae0f1d5.js +1 -0
  129. package/dist/web/standalone/.next/static/chunks/{webpack-6a95bc41e0f7ec89.js → webpack-9a4db269f9ed63ad.js} +1 -1
  130. package/dist/web/standalone/.next/static/css/746ee28c929d1880.css +1 -0
  131. package/package.json +2 -2
  132. package/packages/mcp-server/src/workflow-tools.test.ts +1 -1
  133. package/packages/native/tsconfig.json +2 -1
  134. package/packages/native/tsconfig.tsbuildinfo +1 -1
  135. package/packages/pi-ai/dist/providers/google-gemini-cli.d.ts.map +1 -1
  136. package/packages/pi-ai/dist/providers/google-gemini-cli.js +5 -0
  137. package/packages/pi-ai/dist/providers/google-gemini-cli.js.map +1 -1
  138. package/packages/pi-ai/dist/providers/google-gemini-cli.test.d.ts +2 -0
  139. package/packages/pi-ai/dist/providers/google-gemini-cli.test.d.ts.map +1 -0
  140. package/packages/pi-ai/dist/providers/google-gemini-cli.test.js +41 -0
  141. package/packages/pi-ai/dist/providers/google-gemini-cli.test.js.map +1 -0
  142. package/packages/pi-ai/dist/providers/openai-codex-responses.d.ts.map +1 -1
  143. package/packages/pi-ai/dist/providers/openai-codex-responses.js +82 -1
  144. package/packages/pi-ai/dist/providers/openai-codex-responses.js.map +1 -1
  145. package/packages/pi-ai/dist/providers/openai-codex-responses.test.d.ts +2 -0
  146. package/packages/pi-ai/dist/providers/openai-codex-responses.test.d.ts.map +1 -0
  147. package/packages/pi-ai/dist/providers/openai-codex-responses.test.js +52 -0
  148. package/packages/pi-ai/dist/providers/openai-codex-responses.test.js.map +1 -0
  149. package/packages/pi-ai/dist/providers/simple-options.d.ts +2 -4
  150. package/packages/pi-ai/dist/providers/simple-options.d.ts.map +1 -1
  151. package/packages/pi-ai/dist/providers/simple-options.js +5 -6
  152. package/packages/pi-ai/dist/providers/simple-options.js.map +1 -1
  153. package/packages/pi-ai/dist/providers/simple-options.test.d.ts +2 -0
  154. package/packages/pi-ai/dist/providers/simple-options.test.d.ts.map +1 -0
  155. package/packages/pi-ai/dist/providers/simple-options.test.js +50 -0
  156. package/packages/pi-ai/dist/providers/simple-options.test.js.map +1 -0
  157. package/packages/pi-ai/src/providers/google-gemini-cli.test.ts +49 -0
  158. package/packages/pi-ai/src/providers/google-gemini-cli.ts +7 -0
  159. package/packages/pi-ai/src/providers/openai-codex-responses.test.ts +63 -0
  160. package/packages/pi-ai/src/providers/openai-codex-responses.ts +91 -1
  161. package/packages/pi-ai/src/providers/simple-options.test.ts +60 -0
  162. package/packages/pi-ai/src/providers/simple-options.ts +5 -6
  163. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
  164. package/packages/pi-coding-agent/dist/core/agent-session-thinking-level.test.d.ts +2 -0
  165. package/packages/pi-coding-agent/dist/core/agent-session-thinking-level.test.d.ts.map +1 -0
  166. package/packages/pi-coding-agent/dist/core/agent-session-thinking-level.test.js +66 -0
  167. package/packages/pi-coding-agent/dist/core/agent-session-thinking-level.test.js.map +1 -0
  168. package/packages/pi-coding-agent/dist/core/agent-session.js +1 -1
  169. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  170. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js +44 -3
  171. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js.map +1 -1
  172. package/packages/pi-coding-agent/dist/core/sdk.js +1 -1
  173. package/packages/pi-coding-agent/dist/core/sdk.js.map +1 -1
  174. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.d.ts.map +1 -1
  175. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js +24 -6
  176. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js.map +1 -1
  177. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  178. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +71 -97
  179. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  180. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js +12 -0
  181. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js.map +1 -1
  182. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  183. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +19 -8
  184. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  185. package/packages/pi-coding-agent/src/core/agent-session-thinking-level.test.ts +79 -0
  186. package/packages/pi-coding-agent/src/core/agent-session.ts +1 -1
  187. package/packages/pi-coding-agent/src/core/chat-controller-ordering.test.ts +53 -3
  188. package/packages/pi-coding-agent/src/core/sdk.ts +1 -1
  189. package/packages/pi-coding-agent/src/modes/interactive/components/footer.ts +23 -7
  190. package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +75 -102
  191. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode-ordering.test.ts +14 -0
  192. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +23 -8
  193. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
  194. package/packages/pi-tui/dist/__tests__/terminal.test.d.ts +2 -0
  195. package/packages/pi-tui/dist/__tests__/terminal.test.d.ts.map +1 -0
  196. package/packages/pi-tui/dist/__tests__/terminal.test.js +103 -0
  197. package/packages/pi-tui/dist/__tests__/terminal.test.js.map +1 -0
  198. package/packages/pi-tui/dist/terminal.d.ts +2 -0
  199. package/packages/pi-tui/dist/terminal.d.ts.map +1 -1
  200. package/packages/pi-tui/dist/terminal.js +12 -0
  201. package/packages/pi-tui/dist/terminal.js.map +1 -1
  202. package/packages/pi-tui/src/__tests__/terminal.test.ts +121 -0
  203. package/packages/pi-tui/src/terminal.ts +11 -0
  204. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
  205. package/src/resources/GSD-WORKFLOW.md +7 -0
  206. package/src/resources/extensions/claude-code-cli/partial-builder.ts +2 -1
  207. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +1 -1
  208. package/src/resources/extensions/claude-code-cli/tests/partial-builder.test.ts +19 -2
  209. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +9 -0
  210. package/src/resources/extensions/gsd/auto/contracts.ts +14 -6
  211. package/src/resources/extensions/gsd/auto/infra-errors.ts +9 -3
  212. package/src/resources/extensions/gsd/auto/loop.ts +8 -5
  213. package/src/resources/extensions/gsd/auto/orchestrator.ts +11 -0
  214. package/src/resources/extensions/gsd/auto/phases.ts +90 -38
  215. package/src/resources/extensions/gsd/auto/workflow-memory-pressure.ts +13 -0
  216. package/src/resources/extensions/gsd/auto-dashboard.ts +72 -1
  217. package/src/resources/extensions/gsd/auto-direct-dispatch.ts +1 -0
  218. package/src/resources/extensions/gsd/auto-dispatch.ts +21 -19
  219. package/src/resources/extensions/gsd/auto-model-selection.ts +2 -1
  220. package/src/resources/extensions/gsd/auto-post-unit.ts +78 -8
  221. package/src/resources/extensions/gsd/auto-recovery.ts +74 -11
  222. package/src/resources/extensions/gsd/auto-start.ts +94 -12
  223. package/src/resources/extensions/gsd/auto-verification.ts +22 -2
  224. package/src/resources/extensions/gsd/auto-worktree.ts +193 -10
  225. package/src/resources/extensions/gsd/auto.ts +40 -5
  226. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +42 -7
  227. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +10 -9
  228. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +4 -2
  229. package/src/resources/extensions/gsd/bootstrap/subagent-input.ts +3 -1
  230. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +17 -2
  231. package/src/resources/extensions/gsd/commands/handlers/core.ts +17 -1
  232. package/src/resources/extensions/gsd/commands-prefs-wizard.ts +8 -3
  233. package/src/resources/extensions/gsd/crash-recovery.ts +44 -4
  234. package/src/resources/extensions/gsd/db/milestone-leases.ts +26 -0
  235. package/src/resources/extensions/gsd/db/unit-dispatches.ts +4 -3
  236. package/src/resources/extensions/gsd/dispatch-guard.ts +2 -2
  237. package/src/resources/extensions/gsd/doctor-git-checks.ts +45 -1
  238. package/src/resources/extensions/gsd/doctor-runtime-checks.ts +25 -13
  239. package/src/resources/extensions/gsd/doctor-types.ts +1 -0
  240. package/src/resources/extensions/gsd/doctor.ts +2 -27
  241. package/src/resources/extensions/gsd/export-html.ts +27 -427
  242. package/src/resources/extensions/gsd/forensics.ts +3 -3
  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/state.ts +1 -1
  272. package/src/resources/extensions/gsd/status-guards.ts +13 -0
  273. package/src/resources/extensions/gsd/templates/plan.md +8 -5
  274. package/src/resources/extensions/gsd/templates/task-plan.md +4 -2
  275. package/src/resources/extensions/gsd/tests/auto-dashboard.test.ts +71 -0
  276. package/src/resources/extensions/gsd/tests/auto-deterministic-error-classification-4973.test.ts +116 -0
  277. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +56 -0
  278. package/src/resources/extensions/gsd/tests/auto-orchestrator.test.ts +80 -1
  279. package/src/resources/extensions/gsd/tests/auto-paused-ui-cleanup.test.ts +35 -7
  280. package/src/resources/extensions/gsd/tests/auto-phases-lifecycle.test.ts +53 -2
  281. package/src/resources/extensions/gsd/tests/auto-post-unit-step-message.test.ts +12 -1
  282. package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +91 -6
  283. package/src/resources/extensions/gsd/tests/auto-start-orphan-bootstrap.test.ts +1 -0
  284. package/src/resources/extensions/gsd/tests/auto-stop-notification.test.ts +20 -0
  285. package/src/resources/extensions/gsd/tests/auto-worktree-registry.test.ts +69 -1
  286. package/src/resources/extensions/gsd/tests/checkout-branch-stash-guard.test.ts +87 -0
  287. package/src/resources/extensions/gsd/tests/clear-stale-autostart.test.ts +11 -2
  288. package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +4 -1
  289. package/src/resources/extensions/gsd/tests/complete-slice.test.ts +5 -9
  290. package/src/resources/extensions/gsd/tests/complete-task.test.ts +3 -1
  291. package/src/resources/extensions/gsd/tests/crash-recovery-via-db.test.ts +86 -2
  292. package/src/resources/extensions/gsd/tests/custom-engine-loop-integration.test.ts +2 -0
  293. package/src/resources/extensions/gsd/tests/db-authority-regression.test.ts +208 -0
  294. package/src/resources/extensions/gsd/tests/deep-project-auto-loop.test.ts +59 -2
  295. package/src/resources/extensions/gsd/tests/dispatch-complete-milestone-guard.test.ts +66 -0
  296. package/src/resources/extensions/gsd/tests/dispatch-guard.test.ts +27 -0
  297. package/src/resources/extensions/gsd/tests/doctor-empty-worktree.test.ts +65 -0
  298. package/src/resources/extensions/gsd/tests/export-html-enhancements.test.ts +8 -0
  299. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +11 -0
  300. package/src/resources/extensions/gsd/tests/guided-discuss-project-prompt-rendering.test.ts +2 -0
  301. package/src/resources/extensions/gsd/tests/guided-dispatch-root.test.ts +106 -0
  302. package/src/resources/extensions/gsd/tests/guided-flow-session-isolation.test.ts +59 -11
  303. package/src/resources/extensions/gsd/tests/guided-flow.test.ts +21 -0
  304. package/src/resources/extensions/gsd/tests/guided-tool-contract.test.ts +65 -0
  305. package/src/resources/extensions/gsd/tests/headless-milestone-parity.test.ts +7 -7
  306. package/src/resources/extensions/gsd/tests/hook-model-resolution.test.ts +5 -0
  307. package/src/resources/extensions/gsd/tests/infra-error.test.ts +2 -2
  308. package/src/resources/extensions/gsd/tests/infra-errors-cooldown.test.ts +9 -0
  309. package/src/resources/extensions/gsd/tests/integration/doctor-runtime.test.ts +20 -0
  310. package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +112 -1
  311. package/src/resources/extensions/gsd/tests/integration/state-machine-runtime-failures.test.ts +6 -1
  312. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +46 -0
  313. package/src/resources/extensions/gsd/tests/merge-db-cycle.test.ts +179 -0
  314. package/src/resources/extensions/gsd/tests/migrate-validator-parsers.test.ts +24 -1
  315. package/src/resources/extensions/gsd/tests/migration-auto-check.test.ts +26 -18
  316. package/src/resources/extensions/gsd/tests/native-git-bridge-exec-fallback.test.ts +63 -2
  317. package/src/resources/extensions/gsd/tests/orphaned-worktree-audit.test.ts +121 -1
  318. package/src/resources/extensions/gsd/tests/park-db-sync.test.ts +55 -1
  319. package/src/resources/extensions/gsd/tests/pending-autostart-scope.test.ts +29 -5
  320. package/src/resources/extensions/gsd/tests/pipeline-variant-dispatch.test.ts +2 -1
  321. package/src/resources/extensions/gsd/tests/plan-milestone.test.ts +26 -0
  322. package/src/resources/extensions/gsd/tests/plan-slice-prompt.test.ts +2 -0
  323. package/src/resources/extensions/gsd/tests/plan-slice.test.ts +225 -1
  324. package/src/resources/extensions/gsd/tests/plan-task.test.ts +17 -0
  325. package/src/resources/extensions/gsd/tests/post-execution-checks.test.ts +86 -0
  326. package/src/resources/extensions/gsd/tests/post-unit-git-failure.test.ts +1 -1
  327. package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +53 -0
  328. package/src/resources/extensions/gsd/tests/prefs-wizard-coverage.test.ts +59 -0
  329. package/src/resources/extensions/gsd/tests/prompt-loader.test.ts +23 -0
  330. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +37 -1
  331. package/src/resources/extensions/gsd/tests/queue-reorder-ui.test.ts +54 -0
  332. package/src/resources/extensions/gsd/tests/remediation-completion-guard.test.ts +89 -2
  333. package/src/resources/extensions/gsd/tests/run-uat-replay-cap.test.ts +2 -3
  334. package/src/resources/extensions/gsd/tests/session-switch-abort-misclassification.test.ts +10 -0
  335. package/src/resources/extensions/gsd/tests/smart-entry-routing.test.ts +113 -0
  336. package/src/resources/extensions/gsd/tests/start-auto-detached.test.ts +53 -2
  337. package/src/resources/extensions/gsd/tests/state-reconciliation-drift.test.ts +119 -23
  338. package/src/resources/extensions/gsd/tests/status-guards.test.ts +13 -1
  339. package/src/resources/extensions/gsd/tests/stuck-state-via-db.test.ts +64 -1
  340. package/src/resources/extensions/gsd/tests/summary-render-parity.test.ts +7 -3
  341. package/src/resources/extensions/gsd/tests/unit-context-manifest.test.ts +82 -7
  342. package/src/resources/extensions/gsd/tests/validate-milestone-stuck-guard.test.ts +29 -2
  343. package/src/resources/extensions/gsd/tests/verification-gate.test.ts +110 -1
  344. package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +19 -1
  345. package/src/resources/extensions/gsd/tests/workflow-memory-pressure.test.ts +21 -1
  346. package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +1 -1
  347. package/src/resources/extensions/gsd/tests/worktree-git-pathspec.test.ts +39 -0
  348. package/src/resources/extensions/gsd/tests/worktree-journal-events.test.ts +64 -12
  349. package/src/resources/extensions/gsd/tests/write-gate-planning-unit.test.ts +38 -0
  350. package/src/resources/extensions/gsd/tools/complete-milestone.ts +8 -10
  351. package/src/resources/extensions/gsd/tools/complete-slice.ts +6 -8
  352. package/src/resources/extensions/gsd/tools/plan-milestone.ts +5 -1
  353. package/src/resources/extensions/gsd/tools/plan-slice.ts +98 -12
  354. package/src/resources/extensions/gsd/types.ts +1 -1
  355. package/src/resources/extensions/gsd/unit-context-manifest.ts +12 -9
  356. package/src/resources/extensions/gsd/validation.ts +23 -1
  357. package/src/resources/extensions/gsd/verification-gate.ts +78 -6
  358. package/src/resources/extensions/gsd/workflow-mcp.ts +18 -1
  359. package/src/resources/extensions/gsd/workflow-projections.ts +6 -8
  360. package/src/resources/extensions/gsd/worktree-lifecycle.ts +41 -8
  361. package/src/resources/extensions/gsd/worktree-manager.ts +1 -1
  362. package/src/resources/extensions/shared/html-shell.ts +412 -0
  363. package/src/resources/extensions/visual-brief/page-contract.ts +2 -0
  364. package/src/resources/extensions/visual-brief/prompts.ts +37 -1
  365. package/src/resources/extensions/visual-brief/tests/visual-brief.test.ts +40 -0
  366. package/dist/web/standalone/.next/server/chunks/5822.js +0 -2
  367. package/dist/web/standalone/.next/static/chunks/app/layout-a16c7a7ecdf0c2cf.js +0 -1
  368. package/dist/web/standalone/.next/static/css/0262768ec1b89d34.css +0 -1
  369. package/dist/web/standalone/.next/static/css/de70bee13400563f.css +0 -1
  370. package/dist/web/standalone/.next/static/media/4cf2300e9c8272f7-s.p.woff2 +0 -0
  371. package/dist/web/standalone/.next/static/media/747892c23ea88013-s.woff2 +0 -0
  372. package/dist/web/standalone/.next/static/media/8d697b304b401681-s.woff2 +0 -0
  373. package/dist/web/standalone/.next/static/media/93f479601ee12b01-s.p.woff2 +0 -0
  374. package/dist/web/standalone/.next/static/media/9610d9e46709d722-s.woff2 +0 -0
  375. package/dist/web/standalone/.next/static/media/ba015fad6dcf6784-s.woff2 +0 -0
  376. /package/dist/web/standalone/.next/static/{Qgr2B_MRhPxC0z8fwv4vT → O6femb9LLl3nlgsDaYwS-}/_buildManifest.js +0 -0
  377. /package/dist/web/standalone/.next/static/{Qgr2B_MRhPxC0z8fwv4vT → O6femb9LLl3nlgsDaYwS-}/_ssgManifest.js +0 -0
@@ -544,7 +544,7 @@ test("ADR-017 (#5703): live worker lock is not cleared", async (t) => {
544
544
 
545
545
  // ─── #5704: unregistered-milestone drift ────────────────────────────────────
546
546
 
547
- test("ADR-017 (#5704): unregistered-milestone drift detected and DB row inserted", async (t) => {
547
+ test("ADR-017 (#5704): unregistered-milestone drift fails closed without importing markdown", async (t) => {
548
548
  const base = mkdtempSync(join(tmpdir(), "gsd-adr017-projmd-"));
549
549
  const milestoneDir = join(base, ".gsd", "milestones", "M042");
550
550
  mkdirSync(milestoneDir, { recursive: true });
@@ -571,20 +571,22 @@ test("ADR-017 (#5704): unregistered-milestone drift detected and DB row inserted
571
571
  // Pre-condition: filesystem has the milestone, DB does NOT.
572
572
  assert.equal(getMilestone("M042"), null, "pre: DB has no row for M042");
573
573
 
574
- const result = await reconcileBeforeDispatch(base, {
575
- invalidateStateCache: () => {},
576
- deriveState: async () => makeState(),
577
- });
578
-
579
- assert.equal(result.ok, true);
580
- assert.ok(getMilestone("M042"), "post: DB row inserted for M042");
581
- const milestoneRepaired = result.repaired.find(
582
- (d) => d.kind === "unregistered-milestone",
574
+ await assert.rejects(
575
+ reconcileBeforeDispatch(base, {
576
+ invalidateStateCache: () => {},
577
+ deriveState: async () => makeState(),
578
+ }),
579
+ (err: unknown) => {
580
+ assert.ok(err instanceof ReconciliationFailedError);
581
+ assert.match(String(err.message), /unregistered-milestone/);
582
+ assert.equal(err.failures[0]?.drift.kind, "unregistered-milestone");
583
+ assert.match(String(err.failures[0]?.cause), /M042/);
584
+ assert.match(String(err.failures[0]?.cause), /markdown projection/);
585
+ assert.match(String(err.failures[0]?.cause), /recovery\/migration/);
586
+ return true;
587
+ },
583
588
  );
584
- assert.ok(milestoneRepaired, "repaired list should include the unregistered-milestone drift");
585
- if (milestoneRepaired?.kind === "unregistered-milestone") {
586
- assert.equal(milestoneRepaired.milestoneId, "M042");
587
- }
589
+ assert.equal(getMilestone("M042"), null, "post: DB still has no row for M042");
588
590
  });
589
591
 
590
592
  test("ADR-017 (#5704): registered milestone (DB row present) → no drift", async (t) => {
@@ -627,13 +629,14 @@ test("ADR-017 (#5704): registered milestone (DB row present) → no drift", asyn
627
629
 
628
630
  // ─── #5705: roadmap-divergence drift ─────────────────────────────────────────
629
631
 
630
- test("ADR-017 (#5705): roadmap-divergence drift detected and DB depends synced", async (t) => {
632
+ test("ADR-017 (#5705): roadmap-divergence re-renders projection without syncing depends into DB", async (t) => {
631
633
  const base = mkdtempSync(join(tmpdir(), "gsd-adr017-roadmap-"));
632
634
  const milestoneDir = join(base, ".gsd", "milestones", "M001");
635
+ const roadmapPath = join(milestoneDir, "M001-ROADMAP.md");
633
636
  mkdirSync(milestoneDir, { recursive: true });
634
637
  // ROADMAP.md declares S02 depends on [S01]
635
638
  writeFileSync(
636
- join(milestoneDir, "M001-ROADMAP.md"),
639
+ roadmapPath,
637
640
  [
638
641
  "# M001: Test",
639
642
  "",
@@ -665,7 +668,12 @@ test("ADR-017 (#5705): roadmap-divergence drift detected and DB depends synced",
665
668
  });
666
669
 
667
670
  assert.equal(result.ok, true);
668
- assert.deepEqual(getSlice("M001", "S02")?.depends, ["S01"], "post: DB depends matches ROADMAP.md");
671
+ assert.deepEqual(getSlice("M001", "S02")?.depends, [], "post: DB depends remains authoritative");
672
+ assert.match(
673
+ readFileSync(roadmapPath, "utf-8"),
674
+ /- \[ \] \*\*S02: Feature\*\* `risk:medium` `depends:\[\]`/,
675
+ "post: ROADMAP projection is regenerated from DB depends",
676
+ );
669
677
  const roadmapRepaired = result.repaired.find((d) => d.kind === "roadmap-divergence");
670
678
  assert.ok(roadmapRepaired, "repaired list should include the roadmap-divergence drift");
671
679
  if (roadmapRepaired?.kind === "roadmap-divergence") {
@@ -673,13 +681,14 @@ test("ADR-017 (#5705): roadmap-divergence drift detected and DB depends synced",
673
681
  }
674
682
  });
675
683
 
676
- test("ADR-017 (#5705): ROADMAP declares slice missing from DB slice inserted and drift reported", async (t) => {
684
+ test("ADR-017 (#5705): ROADMAP-only slice is removed from projection and not inserted into DB", async (t) => {
677
685
  const base = mkdtempSync(join(tmpdir(), "gsd-adr017-roadmap-newslice-"));
678
686
  const milestoneDir = join(base, ".gsd", "milestones", "M001");
687
+ const roadmapPath = join(milestoneDir, "M001-ROADMAP.md");
679
688
  mkdirSync(milestoneDir, { recursive: true });
680
689
  // ROADMAP.md declares S01 and S02; DB will only have S01.
681
690
  writeFileSync(
682
- join(milestoneDir, "M001-ROADMAP.md"),
691
+ roadmapPath,
683
692
  [
684
693
  "# M001: Test",
685
694
  "",
@@ -710,10 +719,10 @@ test("ADR-017 (#5705): ROADMAP declares slice missing from DB → slice inserted
710
719
  });
711
720
 
712
721
  assert.equal(result.ok, true);
713
- const s02 = getSlice("M001", "S02");
714
- assert.ok(s02, "post: S02 inserted into DB after repair");
715
- assert.equal(s02?.sequence, 2, "post: S02 sequence matches ROADMAP order");
716
- assert.deepEqual(s02?.depends, ["S01"], "post: S02 depends matches ROADMAP");
722
+ assert.equal(getSlice("M001", "S02"), null, "post: S02 still has no DB row");
723
+ const rendered = readFileSync(roadmapPath, "utf-8");
724
+ assert.match(rendered, /- \[ \] \*\*S01: Foundation\*\*/);
725
+ assert.doesNotMatch(rendered, /S02: Feature/, "post: ROADMAP-only S02 removed from projection");
717
726
  const roadmapRepaired = result.repaired.find((d) => d.kind === "roadmap-divergence");
718
727
  assert.ok(roadmapRepaired, "repaired list should include the roadmap-divergence drift");
719
728
  if (roadmapRepaired?.kind === "roadmap-divergence") {
@@ -721,6 +730,93 @@ test("ADR-017 (#5705): ROADMAP declares slice missing from DB → slice inserted
721
730
  }
722
731
  });
723
732
 
733
+ test("ADR-017 (#5705): ROADMAP sequence drift re-renders from DB order without mutating DB", async (t) => {
734
+ const base = mkdtempSync(join(tmpdir(), "gsd-adr017-roadmap-sequence-"));
735
+ const milestoneDir = join(base, ".gsd", "milestones", "M001");
736
+ const roadmapPath = join(milestoneDir, "M001-ROADMAP.md");
737
+ mkdirSync(milestoneDir, { recursive: true });
738
+ writeFileSync(
739
+ roadmapPath,
740
+ [
741
+ "# M001: Test",
742
+ "",
743
+ "**Vision:** Verify sequence drift",
744
+ "",
745
+ "## Slices",
746
+ "",
747
+ "- [ ] **S02: Feature** `risk:medium` `depends:[]`",
748
+ "- [ ] **S01: Foundation** `risk:medium` `depends:[]`",
749
+ "",
750
+ ].join("\n"),
751
+ );
752
+ t.after(() => {
753
+ try { closeDatabase(); } catch { /* noop */ }
754
+ rmSync(base, { recursive: true, force: true });
755
+ });
756
+
757
+ openDatabase(join(base, ".gsd", "gsd.db"));
758
+ insertMilestone({ id: "M001", title: "Test", status: "active" });
759
+ insertSlice({ id: "S01", milestoneId: "M001", title: "Foundation", status: "pending", risk: "medium", depends: [], demo: "", sequence: 1 });
760
+ insertSlice({ id: "S02", milestoneId: "M001", title: "Feature", status: "pending", risk: "medium", depends: [], demo: "", sequence: 2 });
761
+
762
+ const result = await reconcileBeforeDispatch(base, {
763
+ invalidateStateCache: () => {},
764
+ deriveState: async () => makeState(),
765
+ });
766
+
767
+ assert.equal(result.ok, true);
768
+ assert.equal(getSlice("M001", "S01")?.sequence, 1, "post: S01 DB sequence remains authoritative");
769
+ assert.equal(getSlice("M001", "S02")?.sequence, 2, "post: S02 DB sequence remains authoritative");
770
+ const rendered = readFileSync(roadmapPath, "utf-8");
771
+ assert.ok(
772
+ rendered.indexOf("S01: Foundation") < rendered.indexOf("S02: Feature"),
773
+ "post: ROADMAP projection follows DB sequence",
774
+ );
775
+ assert.ok(result.repaired.some((d) => d.kind === "roadmap-divergence"));
776
+ });
777
+
778
+ test("ADR-017 (#5705): ROADMAP checkbox drift re-renders from DB status without mutating DB", async (t) => {
779
+ const base = mkdtempSync(join(tmpdir(), "gsd-adr017-roadmap-checkbox-"));
780
+ const milestoneDir = join(base, ".gsd", "milestones", "M001");
781
+ const roadmapPath = join(milestoneDir, "M001-ROADMAP.md");
782
+ mkdirSync(milestoneDir, { recursive: true });
783
+ writeFileSync(
784
+ roadmapPath,
785
+ [
786
+ "# M001: Test",
787
+ "",
788
+ "**Vision:** Verify checkbox drift",
789
+ "",
790
+ "## Slices",
791
+ "",
792
+ "- [x] **S01: Foundation** `risk:medium` `depends:[]`",
793
+ "",
794
+ ].join("\n"),
795
+ );
796
+ t.after(() => {
797
+ try { closeDatabase(); } catch { /* noop */ }
798
+ rmSync(base, { recursive: true, force: true });
799
+ });
800
+
801
+ openDatabase(join(base, ".gsd", "gsd.db"));
802
+ insertMilestone({ id: "M001", title: "Test", status: "active" });
803
+ insertSlice({ id: "S01", milestoneId: "M001", title: "Foundation", status: "pending", risk: "medium", depends: [], demo: "", sequence: 1 });
804
+
805
+ const result = await reconcileBeforeDispatch(base, {
806
+ invalidateStateCache: () => {},
807
+ deriveState: async () => makeState(),
808
+ });
809
+
810
+ assert.equal(result.ok, true);
811
+ assert.equal(getSlice("M001", "S01")?.status, "pending", "post: DB status remains authoritative");
812
+ assert.match(
813
+ readFileSync(roadmapPath, "utf-8"),
814
+ /- \[ \] \*\*S01: Foundation\*\*/,
815
+ "post: ROADMAP checkbox reflects DB status",
816
+ );
817
+ assert.ok(result.repaired.some((d) => d.kind === "roadmap-divergence"));
818
+ });
819
+
724
820
  test("ADR-017 (#5705): in-sync ROADMAP and DB → no roadmap-divergence drift", async (t) => {
725
821
  const base = mkdtempSync(join(tmpdir(), "gsd-adr017-roadmap-clean-"));
726
822
  const milestoneDir = join(base, ".gsd", "milestones", "M001");
@@ -3,7 +3,7 @@
3
3
  import test from 'node:test';
4
4
  import assert from 'node:assert/strict';
5
5
 
6
- import { isClosedStatus } from '../status-guards.ts';
6
+ import { isClosedStatus, isFutureMilestoneStatus } from '../status-guards.ts';
7
7
 
8
8
  test('isClosedStatus: "complete" returns true', () => {
9
9
  assert.equal(isClosedStatus('complete'), true);
@@ -32,3 +32,15 @@ test('isClosedStatus: "active" returns false', () => {
32
32
  test('isClosedStatus: "" (empty string) returns false', () => {
33
33
  assert.equal(isClosedStatus(''), false);
34
34
  });
35
+
36
+ test('isFutureMilestoneStatus includes future milestone aliases', () => {
37
+ assert.equal(isFutureMilestoneStatus('pending'), true);
38
+ assert.equal(isFutureMilestoneStatus('queued'), true);
39
+ assert.equal(isFutureMilestoneStatus('planned'), true);
40
+ });
41
+
42
+ test('isFutureMilestoneStatus excludes active and closed milestones', () => {
43
+ assert.equal(isFutureMilestoneStatus('active'), false);
44
+ assert.equal(isFutureMilestoneStatus('complete'), false);
45
+ assert.equal(isFutureMilestoneStatus('parked'), false);
46
+ });
@@ -19,13 +19,17 @@ import {
19
19
  closeDatabase,
20
20
  insertMilestone,
21
21
  } from "../gsd-db.ts";
22
- import { registerAutoWorker } from "../db/auto-workers.ts";
22
+ import { registerAutoWorker, markWorkerCrashed } from "../db/auto-workers.ts";
23
23
  import { claimMilestoneLease } from "../db/milestone-leases.ts";
24
24
  import {
25
25
  recordDispatchClaim,
26
+ markFailed,
27
+ markCanceled,
26
28
  getRecentUnitKeysForWorker,
29
+ getRecentUnitKeysForProjectRoot,
27
30
  } from "../db/unit-dispatches.ts";
28
31
  import { setRuntimeKv, getRuntimeKv } from "../db/runtime-kv.ts";
32
+ import { detectStuck } from "../auto/detect-stuck.ts";
29
33
 
30
34
  function makeBase(): string {
31
35
  const base = mkdtempSync(join(tmpdir(), "gsd-stuck-state-db-"));
@@ -67,6 +71,65 @@ test("getRecentUnitKeysForWorker reconstructs the recentUnits sliding window", (
67
71
  assert.deepEqual(window.map(w => w.key), ["U1", "U2", "U3"]);
68
72
  });
69
73
 
74
+ test("getRecentUnitKeysForProjectRoot restores compound keys used by stuck detection", (t) => {
75
+ const base = makeBase();
76
+ t.after(() => cleanup(base));
77
+ openDatabase(join(base, ".gsd", "gsd.db"));
78
+ insertMilestone({ id: "M001", title: "T", status: "active" });
79
+ insertMilestone({ id: "M002", title: "Crashed", status: "active" });
80
+ const worker = registerAutoWorker({ projectRootRealpath: base });
81
+ const lease = claimMilestoneLease(worker, "M001");
82
+ assert.equal(lease.ok, true);
83
+ if (!lease.ok) return;
84
+
85
+ for (let i = 0; i < 2; i++) {
86
+ const claim = recordDispatchClaim({
87
+ traceId: `t${i}`,
88
+ workerId: worker,
89
+ milestoneLeaseToken: lease.token,
90
+ milestoneId: "M001",
91
+ sliceId: "S01",
92
+ unitType: "complete-slice",
93
+ unitId: "M001/S01",
94
+ });
95
+ assert.equal(claim.ok, true);
96
+ if (!claim.ok) return;
97
+ markCanceled(claim.dispatchId, "pause");
98
+ }
99
+
100
+ const crashedWorker = registerAutoWorker({ projectRootRealpath: base });
101
+ const crashedLease = claimMilestoneLease(crashedWorker, "M002");
102
+ assert.equal(crashedLease.ok, true);
103
+ if (!crashedLease.ok) return;
104
+
105
+ for (let i = 0; i < 3; i++) {
106
+ const claim = recordDispatchClaim({
107
+ traceId: `crashed-${i}`,
108
+ workerId: crashedWorker,
109
+ milestoneLeaseToken: crashedLease.token,
110
+ milestoneId: "M002",
111
+ sliceId: "S01",
112
+ taskId: "T01",
113
+ unitType: "execute-task",
114
+ unitId: "M002/S01/T01",
115
+ });
116
+ assert.equal(claim.ok, true);
117
+ if (!claim.ok) return;
118
+ markFailed(claim.dispatchId, { errorSummary: "worker crashed" });
119
+ }
120
+ markWorkerCrashed(crashedWorker);
121
+
122
+ const window = getRecentUnitKeysForProjectRoot(base, 3);
123
+ assert.deepEqual(window.map(w => w.key), [
124
+ "complete-slice/M001/S01",
125
+ "complete-slice/M001/S01",
126
+ ]);
127
+
128
+ const result = detectStuck([...window, { key: "complete-slice/M001/S01" }]);
129
+ assert.equal(result?.stuck, true);
130
+ assert.match(result?.reason ?? "", /3 consecutive times/);
131
+ });
132
+
70
133
  test("getRecentUnitKeysForWorker honors the limit parameter", (t) => {
71
134
  const base = makeBase();
72
135
  t.after(() => cleanup(base));
@@ -175,13 +175,17 @@ const verificationEvidence = [
175
175
  );
176
176
  }
177
177
 
178
- // Test 11: empty key_files renders YAML placeholder, not empty array
178
+ // Test 11: empty key_files renders an empty YAML list, not a sentinel path
179
179
  {
180
180
  const noFiles = { ...taskRow, key_files: [] };
181
181
  const output = renderSummaryContent(noFiles, SLICE_ID, MILESTONE_ID);
182
182
  assertTrue(
183
- output.includes("key_files:\n - (none)"),
184
- "empty key_files must render as YAML list with (none) placeholder",
183
+ output.includes("key_files: []"),
184
+ "empty key_files must render as an empty YAML list",
185
+ );
186
+ assertTrue(
187
+ !output.includes("key_files:\n - (none)"),
188
+ "empty key_files must not render (none) as a path-like list item",
185
189
  );
186
190
  }
187
191
 
@@ -14,7 +14,10 @@ import {
14
14
  type SkillsPolicy,
15
15
  type UnitContextManifest,
16
16
  } from "../unit-context-manifest.ts";
17
- import { ALLOWED_PLANNING_DISPATCH_AGENTS } from "../bootstrap/write-gate.ts";
17
+ import {
18
+ ALLOWED_PLANNING_DISPATCH_AGENTS,
19
+ shouldBlockPlanningUnit,
20
+ } from "../bootstrap/write-gate.ts";
18
21
  import {
19
22
  getRequiredWorkflowToolsForAutoUnit,
20
23
  getRequiredWorkflowToolsForGuidedUnit,
@@ -217,7 +220,7 @@ test("#4934: every manifest declares a tools policy", () => {
217
220
  });
218
221
 
219
222
  test("#4934: tools.mode is one of the declared policies", () => {
220
- const validModes = new Set(["all", "read-only", "planning", "planning-dispatch", "docs"]);
223
+ const validModes = new Set(["all", "read-only", "planning", "planning-dispatch", "docs", "verification"]);
221
224
  for (const [unitType, manifest] of Object.entries(UNIT_MANIFESTS)) {
222
225
  const mode = (manifest as { tools: { mode: string } }).tools.mode;
223
226
  assert.ok(
@@ -227,28 +230,100 @@ test("#4934: tools.mode is one of the declared policies", () => {
227
230
  }
228
231
  });
229
232
 
230
- test('#4934: only execute-task and reactive-execute may use tools.mode "all" (full source-tree write access)', () => {
231
- const allowedAllUnits = new Set(["execute-task", "reactive-execute"]);
233
+ test('#4934: only execution units and complete-milestone may use tools.mode "all"', () => {
234
+ const allowedAllUnits = new Set(["execute-task", "reactive-execute", "complete-milestone"]);
232
235
  for (const [unitType, manifest] of Object.entries(UNIT_MANIFESTS)) {
233
236
  const mode = (manifest as { tools: { mode: string } }).tools.mode;
234
237
  if (mode === "all") {
235
238
  assert.ok(
236
239
  allowedAllUnits.has(unitType),
237
- `manifest "${unitType}" declares tools.mode = "all" but is not on the execute-track. ` +
238
- 'Only execute-task and reactive-execute should have full source write access; ' +
240
+ `manifest "${unitType}" declares tools.mode = "all" but is not explicitly allowed. ` +
241
+ 'Only execute-task, reactive-execute, and complete-milestone should have full source write access; ' +
239
242
  'planning/discuss/research units must use "planning" or "planning-dispatch" (or "docs" for rewrite-docs).',
240
243
  );
241
244
  }
242
245
  }
243
246
  });
244
247
 
248
+ test("#5453: complete-milestone uses all tools so bash verification is not planning-dispatch blocked", () => {
249
+ const manifest = UNIT_MANIFESTS["complete-milestone"];
250
+
251
+ assert.strictEqual(manifest.tools.mode, "all");
252
+ assert.deepEqual(resolveSubagentPermissionContract("complete-milestone"), {
253
+ allowed: true,
254
+ allowedSubagents: ["*"],
255
+ toolsMode: "all",
256
+ });
257
+ // Runtime gate-level regression: these verification commands were blocked
258
+ // under planning-dispatch in #5453; complete-milestone must bypass that gate.
259
+ for (const cmd of ["git diff --name-only HEAD~1", "git log -n1 --oneline"]) {
260
+ const result = shouldBlockPlanningUnit(
261
+ "bash",
262
+ cmd,
263
+ process.cwd(),
264
+ "complete-milestone",
265
+ manifest.tools,
266
+ );
267
+ assert.strictEqual(
268
+ result.block,
269
+ false,
270
+ `shouldBlockPlanningUnit must not block ${cmd} for complete-milestone: ${result.reason}`,
271
+ );
272
+ }
273
+ });
274
+
275
+ test("#5843: run-uat uses verification tools policy so build/test commands can run", () => {
276
+ const manifest = UNIT_MANIFESTS["run-uat"];
277
+
278
+ assert.strictEqual(manifest.tools.mode, "verification");
279
+
280
+ const buildResult = shouldBlockPlanningUnit(
281
+ "bash",
282
+ "npm run build 2>&1",
283
+ process.cwd(),
284
+ "run-uat",
285
+ manifest.tools,
286
+ );
287
+ assert.strictEqual(
288
+ buildResult.block,
289
+ false,
290
+ `run-uat must allow build verification commands: ${buildResult.reason}`,
291
+ );
292
+
293
+ const sourceWriteResult = shouldBlockPlanningUnit(
294
+ "edit",
295
+ "src/main.ts",
296
+ process.cwd(),
297
+ "run-uat",
298
+ manifest.tools,
299
+ );
300
+ assert.strictEqual(sourceWriteResult.block, true);
301
+ assert.match(sourceWriteResult.reason!, /tools-policy "verification"/);
302
+ });
303
+
304
+ test("planning-dispatch hard block message omits internal tracker references", () => {
305
+ const manifest = UNIT_MANIFESTS["validate-milestone"];
306
+ assert.strictEqual(manifest.tools.mode, "planning-dispatch");
307
+
308
+ const result = shouldBlockPlanningUnit(
309
+ "task",
310
+ "scout",
311
+ process.cwd(),
312
+ "validate-milestone",
313
+ manifest.tools,
314
+ );
315
+
316
+ assert.strictEqual(result.block, true);
317
+ assert.ok(result.reason, "blocked dispatch should include a user-facing reason");
318
+ assert.doesNotMatch(result.reason!, /#[0-9]{3,}/);
319
+ });
320
+
245
321
  test('planning-dispatch mode is reserved for slice-level decomposition and completion units', () => {
246
322
  const allowedDispatchUnits = new Set([
247
323
  "plan-slice",
248
324
  "research-slice",
249
325
  "refine-slice",
250
326
  "complete-slice",
251
- "complete-milestone",
252
327
  "gate-evaluate",
253
328
  // Deep planning mode: research-project orchestrates 4 parallel research
254
329
  // subagents (stack/features/architecture/pitfalls). Subagent dispatch is
@@ -162,7 +162,7 @@ describe("validate-milestone stuck-loop guard (#4094)", () => {
162
162
  assert.equal(pauseAutoMock.mock.callCount(), 0);
163
163
  });
164
164
 
165
- test("continues when no VALIDATION file exists yet", async () => {
165
+ test("retries when no VALIDATION file exists yet", async () => {
166
166
  insertMilestone({ id: "M001" });
167
167
  insertSlice({ id: "S01", milestoneId: "M001", title: "Slice 1", status: "complete" });
168
168
 
@@ -173,7 +173,34 @@ describe("validate-milestone stuck-loop guard (#4094)", () => {
173
173
 
174
174
  const result = await runPostUnitVerification({ s, ctx, pi } as VerificationContext, pauseAutoMock);
175
175
 
176
- assert.equal(result, "continue");
176
+ assert.equal(result, "retry");
177
+ assert.equal(pauseAutoMock.mock.callCount(), 0);
178
+ assert.ok(s.pendingVerificationRetry);
179
+ assert.equal(s.pendingVerificationRetry!.unitId, "M001");
180
+ assert.match(s.pendingVerificationRetry!.failureContext, /gsd_validate_milestone/);
181
+ assert.equal(s.pendingVerificationRetry!.attempt, 1);
182
+ });
183
+
184
+ test("retries when VALIDATION file exists but is empty", async () => {
185
+ insertMilestone({ id: "M001" });
186
+ insertSlice({ id: "S01", milestoneId: "M001", title: "Slice 1", status: "complete" });
187
+
188
+ const path = join(tempDir, ".gsd", "milestones", "M001", "M001-VALIDATION.md");
189
+ writeFileSync(path, "", "utf-8");
190
+ invalidateAllCaches();
191
+
192
+ const ctx = makeMockCtx();
193
+ const pi = makeMockPi();
194
+ const pauseAutoMock = mock.fn(async () => {});
195
+ const s = makeMockSession(tempDir, "validate-milestone", "M001");
196
+
197
+ const result = await runPostUnitVerification({ s, ctx, pi } as VerificationContext, pauseAutoMock);
198
+
199
+ assert.equal(result, "retry");
177
200
  assert.equal(pauseAutoMock.mock.callCount(), 0);
201
+ assert.ok(s.pendingVerificationRetry);
202
+ assert.equal(s.pendingVerificationRetry!.unitId, "M001");
203
+ assert.match(s.pendingVerificationRetry!.failureContext, /exists but is empty/);
204
+ assert.equal(s.pendingVerificationRetry!.attempt, 1);
178
205
  });
179
206
  });
@@ -22,7 +22,7 @@ import { join, dirname } from "node:path";
22
22
  import { tmpdir } from "node:os";
23
23
  import { spawnSync } from "node:child_process";
24
24
  import { fileURLToPath, pathToFileURL } from "node:url";
25
- import { discoverCommands, runVerificationGate, formatFailureContext, captureRuntimeErrors, runDependencyAudit, isLikelyCommand } from "../verification-gate.ts";
25
+ import { discoverCommands, runVerificationGate, formatFailureContext, captureRuntimeErrors, runDependencyAudit, isLikelyCommand, validateVerificationCommand } from "../verification-gate.ts";
26
26
  import type { CaptureRuntimeErrorsOptions, DependencyAuditOptions } from "../verification-gate.ts";
27
27
  import { validatePreferences } from "../preferences.ts";
28
28
 
@@ -215,6 +215,102 @@ describe("verification-gate: discovery", () => {
215
215
  assert.equal(result.source, "task-plan");
216
216
  assert.deepStrictEqual(result.commands, ["npm run test"]);
217
217
  });
218
+
219
+ test("taskPlanVerify rejects piped pytest command", () => {
220
+ const result = discoverCommands({
221
+ taskPlanVerify: "python3 -m pytest tests/ -q --tb=short 2>&1 | tail -5",
222
+ cwd: tmp,
223
+ });
224
+ assert.equal(result.source, "none");
225
+ assert.deepStrictEqual(result.commands, []);
226
+ });
227
+
228
+ test("Python project with tests discovers pytest when package.json is absent", () => {
229
+ mkdirSync(join(tmp, "tests"));
230
+ writeFileSync(join(tmp, "tests", "test_sample.py"), "def test_sample():\n assert True\n");
231
+ writeFileSync(
232
+ join(tmp, "pyproject.toml"),
233
+ `[project]
234
+ name = "sample"
235
+
236
+ [tool.pytest.ini_options]
237
+ pythonpath = ["."]
238
+ `,
239
+ );
240
+
241
+ const result = discoverCommands({ cwd: tmp });
242
+
243
+ assert.equal(result.source, "python-project");
244
+ assert.deepStrictEqual(result.commands, ["python3 -m pytest"]);
245
+ });
246
+
247
+ test("Python project with nested Python test file discovers pytest", () => {
248
+ mkdirSync(join(tmp, "tests", "unit"), { recursive: true });
249
+ writeFileSync(join(tmp, "tests", "unit", "sample_test.py"), "def test_sample():\n assert True\n");
250
+
251
+ const result = discoverCommands({ cwd: tmp });
252
+
253
+ assert.equal(result.source, "python-project");
254
+ assert.deepStrictEqual(result.commands, ["python3 -m pytest"]);
255
+ });
256
+
257
+ test("Python project with pytest.ini discovers pytest", () => {
258
+ writeFileSync(join(tmp, "pytest.ini"), "[pytest]\npythonpath = .\n");
259
+
260
+ const result = discoverCommands({ cwd: tmp });
261
+
262
+ assert.equal(result.source, "python-project");
263
+ assert.deepStrictEqual(result.commands, ["python3 -m pytest"]);
264
+ });
265
+
266
+ test("Python project with explicit pyproject pytest marker discovers pytest", () => {
267
+ writeFileSync(
268
+ join(tmp, "pyproject.toml"),
269
+ `[tool.pytest]
270
+ pythonpath = ["."]
271
+ `,
272
+ );
273
+
274
+ const result = discoverCommands({ cwd: tmp });
275
+
276
+ assert.equal(result.source, "python-project");
277
+ assert.deepStrictEqual(result.commands, ["python3 -m pytest"]);
278
+ });
279
+
280
+ test("Python project markers without pytest evidence do not discover pytest", () => {
281
+ mkdirSync(join(tmp, "tests"));
282
+ writeFileSync(join(tmp, "tests", "README.md"), "# tests\n");
283
+ writeFileSync(
284
+ join(tmp, "pyproject.toml"),
285
+ `[project]
286
+ name = "sample"
287
+ dependencies = ["pytest-cov"]
288
+ `,
289
+ );
290
+
291
+ const result = discoverCommands({ cwd: tmp });
292
+
293
+ assert.equal(result.source, "none");
294
+ assert.deepStrictEqual(result.commands, []);
295
+ });
296
+
297
+ test("Python project with setup.cfg alone does not discover pytest", () => {
298
+ writeFileSync(join(tmp, "setup.cfg"), "[tool:pytest]\npythonpath = .\n");
299
+
300
+ const result = discoverCommands({ cwd: tmp });
301
+
302
+ assert.equal(result.source, "none");
303
+ assert.deepStrictEqual(result.commands, []);
304
+ });
305
+
306
+ test("Python project with tox.ini alone does not discover pytest", () => {
307
+ writeFileSync(join(tmp, "tox.ini"), "[pytest]\npythonpath = .\n");
308
+
309
+ const result = discoverCommands({ cwd: tmp });
310
+
311
+ assert.equal(result.source, "none");
312
+ assert.deepStrictEqual(result.commands, []);
313
+ });
218
314
  });
219
315
 
220
316
  // ─── Execution Tests ─────────────────────────────────────────────────────────
@@ -445,6 +541,10 @@ test("isLikelyCommand: prose descriptions are rejected", () => {
445
541
  assert.equal(isLikelyCommand("Build succeeds without errors or warnings"), false);
446
542
  });
447
543
 
544
+ test("isLikelyCommand: non-ASCII prose descriptions are rejected", () => {
545
+ assert.equal(isLikelyCommand("所有 命令 输出 一行 JSONL go test ./... 通过"), false);
546
+ });
547
+
448
548
  test("isLikelyCommand: empty or whitespace-only strings are rejected", () => {
449
549
  assert.equal(isLikelyCommand(""), false);
450
550
  assert.equal(isLikelyCommand(" "), false);
@@ -455,6 +555,15 @@ test("isLikelyCommand: short lowercase tokens without flags are accepted (could
455
555
  assert.equal(isLikelyCommand("mycheck"), true);
456
556
  });
457
557
 
558
+ test("validateVerificationCommand rejects shell control syntax", () => {
559
+ assert.deepEqual(validateVerificationCommand("python3 -m pytest tests/ -q --tb=short").ok, true);
560
+ const result = validateVerificationCommand("python3 -m pytest tests/ -q --tb=short 2>&1 | tail -5");
561
+ assert.equal(result.ok, false);
562
+ if (!result.ok) {
563
+ assert.match(result.reason, /shell control syntax/);
564
+ }
565
+ });
566
+
458
567
  // ─── Additional Preference Validation Tests (T02) ──────────────────────────
459
568
 
460
569
  test("verification-gate: verification_commands produces no unknown-key warnings", () => {