gsd-pi 2.76.0-dev.fe143342a → 2.77.0-dev.1d17f366c

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 (597) hide show
  1. package/README.md +17 -35
  2. package/dist/claude-cli-check.js +32 -3
  3. package/dist/cli-web-branch.d.ts +1 -0
  4. package/dist/cli-web-branch.js +3 -0
  5. package/dist/cli.js +38 -2
  6. package/dist/extension-discovery.d.ts +6 -0
  7. package/dist/extension-discovery.js +37 -0
  8. package/dist/extension-registry.d.ts +3 -0
  9. package/dist/extension-sort.d.ts +18 -0
  10. package/dist/extension-sort.js +114 -0
  11. package/dist/extension-validator.d.ts +47 -0
  12. package/dist/extension-validator.js +127 -0
  13. package/dist/loader.js +35 -7
  14. package/dist/onboarding.js +45 -0
  15. package/dist/provider-migrations.d.ts +18 -0
  16. package/dist/provider-migrations.js +14 -0
  17. package/dist/resources/extensions/claude-code-cli/readiness.js +4 -3
  18. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +78 -59
  19. package/dist/resources/extensions/cmux/index.js +20 -0
  20. package/dist/resources/extensions/github-sync/templates.js +103 -0
  21. package/dist/resources/extensions/google-search/extension-manifest.json +5 -4
  22. package/dist/resources/extensions/google-search/index.js +3 -375
  23. package/dist/resources/extensions/gsd/abandon-detect.js +44 -0
  24. package/dist/resources/extensions/gsd/auto/loop.js +90 -2
  25. package/dist/resources/extensions/gsd/auto/phases.js +95 -21
  26. package/dist/resources/extensions/gsd/auto/resolve.js +24 -0
  27. package/dist/resources/extensions/gsd/auto/run-unit.js +48 -4
  28. package/dist/resources/extensions/gsd/auto/session.js +18 -1
  29. package/dist/resources/extensions/gsd/auto/turn-epoch.js +95 -0
  30. package/dist/resources/extensions/gsd/auto-dispatch.js +115 -17
  31. package/dist/resources/extensions/gsd/auto-loop.js +1 -1
  32. package/dist/resources/extensions/gsd/auto-model-selection.js +1 -1
  33. package/dist/resources/extensions/gsd/auto-post-unit.js +90 -2
  34. package/dist/resources/extensions/gsd/auto-prompts.js +14 -0
  35. package/dist/resources/extensions/gsd/auto-recovery.js +46 -1
  36. package/dist/resources/extensions/gsd/auto-start.js +45 -39
  37. package/dist/resources/extensions/gsd/auto-timeout-recovery.js +11 -5
  38. package/dist/resources/extensions/gsd/auto-unit-closeout.js +11 -2
  39. package/dist/resources/extensions/gsd/auto-worktree.js +109 -61
  40. package/dist/resources/extensions/gsd/auto.js +97 -31
  41. package/dist/resources/extensions/gsd/bootstrap/agent-end-recovery.js +27 -1
  42. package/dist/resources/extensions/gsd/bootstrap/provider-error-resume.js +4 -2
  43. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +11 -0
  44. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +17 -6
  45. package/dist/resources/extensions/gsd/bootstrap/system-context.js +11 -6
  46. package/dist/resources/extensions/gsd/bootstrap/write-gate.js +34 -2
  47. package/dist/resources/extensions/gsd/clean-root-preflight.js +93 -0
  48. package/dist/resources/extensions/gsd/commands/handlers/workflow.js +31 -4
  49. package/dist/resources/extensions/gsd/commands-cmux.js +9 -6
  50. package/dist/resources/extensions/gsd/commands-extensions.js +634 -43
  51. package/dist/resources/extensions/gsd/commands-prefs-wizard.js +968 -23
  52. package/dist/resources/extensions/gsd/dispatch-guard.js +29 -3
  53. package/dist/resources/extensions/gsd/file-lock.js +49 -9
  54. package/dist/resources/extensions/gsd/git-service.js +1 -0
  55. package/dist/resources/extensions/gsd/gitignore.js +2 -0
  56. package/dist/resources/extensions/gsd/gsd-db.js +90 -30
  57. package/dist/resources/extensions/gsd/guided-flow-queue.js +4 -1
  58. package/dist/resources/extensions/gsd/guided-flow.js +212 -9
  59. package/dist/resources/extensions/gsd/health-widget.js +4 -1
  60. package/dist/resources/extensions/gsd/journal.js +17 -2
  61. package/dist/resources/extensions/gsd/key-manager.js +22 -0
  62. package/dist/resources/extensions/gsd/milestone-actions.js +15 -0
  63. package/dist/resources/extensions/gsd/milestone-summary-classifier.js +37 -0
  64. package/dist/resources/extensions/gsd/model-router.js +36 -3
  65. package/dist/resources/extensions/gsd/notifications.js +30 -16
  66. package/dist/resources/extensions/gsd/pre-execution-checks.js +31 -6
  67. package/dist/resources/extensions/gsd/prompts/discuss-headless.md +29 -2
  68. package/dist/resources/extensions/gsd/prompts/discuss.md +29 -2
  69. package/dist/resources/extensions/gsd/prompts/doctor-heal.md +5 -4
  70. package/dist/resources/extensions/gsd/prompts/parallel-research-slices.md +5 -2
  71. package/dist/resources/extensions/gsd/prompts/system.md +1 -0
  72. package/dist/resources/extensions/gsd/reports.js +5 -4
  73. package/dist/resources/extensions/gsd/safety/evidence-collector.js +96 -0
  74. package/dist/resources/extensions/gsd/safety/file-change-validator.js +12 -4
  75. package/dist/resources/extensions/gsd/safety/safety-harness.js +5 -1
  76. package/dist/resources/extensions/gsd/state-transition-matrix.js +118 -0
  77. package/dist/resources/extensions/gsd/state.js +25 -25
  78. package/dist/resources/extensions/gsd/token-counter.js +22 -5
  79. package/dist/resources/extensions/gsd/tools/complete-milestone.js +16 -10
  80. package/dist/resources/extensions/gsd/tools/complete-slice.js +21 -0
  81. package/dist/resources/extensions/gsd/tools/complete-task.js +31 -0
  82. package/dist/resources/extensions/gsd/uok/audit.js +18 -2
  83. package/dist/resources/extensions/gsd/uok/dispatch-envelope.js +33 -0
  84. package/dist/resources/extensions/gsd/uok/execution-graph.js +10 -0
  85. package/dist/resources/extensions/gsd/uok/gitops.js +2 -1
  86. package/dist/resources/extensions/gsd/uok/loop-adapter.js +37 -10
  87. package/dist/resources/extensions/gsd/uok/parity-report.js +58 -0
  88. package/dist/resources/extensions/gsd/uok/plan-v2.js +30 -7
  89. package/dist/resources/extensions/gsd/uok/writer.js +82 -0
  90. package/dist/resources/extensions/gsd/workflow-logger.js +10 -2
  91. package/dist/resources/extensions/gsd/worktree-manager.js +1 -0
  92. package/dist/resources/extensions/gsd/worktree-resolver.js +50 -10
  93. package/dist/resources/extensions/mcp-client/auth.js +10 -1
  94. package/dist/resources/extensions/mcp-client/index.js +118 -9
  95. package/dist/resources/extensions/shared/cmux-events.js +12 -0
  96. package/dist/resources/extensions/shared/rtk-session-stats.js +1 -2
  97. package/dist/resources/skills/create-skill/SKILL.md +2 -2
  98. package/dist/resources/skills/create-skill/references/gsd-skill-ecosystem.md +4 -4
  99. package/dist/resources/skills/create-skill/workflows/audit-skill.md +4 -4
  100. package/dist/resources/skills/create-skill/workflows/create-new-skill.md +5 -5
  101. package/dist/resources/skills/verify-before-complete/SKILL.md +2 -1
  102. package/dist/resources/skills/write-docs/SKILL.md +2 -1
  103. package/dist/tsconfig.extensions.tsbuildinfo +1 -1
  104. package/dist/web/standalone/.next/BUILD_ID +1 -1
  105. package/dist/web/standalone/.next/app-path-routes-manifest.json +9 -9
  106. package/dist/web/standalone/.next/build-manifest.json +3 -3
  107. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  108. package/dist/web/standalone/.next/react-loadable-manifest.json +1 -1
  109. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  110. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  111. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  112. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  113. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  114. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  115. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  116. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  117. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  118. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  119. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  120. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  121. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  122. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  123. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  124. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  125. package/dist/web/standalone/.next/server/app/api/boot/route.js +1 -1
  126. package/dist/web/standalone/.next/server/app/api/boot/route.js.nft.json +1 -1
  127. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js +1 -1
  128. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js.nft.json +1 -1
  129. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js +1 -1
  130. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js.nft.json +1 -1
  131. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js +1 -1
  132. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js.nft.json +1 -1
  133. package/dist/web/standalone/.next/server/app/api/captures/route.js +1 -1
  134. package/dist/web/standalone/.next/server/app/api/captures/route.js.nft.json +1 -1
  135. package/dist/web/standalone/.next/server/app/api/cleanup/route.js +1 -1
  136. package/dist/web/standalone/.next/server/app/api/cleanup/route.js.nft.json +1 -1
  137. package/dist/web/standalone/.next/server/app/api/doctor/route.js +1 -1
  138. package/dist/web/standalone/.next/server/app/api/doctor/route.js.nft.json +1 -1
  139. package/dist/web/standalone/.next/server/app/api/export-data/route.js +1 -1
  140. package/dist/web/standalone/.next/server/app/api/export-data/route.js.nft.json +1 -1
  141. package/dist/web/standalone/.next/server/app/api/files/route.js +1 -1
  142. package/dist/web/standalone/.next/server/app/api/files/route.js.nft.json +1 -1
  143. package/dist/web/standalone/.next/server/app/api/forensics/route.js +1 -1
  144. package/dist/web/standalone/.next/server/app/api/forensics/route.js.nft.json +1 -1
  145. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  146. package/dist/web/standalone/.next/server/app/api/git/route.js.nft.json +1 -1
  147. package/dist/web/standalone/.next/server/app/api/history/route.js +1 -1
  148. package/dist/web/standalone/.next/server/app/api/history/route.js.nft.json +1 -1
  149. package/dist/web/standalone/.next/server/app/api/hooks/route.js +1 -1
  150. package/dist/web/standalone/.next/server/app/api/hooks/route.js.nft.json +1 -1
  151. package/dist/web/standalone/.next/server/app/api/inspect/route.js +1 -1
  152. package/dist/web/standalone/.next/server/app/api/inspect/route.js.nft.json +1 -1
  153. package/dist/web/standalone/.next/server/app/api/knowledge/route.js +1 -1
  154. package/dist/web/standalone/.next/server/app/api/knowledge/route.js.nft.json +1 -1
  155. package/dist/web/standalone/.next/server/app/api/live-state/route.js +1 -1
  156. package/dist/web/standalone/.next/server/app/api/live-state/route.js.nft.json +1 -1
  157. package/dist/web/standalone/.next/server/app/api/notifications/route.js +1 -1
  158. package/dist/web/standalone/.next/server/app/api/notifications/route.js.nft.json +1 -1
  159. package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
  160. package/dist/web/standalone/.next/server/app/api/onboarding/route.js.nft.json +1 -1
  161. package/dist/web/standalone/.next/server/app/api/projects/route.js +1 -1
  162. package/dist/web/standalone/.next/server/app/api/projects/route.js.nft.json +1 -1
  163. package/dist/web/standalone/.next/server/app/api/recovery/route.js +1 -1
  164. package/dist/web/standalone/.next/server/app/api/recovery/route.js.nft.json +1 -1
  165. package/dist/web/standalone/.next/server/app/api/session/browser/route.js +1 -1
  166. package/dist/web/standalone/.next/server/app/api/session/browser/route.js.nft.json +1 -1
  167. package/dist/web/standalone/.next/server/app/api/session/command/route.js +1 -1
  168. package/dist/web/standalone/.next/server/app/api/session/command/route.js.nft.json +1 -1
  169. package/dist/web/standalone/.next/server/app/api/session/events/route.js +1 -1
  170. package/dist/web/standalone/.next/server/app/api/session/events/route.js.nft.json +1 -1
  171. package/dist/web/standalone/.next/server/app/api/session/manage/route.js +1 -1
  172. package/dist/web/standalone/.next/server/app/api/session/manage/route.js.nft.json +1 -1
  173. package/dist/web/standalone/.next/server/app/api/settings-data/route.js +1 -1
  174. package/dist/web/standalone/.next/server/app/api/settings-data/route.js.nft.json +1 -1
  175. package/dist/web/standalone/.next/server/app/api/skill-health/route.js +1 -1
  176. package/dist/web/standalone/.next/server/app/api/skill-health/route.js.nft.json +1 -1
  177. package/dist/web/standalone/.next/server/app/api/steer/route.js +1 -1
  178. package/dist/web/standalone/.next/server/app/api/steer/route.js.nft.json +1 -1
  179. package/dist/web/standalone/.next/server/app/api/switch-root/route.js +1 -1
  180. package/dist/web/standalone/.next/server/app/api/switch-root/route.js.nft.json +1 -1
  181. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +1 -1
  182. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js.nft.json +1 -1
  183. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +1 -1
  184. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js.nft.json +1 -1
  185. package/dist/web/standalone/.next/server/app/api/undo/route.js +1 -1
  186. package/dist/web/standalone/.next/server/app/api/undo/route.js.nft.json +1 -1
  187. package/dist/web/standalone/.next/server/app/api/visualizer/route.js +1 -1
  188. package/dist/web/standalone/.next/server/app/api/visualizer/route.js.nft.json +1 -1
  189. package/dist/web/standalone/.next/server/app/index.html +1 -1
  190. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  191. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  192. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  193. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  194. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  195. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  196. package/dist/web/standalone/.next/server/app-paths-manifest.json +9 -9
  197. package/dist/web/standalone/.next/server/chunks/1926.js +1 -0
  198. package/dist/web/standalone/.next/server/chunks/6897.js +2 -2
  199. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  200. package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
  201. package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
  202. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  203. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  204. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  205. package/dist/web/standalone/.next/static/chunks/2826.e9f5195e91f9cad2.js +11 -0
  206. package/dist/web/standalone/.next/static/chunks/{webpack-5fc74f13a25fa1bb.js → webpack-2e68521d7c82f7c2.js} +1 -1
  207. package/dist/welcome-screen.js +6 -1
  208. package/dist/wizard.js +2 -0
  209. package/package.json +16 -14
  210. package/packages/daemon/package.json +2 -2
  211. package/packages/mcp-server/README.md +3 -3
  212. package/packages/mcp-server/dist/env-writer.d.ts +1 -0
  213. package/packages/mcp-server/dist/env-writer.d.ts.map +1 -1
  214. package/packages/mcp-server/dist/env-writer.js +74 -6
  215. package/packages/mcp-server/dist/env-writer.js.map +1 -1
  216. package/packages/mcp-server/dist/remote-questions.d.ts +45 -0
  217. package/packages/mcp-server/dist/remote-questions.d.ts.map +1 -0
  218. package/packages/mcp-server/dist/remote-questions.js +732 -0
  219. package/packages/mcp-server/dist/remote-questions.js.map +1 -0
  220. package/packages/mcp-server/dist/server.d.ts +7 -0
  221. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  222. package/packages/mcp-server/dist/server.js +95 -10
  223. package/packages/mcp-server/dist/server.js.map +1 -1
  224. package/packages/mcp-server/dist/session-manager.d.ts +14 -0
  225. package/packages/mcp-server/dist/session-manager.d.ts.map +1 -1
  226. package/packages/mcp-server/dist/session-manager.js +49 -1
  227. package/packages/mcp-server/dist/session-manager.js.map +1 -1
  228. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  229. package/packages/mcp-server/dist/workflow-tools.js +15 -6
  230. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  231. package/packages/mcp-server/package.json +9 -3
  232. package/packages/mcp-server/src/env-writer.test.ts +79 -1
  233. package/packages/mcp-server/src/env-writer.ts +76 -6
  234. package/packages/mcp-server/src/mcp-server.test.ts +67 -0
  235. package/packages/mcp-server/src/readers/readers.test.ts +5 -1
  236. package/packages/mcp-server/src/remote-questions.test.ts +294 -0
  237. package/packages/mcp-server/src/remote-questions.ts +916 -0
  238. package/packages/mcp-server/src/server.ts +118 -16
  239. package/packages/mcp-server/src/session-manager.ts +43 -1
  240. package/packages/mcp-server/src/workflow-tools.test.ts +44 -0
  241. package/packages/mcp-server/src/workflow-tools.ts +19 -6
  242. package/packages/mcp-server/tsconfig.test.json +19 -0
  243. package/packages/mcp-server/tsconfig.tsbuildinfo +1 -1
  244. package/packages/native/package.json +6 -1
  245. package/packages/native/src/__tests__/clipboard.test.mjs +69 -23
  246. package/packages/native/tsconfig.tsbuildinfo +1 -1
  247. package/packages/pi-agent-core/package.json +6 -1
  248. package/packages/pi-agent-core/src/agent-loop.test.ts +220 -15
  249. package/packages/pi-ai/dist/models/custom.d.ts +38 -0
  250. package/packages/pi-ai/dist/models/custom.d.ts.map +1 -1
  251. package/packages/pi-ai/dist/models/custom.js +41 -0
  252. package/packages/pi-ai/dist/models/custom.js.map +1 -1
  253. package/packages/pi-ai/dist/providers/anthropic-auth.test.js +1 -1
  254. package/packages/pi-ai/dist/providers/anthropic-auth.test.js.map +1 -1
  255. package/packages/pi-ai/dist/providers/anthropic-shared.d.ts.map +1 -1
  256. package/packages/pi-ai/dist/providers/anthropic-shared.js +27 -4
  257. package/packages/pi-ai/dist/providers/anthropic-shared.js.map +1 -1
  258. package/packages/pi-ai/dist/providers/anthropic.d.ts.map +1 -1
  259. package/packages/pi-ai/dist/providers/anthropic.js +8 -3
  260. package/packages/pi-ai/dist/providers/anthropic.js.map +1 -1
  261. package/packages/pi-ai/dist/providers/minimax-tool-name.test.d.ts +2 -0
  262. package/packages/pi-ai/dist/providers/minimax-tool-name.test.d.ts.map +1 -0
  263. package/packages/pi-ai/dist/providers/minimax-tool-name.test.js +80 -0
  264. package/packages/pi-ai/dist/providers/minimax-tool-name.test.js.map +1 -0
  265. package/packages/pi-ai/dist/providers/simple-options.d.ts +10 -0
  266. package/packages/pi-ai/dist/providers/simple-options.d.ts.map +1 -1
  267. package/packages/pi-ai/dist/providers/simple-options.js +16 -1
  268. package/packages/pi-ai/dist/providers/simple-options.js.map +1 -1
  269. package/packages/pi-ai/package.json +6 -1
  270. package/packages/pi-ai/src/models/custom.ts +42 -0
  271. package/packages/pi-ai/src/providers/anthropic-auth.test.ts +1 -1
  272. package/packages/pi-ai/src/providers/anthropic-shared.ts +26 -5
  273. package/packages/pi-ai/src/providers/anthropic.ts +9 -3
  274. package/packages/pi-ai/src/providers/minimax-tool-name.test.ts +98 -0
  275. package/packages/pi-ai/src/providers/simple-options.ts +17 -1
  276. package/packages/pi-ai/tsconfig.tsbuildinfo +1 -1
  277. package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js +3 -2
  278. package/packages/pi-coding-agent/dist/core/agent-session-abort-order.test.js.map +1 -1
  279. package/packages/pi-coding-agent/dist/core/agent-session.d.ts +2 -0
  280. package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
  281. package/packages/pi-coding-agent/dist/core/agent-session.js +7 -0
  282. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  283. package/packages/pi-coding-agent/dist/core/compaction/compaction.d.ts +25 -0
  284. package/packages/pi-coding-agent/dist/core/compaction/compaction.d.ts.map +1 -1
  285. package/packages/pi-coding-agent/dist/core/compaction/compaction.js +105 -6
  286. package/packages/pi-coding-agent/dist/core/compaction/compaction.js.map +1 -1
  287. package/packages/pi-coding-agent/dist/core/compaction/compaction.test.js +230 -28
  288. package/packages/pi-coding-agent/dist/core/compaction/compaction.test.js.map +1 -1
  289. package/packages/pi-coding-agent/dist/core/compaction/utils.d.ts +30 -2
  290. package/packages/pi-coding-agent/dist/core/compaction/utils.d.ts.map +1 -1
  291. package/packages/pi-coding-agent/dist/core/compaction/utils.js +113 -12
  292. package/packages/pi-coding-agent/dist/core/compaction/utils.js.map +1 -1
  293. package/packages/pi-coding-agent/dist/core/compaction-orchestrator.d.ts +1 -0
  294. package/packages/pi-coding-agent/dist/core/compaction-orchestrator.d.ts.map +1 -1
  295. package/packages/pi-coding-agent/dist/core/compaction-orchestrator.js +29 -18
  296. package/packages/pi-coding-agent/dist/core/compaction-orchestrator.js.map +1 -1
  297. package/packages/pi-coding-agent/dist/core/compaction-orchestrator.test.d.ts +2 -0
  298. package/packages/pi-coding-agent/dist/core/compaction-orchestrator.test.d.ts.map +1 -0
  299. package/packages/pi-coding-agent/dist/core/compaction-orchestrator.test.js +130 -0
  300. package/packages/pi-coding-agent/dist/core/compaction-orchestrator.test.js.map +1 -0
  301. package/packages/pi-coding-agent/dist/core/compaction-utils.test.js +56 -1
  302. package/packages/pi-coding-agent/dist/core/compaction-utils.test.js.map +1 -1
  303. package/packages/pi-coding-agent/dist/core/discovery-cache.test.js +8 -15
  304. package/packages/pi-coding-agent/dist/core/discovery-cache.test.js.map +1 -1
  305. package/packages/pi-coding-agent/dist/core/extensions/extension-discovery.d.ts +25 -0
  306. package/packages/pi-coding-agent/dist/core/extensions/extension-discovery.d.ts.map +1 -0
  307. package/packages/pi-coding-agent/dist/core/extensions/extension-discovery.js +109 -0
  308. package/packages/pi-coding-agent/dist/core/extensions/extension-discovery.js.map +1 -0
  309. package/packages/pi-coding-agent/dist/core/extensions/extension-registry.d.ts +67 -0
  310. package/packages/pi-coding-agent/dist/core/extensions/extension-registry.d.ts.map +1 -0
  311. package/packages/pi-coding-agent/dist/core/extensions/extension-registry.js +167 -0
  312. package/packages/pi-coding-agent/dist/core/extensions/extension-registry.js.map +1 -0
  313. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts +3 -2
  314. package/packages/pi-coding-agent/dist/core/extensions/loader.d.ts.map +1 -1
  315. package/packages/pi-coding-agent/dist/core/extensions/loader.js +24 -8
  316. package/packages/pi-coding-agent/dist/core/extensions/loader.js.map +1 -1
  317. package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts +2 -0
  318. package/packages/pi-coding-agent/dist/core/extensions/runner.d.ts.map +1 -1
  319. package/packages/pi-coding-agent/dist/core/extensions/runner.js.map +1 -1
  320. package/packages/pi-coding-agent/dist/core/extensions/types.d.ts +14 -0
  321. package/packages/pi-coding-agent/dist/core/extensions/types.d.ts.map +1 -1
  322. package/packages/pi-coding-agent/dist/core/extensions/types.js.map +1 -1
  323. package/packages/pi-coding-agent/dist/core/lsp/lsp-integration.test.js +11 -0
  324. package/packages/pi-coding-agent/dist/core/lsp/lsp-integration.test.js.map +1 -1
  325. package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js +2 -2
  326. package/packages/pi-coding-agent/dist/core/model-registry-auth-mode.test.js.map +1 -1
  327. package/packages/pi-coding-agent/dist/core/model-registry-custom-caps.test.d.ts +2 -0
  328. package/packages/pi-coding-agent/dist/core/model-registry-custom-caps.test.d.ts.map +1 -0
  329. package/packages/pi-coding-agent/dist/core/model-registry-custom-caps.test.js +203 -0
  330. package/packages/pi-coding-agent/dist/core/model-registry-custom-caps.test.js.map +1 -0
  331. package/packages/pi-coding-agent/dist/core/model-registry.d.ts.map +1 -1
  332. package/packages/pi-coding-agent/dist/core/model-registry.js +14 -0
  333. package/packages/pi-coding-agent/dist/core/model-registry.js.map +1 -1
  334. package/packages/pi-coding-agent/dist/core/resource-loader.js +1 -1
  335. package/packages/pi-coding-agent/dist/core/resource-loader.js.map +1 -1
  336. package/packages/pi-coding-agent/dist/core/sdk.d.ts +1 -0
  337. package/packages/pi-coding-agent/dist/core/sdk.d.ts.map +1 -1
  338. package/packages/pi-coding-agent/dist/core/sdk.js +4 -1
  339. package/packages/pi-coding-agent/dist/core/sdk.js.map +1 -1
  340. package/packages/pi-coding-agent/dist/core/sdk.test.js +19 -1
  341. package/packages/pi-coding-agent/dist/core/sdk.test.js.map +1 -1
  342. package/packages/pi-coding-agent/dist/core/session-manager.js +1 -1
  343. package/packages/pi-coding-agent/dist/core/session-manager.js.map +1 -1
  344. package/packages/pi-coding-agent/dist/core/session-manager.test.js +21 -1
  345. package/packages/pi-coding-agent/dist/core/session-manager.test.js.map +1 -1
  346. package/packages/pi-coding-agent/dist/core/system-prompt.js +3 -3
  347. package/packages/pi-coding-agent/dist/core/system-prompt.js.map +1 -1
  348. package/packages/pi-coding-agent/dist/core/tools/path-utils.test.js +2 -1
  349. package/packages/pi-coding-agent/dist/core/tools/path-utils.test.js.map +1 -1
  350. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/provider-display-name.test.js +15 -6
  351. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/provider-display-name.test.js.map +1 -1
  352. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.d.ts.map +1 -1
  353. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js +14 -5
  354. package/packages/pi-coding-agent/dist/modes/interactive/components/footer.js.map +1 -1
  355. package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.d.ts +7 -1
  356. package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.d.ts.map +1 -1
  357. package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.js +31 -9
  358. package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.js.map +1 -1
  359. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js +14 -0
  360. package/packages/pi-coding-agent/dist/modes/interactive/controllers/input-controller.test.js.map +1 -1
  361. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  362. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +13 -1
  363. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  364. package/packages/pi-coding-agent/package.json +6 -1
  365. package/packages/pi-coding-agent/src/core/agent-session-abort-order.test.ts +3 -2
  366. package/packages/pi-coding-agent/src/core/agent-session.ts +11 -0
  367. package/packages/pi-coding-agent/src/core/compaction/compaction.test.ts +368 -28
  368. package/packages/pi-coding-agent/src/core/compaction/compaction.ts +122 -6
  369. package/packages/pi-coding-agent/src/core/compaction/utils.ts +111 -13
  370. package/packages/pi-coding-agent/src/core/compaction-orchestrator.test.ts +154 -0
  371. package/packages/pi-coding-agent/src/core/compaction-orchestrator.ts +32 -18
  372. package/packages/pi-coding-agent/src/core/compaction-utils.test.ts +68 -1
  373. package/packages/pi-coding-agent/src/core/discovery-cache.test.ts +9 -18
  374. package/packages/pi-coding-agent/src/core/extensions/extension-discovery.ts +119 -0
  375. package/packages/pi-coding-agent/src/core/extensions/extension-registry.ts +222 -0
  376. package/packages/pi-coding-agent/src/core/extensions/loader.ts +24 -11
  377. package/packages/pi-coding-agent/src/core/extensions/runner.ts +2 -0
  378. package/packages/pi-coding-agent/src/core/extensions/types.ts +15 -0
  379. package/packages/pi-coding-agent/src/core/lsp/lsp-integration.test.ts +13 -0
  380. package/packages/pi-coding-agent/src/core/model-registry-auth-mode.test.ts +2 -2
  381. package/packages/pi-coding-agent/src/core/model-registry-custom-caps.test.ts +245 -0
  382. package/packages/pi-coding-agent/src/core/model-registry.ts +16 -0
  383. package/packages/pi-coding-agent/src/core/resource-loader.ts +1 -1
  384. package/packages/pi-coding-agent/src/core/sdk.test.ts +25 -1
  385. package/packages/pi-coding-agent/src/core/sdk.ts +10 -3
  386. package/packages/pi-coding-agent/src/core/session-manager.test.ts +30 -1
  387. package/packages/pi-coding-agent/src/core/session-manager.ts +1 -1
  388. package/packages/pi-coding-agent/src/core/system-prompt.ts +3 -3
  389. package/packages/pi-coding-agent/src/core/tools/path-utils.test.ts +2 -1
  390. package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/provider-display-name.test.ts +17 -7
  391. package/packages/pi-coding-agent/src/modes/interactive/components/footer.ts +14 -5
  392. package/packages/pi-coding-agent/src/modes/interactive/components/model-selector.ts +45 -11
  393. package/packages/pi-coding-agent/src/modes/interactive/controllers/input-controller.test.ts +14 -0
  394. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +13 -1
  395. package/packages/pi-coding-agent/tsconfig.tsbuildinfo +1 -1
  396. package/packages/pi-tui/dist/__tests__/autocomplete.test.js +12 -5
  397. package/packages/pi-tui/dist/__tests__/autocomplete.test.js.map +1 -1
  398. package/packages/pi-tui/dist/__tests__/stdin-buffer.test.js +21 -0
  399. package/packages/pi-tui/dist/__tests__/stdin-buffer.test.js.map +1 -1
  400. package/packages/pi-tui/dist/stdin-buffer.d.ts +7 -0
  401. package/packages/pi-tui/dist/stdin-buffer.d.ts.map +1 -1
  402. package/packages/pi-tui/dist/stdin-buffer.js +20 -0
  403. package/packages/pi-tui/dist/stdin-buffer.js.map +1 -1
  404. package/packages/pi-tui/package.json +6 -1
  405. package/packages/pi-tui/src/__tests__/autocomplete.test.ts +20 -5
  406. package/packages/pi-tui/src/__tests__/stdin-buffer.test.ts +27 -0
  407. package/packages/pi-tui/src/stdin-buffer.ts +26 -0
  408. package/packages/pi-tui/tsconfig.tsbuildinfo +1 -1
  409. package/packages/rpc-client/package.json +6 -1
  410. package/pkg/package.json +1 -1
  411. package/scripts/install.js +512 -0
  412. package/scripts/lib/workspace-manifest.cjs +86 -0
  413. package/scripts/link-workspace-packages.cjs +5 -17
  414. package/scripts/postinstall.js +9 -178
  415. package/src/resources/extensions/claude-code-cli/readiness.ts +4 -3
  416. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +91 -63
  417. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +114 -12
  418. package/src/resources/extensions/cmux/index.ts +35 -10
  419. package/src/resources/extensions/github-sync/templates.ts +151 -0
  420. package/src/resources/extensions/github-sync/tests/templates.test.ts +59 -0
  421. package/src/resources/extensions/google-search/extension-manifest.json +5 -4
  422. package/src/resources/extensions/google-search/index.ts +9 -470
  423. package/src/resources/extensions/gsd/abandon-detect.ts +62 -0
  424. package/src/resources/extensions/gsd/auto/loop-deps.ts +14 -1
  425. package/src/resources/extensions/gsd/auto/loop.ts +104 -2
  426. package/src/resources/extensions/gsd/auto/phases.ts +123 -21
  427. package/src/resources/extensions/gsd/auto/resolve.ts +29 -0
  428. package/src/resources/extensions/gsd/auto/run-unit.ts +56 -4
  429. package/src/resources/extensions/gsd/auto/session.ts +28 -1
  430. package/src/resources/extensions/gsd/auto/turn-epoch.ts +108 -0
  431. package/src/resources/extensions/gsd/auto/types.ts +1 -1
  432. package/src/resources/extensions/gsd/auto-dispatch.ts +117 -16
  433. package/src/resources/extensions/gsd/auto-loop.ts +1 -1
  434. package/src/resources/extensions/gsd/auto-model-selection.ts +1 -1
  435. package/src/resources/extensions/gsd/auto-post-unit.ts +92 -3
  436. package/src/resources/extensions/gsd/auto-prompts.ts +28 -1
  437. package/src/resources/extensions/gsd/auto-recovery.ts +40 -1
  438. package/src/resources/extensions/gsd/auto-start.ts +48 -52
  439. package/src/resources/extensions/gsd/auto-timeout-recovery.ts +12 -5
  440. package/src/resources/extensions/gsd/auto-unit-closeout.ts +14 -3
  441. package/src/resources/extensions/gsd/auto-worktree.ts +122 -68
  442. package/src/resources/extensions/gsd/auto.ts +105 -35
  443. package/src/resources/extensions/gsd/bootstrap/agent-end-recovery.ts +34 -1
  444. package/src/resources/extensions/gsd/bootstrap/provider-error-resume.ts +6 -2
  445. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +11 -0
  446. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +18 -6
  447. package/src/resources/extensions/gsd/bootstrap/system-context.ts +13 -9
  448. package/src/resources/extensions/gsd/bootstrap/write-gate.ts +35 -2
  449. package/src/resources/extensions/gsd/clean-root-preflight.ts +111 -0
  450. package/src/resources/extensions/gsd/commands/handlers/workflow.ts +27 -8
  451. package/src/resources/extensions/gsd/commands-cmux.ts +10 -6
  452. package/src/resources/extensions/gsd/commands-extensions.ts +747 -41
  453. package/src/resources/extensions/gsd/commands-prefs-wizard.ts +898 -32
  454. package/src/resources/extensions/gsd/dispatch-guard.ts +26 -2
  455. package/src/resources/extensions/gsd/file-lock.ts +84 -11
  456. package/src/resources/extensions/gsd/git-service.ts +1 -0
  457. package/src/resources/extensions/gsd/gitignore.ts +2 -1
  458. package/src/resources/extensions/gsd/gsd-db.ts +92 -32
  459. package/src/resources/extensions/gsd/guided-flow-queue.ts +4 -1
  460. package/src/resources/extensions/gsd/guided-flow.ts +259 -10
  461. package/src/resources/extensions/gsd/health-widget.ts +3 -1
  462. package/src/resources/extensions/gsd/journal.ts +29 -3
  463. package/src/resources/extensions/gsd/key-manager.ts +22 -0
  464. package/src/resources/extensions/gsd/milestone-actions.ts +18 -0
  465. package/src/resources/extensions/gsd/milestone-summary-classifier.ts +42 -0
  466. package/src/resources/extensions/gsd/model-router.ts +42 -1
  467. package/src/resources/extensions/gsd/notifications.ts +27 -15
  468. package/src/resources/extensions/gsd/pre-execution-checks.ts +33 -7
  469. package/src/resources/extensions/gsd/preferences-types.ts +8 -0
  470. package/src/resources/extensions/gsd/prompts/discuss-headless.md +29 -2
  471. package/src/resources/extensions/gsd/prompts/discuss.md +29 -2
  472. package/src/resources/extensions/gsd/prompts/doctor-heal.md +5 -4
  473. package/src/resources/extensions/gsd/prompts/parallel-research-slices.md +5 -2
  474. package/src/resources/extensions/gsd/prompts/system.md +1 -0
  475. package/src/resources/extensions/gsd/reports.ts +5 -4
  476. package/src/resources/extensions/gsd/safety/evidence-collector.ts +119 -0
  477. package/src/resources/extensions/gsd/safety/file-change-validator.ts +16 -3
  478. package/src/resources/extensions/gsd/safety/safety-harness.ts +9 -0
  479. package/src/resources/extensions/gsd/state-transition-matrix.ts +152 -0
  480. package/src/resources/extensions/gsd/state.ts +35 -30
  481. package/src/resources/extensions/gsd/tests/auto-loop.test.ts +238 -4
  482. package/src/resources/extensions/gsd/tests/auto-mode-guards.test.ts +79 -0
  483. package/src/resources/extensions/gsd/tests/auto-paused-session-validation.test.ts +12 -0
  484. package/src/resources/extensions/gsd/tests/auto-recovery.test.ts +122 -0
  485. package/src/resources/extensions/gsd/tests/auto-start-bootstrap-await-3420.test.ts +141 -0
  486. package/src/resources/extensions/gsd/tests/auto-start-clean-runtime-db-gated.test.ts +63 -0
  487. package/src/resources/extensions/gsd/tests/auto-wrapup-inflight-guard.test.ts +23 -0
  488. package/src/resources/extensions/gsd/tests/clean-root-preflight.test.ts +186 -0
  489. package/src/resources/extensions/gsd/tests/cmux.test.ts +5 -9
  490. package/src/resources/extensions/gsd/tests/complete-milestone-false-merge.test.ts +15 -0
  491. package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +61 -1
  492. package/src/resources/extensions/gsd/tests/complete-slice.test.ts +2 -2
  493. package/src/resources/extensions/gsd/tests/complete-task.test.ts +2 -2
  494. package/src/resources/extensions/gsd/tests/custom-engine-loop-integration.test.ts +161 -0
  495. package/src/resources/extensions/gsd/tests/db-access-guardrails.test.ts +1 -0
  496. package/src/resources/extensions/gsd/tests/derive-state.test.ts +1 -2
  497. package/src/resources/extensions/gsd/tests/dispatch-complete-milestone-guard.test.ts +14 -9
  498. package/src/resources/extensions/gsd/tests/dispatch-guard-summary-db-mismatch.test.ts +77 -0
  499. package/src/resources/extensions/gsd/tests/dispatch-guard.test.ts +25 -0
  500. package/src/resources/extensions/gsd/tests/dispatch-missing-task-plans.test.ts +14 -0
  501. package/src/resources/extensions/gsd/tests/doctor-providers.test.ts +31 -0
  502. package/src/resources/extensions/gsd/tests/double-merge-guard.test.ts +1 -1
  503. package/src/resources/extensions/gsd/tests/ensure-db-open.test.ts +1 -1
  504. package/src/resources/extensions/gsd/tests/escalation.test.ts +1 -1
  505. package/src/resources/extensions/gsd/tests/exec-history.test.ts +113 -0
  506. package/src/resources/extensions/gsd/tests/execution-entry-missing-context-4671.test.ts +173 -0
  507. package/src/resources/extensions/gsd/tests/file-change-validator.test.ts +38 -0
  508. package/src/resources/extensions/gsd/tests/file-lock.test.ts +86 -12
  509. package/src/resources/extensions/gsd/tests/google-search-stub.test.ts +131 -0
  510. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +296 -1
  511. package/src/resources/extensions/gsd/tests/headless-milestone-parity.test.ts +117 -0
  512. package/src/resources/extensions/gsd/tests/integration/auto-worktree-milestone-merge.test.ts +30 -0
  513. package/src/resources/extensions/gsd/tests/integration/git-service.test.ts +4 -2
  514. package/src/resources/extensions/gsd/tests/integration/gitignore-tracked-gsd.test.ts +1 -0
  515. package/src/resources/extensions/gsd/tests/integration/idle-recovery.test.ts +30 -0
  516. package/src/resources/extensions/gsd/tests/integration/worktree-e2e.test.ts +11 -0
  517. package/src/resources/extensions/gsd/tests/issue-4540-regressions.test.ts +288 -0
  518. package/src/resources/extensions/gsd/tests/journal-integration.test.ts +66 -0
  519. package/src/resources/extensions/gsd/tests/key-manager.test.ts +2 -0
  520. package/src/resources/extensions/gsd/tests/mcp-client-security.test.ts +76 -0
  521. package/src/resources/extensions/gsd/tests/md-importer.test.ts +1 -1
  522. package/src/resources/extensions/gsd/tests/memory-pressure-stuck-state.test.ts +12 -0
  523. package/src/resources/extensions/gsd/tests/memory-store.test.ts +2 -2
  524. package/src/resources/extensions/gsd/tests/milestone-status-authoritative.test.ts +3 -3
  525. package/src/resources/extensions/gsd/tests/milestone-summary-classifier.test.ts +30 -0
  526. package/src/resources/extensions/gsd/tests/milestone-transition-state-rebuild.test.ts +4 -2
  527. package/src/resources/extensions/gsd/tests/parallel-commit-scope.test.ts +5 -0
  528. package/src/resources/extensions/gsd/tests/parallel-research-dispatch.test.ts +19 -0
  529. package/src/resources/extensions/gsd/tests/parallel-skill-prompt-integration.test.ts +150 -0
  530. package/src/resources/extensions/gsd/tests/plan-gate-failed-doctor-heal-hint.test.ts +37 -0
  531. package/src/resources/extensions/gsd/tests/pre-exec-gate-loop.test.ts +272 -0
  532. package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +337 -0
  533. package/src/resources/extensions/gsd/tests/prefs-wizard-coverage.test.ts +44 -0
  534. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +39 -25
  535. package/src/resources/extensions/gsd/tests/queue-auto-guard.test.ts +181 -0
  536. package/src/resources/extensions/gsd/tests/quick-auto-guard.test.ts +13 -7
  537. package/src/resources/extensions/gsd/tests/ready-phrase-no-files-4573.test.ts +388 -0
  538. package/src/resources/extensions/gsd/tests/require-slice-discussion-dispatch.test.ts +170 -0
  539. package/src/resources/extensions/gsd/tests/restore-tools-after-discuss.test.ts +9 -3
  540. package/src/resources/extensions/gsd/tests/resume-dispatch-worktree.test.ts +230 -0
  541. package/src/resources/extensions/gsd/tests/rewrite-docs-abandon-detect.test.ts +195 -0
  542. package/src/resources/extensions/gsd/tests/safety-harness-false-positives.test.ts +205 -0
  543. package/src/resources/extensions/gsd/tests/schema-v21-sequence.test.ts +413 -0
  544. package/src/resources/extensions/gsd/tests/session-start-footer.test.ts +32 -40
  545. package/src/resources/extensions/gsd/tests/stale-dirlistcache-4648.test.ts +112 -0
  546. package/src/resources/extensions/gsd/tests/stash-queued-context-files.test.ts +56 -0
  547. package/src/resources/extensions/gsd/tests/state-machine-full-walkthrough.test.ts +24 -0
  548. package/src/resources/extensions/gsd/tests/state-transition-matrix.test.ts +44 -0
  549. package/src/resources/extensions/gsd/tests/stuck-detection-coverage.test.ts +2 -2
  550. package/src/resources/extensions/gsd/tests/token-counter.test.ts +105 -1
  551. package/src/resources/extensions/gsd/tests/tool-compatibility.test.ts +107 -0
  552. package/src/resources/extensions/gsd/tests/triage-resolution.test.ts +50 -2
  553. package/src/resources/extensions/gsd/tests/turn-epoch.test.ts +162 -0
  554. package/src/resources/extensions/gsd/tests/uok-contracts.test.ts +51 -0
  555. package/src/resources/extensions/gsd/tests/uok-execution-graph.test.ts +16 -0
  556. package/src/resources/extensions/gsd/tests/uok-loop-adapter-writer.test.ts +65 -0
  557. package/src/resources/extensions/gsd/tests/uok-parity-report.test.ts +42 -0
  558. package/src/resources/extensions/gsd/tests/uok-plan-v2-wiring.test.ts +42 -2
  559. package/src/resources/extensions/gsd/tests/uok-writer.test.ts +75 -0
  560. package/src/resources/extensions/gsd/tests/validate-extension-package.test.ts +168 -0
  561. package/src/resources/extensions/gsd/tests/validate-milestone.test.ts +138 -5
  562. package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +25 -2
  563. package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +65 -2
  564. package/src/resources/extensions/gsd/tests/worktree-db.test.ts +35 -0
  565. package/src/resources/extensions/gsd/tests/worktree-journal-events.test.ts +6 -1
  566. package/src/resources/extensions/gsd/tests/worktree-resolver.test.ts +78 -5
  567. package/src/resources/extensions/gsd/tests/write-gate.test.ts +64 -0
  568. package/src/resources/extensions/gsd/token-counter.ts +22 -5
  569. package/src/resources/extensions/gsd/tools/complete-milestone.ts +15 -9
  570. package/src/resources/extensions/gsd/tools/complete-slice.ts +38 -0
  571. package/src/resources/extensions/gsd/tools/complete-task.ts +49 -0
  572. package/src/resources/extensions/gsd/uok/audit.ts +20 -2
  573. package/src/resources/extensions/gsd/uok/contracts.ts +65 -0
  574. package/src/resources/extensions/gsd/uok/dispatch-envelope.ts +56 -0
  575. package/src/resources/extensions/gsd/uok/execution-graph.ts +22 -0
  576. package/src/resources/extensions/gsd/uok/gitops.ts +6 -1
  577. package/src/resources/extensions/gsd/uok/loop-adapter.ts +45 -10
  578. package/src/resources/extensions/gsd/uok/parity-report.ts +84 -0
  579. package/src/resources/extensions/gsd/uok/plan-v2.ts +39 -8
  580. package/src/resources/extensions/gsd/uok/writer.ts +113 -0
  581. package/src/resources/extensions/gsd/workflow-logger.ts +23 -3
  582. package/src/resources/extensions/gsd/worktree-manager.ts +1 -0
  583. package/src/resources/extensions/gsd/worktree-resolver.ts +54 -9
  584. package/src/resources/extensions/mcp-client/auth.ts +12 -1
  585. package/src/resources/extensions/mcp-client/index.ts +129 -10
  586. package/src/resources/extensions/shared/cmux-events.ts +59 -0
  587. package/src/resources/extensions/shared/rtk-session-stats.ts +1 -2
  588. package/src/resources/skills/create-skill/SKILL.md +2 -2
  589. package/src/resources/skills/create-skill/references/gsd-skill-ecosystem.md +4 -4
  590. package/src/resources/skills/create-skill/workflows/audit-skill.md +4 -4
  591. package/src/resources/skills/create-skill/workflows/create-new-skill.md +5 -5
  592. package/src/resources/skills/verify-before-complete/SKILL.md +2 -1
  593. package/src/resources/skills/write-docs/SKILL.md +2 -1
  594. package/dist/web/standalone/.next/server/chunks/7461.js +0 -1
  595. package/dist/web/standalone/.next/static/chunks/2826.e59e8578e2e28639.js +0 -9
  596. /package/dist/web/standalone/.next/static/{n21VtX2hZlkpdEUO_nU4z → vidAVJkURvTJ0_V2-64ro}/_buildManifest.js +0 -0
  597. /package/dist/web/standalone/.next/static/{n21VtX2hZlkpdEUO_nU4z → vidAVJkURvTJ0_V2-64ro}/_ssgManifest.js +0 -0
