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
@@ -11,8 +11,15 @@ const source = readFileSync(
11
11
 
12
12
  test("postUnitPreVerification blocks on git action failure", () => {
13
13
  const failureBlock = extractSourceRegion(source, 'if (gitResult.status === "failed")');
14
- assert.ok(failureBlock.includes('ctx.ui.notify(failureMsg, "error")'));
14
+ assert.ok(failureBlock.includes('ctx.ui.notify(failureMsg, opts?.softFailure ? "warning" : "error")'));
15
15
  assert.ok(failureBlock.includes("await pauseAuto(ctx, pi)"));
16
16
  assert.ok(failureBlock.includes('return "dispatched"'));
17
17
  assert.ok(!failureBlock.includes("git-action-failed-nonblocking"));
18
18
  });
19
+
20
+ test("buildTaskCommitContextForUnit filters placeholder key_files entries", () => {
21
+ const keyFilesBlock = extractSourceRegion(source, "keyFiles:");
22
+ assert.ok(keyFilesBlock.includes("normalized.length > 0"));
23
+ assert.ok(keyFilesBlock.includes("!normalized.includes(\"{{\")"));
24
+ assert.ok(keyFilesBlock.includes("/^(?:\\(none\\)|none\\.?|n\\/a)$/i.test(normalized)"));
25
+ });
@@ -8,8 +8,32 @@ import assert from "node:assert/strict";
8
8
  import { existsSync, mkdirSync, mkdtempSync, readFileSync, rmSync, writeFileSync } from "node:fs";
9
9
  import { join } from "node:path";
10
10
  import { tmpdir } from "node:os";
11
+ import { createRequire } from "node:module";
11
12
  import { AutoSession } from "../auto/session.ts";
12
13
  import { postUnitPreVerification } from "../auto-post-unit.ts";
14
+ import {
15
+ _getAdapter,
16
+ closeDatabase,
17
+ getTask,
18
+ insertMilestone,
19
+ insertSlice,
20
+ insertTask,
21
+ openDatabase,
22
+ } from "../gsd-db.ts";
23
+
24
+ const _require = createRequire(import.meta.url);
25
+
26
+ function openRawSqliteForTest(dbPath: string): { exec(sql: string): void; close(): void } {
27
+ try {
28
+ const mod = _require("node:sqlite") as { DatabaseSync: new (path: string) => { exec(sql: string): void; close(): void } };
29
+ return new mod.DatabaseSync(dbPath);
30
+ } catch {
31
+ type SqliteCtor = new (path: string) => { exec(sql: string): void; close(): void };
32
+ const mod = _require("better-sqlite3") as SqliteCtor | { default: SqliteCtor };
33
+ const DatabaseCtor: SqliteCtor = typeof mod === "function" ? mod : mod.default;
34
+ return new DatabaseCtor(dbPath);
35
+ }
36
+ }
13
37
 
14
38
  test("postUnitPreVerification rebuilds STATE.md after a completed unit", async () => {
15
39
  const base = mkdtempSync(join(tmpdir(), "gsd-post-unit-state-"));
@@ -47,3 +71,63 @@ test("postUnitPreVerification rebuilds STATE.md after a completed unit", async (
47
71
  rmSync(base, { recursive: true, force: true });
48
72
  }
49
73
  });
74
+
75
+ test("postUnitPreVerification refreshes DB before checking execute-task completion", async () => {
76
+ const base = mkdtempSync(join(tmpdir(), "gsd-post-unit-db-refresh-"));
77
+ try {
78
+ const sliceDir = join(base, ".gsd", "milestones", "M001", "slices", "S01");
79
+ const tasksDir = join(sliceDir, "tasks");
80
+ mkdirSync(tasksDir, { recursive: true });
81
+ writeFileSync(
82
+ join(base, ".gsd", "milestones", "M001", "M001-ROADMAP.md"),
83
+ "# Roadmap\n\n## Slices\n\n- [ ] **S01: Slice** `risk:low` `depends:[]`\n",
84
+ );
85
+ writeFileSync(
86
+ join(sliceDir, "S01-PLAN.md"),
87
+ "# S01: Slice\n\n## Tasks\n\n- [ ] **T01: Do work** `est:30m`\n",
88
+ );
89
+ writeFileSync(
90
+ join(tasksDir, "T01-SUMMARY.md"),
91
+ "---\nid: T01\nparent: S01\nmilestone: M001\n---\n# T01\nDone.\n",
92
+ );
93
+
94
+ const dbPath = join(base, ".gsd", "gsd.db");
95
+ openDatabase(dbPath);
96
+ insertMilestone({ id: "M001", title: "Milestone", status: "active" });
97
+ insertSlice({ id: "S01", milestoneId: "M001", title: "Slice", status: "pending" });
98
+ insertTask({ id: "T01", sliceId: "S01", milestoneId: "M001", title: "Do work", status: "pending" });
99
+ const adapterBefore = _getAdapter();
100
+
101
+ const externalDb = openRawSqliteForTest(dbPath);
102
+ try {
103
+ externalDb.exec("UPDATE tasks SET status = 'complete', completed_at = '2026-05-14T00:00:00.000Z' WHERE milestone_id = 'M001' AND slice_id = 'S01' AND id = 'T01'");
104
+ } finally {
105
+ externalDb.close();
106
+ }
107
+
108
+ const s = new AutoSession();
109
+ s.basePath = base;
110
+ s.originalBasePath = base;
111
+ s.currentMilestoneId = "M001";
112
+ s.currentUnit = { type: "execute-task", id: "M001/S01/T01", startedAt: Date.now() };
113
+
114
+ const result = await postUnitPreVerification({
115
+ s,
116
+ ctx: { ui: { notify() {} } } as any,
117
+ pi: {} as any,
118
+ buildSnapshotOpts: () => ({}),
119
+ lockBase: () => base,
120
+ stopAuto: async () => {},
121
+ pauseAuto: async () => {},
122
+ updateProgressWidget: () => {},
123
+ }, { skipSettleDelay: true, skipWorktreeSync: true });
124
+
125
+ assert.equal(result, "continue");
126
+ assert.notEqual(_getAdapter(), adapterBefore, "post-unit flow must reopen the DB before deriving state");
127
+ assert.equal(getTask("M001", "S01", "T01")?.status, "complete");
128
+ assert.equal(s.pendingVerificationRetry, null);
129
+ } finally {
130
+ closeDatabase();
131
+ rmSync(base, { recursive: true, force: true });
132
+ }
133
+ });
@@ -9,6 +9,7 @@
9
9
  * 2. File path consistency — files exist vs prior expected_output
10
10
  * 3. Task ordering — detect impossible read-before-create
11
11
  * 4. Interface contracts — contradictory function signatures
12
+ * 5. Verify commands — reject unsafe or non-runnable task verification
12
13
  */
13
14
 
14
15
  import { describe, test, mock } from "node:test";
@@ -22,6 +23,7 @@ import {
22
23
  checkFilePathConsistency,
23
24
  checkTaskOrdering,
24
25
  checkInterfaceContracts,
26
+ checkVerificationCommands,
25
27
  runPreExecutionChecks,
26
28
  normalizeFilePath,
27
29
  type PreExecutionResult,
@@ -126,6 +128,12 @@ import type { Request } from 'express';
126
128
  assert.deepEqual(packages, []);
127
129
  });
128
130
 
131
+ test("ignores @/ path alias imports", () => {
132
+ const desc = `import { handler } from '@/app/api/hello/route';`;
133
+ const packages = extractPackageReferences(desc);
134
+ assert.deepEqual(packages, []);
135
+ });
136
+
129
137
  test("normalizes package subpaths", () => {
130
138
  const desc = "npm install lodash/get";
131
139
  const packages = extractPackageReferences(desc);
@@ -415,6 +423,40 @@ describe("checkFilePathConsistency with path normalization", () => {
415
423
  rmSync(tempDir, { recursive: true, force: true });
416
424
  }
417
425
  });
426
+
427
+ test("absolute input path matches relative expected_output within basePath (#5519)", () => {
428
+ tempDir = join(tmpdir(), `pre-exec-test-${Date.now()}`);
429
+ mkdirSync(tempDir, { recursive: true });
430
+
431
+ try {
432
+ const absGenerated = join(tempDir, "src/new-file.ts");
433
+ const tasks = [
434
+ createTask({
435
+ id: "T01",
436
+ sequence: 0,
437
+ files: [],
438
+ inputs: [],
439
+ expected_output: ["src/new-file.ts"],
440
+ }),
441
+ createTask({
442
+ id: "T02",
443
+ sequence: 1,
444
+ files: [],
445
+ inputs: [absGenerated],
446
+ expected_output: [],
447
+ }),
448
+ ];
449
+
450
+ const results = checkFilePathConsistency(tasks, tempDir);
451
+ assert.deepEqual(
452
+ results,
453
+ [],
454
+ "Should pass because absolute/relative paths under basePath must compare equal",
455
+ );
456
+ } finally {
457
+ rmSync(tempDir, { recursive: true, force: true });
458
+ }
459
+ });
418
460
  });
