gsd-pi 2.71.0-dev.06b86c6 → 2.71.0-dev.246c32d

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 (289) hide show
  1. package/README.md +34 -1
  2. package/dist/cli.js +17 -0
  3. package/dist/headless-events.d.ts +2 -0
  4. package/dist/headless-events.js +7 -0
  5. package/dist/headless.js +16 -3
  6. package/dist/mcp-server.js +37 -14
  7. package/dist/resource-loader.js +6 -3
  8. package/dist/resources/agents/debugger.md +58 -0
  9. package/dist/resources/agents/doc-writer.md +43 -0
  10. package/dist/resources/agents/git-ops.md +56 -0
  11. package/dist/resources/agents/javascript-pro.md +46 -271
  12. package/dist/resources/agents/planner.md +55 -0
  13. package/dist/resources/agents/refactorer.md +47 -0
  14. package/dist/resources/agents/reviewer.md +48 -0
  15. package/dist/resources/agents/security.md +59 -0
  16. package/dist/resources/agents/tester.md +50 -0
  17. package/dist/resources/agents/typescript-pro.md +41 -235
  18. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +113 -10
  19. package/dist/resources/extensions/gsd/auto/infra-errors.js +34 -0
  20. package/dist/resources/extensions/gsd/auto/loop.js +32 -1
  21. package/dist/resources/extensions/gsd/auto/phases.js +5 -1
  22. package/dist/resources/extensions/gsd/auto/session.js +11 -0
  23. package/dist/resources/extensions/gsd/auto-dashboard.js +22 -16
  24. package/dist/resources/extensions/gsd/auto-model-selection.js +10 -2
  25. package/dist/resources/extensions/gsd/auto-prompts.js +88 -33
  26. package/dist/resources/extensions/gsd/auto-start.js +33 -6
  27. package/dist/resources/extensions/gsd/auto-worktree.js +1 -1
  28. package/dist/resources/extensions/gsd/auto.js +56 -0
  29. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +3 -3
  30. package/dist/resources/extensions/gsd/bootstrap/register-shortcuts.js +63 -51
  31. package/dist/resources/extensions/gsd/commands/handlers/auto.js +10 -33
  32. package/dist/resources/extensions/gsd/commands/handlers/core.js +56 -11
  33. package/dist/resources/extensions/gsd/commands/handlers/notifications-handler.js +15 -6
  34. package/dist/resources/extensions/gsd/commands/handlers/workflow.js +4 -10
  35. package/dist/resources/extensions/gsd/dashboard-overlay.js +8 -3
  36. package/dist/resources/extensions/gsd/doctor-providers.js +23 -0
  37. package/dist/resources/extensions/gsd/error-classifier.js +4 -1
  38. package/dist/resources/extensions/gsd/forensics.js +19 -6
  39. package/dist/resources/extensions/gsd/gate-registry.js +208 -0
  40. package/dist/resources/extensions/gsd/gsd-db.js +41 -0
  41. package/dist/resources/extensions/gsd/guided-flow.js +5 -10
  42. package/dist/resources/extensions/gsd/metrics.js +1 -0
  43. package/dist/resources/extensions/gsd/milestone-actions.js +10 -4
  44. package/dist/resources/extensions/gsd/milestone-validation-gates.js +11 -12
  45. package/dist/resources/extensions/gsd/notification-overlay.js +42 -13
  46. package/dist/resources/extensions/gsd/notification-store.js +35 -4
  47. package/dist/resources/extensions/gsd/notification-widget.js +5 -13
  48. package/dist/resources/extensions/gsd/parallel-monitor-overlay.js +8 -3
  49. package/dist/resources/extensions/gsd/prompt-validation.js +126 -0
  50. package/dist/resources/extensions/gsd/prompts/complete-slice.md +3 -1
  51. package/dist/resources/extensions/gsd/prompts/execute-task.md +2 -0
  52. package/dist/resources/extensions/gsd/prompts/validate-milestone.md +2 -0
  53. package/dist/resources/extensions/gsd/session-model-override.js +25 -0
  54. package/dist/resources/extensions/gsd/shortcut-defs.js +40 -0
  55. package/dist/resources/extensions/gsd/state.js +9 -2
  56. package/dist/resources/extensions/gsd/tools/complete-slice.js +52 -1
  57. package/dist/resources/extensions/gsd/tools/complete-task.js +51 -1
  58. package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +4 -1
  59. package/dist/resources/extensions/ollama/index.js +13 -5
  60. package/dist/resources/extensions/shared/gsd-phase-state.js +35 -0
  61. package/dist/resources/extensions/subagent/agents.js +8 -0
  62. package/dist/resources/extensions/subagent/index.js +17 -0
  63. package/dist/startup-model-validation.d.ts +0 -1
  64. package/dist/startup-model-validation.js +6 -2
  65. package/dist/web/standalone/.next/BUILD_ID +1 -1
  66. package/dist/web/standalone/.next/app-path-routes-manifest.json +17 -17
  67. package/dist/web/standalone/.next/build-manifest.json +2 -2
  68. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  69. package/dist/web/standalone/.next/required-server-files.json +1 -1
  70. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  71. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  72. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  73. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  74. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  75. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  76. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  77. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  78. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  79. package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
  80. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  81. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  82. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  83. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  84. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  85. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  86. package/dist/web/standalone/.next/server/app/index.html +1 -1
  87. package/dist/web/standalone/.next/server/app/index.rsc +1 -1
  88. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  89. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
  90. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  91. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
  92. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  93. package/dist/web/standalone/.next/server/app-paths-manifest.json +17 -17
  94. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  95. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  96. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  97. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  98. package/dist/web/standalone/server.js +1 -1
  99. package/package.json +1 -1
  100. package/packages/mcp-server/dist/server.d.ts +12 -1
  101. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  102. package/packages/mcp-server/dist/server.js +90 -42
  103. package/packages/mcp-server/dist/server.js.map +1 -1
  104. package/packages/mcp-server/dist/workflow-tools.js +1 -1
  105. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  106. package/packages/mcp-server/src/server.ts +110 -38
  107. package/packages/mcp-server/src/workflow-tools.ts +1 -1
  108. package/packages/pi-ai/dist/providers/anthropic-auth.test.d.ts +2 -0
  109. package/packages/pi-ai/dist/providers/anthropic-auth.test.d.ts.map +1 -0
  110. package/packages/pi-ai/dist/providers/anthropic-auth.test.js +20 -0
  111. package/packages/pi-ai/dist/providers/anthropic-auth.test.js.map +1 -0
  112. package/packages/pi-ai/dist/providers/anthropic.d.ts +2 -1
  113. package/packages/pi-ai/dist/providers/anthropic.d.ts.map +1 -1
  114. package/packages/pi-ai/dist/providers/anthropic.js +7 -4
  115. package/packages/pi-ai/dist/providers/anthropic.js.map +1 -1
  116. package/packages/pi-ai/src/providers/anthropic-auth.test.ts +32 -0
  117. package/packages/pi-ai/src/providers/anthropic.ts +8 -4
  118. package/packages/pi-coding-agent/dist/core/agent-session-renderable-tools.test.d.ts +2 -0
  119. package/packages/pi-coding-agent/dist/core/agent-session-renderable-tools.test.d.ts.map +1 -0
  120. package/packages/pi-coding-agent/dist/core/agent-session-renderable-tools.test.js +61 -0
  121. package/packages/pi-coding-agent/dist/core/agent-session-renderable-tools.test.js.map +1 -0
  122. package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
  123. package/packages/pi-coding-agent/dist/core/agent-session.js +2 -1
  124. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  125. package/packages/pi-coding-agent/dist/core/auth-storage.d.ts +10 -0
  126. package/packages/pi-coding-agent/dist/core/auth-storage.d.ts.map +1 -1
  127. package/packages/pi-coding-agent/dist/core/auth-storage.js +27 -0
  128. package/packages/pi-coding-agent/dist/core/auth-storage.js.map +1 -1
  129. package/packages/pi-coding-agent/dist/core/auth-storage.test.js +85 -0
  130. package/packages/pi-coding-agent/dist/core/auth-storage.test.js.map +1 -1
  131. package/packages/pi-coding-agent/dist/core/model-resolver-initial-model-auth.test.d.ts +2 -0
  132. package/packages/pi-coding-agent/dist/core/model-resolver-initial-model-auth.test.d.ts.map +1 -0
  133. package/packages/pi-coding-agent/dist/core/model-resolver-initial-model-auth.test.js +64 -0
  134. package/packages/pi-coding-agent/dist/core/model-resolver-initial-model-auth.test.js.map +1 -0
  135. package/packages/pi-coding-agent/dist/core/model-resolver.d.ts.map +1 -1
  136. package/packages/pi-coding-agent/dist/core/model-resolver.js +22 -18
  137. package/packages/pi-coding-agent/dist/core/model-resolver.js.map +1 -1
  138. package/packages/pi-coding-agent/dist/core/model-resolver.test.d.ts +8 -0
  139. package/packages/pi-coding-agent/dist/core/model-resolver.test.d.ts.map +1 -0
  140. package/packages/pi-coding-agent/dist/core/model-resolver.test.js +75 -0
  141. package/packages/pi-coding-agent/dist/core/model-resolver.test.js.map +1 -0
  142. package/packages/pi-coding-agent/dist/core/retry-handler.d.ts +5 -0
  143. package/packages/pi-coding-agent/dist/core/retry-handler.d.ts.map +1 -1
  144. package/packages/pi-coding-agent/dist/core/retry-handler.js +55 -1
  145. package/packages/pi-coding-agent/dist/core/retry-handler.js.map +1 -1
  146. package/packages/pi-coding-agent/dist/core/retry-handler.test.js +57 -0
  147. package/packages/pi-coding-agent/dist/core/retry-handler.test.js.map +1 -1
  148. package/packages/pi-coding-agent/dist/core/sdk.d.ts +11 -0
  149. package/packages/pi-coding-agent/dist/core/sdk.d.ts.map +1 -1
  150. package/packages/pi-coding-agent/dist/core/sdk.js +38 -5
  151. package/packages/pi-coding-agent/dist/core/sdk.js.map +1 -1
  152. package/packages/pi-coding-agent/dist/core/sdk.test.d.ts +2 -0
  153. package/packages/pi-coding-agent/dist/core/sdk.test.d.ts.map +1 -0
  154. package/packages/pi-coding-agent/dist/core/sdk.test.js +71 -0
  155. package/packages/pi-coding-agent/dist/core/sdk.test.js.map +1 -0
  156. package/packages/pi-coding-agent/dist/index.d.ts +1 -1
  157. package/packages/pi-coding-agent/dist/index.d.ts.map +1 -1
  158. package/packages/pi-coding-agent/dist/index.js +1 -1
  159. package/packages/pi-coding-agent/dist/index.js.map +1 -1
  160. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/login-dialog.test.d.ts +2 -0
  161. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/login-dialog.test.d.ts.map +1 -0
  162. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/login-dialog.test.js +13 -0
  163. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/login-dialog.test.js.map +1 -0
  164. package/packages/pi-coding-agent/dist/modes/interactive/components/login-dialog.d.ts +4 -0
  165. package/packages/pi-coding-agent/dist/modes/interactive/components/login-dialog.d.ts.map +1 -1
  166. package/packages/pi-coding-agent/dist/modes/interactive/components/login-dialog.js +24 -2
  167. package/packages/pi-coding-agent/dist/modes/interactive/components/login-dialog.js.map +1 -1
  168. package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.d.ts.map +1 -1
  169. package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.js +9 -2
  170. package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.js.map +1 -1
  171. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts +4 -0
  172. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  173. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +43 -0
  174. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
  175. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  176. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +7 -2
  177. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  178. package/packages/pi-coding-agent/dist/modes/interactive/controllers/model-controller.d.ts.map +1 -1
  179. package/packages/pi-coding-agent/dist/modes/interactive/controllers/model-controller.js +6 -1
  180. package/packages/pi-coding-agent/dist/modes/interactive/controllers/model-controller.js.map +1 -1
  181. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  182. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +4 -3
  183. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  184. package/packages/pi-coding-agent/dist/modes/interactive/slash-command-handlers.js +4 -2
  185. package/packages/pi-coding-agent/dist/modes/interactive/slash-command-handlers.js.map +1 -1
  186. package/packages/pi-coding-agent/src/core/agent-session-renderable-tools.test.ts +70 -0
  187. package/packages/pi-coding-agent/src/core/agent-session.ts +2 -1
  188. package/packages/pi-coding-agent/src/core/auth-storage.test.ts +108 -0
  189. package/packages/pi-coding-agent/src/core/auth-storage.ts +30 -0
  190. package/packages/pi-coding-agent/src/core/model-resolver-initial-model-auth.test.ts +78 -0
  191. package/packages/pi-coding-agent/src/core/model-resolver.test.ts +85 -0
  192. package/packages/pi-coding-agent/src/core/model-resolver.ts +22 -18
  193. package/packages/pi-coding-agent/src/core/retry-handler.test.ts +83 -0
  194. package/packages/pi-coding-agent/src/core/retry-handler.ts +60 -1
  195. package/packages/pi-coding-agent/src/core/sdk.test.ts +89 -0
  196. package/packages/pi-coding-agent/src/core/sdk.ts +45 -9
  197. package/packages/pi-coding-agent/src/index.ts +1 -0
  198. package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/login-dialog.test.ts +24 -0
  199. package/packages/pi-coding-agent/src/modes/interactive/components/login-dialog.ts +30 -2
  200. package/packages/pi-coding-agent/src/modes/interactive/components/model-selector.ts +15 -6
  201. package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +47 -0
  202. package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +7 -2
  203. package/packages/pi-coding-agent/src/modes/interactive/controllers/model-controller.ts +6 -1
  204. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +4 -3
  205. package/packages/pi-coding-agent/src/modes/interactive/slash-command-handlers.ts +4 -2
  206. package/src/resources/agents/debugger.md +58 -0
  207. package/src/resources/agents/doc-writer.md +43 -0
  208. package/src/resources/agents/git-ops.md +56 -0
  209. package/src/resources/agents/javascript-pro.md +46 -271
  210. package/src/resources/agents/planner.md +55 -0
  211. package/src/resources/agents/refactorer.md +47 -0
  212. package/src/resources/agents/reviewer.md +48 -0
  213. package/src/resources/agents/security.md +59 -0
  214. package/src/resources/agents/tester.md +50 -0
  215. package/src/resources/agents/typescript-pro.md +41 -235
  216. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +122 -8
  217. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +189 -6
  218. package/src/resources/extensions/gsd/auto/infra-errors.ts +38 -0
  219. package/src/resources/extensions/gsd/auto/loop-deps.ts +2 -0
  220. package/src/resources/extensions/gsd/auto/loop.ts +45 -1
  221. package/src/resources/extensions/gsd/auto/phases.ts +6 -0
  222. package/src/resources/extensions/gsd/auto/session.ts +11 -0
  223. package/src/resources/extensions/gsd/auto-dashboard.ts +29 -18
  224. package/src/resources/extensions/gsd/auto-model-selection.ts +9 -1
  225. package/src/resources/extensions/gsd/auto-prompts.ts +111 -33
  226. package/src/resources/extensions/gsd/auto-start.ts +40 -6
  227. package/src/resources/extensions/gsd/auto-worktree.ts +1 -1
  228. package/src/resources/extensions/gsd/auto.ts +72 -0
  229. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +3 -3
  230. package/src/resources/extensions/gsd/bootstrap/register-shortcuts.ts +79 -60
  231. package/src/resources/extensions/gsd/commands/handlers/auto.ts +10 -36
  232. package/src/resources/extensions/gsd/commands/handlers/core.ts +58 -11
  233. package/src/resources/extensions/gsd/commands/handlers/notifications-handler.ts +17 -7
  234. package/src/resources/extensions/gsd/commands/handlers/workflow.ts +4 -10
  235. package/src/resources/extensions/gsd/dashboard-overlay.ts +10 -3
  236. package/src/resources/extensions/gsd/doctor-providers.ts +24 -0
  237. package/src/resources/extensions/gsd/error-classifier.ts +4 -1
  238. package/src/resources/extensions/gsd/forensics.ts +23 -7
  239. package/src/resources/extensions/gsd/gate-registry.ts +251 -0
  240. package/src/resources/extensions/gsd/gsd-db.ts +51 -0
  241. package/src/resources/extensions/gsd/guided-flow.ts +5 -10
  242. package/src/resources/extensions/gsd/interrupted-session.ts +1 -0
  243. package/src/resources/extensions/gsd/metrics.ts +12 -1
  244. package/src/resources/extensions/gsd/milestone-actions.ts +10 -3
  245. package/src/resources/extensions/gsd/milestone-validation-gates.ts +11 -13
  246. package/src/resources/extensions/gsd/notification-overlay.ts +47 -14
  247. package/src/resources/extensions/gsd/notification-store.ts +35 -4
  248. package/src/resources/extensions/gsd/notification-widget.ts +5 -14
  249. package/src/resources/extensions/gsd/parallel-monitor-overlay.ts +10 -3
  250. package/src/resources/extensions/gsd/prompt-validation.ts +157 -0
  251. package/src/resources/extensions/gsd/prompts/complete-slice.md +3 -1
  252. package/src/resources/extensions/gsd/prompts/execute-task.md +2 -0
  253. package/src/resources/extensions/gsd/prompts/validate-milestone.md +2 -0
  254. package/src/resources/extensions/gsd/session-model-override.ts +36 -0
  255. package/src/resources/extensions/gsd/shortcut-defs.ts +56 -0
  256. package/src/resources/extensions/gsd/state.ts +13 -2
  257. package/src/resources/extensions/gsd/tests/auto-start-model-capture.test.ts +25 -9
  258. package/src/resources/extensions/gsd/tests/complete-slice-gate-closure.test.ts +167 -0
  259. package/src/resources/extensions/gsd/tests/doctor-providers.test.ts +36 -0
  260. package/src/resources/extensions/gsd/tests/forensics-stuck-loops.test.ts +62 -0
  261. package/src/resources/extensions/gsd/tests/format-shortcut.test.ts +31 -0
  262. package/src/resources/extensions/gsd/tests/gate-dispatch.test.ts +27 -0
  263. package/src/resources/extensions/gsd/tests/gate-registry.test.ts +140 -0
  264. package/src/resources/extensions/gsd/tests/infra-errors-cooldown.test.ts +180 -0
  265. package/src/resources/extensions/gsd/tests/integration/auto-worktree-milestone-merge.test.ts +66 -1
  266. package/src/resources/extensions/gsd/tests/model-isolation.test.ts +36 -51
  267. package/src/resources/extensions/gsd/tests/notification-store.test.ts +18 -0
  268. package/src/resources/extensions/gsd/tests/notification-widget.test.ts +3 -2
  269. package/src/resources/extensions/gsd/tests/notifications-handler.test.ts +90 -0
  270. package/src/resources/extensions/gsd/tests/parallel-monitor-overlay.test.ts +1 -0
  271. package/src/resources/extensions/gsd/tests/park-db-sync.test.ts +18 -0
  272. package/src/resources/extensions/gsd/tests/prompt-system-gate-coverage.test.ts +208 -0
  273. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +9 -0
  274. package/src/resources/extensions/gsd/tests/register-shortcuts.test.ts +63 -5
  275. package/src/resources/extensions/gsd/tests/session-model-override.test.ts +35 -0
  276. package/src/resources/extensions/gsd/tests/start-auto-detached.test.ts +90 -0
  277. package/src/resources/extensions/gsd/tools/complete-slice.ts +63 -0
  278. package/src/resources/extensions/gsd/tools/complete-task.ts +63 -0
  279. package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +4 -1
  280. package/src/resources/extensions/gsd/types.ts +26 -0
  281. package/src/resources/extensions/ollama/index.ts +13 -3
  282. package/src/resources/extensions/ollama/ollama-status-indicator.test.ts +28 -0
  283. package/src/resources/extensions/shared/gsd-phase-state.ts +42 -0
  284. package/src/resources/extensions/shared/tests/gsd-phase-state.test.ts +48 -0
  285. package/src/resources/extensions/subagent/agents.ts +10 -0
  286. package/src/resources/extensions/subagent/index.ts +18 -0
  287. package/src/resources/extensions/subagent/tests/agents-conflicts.test.ts +33 -0
  288. /package/dist/web/standalone/.next/static/{dYVdRaunb2ZSEA8fjkT-V → hnGGkVMxIGGpxSJkf5jIV}/_buildManifest.js +0 -0
  289. /package/dist/web/standalone/.next/static/{dYVdRaunb2ZSEA8fjkT-V → hnGGkVMxIGGpxSJkf5jIV}/_ssgManifest.js +0 -0
