gsd-pi 2.70.1 → 2.71.0-dev.4c35d99

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 (533) hide show
  1. package/README.md +57 -17
  2. package/dist/cli.js +29 -3
  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 +40 -17
  7. package/dist/provider-migrations.d.ts +10 -0
  8. package/dist/provider-migrations.js +12 -0
  9. package/dist/resource-loader.js +139 -13
  10. package/dist/resources/GSD-WORKFLOW.md +1 -1
  11. package/dist/resources/agents/debugger.md +58 -0
  12. package/dist/resources/agents/doc-writer.md +43 -0
  13. package/dist/resources/agents/git-ops.md +56 -0
  14. package/dist/resources/agents/javascript-pro.md +46 -271
  15. package/dist/resources/agents/planner.md +55 -0
  16. package/dist/resources/agents/refactorer.md +47 -0
  17. package/dist/resources/agents/reviewer.md +48 -0
  18. package/dist/resources/agents/security.md +59 -0
  19. package/dist/resources/agents/tester.md +50 -0
  20. package/dist/resources/agents/typescript-pro.md +41 -235
  21. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +242 -40
  22. package/dist/resources/extensions/get-secrets-from-user.js +17 -1
  23. package/dist/resources/extensions/gsd/auto/infra-errors.js +34 -0
  24. package/dist/resources/extensions/gsd/auto/loop.js +32 -1
  25. package/dist/resources/extensions/gsd/auto/phases.js +5 -1
  26. package/dist/resources/extensions/gsd/auto/session.js +11 -0
  27. package/dist/resources/extensions/gsd/auto-dashboard.js +22 -16
  28. package/dist/resources/extensions/gsd/auto-model-selection.js +10 -2
  29. package/dist/resources/extensions/gsd/auto-prompts.js +88 -33
  30. package/dist/resources/extensions/gsd/auto-start.js +37 -18
  31. package/dist/resources/extensions/gsd/auto-tool-tracking.js +1 -1
  32. package/dist/resources/extensions/gsd/auto-worktree.js +1 -1
  33. package/dist/resources/extensions/gsd/auto.js +56 -0
  34. package/dist/resources/extensions/gsd/bootstrap/db-tools.js +3 -3
  35. package/dist/resources/extensions/gsd/bootstrap/register-hooks.js +6 -0
  36. package/dist/resources/extensions/gsd/bootstrap/register-shortcuts.js +63 -51
  37. package/dist/resources/extensions/gsd/bootstrap/system-context.js +6 -0
  38. package/dist/resources/extensions/gsd/commands/context.js +15 -6
  39. package/dist/resources/extensions/gsd/commands/dispatcher.js +12 -2
  40. package/dist/resources/extensions/gsd/commands/handlers/auto.js +10 -33
  41. package/dist/resources/extensions/gsd/commands/handlers/core.js +56 -11
  42. package/dist/resources/extensions/gsd/commands/handlers/notifications-handler.js +15 -6
  43. package/dist/resources/extensions/gsd/commands/handlers/workflow.js +4 -10
  44. package/dist/resources/extensions/gsd/custom-workflow-engine.js +16 -12
  45. package/dist/resources/extensions/gsd/dashboard-overlay.js +8 -3
  46. package/dist/resources/extensions/gsd/dispatch-guard.js +18 -1
  47. package/dist/resources/extensions/gsd/doctor-providers.js +23 -0
  48. package/dist/resources/extensions/gsd/error-classifier.js +1 -1
  49. package/dist/resources/extensions/gsd/file-lock.js +60 -0
  50. package/dist/resources/extensions/gsd/forensics.js +19 -6
  51. package/dist/resources/extensions/gsd/gate-registry.js +208 -0
  52. package/dist/resources/extensions/gsd/gsd-db.js +41 -0
  53. package/dist/resources/extensions/gsd/guided-flow.js +17 -20
  54. package/dist/resources/extensions/gsd/init-wizard.js +3 -11
  55. package/dist/resources/extensions/gsd/metrics.js +1 -0
  56. package/dist/resources/extensions/gsd/milestone-actions.js +10 -4
  57. package/dist/resources/extensions/gsd/milestone-validation-gates.js +11 -12
  58. package/dist/resources/extensions/gsd/notification-overlay.js +42 -13
  59. package/dist/resources/extensions/gsd/notification-store.js +56 -5
  60. package/dist/resources/extensions/gsd/notification-widget.js +5 -13
  61. package/dist/resources/extensions/gsd/parallel-monitor-overlay.js +8 -3
  62. package/dist/resources/extensions/gsd/pre-execution-checks.js +35 -2
  63. package/dist/resources/extensions/gsd/prompt-validation.js +126 -0
  64. package/dist/resources/extensions/gsd/prompts/complete-slice.md +5 -3
  65. package/dist/resources/extensions/gsd/prompts/discuss.md +33 -13
  66. package/dist/resources/extensions/gsd/prompts/execute-task.md +22 -19
  67. package/dist/resources/extensions/gsd/prompts/guided-discuss-milestone.md +2 -0
  68. package/dist/resources/extensions/gsd/prompts/guided-discuss-slice.md +2 -0
  69. package/dist/resources/extensions/gsd/prompts/guided-resume-task.md +1 -1
  70. package/dist/resources/extensions/gsd/prompts/queue.md +3 -2
  71. package/dist/resources/extensions/gsd/prompts/system.md +1 -0
  72. package/dist/resources/extensions/gsd/prompts/validate-milestone.md +4 -1
  73. package/dist/resources/extensions/gsd/session-model-override.js +25 -0
  74. package/dist/resources/extensions/gsd/shortcut-defs.js +40 -0
  75. package/dist/resources/extensions/gsd/state.js +241 -332
  76. package/dist/resources/extensions/gsd/tools/complete-slice.js +52 -1
  77. package/dist/resources/extensions/gsd/tools/complete-task.js +51 -1
  78. package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +38 -1
  79. package/dist/resources/extensions/gsd/workflow-events.js +25 -13
  80. package/dist/resources/extensions/gsd/workflow-mcp-auto-prep.js +56 -0
  81. package/dist/resources/extensions/gsd/workflow-mcp.js +1 -1
  82. package/dist/resources/extensions/ollama/index.js +13 -5
  83. package/dist/resources/extensions/shared/gsd-phase-state.js +35 -0
  84. package/dist/resources/extensions/subagent/agents.js +8 -0
  85. package/dist/resources/extensions/subagent/index.js +17 -0
  86. package/dist/resources/skills/create-skill/SKILL.md +2 -0
  87. package/dist/startup-model-validation.d.ts +0 -1
  88. package/dist/startup-model-validation.js +6 -2
  89. package/dist/web/standalone/.next/BUILD_ID +1 -1
  90. package/dist/web/standalone/.next/app-path-routes-manifest.json +17 -17
  91. package/dist/web/standalone/.next/build-manifest.json +4 -4
  92. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  93. package/dist/web/standalone/.next/react-loadable-manifest.json +1 -1
  94. package/dist/web/standalone/.next/required-server-files.json +3 -3
  95. package/dist/web/standalone/.next/server/app/_global-error/page.js +3 -3
  96. package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  97. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  98. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  99. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  100. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  101. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  102. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  103. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  104. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  105. package/dist/web/standalone/.next/server/app/_not-found/page.js +2 -2
  106. package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  107. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  108. package/dist/web/standalone/.next/server/app/_not-found.rsc +3 -3
  109. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +3 -3
  110. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  111. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +3 -3
  112. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  113. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  114. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  115. package/dist/web/standalone/.next/server/app/api/boot/route.js +1 -1
  116. package/dist/web/standalone/.next/server/app/api/boot/route_client-reference-manifest.js +1 -1
  117. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js +1 -1
  118. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route_client-reference-manifest.js +1 -1
  119. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js +1 -1
  120. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route_client-reference-manifest.js +1 -1
  121. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js +2 -2
  122. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route_client-reference-manifest.js +1 -1
  123. package/dist/web/standalone/.next/server/app/api/browse-directories/route.js +1 -1
  124. package/dist/web/standalone/.next/server/app/api/browse-directories/route_client-reference-manifest.js +1 -1
  125. package/dist/web/standalone/.next/server/app/api/captures/route.js +1 -1
  126. package/dist/web/standalone/.next/server/app/api/captures/route_client-reference-manifest.js +1 -1
  127. package/dist/web/standalone/.next/server/app/api/cleanup/route.js +1 -1
  128. package/dist/web/standalone/.next/server/app/api/cleanup/route_client-reference-manifest.js +1 -1
  129. package/dist/web/standalone/.next/server/app/api/dev-mode/route.js +1 -1
  130. package/dist/web/standalone/.next/server/app/api/dev-mode/route_client-reference-manifest.js +1 -1
  131. package/dist/web/standalone/.next/server/app/api/doctor/route.js +1 -1
  132. package/dist/web/standalone/.next/server/app/api/doctor/route_client-reference-manifest.js +1 -1
  133. package/dist/web/standalone/.next/server/app/api/experimental/route.js +2 -2
  134. package/dist/web/standalone/.next/server/app/api/experimental/route_client-reference-manifest.js +1 -1
  135. package/dist/web/standalone/.next/server/app/api/export-data/route.js +1 -1
  136. package/dist/web/standalone/.next/server/app/api/export-data/route_client-reference-manifest.js +1 -1
  137. package/dist/web/standalone/.next/server/app/api/files/route.js +1 -1
  138. package/dist/web/standalone/.next/server/app/api/files/route_client-reference-manifest.js +1 -1
  139. package/dist/web/standalone/.next/server/app/api/forensics/route.js +1 -1
  140. package/dist/web/standalone/.next/server/app/api/forensics/route_client-reference-manifest.js +1 -1
  141. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  142. package/dist/web/standalone/.next/server/app/api/git/route_client-reference-manifest.js +1 -1
  143. package/dist/web/standalone/.next/server/app/api/history/route.js +1 -1
  144. package/dist/web/standalone/.next/server/app/api/history/route_client-reference-manifest.js +1 -1
  145. package/dist/web/standalone/.next/server/app/api/hooks/route.js +1 -1
  146. package/dist/web/standalone/.next/server/app/api/hooks/route_client-reference-manifest.js +1 -1
  147. package/dist/web/standalone/.next/server/app/api/inspect/route.js +1 -1
  148. package/dist/web/standalone/.next/server/app/api/inspect/route_client-reference-manifest.js +1 -1
  149. package/dist/web/standalone/.next/server/app/api/knowledge/route.js +1 -1
  150. package/dist/web/standalone/.next/server/app/api/knowledge/route_client-reference-manifest.js +1 -1
  151. package/dist/web/standalone/.next/server/app/api/live-state/route.js +1 -1
  152. package/dist/web/standalone/.next/server/app/api/live-state/route_client-reference-manifest.js +1 -1
  153. package/dist/web/standalone/.next/server/app/api/notifications/route.js +2 -2
  154. package/dist/web/standalone/.next/server/app/api/notifications/route_client-reference-manifest.js +1 -1
  155. package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
  156. package/dist/web/standalone/.next/server/app/api/onboarding/route_client-reference-manifest.js +1 -1
  157. package/dist/web/standalone/.next/server/app/api/preferences/route.js +1 -1
  158. package/dist/web/standalone/.next/server/app/api/preferences/route_client-reference-manifest.js +1 -1
  159. package/dist/web/standalone/.next/server/app/api/projects/route.js +1 -1
  160. package/dist/web/standalone/.next/server/app/api/projects/route_client-reference-manifest.js +1 -1
  161. package/dist/web/standalone/.next/server/app/api/recovery/route.js +1 -1
  162. package/dist/web/standalone/.next/server/app/api/recovery/route_client-reference-manifest.js +1 -1
  163. package/dist/web/standalone/.next/server/app/api/remote-questions/route.js +2 -2
  164. package/dist/web/standalone/.next/server/app/api/remote-questions/route_client-reference-manifest.js +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_client-reference-manifest.js +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_client-reference-manifest.js +1 -1
  169. package/dist/web/standalone/.next/server/app/api/session/events/route.js +2 -2
  170. package/dist/web/standalone/.next/server/app/api/session/events/route_client-reference-manifest.js +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_client-reference-manifest.js +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_client-reference-manifest.js +1 -1
  175. package/dist/web/standalone/.next/server/app/api/shutdown/route.js +1 -1
  176. package/dist/web/standalone/.next/server/app/api/shutdown/route_client-reference-manifest.js +1 -1
  177. package/dist/web/standalone/.next/server/app/api/skill-health/route.js +1 -1
  178. package/dist/web/standalone/.next/server/app/api/skill-health/route_client-reference-manifest.js +1 -1
  179. package/dist/web/standalone/.next/server/app/api/steer/route.js +1 -1
  180. package/dist/web/standalone/.next/server/app/api/steer/route_client-reference-manifest.js +1 -1
  181. package/dist/web/standalone/.next/server/app/api/switch-root/route.js +1 -1
  182. package/dist/web/standalone/.next/server/app/api/switch-root/route_client-reference-manifest.js +1 -1
  183. package/dist/web/standalone/.next/server/app/api/terminal/input/route.js +1 -1
  184. package/dist/web/standalone/.next/server/app/api/terminal/input/route_client-reference-manifest.js +1 -1
  185. package/dist/web/standalone/.next/server/app/api/terminal/resize/route.js +2 -2
  186. package/dist/web/standalone/.next/server/app/api/terminal/resize/route_client-reference-manifest.js +1 -1
  187. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +1 -1
  188. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route_client-reference-manifest.js +1 -1
  189. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +2 -2
  190. package/dist/web/standalone/.next/server/app/api/terminal/stream/route_client-reference-manifest.js +1 -1
  191. package/dist/web/standalone/.next/server/app/api/terminal/upload/route.js +1 -1
  192. package/dist/web/standalone/.next/server/app/api/terminal/upload/route_client-reference-manifest.js +1 -1
  193. package/dist/web/standalone/.next/server/app/api/undo/route.js +1 -1
  194. package/dist/web/standalone/.next/server/app/api/undo/route_client-reference-manifest.js +1 -1
  195. package/dist/web/standalone/.next/server/app/api/update/route.js +1 -1
  196. package/dist/web/standalone/.next/server/app/api/update/route_client-reference-manifest.js +1 -1
  197. package/dist/web/standalone/.next/server/app/api/visualizer/route.js +1 -1
  198. package/dist/web/standalone/.next/server/app/api/visualizer/route_client-reference-manifest.js +1 -1
  199. package/dist/web/standalone/.next/server/app/index.html +1 -1
  200. package/dist/web/standalone/.next/server/app/index.rsc +4 -4
  201. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
  202. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +4 -4
  203. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  204. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +3 -3
  205. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  206. package/dist/web/standalone/.next/server/app/page.js +2 -2
  207. package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
  208. package/dist/web/standalone/.next/server/app-paths-manifest.json +17 -17
  209. package/dist/web/standalone/.next/server/chunks/63.js +3 -3
  210. package/dist/web/standalone/.next/server/chunks/6897.js +1 -1
  211. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  212. package/dist/web/standalone/.next/server/middleware-react-loadable-manifest.js +1 -1
  213. package/dist/web/standalone/.next/server/middleware.js +2 -2
  214. package/dist/web/standalone/.next/server/next-font-manifest.js +1 -1
  215. package/dist/web/standalone/.next/server/next-font-manifest.json +1 -1
  216. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  217. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  218. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  219. package/dist/web/standalone/.next/static/chunks/2826.dd3dc8bbd3025fa5.js +9 -0
  220. package/dist/web/standalone/.next/static/chunks/app/_not-found/{page-2f24283c162b6ab3.js → page-f2a7482d42a5614b.js} +1 -1
  221. package/dist/web/standalone/.next/static/chunks/app/{layout-9ecfd95f343793f0.js → layout-a16c7a7ecdf0c2cf.js} +1 -1
  222. package/dist/web/standalone/.next/static/chunks/app/page-f1e30ab6bb269149.js +1 -0
  223. package/dist/web/standalone/.next/static/chunks/main-app-fdab67f7802d7832.js +1 -0
  224. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/global-error-459824ffb8c323dd.js +1 -0
  225. package/dist/web/standalone/.next/static/chunks/{webpack-6e4d7e9a4f57bed4.js → webpack-b868033a5834586d.js} +1 -1
  226. package/dist/web/standalone/node_modules/node-pty/build/Makefile +2 -2
  227. package/dist/web/standalone/node_modules/node-pty/build/Release/pty.node +0 -0
  228. package/dist/web/standalone/node_modules/node-pty/build/pty.target.mk +14 -14
  229. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api.target.mk +14 -14
  230. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api_except.target.mk +14 -14
  231. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api_maybe.target.mk +14 -14
  232. package/dist/web/standalone/server.js +1 -1
  233. package/package.json +1 -1
  234. package/packages/mcp-server/dist/env-writer.d.ts +39 -0
  235. package/packages/mcp-server/dist/env-writer.d.ts.map +1 -0
  236. package/packages/mcp-server/dist/env-writer.js +158 -0
  237. package/packages/mcp-server/dist/env-writer.js.map +1 -0
  238. package/packages/mcp-server/dist/server.d.ts +23 -3
  239. package/packages/mcp-server/dist/server.d.ts.map +1 -1
  240. package/packages/mcp-server/dist/server.js +192 -44
  241. package/packages/mcp-server/dist/server.js.map +1 -1
  242. package/packages/mcp-server/dist/workflow-tools.d.ts.map +1 -1
  243. package/packages/mcp-server/dist/workflow-tools.js +22 -12
  244. package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
  245. package/packages/mcp-server/src/env-writer.test.ts +280 -0
  246. package/packages/mcp-server/src/env-writer.ts +183 -0
  247. package/packages/mcp-server/src/secure-env-collect.test.ts +265 -0
  248. package/packages/mcp-server/src/server.ts +247 -41
  249. package/packages/mcp-server/src/workflow-tools.test.ts +110 -0
  250. package/packages/mcp-server/src/workflow-tools.ts +32 -12
  251. package/packages/pi-ai/dist/providers/amazon-bedrock.js +11 -2
  252. package/packages/pi-ai/dist/providers/amazon-bedrock.js.map +1 -1
  253. package/packages/pi-ai/dist/providers/anthropic-auth.test.d.ts +2 -0
  254. package/packages/pi-ai/dist/providers/anthropic-auth.test.d.ts.map +1 -0
  255. package/packages/pi-ai/dist/providers/anthropic-auth.test.js +20 -0
  256. package/packages/pi-ai/dist/providers/anthropic-auth.test.js.map +1 -0
  257. package/packages/pi-ai/dist/providers/anthropic-shared.d.ts +4 -1
  258. package/packages/pi-ai/dist/providers/anthropic-shared.d.ts.map +1 -1
  259. package/packages/pi-ai/dist/providers/anthropic-shared.js +8 -3
  260. package/packages/pi-ai/dist/providers/anthropic-shared.js.map +1 -1
  261. package/packages/pi-ai/dist/providers/anthropic-shared.test.js +44 -1
  262. package/packages/pi-ai/dist/providers/anthropic-shared.test.js.map +1 -1
  263. package/packages/pi-ai/dist/providers/anthropic.d.ts +2 -1
  264. package/packages/pi-ai/dist/providers/anthropic.d.ts.map +1 -1
  265. package/packages/pi-ai/dist/providers/anthropic.js +7 -4
  266. package/packages/pi-ai/dist/providers/anthropic.js.map +1 -1
  267. package/packages/pi-ai/dist/providers/openai-completions.d.ts.map +1 -1
  268. package/packages/pi-ai/dist/providers/openai-completions.js +11 -0
  269. package/packages/pi-ai/dist/providers/openai-completions.js.map +1 -1
  270. package/packages/pi-ai/src/providers/amazon-bedrock.ts +13 -1
  271. package/packages/pi-ai/src/providers/anthropic-auth.test.ts +32 -0
  272. package/packages/pi-ai/src/providers/anthropic-shared.test.ts +55 -1
  273. package/packages/pi-ai/src/providers/anthropic-shared.ts +14 -3
  274. package/packages/pi-ai/src/providers/anthropic.ts +8 -4
  275. package/packages/pi-ai/src/providers/openai-completions.ts +14 -0
  276. package/packages/pi-coding-agent/dist/core/agent-session-renderable-tools.test.d.ts +2 -0
  277. package/packages/pi-coding-agent/dist/core/agent-session-renderable-tools.test.d.ts.map +1 -0
  278. package/packages/pi-coding-agent/dist/core/agent-session-renderable-tools.test.js +61 -0
  279. package/packages/pi-coding-agent/dist/core/agent-session-renderable-tools.test.js.map +1 -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 +2 -1
  282. package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
  283. package/packages/pi-coding-agent/dist/core/auth-storage.d.ts +10 -0
  284. package/packages/pi-coding-agent/dist/core/auth-storage.d.ts.map +1 -1
  285. package/packages/pi-coding-agent/dist/core/auth-storage.js +27 -0
  286. package/packages/pi-coding-agent/dist/core/auth-storage.js.map +1 -1
  287. package/packages/pi-coding-agent/dist/core/auth-storage.test.js +85 -0
  288. package/packages/pi-coding-agent/dist/core/auth-storage.test.js.map +1 -1
  289. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.d.ts +2 -0
  290. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.d.ts.map +1 -0
  291. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js +388 -0
  292. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js.map +1 -0
  293. package/packages/pi-coding-agent/dist/core/extensions/types.d.ts +2 -0
  294. package/packages/pi-coding-agent/dist/core/extensions/types.d.ts.map +1 -1
  295. package/packages/pi-coding-agent/dist/core/extensions/types.js.map +1 -1
  296. package/packages/pi-coding-agent/dist/core/model-resolver-initial-model-auth.test.d.ts +2 -0
  297. package/packages/pi-coding-agent/dist/core/model-resolver-initial-model-auth.test.d.ts.map +1 -0
  298. package/packages/pi-coding-agent/dist/core/model-resolver-initial-model-auth.test.js +64 -0
  299. package/packages/pi-coding-agent/dist/core/model-resolver-initial-model-auth.test.js.map +1 -0
  300. package/packages/pi-coding-agent/dist/core/model-resolver.d.ts.map +1 -1
  301. package/packages/pi-coding-agent/dist/core/model-resolver.js +22 -18
  302. package/packages/pi-coding-agent/dist/core/model-resolver.js.map +1 -1
  303. package/packages/pi-coding-agent/dist/core/model-resolver.test.d.ts +8 -0
  304. package/packages/pi-coding-agent/dist/core/model-resolver.test.d.ts.map +1 -0
  305. package/packages/pi-coding-agent/dist/core/model-resolver.test.js +75 -0
  306. package/packages/pi-coding-agent/dist/core/model-resolver.test.js.map +1 -0
  307. package/packages/pi-coding-agent/dist/core/sdk.d.ts +11 -0
  308. package/packages/pi-coding-agent/dist/core/sdk.d.ts.map +1 -1
  309. package/packages/pi-coding-agent/dist/core/sdk.js +38 -5
  310. package/packages/pi-coding-agent/dist/core/sdk.js.map +1 -1
  311. package/packages/pi-coding-agent/dist/core/sdk.test.d.ts +2 -0
  312. package/packages/pi-coding-agent/dist/core/sdk.test.d.ts.map +1 -0
  313. package/packages/pi-coding-agent/dist/core/sdk.test.js +71 -0
  314. package/packages/pi-coding-agent/dist/core/sdk.test.js.map +1 -0
  315. package/packages/pi-coding-agent/dist/index.d.ts +1 -1
  316. package/packages/pi-coding-agent/dist/index.d.ts.map +1 -1
  317. package/packages/pi-coding-agent/dist/index.js +1 -1
  318. package/packages/pi-coding-agent/dist/index.js.map +1 -1
  319. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/login-dialog.test.d.ts +2 -0
  320. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/login-dialog.test.d.ts.map +1 -0
  321. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/login-dialog.test.js +13 -0
  322. package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/login-dialog.test.js.map +1 -0
  323. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.d.ts +19 -2
  324. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.d.ts.map +1 -1
  325. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.js +50 -1
  326. package/packages/pi-coding-agent/dist/modes/interactive/components/dynamic-border.js.map +1 -1
  327. package/packages/pi-coding-agent/dist/modes/interactive/components/extension-input.d.ts +1 -0
  328. package/packages/pi-coding-agent/dist/modes/interactive/components/extension-input.d.ts.map +1 -1
  329. package/packages/pi-coding-agent/dist/modes/interactive/components/extension-input.js +1 -0
  330. package/packages/pi-coding-agent/dist/modes/interactive/components/extension-input.js.map +1 -1
  331. package/packages/pi-coding-agent/dist/modes/interactive/components/login-dialog.d.ts +4 -0
  332. package/packages/pi-coding-agent/dist/modes/interactive/components/login-dialog.d.ts.map +1 -1
  333. package/packages/pi-coding-agent/dist/modes/interactive/components/login-dialog.js +24 -2
  334. package/packages/pi-coding-agent/dist/modes/interactive/components/login-dialog.js.map +1 -1
  335. package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.d.ts.map +1 -1
  336. package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.js +9 -2
  337. package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.js.map +1 -1
  338. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts +4 -0
  339. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  340. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +43 -0
  341. package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
  342. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  343. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +175 -25
  344. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  345. package/packages/pi-coding-agent/dist/modes/interactive/controllers/model-controller.d.ts.map +1 -1
  346. package/packages/pi-coding-agent/dist/modes/interactive/controllers/model-controller.js +6 -1
  347. package/packages/pi-coding-agent/dist/modes/interactive/controllers/model-controller.js.map +1 -1
  348. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.d.ts +1 -0
  349. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.d.ts.map +1 -1
  350. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode-state.js.map +1 -1
  351. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +6 -0
  352. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  353. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +62 -5
  354. package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
  355. package/packages/pi-coding-agent/dist/modes/interactive/slash-command-handlers.js +4 -2
  356. package/packages/pi-coding-agent/dist/modes/interactive/slash-command-handlers.js.map +1 -1
  357. package/packages/pi-coding-agent/dist/modes/rpc/rpc-mode.js +1 -1
  358. package/packages/pi-coding-agent/dist/modes/rpc/rpc-mode.js.map +1 -1
  359. package/packages/pi-coding-agent/dist/modes/rpc/rpc-types.d.ts +1 -0
  360. package/packages/pi-coding-agent/dist/modes/rpc/rpc-types.d.ts.map +1 -1
  361. package/packages/pi-coding-agent/dist/modes/rpc/rpc-types.js.map +1 -1
  362. package/packages/pi-coding-agent/package.json +1 -1
  363. package/packages/pi-coding-agent/src/core/agent-session-renderable-tools.test.ts +70 -0
  364. package/packages/pi-coding-agent/src/core/agent-session.ts +2 -1
  365. package/packages/pi-coding-agent/src/core/auth-storage.test.ts +108 -0
  366. package/packages/pi-coding-agent/src/core/auth-storage.ts +30 -0
  367. package/packages/pi-coding-agent/src/core/chat-controller-ordering.test.ts +468 -0
  368. package/packages/pi-coding-agent/src/core/extensions/types.ts +2 -0
  369. package/packages/pi-coding-agent/src/core/model-resolver-initial-model-auth.test.ts +78 -0
  370. package/packages/pi-coding-agent/src/core/model-resolver.test.ts +85 -0
  371. package/packages/pi-coding-agent/src/core/model-resolver.ts +22 -18
  372. package/packages/pi-coding-agent/src/core/sdk.test.ts +89 -0
  373. package/packages/pi-coding-agent/src/core/sdk.ts +45 -9
  374. package/packages/pi-coding-agent/src/index.ts +1 -0
  375. package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/login-dialog.test.ts +24 -0
  376. package/packages/pi-coding-agent/src/modes/interactive/components/dynamic-border.ts +58 -2
  377. package/packages/pi-coding-agent/src/modes/interactive/components/extension-input.ts +2 -0
  378. package/packages/pi-coding-agent/src/modes/interactive/components/login-dialog.ts +30 -2
  379. package/packages/pi-coding-agent/src/modes/interactive/components/model-selector.ts +15 -6
  380. package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +47 -0
  381. package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +205 -31
  382. package/packages/pi-coding-agent/src/modes/interactive/controllers/model-controller.ts +6 -1
  383. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode-state.ts +1 -0
  384. package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +70 -5
  385. package/packages/pi-coding-agent/src/modes/interactive/slash-command-handlers.ts +4 -2
  386. package/packages/pi-coding-agent/src/modes/rpc/rpc-mode.ts +1 -1
  387. package/packages/pi-coding-agent/src/modes/rpc/rpc-types.ts +1 -0
  388. package/packages/pi-tui/dist/components/__tests__/input.test.js +9 -0
  389. package/packages/pi-tui/dist/components/__tests__/input.test.js.map +1 -1
  390. package/packages/pi-tui/dist/components/__tests__/markdown-maxlines.test.d.ts +2 -0
  391. package/packages/pi-tui/dist/components/__tests__/markdown-maxlines.test.d.ts.map +1 -0
  392. package/packages/pi-tui/dist/components/__tests__/markdown-maxlines.test.js +66 -0
  393. package/packages/pi-tui/dist/components/__tests__/markdown-maxlines.test.js.map +1 -0
  394. package/packages/pi-tui/dist/components/input.d.ts +2 -0
  395. package/packages/pi-tui/dist/components/input.d.ts.map +1 -1
  396. package/packages/pi-tui/dist/components/input.js +7 -4
  397. package/packages/pi-tui/dist/components/input.js.map +1 -1
  398. package/packages/pi-tui/dist/components/markdown.d.ts +3 -0
  399. package/packages/pi-tui/dist/components/markdown.d.ts.map +1 -1
  400. package/packages/pi-tui/dist/components/markdown.js +17 -1
  401. package/packages/pi-tui/dist/components/markdown.js.map +1 -1
  402. package/packages/pi-tui/src/components/__tests__/input.test.ts +11 -0
  403. package/packages/pi-tui/src/components/__tests__/markdown-maxlines.test.ts +75 -0
  404. package/packages/pi-tui/src/components/input.ts +7 -4
  405. package/packages/pi-tui/src/components/markdown.ts +22 -1
  406. package/pkg/package.json +1 -1
  407. package/src/resources/GSD-WORKFLOW.md +1 -1
  408. package/src/resources/agents/debugger.md +58 -0
  409. package/src/resources/agents/doc-writer.md +43 -0
  410. package/src/resources/agents/git-ops.md +56 -0
  411. package/src/resources/agents/javascript-pro.md +46 -271
  412. package/src/resources/agents/planner.md +55 -0
  413. package/src/resources/agents/refactorer.md +47 -0
  414. package/src/resources/agents/reviewer.md +48 -0
  415. package/src/resources/agents/security.md +59 -0
  416. package/src/resources/agents/tester.md +50 -0
  417. package/src/resources/agents/typescript-pro.md +41 -235
  418. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +288 -39
  419. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +330 -2
  420. package/src/resources/extensions/get-secrets-from-user.ts +24 -1
  421. package/src/resources/extensions/gsd/auto/infra-errors.ts +38 -0
  422. package/src/resources/extensions/gsd/auto/loop-deps.ts +2 -0
  423. package/src/resources/extensions/gsd/auto/loop.ts +45 -1
  424. package/src/resources/extensions/gsd/auto/phases.ts +6 -0
  425. package/src/resources/extensions/gsd/auto/session.ts +11 -0
  426. package/src/resources/extensions/gsd/auto-dashboard.ts +29 -18
  427. package/src/resources/extensions/gsd/auto-model-selection.ts +9 -1
  428. package/src/resources/extensions/gsd/auto-prompts.ts +111 -33
  429. package/src/resources/extensions/gsd/auto-start.ts +44 -20
  430. package/src/resources/extensions/gsd/auto-tool-tracking.ts +1 -1
  431. package/src/resources/extensions/gsd/auto-worktree.ts +1 -1
  432. package/src/resources/extensions/gsd/auto.ts +72 -0
  433. package/src/resources/extensions/gsd/bootstrap/db-tools.ts +3 -3
  434. package/src/resources/extensions/gsd/bootstrap/register-hooks.ts +6 -0
  435. package/src/resources/extensions/gsd/bootstrap/register-shortcuts.ts +79 -60
  436. package/src/resources/extensions/gsd/bootstrap/system-context.ts +7 -0
  437. package/src/resources/extensions/gsd/commands/context.ts +16 -5
  438. package/src/resources/extensions/gsd/commands/dispatcher.ts +14 -2
  439. package/src/resources/extensions/gsd/commands/handlers/auto.ts +10 -36
  440. package/src/resources/extensions/gsd/commands/handlers/core.ts +58 -11
  441. package/src/resources/extensions/gsd/commands/handlers/notifications-handler.ts +17 -7
  442. package/src/resources/extensions/gsd/commands/handlers/workflow.ts +4 -10
  443. package/src/resources/extensions/gsd/custom-workflow-engine.ts +19 -14
  444. package/src/resources/extensions/gsd/dashboard-overlay.ts +10 -3
  445. package/src/resources/extensions/gsd/dispatch-guard.ts +18 -1
  446. package/src/resources/extensions/gsd/doctor-providers.ts +24 -0
  447. package/src/resources/extensions/gsd/error-classifier.ts +1 -1
  448. package/src/resources/extensions/gsd/file-lock.ts +59 -0
  449. package/src/resources/extensions/gsd/forensics.ts +23 -7
  450. package/src/resources/extensions/gsd/gate-registry.ts +251 -0
  451. package/src/resources/extensions/gsd/gsd-db.ts +51 -0
  452. package/src/resources/extensions/gsd/guided-flow.ts +17 -19
  453. package/src/resources/extensions/gsd/init-wizard.ts +3 -13
  454. package/src/resources/extensions/gsd/interrupted-session.ts +1 -0
  455. package/src/resources/extensions/gsd/metrics.ts +12 -1
  456. package/src/resources/extensions/gsd/milestone-actions.ts +10 -3
  457. package/src/resources/extensions/gsd/milestone-validation-gates.ts +11 -13
  458. package/src/resources/extensions/gsd/notification-overlay.ts +47 -14
  459. package/src/resources/extensions/gsd/notification-store.ts +54 -5
  460. package/src/resources/extensions/gsd/notification-widget.ts +5 -14
  461. package/src/resources/extensions/gsd/parallel-monitor-overlay.ts +10 -3
  462. package/src/resources/extensions/gsd/pre-execution-checks.ts +39 -2
  463. package/src/resources/extensions/gsd/prompt-validation.ts +157 -0
  464. package/src/resources/extensions/gsd/prompts/complete-slice.md +5 -3
  465. package/src/resources/extensions/gsd/prompts/discuss.md +33 -13
  466. package/src/resources/extensions/gsd/prompts/execute-task.md +22 -19
  467. package/src/resources/extensions/gsd/prompts/guided-discuss-milestone.md +2 -0
  468. package/src/resources/extensions/gsd/prompts/guided-discuss-slice.md +2 -0
  469. package/src/resources/extensions/gsd/prompts/guided-resume-task.md +1 -1
  470. package/src/resources/extensions/gsd/prompts/queue.md +3 -2
  471. package/src/resources/extensions/gsd/prompts/system.md +1 -0
  472. package/src/resources/extensions/gsd/prompts/validate-milestone.md +4 -1
  473. package/src/resources/extensions/gsd/session-model-override.ts +36 -0
  474. package/src/resources/extensions/gsd/shortcut-defs.ts +56 -0
  475. package/src/resources/extensions/gsd/state.ts +285 -344
  476. package/src/resources/extensions/gsd/tests/auto-start-model-capture.test.ts +25 -9
  477. package/src/resources/extensions/gsd/tests/auto-start-worktree-db-path.test.ts +28 -0
  478. package/src/resources/extensions/gsd/tests/bootstrap-derive-state-db-open.test.ts +39 -0
  479. package/src/resources/extensions/gsd/tests/complete-slice-gate-closure.test.ts +167 -0
  480. package/src/resources/extensions/gsd/tests/complete-slice-prompt-task-summary-layout.test.ts +18 -0
  481. package/src/resources/extensions/gsd/tests/derive-state-helpers.test.ts +436 -0
  482. package/src/resources/extensions/gsd/tests/discuss-incremental-persistence.test.ts +9 -0
  483. package/src/resources/extensions/gsd/tests/dispatch-guard.test.ts +27 -0
  484. package/src/resources/extensions/gsd/tests/doctor-providers.test.ts +36 -0
  485. package/src/resources/extensions/gsd/tests/execute-task-prompt-existing-artifact-guard.test.ts +33 -0
  486. package/src/resources/extensions/gsd/tests/file-lock.test.ts +103 -0
  487. package/src/resources/extensions/gsd/tests/forensics-stuck-loops.test.ts +62 -0
  488. package/src/resources/extensions/gsd/tests/format-shortcut.test.ts +31 -0
  489. package/src/resources/extensions/gsd/tests/gate-dispatch.test.ts +27 -0
  490. package/src/resources/extensions/gsd/tests/gate-registry.test.ts +140 -0
  491. package/src/resources/extensions/gsd/tests/gsd-no-project-error.test.ts +73 -0
  492. package/src/resources/extensions/gsd/tests/infra-errors-cooldown.test.ts +180 -0
  493. package/src/resources/extensions/gsd/tests/integration/auto-worktree-milestone-merge.test.ts +66 -1
  494. package/src/resources/extensions/gsd/tests/model-isolation.test.ts +36 -51
  495. package/src/resources/extensions/gsd/tests/notification-store.test.ts +35 -0
  496. package/src/resources/extensions/gsd/tests/notification-widget.test.ts +26 -0
  497. package/src/resources/extensions/gsd/tests/notifications-handler.test.ts +90 -0
  498. package/src/resources/extensions/gsd/tests/parallel-monitor-overlay.test.ts +1 -0
  499. package/src/resources/extensions/gsd/tests/park-db-sync.test.ts +18 -0
  500. package/src/resources/extensions/gsd/tests/pre-execution-checks.test.ts +49 -0
  501. package/src/resources/extensions/gsd/tests/prompt-contracts.test.ts +19 -0
  502. package/src/resources/extensions/gsd/tests/prompt-system-gate-coverage.test.ts +208 -0
  503. package/src/resources/extensions/gsd/tests/provider-errors.test.ts +7 -0
  504. package/src/resources/extensions/gsd/tests/register-shortcuts.test.ts +63 -5
  505. package/src/resources/extensions/gsd/tests/secure-env-collect.test.ts +45 -0
  506. package/src/resources/extensions/gsd/tests/session-model-override.test.ts +35 -0
  507. package/src/resources/extensions/gsd/tests/start-auto-detached.test.ts +90 -0
  508. package/src/resources/extensions/gsd/tests/tool-invocation-error-loop-break.test.ts +7 -0
  509. package/src/resources/extensions/gsd/tests/validate-milestone-prompt-verification-classes.test.ts +18 -0
  510. package/src/resources/extensions/gsd/tests/workflow-mcp-auto-prep.test.ts +76 -0
  511. package/src/resources/extensions/gsd/tests/workflow-mcp.test.ts +155 -1
  512. package/src/resources/extensions/gsd/tests/workflow-tool-executors.test.ts +22 -0
  513. package/src/resources/extensions/gsd/tools/complete-slice.ts +63 -0
  514. package/src/resources/extensions/gsd/tools/complete-task.ts +63 -0
  515. package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +64 -26
  516. package/src/resources/extensions/gsd/types.ts +26 -0
  517. package/src/resources/extensions/gsd/workflow-events.ts +34 -25
  518. package/src/resources/extensions/gsd/workflow-mcp-auto-prep.ts +76 -0
  519. package/src/resources/extensions/gsd/workflow-mcp.ts +1 -1
  520. package/src/resources/extensions/ollama/index.ts +13 -3
  521. package/src/resources/extensions/ollama/ollama-status-indicator.test.ts +28 -0
  522. package/src/resources/extensions/shared/gsd-phase-state.ts +42 -0
  523. package/src/resources/extensions/shared/tests/gsd-phase-state.test.ts +48 -0
  524. package/src/resources/extensions/subagent/agents.ts +10 -0
  525. package/src/resources/extensions/subagent/index.ts +18 -0
  526. package/src/resources/extensions/subagent/tests/agents-conflicts.test.ts +33 -0
  527. package/src/resources/skills/create-skill/SKILL.md +2 -0
  528. package/dist/web/standalone/.next/static/chunks/2826.821e01b07d92e948.js +0 -9
  529. package/dist/web/standalone/.next/static/chunks/app/page-7115e62689b5fd84.js +0 -1
  530. package/dist/web/standalone/.next/static/chunks/main-app-d3d4c336195465f9.js +0 -1
  531. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/global-error-ab5a8926e07ec673.js +0 -1
  532. /package/dist/web/standalone/.next/static/{9pw9EXtXjdM7EFrCXUEPf → OI4n_CKC-lM8IQbvGJ_tK}/_buildManifest.js +0 -0
  533. /package/dist/web/standalone/.next/static/{9pw9EXtXjdM7EFrCXUEPf → OI4n_CKC-lM8IQbvGJ_tK}/_ssgManifest.js +0 -0
