@pixelbyte-software/pixcode 1.51.2 → 1.51.4

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 (331) hide show
  1. package/CODE_OF_CONDUCT.md +41 -41
  2. package/CONTRIBUTING.md +155 -155
  3. package/LICENSE +718 -718
  4. package/README.de.md +169 -169
  5. package/README.ja.md +167 -167
  6. package/README.ko.md +167 -167
  7. package/README.md +419 -419
  8. package/README.ru.md +169 -169
  9. package/README.tr.md +298 -298
  10. package/README.zh-CN.md +167 -167
  11. package/SECURITY.md +46 -46
  12. package/dist/api-automation.html +110 -110
  13. package/dist/api-docs.html +548 -548
  14. package/dist/assets/index-B9N-gfOQ.css +32 -0
  15. package/dist/assets/{index-EN9ngyxf.js → index-HfGHXhD6.js} +175 -175
  16. package/dist/clear-cache.html +85 -85
  17. package/dist/convert-icons.md +52 -52
  18. package/dist/docs.html +308 -308
  19. package/dist/favicon.svg +8 -8
  20. package/dist/features.html +133 -133
  21. package/dist/generate-icons.js +48 -48
  22. package/dist/humans.txt +15 -15
  23. package/dist/icons/codex-white.svg +3 -3
  24. package/dist/icons/codex.svg +3 -3
  25. package/dist/icons/cursor-white.svg +11 -11
  26. package/dist/icons/icon-128x128.svg +9 -9
  27. package/dist/icons/icon-144x144.svg +9 -9
  28. package/dist/icons/icon-152x152.svg +9 -9
  29. package/dist/icons/icon-192x192.svg +9 -9
  30. package/dist/icons/icon-384x384.svg +9 -9
  31. package/dist/icons/icon-512x512.svg +9 -9
  32. package/dist/icons/icon-72x72.svg +9 -9
  33. package/dist/icons/icon-96x96.svg +9 -9
  34. package/dist/icons/icon-template.svg +9 -9
  35. package/dist/icons/qwen-logo.svg +14 -14
  36. package/dist/index.html +59 -59
  37. package/dist/landing.html +268 -268
  38. package/dist/llms-full.txt +119 -119
  39. package/dist/llms.txt +53 -53
  40. package/dist/logo.svg +12 -12
  41. package/dist/manifest.json +60 -60
  42. package/dist/openapi.yaml +1696 -1696
  43. package/dist/orchestration.html +125 -125
  44. package/dist/robots.txt +4 -4
  45. package/dist/site.css +692 -692
  46. package/dist/sitemap.xml +51 -51
  47. package/dist/sw.js +132 -132
  48. package/dist-server/server/cli.js +96 -96
  49. package/dist-server/server/daemon/manager.js +33 -33
  50. package/dist-server/server/daemon-manager.js +64 -64
  51. package/dist-server/server/database/db.js +14 -2
  52. package/dist-server/server/database/db.js.map +1 -1
  53. package/dist-server/server/index.js +191 -31
  54. package/dist-server/server/index.js.map +1 -1
  55. package/dist-server/server/middleware/auth.js +16 -5
  56. package/dist-server/server/middleware/auth.js.map +1 -1
  57. package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.js +84 -0
  58. package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.js.map +1 -0
  59. package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.test.js +43 -0
  60. package/dist-server/server/modules/orchestration/a2a/adapters/json-event.adapter.test.js.map +1 -0
  61. package/dist-server/server/modules/orchestration/hermes/hermes.routes.js +55 -1
  62. package/dist-server/server/modules/orchestration/hermes/hermes.routes.js.map +1 -1
  63. package/dist-server/server/modules/orchestration/index.js +1 -0
  64. package/dist-server/server/modules/orchestration/index.js.map +1 -1
  65. package/dist-server/server/routes/auth.js +12 -5
  66. package/dist-server/server/routes/auth.js.map +1 -1
  67. package/dist-server/server/routes/commands.js +25 -25
  68. package/dist-server/server/routes/git.js +29 -17
  69. package/dist-server/server/routes/git.js.map +1 -1
  70. package/dist-server/server/routes/live-view.js +46 -46
  71. package/dist-server/server/routes/platformization.js +7 -6
  72. package/dist-server/server/routes/platformization.js.map +1 -1
  73. package/dist-server/server/services/hermes-gateway.js +310 -0
  74. package/dist-server/server/services/hermes-gateway.js.map +1 -1
  75. package/dist-server/server/services/platformization.js +58 -2
  76. package/dist-server/server/services/platformization.js.map +1 -1
  77. package/dist-server/server/services/public-api-manifest.js +59 -51
  78. package/dist-server/server/services/public-api-manifest.js.map +1 -1
  79. package/package.json +222 -222
  80. package/scripts/fix-node-pty.js +67 -67
  81. package/scripts/github/create-v1.38-issues.mjs +351 -351
  82. package/scripts/github/create-vscode-workbench-issues.mjs +121 -121
  83. package/scripts/hermes/configure-pixcode-mcp.mjs +165 -163
  84. package/scripts/hermes/pixcode-mcp-server.mjs +1009 -958
  85. package/scripts/smoke/changes-panel-layout.mjs +48 -48
  86. package/scripts/smoke/chat-composer-fixed-layout.mjs +55 -55
  87. package/scripts/smoke/chat-message-timeline-order.mjs +41 -41
  88. package/scripts/smoke/chat-realtime-hydration.mjs +44 -44
  89. package/scripts/smoke/chat-session-provider-pools.mjs +35 -35
  90. package/scripts/smoke/chat-session-state.mjs +19 -19
  91. package/scripts/smoke/code-editor-theme.mjs +55 -55
  92. package/scripts/smoke/code-editor-vscode-engine.mjs +91 -91
  93. package/scripts/smoke/command-center-agent-writes.mjs +79 -79
  94. package/scripts/smoke/command-center-non-git.mjs +46 -46
  95. package/scripts/smoke/context-packet.mjs +43 -43
  96. package/scripts/smoke/control-room-ux-redesign.mjs +91 -91
  97. package/scripts/smoke/daemon-entrypoint.mjs +20 -20
  98. package/scripts/smoke/default-landing-routing.mjs +33 -33
  99. package/scripts/smoke/desktop-native-notifications.mjs +30 -30
  100. package/scripts/smoke/desktop-tray-icon.mjs +33 -33
  101. package/scripts/smoke/discord-release-workflow.mjs +24 -24
  102. package/scripts/smoke/git-install-update.mjs +255 -255
  103. package/scripts/smoke/handoff-artifact-protocol.mjs +50 -50
  104. package/scripts/smoke/hermes-api-install.mjs +56 -56
  105. package/scripts/smoke/hermes-gateway-persistence.mjs +104 -104
  106. package/scripts/smoke/hermes-mcp-pixcode-roundtrip.mjs +426 -367
  107. package/scripts/smoke/hermes-rest-chat-api.mjs +162 -162
  108. package/scripts/smoke/hermes-rest-chat-live.mjs +45 -45
  109. package/scripts/smoke/hermes-rest-codex-launch.mjs +209 -209
  110. package/scripts/smoke/hermes-rest-gateway.mjs +79 -70
  111. package/scripts/smoke/hermes-rest-live.mjs +42 -42
  112. package/scripts/smoke/hermes-roundtrip.mjs +167 -167
  113. package/scripts/smoke/hermes-settings-commands.mjs +349 -346
  114. package/scripts/smoke/hermes-smoke-launcher-guard.mjs +34 -34
  115. package/scripts/smoke/live-view-diagnostics.mjs +53 -53
  116. package/scripts/smoke/live-view-environment.mjs +92 -92
  117. package/scripts/smoke/live-view-integration.mjs +450 -450
  118. package/scripts/smoke/mac-desktop-runtime.mjs +37 -37
  119. package/scripts/smoke/mobile-tunnel-guidance.mjs +29 -29
  120. package/scripts/smoke/model-registry.mjs +36 -36
  121. package/scripts/smoke/multi-project-ui.mjs +45 -45
  122. package/scripts/smoke/multi-worker-slots.mjs +42 -42
  123. package/scripts/smoke/notification-center.mjs +87 -87
  124. package/scripts/smoke/notification-inapp-preference.mjs +23 -23
  125. package/scripts/smoke/notification-taxonomy.mjs +58 -58
  126. package/scripts/smoke/orchestration-api.mjs +172 -172
  127. package/scripts/smoke/orchestration-execution-dashboard.mjs +33 -33
  128. package/scripts/smoke/orchestration-live-run.mjs +176 -176
  129. package/scripts/smoke/orchestration-mobile-scroll.mjs +29 -29
  130. package/scripts/smoke/orchestration-model-sync.mjs +30 -30
  131. package/scripts/smoke/orchestration-permission-fallback.mjs +34 -34
  132. package/scripts/smoke/orchestration-runtime-guards.mjs +48 -48
  133. package/scripts/smoke/orchestration-user-facing-output.mjs +25 -25
  134. package/scripts/smoke/permission-policy.mjs +50 -50
  135. package/scripts/smoke/pixcode-workbench-1-48.mjs +167 -167
  136. package/scripts/smoke/provider-models-opencode-live.mjs +66 -66
  137. package/scripts/smoke/provider-rest-api.mjs +124 -124
  138. package/scripts/smoke/provider-selection-status.mjs +52 -52
  139. package/scripts/smoke/run-state-refresh.mjs +52 -52
  140. package/scripts/smoke/runtime-manager.mjs +99 -99
  141. package/scripts/smoke/shell-manual-disconnect.mjs +30 -30
  142. package/scripts/smoke/side-panel-editor-layout.mjs +34 -34
  143. package/scripts/smoke/static-root-routing.mjs +21 -21
  144. package/scripts/smoke/strict-handoff-compact.mjs +60 -60
  145. package/scripts/smoke/taskmaster-config.mjs +24 -24
  146. package/scripts/smoke/taskmaster-execution-telegram.mjs +3 -3
  147. package/scripts/smoke/taskmaster-onboarding.mjs +3 -3
  148. package/scripts/smoke/taskmaster-run-graph.mjs +3 -3
  149. package/scripts/smoke/telegram-control.mjs +242 -242
  150. package/scripts/smoke/tunnel-persistence.mjs +56 -56
  151. package/scripts/smoke/update-issue-progress.mjs +69 -69
  152. package/scripts/smoke/update-ux.mjs +55 -55
  153. package/scripts/smoke/v138-completion.mjs +132 -132
  154. package/scripts/smoke/v138-desktop-release-hardening.mjs +69 -69
  155. package/scripts/smoke/v138-diagnostics.mjs +63 -63
  156. package/scripts/smoke/v138-issue-planner.mjs +33 -33
  157. package/scripts/smoke/v143-remote-control.mjs +76 -76
  158. package/scripts/smoke/v144-production-loop.mjs +47 -47
  159. package/scripts/smoke/v145-platformization.mjs +46 -46
  160. package/scripts/smoke/v146-control-room-ui.mjs +150 -150
  161. package/scripts/smoke/version-modal-autoshow.mjs +29 -29
  162. package/scripts/smoke/vscode-workbench-layout.mjs +63 -63
  163. package/scripts/smoke/vscode-workbench-polish.mjs +461 -436
  164. package/scripts/smoke/workflow-fallback-replay.mjs +56 -56
  165. package/scripts/smoke/workflow-templates.mjs +43 -43
  166. package/scripts/smoke/workflow-trace-timeline.mjs +46 -46
  167. package/scripts/update-git-install.mjs +293 -293
  168. package/server/claude-sdk.js +920 -920
  169. package/server/cli.js +1039 -1039
  170. package/server/constants/config.js +4 -4
  171. package/server/cursor-cli.js +344 -344
  172. package/server/daemon/manager.js +563 -563
  173. package/server/daemon-manager.js +964 -964
  174. package/server/database/db.js +908 -895
  175. package/server/database/json-store.js +197 -197
  176. package/server/gemini-cli.js +550 -550
  177. package/server/gemini-response-handler.js +79 -79
  178. package/server/index.js +201 -30
  179. package/server/load-env.js +35 -35
  180. package/server/middleware/auth.js +171 -156
  181. package/server/modules/orchestration/a2a/adapter-registry.ts +108 -108
  182. package/server/modules/orchestration/a2a/adapters/abstract-a2a.adapter.ts +63 -63
  183. package/server/modules/orchestration/a2a/adapters/claude-code.adapter.ts +286 -286
  184. package/server/modules/orchestration/a2a/adapters/codex.adapter.ts +244 -244
  185. package/server/modules/orchestration/a2a/adapters/cursor.adapter.ts +249 -249
  186. package/server/modules/orchestration/a2a/adapters/gemini.adapter.ts +248 -248
  187. package/server/modules/orchestration/a2a/adapters/json-event.adapter.test.ts +60 -0
  188. package/server/modules/orchestration/a2a/adapters/json-event.adapter.ts +101 -0
  189. package/server/modules/orchestration/a2a/adapters/opencode.adapter.ts +248 -248
  190. package/server/modules/orchestration/a2a/adapters/qwen.adapter.ts +248 -248
  191. package/server/modules/orchestration/a2a/agent-card.ts +55 -55
  192. package/server/modules/orchestration/a2a/routes.ts +590 -590
  193. package/server/modules/orchestration/a2a/task-store.ts +178 -178
  194. package/server/modules/orchestration/a2a/types.ts +126 -126
  195. package/server/modules/orchestration/a2a/validator.ts +113 -113
  196. package/server/modules/orchestration/hermes/hermes.routes.ts +642 -583
  197. package/server/modules/orchestration/index.ts +101 -100
  198. package/server/modules/orchestration/preview/port-watcher.ts +112 -112
  199. package/server/modules/orchestration/preview/preview-proxy.ts +60 -60
  200. package/server/modules/orchestration/preview/types.ts +19 -19
  201. package/server/modules/orchestration/security/permission-policy.ts +401 -401
  202. package/server/modules/orchestration/tasks/orchestration-task-store.ts +41 -41
  203. package/server/modules/orchestration/tasks/orchestration-task.routes.ts +64 -64
  204. package/server/modules/orchestration/tasks/orchestration-task.service.ts +209 -209
  205. package/server/modules/orchestration/tasks/orchestration-task.types.ts +40 -40
  206. package/server/modules/orchestration/tasks/task-run-graph.ts +155 -155
  207. package/server/modules/orchestration/workflows/approval-queue.ts +106 -106
  208. package/server/modules/orchestration/workflows/built-in-workflows.ts +127 -127
  209. package/server/modules/orchestration/workflows/context-packet.ts +186 -186
  210. package/server/modules/orchestration/workflows/handoff-artifact.ts +175 -175
  211. package/server/modules/orchestration/workflows/workflow-fallback-policy.ts +161 -161
  212. package/server/modules/orchestration/workflows/workflow-replay.ts +254 -254
  213. package/server/modules/orchestration/workflows/workflow-runner.ts +2070 -2070
  214. package/server/modules/orchestration/workflows/workflow-store.ts +97 -97
  215. package/server/modules/orchestration/workflows/workflow-templates.ts +272 -272
  216. package/server/modules/orchestration/workflows/workflow-trace.ts +424 -424
  217. package/server/modules/orchestration/workflows/workflow.routes.ts +586 -586
  218. package/server/modules/orchestration/workflows/workflow.types.ts +111 -111
  219. package/server/modules/orchestration/workflows/workspace-target.ts +122 -122
  220. package/server/modules/orchestration/workspace/docker-workspace.ts +136 -136
  221. package/server/modules/orchestration/workspace/path-safety.ts +55 -55
  222. package/server/modules/orchestration/workspace/types.ts +52 -52
  223. package/server/modules/orchestration/workspace/workspace-manager.ts +102 -102
  224. package/server/modules/orchestration/workspace/worktree-workspace.ts +126 -126
  225. package/server/modules/providers/index.ts +2 -2
  226. package/server/modules/providers/list/claude/claude-auth.provider.ts +146 -146
  227. package/server/modules/providers/list/claude/claude-mcp.provider.ts +135 -135
  228. package/server/modules/providers/list/claude/claude-sessions.provider.ts +306 -306
  229. package/server/modules/providers/list/claude/claude.provider.ts +15 -15
  230. package/server/modules/providers/list/codex/codex-auth.provider.ts +117 -117
  231. package/server/modules/providers/list/codex/codex-mcp.provider.ts +135 -135
  232. package/server/modules/providers/list/codex/codex-sessions.provider.ts +319 -319
  233. package/server/modules/providers/list/codex/codex.provider.ts +15 -15
  234. package/server/modules/providers/list/cursor/cursor-auth.provider.ts +147 -147
  235. package/server/modules/providers/list/cursor/cursor-mcp.provider.ts +108 -108
  236. package/server/modules/providers/list/cursor/cursor-sessions.provider.ts +421 -421
  237. package/server/modules/providers/list/cursor/cursor.provider.ts +15 -15
  238. package/server/modules/providers/list/gemini/gemini-auth.provider.ts +173 -173
  239. package/server/modules/providers/list/gemini/gemini-mcp.provider.ts +110 -110
  240. package/server/modules/providers/list/gemini/gemini-sessions.provider.ts +227 -227
  241. package/server/modules/providers/list/gemini/gemini.provider.ts +15 -15
  242. package/server/modules/providers/list/opencode/opencode-auth.provider.ts +131 -131
  243. package/server/modules/providers/list/opencode/opencode-mcp.provider.ts +126 -126
  244. package/server/modules/providers/list/opencode/opencode-sessions.provider.ts +286 -286
  245. package/server/modules/providers/list/opencode/opencode.provider.ts +29 -29
  246. package/server/modules/providers/list/qwen/qwen-auth.provider.ts +146 -146
  247. package/server/modules/providers/list/qwen/qwen-mcp.provider.ts +114 -114
  248. package/server/modules/providers/list/qwen/qwen-sessions.provider.ts +265 -265
  249. package/server/modules/providers/list/qwen/qwen.provider.ts +21 -21
  250. package/server/modules/providers/provider.registry.ts +40 -40
  251. package/server/modules/providers/provider.routes.ts +944 -944
  252. package/server/modules/providers/services/mcp.service.ts +86 -86
  253. package/server/modules/providers/services/provider-auth.service.ts +26 -26
  254. package/server/modules/providers/services/sessions.service.ts +45 -45
  255. package/server/modules/providers/shared/base/abstract.provider.ts +20 -20
  256. package/server/modules/providers/shared/mcp/mcp.provider.ts +151 -151
  257. package/server/modules/providers/shared/provider-configs.ts +142 -142
  258. package/server/modules/providers/tests/mcp.test.ts +293 -293
  259. package/server/openai-codex.js +462 -462
  260. package/server/opencode-cli.js +491 -491
  261. package/server/opencode-response-handler.js +111 -111
  262. package/server/projects.js +3008 -3008
  263. package/server/qwen-code-cli.js +410 -410
  264. package/server/qwen-response-handler.js +73 -73
  265. package/server/routes/agent.js +1435 -1435
  266. package/server/routes/auth.js +154 -146
  267. package/server/routes/codex.js +20 -20
  268. package/server/routes/commands.js +570 -570
  269. package/server/routes/cursor.js +61 -61
  270. package/server/routes/diagnostics.js +41 -41
  271. package/server/routes/gemini.js +25 -25
  272. package/server/routes/git.js +1650 -1635
  273. package/server/routes/live-view.js +411 -411
  274. package/server/routes/mcp-utils.js +13 -13
  275. package/server/routes/messages.js +62 -62
  276. package/server/routes/network.js +125 -125
  277. package/server/routes/platformization.js +198 -197
  278. package/server/routes/plugins.js +320 -320
  279. package/server/routes/production-agent-loop.js +90 -90
  280. package/server/routes/projects.js +917 -917
  281. package/server/routes/public-api.js +34 -34
  282. package/server/routes/qwen.js +27 -27
  283. package/server/routes/remote.js +55 -55
  284. package/server/routes/settings.js +321 -321
  285. package/server/routes/telegram.js +140 -140
  286. package/server/routes/user.js +125 -125
  287. package/server/routes/webhooks.js +63 -63
  288. package/server/services/control-room.js +102 -102
  289. package/server/services/diagnostics.js +165 -165
  290. package/server/services/external-access.js +375 -375
  291. package/server/services/hermes-gateway.js +1562 -1247
  292. package/server/services/hermes-install-jobs.js +729 -729
  293. package/server/services/install-jobs.js +715 -715
  294. package/server/services/live-view.js +956 -956
  295. package/server/services/managed-runtimes.js +493 -493
  296. package/server/services/model-registry.js +144 -144
  297. package/server/services/notification-orchestrator.js +365 -365
  298. package/server/services/notification-taxonomy.js +204 -204
  299. package/server/services/platformization.js +844 -779
  300. package/server/services/production-agent-loop.js +248 -248
  301. package/server/services/provider-cli-versions.js +149 -149
  302. package/server/services/provider-credentials.js +189 -189
  303. package/server/services/provider-models.js +396 -396
  304. package/server/services/public-api-manifest.js +190 -182
  305. package/server/services/remote-connection.js +127 -127
  306. package/server/services/runtime-manager.js +323 -323
  307. package/server/services/startup-update.js +234 -234
  308. package/server/services/telegram/bot.js +331 -331
  309. package/server/services/telegram/control-center.js +979 -979
  310. package/server/services/telegram/telegram-http-client.js +151 -151
  311. package/server/services/telegram/translations.js +340 -340
  312. package/server/services/vapid-keys.js +36 -36
  313. package/server/services/webhooks.js +216 -216
  314. package/server/sessionManager.js +225 -225
  315. package/server/shared/interfaces.ts +54 -54
  316. package/server/shared/types.ts +172 -172
  317. package/server/shared/utils.ts +193 -193
  318. package/server/tsconfig.json +36 -36
  319. package/server/utils/colors.js +21 -21
  320. package/server/utils/commandParser.js +305 -305
  321. package/server/utils/frontmatter.js +18 -18
  322. package/server/utils/gitConfig.js +34 -34
  323. package/server/utils/plugin-loader.js +457 -457
  324. package/server/utils/plugin-process-manager.js +185 -185
  325. package/server/utils/port-access.js +209 -209
  326. package/server/utils/runtime-paths.js +37 -37
  327. package/server/utils/url-detection.js +71 -71
  328. package/server/vite-daemon.js +79 -79
  329. package/shared/modelConstants.js +161 -161
  330. package/shared/networkHosts.js +22 -22
  331. package/dist/assets/index-DMz0zv6T.css +0 -32