419
461
 
420
462
  describe("checkTaskOrdering with path normalization", () => {
@@ -486,6 +528,38 @@ describe("checkTaskOrdering with path normalization", () => {
486
528
  const results = checkTaskOrdering(tasks, "/tmp");
487
529
  assert.deepEqual(results, [], "Should pass - T02 reads file that T01 already created");
488
530
  });
531
+
532
+ test("absolute input matches later relative expected_output and triggers ordering violation (#5519)", () => {
533
+ const tempDir = join(tmpdir(), `pre-exec-ordering-abs-rel-${Date.now()}`);
534
+ mkdirSync(tempDir, { recursive: true });
535
+
536
+ try {
537
+ const tasks = [
538
+ createTask({
539
+ id: "T01",
540
+ sequence: 0,
541
+ files: [],
542
+ inputs: [join(tempDir, "src/new-file.ts")],
543
+ expected_output: [],
544
+ }),
545
+ createTask({
546
+ id: "T02",
547
+ sequence: 1,
548
+ files: [],
549
+ inputs: [],
550
+ expected_output: ["src/new-file.ts"],
551
+ }),
552
+ ];
553
+
554
+ const results = checkTaskOrdering(tasks, tempDir);
555
+ assert.equal(results.length, 1, "Should detect violation for equivalent absolute/relative paths");
556
+ assert.ok(results[0].message.includes("sequence violation"));
557
+ assert.ok(results[0].message.includes("T01"));
558
+ assert.ok(results[0].message.includes("T02"));
559
+ } finally {
560
+ rmSync(tempDir, { recursive: true, force: true });
561
+ }
562
+ });
489
563
  });