@@ -0,0 +1,468 @@
1
+ import assert from "node:assert/strict";
2
+ import { test } from "node:test";
3
+
4
+ import { handleAgentEvent } from "../modes/interactive/controllers/chat-controller.js";
5
+
6
+ function makeUsage() {
7
+ return {
8
+ input: 0,
9
+ output: 0,
10
+ cacheRead: 0,
11
+ cacheWrite: 0,
12
+ totalTokens: 0,
13
+ cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0, total: 0 },
14
+ };
15
+ }
16
+
17
+ function makeAssistant(content: any[]) {
18
+ return {
19
+ role: "assistant",
20
+ content,
21
+ api: "anthropic-messages",
22
+ provider: "claude-code",
23
+ model: "claude-sonnet-4",
24
+ usage: makeUsage(),
25
+ stopReason: "stop",
26
+ timestamp: Date.now(),
27
+ };
28
+ }
29
+
30
+ function createHost() {
31
+ const chatContainer = {
32
+ children: [] as any[],
33
+ addChild(component: any) {
34
+ this.children.push(component);
35
+ },
36
+ removeChild(component: any) {
37
+ const idx = this.children.indexOf(component);
38
+ if (idx !== -1) this.children.splice(idx, 1);
39
+ },
40
+ clear() {
41
+ this.children = [];
42
+ },
43
+ };
44
+
45
+ const pinnedMessageContainer = {
46
+ children: [] as any[],
47
+ addChild(component: any) {
48
+ this.children.push(component);
49
+ },
50
+ removeChild(component: any) {
51
+ const idx = this.children.indexOf(component);
52
+ if (idx !== -1) this.children.splice(idx, 1);
53
+ },
54
+ clear() {
55
+ this.children = [];
56
+ },
57
+ };
58
+
59
+ const host: any = {
60
+ isInitialized: true,
61
+ init: async () => {},
62
+ defaultEditor: { onEscape: undefined },
63
+ editor: {},
64
+ session: { retryAttempt: 0, abortCompaction: () => {}, abortRetry: () => {} },
65
+ ui: { requestRender: () => {}, terminal: { rows: 50 } },
66
+ footer: { invalidate: () => {} },
67
+ keybindings: {},
68
+ statusContainer: { clear: () => {}, addChild: () => {} },
69
+ chatContainer,
70
+ settingsManager: { getTimestampFormat: () => "date-time-iso", getShowImages: () => false },
71
+ pendingTools: new Map(),
72
+ toolOutputExpanded: false,
73
+ hideThinkingBlock: false,
74
+ isBashMode: false,
75
+ defaultWorkingMessage: "Working...",
76
+ compactionQueuedMessages: [],
77
+ editorContainer: {},
78
+ pendingMessagesContainer: { clear: () => {} },
79
+ pinnedMessageContainer,
80
+ addMessageToChat: () => {},
81
+ getMarkdownThemeWithSettings: () => ({}),
82
+ formatWebSearchResult: () => "",
83
+ getRegisteredToolDefinition: () => undefined,
84
+ checkShutdownRequested: async () => {},
85
+ rebuildChatFromMessages: () => {},
86
+ flushCompactionQueue: async () => {},
87
+ showStatus: () => {},
88
+ showError: () => {},
89
+ updatePendingMessagesDisplay: () => {},
90
+ updateTerminalTitle: () => {},
91
+ updateEditorBorderColor: () => {},
92
+ };
93
+
94
+ return host;
95
+ }
96
+
97
+ test("chat-controller keeps tool output ahead of delayed assistant text for external tool streams", async () => {
98
+ // ToolExecutionComponent uses the global theme singleton.
99
+ // Install a minimal no-op theme implementation for this unit test.
100
+ (globalThis as any)[Symbol.for("@gsd/pi-coding-agent:theme")] = {
101
+ fg: (_key: string, text: string) => text,
102
+ bg: (_key: string, text: string) => text,
103
+ bold: (text: string) => text,
104
+ italic: (text: string) => text,
105
+ truncate: (text: string) => text,
106
+ };
107
+
108
+ const host = createHost();
109
+ const toolId = "mcp-tool-1";
110
+ const toolCall = {
111
+ type: "toolCall",
112
+ id: toolId,
113
+ name: "exec_command",
114
+ arguments: { cmd: "echo hi" },
115
+ };
116
+
117
+ await handleAgentEvent(host, { type: "message_start", message: makeAssistant([]) } as any);
118
+
119
+ assert.equal(host.streamingComponent, undefined, "assistant component should be deferred at message_start");
120
+ assert.equal(host.chatContainer.children.length, 0, "nothing should render before content arrives");
121
+
122
+ await handleAgentEvent(
123
+ host,
124
+ {
125
+ type: "message_update",
126
+ message: makeAssistant([toolCall]),
127
+ assistantMessageEvent: {
128
+ type: "toolcall_end",
129
+ contentIndex: 0,
130
+ toolCall: {
131
+ ...toolCall,
132
+ externalResult: {
133
+ content: [{ type: "text", text: "tool output" }],
134
+ details: {},
135
+ isError: false,
136
+ },
137
+ },
138
+ partial: makeAssistant([toolCall]),
139
+ },
140
+ } as any,
141
+ );
142
+
143
+ assert.equal(host.streamingComponent, undefined, "assistant text container should remain deferred for tool-only updates");
144
+ assert.equal(host.chatContainer.children.length, 1, "tool execution block should render immediately");
145
+ assert.equal(host.chatContainer.children[0]?.constructor?.name, "ToolExecutionComponent");
146
+
147
+ // Re-assert required host method before the text-bearing update path.
148
+ host.getMarkdownThemeWithSettings = () => ({});
149
+
150
+ await handleAgentEvent(
151
+ host,
152
+ {
153
+ type: "message_update",
154
+ message: makeAssistant([toolCall, { type: "text", text: "done" }]),
155
+ assistantMessageEvent: {
156
+ type: "text_delta",
157
+ contentIndex: 1,
158
+ delta: "done",
159
+ partial: makeAssistant([toolCall, { type: "text", text: "done" }]),
160
+ },
161
+ } as any,
162
+ );
163
+
164
+ assert.equal(host.chatContainer.children.length, 2, "assistant content should render after existing tool output");
165
+ assert.equal(host.chatContainer.children[0]?.constructor?.name, "ToolExecutionComponent");
166
+ assert.equal(host.chatContainer.children[1]?.constructor?.name, "AssistantMessageComponent");
167
+ });
168
+
169
+ test("chat-controller keeps serverToolUse output ahead of assistant text when external results arrive", async () => {
170
+ (globalThis as any)[Symbol.for("@gsd/pi-coding-agent:theme")] = {
171
+ fg: (_key: string, text: string) => text,
172
+ bg: (_key: string, text: string) => text,
173
+ bold: (text: string) => text,
174
+ italic: (text: string) => text,
175
+ truncate: (text: string) => text,
176
+ };
177
+
178
+ const host = createHost();
179
+ const toolId = "mcp-secure-1";
180
+ const serverToolUse = {
181
+ type: "serverToolUse",
182
+ id: toolId,
183
+ name: "mcp__gsd-workflow__secure_env_collect",
184
+ input: { projectDir: "/tmp/project", keys: [{ key: "SECURE_PASSWORD" }], destination: "dotenv" },
185
+ };
186
+
187
+ await handleAgentEvent(host, { type: "message_start", message: makeAssistant([]) } as any);
188
+
189
+ await handleAgentEvent(
190
+ host,
191
+ {
192
+ type: "message_update",
193
+ message: makeAssistant([serverToolUse]),
194
+ assistantMessageEvent: {
195
+ type: "server_tool_use",
196
+ contentIndex: 0,
197
+ partial: makeAssistant([serverToolUse]),
198
+ },
199
+ } as any,
200
+ );
201
+
202
+ assert.equal(host.streamingComponent, undefined, "assistant content should stay deferred while only tool content streams");
203
+ assert.equal(host.chatContainer.children.length, 1, "server tool block should render immediately");
204
+ assert.equal(host.chatContainer.children[0]?.constructor?.name, "ToolExecutionComponent");
205
+
206
+ host.getMarkdownThemeWithSettings = () => ({});
207
+ const resultMessage = makeAssistant([
208
+ {
209
+ ...serverToolUse,
210
+ externalResult: {
211
+ content: [{ type: "text", text: "secure_env_collect was cancelled by user." }],
212
+ details: {},
213
+ isError: true,
214
+ },
215
+ },
216
+ { type: "text", text: "The secure password collection was cancelled." },
217
+ ]);
218
+
219
+ await handleAgentEvent(
220
+ host,
221
+ {
222
+ type: "message_update",
223
+ message: resultMessage,
224
+ assistantMessageEvent: {
225
+ type: "server_tool_use",
226
+ contentIndex: 0,
227
+ partial: resultMessage,
228
+ },
229
+ } as any,
230
+ );
231
+
232
+ assert.equal(host.chatContainer.children.length, 2, "assistant text should render after existing server tool output");
233
+ assert.equal(host.chatContainer.children[0]?.constructor?.name, "ToolExecutionComponent");
234
+ assert.equal(host.chatContainer.children[1]?.constructor?.name, "AssistantMessageComponent");
235
+ });
236
+
237
+ test("chat-controller pins latest assistant text above editor when tool calls are present", async () => {
238
+ (globalThis as any)[Symbol.for("@gsd/pi-coding-agent:theme")] = {
239
+ fg: (_key: string, text: string) => text,
240
+ bg: (_key: string, text: string) => text,
241
+ bold: (text: string) => text,
242
+ italic: (text: string) => text,
243
+ truncate: (text: string) => text,
244
+ };
245
+
246
+ const host = createHost();
247
+ const toolId = "tool-pin-1";
248
+ const toolCall = {
249
+ type: "toolCall",
250
+ id: toolId,
251
+ name: "exec_command",
252
+ arguments: { cmd: "echo hi" },
253
+ };
254
+
255
+ await handleAgentEvent(host, { type: "message_start", message: makeAssistant([]) } as any);
256
+
257
+ assert.equal(host.pinnedMessageContainer.children.length, 0, "pinned zone should be empty at message_start");
258
+
259
+ // Send a message with text followed by a tool call
260
+ host.getMarkdownThemeWithSettings = () => ({});
261
+ await handleAgentEvent(
262
+ host,
263
+ {
264
+ type: "message_update",
265
+ message: makeAssistant([
266
+ { type: "text", text: "Looking at the files now." },
267
+ toolCall,
268
+ ]),
269
+ assistantMessageEvent: {
270
+ type: "toolcall_end",
271
+ contentIndex: 1,
272
+ toolCall: {
273
+ ...toolCall,
274
+ externalResult: {
275
+ content: [{ type: "text", text: "file contents" }],
276
+ details: {},
277
+ isError: false,
278
+ },
279
+ },
280
+ partial: makeAssistant([{ type: "text", text: "Looking at the files now." }, toolCall]),
281
+ },
282
+ } as any,
283
+ );
284
+
285
+ // Pinned zone should now have a DynamicBorder and a Markdown component
286
+ assert.equal(host.pinnedMessageContainer.children.length, 2, "pinned zone should have border + markdown");
287
+ assert.equal(host.pinnedMessageContainer.children[0]?.constructor?.name, "DynamicBorder");
288
+ assert.equal(host.pinnedMessageContainer.children[1]?.constructor?.name, "Markdown");
289
+ });
290
+
291
+ test("chat-controller clears pinned zone when a new assistant message starts", async () => {
292
+ (globalThis as any)[Symbol.for("@gsd/pi-coding-agent:theme")] = {
293
+ fg: (_key: string, text: string) => text,
294
+ bg: (_key: string, text: string) => text,
295
+ bold: (text: string) => text,
296
+ italic: (text: string) => text,
297
+ truncate: (text: string) => text,
298
+ };
299
+
300
+ const host = createHost();
301
+ const toolCall = {
302
+ type: "toolCall",
303
+ id: "tool-clear-1",
304
+ name: "exec_command",
305
+ arguments: { cmd: "echo hi" },
306
+ };
307
+
308
+ await handleAgentEvent(host, { type: "message_start", message: makeAssistant([]) } as any);
309
+
310
+ // Populate the pinned zone
311
+ host.getMarkdownThemeWithSettings = () => ({});
312
+ await handleAgentEvent(
313
+ host,
314
+ {
315
+ type: "message_update",
316
+ message: makeAssistant([{ type: "text", text: "Working on it." }, toolCall]),
317
+ assistantMessageEvent: {
318
+ type: "toolcall_end",
319
+ contentIndex: 1,
320
+ toolCall: {
321
+ ...toolCall,
322
+ externalResult: {
323
+ content: [{ type: "text", text: "ok" }],
324
+ details: {},
325
+ isError: false,
326
+ },
327
+ },
328
+ partial: makeAssistant([{ type: "text", text: "Working on it." }, toolCall]),
329
+ },
330
+ } as any,
331
+ );
332
+
333
+ assert.ok(host.pinnedMessageContainer.children.length > 0, "pinned zone should be populated");
334
+
335
+ // Start a new assistant message — pinned zone should clear
336
+ await handleAgentEvent(host, { type: "message_start", message: makeAssistant([]) } as any);
337
+
338
+ assert.equal(host.pinnedMessageContainer.children.length, 0, "pinned zone should clear on new assistant message");
339
+ });
340
+
341
+ test("chat-controller clears pinned zone when the agent turn ends", async () => {
342
+ (globalThis as any)[Symbol.for("@gsd/pi-coding-agent:theme")] = {
343
+ fg: (_key: string, text: string) => text,
344
+ bg: (_key: string, text: string) => text,
345
+ bold: (text: string) => text,
346
+ italic: (text: string) => text,
347
+ truncate: (text: string) => text,
348
+ };
349
+
350
+ const host = createHost();
351
+ const toolCall = {
352
+ type: "toolCall",
353
+ id: "tool-clear-on-end-1",
354
+ name: "exec_command",
355
+ arguments: { cmd: "echo hi" },
356
+ };
357
+
358
+ await handleAgentEvent(host, { type: "message_start", message: makeAssistant([]) } as any);
359
+
360
+ host.getMarkdownThemeWithSettings = () => ({});
361
+ await handleAgentEvent(
362
+ host,
363
+ {
364
+ type: "message_update",
365
+ message: makeAssistant([{ type: "text", text: "Working on it." }, toolCall]),
366
+ assistantMessageEvent: {
367
+ type: "toolcall_end",
368
+ contentIndex: 1,
369
+ toolCall: {
370
+ ...toolCall,
371
+ externalResult: {
372
+ content: [{ type: "text", text: "ok" }],
373
+ details: {},
374
+ isError: false,
375
+ },
376
+ },
377
+ partial: makeAssistant([{ type: "text", text: "Working on it." }, toolCall]),
378
+ },
379
+ } as any,
380
+ );
381
+
382
+ assert.ok(host.pinnedMessageContainer.children.length > 0, "pinned zone should be populated before agent_end");
383
+
384
+ await handleAgentEvent(host, { type: "agent_end" } as any);
385
+
386
+ assert.equal(host.pinnedMessageContainer.children.length, 0, "pinned zone should clear on agent_end");
387
+ });
388
+
389
+ test("chat-controller clears pinned zone when assistant message ends", async () => {
390
+ (globalThis as any)[Symbol.for("@gsd/pi-coding-agent:theme")] = {
391
+ fg: (_key: string, text: string) => text,
392
+ bg: (_key: string, text: string) => text,
393
+ bold: (text: string) => text,
394
+ italic: (text: string) => text,
395
+ truncate: (text: string) => text,
396
+ };
397
+
398
+ const host = createHost();
399
+ const toolCall = {
400
+ type: "toolCall",
401
+ id: "tool-msg-end-1",
402
+ name: "exec_command",
403
+ arguments: { cmd: "echo hi" },
404
+ };
405
+
406
+ await handleAgentEvent(host, { type: "message_start", message: makeAssistant([]) } as any);
407
+
408
+ host.getMarkdownThemeWithSettings = () => ({});
409
+ const msgContent = [{ type: "text", text: "Summary after tools." }, toolCall];
410
+ await handleAgentEvent(
411
+ host,
412
+ {
413
+ type: "message_update",
414
+ message: makeAssistant(msgContent),
415
+ assistantMessageEvent: {
416
+ type: "toolcall_end",
417
+ contentIndex: 1,
418
+ toolCall: {
419
+ ...toolCall,
420
+ externalResult: {
421
+ content: [{ type: "text", text: "ok" }],
422
+ details: {},
423
+ isError: false,
424
+ },
425
+ },
426
+ partial: makeAssistant(msgContent),
427
+ },
428
+ } as any,
429
+ );
430
+
431
+ assert.ok(host.pinnedMessageContainer.children.length > 0, "pinned zone should be populated during streaming");
432
+
433
+ // End the assistant message (e.g. before form elicitation) — pinned zone should clear
434
+ await handleAgentEvent(host, { type: "message_end", message: makeAssistant(msgContent) } as any);
435
+
436
+ assert.equal(host.pinnedMessageContainer.children.length, 0, "pinned zone should clear on message_end to prevent duplicate display");
437
+ });
438
+
439
+ test("chat-controller does not pin when there are no tool calls", async () => {
440
+ (globalThis as any)[Symbol.for("@gsd/pi-coding-agent:theme")] = {
441
+ fg: (_key: string, text: string) => text,
442
+ bg: (_key: string, text: string) => text,
443
+ bold: (text: string) => text,
444
+ italic: (text: string) => text,
445
+ truncate: (text: string) => text,
446
+ };
447
+
448
+ const host = createHost();
449
+
450
+ await handleAgentEvent(host, { type: "message_start", message: makeAssistant([]) } as any);
451
+
452
+ host.getMarkdownThemeWithSettings = () => ({});
453
+ await handleAgentEvent(
454
+ host,
455
+ {
456
+ type: "message_update",
457
+ message: makeAssistant([{ type: "text", text: "Just some text, no tools." }]),
458
+ assistantMessageEvent: {
459
+ type: "text_delta",
460
+ contentIndex: 0,
461
+ delta: "Just some text, no tools.",
462
+ partial: makeAssistant([{ type: "text", text: "Just some text, no tools." }]),
463
+ },
464
+ } as any,
465
+ );
466
+
467
+ assert.equal(host.pinnedMessageContainer.children.length, 0, "pinned zone should stay empty without tool calls");
468
+ });
@@ -88,6 +88,8 @@ export interface ExtensionUIDialogOptions {
88
88
  timeout?: number;
89
89
  /** When true, the user can select multiple options. The return type becomes `string[]`. */
90
90
  allowMultiple?: boolean;
91
+ /** When true, text input dialogs should hide typed characters if supported by the client surface. */
92
+ secure?: boolean;
91
93
  }
