@gguf/coder 0.2.9 → 0.3.1

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 (439) hide show
  1. package/.editorconfig +16 -0
  2. package/.env.example +63 -0
  3. package/.gitattributes +1 -0
  4. package/.semgrepignore +19 -0
  5. package/coder-dummy-file.ts +52 -0
  6. package/coder.config.example.json +59 -0
  7. package/coder.config.json +13 -0
  8. package/color_picker.html +36 -0
  9. package/dist/tools/execute-bash.js +3 -3
  10. package/dist/tools/execute-bash.js.map +1 -1
  11. package/dist/tools/fetch-url.js +3 -3
  12. package/dist/tools/fetch-url.js.map +1 -1
  13. package/dist/tools/find-files.d.ts.map +1 -1
  14. package/dist/tools/find-files.js +1 -1
  15. package/dist/tools/find-files.js.map +1 -1
  16. package/dist/tools/lsp-get-diagnostics.js +1 -1
  17. package/dist/tools/lsp-get-diagnostics.js.map +1 -1
  18. package/dist/tools/read-file.d.ts.map +1 -1
  19. package/dist/tools/read-file.js +6 -6
  20. package/dist/tools/read-file.js.map +1 -1
  21. package/dist/tools/search-file-contents.d.ts.map +1 -1
  22. package/dist/tools/search-file-contents.js +1 -1
  23. package/dist/tools/search-file-contents.js.map +1 -1
  24. package/dist/tools/string-replace.js +11 -11
  25. package/dist/tools/string-replace.js.map +1 -1
  26. package/dist/tools/web-search.d.ts.map +1 -1
  27. package/dist/tools/web-search.js +3 -3
  28. package/dist/tools/web-search.js.map +1 -1
  29. package/dist/tools/write-file.js +4 -4
  30. package/dist/tools/write-file.js.map +1 -1
  31. package/dist/utils/tool-result-display.d.ts.map +1 -1
  32. package/dist/utils/tool-result-display.js +3 -3
  33. package/dist/utils/tool-result-display.js.map +1 -1
  34. package/package.json +2 -14
  35. package/scripts/extract-changelog.js +73 -0
  36. package/scripts/fetch-models.js +143 -0
  37. package/scripts/test.sh +40 -0
  38. package/scripts/update-homebrew-formula.sh +125 -0
  39. package/scripts/update-nix-version.sh +157 -0
  40. package/source/ai-sdk-client/AISDKClient.spec.ts +117 -0
  41. package/source/ai-sdk-client/AISDKClient.ts +155 -0
  42. package/source/ai-sdk-client/chat/chat-handler.spec.ts +121 -0
  43. package/source/ai-sdk-client/chat/chat-handler.ts +276 -0
  44. package/source/ai-sdk-client/chat/streaming-handler.spec.ts +173 -0
  45. package/source/ai-sdk-client/chat/streaming-handler.ts +110 -0
  46. package/source/ai-sdk-client/chat/tool-processor.spec.ts +92 -0
  47. package/source/ai-sdk-client/chat/tool-processor.ts +70 -0
  48. package/source/ai-sdk-client/converters/message-converter.spec.ts +220 -0
  49. package/source/ai-sdk-client/converters/message-converter.ts +113 -0
  50. package/source/ai-sdk-client/converters/tool-converter.spec.ts +90 -0
  51. package/source/ai-sdk-client/converters/tool-converter.ts +46 -0
  52. package/source/ai-sdk-client/error-handling/error-extractor.spec.ts +55 -0
  53. package/source/ai-sdk-client/error-handling/error-extractor.ts +15 -0
  54. package/source/ai-sdk-client/error-handling/error-parser.spec.ts +169 -0
  55. package/source/ai-sdk-client/error-handling/error-parser.ts +161 -0
  56. package/source/ai-sdk-client/index.ts +7 -0
  57. package/source/ai-sdk-client/providers/provider-factory.spec.ts +71 -0
  58. package/source/ai-sdk-client/providers/provider-factory.ts +41 -0
  59. package/source/ai-sdk-client/types.ts +9 -0
  60. package/source/ai-sdk-client-empty-message.spec.ts +141 -0
  61. package/source/ai-sdk-client-error-handling.spec.ts +186 -0
  62. package/source/ai-sdk-client-maxretries.spec.ts +114 -0
  63. package/source/ai-sdk-client-preparestep.spec.ts +279 -0
  64. package/source/app/App.spec.tsx +32 -0
  65. package/source/app/App.tsx +480 -0
  66. package/source/app/components/AppContainer.spec.tsx +96 -0
  67. package/source/app/components/AppContainer.tsx +56 -0
  68. package/source/app/components/ChatInterface.spec.tsx +163 -0
  69. package/source/app/components/ChatInterface.tsx +144 -0
  70. package/source/app/components/ModalSelectors.spec.tsx +141 -0
  71. package/source/app/components/ModalSelectors.tsx +135 -0
  72. package/source/app/helpers.spec.ts +97 -0
  73. package/source/app/helpers.ts +63 -0
  74. package/source/app/index.ts +4 -0
  75. package/source/app/types.ts +39 -0
  76. package/source/app/utils/appUtils.ts +294 -0
  77. package/source/app/utils/conversationState.ts +310 -0
  78. package/source/app.spec.tsx +244 -0
  79. package/source/cli.spec.ts +73 -0
  80. package/source/cli.tsx +51 -0
  81. package/source/client-factory.spec.ts +48 -0
  82. package/source/client-factory.ts +178 -0
  83. package/source/command-parser.spec.ts +127 -0
  84. package/source/command-parser.ts +36 -0
  85. package/source/commands/checkpoint.spec.tsx +277 -0
  86. package/source/commands/checkpoint.tsx +366 -0
  87. package/source/commands/clear.tsx +22 -0
  88. package/source/commands/custom-commands.tsx +121 -0
  89. package/source/commands/exit.ts +21 -0
  90. package/source/commands/export.spec.tsx +131 -0
  91. package/source/commands/export.tsx +79 -0
  92. package/source/commands/help.tsx +120 -0
  93. package/source/commands/index.ts +17 -0
  94. package/source/commands/init.tsx +339 -0
  95. package/source/commands/lsp-command.spec.tsx +281 -0
  96. package/source/commands/lsp.tsx +120 -0
  97. package/source/commands/mcp-command.spec.tsx +313 -0
  98. package/source/commands/mcp.tsx +162 -0
  99. package/source/commands/model-database.spec.tsx +758 -0
  100. package/source/commands/model-database.tsx +418 -0
  101. package/source/commands/model.ts +12 -0
  102. package/source/commands/provider.ts +12 -0
  103. package/source/commands/setup-config.tsx +16 -0
  104. package/source/commands/simple-commands.spec.tsx +175 -0
  105. package/source/commands/status.ts +12 -0
  106. package/source/commands/theme.ts +12 -0
  107. package/source/commands/update.spec.tsx +261 -0
  108. package/source/commands/update.tsx +201 -0
  109. package/source/commands/usage.spec.tsx +495 -0
  110. package/source/commands/usage.tsx +100 -0
  111. package/source/commands.spec.ts +436 -0
  112. package/source/commands.ts +83 -0
  113. package/source/components/assistant-message.spec.tsx +796 -0
  114. package/source/components/assistant-message.tsx +34 -0
  115. package/source/components/bash-execution-indicator.tsx +21 -0
  116. package/source/components/cancelling-indicator.tsx +16 -0
  117. package/source/components/chat-queue.spec.tsx +83 -0
  118. package/source/components/chat-queue.tsx +36 -0
  119. package/source/components/checkpoint-display.spec.tsx +219 -0
  120. package/source/components/checkpoint-display.tsx +126 -0
  121. package/source/components/checkpoint-selector.spec.tsx +173 -0
  122. package/source/components/checkpoint-selector.tsx +173 -0
  123. package/source/components/development-mode-indicator.spec.tsx +268 -0
  124. package/source/components/development-mode-indicator.tsx +38 -0
  125. package/source/components/message-box.spec.tsx +427 -0
  126. package/source/components/message-box.tsx +87 -0
  127. package/source/components/model-selector.tsx +132 -0
  128. package/source/components/provider-selector.tsx +75 -0
  129. package/source/components/random-spinner.tsx +19 -0
  130. package/source/components/security-disclaimer.tsx +73 -0
  131. package/source/components/status-connection-display.spec.tsx +133 -0
  132. package/source/components/status.tsx +267 -0
  133. package/source/components/theme-selector.tsx +126 -0
  134. package/source/components/tool-confirmation.tsx +190 -0
  135. package/source/components/tool-execution-indicator.tsx +33 -0
  136. package/source/components/tool-message.tsx +85 -0
  137. package/source/components/ui/titled-box.spec.tsx +207 -0
  138. package/source/components/ui/titled-box.tsx +57 -0
  139. package/source/components/usage/progress-bar.spec.tsx +398 -0
  140. package/source/components/usage/progress-bar.tsx +30 -0
  141. package/source/components/usage/usage-display.spec.tsx +780 -0
  142. package/source/components/usage/usage-display.tsx +291 -0
  143. package/source/components/user-input.spec.tsx +327 -0
  144. package/source/components/user-input.tsx +533 -0
  145. package/source/components/user-message.spec.tsx +230 -0
  146. package/source/components/user-message.tsx +84 -0
  147. package/source/components/welcome-message.tsx +76 -0
  148. package/source/config/env-substitution.ts +65 -0
  149. package/source/config/index.spec.ts +171 -0
  150. package/source/config/index.ts +154 -0
  151. package/source/config/paths.spec.ts +241 -0
  152. package/source/config/paths.ts +55 -0
  153. package/source/config/preferences.ts +51 -0
  154. package/source/config/themes.ts +315 -0
  155. package/source/constants.ts +130 -0
  156. package/source/context/mode-context.spec.ts +79 -0
  157. package/source/context/mode-context.ts +24 -0
  158. package/source/custom-commands/executor.spec.ts +142 -0
  159. package/source/custom-commands/executor.ts +64 -0
  160. package/source/custom-commands/loader.spec.ts +314 -0
  161. package/source/custom-commands/loader.ts +153 -0
  162. package/source/custom-commands/parser.ts +196 -0
  163. package/source/hooks/chat-handler/conversation/conversation-loop.spec.ts +39 -0
  164. package/source/hooks/chat-handler/conversation/conversation-loop.tsx +511 -0
  165. package/source/hooks/chat-handler/conversation/tool-executor.spec.ts +50 -0
  166. package/source/hooks/chat-handler/conversation/tool-executor.tsx +109 -0
  167. package/source/hooks/chat-handler/index.ts +12 -0
  168. package/source/hooks/chat-handler/state/streaming-state.spec.ts +26 -0
  169. package/source/hooks/chat-handler/state/streaming-state.ts +19 -0
  170. package/source/hooks/chat-handler/types.ts +38 -0
  171. package/source/hooks/chat-handler/useChatHandler.spec.tsx +321 -0
  172. package/source/hooks/chat-handler/useChatHandler.tsx +194 -0
  173. package/source/hooks/chat-handler/utils/context-checker.spec.ts +60 -0
  174. package/source/hooks/chat-handler/utils/context-checker.tsx +73 -0
  175. package/source/hooks/chat-handler/utils/message-helpers.spec.ts +42 -0
  176. package/source/hooks/chat-handler/utils/message-helpers.tsx +36 -0
  177. package/source/hooks/chat-handler/utils/tool-filters.spec.ts +109 -0
  178. package/source/hooks/chat-handler/utils/tool-filters.ts +64 -0
  179. package/source/hooks/useAppHandlers.tsx +291 -0
  180. package/source/hooks/useAppInitialization.tsx +422 -0
  181. package/source/hooks/useAppState.tsx +311 -0
  182. package/source/hooks/useDirectoryTrust.tsx +98 -0
  183. package/source/hooks/useInputState.ts +414 -0
  184. package/source/hooks/useModeHandlers.tsx +302 -0
  185. package/source/hooks/useNonInteractiveMode.ts +140 -0
  186. package/source/hooks/useTerminalWidth.tsx +81 -0
  187. package/source/hooks/useTheme.ts +18 -0
  188. package/source/hooks/useToolHandler.tsx +349 -0
  189. package/source/hooks/useUIState.ts +61 -0
  190. package/source/init/agents-template-generator.ts +421 -0
  191. package/source/init/existing-rules-extractor.ts +319 -0
  192. package/source/init/file-scanner.spec.ts +227 -0
  193. package/source/init/file-scanner.ts +238 -0
  194. package/source/init/framework-detector.ts +382 -0
  195. package/source/init/language-detector.ts +269 -0
  196. package/source/init/project-analyzer.spec.ts +231 -0
  197. package/source/init/project-analyzer.ts +458 -0
  198. package/source/lsp/index.ts +31 -0
  199. package/source/lsp/lsp-client.spec.ts +508 -0
  200. package/source/lsp/lsp-client.ts +487 -0
  201. package/source/lsp/lsp-manager.spec.ts +477 -0
  202. package/source/lsp/lsp-manager.ts +419 -0
  203. package/source/lsp/protocol.spec.ts +502 -0
  204. package/source/lsp/protocol.ts +360 -0
  205. package/source/lsp/server-discovery.spec.ts +654 -0
  206. package/source/lsp/server-discovery.ts +515 -0
  207. package/source/markdown-parser/html-entities.spec.ts +88 -0
  208. package/source/markdown-parser/html-entities.ts +45 -0
  209. package/source/markdown-parser/index.spec.ts +281 -0
  210. package/source/markdown-parser/index.ts +126 -0
  211. package/source/markdown-parser/table-parser.spec.ts +133 -0
  212. package/source/markdown-parser/table-parser.ts +114 -0
  213. package/source/markdown-parser/utils.spec.ts +70 -0
  214. package/source/markdown-parser/utils.ts +13 -0
  215. package/source/mcp/mcp-client.spec.ts +81 -0
  216. package/source/mcp/mcp-client.ts +625 -0
  217. package/source/mcp/transport-factory.spec.ts +406 -0
  218. package/source/mcp/transport-factory.ts +312 -0
  219. package/source/message-handler.ts +67 -0
  220. package/source/model-database/database-engine.spec.ts +494 -0
  221. package/source/model-database/database-engine.ts +50 -0
  222. package/source/model-database/model-database.spec.ts +363 -0
  223. package/source/model-database/model-database.ts +91 -0
  224. package/source/model-database/model-engine.spec.ts +447 -0
  225. package/source/model-database/model-engine.ts +65 -0
  226. package/source/model-database/model-fetcher.spec.ts +583 -0
  227. package/source/model-database/model-fetcher.ts +330 -0
  228. package/source/models/index.ts +1 -0
  229. package/source/models/models-cache.spec.ts +214 -0
  230. package/source/models/models-cache.ts +78 -0
  231. package/source/models/models-dev-client.spec.ts +379 -0
  232. package/source/models/models-dev-client.ts +329 -0
  233. package/source/models/models-types.ts +68 -0
  234. package/source/prompt-history.ts +155 -0
  235. package/source/security/command-injection.spec.ts +240 -0
  236. package/source/services/checkpoint-manager.spec.ts +523 -0
  237. package/source/services/checkpoint-manager.ts +466 -0
  238. package/source/services/file-snapshot.spec.ts +569 -0
  239. package/source/services/file-snapshot.ts +220 -0
  240. package/source/test-utils/render-with-theme.tsx +48 -0
  241. package/source/tokenization/index.ts +1 -0
  242. package/source/tokenization/tokenizer-factory.spec.ts +170 -0
  243. package/source/tokenization/tokenizer-factory.ts +125 -0
  244. package/source/tokenization/tokenizers/anthropic-tokenizer.spec.ts +200 -0
  245. package/source/tokenization/tokenizers/anthropic-tokenizer.ts +43 -0
  246. package/source/tokenization/tokenizers/fallback-tokenizer.spec.ts +236 -0
  247. package/source/tokenization/tokenizers/fallback-tokenizer.ts +26 -0
  248. package/source/tokenization/tokenizers/llama-tokenizer.spec.ts +224 -0
  249. package/source/tokenization/tokenizers/llama-tokenizer.ts +41 -0
  250. package/source/tokenization/tokenizers/openai-tokenizer.spec.ts +184 -0
  251. package/source/tokenization/tokenizers/openai-tokenizer.ts +57 -0
  252. package/source/tool-calling/index.ts +5 -0
  253. package/source/tool-calling/json-parser.spec.ts +639 -0
  254. package/source/tool-calling/json-parser.ts +247 -0
  255. package/source/tool-calling/tool-parser.spec.ts +395 -0
  256. package/source/tool-calling/tool-parser.ts +120 -0
  257. package/source/tool-calling/xml-parser.spec.ts +662 -0
  258. package/source/tool-calling/xml-parser.ts +289 -0
  259. package/source/tools/execute-bash.spec.tsx +353 -0
  260. package/source/tools/execute-bash.tsx +219 -0
  261. package/source/tools/execute-function.spec.ts +130 -0
  262. package/source/tools/fetch-url.spec.tsx +342 -0
  263. package/source/tools/fetch-url.tsx +172 -0
  264. package/source/tools/find-files.spec.tsx +924 -0
  265. package/source/tools/find-files.tsx +293 -0
  266. package/source/tools/index.ts +102 -0
  267. package/source/tools/lsp-get-diagnostics.tsx +192 -0
  268. package/source/tools/needs-approval.spec.ts +282 -0
  269. package/source/tools/read-file.spec.tsx +801 -0
  270. package/source/tools/read-file.tsx +387 -0
  271. package/source/tools/search-file-contents.spec.tsx +1273 -0
  272. package/source/tools/search-file-contents.tsx +293 -0
  273. package/source/tools/string-replace.spec.tsx +730 -0
  274. package/source/tools/string-replace.tsx +548 -0
  275. package/source/tools/tool-manager.ts +210 -0
  276. package/source/tools/tool-registry.spec.ts +415 -0
  277. package/source/tools/tool-registry.ts +228 -0
  278. package/source/tools/web-search.tsx +223 -0
  279. package/source/tools/write-file.spec.tsx +559 -0
  280. package/source/tools/write-file.tsx +228 -0
  281. package/source/types/app.ts +37 -0
  282. package/source/types/checkpoint.ts +48 -0
  283. package/source/types/commands.ts +46 -0
  284. package/source/types/components.ts +27 -0
  285. package/source/types/config.ts +103 -0
  286. package/source/types/core-connection-status.spec.ts +67 -0
  287. package/source/types/core.ts +181 -0
  288. package/source/types/hooks.ts +50 -0
  289. package/source/types/index.ts +12 -0
  290. package/source/types/markdown-parser.ts +11 -0
  291. package/source/types/mcp.ts +52 -0
  292. package/source/types/system.ts +16 -0
  293. package/source/types/tokenization.ts +41 -0
  294. package/source/types/ui.ts +40 -0
  295. package/source/types/usage.ts +58 -0
  296. package/source/types/utils.ts +16 -0
  297. package/source/usage/calculator.spec.ts +385 -0
  298. package/source/usage/calculator.ts +104 -0
  299. package/source/usage/storage.spec.ts +703 -0
  300. package/source/usage/storage.ts +238 -0
  301. package/source/usage/tracker.spec.ts +456 -0
  302. package/source/usage/tracker.ts +102 -0
  303. package/source/utils/atomic-deletion.spec.ts +194 -0
  304. package/source/utils/atomic-deletion.ts +127 -0
  305. package/source/utils/bounded-map.spec.ts +300 -0
  306. package/source/utils/bounded-map.ts +193 -0
  307. package/source/utils/checkpoint-utils.spec.ts +222 -0
  308. package/source/utils/checkpoint-utils.ts +92 -0
  309. package/source/utils/error-formatter.spec.ts +169 -0
  310. package/source/utils/error-formatter.ts +194 -0
  311. package/source/utils/file-autocomplete.spec.ts +173 -0
  312. package/source/utils/file-autocomplete.ts +196 -0
  313. package/source/utils/file-cache.spec.ts +309 -0
  314. package/source/utils/file-cache.ts +195 -0
  315. package/source/utils/file-content-loader.spec.ts +180 -0
  316. package/source/utils/file-content-loader.ts +179 -0
  317. package/source/utils/file-mention-handler.spec.ts +261 -0
  318. package/source/utils/file-mention-handler.ts +84 -0
  319. package/source/utils/file-mention-parser.spec.ts +182 -0
  320. package/source/utils/file-mention-parser.ts +170 -0
  321. package/source/utils/fuzzy-matching.spec.ts +149 -0
  322. package/source/utils/fuzzy-matching.ts +146 -0
  323. package/source/utils/indentation-normalizer.spec.ts +216 -0
  324. package/source/utils/indentation-normalizer.ts +76 -0
  325. package/source/utils/installation-detector.spec.ts +178 -0
  326. package/source/utils/installation-detector.ts +153 -0
  327. package/source/utils/logging/config.spec.ts +311 -0
  328. package/source/utils/logging/config.ts +210 -0
  329. package/source/utils/logging/console-facade.spec.ts +184 -0
  330. package/source/utils/logging/console-facade.ts +384 -0
  331. package/source/utils/logging/correlation.spec.ts +679 -0
  332. package/source/utils/logging/correlation.ts +474 -0
  333. package/source/utils/logging/formatters.spec.ts +464 -0
  334. package/source/utils/logging/formatters.ts +207 -0
  335. package/source/utils/logging/health-monitor/alerts/alert-manager.spec.ts +93 -0
  336. package/source/utils/logging/health-monitor/alerts/alert-manager.ts +79 -0
  337. package/source/utils/logging/health-monitor/checks/configuration-check.spec.ts +56 -0
  338. package/source/utils/logging/health-monitor/checks/configuration-check.ts +43 -0
  339. package/source/utils/logging/health-monitor/checks/logging-check.spec.ts +56 -0
  340. package/source/utils/logging/health-monitor/checks/logging-check.ts +58 -0
  341. package/source/utils/logging/health-monitor/checks/memory-check.spec.ts +100 -0
  342. package/source/utils/logging/health-monitor/checks/memory-check.ts +78 -0
  343. package/source/utils/logging/health-monitor/checks/performance-check.spec.ts +56 -0
  344. package/source/utils/logging/health-monitor/checks/performance-check.ts +56 -0
  345. package/source/utils/logging/health-monitor/checks/request-check.spec.ts +56 -0
  346. package/source/utils/logging/health-monitor/checks/request-check.ts +76 -0
  347. package/source/utils/logging/health-monitor/core/health-check-runner.spec.ts +70 -0
  348. package/source/utils/logging/health-monitor/core/health-check-runner.ts +138 -0
  349. package/source/utils/logging/health-monitor/core/health-monitor.spec.ts +58 -0
  350. package/source/utils/logging/health-monitor/core/health-monitor.ts +344 -0
  351. package/source/utils/logging/health-monitor/core/scoring.spec.ts +65 -0
  352. package/source/utils/logging/health-monitor/core/scoring.ts +91 -0
  353. package/source/utils/logging/health-monitor/index.ts +15 -0
  354. package/source/utils/logging/health-monitor/instances.ts +48 -0
  355. package/source/utils/logging/health-monitor/middleware/http-middleware.spec.ts +141 -0
  356. package/source/utils/logging/health-monitor/middleware/http-middleware.ts +75 -0
  357. package/source/utils/logging/health-monitor/types.ts +126 -0
  358. package/source/utils/logging/index.spec.ts +284 -0
  359. package/source/utils/logging/index.ts +236 -0
  360. package/source/utils/logging/integration.spec.ts +441 -0
  361. package/source/utils/logging/log-method-factory.spec.ts +573 -0
  362. package/source/utils/logging/log-method-factory.ts +233 -0
  363. package/source/utils/logging/log-query/aggregation/aggregator.spec.ts +277 -0
  364. package/source/utils/logging/log-query/aggregation/aggregator.ts +159 -0
  365. package/source/utils/logging/log-query/aggregation/facet-generator.spec.ts +159 -0
  366. package/source/utils/logging/log-query/aggregation/facet-generator.ts +47 -0
  367. package/source/utils/logging/log-query/index.ts +23 -0
  368. package/source/utils/logging/log-query/query/filter-predicates.spec.ts +247 -0
  369. package/source/utils/logging/log-query/query/filter-predicates.ts +154 -0
  370. package/source/utils/logging/log-query/query/query-builder.spec.ts +182 -0
  371. package/source/utils/logging/log-query/query/query-builder.ts +151 -0
  372. package/source/utils/logging/log-query/query/query-engine.spec.ts +214 -0
  373. package/source/utils/logging/log-query/query/query-engine.ts +45 -0
  374. package/source/utils/logging/log-query/storage/circular-buffer.spec.ts +143 -0
  375. package/source/utils/logging/log-query/storage/circular-buffer.ts +75 -0
  376. package/source/utils/logging/log-query/storage/index-manager.spec.ts +150 -0
  377. package/source/utils/logging/log-query/storage/index-manager.ts +71 -0
  378. package/source/utils/logging/log-query/storage/log-storage.spec.ts +257 -0
  379. package/source/utils/logging/log-query/storage/log-storage.ts +80 -0
  380. package/source/utils/logging/log-query/types.ts +163 -0
  381. package/source/utils/logging/log-query/utils/helpers.spec.ts +263 -0
  382. package/source/utils/logging/log-query/utils/helpers.ts +72 -0
  383. package/source/utils/logging/log-query/utils/sorting.spec.ts +182 -0
  384. package/source/utils/logging/log-query/utils/sorting.ts +61 -0
  385. package/source/utils/logging/logger-provider.spec.ts +262 -0
  386. package/source/utils/logging/logger-provider.ts +362 -0
  387. package/source/utils/logging/performance.spec.ts +209 -0
  388. package/source/utils/logging/performance.ts +757 -0
  389. package/source/utils/logging/pino-logger.spec.ts +425 -0
  390. package/source/utils/logging/pino-logger.ts +514 -0
  391. package/source/utils/logging/redaction.spec.ts +490 -0
  392. package/source/utils/logging/redaction.ts +267 -0
  393. package/source/utils/logging/request-tracker.spec.ts +1198 -0
  394. package/source/utils/logging/request-tracker.ts +803 -0
  395. package/source/utils/logging/transports.spec.ts +505 -0
  396. package/source/utils/logging/transports.ts +305 -0
  397. package/source/utils/logging/types.ts +216 -0
  398. package/source/utils/message-builder.spec.ts +179 -0
  399. package/source/utils/message-builder.ts +101 -0
  400. package/source/utils/message-queue.tsx +486 -0
  401. package/source/utils/paste-detection.spec.ts +69 -0
  402. package/source/utils/paste-detection.ts +124 -0
  403. package/source/utils/paste-roundtrip.spec.ts +442 -0
  404. package/source/utils/paste-utils.spec.ts +128 -0
  405. package/source/utils/paste-utils.ts +52 -0
  406. package/source/utils/programming-language-helper.spec.ts +74 -0
  407. package/source/utils/programming-language-helper.ts +32 -0
  408. package/source/utils/prompt-assembly.spec.ts +221 -0
  409. package/source/utils/prompt-processor.ts +173 -0
  410. package/source/utils/tool-args-parser.spec.ts +136 -0
  411. package/source/utils/tool-args-parser.ts +54 -0
  412. package/source/utils/tool-cancellation.spec.ts +230 -0
  413. package/source/utils/tool-cancellation.ts +28 -0
  414. package/source/utils/tool-result-display.spec.tsx +469 -0
  415. package/source/utils/tool-result-display.tsx +90 -0
  416. package/source/utils/update-checker.spec.ts +383 -0
  417. package/source/utils/update-checker.ts +183 -0
  418. package/source/wizard/config-wizard.spec.tsx +103 -0
  419. package/source/wizard/config-wizard.tsx +382 -0
  420. package/source/wizard/steps/location-step.spec.tsx +186 -0
  421. package/source/wizard/steps/location-step.tsx +147 -0
  422. package/source/wizard/steps/mcp-step.spec.tsx +607 -0
  423. package/source/wizard/steps/mcp-step.tsx +632 -0
  424. package/source/wizard/steps/provider-step.spec.tsx +342 -0
  425. package/source/wizard/steps/provider-step.tsx +957 -0
  426. package/source/wizard/steps/summary-step.spec.tsx +749 -0
  427. package/source/wizard/steps/summary-step.tsx +228 -0
  428. package/source/wizard/templates/mcp-templates.spec.ts +613 -0
  429. package/source/wizard/templates/mcp-templates.ts +570 -0
  430. package/source/wizard/templates/provider-templates.spec.ts +152 -0
  431. package/source/wizard/templates/provider-templates.ts +485 -0
  432. package/source/wizard/utils/fetch-cloud-models.spec.ts +428 -0
  433. package/source/wizard/utils/fetch-cloud-models.ts +223 -0
  434. package/source/wizard/utils/fetch-local-models.spec.ts +297 -0
  435. package/source/wizard/utils/fetch-local-models.ts +192 -0
  436. package/source/wizard/validation-array.spec.ts +264 -0
  437. package/source/wizard/validation.spec.ts +373 -0
  438. package/source/wizard/validation.ts +232 -0
  439. package/source/app/prompts/main-prompt.md +0 -122