@@ -35,15 +35,20 @@ describe("/gsd quick auto-mode guard (#2417)", () => {
35
35
 
36
36
  const quickBlock = quickBlockMatch[1];
37
37
 
38
- // Verify isAutoActive guard comes BEFORE handleQuick call
39
- const guardIndex = quickBlock.indexOf("isAutoActive()");
38
+ // Verify the shared auto-mode guard comes BEFORE handleQuick call.
39
+ // Accepts either the inline isAutoActive() check (legacy) or the shared
40
+ // requireNotAutoActive() helper (#4712).
41
+ const guardIndex = Math.max(
42
+ quickBlock.indexOf("isAutoActive()"),
43
+ quickBlock.indexOf("requireNotAutoActive("),
44
+ );
40
45
  const handleQuickIndex = quickBlock.indexOf("handleQuick(");
41
46
 
42
- assert.ok(guardIndex !== -1, "isAutoActive() guard exists in quick command block");
47
+ assert.ok(guardIndex !== -1, "auto-mode guard exists in quick command block");
43
48
  assert.ok(handleQuickIndex !== -1, "handleQuick() call exists in quick command block");
44
49
  assert.ok(
45
50
  guardIndex < handleQuickIndex,
46
- "isAutoActive() guard appears before handleQuick() call",
51
+ "auto-mode guard appears before handleQuick() call",
47
52
  );
48
53
  });
