@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,758 @@
1
+ import test from 'ava';
2
+ import {render} from 'ink-testing-library';
3
+ import React from 'react';
4
+ import {themes} from '../config/themes.js';
5
+ import {ThemeContext} from '../hooks/useTheme.js';
6
+ import {clearModelCache} from '../model-database/model-fetcher.js';
7
+ import {ModelDatabaseDisplay, modelDatabaseCommand} from './model-database.js';
8
+
9
+ console.log(`\nmodel-database.spec.tsx – ${React.version}`);
10
+
11
+ // Mock ThemeProvider for testing
12
+ const MockThemeProvider = ({children}: {children: React.ReactNode}) => {
13
+ const mockTheme = {
14
+ currentTheme: 'tokyo-night' as const,
15
+ colors: themes['tokyo-night'].colors,
16
+ setCurrentTheme: () => {},
17
+ };
18
+
19
+ return (
20
+ <ThemeContext.Provider value={mockTheme}>{children}</ThemeContext.Provider>
21
+ );
22
+ };
23
+
24
+ // Helper to create mock OpenRouter API response
25
+ function createMockOpenRouterResponse() {
26
+ return {
27
+ data: [
28
+ {
29
+ id: 'openai/gpt-4',
30
+ name: 'GPT-4',
31
+ description: 'Advanced AI model',
32
+ created: 1700000000,
33
+ context_length: 128000,
34
+ architecture: {
35
+ modality: 'text',
36
+ input_modalities: ['text'],
37
+ output_modalities: ['text'],
38
+ tokenizer: 'unknown',
39
+ },
40
+ pricing: {prompt: '0.00003', completion: '0.00006'},
41
+ supported_parameters: ['tools'],
42
+ },
43
+ {
44
+ id: 'meta-llama/llama-3.1-70b',
45
+ name: 'Llama 3.1 70B',
46
+ description: 'Open source model',
47
+ created: 1710000000,
48
+ context_length: 128000,
49
+ architecture: {
50
+ modality: 'text',
51
+ input_modalities: ['text'],
52
+ output_modalities: ['text'],
53
+ tokenizer: 'unknown',
54
+ },
55
+ pricing: {prompt: '0', completion: '0'},
56
+ supported_parameters: ['tools'],
57
+ },
58
+ {
59
+ id: 'anthropic/claude-3-opus',
60
+ name: 'Claude 3 Opus',
61
+ description: 'Powerful reasoning model',
62
+ created: 1705000000,
63
+ context_length: 200000,
64
+ architecture: {
65
+ modality: 'text',
66
+ input_modalities: ['text'],
67
+ output_modalities: ['text'],
68
+ tokenizer: 'unknown',
69
+ },
70
+ pricing: {prompt: '0.000015', completion: '0.000075'},
71
+ supported_parameters: ['tools'],
72
+ },
73
+ ],
74
+ };
75
+ }
76
+
77
+ // ============================================================================
78
+ // Command Metadata Tests
79
+ // ============================================================================
80
+
81
+ test('modelDatabaseCommand: has correct name', t => {
82
+ t.is(modelDatabaseCommand.name, 'model-database');
83
+ });
84
+
85
+ test('modelDatabaseCommand: has description', t => {
86
+ t.truthy(modelDatabaseCommand.description);
87
+ t.true(modelDatabaseCommand.description.length > 0);
88
+ });
89
+
90
+ test('modelDatabaseCommand: has handler function', t => {
91
+ t.is(typeof modelDatabaseCommand.handler, 'function');
92
+ });
93
+
94
+ test('modelDatabaseCommand: handler is async', t => {
95
+ const result = modelDatabaseCommand.handler([], [], {});
96
+ t.truthy(result);
97
+ t.true(result instanceof Promise);
98
+ });
99
+
100
+ test('modelDatabaseCommand: handler returns React element', async t => {
101
+ const result = await modelDatabaseCommand.handler([], [], {});
102
+ t.truthy(result);
103
+ t.true(React.isValidElement(result));
104
+ });
105
+
106
+ // ============================================================================
107
+ // Component Rendering Tests - Loading State
108
+ // ============================================================================
109
+
110
+ test.serial('ModelDatabaseDisplay: shows loading state initially', async t => {
111
+ clearModelCache();
112
+
113
+ // Mock fetch to delay response
114
+ const originalFetch = globalThis.fetch;
115
+ let resolveFetch: (value: Response) => void;
116
+ const fetchPromise = new Promise<Response>(resolve => {
117
+ resolveFetch = resolve;
118
+ });
119
+ globalThis.fetch = () => fetchPromise;
120
+
121
+ try {
122
+ const {lastFrame} = render(
123
+ <MockThemeProvider>
124
+ <ModelDatabaseDisplay />
125
+ </MockThemeProvider>,
126
+ );
127
+
128
+ const output = lastFrame();
129
+ t.truthy(output);
130
+ t.regex(output!, /Fetching models|OpenRouter/i);
131
+ } finally {
132
+ // Resolve the fetch to clean up
133
+ resolveFetch!({
134
+ ok: true,
135
+ json: async () => createMockOpenRouterResponse(),
136
+ } as Response);
137
+ globalThis.fetch = originalFetch;
138
+ clearModelCache();
139
+ }
140
+ });
141
+
142
+ // ============================================================================
143
+ // Component Rendering Tests - Error State
144
+ // ============================================================================
145
+
146
+ test.serial(
147
+ 'ModelDatabaseDisplay: shows error state on fetch failure',
148
+ async t => {
149
+ clearModelCache();
150
+
151
+ const originalFetch = globalThis.fetch;
152
+ globalThis.fetch = async () => {
153
+ return {
154
+ ok: false,
155
+ status: 500,
156
+ } as Response;
157
+ };
158
+
159
+ try {
160
+ const {lastFrame} = render(
161
+ <MockThemeProvider>
162
+ <ModelDatabaseDisplay />
163
+ </MockThemeProvider>,
164
+ );
165
+
166
+ // Wait for fetch to complete
167
+ await new Promise(resolve => setTimeout(resolve, 100));
168
+
169
+ const output = lastFrame();
170
+ t.truthy(output);
171
+ // After fetch failure, it may show empty state or error
172
+ // The component shows "No models available" if fetch returns empty array
173
+ } finally {
174
+ globalThis.fetch = originalFetch;
175
+ clearModelCache();
176
+ }
177
+ },
178
+ );
179
+
180
+ // ============================================================================
181
+ // Component Rendering Tests - Loaded State
182
+ // ============================================================================
183
+
184
+ test.serial('ModelDatabaseDisplay: renders with model data', async t => {
185
+ clearModelCache();
186
+
187
+ const originalFetch = globalThis.fetch;
188
+ globalThis.fetch = async () => {
189
+ return {
190
+ ok: true,
191
+ json: async () => createMockOpenRouterResponse(),
192
+ } as Response;
193
+ };
194
+
195
+ try {
196
+ const {lastFrame} = render(
197
+ <MockThemeProvider>
198
+ <ModelDatabaseDisplay />
199
+ </MockThemeProvider>,
200
+ );
201
+
202
+ // Wait for async fetch to complete
203
+ await new Promise(resolve => setTimeout(resolve, 200));
204
+
205
+ const output = lastFrame();
206
+ t.truthy(output);
207
+ } finally {
208
+ globalThis.fetch = originalFetch;
209
+ clearModelCache();
210
+ }
211
+ });
212
+
213
+ test.serial('ModelDatabaseDisplay: displays model name', async t => {
214
+ clearModelCache();
215
+
216
+ const originalFetch = globalThis.fetch;
217
+ globalThis.fetch = async () => {
218
+ return {
219
+ ok: true,
220
+ json: async () => createMockOpenRouterResponse(),
221
+ } as Response;
222
+ };
223
+
224
+ try {
225
+ const {lastFrame} = render(
226
+ <MockThemeProvider>
227
+ <ModelDatabaseDisplay />
228
+ </MockThemeProvider>,
229
+ );
230
+
231
+ await new Promise(resolve => setTimeout(resolve, 200));
232
+
233
+ const output = lastFrame();
234
+ t.truthy(output);
235
+ // Should display one of the model names
236
+ t.true(
237
+ output!.includes('GPT-4') ||
238
+ output!.includes('Llama') ||
239
+ output!.includes('Claude'),
240
+ );
241
+ } finally {
242
+ globalThis.fetch = originalFetch;
243
+ clearModelCache();
244
+ }
245
+ });
246
+
247
+ test.serial('ModelDatabaseDisplay: displays tab names', async t => {
248
+ clearModelCache();
249
+
250
+ const originalFetch = globalThis.fetch;
251
+ globalThis.fetch = async () => {
252
+ return {
253
+ ok: true,
254
+ json: async () => createMockOpenRouterResponse(),
255
+ } as Response;
256
+ };
257
+
258
+ try {
259
+ const {lastFrame} = render(
260
+ <MockThemeProvider>
261
+ <ModelDatabaseDisplay />
262
+ </MockThemeProvider>,
263
+ );
264
+
265
+ await new Promise(resolve => setTimeout(resolve, 200));
266
+
267
+ const output = lastFrame();
268
+ t.truthy(output);
269
+ t.regex(output!, /Latest|Open|Proprietary/);
270
+ } finally {
271
+ globalThis.fetch = originalFetch;
272
+ clearModelCache();
273
+ }
274
+ });
275
+
276
+ test.serial(
277
+ 'ModelDatabaseDisplay: displays OpenRouter attribution',
278
+ async t => {
279
+ clearModelCache();
280
+
281
+ const originalFetch = globalThis.fetch;
282
+ globalThis.fetch = async () => {
283
+ return {
284
+ ok: true,
285
+ json: async () => createMockOpenRouterResponse(),
286
+ } as Response;
287
+ };
288
+
289
+ try {
290
+ const {lastFrame} = render(
291
+ <MockThemeProvider>
292
+ <ModelDatabaseDisplay />
293
+ </MockThemeProvider>,
294
+ );
295
+
296
+ await new Promise(resolve => setTimeout(resolve, 200));
297
+
298
+ const output = lastFrame();
299
+ t.truthy(output);
300
+ t.regex(output!, /OpenRouter/);
301
+ } finally {
302
+ globalThis.fetch = originalFetch;
303
+ clearModelCache();
304
+ }
305
+ },
306
+ );
307
+
308
+ test.serial('ModelDatabaseDisplay: displays navigation help', async t => {
309
+ clearModelCache();
310
+
311
+ const originalFetch = globalThis.fetch;
312
+ globalThis.fetch = async () => {
313
+ return {
314
+ ok: true,
315
+ json: async () => createMockOpenRouterResponse(),
316
+ } as Response;
317
+ };
318
+
319
+ try {
320
+ const {lastFrame} = render(
321
+ <MockThemeProvider>
322
+ <ModelDatabaseDisplay />
323
+ </MockThemeProvider>,
324
+ );
325
+
326
+ await new Promise(resolve => setTimeout(resolve, 200));
327
+
328
+ const output = lastFrame();
329
+ t.truthy(output);
330
+ // Should show navigation instructions
331
+ t.regex(output!, /Navigate|Tab|Esc/i);
332
+ } finally {
333
+ globalThis.fetch = originalFetch;
334
+ clearModelCache();
335
+ }
336
+ });
337
+
338
+ // ============================================================================
339
+ // Component Rendering Tests - Model Details
340
+ // ============================================================================
341
+
342
+ test.serial('ModelDatabaseDisplay: displays model details', async t => {
343
+ clearModelCache();
344
+
345
+ const originalFetch = globalThis.fetch;
346
+ globalThis.fetch = async () => {
347
+ return {
348
+ ok: true,
349
+ json: async () => createMockOpenRouterResponse(),
350
+ } as Response;
351
+ };
352
+
353
+ try {
354
+ const {lastFrame} = render(
355
+ <MockThemeProvider>
356
+ <ModelDatabaseDisplay />
357
+ </MockThemeProvider>,
358
+ );
359
+
360
+ await new Promise(resolve => setTimeout(resolve, 200));
361
+
362
+ const output = lastFrame();
363
+ t.truthy(output);
364
+ // Should display model details like Author, Context, Type, Cost
365
+ t.regex(output!, /Author|Context|Type|Cost/);
366
+ } finally {
367
+ globalThis.fetch = originalFetch;
368
+ clearModelCache();
369
+ }
370
+ });
371
+
372
+ test.serial('ModelDatabaseDisplay: displays model count', async t => {
373
+ clearModelCache();
374
+
375
+ const originalFetch = globalThis.fetch;
376
+ globalThis.fetch = async () => {
377
+ return {
378
+ ok: true,
379
+ json: async () => createMockOpenRouterResponse(),
380
+ } as Response;
381
+ };
382
+
383
+ try {
384
+ const {lastFrame} = render(
385
+ <MockThemeProvider>
386
+ <ModelDatabaseDisplay />
387
+ </MockThemeProvider>,
388
+ );
389
+
390
+ await new Promise(resolve => setTimeout(resolve, 200));
391
+
392
+ const output = lastFrame();
393
+ t.truthy(output);
394
+ // Should display model count like "Model 1 of 3"
395
+ t.regex(output!, /Model \d+ of \d+|of \d+/);
396
+ } finally {
397
+ globalThis.fetch = originalFetch;
398
+ clearModelCache();
399
+ }
400
+ });
401
+
402
+ // ============================================================================
403
+ // onCancel Callback Tests
404
+ // ============================================================================
405
+
406
+ test.serial('ModelDatabaseDisplay: calls onCancel callback', async t => {
407
+ clearModelCache();
408
+
409
+ const originalFetch = globalThis.fetch;
410
+ globalThis.fetch = async () => {
411
+ return {
412
+ ok: true,
413
+ json: async () => createMockOpenRouterResponse(),
414
+ } as Response;
415
+ };
416
+
417
+ let cancelCalled = false;
418
+ const onCancel = () => {
419
+ cancelCalled = true;
420
+ };
421
+
422
+ try {
423
+ const {stdin} = render(
424
+ <MockThemeProvider>
425
+ <ModelDatabaseDisplay onCancel={onCancel} />
426
+ </MockThemeProvider>,
427
+ );
428
+
429
+ await new Promise(resolve => setTimeout(resolve, 200));
430
+
431
+ // Press Escape to cancel
432
+ stdin.write('\u001B'); // Escape key
433
+
434
+ await new Promise(resolve => setTimeout(resolve, 50));
435
+
436
+ t.true(cancelCalled);
437
+ } finally {
438
+ globalThis.fetch = originalFetch;
439
+ clearModelCache();
440
+ }
441
+ });
442
+
443
+ test.serial('ModelDatabaseDisplay: calls onCancel on Enter key', async t => {
444
+ clearModelCache();
445
+
446
+ const originalFetch = globalThis.fetch;
447
+ globalThis.fetch = async () => {
448
+ return {
449
+ ok: true,
450
+ json: async () => createMockOpenRouterResponse(),
451
+ } as Response;
452
+ };
453
+
454
+ let cancelCalled = false;
455
+ const onCancel = () => {
456
+ cancelCalled = true;
457
+ };
458
+
459
+ try {
460
+ const {stdin} = render(
461
+ <MockThemeProvider>
462
+ <ModelDatabaseDisplay onCancel={onCancel} />
463
+ </MockThemeProvider>,
464
+ );
465
+
466
+ await new Promise(resolve => setTimeout(resolve, 200));
467
+
468
+ // Press Enter to close
469
+ stdin.write('\r');
470
+
471
+ await new Promise(resolve => setTimeout(resolve, 50));
472
+
473
+ t.true(cancelCalled);
474
+ } finally {
475
+ globalThis.fetch = originalFetch;
476
+ clearModelCache();
477
+ }
478
+ });
479
+
480
+ // ============================================================================
481
+ // Keyboard Navigation Tests
482
+ // ============================================================================
483
+
484
+ test.serial('ModelDatabaseDisplay: handles arrow key navigation', async t => {
485
+ clearModelCache();
486
+
487
+ const originalFetch = globalThis.fetch;
488
+ globalThis.fetch = async () => {
489
+ return {
490
+ ok: true,
491
+ json: async () => createMockOpenRouterResponse(),
492
+ } as Response;
493
+ };
494
+
495
+ try {
496
+ const {stdin, lastFrame} = render(
497
+ <MockThemeProvider>
498
+ <ModelDatabaseDisplay />
499
+ </MockThemeProvider>,
500
+ );
501
+
502
+ await new Promise(resolve => setTimeout(resolve, 200));
503
+
504
+ const initialOutput = lastFrame();
505
+
506
+ // Press down arrow to navigate
507
+ stdin.write('\u001B[B'); // Down arrow
508
+
509
+ await new Promise(resolve => setTimeout(resolve, 50));
510
+
511
+ const afterDownOutput = lastFrame();
512
+
513
+ // The output should potentially show different model or counter
514
+ t.truthy(initialOutput);
515
+ t.truthy(afterDownOutput);
516
+ } finally {
517
+ globalThis.fetch = originalFetch;
518
+ clearModelCache();
519
+ }
520
+ });
521
+
522
+ // ============================================================================
523
+ // Search Mode Tests
524
+ // ============================================================================
525
+
526
+ test.serial('ModelDatabaseDisplay: enters search mode on typing', async t => {
527
+ clearModelCache();
528
+
529
+ const originalFetch = globalThis.fetch;
530
+ globalThis.fetch = async () => {
531
+ return {
532
+ ok: true,
533
+ json: async () => createMockOpenRouterResponse(),
534
+ } as Response;
535
+ };
536
+
537
+ try {
538
+ const {stdin, lastFrame} = render(
539
+ <MockThemeProvider>
540
+ <ModelDatabaseDisplay />
541
+ </MockThemeProvider>,
542
+ );
543
+
544
+ await new Promise(resolve => setTimeout(resolve, 200));
545
+
546
+ // Type a search query
547
+ stdin.write('g');
548
+
549
+ await new Promise(resolve => setTimeout(resolve, 50));
550
+
551
+ const output = lastFrame();
552
+ t.truthy(output);
553
+ // Should show search indicator or the typed character
554
+ t.regex(output!, /Search|g/i);
555
+ } finally {
556
+ globalThis.fetch = originalFetch;
557
+ clearModelCache();
558
+ }
559
+ });
560
+
561
+ test.serial(
562
+ 'ModelDatabaseDisplay: filters models based on search query',
563
+ async t => {
564
+ clearModelCache();
565
+
566
+ const originalFetch = globalThis.fetch;
567
+ globalThis.fetch = async () => {
568
+ return {
569
+ ok: true,
570
+ json: async () => createMockOpenRouterResponse(),
571
+ } as Response;
572
+ };
573
+
574
+ try {
575
+ const {stdin, lastFrame} = render(
576
+ <MockThemeProvider>
577
+ <ModelDatabaseDisplay />
578
+ </MockThemeProvider>,
579
+ );
580
+
581
+ await new Promise(resolve => setTimeout(resolve, 200));
582
+
583
+ // Type search query character by character with delays
584
+ stdin.write('l');
585
+ await new Promise(resolve => setTimeout(resolve, 30));
586
+ stdin.write('l');
587
+ await new Promise(resolve => setTimeout(resolve, 30));
588
+ stdin.write('a');
589
+ await new Promise(resolve => setTimeout(resolve, 30));
590
+ stdin.write('m');
591
+ await new Promise(resolve => setTimeout(resolve, 30));
592
+ stdin.write('a');
593
+
594
+ await new Promise(resolve => setTimeout(resolve, 100));
595
+
596
+ const output = lastFrame();
597
+ t.truthy(output);
598
+ // Should show search mode with query "llama"
599
+ t.regex(output!, /llama|Search/i);
600
+ } finally {
601
+ globalThis.fetch = originalFetch;
602
+ clearModelCache();
603
+ }
604
+ },
605
+ );
606
+
607
+ test.serial(
608
+ 'ModelDatabaseDisplay: search mode can be entered and exited',
609
+ async t => {
610
+ clearModelCache();
611
+
612
+ const originalFetch = globalThis.fetch;
613
+ globalThis.fetch = async () => {
614
+ return {
615
+ ok: true,
616
+ json: async () => createMockOpenRouterResponse(),
617
+ } as Response;
618
+ };
619
+
620
+ try {
621
+ const {stdin, lastFrame} = render(
622
+ <MockThemeProvider>
623
+ <ModelDatabaseDisplay />
624
+ </MockThemeProvider>,
625
+ );
626
+
627
+ await new Promise(resolve => setTimeout(resolve, 200));
628
+
629
+ // Get initial state
630
+ const initialOutput = lastFrame();
631
+ t.truthy(initialOutput);
632
+
633
+ // Enter search mode by typing
634
+ stdin.write('t');
635
+ await new Promise(resolve => setTimeout(resolve, 50));
636
+ stdin.write('e');
637
+ await new Promise(resolve => setTimeout(resolve, 50));
638
+
639
+ const searchOutput = lastFrame();
640
+ t.truthy(searchOutput);
641
+ // In search mode, should show search indicator
642
+ t.regex(searchOutput!, /Search|te/i);
643
+ } finally {
644
+ globalThis.fetch = originalFetch;
645
+ clearModelCache();
646
+ }
647
+ },
648
+ );
649
+
650
+ // ============================================================================
651
+ // Empty State Tests
652
+ // ============================================================================
653
+
654
+ test.serial('ModelDatabaseDisplay: handles empty model list', async t => {
655
+ clearModelCache();
656
+
657
+ const originalFetch = globalThis.fetch;
658
+ globalThis.fetch = async () => {
659
+ return {
660
+ ok: true,
661
+ json: async () => ({data: []}),
662
+ } as Response;
663
+ };
664
+
665
+ try {
666
+ const {lastFrame} = render(
667
+ <MockThemeProvider>
668
+ <ModelDatabaseDisplay />
669
+ </MockThemeProvider>,
670
+ );
671
+
672
+ await new Promise(resolve => setTimeout(resolve, 200));
673
+
674
+ const output = lastFrame();
675
+ t.truthy(output);
676
+ // Should show empty state message
677
+ t.regex(output!, /No models|available/i);
678
+ } finally {
679
+ globalThis.fetch = originalFetch;
680
+ clearModelCache();
681
+ }
682
+ });
683
+
684
+ // ============================================================================
685
+ // Tab Switching Tests
686
+ // ============================================================================
687
+
688
+ test.serial('ModelDatabaseDisplay: switches tabs on Tab key', async t => {
689
+ clearModelCache();
690
+
691
+ const originalFetch = globalThis.fetch;
692
+ globalThis.fetch = async () => {
693
+ return {
694
+ ok: true,
695
+ json: async () => createMockOpenRouterResponse(),
696
+ } as Response;
697
+ };
698
+
699
+ try {
700
+ const {stdin, lastFrame} = render(
701
+ <MockThemeProvider>
702
+ <ModelDatabaseDisplay />
703
+ </MockThemeProvider>,
704
+ );
705
+
706
+ await new Promise(resolve => setTimeout(resolve, 200));
707
+
708
+ const initialOutput = lastFrame();
709
+
710
+ // Press Tab to switch tabs
711
+ stdin.write('\t');
712
+
713
+ await new Promise(resolve => setTimeout(resolve, 50));
714
+
715
+ const afterTabOutput = lastFrame();
716
+
717
+ // Both outputs should be valid
718
+ t.truthy(initialOutput);
719
+ t.truthy(afterTabOutput);
720
+ } finally {
721
+ globalThis.fetch = originalFetch;
722
+ clearModelCache();
723
+ }
724
+ });
725
+
726
+ // ============================================================================
727
+ // Command Title Tests
728
+ // ============================================================================
729
+
730
+ test.serial('ModelDatabaseDisplay: displays command title', async t => {
731
+ clearModelCache();
732
+
733
+ const originalFetch = globalThis.fetch;
734
+ globalThis.fetch = async () => {
735
+ return {
736
+ ok: true,
737
+ json: async () => createMockOpenRouterResponse(),
738
+ } as Response;
739
+ };
740
+
741
+ try {
742
+ const {lastFrame} = render(
743
+ <MockThemeProvider>
744
+ <ModelDatabaseDisplay />
745
+ </MockThemeProvider>,
746
+ );
747
+
748
+ await new Promise(resolve => setTimeout(resolve, 200));
749
+
750
+ const output = lastFrame();
751
+ t.truthy(output);
752
+ // Should display the command title
753
+ t.regex(output!, /model-database|Model Browser/i);
754
+ } finally {
755
+ globalThis.fetch = originalFetch;
756
+ clearModelCache();
757
+ }
758
+ });