@@ -0,0 +1,428 @@
1
+ import test from 'ava';
2
+ import {fetchCloudModels} from './fetch-cloud-models.js';
3
+
4
+ // Store original fetch for restoration
5
+ const originalFetch = globalThis.fetch;
6
+
7
+ // Helper to create standard cloud models response (Anthropic, OpenAI, Mistral)
8
+ function createCloudModelsResponse(
9
+ models: Array<{id: string; name?: string; display_name?: string}>,
10
+ ) {
11
+ return {data: models};
12
+ }
13
+
14
+ // Helper to create GitHub-style response (direct array)
15
+ function createGitHubResponse(models: Array<{id: string; name: string}>) {
16
+ return models;
17
+ }
18
+
19
+ // Helper to create a mock fetch that captures the request
20
+ function createMockFetch(
21
+ response: unknown,
22
+ options: {ok?: boolean; status?: number; statusText?: string} = {},
23
+ ) {
24
+ const {ok = true, status = 200, statusText = 'OK'} = options;
25
+ let capturedUrl: string | undefined;
26
+ let capturedHeaders: Record<string, string> | undefined;
27
+
28
+ const mockFetch = async (url: string | URL | Request, init?: RequestInit) => {
29
+ capturedUrl = url.toString();
30
+ capturedHeaders = init?.headers as Record<string, string>;
31
+ return {
32
+ ok,
33
+ status,
34
+ statusText,
35
+ json: async () => response,
36
+ } as Response;
37
+ };
38
+
39
+ return {
40
+ mockFetch,
41
+ getCapturedUrl: () => capturedUrl,
42
+ getCapturedHeaders: () => capturedHeaders,
43
+ };
44
+ }
45
+
46
+ // Anthropic tests
47
+ test.serial('fetchCloudModels: Anthropic - successful fetch returns models', async t => {
48
+ const response = createCloudModelsResponse([
49
+ {id: 'claude-3-opus', display_name: 'Claude 3 Opus'},
50
+ {id: 'claude-3-sonnet', display_name: 'Claude 3 Sonnet'},
51
+ ]);
52
+ const {mockFetch} = createMockFetch(response);
53
+ globalThis.fetch = mockFetch;
54
+
55
+ try {
56
+ const result = await fetchCloudModels('anthropic', 'sk-ant-test-key');
57
+ t.true(result.success);
58
+ t.is(result.models.length, 2);
59
+ t.is(result.error, undefined);
60
+ } finally {
61
+ globalThis.fetch = originalFetch;
62
+ }
63
+ });
64
+
65
+ test.serial('fetchCloudModels: Anthropic - uses correct endpoint and headers', async t => {
66
+ const {mockFetch, getCapturedUrl, getCapturedHeaders} = createMockFetch(
67
+ createCloudModelsResponse([{id: 'model1'}]),
68
+ );
69
+ globalThis.fetch = mockFetch;
70
+
71
+ try {
72
+ await fetchCloudModels('anthropic', 'sk-ant-test-key');
73
+ t.is(getCapturedUrl(), 'https://api.anthropic.com/v1/models');
74
+ t.is(getCapturedHeaders()?.['X-Api-Key'], 'sk-ant-test-key');
75
+ t.is(getCapturedHeaders()?.['anthropic-version'], '2023-06-01');
76
+ } finally {
77
+ globalThis.fetch = originalFetch;
78
+ }
79
+ });
80
+
81
+ test.serial('fetchCloudModels: Anthropic - uses display_name for model name', async t => {
82
+ const response = createCloudModelsResponse([
83
+ {id: 'claude-3-opus-20240229', display_name: 'Claude 3 Opus'},
84
+ ]);
85
+ const {mockFetch} = createMockFetch(response);
86
+ globalThis.fetch = mockFetch;
87
+
88
+ try {
89
+ const result = await fetchCloudModels('anthropic', 'sk-ant-test-key');
90
+ t.true(result.success);
91
+ t.is(result.models[0]?.id, 'claude-3-opus-20240229');
92
+ t.is(result.models[0]?.name, 'Claude 3 Opus');
93
+ } finally {
94
+ globalThis.fetch = originalFetch;
95
+ }
96
+ });
97
+
98
+ // OpenAI tests
99
+ test.serial('fetchCloudModels: OpenAI - successful fetch returns models', async t => {
100
+ const response = createCloudModelsResponse([
101
+ {id: 'gpt-4'},
102
+ {id: 'gpt-3.5-turbo'},
103
+ ]);
104
+ const {mockFetch} = createMockFetch(response);
105
+ globalThis.fetch = mockFetch;
106
+
107
+ try {
108
+ const result = await fetchCloudModels('openai', 'sk-test-key');
109
+ t.true(result.success);
110
+ t.is(result.models.length, 2);
111
+ } finally {
112
+ globalThis.fetch = originalFetch;
113
+ }
114
+ });
115
+
116
+ test.serial('fetchCloudModels: OpenAI - uses correct endpoint and Bearer auth', async t => {
117
+ const {mockFetch, getCapturedUrl, getCapturedHeaders} = createMockFetch(
118
+ createCloudModelsResponse([{id: 'model1'}]),
119
+ );
120
+ globalThis.fetch = mockFetch;
121
+
122
+ try {
123
+ await fetchCloudModels('openai', 'sk-test-key');
124
+ t.is(getCapturedUrl(), 'https://api.openai.com/v1/models');
125
+ t.is(getCapturedHeaders()?.Authorization, 'Bearer sk-test-key');
126
+ } finally {
127
+ globalThis.fetch = originalFetch;
128
+ }
129
+ });
130
+
131
+ // Mistral tests
132
+ test.serial('fetchCloudModels: Mistral - successful fetch returns models', async t => {
133
+ const response = createCloudModelsResponse([
134
+ {id: 'mistral-large'},
135
+ {id: 'mistral-small'},
136
+ ]);
137
+ const {mockFetch} = createMockFetch(response);
138
+ globalThis.fetch = mockFetch;
139
+
140
+ try {
141
+ const result = await fetchCloudModels('mistral', 'test-api-key');
142
+ t.true(result.success);
143
+ t.is(result.models.length, 2);
144
+ } finally {
145
+ globalThis.fetch = originalFetch;
146
+ }
147
+ });
148
+
149
+ test.serial('fetchCloudModels: Mistral - uses correct endpoint and Bearer auth', async t => {
150
+ const {mockFetch, getCapturedUrl, getCapturedHeaders} = createMockFetch(
151
+ createCloudModelsResponse([{id: 'model1'}]),
152
+ );
153
+ globalThis.fetch = mockFetch;
154
+
155
+ try {
156
+ await fetchCloudModels('mistral', 'test-api-key');
157
+ t.is(getCapturedUrl(), 'https://api.mistral.ai/v1/models');
158
+ t.is(getCapturedHeaders()?.Authorization, 'Bearer test-api-key');
159
+ } finally {
160
+ globalThis.fetch = originalFetch;
161
+ }
162
+ });
163
+
164
+ // GitHub tests
165
+ test.serial('fetchCloudModels: GitHub - successful fetch returns models', async t => {
166
+ const response = createGitHubResponse([
167
+ {id: 'gpt-4o', name: 'GPT-4o'},
168
+ {id: 'claude-3-5-sonnet', name: 'Claude 3.5 Sonnet'},
169
+ ]);
170
+ const {mockFetch} = createMockFetch(response);
171
+ globalThis.fetch = mockFetch;
172
+
173
+ try {
174
+ const result = await fetchCloudModels('github', 'ghp_test-token');
175
+ t.true(result.success);
176
+ t.is(result.models.length, 2);
177
+ } finally {
178
+ globalThis.fetch = originalFetch;
179
+ }
180
+ });
181
+
182
+ test.serial('fetchCloudModels: GitHub - uses correct endpoint and headers', async t => {
183
+ const {mockFetch, getCapturedUrl, getCapturedHeaders} = createMockFetch(
184
+ createGitHubResponse([{id: 'model1', name: 'Model 1'}]),
185
+ );
186
+ globalThis.fetch = mockFetch;
187
+
188
+ try {
189
+ await fetchCloudModels('github', 'ghp_test-token');
190
+ t.is(getCapturedUrl(), 'https://models.github.ai/catalog/models');
191
+ t.is(getCapturedHeaders()?.Authorization, 'Bearer ghp_test-token');
192
+ t.is(getCapturedHeaders()?.['X-GitHub-Api-Version'], '2022-11-28');
193
+ t.is(getCapturedHeaders()?.Accept, 'application/vnd.github+json');
194
+ } finally {
195
+ globalThis.fetch = originalFetch;
196
+ }
197
+ });
198
+
199
+ test.serial('fetchCloudModels: GitHub - uses name field for model name', async t => {
200
+ const response = createGitHubResponse([{id: 'gpt-4o', name: 'GPT-4o'}]);
201
+ const {mockFetch} = createMockFetch(response);
202
+ globalThis.fetch = mockFetch;
203
+
204
+ try {
205
+ const result = await fetchCloudModels('github', 'ghp_test-token');
206
+ t.true(result.success);
207
+ t.is(result.models[0]?.id, 'gpt-4o');
208
+ t.is(result.models[0]?.name, 'GPT-4o');
209
+ } finally {
210
+ globalThis.fetch = originalFetch;
211
+ }
212
+ });
213
+
214
+ // Input validation tests
215
+ test.serial('fetchCloudModels: returns error for empty API key', async t => {
216
+ const result = await fetchCloudModels('openai', '');
217
+ t.false(result.success);
218
+ t.is(result.models.length, 0);
219
+ t.is(result.error, 'API key is required');
220
+ });
221
+
222
+ test.serial('fetchCloudModels: returns error for whitespace-only API key', async t => {
223
+ const result = await fetchCloudModels('openai', ' ');
224
+ t.false(result.success);
225
+ t.is(result.models.length, 0);
226
+ t.is(result.error, 'API key is required');
227
+ });
228
+
229
+ test.serial('fetchCloudModels: returns error for unknown provider', async t => {
230
+ // @ts-expect-error Testing invalid provider type
231
+ const result = await fetchCloudModels('unknown-provider', 'api-key');
232
+ t.false(result.success);
233
+ t.is(result.models.length, 0);
234
+ t.is(result.error, 'Unknown cloud provider: unknown-provider');
235
+ });
236
+
237
+ // HTTP error tests
238
+ test.serial('fetchCloudModels: returns specific error for 401 Unauthorized', async t => {
239
+ const {mockFetch} = createMockFetch({}, {ok: false, status: 401, statusText: 'Unauthorized'});
240
+ globalThis.fetch = mockFetch;
241
+
242
+ try {
243
+ const result = await fetchCloudModels('openai', 'invalid-key');
244
+ t.false(result.success);
245
+ t.is(result.models.length, 0);
246
+ t.is(result.error, 'Invalid API key');
247
+ } finally {
248
+ globalThis.fetch = originalFetch;
249
+ }
250
+ });
251
+
252
+ test.serial('fetchCloudModels: returns specific error for 403 Forbidden', async t => {
253
+ const {mockFetch} = createMockFetch({}, {ok: false, status: 403, statusText: 'Forbidden'});
254
+ globalThis.fetch = mockFetch;
255
+
256
+ try {
257
+ const result = await fetchCloudModels('openai', 'restricted-key');
258
+ t.false(result.success);
259
+ t.is(result.models.length, 0);
260
+ t.is(result.error, 'API key does not have permission to list models');
261
+ } finally {
262
+ globalThis.fetch = originalFetch;
263
+ }
264
+ });
265
+
266
+ test.serial('fetchCloudModels: returns generic error for other HTTP errors', async t => {
267
+ const {mockFetch} = createMockFetch({}, {ok: false, status: 500, statusText: 'Internal Server Error'});
268
+ globalThis.fetch = mockFetch;
269
+
270
+ try {
271
+ const result = await fetchCloudModels('openai', 'api-key');
272
+ t.false(result.success);
273
+ t.is(result.models.length, 0);
274
+ t.is(result.error, 'API returned 500: Internal Server Error');
275
+ } finally {
276
+ globalThis.fetch = originalFetch;
277
+ }
278
+ });
279
+
280
+ // Response format validation tests
281
+ test.serial('fetchCloudModels: returns error for invalid response format (non-GitHub)', async t => {
282
+ const {mockFetch} = createMockFetch({invalid: 'response'}); // Missing data array
283
+ globalThis.fetch = mockFetch;
284
+
285
+ try {
286
+ const result = await fetchCloudModels('openai', 'api-key');
287
+ t.false(result.success);
288
+ t.is(result.models.length, 0);
289
+ t.is(result.error, 'Invalid response format from API');
290
+ } finally {
291
+ globalThis.fetch = originalFetch;
292
+ }
293
+ });
294
+
295
+ test.serial('fetchCloudModels: returns error for invalid GitHub response format', async t => {
296
+ const {mockFetch} = createMockFetch({data: []}); // GitHub expects array, not object
297
+ globalThis.fetch = mockFetch;
298
+
299
+ try {
300
+ const result = await fetchCloudModels('github', 'api-key');
301
+ t.false(result.success);
302
+ t.is(result.models.length, 0);
303
+ t.is(result.error, 'Invalid response format from API');
304
+ } finally {
305
+ globalThis.fetch = originalFetch;
306
+ }
307
+ });
308
+
309
+ test.serial('fetchCloudModels: returns error when no models found', async t => {
310
+ const {mockFetch} = createMockFetch(createCloudModelsResponse([]));
311
+ globalThis.fetch = mockFetch;
312
+
313
+ try {
314
+ const result = await fetchCloudModels('openai', 'api-key');
315
+ t.false(result.success);
316
+ t.is(result.models.length, 0);
317
+ t.is(result.error, 'No models found for this API key');
318
+ } finally {
319
+ globalThis.fetch = originalFetch;
320
+ }
321
+ });
322
+
323
+ // Network error tests
324
+ test.serial('fetchCloudModels: handles network error', async t => {
325
+ globalThis.fetch = async () => {
326
+ throw new Error('Network error');
327
+ };
328
+
329
+ try {
330
+ const result = await fetchCloudModels('openai', 'api-key');
331
+ t.false(result.success);
332
+ t.is(result.models.length, 0);
333
+ t.is(result.error, 'Network error');
334
+ } finally {
335
+ globalThis.fetch = originalFetch;
336
+ }
337
+ });
338
+
339
+ test.serial('fetchCloudModels: handles timeout (AbortError)', async t => {
340
+ globalThis.fetch = async () => {
341
+ const error = new Error('The operation was aborted');
342
+ error.name = 'AbortError';
343
+ throw error;
344
+ };
345
+
346
+ try {
347
+ const result = await fetchCloudModels('openai', 'api-key');
348
+ t.false(result.success);
349
+ t.is(result.models.length, 0);
350
+ t.is(result.error, 'Connection timed out');
351
+ } finally {
352
+ globalThis.fetch = originalFetch;
353
+ }
354
+ });
355
+
356
+ test.serial('fetchCloudModels: handles unknown error type', async t => {
357
+ globalThis.fetch = async () => {
358
+ throw 'string error'; // Non-Error thrown
359
+ };
360
+
361
+ try {
362
+ const result = await fetchCloudModels('openai', 'api-key');
363
+ t.false(result.success);
364
+ t.is(result.models.length, 0);
365
+ t.is(result.error, 'Unknown error occurred');
366
+ } finally {
367
+ globalThis.fetch = originalFetch;
368
+ }
369
+ });
370
+
371
+ // Data validation tests
372
+ test.serial('fetchCloudModels: filters out invalid model entries', async t => {
373
+ const response = createCloudModelsResponse([
374
+ {id: 'valid-model'},
375
+ {id: ''}, // Empty id
376
+ {id: ' '}, // Whitespace only
377
+ {} as {id: string}, // Missing id
378
+ {id: 'another-valid'},
379
+ ]);
380
+ const {mockFetch} = createMockFetch(response);
381
+ globalThis.fetch = mockFetch;
382
+
383
+ try {
384
+ const result = await fetchCloudModels('openai', 'api-key');
385
+ t.true(result.success);
386
+ t.is(result.models.length, 2);
387
+ } finally {
388
+ globalThis.fetch = originalFetch;
389
+ }
390
+ });
391
+
392
+ // Sorting test
393
+ test.serial('fetchCloudModels: returns models sorted alphabetically by name', async t => {
394
+ const response = createCloudModelsResponse([
395
+ {id: 'z-model', display_name: 'Zebra Model'},
396
+ {id: 'a-model', display_name: 'Alpha Model'},
397
+ {id: 'm-model', display_name: 'Middle Model'},
398
+ ]);
399
+ const {mockFetch} = createMockFetch(response);
400
+ globalThis.fetch = mockFetch;
401
+
402
+ try {
403
+ const result = await fetchCloudModels('anthropic', 'api-key');
404
+ t.true(result.success);
405
+ t.is(result.models.length, 3);
406
+ t.is(result.models[0]?.name, 'Alpha Model');
407
+ t.is(result.models[1]?.name, 'Middle Model');
408
+ t.is(result.models[2]?.name, 'Zebra Model');
409
+ } finally {
410
+ globalThis.fetch = originalFetch;
411
+ }
412
+ });
413
+
414
+ // Fallback to id when no name/display_name
415
+ test.serial('fetchCloudModels: falls back to id when no name or display_name', async t => {
416
+ const response = createCloudModelsResponse([{id: 'model-id-only'}]);
417
+ const {mockFetch} = createMockFetch(response);
418
+ globalThis.fetch = mockFetch;
419
+
420
+ try {
421
+ const result = await fetchCloudModels('openai', 'api-key');
422
+ t.true(result.success);
423
+ t.is(result.models[0]?.id, 'model-id-only');
424
+ t.is(result.models[0]?.name, 'model-id-only');
425
+ } finally {
426
+ globalThis.fetch = originalFetch;
427
+ }
428
+ });
@@ -0,0 +1,223 @@
1
+ /**
2
+ * Fetch models from cloud LLM providers
3
+ * - Anthropic: GET /v1/models with X-Api-Key header
4
+ * - OpenAI: GET /v1/models with Authorization: Bearer header
5
+ * - Mistral: GET /v1/models with Authorization: Bearer header
6
+ * - GitHub: GET /catalog/models with Bearer token and X-GitHub-Api-Version header
7
+ */
8
+
9
+ import type {
10
+ CloudModelsEndpointType,
11
+ FetchModelsResult,
12
+ LocalModel,
13
+ } from './fetch-local-models';
14
+
15
+ interface CloudProviderConfig {
16
+ endpoint: string;
17
+ getHeaders: (apiKey: string) => Record<string, string>;
18
+ }
19
+
20
+ const CLOUD_PROVIDERS: Record<CloudModelsEndpointType, CloudProviderConfig> = {
21
+ anthropic: {
22
+ endpoint: 'https://api.anthropic.com/v1/models',
23
+ getHeaders: apiKey => ({
24
+ 'X-Api-Key': apiKey,
25
+ 'anthropic-version': '2023-06-01',
26
+ Accept: 'application/json',
27
+ }),
28
+ },
29
+ openai: {
30
+ endpoint: 'https://api.openai.com/v1/models',
31
+ getHeaders: apiKey => ({
32
+ Authorization: `Bearer ${apiKey}`,
33
+ Accept: 'application/json',
34
+ }),
35
+ },
36
+ mistral: {
37
+ endpoint: 'https://api.mistral.ai/v1/models',
38
+ getHeaders: apiKey => ({
39
+ Authorization: `Bearer ${apiKey}`,
40
+ Accept: 'application/json',
41
+ }),
42
+ },
43
+ github: {
44
+ endpoint: 'https://models.github.ai/catalog/models',
45
+ getHeaders: apiKey => ({
46
+ Accept: 'application/vnd.github+json',
47
+ Authorization: `Bearer ${apiKey}`,
48
+ 'X-GitHub-Api-Version': '2022-11-28',
49
+ }),
50
+ },
51
+ };
52
+
53
+ interface CloudModelsResponse {
54
+ data?: Array<{
55
+ id: string;
56
+ name?: string; // GitHub
57
+ display_name?: string; // Anthropic
58
+ object?: string; // OpenAI/Mistral
59
+ }>;
60
+ }
61
+
62
+ export interface FetchCloudModelsOptions {
63
+ timeoutMs?: number;
64
+ debug?: boolean;
65
+ }
66
+
67
+ /**
68
+ * Fetch available models from a cloud LLM provider
69
+ * @param providerType The cloud provider type ('anthropic', 'openai', 'mistral', 'github')
70
+ * @param apiKey The API key for authentication
71
+ * @param options Optional settings (timeoutMs, debug)
72
+ */
73
+ export async function fetchCloudModels(
74
+ providerType: CloudModelsEndpointType,
75
+ apiKey: string,
76
+ options: FetchCloudModelsOptions = {},
77
+ ): Promise<FetchModelsResult> {
78
+ const {timeoutMs = 10000, debug = false} = options;
79
+
80
+ const log = (message: string) => {
81
+ if (debug) {
82
+ console.log(`[fetch-cloud-models] ${message}`);
83
+ }
84
+ };
85
+
86
+ const providerConfig = CLOUD_PROVIDERS[providerType];
87
+ if (!providerConfig) {
88
+ return {
89
+ success: false,
90
+ models: [],
91
+ error: `Unknown cloud provider: ${providerType}`,
92
+ };
93
+ }
94
+
95
+ if (!apiKey || !apiKey.trim()) {
96
+ return {
97
+ success: false,
98
+ models: [],
99
+ error: 'API key is required',
100
+ };
101
+ }
102
+
103
+ try {
104
+ log(`Fetching models from ${providerType}: ${providerConfig.endpoint}`);
105
+
106
+ const controller = new AbortController();
107
+ const timeoutId = setTimeout(() => controller.abort(), timeoutMs);
108
+
109
+ let data: CloudModelsResponse;
110
+ try {
111
+ const response = await fetch(providerConfig.endpoint, {
112
+ method: 'GET',
113
+ signal: controller.signal,
114
+ headers: providerConfig.getHeaders(apiKey),
115
+ });
116
+
117
+ if (!response.ok) {
118
+ log(`Server error: ${response.status} ${response.statusText}`);
119
+
120
+ // Provide helpful error messages for common cases
121
+ if (response.status === 401) {
122
+ return {
123
+ success: false,
124
+ models: [],
125
+ error: 'Invalid API key',
126
+ };
127
+ }
128
+ if (response.status === 403) {
129
+ return {
130
+ success: false,
131
+ models: [],
132
+ error: 'API key does not have permission to list models',
133
+ };
134
+ }
135
+
136
+ return {
137
+ success: false,
138
+ models: [],
139
+ error: `API returned ${response.status}: ${response.statusText}`,
140
+ };
141
+ }
142
+
143
+ log(`Response OK, parsing JSON...`);
144
+ data = (await response.json()) as CloudModelsResponse;
145
+ log(`JSON parsed successfully`);
146
+ } finally {
147
+ clearTimeout(timeoutId);
148
+ }
149
+
150
+ // GitHub returns a direct array, others return { data: [...] }
151
+ type ModelEntry = {id: string; name?: string; display_name?: string};
152
+ let modelArray: ModelEntry[];
153
+
154
+ if (providerType === 'github') {
155
+ // GitHub returns array directly
156
+ if (!Array.isArray(data)) {
157
+ return {
158
+ success: false,
159
+ models: [],
160
+ error: 'Invalid response format from API',
161
+ };
162
+ }
163
+ modelArray = data as unknown as ModelEntry[];
164
+ } else {
165
+ // Other providers return { data: [...] }
166
+ if (!data.data || !Array.isArray(data.data)) {
167
+ return {
168
+ success: false,
169
+ models: [],
170
+ error: 'Invalid response format from API',
171
+ };
172
+ }
173
+ modelArray = data.data;
174
+ }
175
+
176
+ // Parse models - use name (GitHub) or display_name (Anthropic), otherwise id
177
+ const models: LocalModel[] = modelArray
178
+ .filter(m => m && typeof m.id === 'string' && m.id.trim())
179
+ .map(m => ({
180
+ id: m.id.trim(),
181
+ name: m.name || m.display_name || m.id.trim(),
182
+ }))
183
+ .sort((a, b) => a.name.localeCompare(b.name));
184
+
185
+ log(`Found ${models.length} models (sorted)`);
186
+
187
+ if (models.length === 0) {
188
+ return {
189
+ success: false,
190
+ models: [],
191
+ error: 'No models found for this API key',
192
+ };
193
+ }
194
+
195
+ return {
196
+ success: true,
197
+ models,
198
+ };
199
+ } catch (err) {
200
+ if (err instanceof Error) {
201
+ if (err.name === 'AbortError') {
202
+ log(`Request timed out after ${timeoutMs}ms`);
203
+ return {
204
+ success: false,
205
+ models: [],
206
+ error: 'Connection timed out',
207
+ };
208
+ }
209
+ log(`Error: ${err.message}`);
210
+ return {
211
+ success: false,
212
+ models: [],
213
+ error: err.message,
214
+ };
215
+ }
216
+ log(`Unknown error occurred`);
217
+ return {
218
+ success: false,
219
+ models: [],
220
+ error: 'Unknown error occurred',
221
+ };
222
+ }
223
+ }