49
54
 
@@ -59,10 +64,11 @@ describe("/gsd quick auto-mode guard (#2417)", () => {
59
64
  "utf-8",
60
65
  );
61
66
 
62
- // The error message should tell the user to stop auto-mode first
67
+ // The shared helper assembles the message dynamically from the command name,
68
+ // so assert on the message fragment and the /gsd stop instruction.
63
69
  assert.ok(
64
- src.includes("/gsd quick cannot run while auto-mode is active"),
65
- "error message explains that quick cannot run during auto-mode",
70
+ src.includes("cannot run while auto-mode is active"),
71
+ "error message explains that the command cannot run during auto-mode",
66
72
  );
67
73
  assert.ok(
68
74
  src.includes("/gsd stop"),
@@ -0,0 +1,388 @@
1
+ /**
2
+ * GSD-2 / guided-flow — regression tests for #4573
3
+ *
4
+ * Covers two recovery paths:
5
+ * - maybeHandleReadyPhraseWithoutFiles: nudge when LLM emits
6
+ * "Milestone M001 ready." without writing CONTEXT.md / ROADMAP.md
7
+ * - maybeHandleEmptyIntentTurn: nudge when LLM narrates intent but
8
+ * emits no tool-use blocks
9
+ */
10
+
11
+ import { describe, test, beforeEach } from "node:test";
12
+ import assert from "node:assert/strict";
13
+ import { mkdtempSync, mkdirSync, rmSync, writeFileSync } from "node:fs";
14
+ import { join } from "node:path";
15
+ import { tmpdir } from "node:os";
16
+
17
+ import {
18
+ setPendingAutoStart,
19
+ clearPendingAutoStart,
20
+ maybeHandleReadyPhraseWithoutFiles,
21
+ maybeHandleEmptyIntentTurn,
22
+ resetEmptyTurnCounter,
23
+ } from "../guided-flow.ts";
24
+
25
+ // ─── Test harness ──────────────────────────────────────────────────────────
26
+
27
+ interface MockCapture {
28
+ notifies: Array<{ msg: string; level: string }>;
29
+ messages: Array<{ payload: any; options: any }>;
30
+ }
31
+
32
+ function mkCapture(): MockCapture {
33
+ return { notifies: [], messages: [] };
34
+ }
35
+
36
+ function mkCtx(cap: MockCapture): any {
37
+ return {
38
+ ui: {
39
+ notify: (msg: string, level: string) => {
40
+ cap.notifies.push({ msg, level });
41
+ },
42
+ },
43
+ };
44
+ }
45
+
46
+ function mkPi(cap: MockCapture, opts: { sendThrows?: boolean } = {}): any {
47
+ return {
48
+ sendMessage: (payload: any, options: any) => {
49
+ if (opts.sendThrows) throw new Error("send failed");
50
+ cap.messages.push({ payload, options });
51
+ },
52
+ setActiveTools: () => undefined,
53
+ getActiveTools: () => [],
54
+ };
55
+ }
56
+
57
+ function mkBase(): string {
58
+ const base = mkdtempSync(join(tmpdir(), "gsd-4573-"));
59
+ mkdirSync(join(base, ".gsd", "milestones", "M001"), { recursive: true });
60
+ return base;
61
+ }
62
+
63
+ function assistantMsg(text: string, opts: { toolUse?: boolean } = {}): any {
64
+ const content: any[] = [];
65
+ if (text) content.push({ type: "text", text });
66
+ if (opts.toolUse) content.push({ type: "tool_use", name: "whatever", input: {} });
67
+ return { role: "assistant", content };
68
+ }
69
+
70
+ // ─── ready-phrase recovery (Layer 2) ───────────────────────────────────────
71
+
72
+ describe("#4573 maybeHandleReadyPhraseWithoutFiles", () => {
73
+ beforeEach(() => {
74
+ clearPendingAutoStart();
75
+ resetEmptyTurnCounter();
76
+ });
77
+
78
+ test("no pending entry → no-op", () => {
79
+ const cap = mkCapture();
80
+ const event = { messages: [assistantMsg("Milestone M001 ready.")] };
81
+ const handled = maybeHandleReadyPhraseWithoutFiles(event);
82
+ assert.equal(handled, false);
83
+ assert.equal(cap.messages.length, 0);
84
+ });
85
+
86
+ test("pending entry, ready phrase, no files → notify + sendMessage", () => {
87
+ const base = mkBase();
88
+ try {
89
+ const cap = mkCapture();
90
+ setPendingAutoStart(base, {
91
+ basePath: base,
92
+ milestoneId: "M001",
93
+ ctx: mkCtx(cap),
94
+ pi: mkPi(cap),
95
+ });
96
+ const handled = maybeHandleReadyPhraseWithoutFiles({
97
+ messages: [assistantMsg("Milestone M001 ready.")],
98
+ });
99
+ assert.equal(handled, true);
100
+ assert.equal(cap.messages.length, 1);
101
+ assert.equal(cap.messages[0].payload.customType, "gsd-ready-no-files");
102
+ assert.equal(cap.messages[0].options.triggerTurn, true);
103
+ assert.ok(
104
+ cap.notifies.some((n) => /rejected/.test(n.msg)),
105
+ "user notified about rejection",
106
+ );
107
+ } finally {
108
+ clearPendingAutoStart();
109
+ }
110
+ });
111
+
112
+ test("retry cap — after MAX_READY_REJECTS the nudge stops and entry clears", () => {
113
+ const base = mkBase();
114
+ try {
115
+ const cap = mkCapture();
116
+ setPendingAutoStart(base, {
117
+ basePath: base,
118
+ milestoneId: "M001",
119
+ ctx: mkCtx(cap),
120
+ pi: mkPi(cap),
121
+ });
122
+ const event = { messages: [assistantMsg("Milestone M001 ready.")] };
123
+
124
+ const first = maybeHandleReadyPhraseWithoutFiles(event);
125
+ const second = maybeHandleReadyPhraseWithoutFiles(event);
126
+ const third = maybeHandleReadyPhraseWithoutFiles(event); // > MAX
127
+
128
+ assert.equal(first, true);
129
+ assert.equal(second, true);
130
+ assert.equal(third, true); // still returns true (handled via give-up)
131
+ assert.equal(cap.messages.length, 2, "only 2 nudges sent (MAX_READY_REJECTS=2)");
132
+ assert.ok(
133
+ cap.notifies.some((n) => /Stopping auto-nudge/.test(n.msg)),
134
+ "gives up with error notify",
135
+ );
136
+
137
+ // After giving up, a fresh re-entry starts clean
138
+ const fourth = maybeHandleReadyPhraseWithoutFiles(event);
139
+ assert.equal(fourth, false, "pending entry was cleared — nothing to handle");
140
+ } finally {
141
+ clearPendingAutoStart();
142
+ }
143
+ });
144
+
145
+ test("files present → no nudge (happy path already fired)", () => {
146
+ const base = mkBase();
147
+ try {
148
+ writeFileSync(join(base, ".gsd", "milestones", "M001", "M001-CONTEXT.md"), "# ctx");
149
+ const cap = mkCapture();
150
+ setPendingAutoStart(base, {
151
+ basePath: base,
152
+ milestoneId: "M001",
153
+ ctx: mkCtx(cap),
154
+ pi: mkPi(cap),
155
+ });
156
+ const handled = maybeHandleReadyPhraseWithoutFiles({
157
+ messages: [assistantMsg("Milestone M001 ready.")],
158
+ });
159
+ assert.equal(handled, false);
160
+ assert.equal(cap.messages.length, 0);
161
+ } finally {
162
+ clearPendingAutoStart();
163
+ }
164
+ });
165
+
166
+ test("last message lacks ready phrase → no-op", () => {
167
+ const base = mkBase();
168
+ try {
169
+ const cap = mkCapture();
170
+ setPendingAutoStart(base, {
171
+ basePath: base,
172
+ milestoneId: "M001",
173
+ ctx: mkCtx(cap),
174
+ pi: mkPi(cap),
175
+ });
176
+ const handled = maybeHandleReadyPhraseWithoutFiles({
177
+ messages: [assistantMsg("Let me think about the slices first.")],
178
+ });
179
+ assert.equal(handled, false);
180
+ assert.equal(cap.messages.length, 0);
181
+ } finally {
182
+ clearPendingAutoStart();
183
+ }
184
+ });
185
+
186
+ test("fresh entry after give-up resets counter", () => {
187
+ const base = mkBase();
188
+ try {
189
+ const cap = mkCapture();
190
+ // First cycle: exhaust cap
191
+ setPendingAutoStart(base, {
192
+ basePath: base,
193
+ milestoneId: "M001",
194
+ ctx: mkCtx(cap),
195
+ pi: mkPi(cap),
196
+ });
197
+ const event = { messages: [assistantMsg("Milestone M001 ready.")] };
198
+ maybeHandleReadyPhraseWithoutFiles(event);
199
+ maybeHandleReadyPhraseWithoutFiles(event);
200
+ maybeHandleReadyPhraseWithoutFiles(event); // clears entry
201
+
202
+ // New /gsd run — re-seeds entry; counter must be 0 again
203
+ cap.messages.length = 0;
204
+ setPendingAutoStart(base, {
205
+ basePath: base,
206
+ milestoneId: "M001",
207
+ ctx: mkCtx(cap),
208
+ pi: mkPi(cap),
209
+ });
210
+ const handled = maybeHandleReadyPhraseWithoutFiles(event);
211
+ assert.equal(handled, true);
212
+ assert.equal(cap.messages.length, 1, "fresh entry fires nudge again");
213
+ } finally {
214
+ clearPendingAutoStart();
215
+ }
216
+ });
217
+ });
218
+
219
+ // ─── empty-turn recovery (Layer 3) ────────────────────────────────────────
220
+
221
+ describe("#4573 maybeHandleEmptyIntentTurn", () => {
222
+ beforeEach(() => {
223
+ clearPendingAutoStart();
224
+ resetEmptyTurnCounter();
225
+ });
226
+
227
+ test("no pending entry + isAuto false → no-op (interactive discuss is user-driven)", () => {
228
+ const event = { messages: [assistantMsg("I'll write the CONTEXT.md now.")] };
229
+ const handled = maybeHandleEmptyIntentTurn(event, false);
230
+ assert.equal(handled, false);
231
+ });
232
+
233
+ test("text-only turn WITHOUT commit phrase → not flagged (legitimate text)", () => {
234
+ const base = mkBase();
235
+ try {
236
+ const cap = mkCapture();
237
+ setPendingAutoStart(base, {
238
+ basePath: base,
239
+ milestoneId: "M001",
240
+ ctx: mkCtx(cap),
241
+ pi: mkPi(cap),
242
+ });
243
+ const handled = maybeHandleEmptyIntentTurn(
244
+ { messages: [assistantMsg("Here is the roadmap preview — three slices.")] },
245
+ false,
246
+ );
247
+ assert.equal(handled, false);
248
+ assert.equal(cap.messages.length, 0);
249
+ } finally {
250
+ clearPendingAutoStart();
251
+ }
252
+ });
253
+
254
+ test("text-only turn ending in question → treated as user-handoff, not flagged", () => {
255
+ const base = mkBase();
256
+ try {
257
+ const cap = mkCapture();
258
+ setPendingAutoStart(base, {
259
+ basePath: base,
260
+ milestoneId: "M001",
261
+ ctx: mkCtx(cap),
262
+ pi: mkPi(cap),
263
+ });
264
+ const handled = maybeHandleEmptyIntentTurn(
265
+ { messages: [assistantMsg("Ready to write, or want to adjust?")] },
266
+ false,
267
+ );
268
+ assert.equal(handled, false);
269
+ } finally {
270
+ clearPendingAutoStart();
271
+ }
272
+ });
273
+
274
+ test("commit-intent phrase WITHOUT tool call → nudge fires", () => {
275
+ const base = mkBase();
276
+ try {
277
+ const cap = mkCapture();
278
+ setPendingAutoStart(base, {
279
+ basePath: base,
280
+ milestoneId: "M001",
281
+ ctx: mkCtx(cap),
282
+ pi: mkPi(cap),
283
+ });
284
+ const handled = maybeHandleEmptyIntentTurn(
285
+ { messages: [assistantMsg("I'll now write the CONTEXT.md file.")] },
286
+ false,
287
+ );
288
+ assert.equal(handled, true);
289
+ assert.equal(cap.messages.length, 1);
290
+ assert.equal(cap.messages[0].payload.customType, "gsd-empty-turn-recovery");
291
+ } finally {
292
+ clearPendingAutoStart();
293
+ }
294
+ });
295
+
296
+ test("commit-intent WITH tool-use block → not flagged", () => {
297
+ const base = mkBase();
298
+ try {
299
+ const cap = mkCapture();
300
+ setPendingAutoStart(base, {
301
+ basePath: base,
302
+ milestoneId: "M001",
303
+ ctx: mkCtx(cap),
304
+ pi: mkPi(cap),
305
+ });
306
+ const handled = maybeHandleEmptyIntentTurn(
307
+ { messages: [assistantMsg("I'll write the file now.", { toolUse: true })] },
308
+ false,
309
+ );
310
+ assert.equal(handled, false);
311
+ assert.equal(cap.messages.length, 0);
312
+ } finally {
313
+ clearPendingAutoStart();
314
+ }
315
+ });
316
+
317
+ test("ready phrase is NOT treated as empty-turn (handled by other recovery path)", () => {
318
+ const base = mkBase();
319
+ try {
320
+ const cap = mkCapture();
321
+ setPendingAutoStart(base, {
322
+ basePath: base,
323
+ milestoneId: "M001",
324
+ ctx: mkCtx(cap),
325
+ pi: mkPi(cap),
326
+ });
327
+ const handled = maybeHandleEmptyIntentTurn(
328
+ { messages: [assistantMsg("Milestone M001 ready.")] },
329
+ false,
330
+ );
331
+ assert.equal(handled, false);
332
+ } finally {
333
+ clearPendingAutoStart();
334
+ }
335
+ });
336
+
337
+ test("empty-turn retry cap — stops after MAX_EMPTY_TURN_RETRIES", () => {
338
+ const base = mkBase();
339
+ try {
340
+ const cap = mkCapture();
341
+ setPendingAutoStart(base, {
342
+ basePath: base,
343
+ milestoneId: "M001",
344
+ ctx: mkCtx(cap),
345
+ pi: mkPi(cap),
346
+ });
347
+ const event = { messages: [assistantMsg("I'll write the CONTEXT.md file.")] };
348
+
349
+ maybeHandleEmptyIntentTurn(event, false); // 1
350
+ maybeHandleEmptyIntentTurn(event, false); // 2
351
+ const third = maybeHandleEmptyIntentTurn(event, false); // > cap
352
+
353
+ assert.equal(cap.messages.length, 2, "only 2 nudges sent");
354
+ assert.equal(third, false, "after cap, no further injection");
355
+ assert.ok(
356
+ cap.notifies.some((n) => /Stopping auto-nudge/.test(n.msg)),
357
+ "user notified of give-up",
358
+ );
359
+ } finally {
360
+ clearPendingAutoStart();
361
+ }
362
+ });
363
+
364
+ test("resetEmptyTurnCounter clears state after a successful tool-use turn", () => {
365
+ const base = mkBase();
366
+ try {
367
+ const cap = mkCapture();
368
+ setPendingAutoStart(base, {
369
+ basePath: base,
370
+ milestoneId: "M001",
371
+ ctx: mkCtx(cap),
372
+ pi: mkPi(cap),
373
+ });
374
+ const event = { messages: [assistantMsg("I'll write the CONTEXT.md file.")] };
375
+
376
+ maybeHandleEmptyIntentTurn(event, false); // 1
377
+ maybeHandleEmptyIntentTurn(event, false); // 2 — at cap
378
+ resetEmptyTurnCounter(); // simulate a successful tool-use turn in between
379
+
380
+ cap.messages.length = 0;
381
+ const after = maybeHandleEmptyIntentTurn(event, false);
382
+ assert.equal(after, true, "counter reset — nudge fires again");
383
+ assert.equal(cap.messages.length, 1);
384
+ } finally {
385
+ clearPendingAutoStart();
386
+ }
387
+ });
388
+ });
@@ -0,0 +1,170 @@
1
+ /**
2
+ * Regression tests for #3454: auto-dispatch must honour
3
+ * require_slice_discussion and pause before plan-slice when:
4
+ * 1. state.phase === "planning"
5
+ * 2. require_slice_discussion is enabled in preferences
6
+ * 3. state.activeSlice is non-null
7
+ * 4. the slice has no CONTEXT file on disk
8
+ *
9
+ * Exercises the dispatch rule from DISPATCH_RULES directly with a
10
+ * DispatchContext built against a real temp directory — no source-string
11
+ * matching, no brittle rename dependencies.
12
+ */
13
+ import { test, describe } from "node:test";
14
+ import assert from "node:assert/strict";
15
+ import { mkdtempSync, mkdirSync, writeFileSync, rmSync } from "node:fs";
16
+ import { join } from "node:path";
17
+ import { tmpdir } from "node:os";
18
+ import { DISPATCH_RULES, type DispatchContext } from "../auto-dispatch.ts";
19
+ import type { GSDState } from "../types.ts";
20
+ import type { GSDPreferences } from "../preferences.ts";
21
+
22
+ // Use a stable token (the preference name) for rule lookup instead of the
23
+ // full human-readable title — copy edits to the rule name will not break
24
+ // these tests as long as the rule still references the preference it gates.
25
+ const RULE_NAME_TOKEN = "require_slice_discussion";
26
+
27
+ function findRule() {
28
+ const matches = DISPATCH_RULES.filter(r => r.name.includes(RULE_NAME_TOKEN));
29
+ if (matches.length !== 1) {
30
+ throw new Error(`expected exactly one dispatch rule containing "${RULE_NAME_TOKEN}", found ${matches.length}`);
31
+ }
32
+ return matches[0];
33
+ }
34
+
35
+ function buildState(overrides: Partial<GSDState> = {}): GSDState {
36
+ return {
37
+ activeMilestone: { id: "M001", title: "Test milestone" },
38
+ activeSlice: { id: "S01", title: "Test slice" },
39
+ activeTask: null,
40
+ phase: "planning",
41
+ recentDecisions: [],
42
+ blockers: [],
43
+ nextAction: "",
44
+ registry: [],
45
+ ...overrides,
46
+ };
47
+ }
48
+
49
+ function makeBasePath(prefix: string): string {
50
+ const dir = mkdtempSync(join(tmpdir(), `gsd-req-slice-${prefix}-`));
51
+ mkdirSync(join(dir, ".gsd", "milestones", "M001", "slices", "S01"), { recursive: true });
52
+ return dir;
53
+ }
54
+
55
+ function buildCtx(
56
+ basePath: string,
57
+ prefs: GSDPreferences | undefined,
58
+ state: GSDState = buildState(),
59
+ ): DispatchContext {
60
+ return {
61
+ basePath,
62
+ mid: "M001",
63
+ midTitle: "Test milestone",
64
+ state,
65
+ prefs,
66
+ };
67
+ }
68
+
69
+ describe("require_slice_discussion dispatch rule (#3454)", () => {
70
+ // ─── Positive case: rule fires ────────────────────────────────────────
71
+
72
+ test("returns stop action when preference enabled and CONTEXT missing", async () => {
73
+ const basePath = makeBasePath("fire");
74
+ try {
75
+ const prefs = { phases: { require_slice_discussion: true } } as unknown as GSDPreferences;
76
+ const action = await findRule().match(buildCtx(basePath, prefs));
77
+ assert.ok(action, "rule must return a non-null action when pausing is required");
78
+ assert.strictEqual(action!.action, "stop");
79
+ if (action!.action === "stop") {
80
+ assert.match(action!.reason, /S01/);
81
+ assert.match(action!.reason, /require_slice_discussion/);
82
+ assert.match(action!.reason, /\/gsd discuss/);
83
+ assert.strictEqual(action!.level, "info");
84
+ }
85
+ } finally {
86
+ rmSync(basePath, { recursive: true, force: true });
87
+ }
88
+ });
89
+
90
+ // ─── Negative cases: rule falls through ───────────────────────────────
91
+
92
+ test("falls through (null) when preference is disabled", async () => {
93
+ const basePath = makeBasePath("pref-disabled");
94
+ try {
95
+ const prefs = { phases: { require_slice_discussion: false } } as unknown as GSDPreferences;
96
+ const action = await findRule().match(buildCtx(basePath, prefs));
97
+ assert.strictEqual(action, null);
98
+ } finally {
99
+ rmSync(basePath, { recursive: true, force: true });
100
+ }
101
+ });
102
+
103
+ test("falls through (null) when preference is absent / undefined", async () => {
104
+ const basePath = makeBasePath("pref-absent");
105
+ try {
106
+ const action = await findRule().match(buildCtx(basePath, undefined));
107
+ assert.strictEqual(action, null);
108
+ } finally {
109
+ rmSync(basePath, { recursive: true, force: true });
110
+ }
111
+ });
112
+
113
+ test("falls through (null) when phase is not 'planning'", async () => {
114
+ const basePath = makeBasePath("wrong-phase");
115
+ try {
116
+ const prefs = { phases: { require_slice_discussion: true } } as unknown as GSDPreferences;
117
+ const state = buildState({ phase: "executing" });
118
+ const action = await findRule().match(buildCtx(basePath, prefs, state));
119
+ assert.strictEqual(action, null);
120
+ } finally {
121
+ rmSync(basePath, { recursive: true, force: true });
122
+ }
123
+ });
124
+
125
+ test("falls through (null) when no active slice", async () => {
126
+ const basePath = makeBasePath("no-slice");
127
+ try {
128
+ const prefs = { phases: { require_slice_discussion: true } } as unknown as GSDPreferences;
129
+ const state = buildState({ activeSlice: null });
130
+ const action = await findRule().match(buildCtx(basePath, prefs, state));
131
+ assert.strictEqual(action, null);
132
+ } finally {
133
+ rmSync(basePath, { recursive: true, force: true });
134
+ }
135
+ });
136
+
137
+ // ─── Context-file present: should not pause ───────────────────────────
138
+
139
+ test("falls through (null) when CONTEXT file already exists on disk", async () => {
140
+ const basePath = makeBasePath("ctx-present");
141
+ try {
142
+ // Seed the CONTEXT file that /gsd discuss would have written.
143
+ const sliceDir = join(basePath, ".gsd", "milestones", "M001", "slices", "S01");
144
+ writeFileSync(join(sliceDir, "S01-CONTEXT.md"), "# Discussion notes\n", "utf-8");
145
+
146
+ const prefs = { phases: { require_slice_discussion: true } } as unknown as GSDPreferences;
147
+ const action = await findRule().match(buildCtx(basePath, prefs));
148
+ assert.strictEqual(
149
+ action,
150
+ null,
151
+ "once the slice has a CONTEXT file, the rule must fall through so planning proceeds",
152
+ );
153
+ } finally {
154
+ rmSync(basePath, { recursive: true, force: true });
155
+ }
156
+ });
157
+
158
+ // ─── Rule ordering: must run before the plan-slice rule ──────────────
159
+
160
+ test("rule is ordered before the 'planning → plan-slice' rule", () => {
161
+ const discussIdx = DISPATCH_RULES.findIndex(r => r.name.includes(RULE_NAME_TOKEN));
162
+ const planIdx = DISPATCH_RULES.findIndex(r => r.name.startsWith("planning → plan-slice"));
163
+ assert.ok(discussIdx >= 0, "require_slice_discussion rule must be registered");
164
+ assert.ok(planIdx >= 0, "plan-slice rule must be registered");
165
+ assert.ok(
166
+ discussIdx < planIdx,
167
+ "require_slice_discussion rule must be ordered before plan-slice so it preempts dispatch",
168
+ );
169
+ });
170
+ });
@@ -45,9 +45,15 @@ describe('restore tools after discuss flow scoping (#3628)', () => {
45
45
  })
46
46
 
47
47
  it('savedTools is restored after sendMessage', () => {
48
- // Find the sendMessage call
49
- const sendMsg = src.indexOf('triggerTurn: true')
50
- assert.ok(sendMsg !== -1, 'sendMessage with triggerTurn must exist')
48
+ // #4573: guided-flow.ts now contains multiple `triggerTurn: true` calls
49
+ // (ready-phrase and empty-turn recovery paths). The discuss-flow scoping
50
+ // sendMessage is the one that follows `savedTools = currentTools`, so
51
+ // anchor the search there rather than at the first `triggerTurn: true`.
52
+ const savedToolsAssign = src.indexOf('savedTools = currentTools')
53
+ assert.ok(savedToolsAssign !== -1, 'savedTools = currentTools must exist')
54
+
55
+ const sendMsg = src.indexOf('triggerTurn: true', savedToolsAssign)
56
+ assert.ok(sendMsg !== -1, 'discuss-flow sendMessage with triggerTurn must exist after savedTools capture')
51
57
 
52
58
  // After sendMessage, savedTools should be restored via setActiveTools
53
59
  const afterSend = src.slice(sendMsg, sendMsg + 500)