@gguf/coder 0.3.0 → 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 (414) 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/package.json +2 -14
  10. package/scripts/extract-changelog.js +73 -0
  11. package/scripts/fetch-models.js +143 -0
  12. package/scripts/test.sh +40 -0
  13. package/scripts/update-homebrew-formula.sh +125 -0
  14. package/scripts/update-nix-version.sh +157 -0
  15. package/source/ai-sdk-client/AISDKClient.spec.ts +117 -0
  16. package/source/ai-sdk-client/AISDKClient.ts +155 -0
  17. package/source/ai-sdk-client/chat/chat-handler.spec.ts +121 -0
  18. package/source/ai-sdk-client/chat/chat-handler.ts +276 -0
  19. package/source/ai-sdk-client/chat/streaming-handler.spec.ts +173 -0
  20. package/source/ai-sdk-client/chat/streaming-handler.ts +110 -0
  21. package/source/ai-sdk-client/chat/tool-processor.spec.ts +92 -0
  22. package/source/ai-sdk-client/chat/tool-processor.ts +70 -0
  23. package/source/ai-sdk-client/converters/message-converter.spec.ts +220 -0
  24. package/source/ai-sdk-client/converters/message-converter.ts +113 -0
  25. package/source/ai-sdk-client/converters/tool-converter.spec.ts +90 -0
  26. package/source/ai-sdk-client/converters/tool-converter.ts +46 -0
  27. package/source/ai-sdk-client/error-handling/error-extractor.spec.ts +55 -0
  28. package/source/ai-sdk-client/error-handling/error-extractor.ts +15 -0
  29. package/source/ai-sdk-client/error-handling/error-parser.spec.ts +169 -0
  30. package/source/ai-sdk-client/error-handling/error-parser.ts +161 -0
  31. package/source/ai-sdk-client/index.ts +7 -0
  32. package/source/ai-sdk-client/providers/provider-factory.spec.ts +71 -0
  33. package/source/ai-sdk-client/providers/provider-factory.ts +41 -0
  34. package/source/ai-sdk-client/types.ts +9 -0
  35. package/source/ai-sdk-client-empty-message.spec.ts +141 -0
  36. package/source/ai-sdk-client-error-handling.spec.ts +186 -0
  37. package/source/ai-sdk-client-maxretries.spec.ts +114 -0
  38. package/source/ai-sdk-client-preparestep.spec.ts +279 -0
  39. package/source/app/App.spec.tsx +32 -0
  40. package/source/app/App.tsx +480 -0
  41. package/source/app/components/AppContainer.spec.tsx +96 -0
  42. package/source/app/components/AppContainer.tsx +56 -0
  43. package/source/app/components/ChatInterface.spec.tsx +163 -0
  44. package/source/app/components/ChatInterface.tsx +144 -0
  45. package/source/app/components/ModalSelectors.spec.tsx +141 -0
  46. package/source/app/components/ModalSelectors.tsx +135 -0
  47. package/source/app/helpers.spec.ts +97 -0
  48. package/source/app/helpers.ts +63 -0
  49. package/source/app/index.ts +4 -0
  50. package/source/app/types.ts +39 -0
  51. package/source/app/utils/appUtils.ts +294 -0
  52. package/source/app/utils/conversationState.ts +310 -0
  53. package/source/app.spec.tsx +244 -0
  54. package/source/cli.spec.ts +73 -0
  55. package/source/cli.tsx +51 -0
  56. package/source/client-factory.spec.ts +48 -0
  57. package/source/client-factory.ts +178 -0
  58. package/source/command-parser.spec.ts +127 -0
  59. package/source/command-parser.ts +36 -0
  60. package/source/commands/checkpoint.spec.tsx +277 -0
  61. package/source/commands/checkpoint.tsx +366 -0
  62. package/source/commands/clear.tsx +22 -0
  63. package/source/commands/custom-commands.tsx +121 -0
  64. package/source/commands/exit.ts +21 -0
  65. package/source/commands/export.spec.tsx +131 -0
  66. package/source/commands/export.tsx +79 -0
  67. package/source/commands/help.tsx +120 -0
  68. package/source/commands/index.ts +17 -0
  69. package/source/commands/init.tsx +339 -0
  70. package/source/commands/lsp-command.spec.tsx +281 -0
  71. package/source/commands/lsp.tsx +120 -0
  72. package/source/commands/mcp-command.spec.tsx +313 -0
  73. package/source/commands/mcp.tsx +162 -0
  74. package/source/commands/model-database.spec.tsx +758 -0
  75. package/source/commands/model-database.tsx +418 -0
  76. package/source/commands/model.ts +12 -0
  77. package/source/commands/provider.ts +12 -0
  78. package/source/commands/setup-config.tsx +16 -0
  79. package/source/commands/simple-commands.spec.tsx +175 -0
  80. package/source/commands/status.ts +12 -0
  81. package/source/commands/theme.ts +12 -0
  82. package/source/commands/update.spec.tsx +261 -0
  83. package/source/commands/update.tsx +201 -0
  84. package/source/commands/usage.spec.tsx +495 -0
  85. package/source/commands/usage.tsx +100 -0
  86. package/source/commands.spec.ts +436 -0
  87. package/source/commands.ts +83 -0
  88. package/source/components/assistant-message.spec.tsx +796 -0
  89. package/source/components/assistant-message.tsx +34 -0
  90. package/source/components/bash-execution-indicator.tsx +21 -0
  91. package/source/components/cancelling-indicator.tsx +16 -0
  92. package/source/components/chat-queue.spec.tsx +83 -0
  93. package/source/components/chat-queue.tsx +36 -0
  94. package/source/components/checkpoint-display.spec.tsx +219 -0
  95. package/source/components/checkpoint-display.tsx +126 -0
  96. package/source/components/checkpoint-selector.spec.tsx +173 -0
  97. package/source/components/checkpoint-selector.tsx +173 -0
  98. package/source/components/development-mode-indicator.spec.tsx +268 -0
  99. package/source/components/development-mode-indicator.tsx +38 -0
  100. package/source/components/message-box.spec.tsx +427 -0
  101. package/source/components/message-box.tsx +87 -0
  102. package/source/components/model-selector.tsx +132 -0
  103. package/source/components/provider-selector.tsx +75 -0
  104. package/source/components/random-spinner.tsx +19 -0
  105. package/source/components/security-disclaimer.tsx +73 -0
  106. package/source/components/status-connection-display.spec.tsx +133 -0
  107. package/source/components/status.tsx +267 -0
  108. package/source/components/theme-selector.tsx +126 -0
  109. package/source/components/tool-confirmation.tsx +190 -0
  110. package/source/components/tool-execution-indicator.tsx +33 -0
  111. package/source/components/tool-message.tsx +85 -0
  112. package/source/components/ui/titled-box.spec.tsx +207 -0
  113. package/source/components/ui/titled-box.tsx +57 -0
  114. package/source/components/usage/progress-bar.spec.tsx +398 -0
  115. package/source/components/usage/progress-bar.tsx +30 -0
  116. package/source/components/usage/usage-display.spec.tsx +780 -0
  117. package/source/components/usage/usage-display.tsx +291 -0
  118. package/source/components/user-input.spec.tsx +327 -0
  119. package/source/components/user-input.tsx +533 -0
  120. package/source/components/user-message.spec.tsx +230 -0
  121. package/source/components/user-message.tsx +84 -0
  122. package/source/components/welcome-message.tsx +76 -0
  123. package/source/config/env-substitution.ts +65 -0
  124. package/source/config/index.spec.ts +171 -0
  125. package/source/config/index.ts +154 -0
  126. package/source/config/paths.spec.ts +241 -0
  127. package/source/config/paths.ts +55 -0
  128. package/source/config/preferences.ts +51 -0
  129. package/source/config/themes.ts +315 -0
  130. package/source/constants.ts +130 -0
  131. package/source/context/mode-context.spec.ts +79 -0
  132. package/source/context/mode-context.ts +24 -0
  133. package/source/custom-commands/executor.spec.ts +142 -0
  134. package/source/custom-commands/executor.ts +64 -0
  135. package/source/custom-commands/loader.spec.ts +314 -0
  136. package/source/custom-commands/loader.ts +153 -0
  137. package/source/custom-commands/parser.ts +196 -0
  138. package/source/hooks/chat-handler/conversation/conversation-loop.spec.ts +39 -0
  139. package/source/hooks/chat-handler/conversation/conversation-loop.tsx +511 -0
  140. package/source/hooks/chat-handler/conversation/tool-executor.spec.ts +50 -0
  141. package/source/hooks/chat-handler/conversation/tool-executor.tsx +109 -0
  142. package/source/hooks/chat-handler/index.ts +12 -0
  143. package/source/hooks/chat-handler/state/streaming-state.spec.ts +26 -0
  144. package/source/hooks/chat-handler/state/streaming-state.ts +19 -0
  145. package/source/hooks/chat-handler/types.ts +38 -0
  146. package/source/hooks/chat-handler/useChatHandler.spec.tsx +321 -0
  147. package/source/hooks/chat-handler/useChatHandler.tsx +194 -0
  148. package/source/hooks/chat-handler/utils/context-checker.spec.ts +60 -0
  149. package/source/hooks/chat-handler/utils/context-checker.tsx +73 -0
  150. package/source/hooks/chat-handler/utils/message-helpers.spec.ts +42 -0
  151. package/source/hooks/chat-handler/utils/message-helpers.tsx +36 -0
  152. package/source/hooks/chat-handler/utils/tool-filters.spec.ts +109 -0
  153. package/source/hooks/chat-handler/utils/tool-filters.ts +64 -0
  154. package/source/hooks/useAppHandlers.tsx +291 -0
  155. package/source/hooks/useAppInitialization.tsx +422 -0
  156. package/source/hooks/useAppState.tsx +311 -0
  157. package/source/hooks/useDirectoryTrust.tsx +98 -0
  158. package/source/hooks/useInputState.ts +414 -0
  159. package/source/hooks/useModeHandlers.tsx +302 -0
  160. package/source/hooks/useNonInteractiveMode.ts +140 -0
  161. package/source/hooks/useTerminalWidth.tsx +81 -0
  162. package/source/hooks/useTheme.ts +18 -0
  163. package/source/hooks/useToolHandler.tsx +349 -0
  164. package/source/hooks/useUIState.ts +61 -0
  165. package/source/init/agents-template-generator.ts +421 -0
  166. package/source/init/existing-rules-extractor.ts +319 -0
  167. package/source/init/file-scanner.spec.ts +227 -0
  168. package/source/init/file-scanner.ts +238 -0
  169. package/source/init/framework-detector.ts +382 -0
  170. package/source/init/language-detector.ts +269 -0
  171. package/source/init/project-analyzer.spec.ts +231 -0
  172. package/source/init/project-analyzer.ts +458 -0
  173. package/source/lsp/index.ts +31 -0
  174. package/source/lsp/lsp-client.spec.ts +508 -0
  175. package/source/lsp/lsp-client.ts +487 -0
  176. package/source/lsp/lsp-manager.spec.ts +477 -0
  177. package/source/lsp/lsp-manager.ts +419 -0
  178. package/source/lsp/protocol.spec.ts +502 -0
  179. package/source/lsp/protocol.ts +360 -0
  180. package/source/lsp/server-discovery.spec.ts +654 -0
  181. package/source/lsp/server-discovery.ts +515 -0
  182. package/source/markdown-parser/html-entities.spec.ts +88 -0
  183. package/source/markdown-parser/html-entities.ts +45 -0
  184. package/source/markdown-parser/index.spec.ts +281 -0
  185. package/source/markdown-parser/index.ts +126 -0
  186. package/source/markdown-parser/table-parser.spec.ts +133 -0
  187. package/source/markdown-parser/table-parser.ts +114 -0
  188. package/source/markdown-parser/utils.spec.ts +70 -0
  189. package/source/markdown-parser/utils.ts +13 -0
  190. package/source/mcp/mcp-client.spec.ts +81 -0
  191. package/source/mcp/mcp-client.ts +625 -0
  192. package/source/mcp/transport-factory.spec.ts +406 -0
  193. package/source/mcp/transport-factory.ts +312 -0
  194. package/source/message-handler.ts +67 -0
  195. package/source/model-database/database-engine.spec.ts +494 -0
  196. package/source/model-database/database-engine.ts +50 -0
  197. package/source/model-database/model-database.spec.ts +363 -0
  198. package/source/model-database/model-database.ts +91 -0
  199. package/source/model-database/model-engine.spec.ts +447 -0
  200. package/source/model-database/model-engine.ts +65 -0
  201. package/source/model-database/model-fetcher.spec.ts +583 -0
  202. package/source/model-database/model-fetcher.ts +330 -0
  203. package/source/models/index.ts +1 -0
  204. package/source/models/models-cache.spec.ts +214 -0
  205. package/source/models/models-cache.ts +78 -0
  206. package/source/models/models-dev-client.spec.ts +379 -0
  207. package/source/models/models-dev-client.ts +329 -0
  208. package/source/models/models-types.ts +68 -0
  209. package/source/prompt-history.ts +155 -0
  210. package/source/security/command-injection.spec.ts +240 -0
  211. package/source/services/checkpoint-manager.spec.ts +523 -0
  212. package/source/services/checkpoint-manager.ts +466 -0
  213. package/source/services/file-snapshot.spec.ts +569 -0
  214. package/source/services/file-snapshot.ts +220 -0
  215. package/source/test-utils/render-with-theme.tsx +48 -0
  216. package/source/tokenization/index.ts +1 -0
  217. package/source/tokenization/tokenizer-factory.spec.ts +170 -0
  218. package/source/tokenization/tokenizer-factory.ts +125 -0
  219. package/source/tokenization/tokenizers/anthropic-tokenizer.spec.ts +200 -0
  220. package/source/tokenization/tokenizers/anthropic-tokenizer.ts +43 -0
  221. package/source/tokenization/tokenizers/fallback-tokenizer.spec.ts +236 -0
  222. package/source/tokenization/tokenizers/fallback-tokenizer.ts +26 -0
  223. package/source/tokenization/tokenizers/llama-tokenizer.spec.ts +224 -0
  224. package/source/tokenization/tokenizers/llama-tokenizer.ts +41 -0
  225. package/source/tokenization/tokenizers/openai-tokenizer.spec.ts +184 -0
  226. package/source/tokenization/tokenizers/openai-tokenizer.ts +57 -0
  227. package/source/tool-calling/index.ts +5 -0
  228. package/source/tool-calling/json-parser.spec.ts +639 -0
  229. package/source/tool-calling/json-parser.ts +247 -0
  230. package/source/tool-calling/tool-parser.spec.ts +395 -0
  231. package/source/tool-calling/tool-parser.ts +120 -0
  232. package/source/tool-calling/xml-parser.spec.ts +662 -0
  233. package/source/tool-calling/xml-parser.ts +289 -0
  234. package/source/tools/execute-bash.spec.tsx +353 -0
  235. package/source/tools/execute-bash.tsx +219 -0
  236. package/source/tools/execute-function.spec.ts +130 -0
  237. package/source/tools/fetch-url.spec.tsx +342 -0
  238. package/source/tools/fetch-url.tsx +172 -0
  239. package/source/tools/find-files.spec.tsx +924 -0
  240. package/source/tools/find-files.tsx +293 -0
  241. package/source/tools/index.ts +102 -0
  242. package/source/tools/lsp-get-diagnostics.tsx +192 -0
  243. package/source/tools/needs-approval.spec.ts +282 -0
  244. package/source/tools/read-file.spec.tsx +801 -0
  245. package/source/tools/read-file.tsx +387 -0
  246. package/source/tools/search-file-contents.spec.tsx +1273 -0
  247. package/source/tools/search-file-contents.tsx +293 -0
  248. package/source/tools/string-replace.spec.tsx +730 -0
  249. package/source/tools/string-replace.tsx +548 -0
  250. package/source/tools/tool-manager.ts +210 -0
  251. package/source/tools/tool-registry.spec.ts +415 -0
  252. package/source/tools/tool-registry.ts +228 -0
  253. package/source/tools/web-search.tsx +223 -0
  254. package/source/tools/write-file.spec.tsx +559 -0
  255. package/source/tools/write-file.tsx +228 -0
  256. package/source/types/app.ts +37 -0
  257. package/source/types/checkpoint.ts +48 -0
  258. package/source/types/commands.ts +46 -0
  259. package/source/types/components.ts +27 -0
  260. package/source/types/config.ts +103 -0
  261. package/source/types/core-connection-status.spec.ts +67 -0
  262. package/source/types/core.ts +181 -0
  263. package/source/types/hooks.ts +50 -0
  264. package/source/types/index.ts +12 -0
  265. package/source/types/markdown-parser.ts +11 -0
  266. package/source/types/mcp.ts +52 -0
  267. package/source/types/system.ts +16 -0
  268. package/source/types/tokenization.ts +41 -0
  269. package/source/types/ui.ts +40 -0
  270. package/source/types/usage.ts +58 -0
  271. package/source/types/utils.ts +16 -0
  272. package/source/usage/calculator.spec.ts +385 -0
  273. package/source/usage/calculator.ts +104 -0
  274. package/source/usage/storage.spec.ts +703 -0
  275. package/source/usage/storage.ts +238 -0
  276. package/source/usage/tracker.spec.ts +456 -0
  277. package/source/usage/tracker.ts +102 -0
  278. package/source/utils/atomic-deletion.spec.ts +194 -0
  279. package/source/utils/atomic-deletion.ts +127 -0
  280. package/source/utils/bounded-map.spec.ts +300 -0
  281. package/source/utils/bounded-map.ts +193 -0
  282. package/source/utils/checkpoint-utils.spec.ts +222 -0
  283. package/source/utils/checkpoint-utils.ts +92 -0
  284. package/source/utils/error-formatter.spec.ts +169 -0
  285. package/source/utils/error-formatter.ts +194 -0
  286. package/source/utils/file-autocomplete.spec.ts +173 -0
  287. package/source/utils/file-autocomplete.ts +196 -0
  288. package/source/utils/file-cache.spec.ts +309 -0
  289. package/source/utils/file-cache.ts +195 -0
  290. package/source/utils/file-content-loader.spec.ts +180 -0
  291. package/source/utils/file-content-loader.ts +179 -0
  292. package/source/utils/file-mention-handler.spec.ts +261 -0
  293. package/source/utils/file-mention-handler.ts +84 -0
  294. package/source/utils/file-mention-parser.spec.ts +182 -0
  295. package/source/utils/file-mention-parser.ts +170 -0
  296. package/source/utils/fuzzy-matching.spec.ts +149 -0
  297. package/source/utils/fuzzy-matching.ts +146 -0
  298. package/source/utils/indentation-normalizer.spec.ts +216 -0
  299. package/source/utils/indentation-normalizer.ts +76 -0
  300. package/source/utils/installation-detector.spec.ts +178 -0
  301. package/source/utils/installation-detector.ts +153 -0
  302. package/source/utils/logging/config.spec.ts +311 -0
  303. package/source/utils/logging/config.ts +210 -0
  304. package/source/utils/logging/console-facade.spec.ts +184 -0
  305. package/source/utils/logging/console-facade.ts +384 -0
  306. package/source/utils/logging/correlation.spec.ts +679 -0
  307. package/source/utils/logging/correlation.ts +474 -0
  308. package/source/utils/logging/formatters.spec.ts +464 -0
  309. package/source/utils/logging/formatters.ts +207 -0
  310. package/source/utils/logging/health-monitor/alerts/alert-manager.spec.ts +93 -0
  311. package/source/utils/logging/health-monitor/alerts/alert-manager.ts +79 -0
  312. package/source/utils/logging/health-monitor/checks/configuration-check.spec.ts +56 -0
  313. package/source/utils/logging/health-monitor/checks/configuration-check.ts +43 -0
  314. package/source/utils/logging/health-monitor/checks/logging-check.spec.ts +56 -0
  315. package/source/utils/logging/health-monitor/checks/logging-check.ts +58 -0
  316. package/source/utils/logging/health-monitor/checks/memory-check.spec.ts +100 -0
  317. package/source/utils/logging/health-monitor/checks/memory-check.ts +78 -0
  318. package/source/utils/logging/health-monitor/checks/performance-check.spec.ts +56 -0
  319. package/source/utils/logging/health-monitor/checks/performance-check.ts +56 -0
  320. package/source/utils/logging/health-monitor/checks/request-check.spec.ts +56 -0
  321. package/source/utils/logging/health-monitor/checks/request-check.ts +76 -0
  322. package/source/utils/logging/health-monitor/core/health-check-runner.spec.ts +70 -0
  323. package/source/utils/logging/health-monitor/core/health-check-runner.ts +138 -0
  324. package/source/utils/logging/health-monitor/core/health-monitor.spec.ts +58 -0
  325. package/source/utils/logging/health-monitor/core/health-monitor.ts +344 -0
  326. package/source/utils/logging/health-monitor/core/scoring.spec.ts +65 -0
  327. package/source/utils/logging/health-monitor/core/scoring.ts +91 -0
  328. package/source/utils/logging/health-monitor/index.ts +15 -0
  329. package/source/utils/logging/health-monitor/instances.ts +48 -0
  330. package/source/utils/logging/health-monitor/middleware/http-middleware.spec.ts +141 -0
  331. package/source/utils/logging/health-monitor/middleware/http-middleware.ts +75 -0
  332. package/source/utils/logging/health-monitor/types.ts +126 -0
  333. package/source/utils/logging/index.spec.ts +284 -0
  334. package/source/utils/logging/index.ts +236 -0
  335. package/source/utils/logging/integration.spec.ts +441 -0
  336. package/source/utils/logging/log-method-factory.spec.ts +573 -0
  337. package/source/utils/logging/log-method-factory.ts +233 -0
  338. package/source/utils/logging/log-query/aggregation/aggregator.spec.ts +277 -0
  339. package/source/utils/logging/log-query/aggregation/aggregator.ts +159 -0
  340. package/source/utils/logging/log-query/aggregation/facet-generator.spec.ts +159 -0
  341. package/source/utils/logging/log-query/aggregation/facet-generator.ts +47 -0
  342. package/source/utils/logging/log-query/index.ts +23 -0
  343. package/source/utils/logging/log-query/query/filter-predicates.spec.ts +247 -0
  344. package/source/utils/logging/log-query/query/filter-predicates.ts +154 -0
  345. package/source/utils/logging/log-query/query/query-builder.spec.ts +182 -0
  346. package/source/utils/logging/log-query/query/query-builder.ts +151 -0
  347. package/source/utils/logging/log-query/query/query-engine.spec.ts +214 -0
  348. package/source/utils/logging/log-query/query/query-engine.ts +45 -0
  349. package/source/utils/logging/log-query/storage/circular-buffer.spec.ts +143 -0
  350. package/source/utils/logging/log-query/storage/circular-buffer.ts +75 -0
  351. package/source/utils/logging/log-query/storage/index-manager.spec.ts +150 -0
  352. package/source/utils/logging/log-query/storage/index-manager.ts +71 -0
  353. package/source/utils/logging/log-query/storage/log-storage.spec.ts +257 -0
  354. package/source/utils/logging/log-query/storage/log-storage.ts +80 -0
  355. package/source/utils/logging/log-query/types.ts +163 -0
  356. package/source/utils/logging/log-query/utils/helpers.spec.ts +263 -0
  357. package/source/utils/logging/log-query/utils/helpers.ts +72 -0
  358. package/source/utils/logging/log-query/utils/sorting.spec.ts +182 -0
  359. package/source/utils/logging/log-query/utils/sorting.ts +61 -0
  360. package/source/utils/logging/logger-provider.spec.ts +262 -0
  361. package/source/utils/logging/logger-provider.ts +362 -0
  362. package/source/utils/logging/performance.spec.ts +209 -0
  363. package/source/utils/logging/performance.ts +757 -0
  364. package/source/utils/logging/pino-logger.spec.ts +425 -0
  365. package/source/utils/logging/pino-logger.ts +514 -0
  366. package/source/utils/logging/redaction.spec.ts +490 -0
  367. package/source/utils/logging/redaction.ts +267 -0
  368. package/source/utils/logging/request-tracker.spec.ts +1198 -0
  369. package/source/utils/logging/request-tracker.ts +803 -0
  370. package/source/utils/logging/transports.spec.ts +505 -0
  371. package/source/utils/logging/transports.ts +305 -0
  372. package/source/utils/logging/types.ts +216 -0
  373. package/source/utils/message-builder.spec.ts +179 -0
  374. package/source/utils/message-builder.ts +101 -0
  375. package/source/utils/message-queue.tsx +486 -0
  376. package/source/utils/paste-detection.spec.ts +69 -0
  377. package/source/utils/paste-detection.ts +124 -0
  378. package/source/utils/paste-roundtrip.spec.ts +442 -0
  379. package/source/utils/paste-utils.spec.ts +128 -0
  380. package/source/utils/paste-utils.ts +52 -0
  381. package/source/utils/programming-language-helper.spec.ts +74 -0
  382. package/source/utils/programming-language-helper.ts +32 -0
  383. package/source/utils/prompt-assembly.spec.ts +221 -0
  384. package/source/utils/prompt-processor.ts +173 -0
  385. package/source/utils/tool-args-parser.spec.ts +136 -0
  386. package/source/utils/tool-args-parser.ts +54 -0
  387. package/source/utils/tool-cancellation.spec.ts +230 -0
  388. package/source/utils/tool-cancellation.ts +28 -0
  389. package/source/utils/tool-result-display.spec.tsx +469 -0
  390. package/source/utils/tool-result-display.tsx +90 -0
  391. package/source/utils/update-checker.spec.ts +383 -0
  392. package/source/utils/update-checker.ts +183 -0
  393. package/source/wizard/config-wizard.spec.tsx +103 -0
  394. package/source/wizard/config-wizard.tsx +382 -0
  395. package/source/wizard/steps/location-step.spec.tsx +186 -0
  396. package/source/wizard/steps/location-step.tsx +147 -0
  397. package/source/wizard/steps/mcp-step.spec.tsx +607 -0
  398. package/source/wizard/steps/mcp-step.tsx +632 -0
  399. package/source/wizard/steps/provider-step.spec.tsx +342 -0
  400. package/source/wizard/steps/provider-step.tsx +957 -0
  401. package/source/wizard/steps/summary-step.spec.tsx +749 -0
  402. package/source/wizard/steps/summary-step.tsx +228 -0
  403. package/source/wizard/templates/mcp-templates.spec.ts +613 -0
  404. package/source/wizard/templates/mcp-templates.ts +570 -0
  405. package/source/wizard/templates/provider-templates.spec.ts +152 -0
  406. package/source/wizard/templates/provider-templates.ts +485 -0
  407. package/source/wizard/utils/fetch-cloud-models.spec.ts +428 -0
  408. package/source/wizard/utils/fetch-cloud-models.ts +223 -0
  409. package/source/wizard/utils/fetch-local-models.spec.ts +297 -0
  410. package/source/wizard/utils/fetch-local-models.ts +192 -0
  411. package/source/wizard/validation-array.spec.ts +264 -0
  412. package/source/wizard/validation.spec.ts +373 -0
  413. package/source/wizard/validation.ts +232 -0
  414. package/source/app/prompts/main-prompt.md +0 -122