@@ -1,396 +1,396 @@
1
- /**
2
- * Provider model catalog — dynamic discovery + persistent cache.
3
- *
4
- * Each CLI provider offers a different way to list the models it can run:
5
- *
6
- * - Anthropic: https://api.anthropic.com/v1/models (API key required)
7
- * - OpenAI : https://api.openai.com/v1/models (or a custom baseUrl when
8
- * the user points Codex at an OpenAI-compatible proxy)
9
- * - Google : https://generativelanguage.googleapis.com/v1beta/models
10
- * - Qwen : same OpenAI-compat shape when users BYOK through
11
- * ModelStudio / ModelScope / OpenRouter etc.
12
- *
13
- * The UI still keeps a hardcoded "known good" catalog as a baseline so
14
- * brand-new installs don't show an empty dropdown; discovery merges the
15
- * server-reported list on top, dedupes, and persists to
16
- * `~/.pixcode/provider-models.json` with a 6-hour freshness window.
17
- */
18
- import { promises as fs } from 'node:fs';
19
- import os from 'node:os';
20
- import path from 'node:path';
21
-
22
- import { getProviderCredentials } from './provider-credentials.js';
23
-
24
- const CACHE_FILE = path.join(os.homedir(), '.pixcode', 'provider-models.json');
25
- const CACHE_TTL_MS = 6 * 60 * 60 * 1000; // 6 hours
26
-
27
- async function readCache() {
28
- try {
29
- const raw = await fs.readFile(CACHE_FILE, 'utf8');
30
- const parsed = JSON.parse(raw);
31
- return parsed && typeof parsed === 'object' ? parsed : {};
32
- } catch {
33
- return {};
34
- }
35
- }
36
-
37
- async function writeCache(next) {
38
- await fs.mkdir(path.dirname(CACHE_FILE), { recursive: true });
39
- await fs.writeFile(CACHE_FILE, JSON.stringify(next, null, 2), { mode: 0o600 });
40
- }
41
-
42
- /**
43
- * Cache entry shape:
44
- * { models: [{ value, label, source: 'static' | 'api' }],
45
- * fetchedAt: '<iso date>',
46
- * error?: '...' }
47
- *
48
- * `source` tells the UI which entries came from the live API so a
49
- * refresh can prune stale ones without losing the hand-maintained
50
- * defaults.
51
- */
52
- async function loadCachedEntry(provider) {
53
- const cache = await readCache();
54
- return cache[provider] || null;
55
- }
56
-
57
- async function saveCacheEntry(provider, entry) {
58
- const cache = await readCache();
59
- cache[provider] = { ...entry, fetchedAt: new Date().toISOString() };
60
- await writeCache(cache);
61
- }
62
-
63
- function normalizeList(list) {
64
- if (!Array.isArray(list)) return [];
65
- const seen = new Set();
66
- const out = [];
67
- for (const item of list) {
68
- if (!item || typeof item !== 'object') continue;
69
- const value = typeof item.value === 'string' ? item.value.trim() : '';
70
- if (!value || seen.has(value)) continue;
71
- seen.add(value);
72
- const label = typeof item.label === 'string' && item.label.trim() ? item.label.trim() : value;
73
- const source = item.source === 'api' ? 'api' : 'static';
74
- const entry = { value, label, source };
75
- if (typeof item.free === 'boolean') entry.free = item.free;
76
- out.push(entry);
77
- }
78
- return out;
79
- }
80
-
81
- function mergeCatalogs(primary, secondary) {
82
- const seen = new Map();
83
- for (const item of [...primary, ...secondary]) {
84
- if (!seen.has(item.value)) seen.set(item.value, item);
85
- }
86
- return Array.from(seen.values());
87
- }
88
-
89
- function mergeProviderCatalogs(provider, primary, staticCatalog) {
90
- const normalizedPrimary = normalizeList(primary);
91
-
92
- // OpenCode Zen free models rotate often. When models.dev succeeds, treat
93
- // that live catalog as authoritative; otherwise stale static freebies can
94
- // leak back into the UI and fail later with ProviderModelNotFoundError.
95
- if (provider === 'opencode') {
96
- const liveModels = normalizedPrimary.filter((item) => item.source === 'api');
97
- if (liveModels.length > 0) return liveModels;
98
- }
99
-
100
- return mergeCatalogs(normalizedPrimary, staticCatalog);
101
- }
102
-
103
- // ---------------- Per-provider live discovery ----------------
104
-
105
- async function discoverAnthropic(apiKey, baseUrl) {
106
- const endpoint = (baseUrl?.replace(/\/+$/, '') || 'https://api.anthropic.com') + '/v1/models';
107
- const response = await fetch(endpoint, {
108
- headers: {
109
- 'x-api-key': apiKey,
110
- 'anthropic-version': '2023-06-01',
111
- },
112
- });
113
- if (!response.ok) throw new Error(`Anthropic /v1/models returned ${response.status}`);
114
- const data = await response.json();
115
- const rows = Array.isArray(data?.data) ? data.data : [];
116
- return rows
117
- .filter((m) => typeof m?.id === 'string')
118
- .map((m) => ({
119
- value: m.id,
120
- label: typeof m.display_name === 'string' && m.display_name.trim() ? m.display_name : m.id,
121
- source: 'api',
122
- }));
123
- }
124
-
125
- async function discoverOpenAiCompat(apiKey, baseUrl, fallbackBase) {
126
- const endpoint = (baseUrl?.replace(/\/+$/, '') || fallbackBase) + '/models';
127
- const response = await fetch(endpoint, {
128
- headers: { Authorization: `Bearer ${apiKey}` },
129
- });
130
- if (!response.ok) {
131
- // 401 specifically means our key is bad — but for codex/qwen/etc.
132
- // users often log in via OAuth (`codex login`, `qwen auth`) which
133
- // doesn't expose an OpenAI-compatible API key. Surface a clean
134
- // "no live discovery available" rather than a scary 401 trace.
135
- if (response.status === 401) {
136
- const err = new Error('OpenAI-compatible /v1/models requires an API key. The static catalog is shown instead — that\'s expected when you signed in via OAuth (e.g. `codex login`).');
137
- err.code = 'OAUTH_NO_API_KEY';
138
- throw err;
139
- }
140
- throw new Error(`${endpoint} returned ${response.status}`);
141
- }
142
- const data = await response.json();
143
- const rows = Array.isArray(data?.data) ? data.data : [];
144
- return rows
145
- .filter((m) => typeof m?.id === 'string')
146
- .map((m) => ({
147
- value: m.id,
148
- label: m.id,
149
- source: 'api',
150
- }));
151
- }
152
-
153
- /**
154
- * Detect whether the user is authenticated via the provider's OAuth flow
155
- * (codex login / qwen auth) so we can skip live model discovery silently
156
- * — those flows don't surface a usable OpenAI-compatible API key, and the
157
- * SDK calls the upstream APIs through its own internal auth path.
158
- */
159
- async function hasProviderOauthAuth(provider) {
160
- if (provider === 'codex') {
161
- try {
162
- await fs.access(path.join(os.homedir(), '.codex', 'auth.json'));
163
- return true;
164
- } catch { return false; }
165
- }
166
- if (provider === 'qwen') {
167
- try {
168
- await fs.access(path.join(os.homedir(), '.qwen', 'oauth_creds.json'));
169
- return true;
170
- } catch { return false; }
171
- }
172
- return false;
173
- }
174
-
175
- /**
176
- * OpenCode is multi-provider — its "model" picker isn't a single API list,
177
- * it's the union of every provider it can route to (Anthropic, OpenAI,
178
- * Google, xAI, OpenRouter, OpenCode Zen, Ollama, etc.). The canonical
179
- * catalog lives at https://models.dev/api.json (no auth, ~1.8 MB JSON, 115
180
- * providers as of 2026-04). We pull that, filter to providers the user
181
- * has authenticated with (read `~/.local/share/opencode/auth.json`) plus
182
- * always include the OpenCode Zen tier (works without explicit auth on
183
- * the free models), drop deprecated entries, and tag free models.
184
- */
185
- async function discoverOpencode() {
186
- const url = process.env.OPENCODE_MODELS_URL || 'https://models.dev/api.json';
187
- const response = await fetch(url, {
188
- // OpenCode itself caches this for hours; we cache for 6h via the
189
- // outer wrapper so a single 7s fetch on cold start is acceptable.
190
- signal: AbortSignal.timeout(15000),
191
- });
192
- if (!response.ok) throw new Error(`models.dev/api.json returned ${response.status}`);
193
- const data = await response.json();
194
- if (!data || typeof data !== 'object') throw new Error('models.dev returned a non-object payload');
195
-
196
- // Read OpenCode's auth.json to know which providers the user can
197
- // actually call. Missing file → only show always-free Zen.
198
- const authedProviders = new Set(['opencode']);
199
- try {
200
- const authPath = path.join(os.homedir(), '.local', 'share', 'opencode', 'auth.json');
201
- const raw = await fs.readFile(authPath, 'utf8');
202
- const auth = JSON.parse(raw);
203
- if (auth && typeof auth === 'object') {
204
- for (const k of Object.keys(auth)) authedProviders.add(k);
205
- }
206
- } catch { /* no auth.json → only Zen free models surface */ }
207
-
208
- // Common env-var providers OpenCode picks up automatically. If the user
209
- // exported one in their shell, surface those models too even without
210
- // auth.json. Mirrors the env list in opencode-auth.provider.ts.
211
- const envProviderHints = {
212
- anthropic: ['ANTHROPIC_API_KEY'],
213
- openai: ['OPENAI_API_KEY'],
214
- google: ['GOOGLE_GENERATIVE_AI_API_KEY', 'GEMINI_API_KEY'],
215
- 'google-vertex': ['GOOGLE_APPLICATION_CREDENTIALS'],
216
- xai: ['XAI_API_KEY'],
217
- groq: ['GROQ_API_KEY'],
218
- cerebras: ['CEREBRAS_API_KEY'],
219
- openrouter: ['OPENROUTER_API_KEY'],
220
- };
221
- for (const [providerId, envVars] of Object.entries(envProviderHints)) {
222
- if (envVars.some((v) => process.env[v]?.trim())) authedProviders.add(providerId);
223
- }
224
-
225
- const out = [];
226
- for (const [providerId, providerCfg] of Object.entries(data)) {
227
- if (!authedProviders.has(providerId)) continue;
228
- if (!providerCfg || typeof providerCfg !== 'object') continue;
229
- const models = providerCfg.models;
230
- if (!models || typeof models !== 'object') continue;
231
-
232
- const providerName = typeof providerCfg.name === 'string' && providerCfg.name.trim()
233
- ? providerCfg.name
234
- : providerId;
235
-
236
- for (const [modelId, modelCfg] of Object.entries(models)) {
237
- if (!modelCfg || typeof modelCfg !== 'object') continue;
238
- // Skip deprecated entries from the default list — users can
239
- // still hand-type them if they really need to.
240
- if (modelCfg.status === 'deprecated') continue;
241
- const cost = modelCfg.cost && typeof modelCfg.cost === 'object' ? modelCfg.cost : null;
242
- const free = !cost || (Number(cost.input) === 0 && Number(cost.output) === 0);
243
- const ctx = modelCfg.limit?.context;
244
- const ctxLabel = typeof ctx === 'number' && ctx > 0
245
- ? ` · ${ctx >= 1_000_000 ? `${(ctx / 1_000_000).toFixed(1)}M` : `${Math.round(ctx / 1000)}K`}`
246
- : '';
247
- const freeLabel = free ? ' · Free' : '';
248
- const modelName = typeof modelCfg.name === 'string' && modelCfg.name.trim()
249
- ? modelCfg.name
250
- : modelId;
251
-
252
- out.push({
253
- value: `${providerId}/${modelId}`,
254
- label: `${providerName} · ${modelName}${ctxLabel}${freeLabel}`,
255
- source: 'api',
256
- free,
257
- });
258
- }
259
- }
260
-
261
- // Sort: free first (handy when the user is unauthed), then by label.
262
- out.sort((a, b) => {
263
- if (a.free !== b.free) return a.free ? -1 : 1;
264
- return a.label.localeCompare(b.label);
265
- });
266
-
267
- return out;
268
- }
269
-
270
- async function discoverGoogle(apiKey) {
271
- // Google Generative Language API — public models list, API key as query.
272
- const endpoint = `https://generativelanguage.googleapis.com/v1beta/models?key=${encodeURIComponent(apiKey)}`;
273
- const response = await fetch(endpoint);
274
- if (!response.ok) throw new Error(`Google /v1beta/models returned ${response.status}`);
275
- const data = await response.json();
276
- const rows = Array.isArray(data?.models) ? data.models : [];
277
- return rows
278
- .filter((m) => typeof m?.name === 'string' && m.name.includes('models/'))
279
- .map((m) => {
280
- const id = m.name.replace(/^models\//, '');
281
- return {
282
- value: id,
283
- label: typeof m.displayName === 'string' && m.displayName.trim() ? m.displayName : id,
284
- source: 'api',
285
- };
286
- });
287
- }
288
-
289
- /**
290
- * Returns the merged catalog for a provider.
291
- * opts.forceRefresh: ignore cache and hit the upstream API
292
- * opts.staticList: hardcoded fallback from shared/modelConstants.js
293
- */
294
- export async function getProviderModels(provider, opts = {}) {
295
- const { forceRefresh = false, staticList = [] } = opts;
296
- const staticCatalog = normalizeList(staticList.map((m) => ({ ...m, source: 'static' })));
297
-
298
- const cached = await loadCachedEntry(provider);
299
- const cacheFresh = cached?.fetchedAt
300
- ? Date.now() - Date.parse(cached.fetchedAt) < CACHE_TTL_MS
301
- : false;
302
-
303
- if (!forceRefresh && cacheFresh && Array.isArray(cached?.models)) {
304
- const merged = mergeProviderCatalogs(provider, cached.models, staticCatalog);
305
- return {
306
- models: merged,
307
- fetchedAt: cached.fetchedAt,
308
- error: cached.error,
309
- fromCache: true,
310
- };
311
- }
312
-
313
- // OpenCode is the odd one out: its catalog is models.dev, not a per-key
314
- // API endpoint. Skip the credential plumbing and dispatch straight.
315
- let liveModels = [];
316
- let error;
317
- if (provider === 'opencode') {
318
- try {
319
- liveModels = await discoverOpencode();
320
- } catch (err) {
321
- error = err?.message || String(err);
322
- }
323
- const merged = mergeProviderCatalogs(provider, liveModels, staticCatalog);
324
- const entry = { models: merged, error };
325
- await saveCacheEntry(provider, entry).catch(() => { /* non-fatal */ });
326
- return { models: merged, fetchedAt: new Date().toISOString(), error, fromCache: false };
327
- }
328
-
329
- // Pick up credentials from Pixcode's UI store first, then fall back to
330
- // the native env vars so a user who already exported ANTHROPIC_API_KEY
331
- // (or authenticated Claude Code via OAuth — the SDK writes the key into
332
- // process.env on boot) gets live models without re-entering anything.
333
- const creds = await getProviderCredentials(provider);
334
- const envKey = {
335
- claude: process.env.ANTHROPIC_API_KEY,
336
- codex: process.env.OPENAI_API_KEY,
337
- qwen: process.env.OPENAI_API_KEY || process.env.QWEN_API_KEY,
338
- gemini: process.env.GEMINI_API_KEY,
339
- }[provider];
340
- const envBase = {
341
- claude: process.env.ANTHROPIC_BASE_URL,
342
- codex: process.env.OPENAI_BASE_URL,
343
- qwen: process.env.OPENAI_BASE_URL,
344
- gemini: undefined,
345
- }[provider];
346
- const apiKey = creds?.apiKey || envKey;
347
- const baseUrl = creds?.baseUrl || envBase || undefined;
348
-
349
- if (!apiKey) {
350
- // Codex and Qwen support OAuth (`codex login`, `qwen auth`) which
351
- // DOESN'T expose a usable API key — the SDK auths against the
352
- // upstream API directly. Skip the discovery step silently in that
353
- // case; the static catalog is the right answer.
354
- const oauthOnly = await hasProviderOauthAuth(provider);
355
- if (!oauthOnly) {
356
- // Be explicit so the UI can surface a useful hint rather than just
357
- // showing the static baseline with no reason given.
358
- error = `No ${provider} API key configured. Save one in Settings > Agents > API Key, or sign in via the CLI (e.g. \`codex login\`).`;
359
- }
360
- } else {
361
- try {
362
- if (provider === 'claude') {
363
- liveModels = await discoverAnthropic(apiKey, baseUrl);
364
- } else if (provider === 'codex') {
365
- liveModels = await discoverOpenAiCompat(apiKey, baseUrl, 'https://api.openai.com/v1');
366
- } else if (provider === 'qwen') {
367
- liveModels = await discoverOpenAiCompat(
368
- apiKey,
369
- baseUrl,
370
- 'https://dashscope-intl.aliyuncs.com/compatible-mode/v1',
371
- );
372
- } else if (provider === 'gemini') {
373
- liveModels = await discoverGoogle(apiKey);
374
- }
375
- } catch (err) {
376
- // OAuth users get a clean message instead of a raw 401 stack.
377
- if (err?.code === 'OAUTH_NO_API_KEY') {
378
- error = err.message;
379
- } else {
380
- error = err?.message || String(err);
381
- }
382
- }
383
- }
384
-
385
- const merged = mergeProviderCatalogs(provider, liveModels, staticCatalog);
386
- const entry = { models: merged, error };
387
- await saveCacheEntry(provider, entry).catch(() => { /* non-fatal */ });
388
- return { models: merged, fetchedAt: new Date().toISOString(), error, fromCache: false };
389
- }
390
-
391
- export async function clearProviderModelCache(provider) {
392
- const cache = await readCache();
393
- if (provider) delete cache[provider];
394
- else Object.keys(cache).forEach((k) => delete cache[k]);
395
- await writeCache(cache);
396
- }
1
+ /**
2
+ * Provider model catalog — dynamic discovery + persistent cache.
3
+ *
4
+ * Each CLI provider offers a different way to list the models it can run:
5
+ *
6
+ * - Anthropic: https://api.anthropic.com/v1/models (API key required)
7
+ * - OpenAI : https://api.openai.com/v1/models (or a custom baseUrl when
8
+ * the user points Codex at an OpenAI-compatible proxy)
9
+ * - Google : https://generativelanguage.googleapis.com/v1beta/models
10
+ * - Qwen : same OpenAI-compat shape when users BYOK through
11
+ * ModelStudio / ModelScope / OpenRouter etc.
12
+ *
13
+ * The UI still keeps a hardcoded "known good" catalog as a baseline so
14
+ * brand-new installs don't show an empty dropdown; discovery merges the
15
+ * server-reported list on top, dedupes, and persists to
16
+ * `~/.pixcode/provider-models.json` with a 6-hour freshness window.
17
+ */
18
+ import { promises as fs } from 'node:fs';
19
+ import os from 'node:os';
20
+ import path from 'node:path';
21
+
22
+ import { getProviderCredentials } from './provider-credentials.js';
23
+
24
+ const CACHE_FILE = path.join(os.homedir(), '.pixcode', 'provider-models.json');
25
+ const CACHE_TTL_MS = 6 * 60 * 60 * 1000; // 6 hours
26
+
27
+ async function readCache() {
28
+ try {
29
+ const raw = await fs.readFile(CACHE_FILE, 'utf8');
30
+ const parsed = JSON.parse(raw);
31
+ return parsed && typeof parsed === 'object' ? parsed : {};
32
+ } catch {
33
+ return {};
34
+ }
35
+ }
36
+
37
+ async function writeCache(next) {
38
+ await fs.mkdir(path.dirname(CACHE_FILE), { recursive: true });
39
+ await fs.writeFile(CACHE_FILE, JSON.stringify(next, null, 2), { mode: 0o600 });
40
+ }
41
+
42
+ /**
43
+ * Cache entry shape:
44
+ * { models: [{ value, label, source: 'static' | 'api' }],
45
+ * fetchedAt: '<iso date>',
46
+ * error?: '...' }
47
+ *
48
+ * `source` tells the UI which entries came from the live API so a
49
+ * refresh can prune stale ones without losing the hand-maintained
50
+ * defaults.
51
+ */
52
+ async function loadCachedEntry(provider) {
53
+ const cache = await readCache();
54
+ return cache[provider] || null;
55
+ }
56
+
57
+ async function saveCacheEntry(provider, entry) {
58
+ const cache = await readCache();
59
+ cache[provider] = { ...entry, fetchedAt: new Date().toISOString() };
60
+ await writeCache(cache);
61
+ }
62
+
63
+ function normalizeList(list) {
64
+ if (!Array.isArray(list)) return [];
65
+ const seen = new Set();
66
+ const out = [];
67
+ for (const item of list) {
68
+ if (!item || typeof item !== 'object') continue;
69
+ const value = typeof item.value === 'string' ? item.value.trim() : '';
70
+ if (!value || seen.has(value)) continue;
71
+ seen.add(value);
72
+ const label = typeof item.label === 'string' && item.label.trim() ? item.label.trim() : value;
73
+ const source = item.source === 'api' ? 'api' : 'static';
74
+ const entry = { value, label, source };
75
+ if (typeof item.free === 'boolean') entry.free = item.free;
76
+ out.push(entry);
77
+ }
78
+ return out;
79
+ }
80
+
81
+ function mergeCatalogs(primary, secondary) {
82
+ const seen = new Map();
83
+ for (const item of [...primary, ...secondary]) {
84
+ if (!seen.has(item.value)) seen.set(item.value, item);
85
+ }
86
+ return Array.from(seen.values());
87
+ }
88
+
89
+ function mergeProviderCatalogs(provider, primary, staticCatalog) {
90
+ const normalizedPrimary = normalizeList(primary);
91
+
92
+ // OpenCode Zen free models rotate often. When models.dev succeeds, treat
93
+ // that live catalog as authoritative; otherwise stale static freebies can
94
+ // leak back into the UI and fail later with ProviderModelNotFoundError.
95
+ if (provider === 'opencode') {
96
+ const liveModels = normalizedPrimary.filter((item) => item.source === 'api');
97
+ if (liveModels.length > 0) return liveModels;
98
+ }
99
+
100
+ return mergeCatalogs(normalizedPrimary, staticCatalog);
101
+ }
102
+
103
+ // ---------------- Per-provider live discovery ----------------
104
+
105
+ async function discoverAnthropic(apiKey, baseUrl) {
106
+ const endpoint = (baseUrl?.replace(/\/+$/, '') || 'https://api.anthropic.com') + '/v1/models';
107
+ const response = await fetch(endpoint, {
108
+ headers: {
109
+ 'x-api-key': apiKey,
110
+ 'anthropic-version': '2023-06-01',
111
+ },
112
+ });
113
+ if (!response.ok) throw new Error(`Anthropic /v1/models returned ${response.status}`);
114
+ const data = await response.json();
115
+ const rows = Array.isArray(data?.data) ? data.data : [];
116
+ return rows
117
+ .filter((m) => typeof m?.id === 'string')
118
+ .map((m) => ({
119
+ value: m.id,
120
+ label: typeof m.display_name === 'string' && m.display_name.trim() ? m.display_name : m.id,
121
+ source: 'api',
122
+ }));
123
+ }
124
+
125
+ async function discoverOpenAiCompat(apiKey, baseUrl, fallbackBase) {
126
+ const endpoint = (baseUrl?.replace(/\/+$/, '') || fallbackBase) + '/models';
127
+ const response = await fetch(endpoint, {
128
+ headers: { Authorization: `Bearer ${apiKey}` },
129
+ });
130
+ if (!response.ok) {
131
+ // 401 specifically means our key is bad — but for codex/qwen/etc.
132
+ // users often log in via OAuth (`codex login`, `qwen auth`) which
133
+ // doesn't expose an OpenAI-compatible API key. Surface a clean
134
+ // "no live discovery available" rather than a scary 401 trace.
135
+ if (response.status === 401) {
136
+ const err = new Error('OpenAI-compatible /v1/models requires an API key. The static catalog is shown instead — that\'s expected when you signed in via OAuth (e.g. `codex login`).');
137
+ err.code = 'OAUTH_NO_API_KEY';
138
+ throw err;
139
+ }
140
+ throw new Error(`${endpoint} returned ${response.status}`);
141
+ }
142
+ const data = await response.json();
143
+ const rows = Array.isArray(data?.data) ? data.data : [];
144
+ return rows
145
+ .filter((m) => typeof m?.id === 'string')
146
+ .map((m) => ({
147
+ value: m.id,
148
+ label: m.id,
149
+ source: 'api',
150
+ }));
151
+ }
152
+
153
+ /**
154
+ * Detect whether the user is authenticated via the provider's OAuth flow
155
+ * (codex login / qwen auth) so we can skip live model discovery silently
156
+ * — those flows don't surface a usable OpenAI-compatible API key, and the
157
+ * SDK calls the upstream APIs through its own internal auth path.
158
+ */
159
+ async function hasProviderOauthAuth(provider) {
160
+ if (provider === 'codex') {
161
+ try {
162
+ await fs.access(path.join(os.homedir(), '.codex', 'auth.json'));
163
+ return true;
164
+ } catch { return false; }
165
+ }
166
+ if (provider === 'qwen') {
167
+ try {
168
+ await fs.access(path.join(os.homedir(), '.qwen', 'oauth_creds.json'));
169
+ return true;
170
+ } catch { return false; }
171
+ }
172
+ return false;
173
+ }
174
+
175
+ /**
176
+ * OpenCode is multi-provider — its "model" picker isn't a single API list,
177
+ * it's the union of every provider it can route to (Anthropic, OpenAI,
178
+ * Google, xAI, OpenRouter, OpenCode Zen, Ollama, etc.). The canonical
179
+ * catalog lives at https://models.dev/api.json (no auth, ~1.8 MB JSON, 115
180
+ * providers as of 2026-04). We pull that, filter to providers the user
181
+ * has authenticated with (read `~/.local/share/opencode/auth.json`) plus
182
+ * always include the OpenCode Zen tier (works without explicit auth on
183
+ * the free models), drop deprecated entries, and tag free models.
184
+ */
185
+ async function discoverOpencode() {
186
+ const url = process.env.OPENCODE_MODELS_URL || 'https://models.dev/api.json';
187
+ const response = await fetch(url, {
188
+ // OpenCode itself caches this for hours; we cache for 6h via the
189
+ // outer wrapper so a single 7s fetch on cold start is acceptable.
190
+ signal: AbortSignal.timeout(15000),
191
+ });
192
+ if (!response.ok) throw new Error(`models.dev/api.json returned ${response.status}`);
193
+ const data = await response.json();
194
+ if (!data || typeof data !== 'object') throw new Error('models.dev returned a non-object payload');
195
+
196
+ // Read OpenCode's auth.json to know which providers the user can
197
+ // actually call. Missing file → only show always-free Zen.
198
+ const authedProviders = new Set(['opencode']);
199
+ try {
200
+ const authPath = path.join(os.homedir(), '.local', 'share', 'opencode', 'auth.json');
201
+ const raw = await fs.readFile(authPath, 'utf8');
202
+ const auth = JSON.parse(raw);
203
+ if (auth && typeof auth === 'object') {
204
+ for (const k of Object.keys(auth)) authedProviders.add(k);
205
+ }
206
+ } catch { /* no auth.json → only Zen free models surface */ }
207
+
208
+ // Common env-var providers OpenCode picks up automatically. If the user
209
+ // exported one in their shell, surface those models too even without
210
+ // auth.json. Mirrors the env list in opencode-auth.provider.ts.
211
+ const envProviderHints = {
212
+ anthropic: ['ANTHROPIC_API_KEY'],
213
+ openai: ['OPENAI_API_KEY'],
214
+ google: ['GOOGLE_GENERATIVE_AI_API_KEY', 'GEMINI_API_KEY'],
215
+ 'google-vertex': ['GOOGLE_APPLICATION_CREDENTIALS'],
216
+ xai: ['XAI_API_KEY'],
217
+ groq: ['GROQ_API_KEY'],
218
+ cerebras: ['CEREBRAS_API_KEY'],
219
+ openrouter: ['OPENROUTER_API_KEY'],
220
+ };
221
+ for (const [providerId, envVars] of Object.entries(envProviderHints)) {
222
+ if (envVars.some((v) => process.env[v]?.trim())) authedProviders.add(providerId);
223
+ }
224
+
225
+ const out = [];
226
+ for (const [providerId, providerCfg] of Object.entries(data)) {
227
+ if (!authedProviders.has(providerId)) continue;
228
+ if (!providerCfg || typeof providerCfg !== 'object') continue;
229
+ const models = providerCfg.models;
230
+ if (!models || typeof models !== 'object') continue;
231
+
232
+ const providerName = typeof providerCfg.name === 'string' && providerCfg.name.trim()
233
+ ? providerCfg.name
234
+ : providerId;
235
+
236
+ for (const [modelId, modelCfg] of Object.entries(models)) {
237
+ if (!modelCfg || typeof modelCfg !== 'object') continue;
238
+ // Skip deprecated entries from the default list — users can
239
+ // still hand-type them if they really need to.
240
+ if (modelCfg.status === 'deprecated') continue;
241
+ const cost = modelCfg.cost && typeof modelCfg.cost === 'object' ? modelCfg.cost : null;
242
+ const free = !cost || (Number(cost.input) === 0 && Number(cost.output) === 0);
243
+ const ctx = modelCfg.limit?.context;
244
+ const ctxLabel = typeof ctx === 'number' && ctx > 0
245
+ ? ` · ${ctx >= 1_000_000 ? `${(ctx / 1_000_000).toFixed(1)}M` : `${Math.round(ctx / 1000)}K`}`
246
+ : '';
247
+ const freeLabel = free ? ' · Free' : '';
248
+ const modelName = typeof modelCfg.name === 'string' && modelCfg.name.trim()
249
+ ? modelCfg.name
250
+ : modelId;
251
+
252
+ out.push({
253
+ value: `${providerId}/${modelId}`,
254
+ label: `${providerName} · ${modelName}${ctxLabel}${freeLabel}`,
255
+ source: 'api',
256
+ free,
257
+ });
258
+ }
259
+ }
260
+
261
+ // Sort: free first (handy when the user is unauthed), then by label.
262
+ out.sort((a, b) => {
263
+ if (a.free !== b.free) return a.free ? -1 : 1;
264
+ return a.label.localeCompare(b.label);
265
+ });
266
+
267
+ return out;
268
+ }
269
+
270
+ async function discoverGoogle(apiKey) {
271
+ // Google Generative Language API — public models list, API key as query.
272
+ const endpoint = `https://generativelanguage.googleapis.com/v1beta/models?key=${encodeURIComponent(apiKey)}`;
273
+ const response = await fetch(endpoint);
274
+ if (!response.ok) throw new Error(`Google /v1beta/models returned ${response.status}`);
275
+ const data = await response.json();
276
+ const rows = Array.isArray(data?.models) ? data.models : [];
277
+ return rows
278
+ .filter((m) => typeof m?.name === 'string' && m.name.includes('models/'))
279
+ .map((m) => {
280
+ const id = m.name.replace(/^models\//, '');
281
+ return {
282
+ value: id,
283
+ label: typeof m.displayName === 'string' && m.displayName.trim() ? m.displayName : id,
284
+ source: 'api',
285
+ };
286
+ });
287
+ }
288
+
289
+ /**
290
+ * Returns the merged catalog for a provider.
291
+ * opts.forceRefresh: ignore cache and hit the upstream API
292
+ * opts.staticList: hardcoded fallback from shared/modelConstants.js
293
+ */
294
+ export async function getProviderModels(provider, opts = {}) {
295
+ const { forceRefresh = false, staticList = [] } = opts;
296
+ const staticCatalog = normalizeList(staticList.map((m) => ({ ...m, source: 'static' })));
297
+
298
+ const cached = await loadCachedEntry(provider);
299
+ const cacheFresh = cached?.fetchedAt
300
+ ? Date.now() - Date.parse(cached.fetchedAt) < CACHE_TTL_MS
301
+ : false;
302
+
303
+ if (!forceRefresh && cacheFresh && Array.isArray(cached?.models)) {
304
+ const merged = mergeProviderCatalogs(provider, cached.models, staticCatalog);
305
+ return {
306
+ models: merged,
307
+ fetchedAt: cached.fetchedAt,
308
+ error: cached.error,
309
+ fromCache: true,
310
+ };
311
+ }
312
+
313
+ // OpenCode is the odd one out: its catalog is models.dev, not a per-key
314
+ // API endpoint. Skip the credential plumbing and dispatch straight.
315
+ let liveModels = [];
316
+ let error;
317
+ if (provider === 'opencode') {
318
+ try {
319
+ liveModels = await discoverOpencode();
320
+ } catch (err) {
321
+ error = err?.message || String(err);
322
+ }
323
+ const merged = mergeProviderCatalogs(provider, liveModels, staticCatalog);
324
+ const entry = { models: merged, error };
325
+ await saveCacheEntry(provider, entry).catch(() => { /* non-fatal */ });
326
+ return { models: merged, fetchedAt: new Date().toISOString(), error, fromCache: false };
327
+ }
328
+
329
+ // Pick up credentials from Pixcode's UI store first, then fall back to
330
+ // the native env vars so a user who already exported ANTHROPIC_API_KEY
331
+ // (or authenticated Claude Code via OAuth — the SDK writes the key into
332
+ // process.env on boot) gets live models without re-entering anything.
333
+ const creds = await getProviderCredentials(provider);
334
+ const envKey = {
335
+ claude: process.env.ANTHROPIC_API_KEY,
336
+ codex: process.env.OPENAI_API_KEY,
337
+ qwen: process.env.OPENAI_API_KEY || process.env.QWEN_API_KEY,
338
+ gemini: process.env.GEMINI_API_KEY,
339
+ }[provider];
340
+ const envBase = {
341
+ claude: process.env.ANTHROPIC_BASE_URL,
342
+ codex: process.env.OPENAI_BASE_URL,
343
+ qwen: process.env.OPENAI_BASE_URL,
344
+ gemini: undefined,
345
+ }[provider];
346
+ const apiKey = creds?.apiKey || envKey;
347
+ const baseUrl = creds?.baseUrl || envBase || undefined;
348
+
349
+ if (!apiKey) {
350
+ // Codex and Qwen support OAuth (`codex login`, `qwen auth`) which
351
+ // DOESN'T expose a usable API key — the SDK auths against the
352
+ // upstream API directly. Skip the discovery step silently in that
353
+ // case; the static catalog is the right answer.
354
+ const oauthOnly = await hasProviderOauthAuth(provider);
355
+ if (!oauthOnly) {
356
+ // Be explicit so the UI can surface a useful hint rather than just
357
+ // showing the static baseline with no reason given.
358
+ error = `No ${provider} API key configured. Save one in Settings > Agents > API Key, or sign in via the CLI (e.g. \`codex login\`).`;
359
+ }
360
+ } else {
361
+ try {
362
+ if (provider === 'claude') {
363
+ liveModels = await discoverAnthropic(apiKey, baseUrl);
364
+ } else if (provider === 'codex') {
365
+ liveModels = await discoverOpenAiCompat(apiKey, baseUrl, 'https://api.openai.com/v1');
366
+ } else if (provider === 'qwen') {
367
+ liveModels = await discoverOpenAiCompat(
368
+ apiKey,
369
+ baseUrl,
370
+ 'https://dashscope-intl.aliyuncs.com/compatible-mode/v1',
371
+ );
372
+ } else if (provider === 'gemini') {
373
+ liveModels = await discoverGoogle(apiKey);
374
+ }
375
+ } catch (err) {
376
+ // OAuth users get a clean message instead of a raw 401 stack.
377
+ if (err?.code === 'OAUTH_NO_API_KEY') {
378
+ error = err.message;
379
+ } else {
380
+ error = err?.message || String(err);
381
+ }
382
+ }
383
+ }
384
+
385
+ const merged = mergeProviderCatalogs(provider, liveModels, staticCatalog);
386
+ const entry = { models: merged, error };
387
+ await saveCacheEntry(provider, entry).catch(() => { /* non-fatal */ });
388
+ return { models: merged, fetchedAt: new Date().toISOString(), error, fromCache: false };
389
+ }
390
+
391
+ export async function clearProviderModelCache(provider) {
392
+ const cache = await readCache();
393
+ if (provider) delete cache[provider];
394
+ else Object.keys(cache).forEach((k) => delete cache[k]);
395
+ await writeCache(cache);
396
+ }