490
564
 
491
565
  // ─── Task Ordering Tests ─────────────────────────────────────────────────────
@@ -812,6 +886,33 @@ function process(a: number): number
812
886
  });
813
887
  });
814
888
 
889
+ describe("checkVerificationCommands", () => {
890
+ test("accepts pipe-free pytest Verify command", () => {
891
+ const results = checkVerificationCommands([
892
+ createTask({
893
+ id: "T01",
894
+ verify: "python3 -m pytest tests/ -q --tb=short",
895
+ }),
896
+ ]);
897
+
898
+ assert.deepEqual(results, []);
899
+ });
900
+
901
+ test("rejects piped pytest Verify command", () => {
902
+ const results = checkVerificationCommands([
903
+ createTask({
904
+ id: "T01",
905
+ verify: "python3 -m pytest tests/ -q --tb=short 2>&1 | tail -5",
906
+ }),
907
+ ]);
908
+
909
+ assert.equal(results.length, 1);
910
+ assert.equal(results[0]?.category, "tool");
911
+ assert.equal(results[0]?.blocking, true);
912
+ assert.match(results[0]?.message ?? "", /shell control syntax/);
913
+ });
914
+ });
915
+
815
916
  // ─── runPreExecutionChecks Integration Tests ─────────────────────────────────
816
917
 