@@ -1,79 +1,98 @@
1
1
  import { existsSync } from "node:fs";
2
2
  import { join } from "node:path";
3
3
 
4
- import type { ExtensionAPI } from "@gsd/pi-coding-agent";
4
+ import type { ExtensionAPI, ExtensionContext } from "@gsd/pi-coding-agent";
5
5
  import { Key } from "@gsd/pi-tui";
6
6
 
7
7
  import { GSDDashboardOverlay } from "../dashboard-overlay.js";
8
8
  import { GSDNotificationOverlay } from "../notification-overlay.js";
9
9
  import { ParallelMonitorOverlay } from "../parallel-monitor-overlay.js";
10
+ import { GSD_SHORTCUTS } from "../shortcut-defs.js";
10
11
  import { projectRoot } from "../commands/context.js";
11
12
  import { shortcutDesc } from "../../shared/mod.js";
12
13
 
13
14
  export function registerShortcuts(pi: ExtensionAPI): void {
14
- pi.registerShortcut(Key.ctrlAlt("g"), {
15
- description: shortcutDesc("Open GSD dashboard", "/gsd status"),
16
- handler: async (ctx) => {
17
- const basePath = projectRoot();
18
- if (!existsSync(join(basePath, ".gsd"))) {
19
- ctx.ui.notify("No .gsd/ directory found. Run /gsd to start.", "info");
20
- return;
21
- }
22
- await ctx.ui.custom<boolean>(
23
- (tui, theme, _kb, done) => new GSDDashboardOverlay(tui, theme, () => done(true)),
24
- {
25
- overlay: true,
26
- overlayOptions: {
27
- width: "90%",
28
- minWidth: 80,
29
- maxHeight: "92%",
30
- anchor: "center",
31
- },
15
+ const overlayOptions = {
16
+ width: "90%",
17
+ minWidth: 80,
18
+ maxHeight: "92%",
19
+ anchor: "center",
20
+ } as const;
21
+
22
+ const openDashboardOverlay = async (ctx: ExtensionContext) => {
23
+ const basePath = projectRoot();
24
+ if (!existsSync(join(basePath, ".gsd"))) {
25
+ ctx.ui.notify("No .gsd/ directory found. Run /gsd to start.", "info");
26
+ return;
27
+ }
28
+ await ctx.ui.custom<boolean>(
29
+ (tui, theme, _kb, done) => new GSDDashboardOverlay(tui, theme, () => done(true)),
30
+ {
31
+ overlay: true,
32
+ overlayOptions,
33
+ },
34
+ );
35
+ };
36
+
37
+ const openNotificationsOverlay = async (ctx: ExtensionContext) => {
38
+ await ctx.ui.custom<boolean>(
39
+ (tui, theme, _kb, done) => new GSDNotificationOverlay(tui, theme, () => done(true)),
40
+ {
41
+ overlay: true,
42
+ overlayOptions: {
43
+ width: "80%",
44
+ minWidth: 60,
45
+ maxHeight: "88%",
46
+ anchor: "center",
47
+ backdrop: true,
32
48
  },
33
- );
34
- },
49
+ },
50
+ );
51
+ };
52
+
53
+ const openParallelOverlay = async (ctx: ExtensionContext) => {
54
+ const basePath = projectRoot();
55
+ const parallelDir = join(basePath, ".gsd", "parallel");
56
+ if (!existsSync(parallelDir)) {
57
+ ctx.ui.notify("No parallel workers found. Run /gsd parallel start first.", "info");
58
+ return;
59
+ }
60
+ await ctx.ui.custom<boolean>(
61
+ (tui, theme, _kb, done) => new ParallelMonitorOverlay(tui, theme, () => done(true), basePath),
62
+ {
63
+ overlay: true,
64
+ overlayOptions,
65
+ },
66
+ );
67
+ };
68
+
69
+ pi.registerShortcut(Key.ctrlAlt(GSD_SHORTCUTS.dashboard.key), {
70
+ description: shortcutDesc(GSD_SHORTCUTS.dashboard.action, GSD_SHORTCUTS.dashboard.command),
71
+ handler: openDashboardOverlay,
35
72
  });
36
73
 
37
- pi.registerShortcut(Key.ctrlAlt("n"), {
38
- description: shortcutDesc("Open notification history", "/gsd notifications"),
39
- handler: async (ctx) => {
40
- await ctx.ui.custom<boolean>(
41
- (tui, theme, _kb, done) => new GSDNotificationOverlay(tui, theme, () => done(true)),
42
- {
43
- overlay: true,
44
- overlayOptions: {
45
- width: "80%",
46
- minWidth: 60,
47
- maxHeight: "88%",
48
- anchor: "center",
49
- backdrop: true,
50
- },
51
- },
52
- );
53
- },
74
+ // Fallback for terminals where Ctrl+Alt letter chords are not forwarded reliably.
75
+ pi.registerShortcut(Key.ctrlShift(GSD_SHORTCUTS.dashboard.key), {
76
+ description: shortcutDesc(`${GSD_SHORTCUTS.dashboard.action} (fallback)`, GSD_SHORTCUTS.dashboard.command),
77
+ handler: openDashboardOverlay,
54
78
  });
55
79
 
56
- pi.registerShortcut(Key.ctrlAlt("p"), {
57
- description: shortcutDesc("Open parallel worker monitor", "/gsd parallel watch"),
58
- handler: async (ctx) => {
59
- const basePath = projectRoot();
60
- const parallelDir = join(basePath, ".gsd", "parallel");
61
- if (!existsSync(parallelDir)) {
62
- ctx.ui.notify("No parallel workers found. Run /gsd parallel start first.", "info");
63
- return;
64
- }
65
- await ctx.ui.custom<boolean>(
66
- (tui, theme, _kb, done) => new ParallelMonitorOverlay(tui, theme, () => done(true)),
67
- {
68
- overlay: true,
69
- overlayOptions: {
70
- width: "90%",
71
- minWidth: 80,
72
- maxHeight: "92%",
73
- anchor: "center",
74
- },
75
- },
76
- );
77
- },
80
+ pi.registerShortcut(Key.ctrlAlt(GSD_SHORTCUTS.notifications.key), {
81
+ description: shortcutDesc(GSD_SHORTCUTS.notifications.action, GSD_SHORTCUTS.notifications.command),
82
+ handler: openNotificationsOverlay,
78
83
  });
84
+
85
+ // Fallback for terminals where Ctrl+Alt letter chords are not forwarded reliably.
86
+ pi.registerShortcut(Key.ctrlShift(GSD_SHORTCUTS.notifications.key), {
87
+ description: shortcutDesc(`${GSD_SHORTCUTS.notifications.action} (fallback)`, GSD_SHORTCUTS.notifications.command),
88
+ handler: openNotificationsOverlay,
89
+ });
90
+
91
+ pi.registerShortcut(Key.ctrlAlt(GSD_SHORTCUTS.parallel.key), {
92
+ description: shortcutDesc(GSD_SHORTCUTS.parallel.action, GSD_SHORTCUTS.parallel.command),
93
+ handler: openParallelOverlay,
94
+ });
95
+
96
+ // No Ctrl+Shift+P fallback — conflicts with cycleModelBackward (shift+ctrl+p).
97
+ // Use Ctrl+Alt+P or /gsd parallel watch instead.
79
98
  }
@@ -4,7 +4,7 @@ import { existsSync, readFileSync } from "node:fs";
4
4
  import { resolve } from "node:path";
5
5
 
6
6
  import { enableDebug } from "../../debug-logger.js";
7
- import { getAutoDashboardData, isAutoActive, isAutoPaused, pauseAuto, startAuto, stopAuto, stopAutoRemote } from "../../auto.js";
7
+ import { getAutoDashboardData, isAutoActive, isAutoPaused, pauseAuto, startAutoDetached, stopAuto, stopAutoRemote } from "../../auto.js";
8
8
  import { handleRate } from "../../commands-rate.js";
9
9
  import { guardRemoteSession, projectRoot } from "../context.js";
10
10
  import { findMilestoneIds } from "../../milestone-id-utils.js";
@@ -42,26 +42,6 @@ export function parseMilestoneTarget(input: string): { milestoneId: string | nul
42
42
  return { milestoneId: match[1], rest };
43
43
  }
44
44
 
45
- /**
46
- * Set GSD_MILESTONE_LOCK to target a specific milestone, then run `fn`.
47
- * Clears the env var when `fn` resolves or rejects, so the lock does not
48
- * leak into subsequent commands in the same process.
49
- */
50
- async function withMilestoneLock(milestoneId: string, fn: () => Promise<void>): Promise<void> {
51
- const previous = process.env.GSD_MILESTONE_LOCK;
52
- process.env.GSD_MILESTONE_LOCK = milestoneId;
53
- try {
54
- await fn();
55
- } finally {
56
- // Restore previous value (undefined → delete, else restore).
57
- if (previous === undefined) {
58
- delete process.env.GSD_MILESTONE_LOCK;
59
- } else {
60
- process.env.GSD_MILESTONE_LOCK = previous;
61
- }
62
- }
63
- }
64
-
65
45
  export async function handleAutoCommand(trimmed: string, ctx: ExtensionCommandContext, pi: ExtensionAPI): Promise<boolean> {
66
46
  if (trimmed === "next" || trimmed.startsWith("next ")) {
67
47
  if (trimmed.includes("--dry-run")) {
@@ -84,13 +64,10 @@ export async function handleAutoCommand(trimmed: string, ctx: ExtensionCommandCo
84
64
  }
85
65
  }
86
66
 
87
- if (milestoneId) {
88
- await withMilestoneLock(milestoneId, () =>
89
- startAuto(ctx, pi, projectRoot(), verboseMode, { step: true }),
90
- );
91
- } else {
92
- await startAuto(ctx, pi, projectRoot(), verboseMode, { step: true });
93
- }
67
+ startAutoDetached(ctx, pi, projectRoot(), verboseMode, {
68
+ step: true,
69
+ milestoneLock: milestoneId,
70
+ });
94
71
  return true;
95
72
  }
96
73
 
@@ -128,13 +105,11 @@ export async function handleAutoCommand(trimmed: string, ctx: ExtensionCommandCo
128
105
  const { showHeadlessMilestoneCreation } = await import("../../guided-flow.js");
129
106
  await showHeadlessMilestoneCreation(ctx, pi, projectRoot(), seedContent);
130
107
  } else if (milestoneId) {
131
- // Target a specific milestone — use GSD_MILESTONE_LOCK so state
132
- // derivation only sees this milestone (#2521).
133
- await withMilestoneLock(milestoneId, () =>
134
- startAuto(ctx, pi, projectRoot(), verboseMode),
135
- );
108
+ startAutoDetached(ctx, pi, projectRoot(), verboseMode, {
109
+ milestoneLock: milestoneId,
110
+ });
136
111
  } else {
137
- await startAuto(ctx, pi, projectRoot(), verboseMode);
112
+ startAutoDetached(ctx, pi, projectRoot(), verboseMode);
138
113
  }
139
114
  return true;
140
115
  }
@@ -175,10 +150,9 @@ export async function handleAutoCommand(trimmed: string, ctx: ExtensionCommandCo
175
150
 
176
151
  if (trimmed === "") {
177
152
  if (!(await guardRemoteSession(ctx, pi))) return true;
178
- await startAuto(ctx, pi, projectRoot(), false, { step: true });
153
+ startAutoDetached(ctx, pi, projectRoot(), false, { step: true });
179
154
  return true;
180
155
  }
181
156
 
182
157
  return false;
183
158
  }
184
-
@@ -8,11 +8,45 @@ import { ensurePreferencesFile, handlePrefs, handlePrefsMode, handlePrefsWizard
8
8
  import { runEnvironmentChecks } from "../../doctor-environment.js";
9
9
  import { deriveState } from "../../state.js";
10
10
  import { handleCmux } from "../../commands-cmux.js";
11
+ import { setSessionModelOverride } from "../../session-model-override.js";
11
12
  import { projectRoot } from "../context.js";
12
- import { formatShortcut } from "../../files.js";
13
+ import { formattedShortcutPair } from "../../shortcut-defs.js";
13
14
 
14
- export function showHelp(ctx: ExtensionCommandContext): void {
15
- const lines = [
15
+ export function showHelp(ctx: ExtensionCommandContext, args = ""): void {
16
+ const summaryLines = [
17
+ "GSD — Get Shit Done\n",
18
+ "QUICK START",
19
+ " /gsd start <tpl> Start a workflow template",
20
+ " /gsd Run next unit (same as /gsd next)",
21
+ " /gsd auto Run all queued units continuously",
22
+ " /gsd pause Pause auto-mode",
23
+ " /gsd stop Stop auto-mode gracefully",
24
+ "",
25
+ "VISIBILITY",
26
+ ` /gsd status Dashboard (${formattedShortcutPair("dashboard")})`,
27
+ ` /gsd parallel watch Parallel monitor (${formattedShortcutPair("parallel")})`,
28
+ ` /gsd notifications Notification history (${formattedShortcutPair("notifications")})`,
29
+ " /gsd visualize Interactive 10-tab TUI",
30
+ " /gsd queue Show queued/dispatched units",
31
+ "",
32
+ "COURSE CORRECTION",
33
+ " /gsd steer <desc> Apply user override to active work",
34
+ " /gsd capture <text> Quick-capture a thought to CAPTURES.md",
35
+ " /gsd triage Classify and route pending captures",
36
+ " /gsd undo Revert last completed unit [--force]",
37
+ " /gsd rethink Conversational project reorganization",
38
+ "",
39
+ "SETUP",
40
+ " /gsd init Project init wizard",
41
+ " /gsd setup Global setup status [llm|search|remote|keys|prefs]",
42
+ " /gsd model Switch active session model",
43
+ " /gsd prefs Manage preferences",
44
+ " /gsd doctor Diagnose and repair .gsd/ state",
45
+ "",
46
+ "Use /gsd help full for the complete command reference.",
47
+ ];
48
+
49
+ const fullLines = [
16
50
  "GSD — Get Shit Done\n",
17
51
  "WORKFLOW",
18
52
  " /gsd start <tpl> Start a workflow template (bugfix, spike, feature, hotfix, etc.)",
@@ -26,12 +60,13 @@ export function showHelp(ctx: ExtensionCommandContext): void {
26
60
  " /gsd new-milestone Create milestone from headless context (used by gsd headless)",
27
61
  "",
28
62
  "VISIBILITY",
29
- ` /gsd status Show progress dashboard (${formatShortcut("Ctrl+Alt+G")})`,
63
+ ` /gsd status Show progress dashboard (${formattedShortcutPair("dashboard")})`,
64
+ ` /gsd parallel watch Open parallel worker monitor (${formattedShortcutPair("parallel")})`,
30
65
  " /gsd visualize Interactive 10-tab TUI (progress, timeline, deps, metrics, health, agent, changes, knowledge, captures, export)",
31
66
  " /gsd queue Show queued/dispatched units and execution order",
32
67
  " /gsd history View execution history [--cost] [--phase] [--model] [N]",
33
68
  " /gsd changelog Show categorized release notes [version]",
34
- ` /gsd notifications View persistent notification history [clear|tail|filter] (${formatShortcut("Ctrl+Alt+N")})`,
69
+ ` /gsd notifications View persistent notification history [clear|tail|filter] (${formattedShortcutPair("notifications")})`,
35
70
  "",
36
71
  "COURSE CORRECTION",
37
72
  " /gsd steer <desc> Apply user override to active work",
@@ -71,7 +106,8 @@ export function showHelp(ctx: ExtensionCommandContext): void {
71
106
  " /gsd inspect Show SQLite DB diagnostics (schema, row counts, recent entries)",
72
107
  " /gsd update Update GSD to the latest version via npm",
73
108
  ];
74
- ctx.ui.notify(lines.join("\n"), "info");
109
+ const full = ["full", "--full", "all"].includes(args.trim().toLowerCase());
110
+ ctx.ui.notify((full ? fullLines : summaryLines).join("\n"), "info");
75
111
  }
76
112
 
77
113
  export async function handleStatus(ctx: ExtensionCommandContext): Promise<void> {
@@ -92,9 +128,9 @@ export async function handleStatus(ctx: ExtensionCommandContext): Promise<void>
92
128
  {
93
129
  overlay: true,
94
130
  overlayOptions: {
95
- width: "70%",
96
- minWidth: 60,
97
- maxHeight: "90%",
131
+ width: "90%",
132
+ minWidth: 80,
133
+ maxHeight: "92%",
98
134
  anchor: "center",
99
135
  },
100
136
  },
@@ -301,6 +337,17 @@ async function handleModel(trimmedArgs: string, ctx: ExtensionCommandContext, pi
301
337
  return;
302
338
  }
303
339
 
340
+ // /gsd model is an explicit per-session pin for GSD dispatches.
341
+ // This is captured at auto bootstrap so it survives internal session
342
+ // switches during /gsd auto and /gsd next runs.
343
+ const sessionId = ctx.sessionManager?.getSessionId?.();
344
+ if (sessionId) {
345
+ setSessionModelOverride(sessionId, {
346
+ provider: targetModel.provider,
347
+ id: targetModel.id,
348
+ });
349
+ }
350
+
304
351
  ctx.ui.notify(`Model: ${targetModel.provider}/${targetModel.id}`, "info");
305
352
  }
306
353
 
@@ -309,8 +356,8 @@ export async function handleCoreCommand(
309
356
  ctx: ExtensionCommandContext,
310
357
  pi?: ExtensionAPI,
311
358
  ): Promise<boolean> {
312
- if (trimmed === "help" || trimmed === "h" || trimmed === "?") {
313
- showHelp(ctx);
359
+ if (trimmed === "help" || trimmed === "h" || trimmed === "?" || trimmed.startsWith("help ")) {
360
+ showHelp(ctx, trimmed.startsWith("help ") ? trimmed.slice(5).trim() : "");
314
361
  return true;
315
362
  }
316
363
  if (trimmed === "status") {
@@ -13,6 +13,8 @@ import {
13
13
  } from "../../notification-store.js";
14
14
  import { GSDNotificationOverlay } from "../../notification-overlay.js";
15
15
 
16
+ const MAX_INLINE_ENTRIES = 40;
17
+
16
18
  function severityIcon(severity: NotifySeverity): string {
17
19
  switch (severity) {
18
20
  case "error": return "✗";
@@ -54,8 +56,9 @@ export async function handleNotificationsCommand(
54
56
  if (args === "tail" || args.startsWith("tail ")) {
55
57
  const countStr = args.replace(/^tail\s*/, "").trim();
56
58
  const count = countStr ? parseInt(countStr, 10) : 20;
57
- const n = isNaN(count) || count < 1 ? 20 : Math.min(count, 100);
58
- const entries = readNotifications().slice(0, n);
59
+ const all = readNotifications();
60
+ const n = isNaN(count) || count < 1 ? 20 : Math.min(count, MAX_INLINE_ENTRIES);
61
+ const entries = all.slice(0, n);
59
62
 
60
63
  if (entries.length === 0) {
61
64
  ctx.ui.notify("No notifications.", "info");
@@ -65,7 +68,10 @@ export async function handleNotificationsCommand(
65
68
  const lines = entries.map((e) =>
66
69
  `${severityIcon(e.severity)} [${formatTimestamp(e.ts)}] ${e.message}`,
67
70
  );
68
- ctx.ui.notify(`Last ${entries.length} notification(s):\n${lines.join("\n")}`, "info");
71
+ const suffix = all.length > entries.length
72
+ ? `\n... and ${all.length - entries.length} more (open /gsd notifications to browse all)`
73
+ : "";
74
+ ctx.ui.notify(`Last ${entries.length} notification(s):\n${lines.join("\n")}${suffix}`, "info");
69
75
  return true;
70
76
  }
71
77
 
@@ -86,7 +92,9 @@ export async function handleNotificationsCommand(
86
92
  const lines = entries.slice(0, 20).map((e) =>
87
93
  `${severityIcon(e.severity)} [${formatTimestamp(e.ts)}] ${e.message}`,
88
94
  );
89
- const suffix = entries.length > 20 ? `\n... and ${entries.length - 20} more` : "";
95
+ const suffix = entries.length > 20
96
+ ? `\n... and ${entries.length - 20} more (open /gsd notifications to browse all)`
97
+ : "";
90
98
  ctx.ui.notify(`${severity} notifications (${entries.length}):\n${lines.join("\n")}${suffix}`, "info");
91
99
  return true;
92
100
  }
@@ -96,8 +104,8 @@ export async function handleNotificationsCommand(
96
104
  // Try overlay first (TUI mode)
97
105
  if (ctx.hasUI) {
98
106
  try {
99
- await ctx.ui.custom<void>(
100
- (tui, theme, _kb, done) => new GSDNotificationOverlay(tui, theme, () => done()),
107
+ const result = await ctx.ui.custom<boolean>(
108
+ (tui, theme, _kb, done) => new GSDNotificationOverlay(tui, theme, () => done(true)),
101
109
  {
102
110
  overlay: true,
103
111
  overlayOptions: {
@@ -109,7 +117,9 @@ export async function handleNotificationsCommand(
109
117
  },
110
118
  },
111
119
  );
112
- return true;
120
+ if (result !== undefined) {
121
+ return true;
122
+ }
113
123
  } catch {
114
124
  // Fall through to text output if overlay fails
115
125
  }
@@ -18,7 +18,7 @@ import { createRun, listRuns } from "../../run-manager.js";
18
18
  import {
19
19
  setActiveEngineId,
20
20
  setActiveRunDir,
21
- startAuto,
21
+ startAutoDetached,
22
22
  pauseAuto,
23
23
  isAutoActive,
24
24
  getActiveEngineId,
@@ -77,7 +77,7 @@ async function handleCustomWorkflow(
77
77
  setActiveEngineId("custom");
78
78
  setActiveRunDir(runDir);
79
79
  ctx.ui.notify(`Created workflow run: ${defName}\nRun dir: ${runDir}`, "info");
80
- await startAuto(ctx, pi, base, false);
80
+ startAutoDetached(ctx, pi, base, false);
81
81
  } catch (err) {
82
82
  // Clean up engine state so a failed workflow run doesn't pollute the next /gsd auto
83
83
  setActiveEngineId(null);
@@ -157,13 +157,8 @@ async function handleCustomWorkflow(
157
157
  ctx.ui.notify("No custom workflow to resume. Use /gsd auto for dev workflow.", "warning");
158
158
  return true;
159
159
  }
160
- try {
161
- await startAuto(ctx, pi, projectRoot(), false);
162
- ctx.ui.notify("Custom workflow resumed.", "info");
163
- } catch (err) {
164
- const msg = err instanceof Error ? err.message : String(err);
165
- ctx.ui.notify(`Failed to resume workflow: ${msg}`, "error");
166
- }
160
+ startAutoDetached(ctx, pi, projectRoot(), false);
161
+ ctx.ui.notify("Custom workflow resumed.", "info");
167
162
  return true;
168
163
  }
169
164
 
@@ -278,4 +273,3 @@ export function getNextMilestoneId(basePath: string): string {
278
273
  const uniqueIds = !!loadEffectiveGSDPreferences()?.preferences?.unique_milestone_ids;
279
274
  return nextMilestoneId(milestoneIds, uniqueIds);
280
275
  }
281
-
@@ -3,7 +3,8 @@
3
3
  *
4
4
  * Full-screen overlay showing auto-mode progress: milestone/slice/task
5
5
  * breakdown, current unit, completed units, timing, and activity log.
6
- * Toggled with Ctrl+Alt+G (⌃⌥G on macOS) or opened from /gsd status.
6
+ * Toggled with Ctrl+Alt+G (⌃⌥G on macOS), Ctrl+Shift+G fallback,
7
+ * or opened from /gsd status.
7
8
  */
8
9
 
9
10
  import type { Theme } from "@gsd/pi-coding-agent";
@@ -26,6 +27,7 @@ import { formatDuration, padRight, joinColumns, centerLine, fitColumns, STATUS_G
26
27
  import { estimateTimeRemaining } from "./auto-dashboard.js";
27
28
  import { computeProgressScore, formatProgressLine } from "./progress-score.js";
28
29
  import { runEnvironmentChecks, type EnvironmentCheckResult } from "./doctor-environment.js";
30
+ import { formattedShortcutPair } from "./shortcut-defs.js";
29
31
 
30
32
  function unitLabel(type: string): string {
31
33
  switch (type) {
@@ -203,7 +205,12 @@ export class GSDDashboardOverlay {
203
205
  }
204
206
 
205
207
  handleInput(data: string): void {
206
- if (matchesKey(data, Key.escape) || matchesKey(data, Key.ctrl("c")) || matchesKey(data, Key.ctrlAlt("g"))) {
208
+ if (
209
+ matchesKey(data, Key.escape) ||
210
+ matchesKey(data, Key.ctrl("c")) ||
211
+ matchesKey(data, Key.ctrlAlt("g")) ||
212
+ matchesKey(data, Key.ctrlShift("g"))
213
+ ) {
207
214
  this.dispose();
208
215
  this.onClose();
209
216
  return;
@@ -587,7 +594,7 @@ export class GSDDashboardOverlay {
587
594
 
588
595
  lines.push(blank());
589
596
  lines.push(hr());
590
- lines.push(centered(th.fg("dim", "↑↓ scroll · g/G top/end · esc close")));
597
+ lines.push(centered(th.fg("dim", `↑↓ scroll · g/G top/end · Esc/${formattedShortcutPair("dashboard")} close`)));
591
598
 
592
599
  return lines;
593
600
  }
@@ -185,11 +185,35 @@ const PROVIDER_ROUTES: Record<string, string[]> = {
185
185
  google: ["google-gemini-cli"],
186
186
  };
187
187
 
188
+ /**
189
+ * Providers that use external CLI authentication (not API keys).
190
+ * These are always considered "ok" — the host CLI handles auth.
191
+ */
192
+ const CLI_AUTH_PROVIDERS = new Set([
193
+ "claude-code",
194
+ "openai-codex",
195
+ "google-gemini-cli",
196
+ "google-antigravity",
197
+ ]);
198
+
188
199
  function checkLlmProviders(): ProviderCheckResult[] {
189
200
  const required = collectConfiguredModelProviders();
190
201
  const results: ProviderCheckResult[] = [];
191
202
 
192
203
  for (const providerId of required) {
204
+ // CLI-authenticated providers don't need API keys — skip key check
205
+ if (CLI_AUTH_PROVIDERS.has(providerId)) {
206
+ const info = PROVIDER_REGISTRY.find(p => p.id === providerId);
207
+ results.push({
208
+ name: providerId,
209
+ label: info?.label ?? providerId,
210
+ category: "llm",
211
+ status: "ok",
212
+ message: `${info?.label ?? providerId} — CLI auth (no key needed)`,
213
+ required: true,
214
+ });
215
+ continue;
216
+ }
193
217
  const info = PROVIDER_REGISTRY.find(p => p.id === providerId);
194
218
  const label = providerId === "anthropic-vertex"
195
219
  ? "Anthropic Vertex"
@@ -44,6 +44,9 @@ export function resetRetryState(state: RetryState): void {
44
44
 
45
45
  const PERMANENT_RE = /auth|unauthorized|forbidden|invalid.*key|invalid.*api|billing|quota exceeded|account/i;
46
46
  const RATE_LIMIT_RE = /rate.?limit|too many requests|429/i;
47
+ // OpenRouter affordability-style quota errors should be treated as transient
48
+ // so core retry logic can lower maxTokens and continue in-session.
49
+ const AFFORDABILITY_RE = /requires more credits|can only afford|insufficient credits|not enough credits|fewer max_tokens/i;
47
50
  const NETWORK_RE = /network|ECONNRESET|ETIMEDOUT|ECONNREFUSED|socket hang up|fetch failed|connection.*reset|dns/i;
48
51
  const SERVER_RE = /internal server error|500|502|503|overloaded|server_error|api_error|service.?unavailable/i;
49
52
  // ECONNRESET/ECONNREFUSED are in NETWORK_RE (same-model retry first).
@@ -67,7 +70,7 @@ const RESET_DELAY_RE = /reset in (\d+)s/i;
67
70
  */
68
71
  export function classifyError(errorMsg: string, retryAfterMs?: number): ErrorClass {
69
72
  const isPermanent = PERMANENT_RE.test(errorMsg);
70
- const isRateLimit = RATE_LIMIT_RE.test(errorMsg);
73
+ const isRateLimit = RATE_LIMIT_RE.test(errorMsg) || AFFORDABILITY_RE.test(errorMsg);
71
74
 
72
75
  // 1. Permanent — but rate limit takes precedence
73
76
  if (isPermanent && !isRateLimit) {
@@ -650,19 +650,33 @@ function getDbCompletionCounts(): DbCompletionCounts | null {
650
650
  * Exported for testability.
651
651
  */
652
652
  export function detectStuckLoops(units: UnitMetrics[], anomalies: ForensicAnomaly[]): void {
653
- // First, collect unique startedAt values per type/id key
654
- const dispatchMap = new Map<string, Set<number>>();
653
+ // First, collect unique startedAt values per type/id key, bucketed by
654
+ // autoSessionKey when available so cross-session recovery does not look
655
+ // like a within-session stuck loop.
656
+ const dispatchMap = new Map<string, Map<string, Set<number>>>();
655
657
  for (const u of units) {
656
658
  const key = `${u.type}/${u.id}`;
657
- let starts = dispatchMap.get(key);
659
+ let sessionBuckets = dispatchMap.get(key);
660
+ if (!sessionBuckets) {
661
+ sessionBuckets = new Map();
662
+ dispatchMap.set(key, sessionBuckets);
663
+ }
664
+
665
+ const sessionKey = u.autoSessionKey ?? "__legacy__";
666
+ let starts = sessionBuckets.get(sessionKey);
658
667
  if (!starts) {
659
668
  starts = new Set();
660
- dispatchMap.set(key, starts);
669
+ sessionBuckets.set(sessionKey, starts);
661
670
  }
662
671
  starts.add(u.startedAt);
663
672
  }
664
- for (const [key, starts] of dispatchMap) {
665
- const count = starts.size;
673
+
674
+ for (const [key, sessionBuckets] of dispatchMap) {
675
+ const hasSessionAwareData = Array.from(sessionBuckets.keys()).some((sessionKey) => sessionKey !== "__legacy__");
676
+ const count = hasSessionAwareData
677
+ ? Math.max(...Array.from(sessionBuckets.values(), (starts) => starts.size))
678
+ : (sessionBuckets.get("__legacy__")?.size ?? 0);
679
+
666
680
  if (count > 1) {
667
681
  const [unitType, ...idParts] = key.split("/");
668
682
  anomalies.push({
@@ -671,7 +685,9 @@ export function detectStuckLoops(units: UnitMetrics[], anomalies: ForensicAnomal
671
685
  unitType,
672
686
  unitId: idParts.join("/"),
673
687
  summary: `Unit ${key} was dispatched ${count} times`,
674
- details: `Repeated dispatch suggests the unit completed but its artifacts weren't verified, or the state machine kept returning it.`,
688
+ details: hasSessionAwareData
689
+ ? `Repeated dispatch within the same auto session suggests the unit completed but its artifacts were not verified, or the state machine kept returning it. Cross-session recovery runs are ignored.`
690
+ : `Repeated dispatch suggests the unit completed but its artifacts weren't verified, or the state machine kept returning it.`,
675
691
  });
676
692
  }
677
693
  }