@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,330 @@
1
+ import {CACHE_MODELS_EXPIRATION_MS} from '@/constants';
2
+ import {ModelEntry} from '@/types/index';
3
+ import {logError} from '@/utils/message-queue';
4
+
5
+ // Cache for fetched model data
6
+ interface ModelCache {
7
+ models: ModelEntry[];
8
+ timestamp: number;
9
+ }
10
+
11
+ // OpenRouter API response structure
12
+ interface OpenRouterModel {
13
+ id: string;
14
+ name: string;
15
+ description: string;
16
+ created: number;
17
+ context_length: number;
18
+ architecture: {
19
+ modality: string;
20
+ input_modalities: string[];
21
+ output_modalities: string[];
22
+ tokenizer: string;
23
+ };
24
+ pricing: {
25
+ prompt: string;
26
+ completion: string;
27
+ };
28
+ supported_parameters?: string[];
29
+ }
30
+
31
+ interface OpenRouterResponse {
32
+ data: OpenRouterModel[];
33
+ }
34
+
35
+ const OPENROUTER_API = 'https://openrouter.ai/api/v1/models';
36
+
37
+ let modelCache: ModelCache | null = null;
38
+
39
+ /**
40
+ * Known open-weight model prefixes/patterns
41
+ */
42
+ const OPEN_WEIGHT_PATTERNS = [
43
+ 'meta-llama/',
44
+ 'mistralai/',
45
+ 'qwen/',
46
+ 'deepseek/',
47
+ 'google/gemma',
48
+ 'microsoft/phi',
49
+ 'nvidia/',
50
+ 'cohere/command-r',
51
+ 'databricks/',
52
+ 'allenai/',
53
+ 'huggingfaceh4/',
54
+ 'openchat/',
55
+ 'teknium/',
56
+ 'nousresearch/',
57
+ 'cognitivecomputations/',
58
+ 'thebloke/',
59
+ 'codellama/',
60
+ '/llama',
61
+ '/mistral',
62
+ '/qwen',
63
+ '/deepseek',
64
+ '/gemma',
65
+ '/phi-',
66
+ '/wizardlm',
67
+ '/vicuna',
68
+ '/falcon',
69
+ '/starcoder',
70
+ '/codestral',
71
+ '/devstral',
72
+ ];
73
+
74
+ /**
75
+ * Check if a model is open-weight based on ID
76
+ */
77
+ function isOpenWeight(modelId: string): boolean {
78
+ const lowerId = modelId.toLowerCase();
79
+ return OPEN_WEIGHT_PATTERNS.some(pattern => lowerId.includes(pattern));
80
+ }
81
+
82
+ /**
83
+ * Extract author from model ID
84
+ */
85
+ function extractAuthor(modelId: string): string {
86
+ const parts = modelId.split('/');
87
+ if (parts.length >= 2) {
88
+ const author = parts[0];
89
+ // Capitalize first letter
90
+ return author.charAt(0).toUpperCase() + author.slice(1);
91
+ }
92
+ return 'Unknown';
93
+ }
94
+
95
+ /**
96
+ * Check if model supports text input and output
97
+ */
98
+ function supportsTextInTextOut(model: OpenRouterModel): boolean {
99
+ const inputMods = model.architecture?.input_modalities || [];
100
+ const outputMods = model.architecture?.output_modalities || [];
101
+
102
+ const hasTextInput = inputMods.length === 0 || inputMods.includes('text');
103
+ const hasTextOutput = outputMods.length === 0 || outputMods.includes('text');
104
+
105
+ return hasTextInput && hasTextOutput;
106
+ }
107
+
108
+ /**
109
+ * Check if model is relevant for coding (has tool support or is a known coding model)
110
+ */
111
+ function isRelevantForCoding(model: OpenRouterModel): boolean {
112
+ const modelId = model.id.toLowerCase();
113
+ const modelName = model.name.toLowerCase();
114
+ const description = (model.description || '').toLowerCase();
115
+
116
+ // Must support text in/out
117
+ if (!supportsTextInTextOut(model)) return false;
118
+
119
+ // Skip embedding models
120
+ if (modelId.includes('embed') || modelName.includes('embed')) return false;
121
+
122
+ // Skip image/audio only models
123
+ if (
124
+ modelId.includes('dall-e') ||
125
+ modelId.includes('stable-diffusion') ||
126
+ modelId.includes('whisper') ||
127
+ modelId.includes('tts') ||
128
+ modelId.includes('imagen')
129
+ ) {
130
+ return false;
131
+ }
132
+
133
+ // Has tool calling support
134
+ if (model.supported_parameters?.includes('tools')) return true;
135
+
136
+ // Known coding-capable models
137
+ const codingPatterns = [
138
+ 'gpt',
139
+ 'claude',
140
+ 'gemini',
141
+ 'deepseek',
142
+ 'qwen',
143
+ 'coder',
144
+ 'codestral',
145
+ 'devstral',
146
+ 'llama',
147
+ 'mistral',
148
+ 'phi',
149
+ 'command',
150
+ 'grok',
151
+ 'codex',
152
+ ];
153
+
154
+ if (codingPatterns.some(p => modelId.includes(p) || modelName.includes(p))) {
155
+ return true;
156
+ }
157
+
158
+ // Description mentions coding
159
+ if (
160
+ description.includes('code') ||
161
+ description.includes('programming') ||
162
+ description.includes('developer')
163
+ ) {
164
+ return true;
165
+ }
166
+
167
+ return false;
168
+ }
169
+
170
+ /**
171
+ * Calculate cost score (0-10 scale, 10 = free/cheapest)
172
+ */
173
+ function calculateCostScore(
174
+ promptPrice: string,
175
+ completionPrice: string,
176
+ ): number {
177
+ const input = parseFloat(promptPrice) || 0;
178
+ const output = parseFloat(completionPrice) || 0;
179
+
180
+ // Price per million tokens (prices are per token)
181
+ const inputPerMillion = input * 1_000_000;
182
+ const outputPerMillion = output * 1_000_000;
183
+
184
+ // Weighted average (output typically more important)
185
+ const avgCost = inputPerMillion * 0.3 + outputPerMillion * 0.7;
186
+
187
+ if (avgCost === 0) return 10;
188
+ if (avgCost < 0.1) return 9;
189
+ if (avgCost < 0.5) return 8;
190
+ if (avgCost < 1) return 7;
191
+ if (avgCost < 2) return 6;
192
+ if (avgCost < 5) return 5;
193
+ if (avgCost < 10) return 4;
194
+ if (avgCost < 20) return 3;
195
+ if (avgCost < 50) return 2;
196
+ return 1;
197
+ }
198
+
199
+ /**
200
+ * Format cost details string
201
+ */
202
+ function formatCostDetails(
203
+ promptPrice: string,
204
+ completionPrice: string,
205
+ isLocal: boolean,
206
+ ): string {
207
+ const input = parseFloat(promptPrice) || 0;
208
+ const output = parseFloat(completionPrice) || 0;
209
+
210
+ if (input === 0 && output === 0) {
211
+ return isLocal ? 'Free (open weights)' : 'Free';
212
+ }
213
+
214
+ // Convert to per million tokens
215
+ const inputPerMillion = input * 1_000_000;
216
+ const outputPerMillion = output * 1_000_000;
217
+
218
+ return `$${inputPerMillion.toFixed(2)}/M in, $${outputPerMillion.toFixed(
219
+ 2,
220
+ )}/M out`;
221
+ }
222
+
223
+ /**
224
+ * Format context length for display
225
+ */
226
+ function formatContextLength(contextLength: number): string {
227
+ if (contextLength >= 1_000_000) {
228
+ return `${(contextLength / 1_000_000).toFixed(1)}M`;
229
+ }
230
+ if (contextLength >= 1000) {
231
+ return `${Math.round(contextLength / 1000)}K`;
232
+ }
233
+ return `${contextLength}`;
234
+ }
235
+
236
+ /**
237
+ * Fetch models from OpenRouter API
238
+ */
239
+ async function fetchOpenRouterModels(): Promise<OpenRouterModel[]> {
240
+ try {
241
+ const response = await fetch(OPENROUTER_API);
242
+ if (!response.ok) {
243
+ throw new Error(`OpenRouter API returned ${response.status}`);
244
+ }
245
+
246
+ const data = (await response.json()) as OpenRouterResponse;
247
+ return data.data || [];
248
+ } catch (error) {
249
+ logError('Failed to fetch OpenRouter models', true, {error});
250
+ return [];
251
+ }
252
+ }
253
+
254
+ /**
255
+ * Fetch and process models from OpenRouter
256
+ */
257
+ export async function fetchModels(): Promise<ModelEntry[]> {
258
+ // Check cache first
259
+ if (
260
+ modelCache &&
261
+ Date.now() - modelCache.timestamp < CACHE_MODELS_EXPIRATION_MS
262
+ ) {
263
+ return modelCache.models;
264
+ }
265
+
266
+ const openRouterModels = await fetchOpenRouterModels();
267
+ const models: ModelEntry[] = [];
268
+
269
+ for (const model of openRouterModels) {
270
+ // Skip if not relevant for coding
271
+ if (!isRelevantForCoding(model)) continue;
272
+
273
+ const isLocal = isOpenWeight(model.id);
274
+ const costScore = calculateCostScore(
275
+ model.pricing.prompt,
276
+ model.pricing.completion,
277
+ );
278
+
279
+ const entry: ModelEntry = {
280
+ id: model.id,
281
+ name: model.name,
282
+ author: extractAuthor(model.id),
283
+ size: formatContextLength(model.context_length),
284
+ local: isLocal,
285
+ api: true,
286
+ contextLength: model.context_length,
287
+ created: model.created,
288
+ quality: {
289
+ cost: costScore,
290
+ },
291
+ costType: costScore >= 9 ? ('free' as const) : ('paid' as const),
292
+ costDetails: formatCostDetails(
293
+ model.pricing.prompt,
294
+ model.pricing.completion,
295
+ isLocal,
296
+ ),
297
+ hasToolSupport: model.supported_parameters?.includes('tools') || false,
298
+ };
299
+
300
+ models.push(entry);
301
+ }
302
+
303
+ // Sort by created date (newest first)
304
+ models.sort((a, b) => (b.created || 0) - (a.created || 0));
305
+
306
+ // Update cache
307
+ modelCache = {
308
+ models,
309
+ timestamp: Date.now(),
310
+ };
311
+
312
+ return models;
313
+ }
314
+
315
+ /**
316
+ * Clear the model cache (useful for forcing a refresh)
317
+ */
318
+ export function clearModelCache(): void {
319
+ modelCache = null;
320
+ }
321
+
322
+ /**
323
+ * Check if models are currently cached
324
+ */
325
+ export function isModelsCached(): boolean {
326
+ return (
327
+ modelCache !== null &&
328
+ Date.now() - modelCache.timestamp < CACHE_MODELS_EXPIRATION_MS
329
+ );
330
+ }
@@ -0,0 +1 @@
1
+ export {getModelContextLimit} from './models-dev-client.js';
@@ -0,0 +1,214 @@
1
+ import test from 'ava';
2
+ import { unlink, writeFile, readFile, mkdir } from 'node:fs/promises';
3
+ import { existsSync } from 'node:fs';
4
+ import { join } from 'node:path';
5
+ import { tmpdir } from 'node:os';
6
+ import type { CachedModelsData } from './models-types.js';
7
+ import {
8
+ readCache,
9
+ writeCache,
10
+ } from './models-cache.js';
11
+
12
+ // Get the actual cache file path that will be used
13
+ // We need to import dynamically to get the actual path at runtime
14
+ let actualCachePath: string;
15
+
16
+ async function getCacheFilePath(): Promise<string> {
17
+ // Import dynamically to avoid module resolution issues during testing
18
+ const { xdgCache } = await import('xdg-basedir');
19
+ const path = await import('node:path');
20
+
21
+ const DEFAULT_CACHE_DIR =
22
+ process.platform === 'darwin'
23
+ ? path.join(process.env.HOME || '~', 'Library', 'Caches')
24
+ : path.join(process.env.HOME || '~', '.cache');
25
+
26
+ const cacheBase = xdgCache || DEFAULT_CACHE_DIR;
27
+ return path.join(cacheBase, 'coder', 'models.json');
28
+ }
29
+
30
+ // ============================================================================
31
+ // Setup and Teardown
32
+ // ============================================================================
33
+
34
+ test.before(async () => {
35
+ actualCachePath = await getCacheFilePath();
36
+ });
37
+
38
+ test.afterEach.always(async () => {
39
+ // Clean up test cache file after each test
40
+ if (actualCachePath && existsSync(actualCachePath)) {
41
+ try {
42
+ await unlink(actualCachePath);
43
+ } catch {
44
+ // Ignore cleanup errors
45
+ }
46
+ }
47
+ });
48
+
49
+ // ============================================================================
50
+ // Tests for readCache
51
+ // ============================================================================
52
+
53
+ test('readCache returns null when cache file does not exist', async t => {
54
+ // Ensure cache file doesn't exist
55
+ if (existsSync(actualCachePath)) {
56
+ await unlink(actualCachePath);
57
+ }
58
+
59
+ const result = await readCache();
60
+ t.is(result, null);
61
+ });
62
+
63
+ test('readCache returns valid cached data', async t => {
64
+ const mockData: CachedModelsData = {
65
+ data: { models: [] } as any,
66
+ fetchedAt: Date.now() - 1000,
67
+ expiresAt: Date.now() + 1000000, // Not expired
68
+ };
69
+
70
+ // Ensure directory exists
71
+ const dir = join(actualCachePath, '..');
72
+ try {
73
+ await mkdir(dir, { recursive: true });
74
+ } catch {
75
+ // Directory may already exist
76
+ }
77
+
78
+ await writeFile(actualCachePath, JSON.stringify(mockData), 'utf-8');
79
+
80
+ const result = await readCache();
81
+
82
+ t.truthy(result);
83
+ t.deepEqual(result?.data, mockData.data);
84
+ t.is(result?.fetchedAt, mockData.fetchedAt);
85
+ t.is(result?.expiresAt, mockData.expiresAt);
86
+ });
87
+
88
+ test('readCache returns null when cache is expired', async t => {
89
+ const expiredData: CachedModelsData = {
90
+ data: { models: [] } as any,
91
+ fetchedAt: Date.now() - 1000000,
92
+ expiresAt: Date.now() - 1000, // Expired
93
+ };
94
+
95
+ // Ensure directory exists
96
+ const dir = join(actualCachePath, '..');
97
+ try {
98
+ await mkdir(dir, { recursive: true });
99
+ } catch {
100
+ // Directory may already exist
101
+ }
102
+
103
+ await writeFile(actualCachePath, JSON.stringify(expiredData), 'utf-8');
104
+
105
+ const result = await readCache();
106
+
107
+ t.is(result, null);
108
+ });
109
+
110
+ test('readCache returns null on JSON parse error', async t => {
111
+ // Ensure directory exists
112
+ const dir = join(actualCachePath, '..');
113
+ try {
114
+ await mkdir(dir, { recursive: true });
115
+ } catch {
116
+ // Directory may already exist
117
+ }
118
+
119
+ await writeFile(actualCachePath, 'invalid json{{{', 'utf-8');
120
+
121
+ const result = await readCache();
122
+
123
+ t.is(result, null);
124
+ });
125
+
126
+ // ============================================================================
127
+ // Tests for writeCache
128
+ // ============================================================================
129
+
130
+ test('writeCache writes cache data correctly', async t => {
131
+ const mockData = { models: [{ id: 'test-model' }] } as any;
132
+
133
+ await writeCache(mockData);
134
+
135
+ // Verify the file was written
136
+ const exists = existsSync(actualCachePath);
137
+ t.true(exists);
138
+
139
+ // Verify the content
140
+ const content = await readFile(actualCachePath, 'utf-8');
141
+ const writtenData = JSON.parse(content) as CachedModelsData;
142
+
143
+ t.truthy(writtenData.data);
144
+ t.truthy(writtenData.fetchedAt);
145
+ t.truthy(writtenData.expiresAt);
146
+ t.true(writtenData.expiresAt > Date.now());
147
+ });
148
+
149
+ test('writeCache creates directory if it does not exist', async t => {
150
+ const mockData = { models: [] } as any;
151
+
152
+ // This should not throw even if directory doesn't exist
153
+ await t.notThrowsAsync(async () => {
154
+ await writeCache(mockData);
155
+ });
156
+
157
+ // Verify the file was created
158
+ t.true(existsSync(actualCachePath));
159
+ });
160
+
161
+ // ============================================================================
162
+ // Integration-like tests
163
+ // ============================================================================
164
+
165
+ test('cache expiration is calculated correctly', async t => {
166
+ const mockData = { models: [] } as any;
167
+ const now = Date.now();
168
+
169
+ await writeCache(mockData);
170
+
171
+ const content = await readFile(actualCachePath, 'utf-8');
172
+ const writtenData = JSON.parse(content) as CachedModelsData;
173
+
174
+ // Get the expiration constant
175
+ const { CACHE_MODELS_EXPIRATION_MS } = await import('@/constants');
176
+
177
+ // Expiration should be roughly CACHE_MODELS_EXPIRATION_MS in the future
178
+ t.true(writtenData.expiresAt >= now + CACHE_MODELS_EXPIRATION_MS - 100);
179
+ t.true(writtenData.expiresAt <= now + CACHE_MODELS_EXPIRATION_MS + 100);
180
+ });
181
+
182
+ test('readCache and writeCache round-trip correctly', async t => {
183
+ const originalData = {
184
+ models: [
185
+ { id: 'model1', name: 'Test Model 1' },
186
+ { id: 'model2', name: 'Test Model 2' },
187
+ ],
188
+ } as any;
189
+
190
+ // Write the cache
191
+ await writeCache(originalData);
192
+
193
+ // Read it back
194
+ const readResult = await readCache();
195
+
196
+ t.truthy(readResult);
197
+ t.deepEqual(readResult?.data, originalData);
198
+ });
199
+
200
+ test('writeCache overwrites existing cache', async t => {
201
+ // Write first cache
202
+ const firstData = { models: [{ id: 'model1' }] } as any;
203
+ await writeCache(firstData);
204
+
205
+ // Write second cache (should overwrite)
206
+ const secondData = { models: [{ id: 'model2' }, { id: 'model3' }] } as any;
207
+ await writeCache(secondData);
208
+
209
+ // Read back
210
+ const readResult = await readCache();
211
+
212
+ t.truthy(readResult);
213
+ t.deepEqual(readResult?.data, secondData);
214
+ });
@@ -0,0 +1,78 @@
1
+ /**
2
+ * Cache management for models.dev data
3
+ * Stores model database in XDG_CACHE_HOME for fast lookup
4
+ */
5
+
6
+ import {constants} from 'node:fs';
7
+ import {access, mkdir, readFile, writeFile} from 'node:fs/promises';
8
+ import * as path from 'node:path';
9
+ import {CACHE_MODELS_EXPIRATION_MS} from '@/constants';
10
+ import {formatError} from '@/utils/error-formatter';
11
+ import {getLogger} from '@/utils/logging';
12
+ import {xdgCache} from 'xdg-basedir';
13
+ import type {CachedModelsData, ModelsDevDatabase} from './models-types.js';
14
+
15
+ const DEFAULT_CACHE_DIR =
16
+ process.platform === 'darwin'
17
+ ? path.join(process.env.HOME || '~', 'Library', 'Caches')
18
+ : path.join(process.env.HOME || '~', '.cache');
19
+
20
+ function getCacheDir(): string {
21
+ const cacheBase = xdgCache || DEFAULT_CACHE_DIR;
22
+ return path.join(cacheBase, 'coder');
23
+ }
24
+
25
+ function getCacheFilePath(): string {
26
+ return path.join(getCacheDir(), 'models.json');
27
+ }
28
+
29
+ async function ensureCacheDir(): Promise<void> {
30
+ const dir = getCacheDir();
31
+
32
+ try {
33
+ await access(dir, constants.F_OK);
34
+ } catch {
35
+ await mkdir(dir, {recursive: true});
36
+ }
37
+ }
38
+
39
+ export async function readCache(): Promise<CachedModelsData | null> {
40
+ try {
41
+ const cachePath = getCacheFilePath();
42
+
43
+ await access(cachePath, constants.F_OK);
44
+
45
+ const content = await readFile(cachePath, 'utf-8');
46
+ const cached = JSON.parse(content) as CachedModelsData;
47
+
48
+ // Check if cache is expired
49
+ if (Date.now() > cached.expiresAt) {
50
+ return null;
51
+ }
52
+
53
+ return cached;
54
+ } catch (error) {
55
+ // If there's any error reading cache, return null to trigger fresh fetch
56
+ const logger = getLogger();
57
+ logger.warn({error: formatError(error)}, 'Failed to read models cache');
58
+ return null;
59
+ }
60
+ }
61
+
62
+ export async function writeCache(data: ModelsDevDatabase): Promise<void> {
63
+ try {
64
+ await ensureCacheDir();
65
+
66
+ const cached: CachedModelsData = {
67
+ data,
68
+ fetchedAt: Date.now(),
69
+ expiresAt: Date.now() + CACHE_MODELS_EXPIRATION_MS,
70
+ };
71
+
72
+ const cachePath = getCacheFilePath();
73
+ await writeFile(cachePath, JSON.stringify(cached, null, 2), 'utf-8');
74
+ } catch (error) {
75
+ const logger = getLogger();
76
+ logger.warn({error: formatError(error)}, 'Failed to write models cache');
77
+ }
78
+ }