817
918
  describe("runPreExecutionChecks", () => {
@@ -847,6 +948,30 @@ describe("runPreExecutionChecks", () => {
847
948
  }
848
949
  });
849
950
 
951
+ test("returns fail status for unsafe Verify command before execution", async () => {
952
+ tempDir = join(tmpdir(), `pre-exec-test-${Date.now()}`);
953
+ mkdirSync(tempDir, { recursive: true });
954
+
955
+ try {
956
+ const tasks = [
957
+ createTask({
958
+ id: "T01",
959
+ verify: "python3 -m pytest tests/ -q --tb=short 2>&1 | tail -5",
960
+ }),
961
+ ];
962
+
963
+ const result = await runPreExecutionChecks(tasks, tempDir);
964
+
965
+ assert.equal(result.status, "fail");
966
+ assert.equal(result.checks.length, 1);
967
+ assert.equal(result.checks[0]?.category, "tool");
968
+ assert.equal(result.checks[0]?.blocking, true);
969
+ assert.match(result.checks[0]?.message ?? "", /Unsafe or non-runnable Verify command/);
970
+ } finally {
971
+ rmSync(tempDir, { recursive: true, force: true });
972
+ }
973
+ });
974
+
850
975
  test("returns fail status when blocking failure exists", async () => {
851
976
  tempDir = join(tmpdir(), `pre-exec-test-${Date.now()}`);
852
977
  mkdirSync(tempDir, { recursive: true });
@@ -1993,6 +2118,28 @@ describe("checkFilePathConsistency quote-wrapped annotation (#3747)", () => {
1993
2118
  );
1994
2119
  });
1995
2120
 
2121
+ test("bare path with parenthetical annotation strips suffix before path check", (t) => {
2122
+ const tempDir = join(tmpdir(), `pre-exec-paren-${Date.now()}`);
2123
+ mkdirSync(join(tempDir, "src"), { recursive: true });
2124
+ writeFileSync(join(tempDir, "src/paren.ts"), "// content");
2125
+ t.after(() => rmSync(tempDir, { recursive: true, force: true }));
2126
+
2127
+ const tasks = [
2128
+ createTask({
2129
+ id: "T01",
2130
+ inputs: ["src/paren.ts (note)"],
2131
+ expected_output: [],
2132
+ }),
2133
+ ];
2134
+
2135
+ const results = checkFilePathConsistency(tasks, tempDir);
2136
+ assert.equal(
2137
+ results.length,
2138
+ 0,
2139
+ "Parenthetical suffix should be stripped and resolved to the real file",
2140
+ );
2141
+ });
2142
+
1996
2143
  test("prose value with spaces inside quotes is skipped (not a path)", () => {
1997
2144
  // "some description text" contains spaces — should not be checked as a path
1998
2145
  const tasks = [
@@ -556,6 +556,60 @@ describe("Pre-execution checks → pauseAuto wiring", () => {
556
556
  );
557
557
  });
558
558
 
559
+ test("files absent from s.basePath but present in canonicalProjectRoot do not block", async () => {
560
+ writePreferences({
561
+ enhanced_verification: true,
562
+ enhanced_verification_pre: true,
563
+ enhanced_verification_strict: false,
564
+ });
565
+
566
+ const worktreeDir = join(tempDir, "worktree-missing-src");
567
+ mkdirSync(join(worktreeDir, ".gsd", "milestones", "M001", "slices", "S01", "tasks"), { recursive: true });
568
+
569
+ mkdirSync(join(tempDir, "src", "engine"), { recursive: true });
570
+ writeFileSync(join(tempDir, "src", "engine", "bus.ts"), "export const bus = {};");
571
+
572
+ insertMilestone({ id: "M001" });
573
+ insertSlice({ id: "S01", milestoneId: "M001", title: "Test Slice", risk: "low" });
574
+ insertTask({
575
+ id: "T01",
576
+ sliceId: "S01",
577
+ milestoneId: "M001",
578
+ title: "Task that reads canonical-root source files",
579
+ status: "pending",
580
+ planning: {
581
+ description: "Reads src/engine/bus.ts from canonical root",
582
+ estimate: "1h",
583
+ files: [],
584
+ verify: "npm test",
585
+ inputs: ["`src/engine/bus.ts` — lifecycle helper subscription API"],
586
+ expectedOutput: ["src/engine/helper.ts"],
587
+ observabilityImpact: "",
588
+ },
589
+ sequence: 0,
590
+ });
591
+
592
+ const ctx = makeMockCtx();
593
+ const pi = makeMockPi();
594
+ const pauseAutoMock = mock.fn(async () => {});
595
+ const s = makeMockSession(worktreeDir, { type: "plan-slice", id: "M001/S01" });
596
+ Object.defineProperty(s, "canonicalProjectRoot", { get: () => tempDir });
597
+
598
+ const pctx = makePostUnitContext(s, ctx, pi, pauseAutoMock);
599
+ const result = await postUnitPostVerification(pctx);
600
+
601
+ assert.equal(
602
+ pauseAutoMock.mock.callCount(),
603
+ 0,
604
+ "pauseAuto should NOT be called when referenced files exist in canonicalProjectRoot",
605
+ );
606
+ assert.equal(
607
+ result,
608
+ "continue",
609
+ "postUnitPostVerification should return 'continue' when canonical root files satisfy pre-exec inputs",
610
+ );
611
+ });
612
+
559
613
  test("uok gate runner persists pre-execution gate outcomes when enabled", async () => {
560
614
  writePreferences({
561
615
  enhanced_verification: true,
@@ -0,0 +1,128 @@
1
+ import { describe, it } from "node:test";
2
+ import assert from "node:assert/strict";
3
+ import { resolveModelMcpConfig } from "../preferences-mcp.ts";
4
+ import { validatePreferences } from "../preferences-validation.ts";
5
+ import type { ClaudeCodeMcpConfig } from "../preferences-types.ts";
6
+
7
+ // ─── resolveModelMcpConfig ──────────────────────────────────────────────────
8
+
9
+ describe("resolveModelMcpConfig", () => {
10
+ it("returns entry when modelId starts with configured prefix", () => {
11
+ const config: ClaudeCodeMcpConfig = {
12
+ per_model: {
13
+ "claude-haiku": { allowed_servers: ["a"] },
14
+ },
15
+ };
16
+ const result = resolveModelMcpConfig("claude-haiku-4-5-20251001", config);
17
+ assert.deepEqual(result, { allowed_servers: ["a"] });
18
+ });
19
+
20
+ it("longest-prefix-wins when multiple prefixes match", () => {
21
+ const config: ClaudeCodeMcpConfig = {
22
+ per_model: {
23
+ "claude-haiku": { allowed_servers: ["short"] },
24
+ "claude-haiku-4-5": { allowed_servers: ["long"] },
25
+ },
26
+ };
27
+ const result = resolveModelMcpConfig("claude-haiku-4-5-20251001", config);
28
+ assert.deepEqual(result, { allowed_servers: ["long"] });
29
+ });
30
+
31
+ it("returns undefined when no prefix matches", () => {
32
+ const config: ClaudeCodeMcpConfig = {
33
+ per_model: {
34
+ "claude-haiku": { allowed_servers: ["a"] },
35
+ },
36
+ };
37
+ const result = resolveModelMcpConfig("claude-opus-4-7", config);
38
+ assert.equal(result, undefined);
39
+ });
40
+
41
+ it("returns undefined for empty per_model", () => {
42
+ const config: ClaudeCodeMcpConfig = { per_model: {} };
43
+ const result = resolveModelMcpConfig("claude-sonnet-4-6", config);
44
+ assert.equal(result, undefined);
45
+ });
46
+
47
+ it("returns entry when modelId exactly equals key", () => {
48
+ const config: ClaudeCodeMcpConfig = {
49
+ per_model: {
50
+ "claude-haiku-4-5-20251001": { blocked_servers: ["x"] },
51
+ },
52
+ };
53
+ const result = resolveModelMcpConfig("claude-haiku-4-5-20251001", config);
54
+ assert.deepEqual(result, { blocked_servers: ["x"] });
55
+ });
56
+
57
+ it("returns entry with both allowed_servers and blocked_servers", () => {
58
+ const config: ClaudeCodeMcpConfig = {
59
+ per_model: {
60
+ "claude-sonnet": { allowed_servers: ["a", "b"], blocked_servers: ["c"] },
61
+ },
62
+ };
63
+ const result = resolveModelMcpConfig("claude-sonnet-4-6", config);
64
+ assert.deepEqual(result, { allowed_servers: ["a", "b"], blocked_servers: ["c"] });
65
+ });
66
+ });
67
+
68
+ // ─── validatePreferences — claude_code_mcp ─────────────────────────────────
69
+
70
+ describe("validatePreferences — claude_code_mcp", () => {
71
+ it("passes with a valid claude_code_mcp block", () => {
72
+ const { errors, warnings, preferences } = validatePreferences({
73
+ claude_code_mcp: {
74
+ per_model: {
75
+ "claude-haiku": { allowed_servers: ["mcp-a"], blocked_servers: ["mcp-b"] },
76
+ },
77
+ },
78
+ });
79
+ assert.deepEqual(errors, []);
80
+ assert.deepEqual(warnings, []);
81
+ assert.deepEqual(preferences.claude_code_mcp, {
82
+ per_model: {
83
+ "claude-haiku": { allowed_servers: ["mcp-a"], blocked_servers: ["mcp-b"] },
84
+ },
85
+ });
86
+ });
87
+
88
+ it("warns and ignores when claude_code_mcp is not an object", () => {
89
+ const { errors, warnings } = validatePreferences({
90
+ claude_code_mcp: "bad-value" as unknown as object,
91
+ });
92
+ assert.deepEqual(errors, []);
93
+ assert.ok(
94
+ warnings.some((w) => w.includes("claude_code_mcp must be an object")),
95
+ `expected warning about non-object, got: ${JSON.stringify(warnings)}`,
96
+ );
97
+ });
98
+
99
+ it("warns when per_model entry has non-array allowed_servers", () => {
100
+ const { errors, warnings } = validatePreferences({
101
+ claude_code_mcp: {
102
+ per_model: {
103
+ "claude-haiku": { allowed_servers: "not-an-array" as unknown as string[] },
104
+ },
105
+ },
106
+ });
107
+ assert.deepEqual(errors, []);
108
+ assert.ok(
109
+ warnings.some((w) => w.includes("allowed_servers")),
110
+ `expected warning about allowed_servers, got: ${JSON.stringify(warnings)}`,
111
+ );
112
+ });
113
+
114
+ it("warns when per_model entry has non-array blocked_servers", () => {
115
+ const { errors, warnings } = validatePreferences({
116
+ claude_code_mcp: {
117
+ per_model: {
118
+ "claude-haiku": { blocked_servers: 42 as unknown as string[] },
119
+ },
120
+ },
121
+ });
122
+ assert.deepEqual(errors, []);
123
+ assert.ok(
124
+ warnings.some((w) => w.includes("blocked_servers")),
125
+ `expected warning about blocked_servers, got: ${JSON.stringify(warnings)}`,
126
+ );
127
+ });
128
+ });
@@ -225,6 +225,81 @@ test("slice_parallel numeric max_workers is bounded to 1..8", () => {
225
225
  assert.ok(tooHigh.errors.some(e => e.includes("slice_parallel.max_workers")));
226
226
  });
227
227
 
228
+ test("workspace.repositories validates and is preserved", () => {
229
+ const { preferences, errors, warnings } = validatePreferences({
230
+ workspace: {
231
+ mode: "parent",
232
+ repositories: {
233
+ frontend: {
234
+ path: "frontend",
235
+ role: "web UI",
236
+ verification: ["npm test"],
237
+ commit_policy: "skip",
238
+ },
239
+ backend: {
240
+ path: "backend",
241
+ role: "API",
242
+ },
243
+ },
244
+ },
245
+ });
246
+
247
+ assert.equal(errors.length, 0);
248
+ assert.equal(warnings.filter(w => w.includes("workspace")).length, 0);
249
+ assert.equal(preferences.workspace?.mode, "parent");
250
+ assert.deepEqual(preferences.workspace?.repositories?.frontend, {
251
+ path: "frontend",
252
+ role: "web UI",
253
+ verification: ["npm test"],
254
+ commit_policy: "skip",
255
+ });
256
+ assert.deepEqual(preferences.workspace?.repositories?.backend, {
257
+ path: "backend",
258
+ role: "API",
259
+ });
260
+ });
261
+
262
+ test("workspace.repositories.commit_policy rejects invalid values", () => {
263
+ const { errors } = validatePreferences({
264
+ workspace: {
265
+ repositories: {
266
+ frontend: {
267
+ path: "frontend",
268
+ commit_policy: "manual",
269
+ },
270
+ },
271
+ },
272
+ } as any);
273
+
274
+ assert.ok(errors.some((e) => e.includes("workspace.repositories.frontend.commit_policy")));
275
+ });
276
+
277
+ test("workspace.repositories rejects invalid shapes", () => {
278
+ const { errors } = validatePreferences({
279
+ workspace: {
280
+ mode: "invalid-mode",
281
+ repositories: {
282
+ "bad id": { path: "frontend" },
283
+ backend: { path: "" },
284
+ },
285
+ },
286
+ } as any);
287
+
288
+ assert.ok(errors.some(e => e.includes("workspace.mode")));
289
+ assert.ok(errors.some(e => e.includes('workspace.repositories key "bad id"')));
290
+ assert.ok(errors.some(e => e.includes("workspace.repositories.backend.path")));
291
+ });
292
+
293
+ test("workspace is a recognized preference key (no unknown warning)", () => {
294
+ const { warnings } = validatePreferences({
295
+ workspace: { mode: "project" },
296
+ });
297
+ assert.equal(
298
+ warnings.filter(w => w.includes("unknown preference key \"workspace\"")).length,
299
+ 0,
300
+ );
301
+ });
302
+
228
303
  test("valid values pass through correctly", () => {
229
304
  const { preferences: p1 } = validatePreferences({ budget_enforcement: "halt" });
230
305
  assert.equal(p1.budget_enforcement, "halt");
@@ -58,6 +58,7 @@ const PREF_SAMPLE_VALUES: Record<string, unknown> = {
58
58
  verification_commands: ["npm test"],
59
59
  verification_auto_fix: true,
60
60
  verification_max_retries: 1,
61
+ per_unit_cost_cap_usd: 5,
61
62
  search_provider: "web",
62
63
  context_selection: "auto",
63
64
  widget_mode: "small",
@@ -86,6 +87,17 @@ const PREF_SAMPLE_VALUES: Record<string, unknown> = {
86
87
  context_window_override: 128000,
87
88
  context_mode: { enabled: true },
88
89
  planning_depth: "deep",
90
+ claude_code_mcp: { per_model: { "claude-haiku": { allowed_servers: ["gsd-workflow"] } } },
91
+ workspace: {
92
+ mode: "parent",
93
+ repositories: {
94
+ frontend: {
95
+ path: "frontend",
96
+ role: "web UI",
97
+ verification: ["npm test"],
98
+ },
99
+ },
100
+ },
89
101
  };
90
102
 
91
103
  test("prefs wizard save path preserves every known preference key", async () => {
@@ -141,3 +153,70 @@ test("category summaries expose the wizard menu surface for configured prefs", (
141
153
  assert.match(summaries.integrations, /remote: C123/);
142
154
  assert.match(summaries.verification, /1 cmd/);
143
155
  });
156
+
157
+ test("models wizard offers discovered models for enabled providers", async () => {
158
+ const dir = mkdtempSync(join(tmpdir(), "gsd-prefs-wizard-"));
159
+ const prefsPath = join(dir, "PREFERENCES.md");
160
+ const choices = [
161
+ "Models",
162
+ "local (2 models)",
163
+ "discovered-model",
164
+ "(keep current)",
165
+ "(keep current)",
166
+ "(keep current)",
167
+ "(keep current)",
168
+ "(keep current)",
169
+ "(keep current)",
170
+ "(keep current)",
171
+ ];
172
+ const ctx = {
173
+ // `getAllWithDiscovered` reads `this._all` so the wizard must call it as a
174
+ // method — invoking a detached reference would lose `this` and throw,
175
+ // mirroring the real ModelRegistry implementation.
176
+ modelRegistry: {
177
+ _all: [
178
+ { provider: "local", id: "baseline-model" },
179
+ { provider: "local", id: "discovered-model" },
180
+ { provider: "disabled", id: "hidden-model" },
181
+ ],
182
+ getAvailable() {
183
+ return [{ provider: "local", id: "baseline-model" }];
184
+ },
185
+ getAllWithDiscovered() {
186
+ return this._all;
187
+ },
188
+ },
189
+ ui: {
190
+ notify() {},
191
+ select: async (label: string, options: string[]) => {
192
+ const choice = choices.shift();
193
+ if (!choice && label === "GSD Preferences") return "── Save & Exit ──";
194
+ if (!choice && options.includes("(keep current)")) return "(keep current)";
195
+ if (!choice && options.includes("Done")) return "Done";
196
+ assert.ok(choice, `Unexpected prompt: ${label}`);
197
+ if (choice === "Models") {
198
+ const modelsOption = options.find((option) => option.startsWith("Models"));
199
+ assert.ok(modelsOption, "Expected Models category option");
200
+ return modelsOption;
201
+ }
202
+ assert.ok(options.includes(choice), `"${choice}" must be offered by "${label}"`);
203
+ assert.ok(!options.includes("hidden-model"), "models from disabled providers must not be offered");
204
+ return choice;
205
+ },
206
+ input: async () => null,
207
+ },
208
+ waitForIdle: async () => {},
209
+ reload: async () => {},
210
+ } as any;
211
+
212
+ try {
213
+ await handlePrefsWizard(ctx, "project", {}, { pathOverride: prefsPath });
214
+
215
+ assert.equal(choices.length, 0, "Expected all queued wizard choices to be consumed");
216
+ const saved = readFileSync(prefsPath, "utf-8");
217
+ assert.match(saved, /research:\s+local\/discovered-model/);
218
+ assert.doesNotMatch(saved, /hidden-model/);
219
+ } finally {
220
+ rmSync(dir, { recursive: true, force: true });
221
+ }
222
+ });