gsd-pi 2.82.0-dev.ed17d078d → 3.0.0-dev.1b44e695b

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 (619) hide show
  1. package/README.md +93 -18
  2. package/dist/cli.js +20 -9
  3. package/dist/headless-ui.js +13 -6
  4. package/dist/headless.js +9 -2
  5. package/dist/resources/.managed-resources-content-hash +1 -1
  6. package/dist/resources/GSD-WORKFLOW.md +10 -1
  7. package/dist/resources/extensions/claude-code-cli/partial-builder.js +2 -1
  8. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +44 -6
  9. package/dist/resources/extensions/cmux/index.js +5 -0
  10. package/dist/resources/extensions/gsd/auto/detect-stuck.js +1 -1
  11. package/dist/resources/extensions/gsd/auto/infra-errors.js +9 -3
  12. package/dist/resources/extensions/gsd/auto/loop.js +122 -40
  13. package/dist/resources/extensions/gsd/auto/orchestrator.js +15 -4
  14. package/dist/resources/extensions/gsd/auto/phases.js +134 -49
  15. package/dist/resources/extensions/gsd/auto/session.js +6 -0
  16. package/dist/resources/extensions/gsd/auto/unit-runner-events.js +7 -1
  17. package/dist/resources/extensions/gsd/auto/workflow-kernel.js +3 -0
  18. package/dist/resources/extensions/gsd/auto/workflow-memory-pressure.js +12 -0
  19. package/dist/resources/extensions/gsd/auto-budget.js +9 -0
  20. package/dist/resources/extensions/gsd/auto-dashboard.js +66 -1
  21. package/dist/resources/extensions/gsd/auto-direct-dispatch.js +1 -0
  22. package/dist/resources/extensions/gsd/auto-dispatch.js +144 -30
  23. package/dist/resources/extensions/gsd/auto-model-selection.js +2 -0
  24. package/dist/resources/extensions/gsd/auto-post-unit.js +329 -137
  25. package/dist/resources/extensions/gsd/auto-prompts.js +36 -10
  26. package/dist/resources/extensions/gsd/auto-recovery.js +82 -16
  27. package/dist/resources/extensions/gsd/auto-start.js +99 -16
  28. package/dist/resources/extensions/gsd/auto-timers.js +11 -3
  29. package/dist/resources/extensions/gsd/auto-verification.js +146 -34
  30. package/dist/resources/extensions/gsd/auto-worktree.js +185 -26
  31. package/dist/resources/extensions/gsd/auto.js +135 -74
  32. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +65 -10
  33. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +13 -10
  34. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +14 -4
  35. package/dist/resources/extensions/gsd/bootstrap/subagent-input.js +21 -9
  36. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +24 -9
  37. package/dist/resources/extensions/gsd/clean-root-preflight.js +170 -8
  38. package/dist/resources/extensions/gsd/commands/catalog.js +10 -1
  39. package/dist/resources/extensions/gsd/commands/handlers/core.js +38 -0
  40. package/dist/resources/extensions/gsd/commands/handlers/ops.js +20 -0
  41. package/dist/resources/extensions/gsd/commands-bootstrap.js +5 -0
  42. package/dist/resources/extensions/gsd/commands-handlers.js +2 -0
  43. package/dist/resources/extensions/gsd/commands-mcp-status.js +9 -0
  44. package/dist/resources/extensions/gsd/commands-prefs-wizard.js +10 -2
  45. package/dist/resources/extensions/gsd/commands-verdict.js +139 -0
  46. package/dist/resources/extensions/gsd/crash-recovery.js +55 -7
  47. package/dist/resources/extensions/gsd/db/auto-workers.js +30 -0
  48. package/dist/resources/extensions/gsd/db/milestone-leases.js +24 -0
  49. package/dist/resources/extensions/gsd/db/unit-dispatches.js +3 -2
  50. package/dist/resources/extensions/gsd/db-base-schema.js +2 -0
  51. package/dist/resources/extensions/gsd/db-migration-steps.js +4 -0
  52. package/dist/resources/extensions/gsd/db-task-slice-rows.js +2 -0
  53. package/dist/resources/extensions/gsd/dispatch-guard.js +46 -2
  54. package/dist/resources/extensions/gsd/docs/preferences-reference.md +10 -0
  55. package/dist/resources/extensions/gsd/doctor-git-checks.js +46 -1
  56. package/dist/resources/extensions/gsd/doctor-runtime-checks.js +28 -11
  57. package/dist/resources/extensions/gsd/doctor.js +2 -28
  58. package/dist/resources/extensions/gsd/export-html.js +27 -425
  59. package/dist/resources/extensions/gsd/forensics.js +10 -3
  60. package/dist/resources/extensions/gsd/git-service.js +152 -15
  61. package/dist/resources/extensions/gsd/gsd-db.js +76 -33
  62. package/dist/resources/extensions/gsd/guided-flow-queue.js +4 -3
  63. package/dist/resources/extensions/gsd/guided-flow.js +110 -117
  64. package/dist/resources/extensions/gsd/guided-unit-context.js +23 -0
  65. package/dist/resources/extensions/gsd/init-wizard.js +17 -2
  66. package/dist/resources/extensions/gsd/markdown-renderer.js +14 -11
  67. package/dist/resources/extensions/gsd/mcp-filter.js +58 -0
  68. package/dist/resources/extensions/gsd/migrate/parsers.js +121 -2
  69. package/dist/resources/extensions/gsd/migration-auto-check.js +12 -17
  70. package/dist/resources/extensions/gsd/milestone-actions.js +11 -4
  71. package/dist/resources/extensions/gsd/native-git-bridge.js +57 -14
  72. package/dist/resources/extensions/gsd/parallel-orchestrator.js +3 -0
  73. package/dist/resources/extensions/gsd/paths.js +4 -0
  74. package/dist/resources/extensions/gsd/pending-auto-start.js +52 -0
  75. package/dist/resources/extensions/gsd/planning-path-scope.js +9 -3
  76. package/dist/resources/extensions/gsd/post-execution-checks.js +73 -9
  77. package/dist/resources/extensions/gsd/pre-execution-checks.js +54 -19
  78. package/dist/resources/extensions/gsd/preferences-mcp.js +19 -0
  79. package/dist/resources/extensions/gsd/preferences-types.js +3 -0
  80. package/dist/resources/extensions/gsd/preferences-validation.js +147 -0
  81. package/dist/resources/extensions/gsd/preferences.js +6 -0
  82. package/dist/resources/extensions/gsd/prompt-loader.js +1 -1
  83. package/dist/resources/extensions/gsd/prompts/complete-milestone.md +1 -1
  84. package/dist/resources/extensions/gsd/prompts/complete-slice.md +1 -1
  85. package/dist/resources/extensions/gsd/prompts/discuss-headless.md +8 -8
  86. package/dist/resources/extensions/gsd/prompts/discuss.md +9 -9
  87. package/dist/resources/extensions/gsd/prompts/forensics.md +3 -3
  88. package/dist/resources/extensions/gsd/prompts/guided-discuss-project.md +4 -4
  89. package/dist/resources/extensions/gsd/prompts/guided-discuss-requirements.md +3 -3
  90. package/dist/resources/extensions/gsd/prompts/plan-slice.md +4 -4
  91. package/dist/resources/extensions/gsd/prompts/queue.md +4 -4
  92. package/dist/resources/extensions/gsd/prompts/reactive-execute.md +1 -1
  93. package/dist/resources/extensions/gsd/prompts/refine-slice.md +2 -2
  94. package/dist/resources/extensions/gsd/prompts/rewrite-docs.md +1 -1
  95. package/dist/resources/extensions/gsd/queue-reorder-ui.js +30 -13
  96. package/dist/resources/extensions/gsd/repo-identity.js +39 -22
  97. package/dist/resources/extensions/gsd/repository-registry.js +44 -0
  98. package/dist/resources/extensions/gsd/safety/evidence-collector.js +2 -0
  99. package/dist/resources/extensions/gsd/safety/evidence-cross-ref.js +42 -18
  100. package/dist/resources/extensions/gsd/session-lock.js +15 -2
  101. package/dist/resources/extensions/gsd/slice-parallel-conflict.js +2 -2
  102. package/dist/resources/extensions/gsd/slice-parallel-orchestrator.js +84 -5
  103. package/dist/resources/extensions/gsd/smart-entry-routing.js +36 -0
  104. package/dist/resources/extensions/gsd/state-reconciliation/drift/merge-state.js +6 -1
  105. package/dist/resources/extensions/gsd/state-reconciliation/drift/project-md.js +9 -14
  106. package/dist/resources/extensions/gsd/state-reconciliation/drift/roadmap.js +19 -24
  107. package/dist/resources/extensions/gsd/state.js +28 -7
  108. package/dist/resources/extensions/gsd/status-guards.js +14 -2
  109. package/dist/resources/extensions/gsd/templates/PREFERENCES.md +1 -0
  110. package/dist/resources/extensions/gsd/templates/plan.md +9 -5
  111. package/dist/resources/extensions/gsd/templates/task-plan.md +10 -2
  112. package/dist/resources/extensions/gsd/tools/complete-milestone.js +15 -9
  113. package/dist/resources/extensions/gsd/tools/complete-slice.js +56 -10
  114. package/dist/resources/extensions/gsd/tools/exec-tool.js +87 -5
  115. package/dist/resources/extensions/gsd/tools/plan-milestone.js +7 -1
  116. package/dist/resources/extensions/gsd/tools/plan-slice.js +151 -15
  117. package/dist/resources/extensions/gsd/tools/validate-milestone.js +32 -1
  118. package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +185 -40
  119. package/dist/resources/extensions/gsd/unit-context-composer.js +2 -0
  120. package/dist/resources/extensions/gsd/unit-context-manifest.js +69 -17
  121. package/dist/resources/extensions/gsd/validation.js +23 -1
  122. package/dist/resources/extensions/gsd/verification-gate.js +142 -7
  123. package/dist/resources/extensions/gsd/verification-verdict.js +26 -0
  124. package/dist/resources/extensions/gsd/workflow-manifest.js +2 -0
  125. package/dist/resources/extensions/gsd/workflow-mcp.js +17 -1
  126. package/dist/resources/extensions/gsd/workflow-projections.js +6 -8
  127. package/dist/resources/extensions/gsd/worktree-lifecycle.js +86 -19
  128. package/dist/resources/extensions/gsd/worktree-manager.js +11 -2
  129. package/dist/resources/extensions/gsd/worktree-safety.js +43 -4
  130. package/dist/resources/extensions/gsd/worktree-state-projection.js +31 -0
  131. package/dist/resources/extensions/gsd/worktree-telemetry.js +32 -0
  132. package/dist/resources/extensions/shared/html-shell.js +388 -0
  133. package/dist/resources/extensions/shared/interview-ui.js +6 -4
  134. package/dist/resources/extensions/shared/next-action-ui.js +13 -5
  135. package/dist/resources/extensions/subagent/index.js +448 -78
  136. package/dist/resources/extensions/subagent/launch.js +77 -0
  137. package/dist/resources/extensions/subagent/run-store.js +148 -0
  138. package/dist/resources/extensions/ttsr/ttsr-manager.js +3 -1
  139. package/dist/resources/extensions/visual-brief/artifact-policy.js +29 -0
  140. package/dist/resources/extensions/visual-brief/extension-manifest.json +8 -0
  141. package/dist/resources/extensions/visual-brief/index.js +5 -0
  142. package/dist/resources/extensions/visual-brief/page-contract.js +124 -0
  143. package/dist/resources/extensions/visual-brief/prompts.js +140 -0
  144. package/dist/resources/skills/forensics/SKILL.md +1 -1
  145. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  146. package/dist/web/standalone/.next/BUILD_ID +1 -1
  147. package/dist/web/standalone/.next/app-path-routes-manifest.json +7 -7
  148. package/dist/web/standalone/.next/build-manifest.json +3 -3
  149. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  150. package/dist/web/standalone/.next/react-loadable-manifest.json +5 -5
  151. package/dist/web/standalone/.next/required-server-files.json +1 -1
  152. package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  153. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  154. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  155. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  156. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  157. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  158. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  159. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  160. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  161. package/dist/web/standalone/.next/server/app/_not-found/page.js +2 -2
  162. package/dist/web/standalone/.next/server/app/_not-found/page.js.nft.json +1 -1
  163. package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  164. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  165. package/dist/web/standalone/.next/server/app/_not-found.rsc +4 -7
  166. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +4 -7
  167. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  168. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +4 -5
  169. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  170. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  171. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +2 -5
  172. package/dist/web/standalone/.next/server/app/api/browse-directories/route.js +1 -1
  173. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  174. package/dist/web/standalone/.next/server/app/index.html +1 -1
  175. package/dist/web/standalone/.next/server/app/index.rsc +4 -7
  176. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  177. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +4 -7
  178. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  179. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +4 -5
  180. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +2 -5
  181. package/dist/web/standalone/.next/server/app/page.js +2 -2
  182. package/dist/web/standalone/.next/server/app/page.js.nft.json +1 -1
  183. package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
  184. package/dist/web/standalone/.next/server/app-paths-manifest.json +7 -7
  185. package/dist/web/standalone/.next/server/chunks/4266.js +2 -0
  186. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  187. package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
  188. package/dist/web/standalone/.next/server/next-font-manifest.js +1 -1
  189. package/dist/web/standalone/.next/server/next-font-manifest.json +1 -1
  190. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  191. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  192. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  193. package/dist/web/standalone/.next/static/chunks/2973.33f26573894b6153.js +2 -0
  194. package/dist/web/standalone/.next/static/chunks/8359.65b24fac92188a6b.js +10 -0
  195. package/dist/web/standalone/.next/static/chunks/9441.ff70bb53f6835771.js +1 -0
  196. package/dist/web/standalone/.next/static/chunks/app/layout-8c10ec293ae0f1d5.js +1 -0
  197. package/dist/web/standalone/.next/static/chunks/{webpack-de742b64187e13fe.js → webpack-855d616060cb6e59.js} +1 -1
  198. package/dist/web/standalone/.next/static/css/746ee28c929d1880.css +1 -0
  199. package/dist/web/standalone/server.js +1 -1
  200. package/package.json +4 -4
  201. package/packages/contracts/dist/rpc.test.js +7 -0
  202. package/packages/contracts/dist/rpc.test.js.map +1 -1
  203. package/packages/contracts/dist/workflow.d.ts +21 -0
  204. package/packages/contracts/dist/workflow.d.ts.map +1 -1
  205. package/packages/contracts/dist/workflow.js +24 -0
  206. package/packages/contracts/dist/workflow.js.map +1 -1
  207. package/packages/contracts/src/rpc.test.ts +8 -0
  208. package/packages/contracts/src/workflow.ts +24 -0
  209. package/packages/daemon/package.json +2 -2
  210. package/packages/mcp-server/README.md +13 -4
  211. package/packages/mcp-server/dist/workflow-tools.d.ts +0 -3
  212. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  213. package/packages/mcp-server/dist/workflow-tools.js +80 -0
  214. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  215. package/packages/mcp-server/package.json +2 -2
  216. package/packages/mcp-server/src/workflow-tools.test.ts +23 -1
  217. package/packages/mcp-server/src/workflow-tools.ts +168 -0
  218. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
  219. package/packages/native/package.json +1 -1
  220. package/packages/native/tsconfig.json +2 -1
  221. package/packages/native/tsconfig.tsbuildinfo +1 -1
  222. package/packages/pi-agent-core/package.json +1 -1
  223. package/packages/pi-ai/dist/providers/google-gemini-cli.d.ts.map +1 -1
  224. package/packages/pi-ai/dist/providers/google-gemini-cli.js +5 -0
  225. package/packages/pi-ai/dist/providers/google-gemini-cli.js.map +1 -1
  226. package/packages/pi-ai/dist/providers/google-gemini-cli.test.d.ts +2 -0
  227. package/packages/pi-ai/dist/providers/google-gemini-cli.test.d.ts.map +1 -0
  228. package/packages/pi-ai/dist/providers/google-gemini-cli.test.js +41 -0
  229. package/packages/pi-ai/dist/providers/google-gemini-cli.test.js.map +1 -0
  230. package/packages/pi-ai/dist/providers/openai-codex-responses.d.ts.map +1 -1
  231. package/packages/pi-ai/dist/providers/openai-codex-responses.js +82 -1
  232. package/packages/pi-ai/dist/providers/openai-codex-responses.js.map +1 -1
  233. package/packages/pi-ai/dist/providers/openai-codex-responses.test.d.ts +2 -0
  234. package/packages/pi-ai/dist/providers/openai-codex-responses.test.d.ts.map +1 -0
  235. package/packages/pi-ai/dist/providers/openai-codex-responses.test.js +52 -0
  236. package/packages/pi-ai/dist/providers/openai-codex-responses.test.js.map +1 -0
  237. package/packages/pi-ai/dist/providers/simple-options.d.ts +2 -4
  238. package/packages/pi-ai/dist/providers/simple-options.d.ts.map +1 -1
  239. package/packages/pi-ai/dist/providers/simple-options.js +5 -6
  240. package/packages/pi-ai/dist/providers/simple-options.js.map +1 -1
  241. package/packages/pi-ai/dist/providers/simple-options.test.d.ts +2 -0
  242. package/packages/pi-ai/dist/providers/simple-options.test.d.ts.map +1 -0
  243. package/packages/pi-ai/dist/providers/simple-options.test.js +50 -0
  244. package/packages/pi-ai/dist/providers/simple-options.test.js.map +1 -0
  245. package/packages/pi-ai/package.json +1 -1
  246. package/packages/pi-ai/src/providers/google-gemini-cli.test.ts +49 -0
  247. package/packages/pi-ai/src/providers/google-gemini-cli.ts +7 -0
  248. package/packages/pi-ai/src/providers/openai-codex-responses.test.ts +63 -0
  249. package/packages/pi-ai/src/providers/openai-codex-responses.ts +91 -1
  250. package/packages/pi-ai/src/providers/simple-options.test.ts +60 -0
  251. package/packages/pi-ai/src/providers/simple-options.ts +5 -6
  252. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
  253. package/packages/pi-coding-agent/dist/core/agent-session-thinking-level.test.d.ts +2 -0
  254. package/packages/pi-coding-agent/dist/core/agent-session-thinking-level.test.d.ts.map +1 -0
  255. package/packages/pi-coding-agent/dist/core/agent-session-thinking-level.test.js +66 -0
  256. package/packages/pi-coding-agent/dist/core/agent-session-thinking-level.test.js.map +1 -0
  257. package/packages/pi-coding-agent/dist/core/agent-session.js +1 -1
  258. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  259. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js +44 -3
  260. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js.map +1 -1
  261. package/packages/pi-coding-agent/dist/core/compaction/compaction.d.ts +6 -1
  262. package/packages/pi-coding-agent/dist/core/compaction/compaction.d.ts.map +1 -1
  263. package/packages/pi-coding-agent/dist/core/compaction/compaction.js +7 -2
  264. package/packages/pi-coding-agent/dist/core/compaction/compaction.js.map +1 -1
  265. package/packages/pi-coding-agent/dist/core/compaction/compaction.test.js +14 -1
  266. package/packages/pi-coding-agent/dist/core/compaction/compaction.test.js.map +1 -1
  267. package/packages/pi-coding-agent/dist/core/sdk.js +1 -1
  268. package/packages/pi-coding-agent/dist/core/sdk.js.map +1 -1
  269. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js +8 -2
  270. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js.map +1 -1
  271. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.d.ts.map +1 -1
  272. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js +24 -6
  273. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js.map +1 -1
  274. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +1 -1
  275. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
  276. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  277. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +82 -97
  278. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  279. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js +7 -7
  280. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.js.map +1 -1
  281. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js +11 -0
  282. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js.map +1 -1
  283. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js +25 -1
  284. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-ordering.test.js.map +1 -1
  285. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +2 -0
  286. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  287. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +24 -10
  288. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  289. package/packages/pi-coding-agent/package.json +1 -1
  290. package/packages/pi-coding-agent/src/core/agent-session-thinking-level.test.ts +79 -0
  291. package/packages/pi-coding-agent/src/core/agent-session.ts +1 -1
  292. package/packages/pi-coding-agent/src/core/chat-controller-ordering.test.ts +53 -3
  293. package/packages/pi-coding-agent/src/core/compaction/compaction.test.ts +23 -1
  294. package/packages/pi-coding-agent/src/core/compaction/compaction.ts +7 -2
  295. package/packages/pi-coding-agent/src/core/sdk.ts +1 -1
  296. package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/tool-execution.test.ts +17 -1
  297. package/packages/pi-coding-agent/src/modes/interactive/components/footer.ts +23 -7
  298. package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +1 -1
  299. package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +91 -102
  300. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.test.ts +15 -1
  301. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.ts +9 -9
  302. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode-ordering.test.ts +30 -1
  303. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +29 -10
  304. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
  305. package/packages/pi-tui/dist/__tests__/terminal.test.d.ts +2 -0
  306. package/packages/pi-tui/dist/__tests__/terminal.test.d.ts.map +1 -0
  307. package/packages/pi-tui/dist/__tests__/terminal.test.js +103 -0
  308. package/packages/pi-tui/dist/__tests__/terminal.test.js.map +1 -0
  309. package/packages/pi-tui/dist/__tests__/tui.test.js +45 -2
  310. package/packages/pi-tui/dist/__tests__/tui.test.js.map +1 -1
  311. package/packages/pi-tui/dist/terminal.d.ts +2 -0
  312. package/packages/pi-tui/dist/terminal.d.ts.map +1 -1
  313. package/packages/pi-tui/dist/terminal.js +12 -0
  314. package/packages/pi-tui/dist/terminal.js.map +1 -1
  315. package/packages/pi-tui/dist/tui.d.ts.map +1 -1
  316. package/packages/pi-tui/dist/tui.js +106 -27
  317. package/packages/pi-tui/dist/tui.js.map +1 -1
  318. package/packages/pi-tui/package.json +1 -1
  319. package/packages/pi-tui/src/__tests__/terminal.test.ts +121 -0
  320. package/packages/pi-tui/src/__tests__/tui.test.ts +59 -2
  321. package/packages/pi-tui/src/terminal.ts +11 -0
  322. package/packages/pi-tui/src/tui.ts +108 -27
  323. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
  324. package/packages/rpc-client/package.json +1 -1
  325. package/packages/rpc-client/tsconfig.tsbuildinfo +1 -1
  326. package/pkg/package.json +1 -1
  327. package/src/resources/GSD-WORKFLOW.md +10 -1
  328. package/src/resources/extensions/claude-code-cli/partial-builder.ts +2 -1
  329. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +52 -6
  330. package/src/resources/extensions/claude-code-cli/tests/partial-builder.test.ts +19 -2
  331. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +49 -2
  332. package/src/resources/extensions/cmux/index.ts +6 -0
  333. package/src/resources/extensions/gsd/auto/contracts.ts +19 -6
  334. package/src/resources/extensions/gsd/auto/detect-stuck.ts +1 -0
  335. package/src/resources/extensions/gsd/auto/infra-errors.ts +9 -3
  336. package/src/resources/extensions/gsd/auto/loop.ts +123 -40
  337. package/src/resources/extensions/gsd/auto/orchestrator.ts +15 -4
  338. package/src/resources/extensions/gsd/auto/phases.ts +160 -60
  339. package/src/resources/extensions/gsd/auto/session.ts +16 -0
  340. package/src/resources/extensions/gsd/auto/types.ts +3 -0
  341. package/src/resources/extensions/gsd/auto/unit-runner-events.ts +6 -2
  342. package/src/resources/extensions/gsd/auto/workflow-kernel.ts +5 -1
  343. package/src/resources/extensions/gsd/auto/workflow-memory-pressure.ts +13 -0
  344. package/src/resources/extensions/gsd/auto-budget.ts +11 -0
  345. package/src/resources/extensions/gsd/auto-dashboard.ts +72 -1
  346. package/src/resources/extensions/gsd/auto-direct-dispatch.ts +1 -0
  347. package/src/resources/extensions/gsd/auto-dispatch.ts +164 -29
  348. package/src/resources/extensions/gsd/auto-model-selection.ts +2 -1
  349. package/src/resources/extensions/gsd/auto-post-unit.ts +369 -148
  350. package/src/resources/extensions/gsd/auto-prompts.ts +36 -13
  351. package/src/resources/extensions/gsd/auto-recovery.ts +86 -13
  352. package/src/resources/extensions/gsd/auto-start.ts +109 -14
  353. package/src/resources/extensions/gsd/auto-timers.ts +10 -3
  354. package/src/resources/extensions/gsd/auto-verification.ts +174 -42
  355. package/src/resources/extensions/gsd/auto-worktree.ts +202 -30
  356. package/src/resources/extensions/gsd/auto.ts +172 -81
  357. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +66 -10
  358. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +13 -10
  359. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +13 -4
  360. package/src/resources/extensions/gsd/bootstrap/subagent-input.ts +19 -7
  361. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +27 -10
  362. package/src/resources/extensions/gsd/clean-root-preflight.ts +174 -8
  363. package/src/resources/extensions/gsd/commands/catalog.ts +10 -1
  364. package/src/resources/extensions/gsd/commands/handlers/core.ts +41 -0
  365. package/src/resources/extensions/gsd/commands/handlers/ops.ts +21 -0
  366. package/src/resources/extensions/gsd/commands-bootstrap.ts +10 -0
  367. package/src/resources/extensions/gsd/commands-handlers.ts +2 -0
  368. package/src/resources/extensions/gsd/commands-mcp-status.ts +8 -0
  369. package/src/resources/extensions/gsd/commands-prefs-wizard.ts +11 -3
  370. package/src/resources/extensions/gsd/commands-verdict.ts +202 -0
  371. package/src/resources/extensions/gsd/crash-recovery.ts +55 -6
  372. package/src/resources/extensions/gsd/db/auto-workers.ts +37 -0
  373. package/src/resources/extensions/gsd/db/milestone-leases.ts +26 -0
  374. package/src/resources/extensions/gsd/db/unit-dispatches.ts +4 -3
  375. package/src/resources/extensions/gsd/db-base-schema.ts +2 -0
  376. package/src/resources/extensions/gsd/db-migration-steps.ts +5 -0
  377. package/src/resources/extensions/gsd/db-task-slice-rows.ts +4 -0
  378. package/src/resources/extensions/gsd/dispatch-guard.ts +60 -2
  379. package/src/resources/extensions/gsd/docs/preferences-reference.md +10 -0
  380. package/src/resources/extensions/gsd/doctor-git-checks.ts +45 -1
  381. package/src/resources/extensions/gsd/doctor-runtime-checks.ts +25 -13
  382. package/src/resources/extensions/gsd/doctor-types.ts +1 -0
  383. package/src/resources/extensions/gsd/doctor.ts +2 -27
  384. package/src/resources/extensions/gsd/export-html.ts +27 -427
  385. package/src/resources/extensions/gsd/forensics.ts +9 -3
  386. package/src/resources/extensions/gsd/git-service.ts +182 -16
  387. package/src/resources/extensions/gsd/gsd-db.ts +80 -31
  388. package/src/resources/extensions/gsd/guided-flow-queue.ts +4 -3
  389. package/src/resources/extensions/gsd/guided-flow.ts +142 -134
  390. package/src/resources/extensions/gsd/guided-unit-context.ts +30 -0
  391. package/src/resources/extensions/gsd/init-wizard.ts +17 -2
  392. package/src/resources/extensions/gsd/journal.ts +8 -1
  393. package/src/resources/extensions/gsd/markdown-renderer.ts +14 -11
  394. package/src/resources/extensions/gsd/mcp-filter.ts +80 -0
  395. package/src/resources/extensions/gsd/migrate/parsers.ts +139 -2
  396. package/src/resources/extensions/gsd/migration-auto-check.ts +15 -23
  397. package/src/resources/extensions/gsd/milestone-actions.ts +10 -4
  398. package/src/resources/extensions/gsd/native-git-bridge.ts +63 -14
  399. package/src/resources/extensions/gsd/parallel-orchestrator.ts +3 -0
  400. package/src/resources/extensions/gsd/paths.ts +5 -0
  401. package/src/resources/extensions/gsd/pending-auto-start.ts +79 -0
  402. package/src/resources/extensions/gsd/planning-path-scope.ts +10 -2
  403. package/src/resources/extensions/gsd/post-execution-checks.ts +87 -12
  404. package/src/resources/extensions/gsd/pre-execution-checks.ts +67 -19
  405. package/src/resources/extensions/gsd/preferences-mcp.ts +27 -0
  406. package/src/resources/extensions/gsd/preferences-types.ts +35 -0
  407. package/src/resources/extensions/gsd/preferences-validation.ts +154 -0
  408. package/src/resources/extensions/gsd/preferences.ts +9 -0
  409. package/src/resources/extensions/gsd/prompt-loader.ts +1 -1
  410. package/src/resources/extensions/gsd/prompts/complete-milestone.md +1 -1
  411. package/src/resources/extensions/gsd/prompts/complete-slice.md +1 -1
  412. package/src/resources/extensions/gsd/prompts/discuss-headless.md +8 -8
  413. package/src/resources/extensions/gsd/prompts/discuss.md +9 -9
  414. package/src/resources/extensions/gsd/prompts/forensics.md +3 -3
  415. package/src/resources/extensions/gsd/prompts/guided-discuss-project.md +4 -4
  416. package/src/resources/extensions/gsd/prompts/guided-discuss-requirements.md +3 -3
  417. package/src/resources/extensions/gsd/prompts/plan-slice.md +4 -4
  418. package/src/resources/extensions/gsd/prompts/queue.md +4 -4
  419. package/src/resources/extensions/gsd/prompts/reactive-execute.md +1 -1
  420. package/src/resources/extensions/gsd/prompts/refine-slice.md +2 -2
  421. package/src/resources/extensions/gsd/prompts/rewrite-docs.md +1 -1
  422. package/src/resources/extensions/gsd/queue-reorder-ui.ts +31 -13
  423. package/src/resources/extensions/gsd/repo-identity.ts +45 -25
  424. package/src/resources/extensions/gsd/repository-registry.ts +77 -0
  425. package/src/resources/extensions/gsd/safety/evidence-collector.ts +2 -0
  426. package/src/resources/extensions/gsd/safety/evidence-cross-ref.ts +54 -19
  427. package/src/resources/extensions/gsd/session-lock.ts +15 -2
  428. package/src/resources/extensions/gsd/slice-parallel-conflict.ts +2 -2
  429. package/src/resources/extensions/gsd/slice-parallel-orchestrator.ts +75 -3
  430. package/src/resources/extensions/gsd/smart-entry-routing.ts +77 -0
  431. package/src/resources/extensions/gsd/state-reconciliation/drift/merge-state.ts +8 -1
  432. package/src/resources/extensions/gsd/state-reconciliation/drift/project-md.ts +12 -15
  433. package/src/resources/extensions/gsd/state-reconciliation/drift/roadmap.ts +17 -25
  434. package/src/resources/extensions/gsd/state.ts +33 -7
  435. package/src/resources/extensions/gsd/status-guards.ts +16 -2
  436. package/src/resources/extensions/gsd/templates/PREFERENCES.md +1 -0
  437. package/src/resources/extensions/gsd/templates/plan.md +9 -5
  438. package/src/resources/extensions/gsd/templates/task-plan.md +10 -2
  439. package/src/resources/extensions/gsd/tests/artifact-retry-cap.test.ts +1 -1
  440. package/src/resources/extensions/gsd/tests/auto-abort-pause-regression.test.ts +10 -1
  441. package/src/resources/extensions/gsd/tests/auto-dashboard.test.ts +71 -0
  442. package/src/resources/extensions/gsd/tests/auto-deterministic-error-classification-4973.test.ts +116 -0
  443. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +775 -34
  444. package/src/resources/extensions/gsd/tests/auto-orchestrator.test.ts +245 -28
  445. package/src/resources/extensions/gsd/tests/auto-paused-ui-cleanup.test.ts +151 -12
  446. package/src/resources/extensions/gsd/tests/auto-phases-lifecycle.test.ts +53 -2
  447. package/src/resources/extensions/gsd/tests/auto-post-unit-step-message.test.ts +18 -6
  448. package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +136 -13
  449. package/src/resources/extensions/gsd/tests/auto-remote-session-lock-cleanup.test.ts +64 -0
  450. package/src/resources/extensions/gsd/tests/auto-retry-mcp-churn-fixes.test.ts +12 -0
  451. package/src/resources/extensions/gsd/tests/auto-start-orphan-bootstrap.test.ts +1 -0
  452. package/src/resources/extensions/gsd/tests/auto-stop-notification.test.ts +20 -0
  453. package/src/resources/extensions/gsd/tests/auto-workers.test.ts +29 -0
  454. package/src/resources/extensions/gsd/tests/auto-worktree-registry.test.ts +69 -1
  455. package/src/resources/extensions/gsd/tests/autocomplete-regressions-1675.test.ts +21 -0
  456. package/src/resources/extensions/gsd/tests/brief-command.test.ts +89 -0
  457. package/src/resources/extensions/gsd/tests/checkout-branch-stash-guard.test.ts +87 -0
  458. package/src/resources/extensions/gsd/tests/clean-root-preflight.test.ts +107 -2
  459. package/src/resources/extensions/gsd/tests/clear-stale-autostart.test.ts +32 -4
  460. package/src/resources/extensions/gsd/tests/closeout-git-deferral.test.ts +16 -0
  461. package/src/resources/extensions/gsd/tests/commands-verdict.test.ts +378 -0
  462. package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +55 -2
  463. package/src/resources/extensions/gsd/tests/complete-slice.test.ts +60 -9
  464. package/src/resources/extensions/gsd/tests/complete-task.test.ts +3 -1
  465. package/src/resources/extensions/gsd/tests/crash-recovery-via-db.test.ts +104 -2
  466. package/src/resources/extensions/gsd/tests/custom-engine-loop-integration.test.ts +2 -0
  467. package/src/resources/extensions/gsd/tests/db-authority-regression.test.ts +208 -0
  468. package/src/resources/extensions/gsd/tests/db-task-slice-rows.test.ts +1 -0
  469. package/src/resources/extensions/gsd/tests/deep-project-auto-loop.test.ts +61 -2
  470. package/src/resources/extensions/gsd/tests/dispatch-complete-milestone-guard.test.ts +111 -1
  471. package/src/resources/extensions/gsd/tests/dispatch-guard.test.ts +68 -1
  472. package/src/resources/extensions/gsd/tests/dispatch-missing-task-plans.test.ts +140 -1
  473. package/src/resources/extensions/gsd/tests/doctor-empty-worktree.test.ts +65 -0
  474. package/src/resources/extensions/gsd/tests/doctor-forensics-db-open-regression.test.ts +50 -0
  475. package/src/resources/extensions/gsd/tests/evidence-cross-ref.test.ts +97 -0
  476. package/src/resources/extensions/gsd/tests/exec-sandbox.test.ts +99 -1
  477. package/src/resources/extensions/gsd/tests/execution-entry-missing-context-4671.test.ts +15 -1
  478. package/src/resources/extensions/gsd/tests/export-html-enhancements.test.ts +8 -0
  479. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +28 -0
  480. package/src/resources/extensions/gsd/tests/gsdroot-worktree-detection.test.ts +5 -2
  481. package/src/resources/extensions/gsd/tests/guided-discuss-project-prompt-rendering.test.ts +2 -0
  482. package/src/resources/extensions/gsd/tests/guided-dispatch-root.test.ts +106 -0
  483. package/src/resources/extensions/gsd/tests/guided-flow-session-isolation.test.ts +59 -11
  484. package/src/resources/extensions/gsd/tests/guided-flow.test.ts +21 -0
  485. package/src/resources/extensions/gsd/tests/guided-tool-contract.test.ts +65 -0
  486. package/src/resources/extensions/gsd/tests/headless-milestone-parity.test.ts +7 -7
  487. package/src/resources/extensions/gsd/tests/hook-model-resolution.test.ts +5 -0
  488. package/src/resources/extensions/gsd/tests/infra-error.test.ts +2 -2
  489. package/src/resources/extensions/gsd/tests/infra-errors-cooldown.test.ts +9 -0
  490. package/src/resources/extensions/gsd/tests/init-prefs-routing.test.ts +22 -0
  491. package/src/resources/extensions/gsd/tests/integration/doctor-runtime.test.ts +20 -0
  492. package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +226 -2
  493. package/src/resources/extensions/gsd/tests/integration/state-machine-edge-cases.test.ts +5 -21
  494. package/src/resources/extensions/gsd/tests/integration/state-machine-live-validation.test.ts +15 -0
  495. package/src/resources/extensions/gsd/tests/integration/state-machine-runtime-failures.test.ts +6 -1
  496. package/src/resources/extensions/gsd/tests/interrupted-session-auto.test.ts +40 -0
  497. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +49 -3
  498. package/src/resources/extensions/gsd/tests/journal.test.ts +32 -0
  499. package/src/resources/extensions/gsd/tests/mcp-filter.test.ts +287 -0
  500. package/src/resources/extensions/gsd/tests/mcp-status.test.ts +11 -0
  501. package/src/resources/extensions/gsd/tests/merge-db-cycle.test.ts +179 -0
  502. package/src/resources/extensions/gsd/tests/merge-self-branch-guard.test.ts +21 -40
  503. package/src/resources/extensions/gsd/tests/migrate-validator-parsers.test.ts +59 -1
  504. package/src/resources/extensions/gsd/tests/migration-auto-check.test.ts +26 -18
  505. package/src/resources/extensions/gsd/tests/native-git-bridge-exec-fallback.test.ts +80 -2
  506. package/src/resources/extensions/gsd/tests/orphaned-worktree-audit.test.ts +121 -1
  507. package/src/resources/extensions/gsd/tests/parallel-orchestrator-zombie-cleanup.test.ts +55 -0
  508. package/src/resources/extensions/gsd/tests/park-db-sync.test.ts +55 -1
  509. package/src/resources/extensions/gsd/tests/pending-autostart-scope.test.ts +29 -5
  510. package/src/resources/extensions/gsd/tests/pipeline-variant-dispatch.test.ts +2 -1
  511. package/src/resources/extensions/gsd/tests/plan-milestone.test.ts +26 -0
  512. package/src/resources/extensions/gsd/tests/plan-slice-prompt.test.ts +2 -0
  513. package/src/resources/extensions/gsd/tests/plan-slice.test.ts +343 -3
  514. package/src/resources/extensions/gsd/tests/plan-task.test.ts +18 -1
  515. package/src/resources/extensions/gsd/tests/post-exec-retry-bypass.test.ts +79 -1
  516. package/src/resources/extensions/gsd/tests/post-execution-checks.test.ts +105 -3
  517. package/src/resources/extensions/gsd/tests/post-unit-git-failure.test.ts +8 -1
  518. package/src/resources/extensions/gsd/tests/post-unit-state-rebuild.test.ts +84 -0
  519. package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +147 -0
  520. package/src/resources/extensions/gsd/tests/pre-execution-pause-wiring.test.ts +54 -0
  521. package/src/resources/extensions/gsd/tests/preferences-mcp.test.ts +128 -0
  522. package/src/resources/extensions/gsd/tests/preferences.test.ts +75 -0
  523. package/src/resources/extensions/gsd/tests/prefs-wizard-coverage.test.ts +79 -0
  524. package/src/resources/extensions/gsd/tests/progressive-planning.test.ts +42 -1
  525. package/src/resources/extensions/gsd/tests/prompt-loader.test.ts +23 -0
  526. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +37 -1
  527. package/src/resources/extensions/gsd/tests/quality-gates.test.ts +6 -0
  528. package/src/resources/extensions/gsd/tests/queue-reorder-ui.test.ts +54 -0
  529. package/src/resources/extensions/gsd/tests/remediation-completion-guard.test.ts +89 -2
  530. package/src/resources/extensions/gsd/tests/repo-identity-worktree.test.ts +28 -1
  531. package/src/resources/extensions/gsd/tests/repository-registry.test.ts +52 -0
  532. package/src/resources/extensions/gsd/tests/run-uat-replay-cap.test.ts +2 -3
  533. package/src/resources/extensions/gsd/tests/session-lock-regression.test.ts +35 -0
  534. package/src/resources/extensions/gsd/tests/session-switch-abort-misclassification.test.ts +59 -1
  535. package/src/resources/extensions/gsd/tests/slice-parallel-conflict.test.ts +2 -2
  536. package/src/resources/extensions/gsd/tests/slice-parallel-orchestrator.test.ts +112 -1
  537. package/src/resources/extensions/gsd/tests/smart-entry-routing.test.ts +113 -0
  538. package/src/resources/extensions/gsd/tests/start-auto-detached.test.ts +94 -2
  539. package/src/resources/extensions/gsd/tests/state-corruption-2945.test.ts +6 -0
  540. package/src/resources/extensions/gsd/tests/state-reconciliation-drift.test.ts +119 -23
  541. package/src/resources/extensions/gsd/tests/status-guards.test.ts +17 -1
  542. package/src/resources/extensions/gsd/tests/stuck-state-via-db.test.ts +64 -1
  543. package/src/resources/extensions/gsd/tests/subagent-model-dispatch.test.ts +2 -1
  544. package/src/resources/extensions/gsd/tests/summary-render-parity.test.ts +7 -3
  545. package/src/resources/extensions/gsd/tests/unit-context-composer.test.ts +1 -0
  546. package/src/resources/extensions/gsd/tests/unit-context-manifest.test.ts +131 -9
  547. package/src/resources/extensions/gsd/tests/uok-gitops-turn-action.test.ts +111 -1
  548. package/src/resources/extensions/gsd/tests/validate-milestone-stuck-guard.test.ts +29 -2
  549. package/src/resources/extensions/gsd/tests/validate-milestone-write-order.test.ts +68 -0
  550. package/src/resources/extensions/gsd/tests/validate-milestone.test.ts +42 -1
  551. package/src/resources/extensions/gsd/tests/verification-gate.test.ts +188 -1
  552. package/src/resources/extensions/gsd/tests/verification-verdict.test.ts +78 -0
  553. package/src/resources/extensions/gsd/tests/workflow-kernel.test.ts +7 -0
  554. package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +19 -1
  555. package/src/resources/extensions/gsd/tests/workflow-memory-pressure.test.ts +21 -1
  556. package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +153 -1
  557. package/src/resources/extensions/gsd/tests/worktree-git-pathspec.test.ts +39 -0
  558. package/src/resources/extensions/gsd/tests/worktree-journal-events.test.ts +64 -12
  559. package/src/resources/extensions/gsd/tests/worktree-lifecycle.test.ts +73 -2
  560. package/src/resources/extensions/gsd/tests/worktree-preferences-sync.test.ts +25 -0
  561. package/src/resources/extensions/gsd/tests/worktree-safety.test.ts +90 -0
  562. package/src/resources/extensions/gsd/tests/worktree-sync-milestones.test.ts +95 -0
  563. package/src/resources/extensions/gsd/tests/worktree-telemetry.test.ts +16 -0
  564. package/src/resources/extensions/gsd/tests/worktree-write-gate.test.ts +27 -0
  565. package/src/resources/extensions/gsd/tests/write-gate-planning-unit.test.ts +59 -0
  566. package/src/resources/extensions/gsd/tools/complete-milestone.ts +18 -10
  567. package/src/resources/extensions/gsd/tools/complete-slice.ts +57 -10
  568. package/src/resources/extensions/gsd/tools/exec-tool.ts +98 -5
  569. package/src/resources/extensions/gsd/tools/plan-milestone.ts +5 -1
  570. package/src/resources/extensions/gsd/tools/plan-slice.ts +172 -12
  571. package/src/resources/extensions/gsd/tools/validate-milestone.ts +31 -0
  572. package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +166 -17
  573. package/src/resources/extensions/gsd/types.ts +1 -1
  574. package/src/resources/extensions/gsd/unit-context-composer.ts +3 -0
  575. package/src/resources/extensions/gsd/unit-context-manifest.ts +86 -19
  576. package/src/resources/extensions/gsd/validation.ts +23 -1
  577. package/src/resources/extensions/gsd/verification-gate.ts +170 -6
  578. package/src/resources/extensions/gsd/verification-verdict.ts +47 -0
  579. package/src/resources/extensions/gsd/workflow-manifest.ts +2 -0
  580. package/src/resources/extensions/gsd/workflow-mcp.ts +18 -1
  581. package/src/resources/extensions/gsd/workflow-projections.ts +6 -8
  582. package/src/resources/extensions/gsd/worktree-lifecycle.ts +98 -20
  583. package/src/resources/extensions/gsd/worktree-manager.ts +14 -2
  584. package/src/resources/extensions/gsd/worktree-safety.ts +57 -10
  585. package/src/resources/extensions/gsd/worktree-state-projection.ts +43 -0
  586. package/src/resources/extensions/gsd/worktree-telemetry.ts +39 -0
  587. package/src/resources/extensions/shared/html-shell.ts +412 -0
  588. package/src/resources/extensions/shared/interview-ui.ts +6 -4
  589. package/src/resources/extensions/shared/next-action-ui.ts +11 -5
  590. package/src/resources/extensions/shared/tests/interview-notes-loop.test.ts +15 -0
  591. package/src/resources/extensions/shared/tests/next-action-ui-hasui.test.ts +32 -0
  592. package/src/resources/extensions/subagent/index.ts +567 -103
  593. package/src/resources/extensions/subagent/launch.ts +131 -0
  594. package/src/resources/extensions/subagent/run-store.ts +218 -0
  595. package/src/resources/extensions/subagent/tests/launch.test.ts +115 -0
  596. package/src/resources/extensions/subagent/tests/run-store.test.ts +111 -0
  597. package/src/resources/extensions/ttsr/ttsr-manager.ts +5 -1
  598. package/src/resources/extensions/visual-brief/artifact-policy.ts +41 -0
  599. package/src/resources/extensions/visual-brief/extension-manifest.json +8 -0
  600. package/src/resources/extensions/visual-brief/index.ts +8 -0
  601. package/src/resources/extensions/visual-brief/page-contract.ts +136 -0
  602. package/src/resources/extensions/visual-brief/prompts.ts +183 -0
  603. package/src/resources/extensions/visual-brief/tests/visual-brief.test.ts +212 -0
  604. package/src/resources/skills/forensics/SKILL.md +1 -1
  605. package/dist/web/standalone/.next/server/chunks/5822.js +0 -2
  606. package/dist/web/standalone/.next/static/chunks/2556.0527fea66e123b7f.js +0 -1
  607. package/dist/web/standalone/.next/static/chunks/8359.e059d86b255fce1c.js +0 -10
  608. package/dist/web/standalone/.next/static/chunks/9441.1081da1125d1764f.js +0 -1
  609. package/dist/web/standalone/.next/static/chunks/app/layout-a16c7a7ecdf0c2cf.js +0 -1
  610. package/dist/web/standalone/.next/static/css/54ec2745c1da488b.css +0 -1
  611. package/dist/web/standalone/.next/static/css/de70bee13400563f.css +0 -1
  612. package/dist/web/standalone/.next/static/media/4cf2300e9c8272f7-s.p.woff2 +0 -0
  613. package/dist/web/standalone/.next/static/media/747892c23ea88013-s.woff2 +0 -0
  614. package/dist/web/standalone/.next/static/media/8d697b304b401681-s.woff2 +0 -0
  615. package/dist/web/standalone/.next/static/media/93f479601ee12b01-s.p.woff2 +0 -0
  616. package/dist/web/standalone/.next/static/media/9610d9e46709d722-s.woff2 +0 -0
  617. package/dist/web/standalone/.next/static/media/ba015fad6dcf6784-s.woff2 +0 -0
  618. /package/dist/web/standalone/.next/static/{YEvjuT-fsFfYQhDSWtueS → Z8H5evS-hDo0qdP22XJPA}/_buildManifest.js +0 -0
  619. /package/dist/web/standalone/.next/static/{YEvjuT-fsFfYQhDSWtueS → Z8H5evS-hDo0qdP22XJPA}/_ssgManifest.js +0 -0