92
94
 
93
95
  /** Placement for extension widgets. */
@@ -0,0 +1,78 @@
1
+ import assert from "node:assert/strict";
2
+ import { describe, it } from "node:test";
3
+ import type { Api, Model } from "@gsd/pi-ai";
4
+ import type { ModelRegistry } from "./model-registry.js";
5
+ import { findInitialModel } from "./model-resolver.js";
6
+
7
+ function makeModel(provider: string, id: string): Model<Api> {
8
+ return {
9
+ id,
10
+ name: id,
11
+ provider,
12
+ api: "openai-responses",
13
+ reasoning: false,
14
+ input: ["text"],
15
+ cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },
16
+ contextWindow: 128000,
17
+ maxTokens: 8192,
18
+ } as Model<Api>;
19
+ }
20
+
21
+ function makeRegistry(opts: {
22
+ readyProviders?: Set<string>;
23
+ byProviderAndId?: Map<string, Model<Api>>;
24
+ available?: Model<Api>[];
25
+ }): ModelRegistry {
26
+ const readyProviders = opts.readyProviders ?? new Set<string>();
27
+ const byProviderAndId = opts.byProviderAndId ?? new Map<string, Model<Api>>();
28
+ const available = opts.available ?? [];
29
+
30
+ return {
31
+ find: (provider: string, modelId: string) => byProviderAndId.get(`${provider}/${modelId}`),
32
+ getAvailable: async () => available,
33
+ isProviderRequestReady: (provider: string) => readyProviders.has(provider),
34
+ } as unknown as ModelRegistry;
35
+ }
36
+
37
+ describe("findInitialModel auth gating for saved defaults", () => {
38
+ it("uses saved default when provider is request-ready", async () => {
39
+ const saved = makeModel("anthropic", "claude-opus-4-6");
40
+ const registry = makeRegistry({
41
+ readyProviders: new Set(["anthropic"]),
42
+ byProviderAndId: new Map([[`anthropic/claude-opus-4-6`, saved]]),
43
+ available: [saved],
44
+ });
45
+
46
+ const result = await findInitialModel({
47
+ scopedModels: [],
48
+ isContinuing: false,
49
+ defaultProvider: "anthropic",
50
+ defaultModelId: "claude-opus-4-6",
51
+ modelRegistry: registry,
52
+ });
53
+
54
+ assert.equal(result.model?.provider, "anthropic");
55
+ assert.equal(result.model?.id, "claude-opus-4-6");
56
+ });
57
+
58
+ it("skips saved default when provider is not request-ready and falls back to available", async () => {
59
+ const staleDefault = makeModel("anthropic", "claude-opus-4-6");
60
+ const fallback = makeModel("openai", "gpt-5.4");
61
+ const registry = makeRegistry({
62
+ readyProviders: new Set(["openai"]),
63
+ byProviderAndId: new Map([[`anthropic/claude-opus-4-6`, staleDefault]]),
64
+ available: [fallback],
65
+ });
66
+
67
+ const result = await findInitialModel({
68
+ scopedModels: [],
69
+ isContinuing: false,
70
+ defaultProvider: "anthropic",
71
+ defaultModelId: "claude-opus-4-6",
72
+ modelRegistry: registry,
73
+ });
74
+
75
+ assert.equal(result.model?.provider, "openai");
76
+ assert.equal(result.model?.id, "gpt-5.4");
77
+ });
78
+ });
@@ -0,0 +1,85 @@
1
+ /**
2
+ * Regression test for the #unconfigured-models fix: findInitialModel() must
3
+ * skip the saved default when its provider has no working auth, rather than
4
+ * returning an unusable model that every selector surface would display as
5
+ * "current".
6
+ */
7
+
8
+ import test from "node:test";
9
+ import assert from "node:assert/strict";
10
+ import { findInitialModel } from "./model-resolver.js";
11
+
12
+ function fakeRegistry(options: {
13
+ models: Array<{ provider: string; id: string }>;
14
+ readyProviders: Set<string>;
15
+ }) {
16
+ const fullModels = options.models.map((m) => ({
17
+ ...m,
18
+ name: m.id,
19
+ api: "anthropic-messages",
20
+ baseUrl: "",
21
+ reasoning: false,
22
+ input: ["text"],
23
+ cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },
24
+ contextWindow: 128_000,
25
+ maxTokens: 4096,
26
+ }));
27
+ const available = fullModels.filter((m) => options.readyProviders.has(m.provider));
28
+ return {
29
+ find(provider: string, id: string) {
30
+ return fullModels.find((m) => m.provider === provider && m.id === id);
31
+ },
32
+ getAvailable() {
33
+ return available;
34
+ },
35
+ isProviderRequestReady(provider: string) {
36
+ return options.readyProviders.has(provider);
37
+ },
38
+ };
39
+ }
40
+
41
+ test("findInitialModel skips saved default when provider has no auth", async () => {
42
+ // User saved xai/grok-4 as default, but XAI_API_KEY is unset so xai is
43
+ // in the registry but not ready. Previously findInitialModel() step 3
44
+ // returned xai anyway — now it must fall through to step 4 and pick
45
+ // an available model.
46
+ const registry = fakeRegistry({
47
+ models: [
48
+ { provider: "xai", id: "grok-4-fast-non-reasoning" },
49
+ { provider: "anthropic", id: "claude-opus-4-6" },
50
+ ],
51
+ readyProviders: new Set(["anthropic"]),
52
+ });
53
+
54
+ const result = await findInitialModel({
55
+ scopedModels: [],
56
+ isContinuing: false,
57
+ defaultProvider: "xai",
58
+ defaultModelId: "grok-4-fast-non-reasoning",
59
+ modelRegistry: registry as any,
60
+ });
61
+
62
+ assert.ok(result.model, "a model must be returned");
63
+ assert.equal(result.model!.provider, "anthropic", "unauth'd saved default must be skipped");
64
+ });
65
+
66
+ test("findInitialModel keeps saved default when provider has auth", async () => {
67
+ const registry = fakeRegistry({
68
+ models: [
69
+ { provider: "anthropic", id: "claude-opus-4-6" },
70
+ { provider: "openai", id: "gpt-5.4" },
71
+ ],
72
+ readyProviders: new Set(["anthropic", "openai"]),
73
+ });
74
+
75
+ const result = await findInitialModel({
76
+ scopedModels: [],
77
+ isContinuing: false,
78
+ defaultProvider: "openai",
79
+ defaultModelId: "gpt-5.4",
80
+ modelRegistry: registry as any,
81
+ });
82
+
83
+ assert.equal(result.model?.provider, "openai");
84
+ assert.equal(result.model?.id, "gpt-5.4");
85
+ });
@@ -504,27 +504,31 @@ export async function findInitialModel(options: {
504
504
 
505
505
  // 3. Try saved default from settings
506
506
  if (defaultProvider && defaultModelId) {
507
- const found = modelRegistry.find(defaultProvider, defaultModelId);
508
- if (found) {
509
- // Check if the provider's recommended default is a higher-capability variant
510
- // of the saved model (e.g. saved "claude-opus-4-6" vs recommended "claude-opus-4-6-extended").
511
- // If so, prefer the recommended variant to avoid using a smaller context window (#1125).
512
- const recommendedId = defaultModelPerProvider[defaultProvider as KnownProvider];
513
- if (recommendedId && recommendedId !== defaultModelId && recommendedId.startsWith(defaultModelId)) {
514
- const recommended = modelRegistry.find(defaultProvider, recommendedId);
515
- if (recommended) {
516
- model = recommended;
517
- if (defaultThinkingLevel) {
518
- thinkingLevel = defaultThinkingLevel;
507
+ // Guard against stale settings defaults: only use the saved provider/model
508
+ // if the provider is actually request-ready (auth/OAuth/CLI ready).
509
+ if (modelRegistry.isProviderRequestReady(defaultProvider)) {
510
+ const found = modelRegistry.find(defaultProvider, defaultModelId);
511
+ if (found) {
512
+ // Check if the provider's recommended default is a higher-capability variant
513
+ // of the saved model (e.g. saved "claude-opus-4-6" vs recommended "claude-opus-4-6-extended").
514
+ // If so, prefer the recommended variant to avoid using a smaller context window (#1125).
515
+ const recommendedId = defaultModelPerProvider[defaultProvider as KnownProvider];
516
+ if (recommendedId && recommendedId !== defaultModelId && recommendedId.startsWith(defaultModelId)) {
517
+ const recommended = modelRegistry.find(defaultProvider, recommendedId);
518
+ if (recommended) {
519
+ model = recommended;
520
+ if (defaultThinkingLevel) {
521
+ thinkingLevel = defaultThinkingLevel;
522
+ }
523
+ return { model, thinkingLevel, fallbackMessage: undefined };
519
524
  }
520
- return { model, thinkingLevel, fallbackMessage: undefined };
521
525
  }
526
+ model = found;
527
+ if (defaultThinkingLevel) {
528
+ thinkingLevel = defaultThinkingLevel;
529
+ }
530
+ return { model, thinkingLevel, fallbackMessage: undefined };
522
531
  }
523
- model = found;
524
- if (defaultThinkingLevel) {
525
- thinkingLevel = defaultThinkingLevel;
526
- }
527
- return { model, thinkingLevel, fallbackMessage: undefined };
528
532
  }
529
533
  }
530
534