@@ -0,0 +1,570 @@
1
+ import {TIMEOUT_MCP_DEFAULT_MS} from '@/constants';
2
+ import type {TemplateField} from './provider-templates';
3
+
4
+ export type McpTransportType = 'stdio' | 'websocket' | 'http';
5
+
6
+ export interface McpServerConfig {
7
+ name: string;
8
+ transport: McpTransportType;
9
+
10
+ // STDIO-specific
11
+ command?: string;
12
+ args?: string[];
13
+ env?: Record<string, string>;
14
+
15
+ // Remote transport-specific
16
+ url?: string;
17
+ headers?: Record<string, string>;
18
+ auth?: {
19
+ type: 'bearer' | 'basic' | 'api-key' | 'custom';
20
+ token?: string;
21
+ username?: string;
22
+ password?: string;
23
+ apiKey?: string;
24
+ customHeaders?: Record<string, string>;
25
+ };
26
+ timeout?: number;
27
+
28
+ // Common
29
+ description?: string;
30
+ tags?: string[];
31
+ enabled?: boolean;
32
+ }
33
+
34
+ export interface McpTemplate {
35
+ id: string;
36
+ name: string;
37
+ description: string;
38
+ command: string;
39
+ fields: TemplateField[];
40
+ buildConfig: (answers: Record<string, string>) => McpServerConfig;
41
+ category?: 'local' | 'remote';
42
+ transportType: McpTransportType;
43
+ }
44
+
45
+ export const MCP_TEMPLATES: McpTemplate[] = [
46
+ {
47
+ id: 'filesystem',
48
+ name: 'Filesystem',
49
+ description: 'Read/write files and directories',
50
+ command: 'npx',
51
+ fields: [
52
+ {
53
+ name: 'allowedDirs',
54
+ prompt: 'Allowed directories (comma-separated paths)',
55
+ required: true,
56
+ },
57
+ ],
58
+ buildConfig: answers => ({
59
+ name: 'filesystem',
60
+ transport: 'stdio' as McpTransportType,
61
+ command: 'npx',
62
+ args: [
63
+ '-y',
64
+ '@modelcontextprotocol/server-filesystem',
65
+ ...answers.allowedDirs
66
+ .split(',')
67
+ .map(d => d.trim())
68
+ .filter(Boolean),
69
+ ],
70
+ description: 'Read/write files and directories',
71
+ tags: ['filesystem', 'local'],
72
+ }),
73
+ category: 'local',
74
+ transportType: 'stdio',
75
+ },
76
+ {
77
+ id: 'github',
78
+ name: 'GitHub',
79
+ description: 'Repository management and operations',
80
+ command: 'npx',
81
+ fields: [
82
+ {
83
+ name: 'githubToken',
84
+ prompt: 'GitHub Personal Access Token (scopes: repo, read:org)',
85
+ required: true,
86
+ sensitive: true,
87
+ },
88
+ ],
89
+ buildConfig: answers => ({
90
+ name: 'github',
91
+ transport: 'stdio' as McpTransportType,
92
+ command: 'npx',
93
+ args: ['-y', '@modelcontextprotocol/server-github'],
94
+ env: {
95
+ GITHUB_PERSONAL_ACCESS_TOKEN: answers.githubToken,
96
+ },
97
+ description: 'Repository management and operations',
98
+ tags: ['github', 'git', 'repository', 'stdio'],
99
+ }),
100
+ category: 'local',
101
+ transportType: 'stdio',
102
+ },
103
+ {
104
+ id: 'postgres',
105
+ name: 'PostgreSQL',
106
+ description: 'Database queries and management',
107
+ command: 'npx',
108
+ fields: [
109
+ {
110
+ name: 'connectionString',
111
+ prompt: 'Connection string (postgresql://user:pass@host:port/db)',
112
+ required: true,
113
+ sensitive: true,
114
+ },
115
+ ],
116
+ buildConfig: answers => ({
117
+ name: 'postgres',
118
+ transport: 'stdio' as McpTransportType,
119
+ command: 'npx',
120
+ args: ['-y', '@modelcontextprotocol/server-postgres'],
121
+ env: {
122
+ POSTGRES_CONNECTION_STRING: answers.connectionString,
123
+ },
124
+ description: 'Database queries and management',
125
+ tags: ['database', 'postgres', 'sql'],
126
+ }),
127
+ category: 'local',
128
+ transportType: 'stdio',
129
+ },
130
+ {
131
+ id: 'brave-search',
132
+ name: 'Brave Search',
133
+ description: 'Web search capabilities',
134
+ command: 'npx',
135
+ fields: [
136
+ {
137
+ name: 'braveApiKey',
138
+ prompt: 'Brave Search API Key',
139
+ required: true,
140
+ sensitive: true,
141
+ },
142
+ ],
143
+ buildConfig: answers => ({
144
+ name: 'brave-search',
145
+ transport: 'stdio' as McpTransportType,
146
+ command: 'npx',
147
+ args: ['-y', '@modelcontextprotocol/server-brave-search'],
148
+ env: {
149
+ BRAVE_API_KEY: answers.braveApiKey,
150
+ },
151
+ description: 'Web search capabilities',
152
+ tags: ['search', 'web', 'brave'],
153
+ }),
154
+ category: 'local',
155
+ transportType: 'stdio',
156
+ },
157
+ {
158
+ id: 'fetch',
159
+ name: 'Fetch',
160
+ description: 'HTTP requests and web scraping',
161
+ command: 'npx',
162
+ fields: [
163
+ {
164
+ name: 'userAgent',
165
+ prompt: 'User-Agent string (optional)',
166
+ required: false,
167
+ default: 'ModelContextProtocol/1.0',
168
+ },
169
+ ],
170
+ buildConfig: answers => {
171
+ const config: McpServerConfig = {
172
+ name: 'fetch',
173
+ transport: 'stdio' as McpTransportType,
174
+ command: 'npx',
175
+ args: ['-y', '@modelcontextprotocol/server-fetch'],
176
+ description: 'HTTP requests and web scraping',
177
+ tags: ['http', 'scraping', 'fetch', 'stdio'],
178
+ };
179
+ if (answers.userAgent) {
180
+ config.env = {USER_AGENT: answers.userAgent};
181
+ }
182
+ return config;
183
+ },
184
+ category: 'local',
185
+ transportType: 'stdio',
186
+ },
187
+ {
188
+ id: 'deepwiki',
189
+ name: 'DeepWiki',
190
+ description:
191
+ 'DeepWiki provides up-to-date documentation you can talk to, for every repo in the world.',
192
+ command: '',
193
+ fields: [
194
+ {
195
+ name: 'serverName',
196
+ prompt: 'Server name',
197
+ required: true,
198
+ default: 'deepwiki',
199
+ },
200
+ {
201
+ name: 'url',
202
+ prompt: 'Server URL',
203
+ required: true,
204
+ default: 'https://mcp.deepwiki.com/mcp',
205
+ },
206
+ ],
207
+ buildConfig: answers => ({
208
+ name: answers.serverName || 'deepwiki',
209
+ transport: 'http' as McpTransportType,
210
+ url: answers.url || 'https://mcp.deepwiki.com/mcp',
211
+ description:
212
+ 'DeepWiki provides up-to-date documentation you can talk to, for every repo in the world.',
213
+ tags: ['remote', 'wiki', 'documentation', 'http'],
214
+ timeout: TIMEOUT_MCP_DEFAULT_MS,
215
+ }),
216
+ category: 'remote',
217
+ transportType: 'http',
218
+ },
219
+ {
220
+ id: 'sequential-thinking',
221
+ name: 'Sequential Thinking',
222
+ description:
223
+ 'Dynamic and reflective problem-solving through thought sequences.',
224
+ command: '',
225
+ fields: [
226
+ {
227
+ name: 'serverName',
228
+ prompt: 'Server name',
229
+ required: true,
230
+ default: 'sequential-thinking',
231
+ },
232
+ {
233
+ name: 'url',
234
+ prompt: 'Server URL',
235
+ required: true,
236
+ default: 'https://remote.mcpservers.org/sequentialthinking/mcp',
237
+ },
238
+ ],
239
+ buildConfig: answers => ({
240
+ name: answers.serverName || 'sequential-thinking',
241
+ transport: 'http' as McpTransportType,
242
+ url:
243
+ answers.url || 'https://remote.mcpservers.org/sequentialthinking/mcp',
244
+ description:
245
+ 'Dynamic and reflective problem-solving through thought sequences.',
246
+ tags: ['remote', 'reasoning', 'analysis', 'http'],
247
+ timeout: TIMEOUT_MCP_DEFAULT_MS,
248
+ }),
249
+ category: 'remote',
250
+ transportType: 'http',
251
+ },
252
+ {
253
+ id: 'context7',
254
+ name: 'Context7',
255
+ description: 'Up-to-date code documentation for LLMs and AI code editors.',
256
+ command: '',
257
+ fields: [
258
+ {
259
+ name: 'serverName',
260
+ prompt: 'Server name',
261
+ required: true,
262
+ default: 'context7',
263
+ },
264
+ {
265
+ name: 'url',
266
+ prompt: 'Server URL',
267
+ required: true,
268
+ default: 'https://mcp.context7.com/mcp',
269
+ },
270
+ ],
271
+ buildConfig: answers => ({
272
+ name: answers.serverName || 'context7',
273
+ transport: 'http' as McpTransportType,
274
+ url: answers.url || 'https://mcp.context7.com/mcp',
275
+ description:
276
+ 'Up-to-date code documentation for LLMs and AI code editors.',
277
+ tags: ['remote', 'context', 'information', 'http'],
278
+ timeout: TIMEOUT_MCP_DEFAULT_MS,
279
+ }),
280
+ category: 'remote',
281
+ transportType: 'http',
282
+ },
283
+ {
284
+ id: 'remote-fetch',
285
+ name: 'Fetch',
286
+ description: 'Web content fetching and conversion for efficient LLM usage',
287
+ command: '',
288
+ fields: [
289
+ {
290
+ name: 'serverName',
291
+ prompt: 'Server name',
292
+ required: true,
293
+ default: 'remote-fetch',
294
+ },
295
+ {
296
+ name: 'url',
297
+ prompt: 'Server URL',
298
+ required: true,
299
+ default: 'https://remote.mcpservers.org/fetch/mcp',
300
+ },
301
+ ],
302
+ buildConfig: answers => ({
303
+ name: answers.serverName || 'remote-fetch',
304
+ transport: 'http' as McpTransportType,
305
+ url: answers.url || 'https://remote.mcpservers.org/fetch/mcp',
306
+ description:
307
+ 'Web content fetching and conversion for efficient LLM usage',
308
+ tags: ['remote', 'http', 'scraping', 'fetch'],
309
+ timeout: TIMEOUT_MCP_DEFAULT_MS,
310
+ }),
311
+ category: 'remote',
312
+ transportType: 'http',
313
+ },
314
+ {
315
+ id: 'github-remote',
316
+ name: 'GitHub (Remote)',
317
+ description:
318
+ 'Remote GitHub MCP server for repository management and operations',
319
+ command: '',
320
+ fields: [
321
+ {
322
+ name: 'serverName',
323
+ prompt: 'Server name',
324
+ required: true,
325
+ default: 'github-remote',
326
+ },
327
+ {
328
+ name: 'githubToken',
329
+ prompt: 'GitHub Personal Access Token (requires repo, read:org scopes)',
330
+ required: true,
331
+ sensitive: true,
332
+ },
333
+ ],
334
+ buildConfig: answers => ({
335
+ name: answers.serverName || 'github-remote',
336
+ transport: 'http' as McpTransportType,
337
+ url: 'https://api.githubcopilot.com/mcp/',
338
+ description:
339
+ 'Remote GitHub MCP server for repository management and operations',
340
+ tags: ['remote', 'github', 'git', 'repository', 'http'],
341
+ timeout: TIMEOUT_MCP_DEFAULT_MS,
342
+ headers: {
343
+ Authorization: `Bearer ${answers.githubToken}`,
344
+ },
345
+ }),
346
+ category: 'remote',
347
+ transportType: 'http',
348
+ },
349
+ {
350
+ id: 'gitlab',
351
+ name: 'GitLab',
352
+ description: 'GitLab MCP server for repository management and operations',
353
+ command: 'npx',
354
+ fields: [
355
+ {
356
+ name: 'gitlabToken',
357
+ prompt: 'GitLab Personal Access Token',
358
+ required: true,
359
+ sensitive: true,
360
+ },
361
+ {
362
+ name: 'gitlabApiUrl',
363
+ prompt: 'GitLab API URL (default: https://gitlab.com/api/v4)',
364
+ required: false,
365
+ default: 'https://gitlab.com/api/v4',
366
+ },
367
+ ],
368
+ buildConfig: answers => ({
369
+ name: 'gitlab',
370
+ transport: 'stdio' as McpTransportType,
371
+ command: 'npx',
372
+ args: ['-y', '@zereight/mcp-gitlab'],
373
+ env: {
374
+ GITLAB_PERSONAL_ACCESS_TOKEN: answers.gitlabToken,
375
+ GITLAB_API_URL: answers.gitlabApiUrl || 'https://gitlab.com/api/v4',
376
+ },
377
+ description: 'GitLab MCP server for repository management and operations',
378
+ tags: ['gitlab', 'git', 'repository', 'stdio'],
379
+ }),
380
+ category: 'local',
381
+ transportType: 'stdio',
382
+ },
383
+ {
384
+ id: 'playwright',
385
+ name: 'Playwright',
386
+ description: 'Playwright MCP server for browser automation',
387
+ command: 'npx',
388
+ fields: [],
389
+ buildConfig: _answers => ({
390
+ name: 'playwright',
391
+ transport: 'stdio' as McpTransportType,
392
+ command: 'npx',
393
+ args: ['@playwright/mcp@latest'],
394
+ description: 'Playwright MCP server for browser automation',
395
+ tags: ['playwright', 'browser', 'automation', 'stdio'],
396
+ }),
397
+ category: 'local',
398
+ transportType: 'stdio',
399
+ },
400
+ {
401
+ id: 'chrome-devtools',
402
+ name: 'Chrome DevTools',
403
+ description: 'Chrome DevTools MCP server for browser automation',
404
+ command: 'npx',
405
+ fields: [
406
+ {
407
+ name: 'headless',
408
+ prompt: 'Run Chrome in headless mode? (true/false)',
409
+ required: false,
410
+ default: 'true',
411
+ },
412
+ ],
413
+ buildConfig: answers => ({
414
+ name: 'chrome-devtools',
415
+ transport: 'stdio' as McpTransportType,
416
+ command: 'npx',
417
+ args: [
418
+ '-y',
419
+ 'chrome-devtools-mcp@latest',
420
+ ...(answers.headless === 'true' ? ['--headless=true'] : []),
421
+ ],
422
+ description: 'Chrome DevTools MCP server for browser automation',
423
+ tags: ['chrome', 'devtools', 'browser', 'automation', 'stdio'],
424
+ }),
425
+ category: 'local',
426
+ transportType: 'stdio',
427
+ },
428
+ {
429
+ id: 'duckduckgo',
430
+ name: 'DuckDuckGo Search',
431
+ description: 'DuckDuckGo search MCP server',
432
+ command: 'uvx',
433
+ fields: [],
434
+ buildConfig: _answers => ({
435
+ name: 'duckduckgo',
436
+ transport: 'stdio' as McpTransportType,
437
+ command: 'uvx',
438
+ args: ['duckduckgo-mcp-server'],
439
+ description: 'DuckDuckGo search MCP server',
440
+ tags: ['duckduckgo', 'search', 'stdio'],
441
+ }),
442
+ category: 'local',
443
+ transportType: 'stdio',
444
+ },
445
+ {
446
+ id: 'git',
447
+ name: 'Git',
448
+ description: 'Git MCP server for local repository operations',
449
+ command: 'uvx',
450
+ fields: [
451
+ {
452
+ name: 'repositoryPath',
453
+ prompt: 'Path to Git repository',
454
+ required: true,
455
+ },
456
+ ],
457
+ buildConfig: answers => ({
458
+ name: 'git',
459
+ transport: 'stdio' as McpTransportType,
460
+ command: 'uvx',
461
+ args: ['mcp-server-git', '--repository', answers.repositoryPath],
462
+ description: 'Git MCP server for local repository operations',
463
+ tags: ['git', 'repository', 'stdio'],
464
+ }),
465
+ category: 'local',
466
+ transportType: 'stdio',
467
+ },
468
+ {
469
+ id: 'memory',
470
+ name: 'Memory',
471
+ description: 'Memory MCP server for persistent storage',
472
+ command: 'npx',
473
+ fields: [],
474
+ buildConfig: _answers => ({
475
+ name: 'memory',
476
+ transport: 'stdio' as McpTransportType,
477
+ command: 'npx',
478
+ args: ['-y', '@modelcontextprotocol/server-memory'],
479
+ description: 'Memory MCP server for persistent storage',
480
+ tags: ['memory', 'storage', 'stdio'],
481
+ }),
482
+ category: 'local',
483
+ transportType: 'stdio',
484
+ },
485
+ {
486
+ id: 'custom',
487
+ name: 'Custom MCP Server',
488
+ description: 'Custom MCP server configuration',
489
+ command: '',
490
+ fields: [
491
+ {
492
+ name: 'transport',
493
+ prompt: 'Transport type (stdio, http, websocket)',
494
+ required: true,
495
+ default: 'stdio',
496
+ },
497
+ {
498
+ name: 'serverName',
499
+ prompt: 'Server name',
500
+ required: true,
501
+ },
502
+ {
503
+ name: 'url',
504
+ prompt: 'Server URL (for http/websocket transports)',
505
+ required: false,
506
+ },
507
+ {
508
+ name: 'command',
509
+ prompt: 'Command (for stdio transport)',
510
+ required: false,
511
+ },
512
+ {
513
+ name: 'args',
514
+ prompt: 'Arguments (space-separated, for stdio transport)',
515
+ required: false,
516
+ },
517
+ {
518
+ name: 'envVars',
519
+ prompt: 'Environment variables (KEY=VALUE, one per line, optional)',
520
+ required: false,
521
+ },
522
+ ],
523
+ buildConfig: answers => {
524
+ const config: McpServerConfig = {
525
+ name: answers.serverName,
526
+ transport: (answers.transport || 'stdio') as McpTransportType,
527
+ description: 'Custom MCP server configuration',
528
+ tags: ['custom'],
529
+ };
530
+
531
+ // Configure based on transport type
532
+ const transport = answers.transport || 'stdio';
533
+ if (transport === 'stdio') {
534
+ if (!answers.command) {
535
+ throw new Error('Command is required for stdio transport');
536
+ }
537
+ config.command = answers.command;
538
+ config.args = answers.args
539
+ ? answers.args
540
+ .split(' ')
541
+ .map(arg => arg.trim())
542
+ .filter(Boolean)
543
+ : [];
544
+ } else if (transport === 'http' || transport === 'websocket') {
545
+ if (!answers.url) {
546
+ throw new Error('URL is required for http/websocket transports');
547
+ }
548
+ config.url = answers.url;
549
+ config.timeout = TIMEOUT_MCP_DEFAULT_MS;
550
+ }
551
+
552
+ if (answers.envVars) {
553
+ config.env = {};
554
+ const lines = answers.envVars.split('\n');
555
+ for (const line of lines) {
556
+ const trimmed = line.trim();
557
+ if (!trimmed) continue;
558
+ const [key, ...valueParts] = trimmed.split('=');
559
+ if (key && valueParts.length > 0) {
560
+ config.env[key.trim()] = valueParts.join('=').trim();
561
+ }
562
+ }
563
+ }
564
+
565
+ return config;
566
+ },
567
+ category: 'local', // Default to local, but can be remote based on transport
568
+ transportType: 'stdio', // Default to stdio, but can be http/websocket based on transport
569
+ },
570
+ ];
@@ -0,0 +1,152 @@
1
+ import test from 'ava';
2
+ import {PROVIDER_TEMPLATES} from './provider-templates.js';
3
+
4
+ test('ollama template: single model', t => {
5
+ const template = PROVIDER_TEMPLATES.find(t => t.id === 'ollama');
6
+ t.truthy(template);
7
+
8
+ const config = template!.buildConfig({
9
+ providerName: 'ollama',
10
+ baseUrl: 'http://localhost:11434/v1',
11
+ model: 'llama2',
12
+ });
13
+
14
+ t.deepEqual(config.models, ['llama2']);
15
+ });
16
+
17
+ test('ollama template: multiple comma-separated models', t => {
18
+ const template = PROVIDER_TEMPLATES.find(t => t.id === 'ollama');
19
+ t.truthy(template);
20
+
21
+ const config = template!.buildConfig({
22
+ providerName: 'ollama',
23
+ baseUrl: 'http://localhost:11434/v1',
24
+ model: 'llama2, codellama, mistral',
25
+ });
26
+
27
+ t.deepEqual(config.models, ['llama2', 'codellama', 'mistral']);
28
+ });
29
+
30
+ test('ollama template: handles extra whitespace', t => {
31
+ const template = PROVIDER_TEMPLATES.find(t => t.id === 'ollama');
32
+ t.truthy(template);
33
+
34
+ const config = template!.buildConfig({
35
+ providerName: 'ollama',
36
+ baseUrl: 'http://localhost:11434/v1',
37
+ model: ' llama2 , codellama , mistral ',
38
+ });
39
+
40
+ t.deepEqual(config.models, ['llama2', 'codellama', 'mistral']);
41
+ });
42
+
43
+ test('ollama template: filters empty strings', t => {
44
+ const template = PROVIDER_TEMPLATES.find(t => t.id === 'ollama');
45
+ t.truthy(template);
46
+
47
+ const config = template!.buildConfig({
48
+ providerName: 'ollama',
49
+ baseUrl: 'http://localhost:11434/v1',
50
+ model: 'llama2,,codellama,',
51
+ });
52
+
53
+ t.deepEqual(config.models, ['llama2', 'codellama']);
54
+ });
55
+
56
+ test('custom template: single model', t => {
57
+ const template = PROVIDER_TEMPLATES.find(t => t.id === 'custom');
58
+ t.truthy(template);
59
+
60
+ const config = template!.buildConfig({
61
+ providerName: 'custom-provider',
62
+ baseUrl: 'http://localhost:8000/v1',
63
+ model: 'my-model',
64
+ });
65
+
66
+ t.deepEqual(config.models, ['my-model']);
67
+ });
68
+
69
+ test('custom template: multiple comma-separated models', t => {
70
+ const template = PROVIDER_TEMPLATES.find(t => t.id === 'custom');
71
+ t.truthy(template);
72
+
73
+ const config = template!.buildConfig({
74
+ providerName: 'custom-provider',
75
+ baseUrl: 'http://localhost:8000/v1',
76
+ model: 'model1, model2, model3',
77
+ });
78
+
79
+ t.deepEqual(config.models, ['model1', 'model2', 'model3']);
80
+ });
81
+
82
+ test('openrouter template: single model', t => {
83
+ const template = PROVIDER_TEMPLATES.find(t => t.id === 'openrouter');
84
+ t.truthy(template);
85
+
86
+ const config = template!.buildConfig({
87
+ providerName: 'OpenRouter',
88
+ apiKey: 'test-key',
89
+ model: 'z-ai/glm-4.7',
90
+ });
91
+
92
+ t.deepEqual(config.models, ['z-ai/glm-4.7']);
93
+ });
94
+
95
+ test('openrouter template: multiple comma-separated models', t => {
96
+ const template = PROVIDER_TEMPLATES.find(t => t.id === 'openrouter');
97
+ t.truthy(template);
98
+
99
+ const config = template!.buildConfig({
100
+ providerName: 'OpenRouter',
101
+ apiKey: 'test-key',
102
+ model: 'z-ai/glm-4.7, anthropic/claude-3-opus, openai/gpt-4',
103
+ });
104
+
105
+ t.deepEqual(config.models, [
106
+ 'z-ai/glm-4.7',
107
+ 'anthropic/claude-3-opus',
108
+ 'openai/gpt-4',
109
+ ]);
110
+ });
111
+
112
+ test('openai template: preserves organizationId', t => {
113
+ const template = PROVIDER_TEMPLATES.find(t => t.id === 'openai');
114
+ t.truthy(template);
115
+
116
+ const config = template!.buildConfig({
117
+ providerName: 'openai',
118
+ apiKey: 'test-key',
119
+ model: 'gpt-5-codex',
120
+ organizationId: 'org-123',
121
+ });
122
+
123
+ t.is(config.organizationId, 'org-123');
124
+ t.deepEqual(config.models, ['gpt-5-codex']);
125
+ });
126
+
127
+ test('openai template: handles multiple models', t => {
128
+ const template = PROVIDER_TEMPLATES.find(t => t.id === 'openai');
129
+ t.truthy(template);
130
+
131
+ const config = template!.buildConfig({
132
+ providerName: 'openai',
133
+ apiKey: 'test-key',
134
+ model: 'gpt-5-codex, gpt-4-turbo, gpt-4',
135
+ });
136
+
137
+ t.deepEqual(config.models, ['gpt-5-codex', 'gpt-4-turbo', 'gpt-4']);
138
+ });
139
+
140
+ test('custom template: includes timeout', t => {
141
+ const template = PROVIDER_TEMPLATES.find(t => t.id === 'custom');
142
+ t.truthy(template);
143
+
144
+ const config = template!.buildConfig({
145
+ providerName: 'custom-provider',
146
+ baseUrl: 'http://localhost:8000/v1',
147
+ model: 'my-model',
148
+ timeout: '60000',
149
+ });
150
+
151
+ t.is(config.timeout, 60000);
152
+ });