@@ -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));
@@ -91,8 +91,9 @@ test("buildReactiveExecutePrompt injects subagent model when provided", async (t
91
91
  );
92
92
 
93
93
  assert.match(prompt, /model: "claude-opus-4-6"/);
94
- assert.match(prompt, /Context Mode \(execution lane\):/);
94
+ assert.match(prompt, /Lane: \*\*execution lane\*\*\./);
95
95
  assert.match(prompt, /## Context Mode/);
96
+ assert.doesNotMatch(prompt, /You are executing GSD auto-mode\./);
96
97
  });
97
98
 
98
99
  test("buildReactiveExecutePrompt omits model instruction when subagentModel is omitted", async (t) => {
@@ -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
 
@@ -153,6 +153,7 @@ const laneLabelByMode: Record<string, string> = {
153
153
  verification: "verification",
154
154
  orchestration: "orchestration",
155
155
  docs: "documentation",
156
+ triage: "triage",
156
157
  };
157
158
 
158
159
  test("Context Mode composer: every known eligible unit renders its configured lane and required tools", () => {
@@ -7,13 +7,17 @@ import {
7
7
  ARTIFACT_KEYS,
8
8
  KNOWN_UNIT_TYPES,
9
9
  UNIT_MANIFESTS,
10
+ resolveSubagentPermissionContract,
10
11
  resolveManifest,
11
12
  type ArtifactKey,
12
13
  type ContextModePolicy,
13
14
  type SkillsPolicy,
14
15
  type UnitContextManifest,
15
16
  } from "../unit-context-manifest.ts";
16
- 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";
17
21
  import {
18
22
  getRequiredWorkflowToolsForAutoUnit,
19
23
  getRequiredWorkflowToolsForGuidedUnit,
@@ -106,8 +110,10 @@ test("Context Mode: every manifest declares the expected contextMode lane", () =
106
110
  "reassess-roadmap": "planning",
107
111
  "execute-task": "execution",
108
112
  "reactive-execute": "execution",
113
+ "quick-task": "execution",
109
114
  "run-uat": "verification",
110
115
  "gate-evaluate": "verification",
116
+ "triage-captures": "triage",
111
117
  "validate-milestone": "verification",
112
118
  "complete-slice": "verification",
113
119
  "complete-milestone": "verification",
@@ -216,7 +222,7 @@ test("#4934: every manifest declares a tools policy", () => {
216
222
  });
217
223
 
218
224
  test("#4934: tools.mode is one of the declared policies", () => {
219
- const validModes = new Set(["all", "read-only", "planning", "planning-dispatch", "docs"]);
225
+ const validModes = new Set(["all", "read-only", "planning", "planning-dispatch", "docs", "verification"]);
220
226
  for (const [unitType, manifest] of Object.entries(UNIT_MANIFESTS)) {
221
227
  const mode = (manifest as { tools: { mode: string } }).tools.mode;
222
228
  assert.ok(
@@ -226,32 +232,130 @@ test("#4934: tools.mode is one of the declared policies", () => {
226
232
  }
227
233
  });
228
234
 
229
- test('#4934: only execute-task and reactive-execute may use tools.mode "all" (full source-tree write access)', () => {
230
- const allowedAllUnits = new Set(["execute-task", "reactive-execute"]);
235
+ test('#4934: only execution units, quick-task, and closeout units may use tools.mode "all"', () => {
236
+ const allowedAllUnits = new Set([
237
+ "execute-task",
238
+ "reactive-execute",
239
+ "quick-task",
240
+ "validate-milestone",
241
+ "complete-milestone",
242
+ "complete-slice",
243
+ ]);
231
244
  for (const [unitType, manifest] of Object.entries(UNIT_MANIFESTS)) {
232
245
  const mode = (manifest as { tools: { mode: string } }).tools.mode;
233
246
  if (mode === "all") {
234
247
  assert.ok(
235
248
  allowedAllUnits.has(unitType),
236
- `manifest "${unitType}" declares tools.mode = "all" but is not on the execute-track. ` +
237
- 'Only execute-task and reactive-execute should have full source write access; ' +
249
+ `manifest "${unitType}" declares tools.mode = "all" but is not explicitly allowed. ` +
250
+ 'Only execute-task/reactive-execute, quick-task, and closeout units should have full source write access; ' +
238
251
  'planning/discuss/research units must use "planning" or "planning-dispatch" (or "docs" for rewrite-docs).',
239
252
  );
240
253
  }
241
254
  }
242
255
  });
243
256
 
257
+ test("#5453: complete-milestone uses all tools so bash verification is not planning-dispatch blocked", () => {
258
+ const manifest = UNIT_MANIFESTS["complete-milestone"];
259
+
260
+ assert.strictEqual(manifest.tools.mode, "all");
261
+ assert.deepEqual(resolveSubagentPermissionContract("complete-milestone"), {
262
+ allowed: true,
263
+ allowedSubagents: ["*"],
264
+ toolsMode: "all",
265
+ });
266
+ // Runtime gate-level regression: these verification commands were blocked
267
+ // under planning-dispatch in #5453; complete-milestone must bypass that gate.
268
+ for (const cmd of ["git diff --name-only HEAD~1", "git log -n1 --oneline"]) {
269
+ const result = shouldBlockPlanningUnit(
270
+ "bash",
271
+ cmd,
272
+ process.cwd(),
273
+ "complete-milestone",
274
+ manifest.tools,
275
+ );
276
+ assert.strictEqual(
277
+ result.block,
278
+ false,
279
+ `shouldBlockPlanningUnit must not block ${cmd} for complete-milestone: ${result.reason}`,
280
+ );
281
+ }
282
+ });
283
+
284
+ test("#5731: validate-milestone and complete-slice use all tools so closeout verification is not planning-dispatch blocked", () => {
285
+ for (const unitType of ["validate-milestone", "complete-slice"] as const) {
286
+ const manifest = UNIT_MANIFESTS[unitType];
287
+ assert.strictEqual(manifest.tools.mode, "all");
288
+ const result = shouldBlockPlanningUnit(
289
+ "bash",
290
+ "go test -short -count=1 ./...",
291
+ process.cwd(),
292
+ unitType,
293
+ manifest.tools,
294
+ );
295
+ assert.strictEqual(
296
+ result.block,
297
+ false,
298
+ `${unitType} must allow verification bash commands: ${result.reason}`,
299
+ );
300
+ }
301
+ });
302
+
303
+ test("#5843: run-uat uses verification tools policy so build/test commands can run", () => {
304
+ const manifest = UNIT_MANIFESTS["run-uat"];
305
+
306
+ assert.strictEqual(manifest.tools.mode, "verification");
307
+
308
+ const buildResult = shouldBlockPlanningUnit(
309
+ "bash",
310
+ "npm run build 2>&1",
311
+ process.cwd(),
312
+ "run-uat",
313
+ manifest.tools,
314
+ );
315
+ assert.strictEqual(
316
+ buildResult.block,
317
+ false,
318
+ `run-uat must allow build verification commands: ${buildResult.reason}`,
319
+ );
320
+
321
+ const sourceWriteResult = shouldBlockPlanningUnit(
322
+ "edit",
323
+ "src/main.ts",
324
+ process.cwd(),
325
+ "run-uat",
326
+ manifest.tools,
327
+ );
328
+ assert.strictEqual(sourceWriteResult.block, true);
329
+ assert.match(sourceWriteResult.reason!, /tools-policy "verification"/);
330
+ });
331
+
332
+ test("planning-dispatch hard block message omits internal tracker references", () => {
333
+ const manifest = UNIT_MANIFESTS["plan-slice"];
334
+ assert.strictEqual(manifest.tools.mode, "planning-dispatch");
335
+
336
+ const result = shouldBlockPlanningUnit(
337
+ "task",
338
+ "scout",
339
+ process.cwd(),
340
+ "plan-slice",
341
+ manifest.tools,
342
+ );
343
+
344
+ assert.strictEqual(result.block, true);
345
+ assert.ok(result.reason, "blocked dispatch should include a user-facing reason");
346
+ assert.doesNotMatch(result.reason!, /#[0-9]{3,}/);
347
+ });
348
+
244
349
  test('planning-dispatch mode is reserved for slice-level decomposition and completion units', () => {
245
350
  const allowedDispatchUnits = new Set([
246
351
  "plan-slice",
352
+ "research-slice",
247
353
  "refine-slice",
248
- "complete-slice",
249
- "complete-milestone",
354
+ "gate-evaluate",
250
355
  // Deep planning mode: research-project orchestrates 4 parallel research
251
356
  // subagents (stack/features/architecture/pitfalls). Subagent dispatch is
252
357
  // the unit's core mechanism — without it, the unit cannot do its job.
253
358
  "research-project",
254
- "validate-milestone",
255
359
  ]);
256
360
  for (const [unitType, manifest] of Object.entries(UNIT_MANIFESTS)) {
257
361
  const mode = (manifest as { tools: { mode: string } }).tools.mode;
@@ -265,6 +369,24 @@ test('planning-dispatch mode is reserved for slice-level decomposition and compl
265
369
  }
266
370
  });
267
371
 
372
+ test('Unit Tool Contract exposes subagent dispatch permissions', () => {
373
+ assert.deepEqual(resolveSubagentPermissionContract("plan-slice"), {
374
+ allowed: true,
375
+ allowedSubagents: ["scout", "planner"],
376
+ toolsMode: "planning-dispatch",
377
+ });
378
+ assert.deepEqual(resolveSubagentPermissionContract("gate-evaluate"), {
379
+ allowed: true,
380
+ allowedSubagents: ["reviewer", "security", "tester"],
381
+ toolsMode: "planning-dispatch",
382
+ });
383
+ assert.deepEqual(resolveSubagentPermissionContract("discuss-milestone"), {
384
+ allowed: false,
385
+ allowedSubagents: [],
386
+ toolsMode: "planning",
387
+ });
388
+ });
389
+
268
390
  test('planning-dispatch manifests declare non-empty allowedSubagents lists', () => {
269
391
  for (const [unitType, manifest] of Object.entries(UNIT_MANIFESTS)) {
270
392
  if (manifest.tools.mode !== "planning-dispatch") continue;
@@ -1,6 +1,6 @@
1
1
  import test from "node:test";
2
2
  import assert from "node:assert/strict";
3
- import { mkdtempSync, rmSync, writeFileSync } from "node:fs";
3
+ import { mkdtempSync, mkdirSync, rmSync, writeFileSync } from "node:fs";
4
4
  import { tmpdir } from "node:os";
5
5
  import { join } from "node:path";
6
6
  import { execSync } from "node:child_process";
@@ -12,6 +12,10 @@ function run(cmd: string, cwd: string): string {
12
12
 
13
13
  function makeRepo(): string {
14
14
  const repo = mkdtempSync(join(tmpdir(), "gsd-uok-gitops-"));
15
+ return initRepo(repo);
16
+ }
17
+
18
+ function initRepo(repo: string): string {
15
19
  run("git init", repo);
16
20
  run('git config user.email "test@example.com"', repo);
17
21
  run('git config user.name "Test User"', repo);
@@ -32,6 +36,7 @@ test("uok gitops turn action status-only reports working tree dirtiness", () =>
32
36
  });
33
37
  assert.equal(clean.status, "ok");
34
38
  assert.equal(clean.dirty, false);
39
+ assert.deepEqual(clean.dirtyRepositories, { project: false });
35
40
 
36
41
  writeFileSync(join(repo, "README.md"), "# Dirty\n", "utf-8");
37
42
  const dirty = runTurnGitAction({
@@ -42,11 +47,57 @@ test("uok gitops turn action status-only reports working tree dirtiness", () =>
42
47
  });
43
48
  assert.equal(dirty.status, "ok");
44
49
  assert.equal(dirty.dirty, true);
50
+ assert.deepEqual(dirty.dirtyRepositories, { project: true });
45
51
  } finally {
46
52
  rmSync(repo, { recursive: true, force: true });
47
53
  }
48
54
  });
49
55
 
56
+ test("uok gitops turn action status-only reports per-repository dirtiness in parent mode", () => {
57
+ const root = mkdtempSync(join(tmpdir(), "gsd-uok-gitops-parent-"));
58
+ try {
59
+ initRepo(root);
60
+ mkdirSync(join(root, ".gsd"), { recursive: true });
61
+ mkdirSync(join(root, "frontend"), { recursive: true });
62
+ mkdirSync(join(root, "backend"), { recursive: true });
63
+ initRepo(join(root, "frontend"));
64
+ initRepo(join(root, "backend"));
65
+ writeFileSync(join(root, ".gitignore"), "frontend/\nbackend/\n", "utf-8");
66
+ run("git add .gitignore", root);
67
+ run('git commit -m "chore: ignore nested repos"', root);
68
+ writeFileSync(join(root, ".gsd", "PREFERENCES.md"), `---
69
+ version: 1
70
+ workspace:
71
+ mode: parent
72
+ repositories:
73
+ frontend:
74
+ path: frontend
75
+ backend:
76
+ path: backend
77
+ ---
78
+ `, "utf-8");
79
+ run("git add .gsd/PREFERENCES.md", root);
80
+ run('git commit -m "chore: configure parent workspace repos"', root);
81
+
82
+ writeFileSync(join(root, "frontend", "README.md"), "# Dirty frontend\n", "utf-8");
83
+
84
+ const result = runTurnGitAction({
85
+ basePath: root,
86
+ action: "status-only",
87
+ unitType: "execute-task",
88
+ unitId: "M001/S01/T01",
89
+ });
90
+
91
+ assert.equal(result.status, "ok");
92
+ assert.equal(result.dirty, true);
93
+ assert.equal(result.dirtyRepositories?.project, false);
94
+ assert.equal(result.dirtyRepositories?.frontend, true);
95
+ assert.equal(result.dirtyRepositories?.backend, false);
96
+ } finally {
97
+ rmSync(root, { recursive: true, force: true });
98
+ }
99
+ });
100
+
50
101
  test("uok gitops turn action snapshot writes snapshot refs", () => {
51
102
  const repo = makeRepo();
52
103
  try {
@@ -84,6 +135,55 @@ test("uok gitops turn action commit creates commit with unit trailer", () => {
84
135
  }
85
136
  });
86
137
 
138
+ test("uok gitops turn action commit honors per-repo commit_policy skip", () => {
139
+ const root = mkdtempSync(join(tmpdir(), "gsd-uok-gitops-commit-policy-"));
140
+ try {
141
+ initRepo(root);
142
+ mkdirSync(join(root, ".gsd"), { recursive: true });
143
+ mkdirSync(join(root, "frontend"), { recursive: true });
144
+ mkdirSync(join(root, "backend"), { recursive: true });
145
+ initRepo(join(root, "frontend"));
146
+ initRepo(join(root, "backend"));
147
+ writeFileSync(join(root, ".gitignore"), "frontend/\nbackend/\n", "utf-8");
148
+ run("git add .gitignore", root);
149
+ run('git commit -m "chore: ignore nested repos"', root);
150
+ writeFileSync(join(root, ".gsd", "PREFERENCES.md"), `---
151
+ version: 1
152
+ workspace:
153
+ mode: parent
154
+ repositories:
155
+ frontend:
156
+ path: frontend
157
+ commit_policy: skip
158
+ backend:
159
+ path: backend
160
+ ---
161
+ `, "utf-8");
162
+ run("git add .gsd/PREFERENCES.md", root);
163
+ run('git commit -m "chore: configure commit policies"', root);
164
+
165
+ writeFileSync(join(root, "frontend", "README.md"), "# frontend dirty\n", "utf-8");
166
+ writeFileSync(join(root, "backend", "README.md"), "# backend dirty\n", "utf-8");
167
+
168
+ const result = runTurnGitAction({
169
+ basePath: root,
170
+ action: "commit",
171
+ unitType: "execute-task",
172
+ unitId: "M001/S01/T03",
173
+ targetRepositories: ["frontend", "backend"],
174
+ });
175
+
176
+ assert.equal(result.status, "ok");
177
+ assert.deepEqual(result.skippedRepositories, ["frontend"]);
178
+ assert.equal(typeof result.commitMessages?.backend, "string");
179
+ assert.equal(result.commitMessages?.frontend, undefined);
180
+ assert.equal(run("git status --porcelain", join(root, "frontend")).length > 0, true);
181
+ assert.equal(run("git status --porcelain", join(root, "backend")), "");
182
+ } finally {
183
+ rmSync(root, { recursive: true, force: true });
184
+ }
185
+ });
186
+
87
187
  test("uok gitops turn action rethrows infrastructure failures", () => {
88
188
  const err = Object.assign(new Error("ENFILE: file table overflow"), { code: "ENFILE" });
89
189
 
@@ -97,3 +197,13 @@ test("uok gitops turn action keeps non-infrastructure git failures recoverable",
97
197
  assert.equal(result.status, "failed");
98
198
  assert.equal(result.error, "nothing to commit");
99
199
  });
200
+
201
+ test("uok gitops turn action prefers stderr details for git failures", () => {
202
+ const err = Object.assign(new Error("Command failed: git commit -F -"), {
203
+ stderr: "fatal: unable to auto-detect email address",
204
+ });
205
+
206
+ const result = handleTurnGitActionError("commit", err);
207
+ assert.equal(result.status, "failed");
208
+ assert.equal(result.error, "fatal: unable to auto-detect email address");
209
+ });
@@ -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
  });
@@ -150,4 +150,72 @@ describe("handleValidateMilestone write ordering (#2725)", () => {
150
150
  assert.equal(row?.trace_id, "trace-val-1");
151
151
  assert.equal(row?.turn_id, "turn-val-1");
152
152
  });
153
+
154
+ it("rejects verificationClasses that omit planned Operational class", async () => {
155
+ base = makeTmpBase();
156
+ const dbPath = join(base, ".gsd", "gsd.db");
157
+ openDatabase(dbPath);
158
+ insertMilestone({
159
+ id: "M001",
160
+ planning: {
161
+ verificationOperational: "Camoufox subprocess lifecycle/cleanup proof",
162
+ },
163
+ });
164
+ insertSlice({ id: "S01", milestoneId: "M001" });
165
+
166
+ const result = await handleValidateMilestone(
167
+ { ...VALID_PARAMS, verificationClasses: "| Check | Result |\n| --- | --- |\n| Generic verification | PASS |" },
168
+ base,
169
+ );
170
+ assert.ok("error" in result, "expected validation to fail");
171
+ assert.match(result.error, /must include canonical row "Operational"/);
172
+
173
+ const adapter = _getAdapter()!;
174
+ const row = adapter.prepare(
175
+ `SELECT status FROM assessments WHERE milestone_id = 'M001' AND scope = 'milestone-validation'`,
176
+ ).get() as { status: string } | undefined;
177
+ assert.equal(row, undefined, "assessment row should not be written when verification classes are invalid");
178
+ });
179
+
180
+ it("accepts verificationClasses when planned Operational class is present", async () => {
181
+ base = makeTmpBase();
182
+ const dbPath = join(base, ".gsd", "gsd.db");
183
+ openDatabase(dbPath);
184
+ insertMilestone({
185
+ id: "M001",
186
+ planning: {
187
+ verificationOperational: "Camoufox subprocess lifecycle/cleanup proof",
188
+ },
189
+ });
190
+ insertSlice({ id: "S01", milestoneId: "M001" });
191
+
192
+ const result = await handleValidateMilestone(
193
+ {
194
+ ...VALID_PARAMS,
195
+ verificationClasses:
196
+ "| Class | Planned Check | Evidence | Verdict |\n| --- | --- | --- | --- |\n| Operational | Camoufox subprocess lifecycle/cleanup proof | S01 + process-death evidence | NEEDS-ATTENTION |",
197
+ },
198
+ base,
199
+ );
200
+ assert.ok(!("error" in result), `unexpected error: ${"error" in result ? result.error : ""}`);
201
+ });
202
+
203
+ it("treats 'not required - ...' verification values as not applicable", async () => {
204
+ base = makeTmpBase();
205
+ const dbPath = join(base, ".gsd", "gsd.db");
206
+ openDatabase(dbPath);
207
+ insertMilestone({
208
+ id: "M001",
209
+ planning: {
210
+ verificationOperational: "not required - backend-only",
211
+ },
212
+ });
213
+ insertSlice({ id: "S01", milestoneId: "M001" });
214
+
215
+ const result = await handleValidateMilestone(
216
+ { ...VALID_PARAMS, verificationClasses: "| Check | Result |\n| --- | --- |\n| Generic verification | PASS |" },
217
+ base,
218
+ );
219
+ assert.ok(!("error" in result), `unexpected error: ${"error" in result ? result.error : ""}`);
220
+ });
153
221
  });
@@ -13,7 +13,7 @@ import { buildCompleteMilestonePrompt, buildValidateMilestonePrompt } from "../a
13
13
  import type { GSDState } from "../types.ts";
14
14
  import { clearPathCache } from "../paths.ts";
15
15
  import { clearParseCache } from "../files.ts";
16
- import { closeDatabase, insertMilestone, insertSlice, openDatabase, getMilestone } from "../gsd-db.ts";
16
+ import { closeDatabase, insertAssessment, insertMilestone, insertSlice, openDatabase, getMilestone } from "../gsd-db.ts";
17
17
 
18
18
  // ─── Helpers ──────────────────────────────────────────────────────────────
19
19
 
@@ -205,6 +205,47 @@ test("deriveState returns blocked when needs-remediation has no incomplete slice
205
205
  }
206
206
  });
207
207
 
208
+ test("deriveState parks milestone when validation verdict is needs-attention and no summary (#6136)", async () => {
209
+ const base = makeTmpBase();
210
+ try {
211
+ writeRoadmap(base, "M001", ALL_DONE_ROADMAP);
212
+ writeValidation(base, "M001", "---\nverdict: needs-attention\nremediation_round: 0\n---\n\n# Validation\nNeeds attention.");
213
+
214
+ const state = await deriveState(base);
215
+ assert.equal(state.phase, "pre-planning");
216
+ assert.equal(state.activeMilestone, null);
217
+ assert.equal(state.registry.find(entry => entry.id === "M001")?.status, "parked");
218
+ assert.ok(state.nextAction.includes("parked"));
219
+ } finally {
220
+ cleanup(base);
221
+ }
222
+ });
223
+
224
+ test("deriveState parks DB-backed milestone when validation verdict is needs-attention (#6136)", async () => {
225
+ const base = makeTmpBase();
226
+ try {
227
+ openTestDb(base);
228
+ insertMilestone({ id: "M001", title: "Test Milestone", status: "active" });
229
+ insertSlice({ id: "S01", milestoneId: "M001", title: "First slice", status: "complete", depends: [] });
230
+ writeRoadmap(base, "M001", ALL_DONE_ROADMAP);
231
+ insertAssessment({
232
+ path: join(".gsd", "milestones", "M001", "M001-VALIDATION.md"),
233
+ milestoneId: "M001",
234
+ status: "needs-attention",
235
+ scope: "milestone-validation",
236
+ fullContent: "---\nverdict: needs-attention\nremediation_round: 0\n---\n\n# Validation\nNeeds attention.",
237
+ });
238
+
239
+ const state = await deriveState(base);
240
+ assert.equal(state.phase, "pre-planning");
241
+ assert.equal(state.activeMilestone, null);
242
+ assert.equal(state.registry.find(entry => entry.id === "M001")?.status, "parked");
243
+ assert.ok(state.nextAction.includes("parked"));
244
+ } finally {
245
+ cleanup(base);
246
+ }
247
+ });
248
+
208
249
  test("deriveState returns complete when both VALIDATION and SUMMARY exist", async () => {
209
250
  const base = makeTmpBase();
210
251
  try {