@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,281 @@
1
+ import test from 'ava';
2
+ import {render} from 'ink-testing-library';
3
+ import React from 'react';
4
+ import {themes} from '../config/themes';
5
+ import {ThemeContext} from '../hooks/useTheme';
6
+ import {type LSPInitResult, getLSPManager} from '../lsp/lsp-manager';
7
+ import {LSP, lspCommand} from './lsp';
8
+
9
+ console.log(`\nlsp-command.spec.tsx`);
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
+ return (
19
+ <ThemeContext.Provider value={mockTheme}>{children}</ThemeContext.Provider>
20
+ );
21
+ };
22
+
23
+ // Mock LSP manager status
24
+ const mockLSPStatus = {
25
+ initialized: true,
26
+ servers: [
27
+ {
28
+ name: 'typescript-language-server',
29
+ ready: true,
30
+ languages: ['ts', 'js', 'tsx', 'jsx'],
31
+ },
32
+ {
33
+ name: 'gopls',
34
+ ready: true,
35
+ languages: ['go'],
36
+ },
37
+ {
38
+ name: 'rust-analyzer',
39
+ ready: false,
40
+ languages: ['rs'],
41
+ },
42
+ ],
43
+ };
44
+
45
+ // ============================================================================
46
+ // Tests for LSP Command Display
47
+ // ============================================================================
48
+
49
+ test('LSP command: shows no servers when none connected', t => {
50
+ const emptyStatus = {
51
+ initialized: false,
52
+ servers: [],
53
+ };
54
+
55
+ const {lastFrame} = render(
56
+ <MockThemeProvider>
57
+ <LSP status={emptyStatus} />
58
+ </MockThemeProvider>,
59
+ );
60
+
61
+ const output = lastFrame();
62
+ t.truthy(output);
63
+ t.regex(output!, /No LSP servers connected/);
64
+ t.regex(output!, /agents\.config\.json/);
65
+ t.regex(
66
+ output!,
67
+ /LSP servers will auto-discover based on your project files/,
68
+ );
69
+ });
70
+
71
+ test('LSP command: displays server status correctly', t => {
72
+ const {lastFrame} = render(
73
+ <MockThemeProvider>
74
+ <LSP status={mockLSPStatus} />
75
+ </MockThemeProvider>,
76
+ );
77
+
78
+ const output = lastFrame();
79
+ t.truthy(output);
80
+
81
+ // Should show connected servers count
82
+ t.regex(output!, /Connected LSP Servers \(3\):/);
83
+
84
+ // Should show server names
85
+ t.regex(output!, /typescript-language-server/);
86
+ t.regex(output!, /gopls/);
87
+ t.regex(output!, /rust-analyzer/);
88
+
89
+ // Should show status icons
90
+ t.regex(output!, /🟢/); // Ready servers
91
+ t.regex(output!, /🔴/); // Initializing server
92
+
93
+ // Should show status text
94
+ t.regex(output!, /\(Ready\)/);
95
+ t.regex(output!, /\(Initializing\)/);
96
+ });
97
+
98
+ test('LSP command: displays associated languages correctly', t => {
99
+ const {lastFrame} = render(
100
+ <MockThemeProvider>
101
+ <LSP status={mockLSPStatus} />
102
+ </MockThemeProvider>,
103
+ );
104
+
105
+ const output = lastFrame();
106
+ t.truthy(output);
107
+
108
+ // Should show languages for each server
109
+ t.regex(output!, /Languages: ts, js, tsx, jsx/);
110
+ t.regex(output!, /Languages: go/);
111
+ t.regex(output!, /Languages: rs/);
112
+ });
113
+
114
+ test('LSP command: handles single language correctly', t => {
115
+ const singleLangStatus = {
116
+ initialized: true,
117
+ servers: [
118
+ {
119
+ name: 'single-lang-server',
120
+ ready: true,
121
+ languages: ['python'],
122
+ },
123
+ ],
124
+ };
125
+
126
+ const {lastFrame} = render(
127
+ <MockThemeProvider>
128
+ <LSP status={singleLangStatus} />
129
+ </MockThemeProvider>,
130
+ );
131
+
132
+ const output = lastFrame();
133
+ t.truthy(output);
134
+ t.regex(output!, /Languages: python/);
135
+ });
136
+
137
+ test('LSP command: handles multiple languages correctly', t => {
138
+ const multiLangStatus = {
139
+ initialized: true,
140
+ servers: [
141
+ {
142
+ name: 'multi-lang-server',
143
+ ready: true,
144
+ languages: ['js', 'ts', 'jsx', 'tsx', 'vue'],
145
+ },
146
+ ],
147
+ };
148
+
149
+ const {lastFrame} = render(
150
+ <MockThemeProvider>
151
+ <LSP status={multiLangStatus} />
152
+ </MockThemeProvider>,
153
+ );
154
+
155
+ const output = lastFrame();
156
+ t.truthy(output);
157
+ t.regex(output!, /Languages: js, ts, jsx, tsx, vue/);
158
+ });
159
+
160
+ test('LSP command: shows correct status icons', t => {
161
+ const testCases = [
162
+ {ready: true, expectedIcon: '🟢'},
163
+ {ready: false, expectedIcon: '🔴'},
164
+ ];
165
+
166
+ for (const testCase of testCases) {
167
+ const status = {
168
+ initialized: true,
169
+ servers: [
170
+ {
171
+ name: 'test-server',
172
+ ready: testCase.ready,
173
+ languages: ['test'],
174
+ },
175
+ ],
176
+ };
177
+
178
+ const {lastFrame} = render(
179
+ <MockThemeProvider>
180
+ <LSP status={status} />
181
+ </MockThemeProvider>,
182
+ );
183
+
184
+ const output = lastFrame();
185
+ t.truthy(output);
186
+ t.regex(
187
+ output!,
188
+ new RegExp(testCase.expectedIcon),
189
+ `Should show ${testCase.expectedIcon} for ready=${testCase.ready}`,
190
+ );
191
+ }
192
+ });
193
+
194
+ test('LSP command: shows correct status text', t => {
195
+ const testCases = [
196
+ {ready: true, expectedText: 'Ready'},
197
+ {ready: false, expectedText: 'Initializing'},
198
+ ];
199
+
200
+ for (const testCase of testCases) {
201
+ const status = {
202
+ initialized: true,
203
+ servers: [
204
+ {
205
+ name: 'test-server',
206
+ ready: testCase.ready,
207
+ languages: ['test'],
208
+ },
209
+ ],
210
+ };
211
+
212
+ const {lastFrame} = render(
213
+ <MockThemeProvider>
214
+ <LSP status={status} />
215
+ </MockThemeProvider>,
216
+ );
217
+
218
+ const output = lastFrame();
219
+ t.truthy(output);
220
+ t.regex(
221
+ output!,
222
+ new RegExp(`\\(${testCase.expectedText}\\)`),
223
+ `Should show (${testCase.expectedText}) for ready=${testCase.ready}`,
224
+ );
225
+ }
226
+ });
227
+
228
+ // ============================================================================
229
+ // Command Handler Tests
230
+ // ============================================================================
231
+
232
+ test('lspCommand has correct name', t => {
233
+ t.is(lspCommand.name, 'lsp');
234
+ });
235
+
236
+ test('lspCommand has description', t => {
237
+ t.truthy(lspCommand.description);
238
+ t.is(typeof lspCommand.description, 'string');
239
+ t.true(lspCommand.description.length > 0);
240
+ });
241
+
242
+ test('lspCommand handler is a function', t => {
243
+ t.is(typeof lspCommand.handler, 'function');
244
+ });
245
+
246
+ test('lspCommand handler returns valid React element', async t => {
247
+ // Mock the LSP manager to return our test status
248
+ const originalGetLSPManager = getLSPManager;
249
+ const mockLSPManager = {
250
+ getStatus: () => mockLSPStatus,
251
+ };
252
+ // Since we can't directly mock the singleton function, we'll just call the handler
253
+ // which will use the actual LSP manager (but it will return an element regardless)
254
+ const mockMessages: any[] = [];
255
+ const mockMetadata: any = {
256
+ provider: 'test',
257
+ model: 'test',
258
+ tokens: 0,
259
+ getMessageTokens: () => 0,
260
+ };
261
+
262
+ const result = await lspCommand.handler([], mockMessages, mockMetadata);
263
+
264
+ t.truthy(result);
265
+ t.true(React.isValidElement(result));
266
+ });
267
+
268
+ test('lspCommand handler returns React element when no servers connected', async t => {
269
+ const mockMessages: any[] = [];
270
+ const mockMetadata: any = {
271
+ provider: 'test',
272
+ model: 'test',
273
+ tokens: 0,
274
+ getMessageTokens: () => 0,
275
+ };
276
+
277
+ const result = await lspCommand.handler([], mockMessages, mockMetadata);
278
+
279
+ t.truthy(result);
280
+ t.true(React.isValidElement(result));
281
+ });
@@ -0,0 +1,120 @@
1
+ import { TitledBox } from '@/components/ui/titled-box';
2
+ import { useTerminalWidth } from '@/hooks/useTerminalWidth';
3
+ import { useTheme } from '@/hooks/useTheme';
4
+ import { getLSPManager } from '@/lsp/lsp-manager';
5
+ import type { Command } from '@/types/index';
6
+ import { Box, Text } from 'ink';
7
+ import React from 'react';
8
+
9
+ interface LSPProps {
10
+ status: {
11
+ initialized: boolean;
12
+ servers: Array<{ name: string; ready: boolean; languages: string[] }>;
13
+ };
14
+ }
15
+
16
+ export function LSP({ status }: LSPProps) {
17
+ const boxWidth = useTerminalWidth();
18
+ const { colors } = useTheme();
19
+ const { servers } = status;
20
+
21
+ return (
22
+ <TitledBox
23
+ title="/lsp"
24
+ width={boxWidth}
25
+ borderColor={colors.primary}
26
+ paddingX={2}
27
+ paddingY={1}
28
+ flexDirection="column"
29
+ marginBottom={1}
30
+ >
31
+ {servers.length === 0 ? (
32
+ <>
33
+ <Box marginBottom={1}>
34
+ <Text color={colors.white} bold>
35
+ No LSP servers connected
36
+ </Text>
37
+ </Box>
38
+
39
+ <Text color={colors.white}>
40
+ To connect LSP servers, configure them in your{' '}
41
+ <Text color={colors.primary}>coder.config.json</Text> file:
42
+ </Text>
43
+
44
+ <Box marginTop={1} marginBottom={1}>
45
+ <Text color={colors.secondary}>
46
+ {`{
47
+ "coder": {
48
+ "lsp": {
49
+ "servers": [
50
+ {
51
+ "name": "typescript-language-server",
52
+ "command": "typescript-language-server",
53
+ "args": ["--stdio"],
54
+ "languages": ["ts", "js", "tsx", "jsx"]
55
+ }
56
+ ]
57
+ }
58
+ }
59
+ }`}
60
+ </Text>
61
+ </Box>
62
+
63
+ <Text color={colors.secondary}>
64
+ LSP servers will auto-discover based on your project files.
65
+ </Text>
66
+ </>
67
+ ) : (
68
+ <>
69
+ <Box marginBottom={1}>
70
+ <Text color={colors.primary}>
71
+ Connected LSP Servers ({servers.length}):
72
+ </Text>
73
+ </Box>
74
+
75
+ {servers.map((server, index) => {
76
+ // Determine status icon and text based on readiness
77
+ const statusIcon = server.ready ? '🟢' : '🔴';
78
+ const statusText = server.ready ? 'Ready' : 'Initializing';
79
+
80
+ return (
81
+ <Box key={index} marginBottom={1}>
82
+ <Box flexDirection="column">
83
+ <Text color={colors.white}>
84
+ • {statusIcon}{' '}
85
+ <Text color={colors.primary}>{server.name}</Text>:{' '}
86
+ <Text color={colors.secondary}>({statusText})</Text>
87
+ </Text>
88
+
89
+ {server.languages.length > 0 && (
90
+ <Text color={colors.secondary}>
91
+ Languages: {server.languages.join(', ')}
92
+ </Text>
93
+ )}
94
+ </Box>
95
+ </Box>
96
+ );
97
+ })}
98
+ </>
99
+ )}
100
+ </TitledBox>
101
+ );
102
+ }
103
+
104
+ export const lspCommand: Command = {
105
+ name: 'lsp',
106
+ description: 'Show connected LSP servers and their status',
107
+ handler: (_args: string[], _messages, _metadata) => {
108
+ const lspManager = getLSPManager();
109
+
110
+ // Get the current status of LSP servers
111
+ const status = lspManager.getStatus();
112
+
113
+ return Promise.resolve(
114
+ React.createElement(LSP, {
115
+ key: `lsp-${Date.now()}`,
116
+ status: status,
117
+ }),
118
+ );
119
+ },
120
+ };
@@ -0,0 +1,313 @@
1
+ import test from 'ava';
2
+ import {render} from 'ink-testing-library';
3
+ import React from 'react';
4
+ import {themes} from '../config/themes';
5
+ import {ThemeContext} from '../hooks/useTheme';
6
+ import {ToolManager} from '../tools/tool-manager';
7
+ import {MCP} from './mcp';
8
+
9
+ console.log(`\nmcp-command.spec.tsx`);
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
+ return (
19
+ <ThemeContext.Provider value={mockTheme}>{children}</ThemeContext.Provider>
20
+ );
21
+ };
22
+
23
+ // ============================================================================
24
+ // Tests for Enhanced MCP Command Display
25
+ // ============================================================================
26
+
27
+ test('MCP command: shows no servers when none connected', t => {
28
+ const mockToolManager = {
29
+ getConnectedServers: () => [],
30
+ getServerTools: () => [],
31
+ getServerInfo: () => undefined,
32
+ } as unknown as ToolManager;
33
+
34
+ const {lastFrame} = render(
35
+ <MockThemeProvider>
36
+ <MCP toolManager={mockToolManager} />
37
+ </MockThemeProvider>,
38
+ );
39
+
40
+ const output = lastFrame();
41
+ t.truthy(output);
42
+ t.regex(output!, /No MCP servers connected/);
43
+ });
44
+
45
+ test('MCP command: displays transport type icons', t => {
46
+ const mockToolManager = {
47
+ getConnectedServers: () => [
48
+ 'stdio-server',
49
+ 'websocket-server',
50
+ 'http-server',
51
+ ],
52
+ getServerTools: (serverName: string) => [
53
+ {name: `tool-${serverName}`, description: 'Test tool'},
54
+ ],
55
+ getServerInfo: (serverName: string) => ({
56
+ name: serverName,
57
+ transport: serverName.includes('stdio')
58
+ ? 'stdio'
59
+ : serverName.includes('websocket')
60
+ ? 'websocket'
61
+ : 'http',
62
+ toolCount: 1,
63
+ connected: true,
64
+ }),
65
+ } as unknown as ToolManager;
66
+
67
+ const {lastFrame} = render(
68
+ <MockThemeProvider>
69
+ <MCP toolManager={mockToolManager} />
70
+ </MockThemeProvider>,
71
+ );
72
+
73
+ const output = lastFrame();
74
+ t.truthy(output);
75
+
76
+ // Should show transport icons
77
+ t.regex(output!, /💻/); // stdio icon
78
+ t.regex(output!, /🔄/); // websocket icon
79
+ t.regex(output!, /🌐/); // http icon
80
+
81
+ // Should show transport type names
82
+ t.regex(output!, /STDIO/);
83
+ t.regex(output!, /WEBSOCKET/);
84
+ t.regex(output!, /HTTP/);
85
+ });
86
+
87
+ test('MCP command: displays URLs for remote servers', t => {
88
+ const mockToolManager = {
89
+ getConnectedServers: () => ['remote-server'],
90
+ getServerTools: () => [{name: 'remote-tool', description: 'Remote tool'}],
91
+ getServerInfo: () => ({
92
+ name: 'remote-server',
93
+ transport: 'http',
94
+ url: 'https://example.com/mcp',
95
+ toolCount: 1,
96
+ connected: true,
97
+ }),
98
+ } as unknown as ToolManager;
99
+
100
+ const {lastFrame} = render(
101
+ <MockThemeProvider>
102
+ <MCP toolManager={mockToolManager} />
103
+ </MockThemeProvider>,
104
+ );
105
+
106
+ const output = lastFrame();
107
+ t.truthy(output);
108
+
109
+ // Should show the URL for remote server
110
+ t.regex(output!, /URL: https:\/\/example\.com\/mcp/);
111
+ });
112
+
113
+ test('MCP command: displays server descriptions', t => {
114
+ const mockToolManager = {
115
+ getConnectedServers: () => ['server-with-description'],
116
+ getServerTools: () => [{name: 'test-tool', description: 'Test tool'}],
117
+ getServerInfo: () => ({
118
+ name: 'server-with-description',
119
+ transport: 'stdio',
120
+ toolCount: 1,
121
+ connected: true,
122
+ description: 'This is a test server description',
123
+ }),
124
+ } as unknown as ToolManager;
125
+
126
+ const {lastFrame} = render(
127
+ <MockThemeProvider>
128
+ <MCP toolManager={mockToolManager} />
129
+ </MockThemeProvider>,
130
+ );
131
+
132
+ const output = lastFrame();
133
+ t.truthy(output);
134
+
135
+ // Should show the description
136
+ t.regex(output!, /This is a test server description/);
137
+ });
138
+
139
+ test('MCP command: displays server tags', t => {
140
+ const mockToolManager = {
141
+ getConnectedServers: () => ['server-with-tags'],
142
+ getServerTools: () => [{name: 'test-tool', description: 'Test tool'}],
143
+ getServerInfo: () => ({
144
+ name: 'server-with-tags',
145
+ transport: 'http',
146
+ toolCount: 1,
147
+ connected: true,
148
+ tags: ['documentation', 'remote', 'http'],
149
+ }),
150
+ } as unknown as ToolManager;
151
+
152
+ const {lastFrame} = render(
153
+ <MockThemeProvider>
154
+ <MCP toolManager={mockToolManager} />
155
+ </MockThemeProvider>,
156
+ );
157
+
158
+ const output = lastFrame();
159
+ t.truthy(output);
160
+
161
+ // Should show the tags with # prefix
162
+ t.regex(output!, /Tags: #documentation #remote #http/);
163
+ });
164
+
165
+ test('MCP command: displays tool information correctly', t => {
166
+ const mockToolManager = {
167
+ getConnectedServers: () => ['multi-tool-server'],
168
+ getServerTools: () => [
169
+ {name: 'tool-1', description: 'First tool'},
170
+ {name: 'tool-2', description: 'Second tool'},
171
+ {name: 'tool-3', description: 'Third tool'},
172
+ ],
173
+ getServerInfo: () => ({
174
+ name: 'multi-tool-server',
175
+ transport: 'stdio',
176
+ toolCount: 3,
177
+ connected: true,
178
+ }),
179
+ } as unknown as ToolManager;
180
+
181
+ const {lastFrame} = render(
182
+ <MockThemeProvider>
183
+ <MCP toolManager={mockToolManager} />
184
+ </MockThemeProvider>,
185
+ );
186
+
187
+ const output = lastFrame();
188
+ t.truthy(output);
189
+
190
+ // Should show correct tool count
191
+ t.regex(output!, /3 tools/);
192
+
193
+ // Should list tool names
194
+ t.regex(output!, /Tools:/);
195
+ t.regex(output!, /tool-1/);
196
+ t.regex(output!, /tool-2/);
197
+ t.regex(output!, /tool-3/);
198
+ });
199
+
200
+ test('MCP command: handles singular tool count', t => {
201
+ const mockToolManager = {
202
+ getConnectedServers: () => ['single-tool-server'],
203
+ getServerTools: () => [{name: 'only-tool', description: 'Only tool'}],
204
+ getServerInfo: () => ({
205
+ name: 'single-tool-server',
206
+ transport: 'websocket',
207
+ toolCount: 1,
208
+ connected: true,
209
+ }),
210
+ } as unknown as ToolManager;
211
+
212
+ const {lastFrame} = render(
213
+ <MockThemeProvider>
214
+ <MCP toolManager={mockToolManager} />
215
+ </MockThemeProvider>,
216
+ );
217
+
218
+ const output = lastFrame();
219
+ t.truthy(output);
220
+
221
+ // Should show singular "tool" (not "tools")
222
+ t.regex(output!, /1 tool/);
223
+ t.notRegex(output!, /1 tools/);
224
+ });
225
+
226
+ test('MCP command: shows server count header', t => {
227
+ const mockToolManager = {
228
+ getConnectedServers: () => ['server-1', 'server-2', 'server-3'],
229
+ getServerTools: () => [],
230
+ getServerInfo: () => ({
231
+ name: 'test-server',
232
+ transport: 'stdio',
233
+ toolCount: 0,
234
+ connected: true,
235
+ }),
236
+ } as unknown as ToolManager;
237
+
238
+ const {lastFrame} = render(
239
+ <MockThemeProvider>
240
+ <MCP toolManager={mockToolManager} />
241
+ </MockThemeProvider>,
242
+ );
243
+
244
+ const output = lastFrame();
245
+ t.truthy(output);
246
+
247
+ // Should show connected servers count
248
+ t.regex(output!, /Connected MCP Servers \(3\):/);
249
+ });
250
+
251
+ test('MCP command: shows configuration examples', t => {
252
+ const mockToolManager = {
253
+ getConnectedServers: () => [],
254
+ getServerTools: () => [],
255
+ getServerInfo: () => undefined,
256
+ } as unknown as ToolManager;
257
+
258
+ const {lastFrame} = render(
259
+ <MockThemeProvider>
260
+ <MCP toolManager={mockToolManager} />
261
+ </MockThemeProvider>,
262
+ );
263
+
264
+ const output = lastFrame();
265
+ t.truthy(output);
266
+
267
+ // Should show configuration examples with transport field
268
+ t.regex(output!, /"transport": "stdio"/);
269
+ t.regex(output!, /"transport": "http"/);
270
+
271
+ // Should include transport field in examples
272
+ t.regex(output!, /"command":/);
273
+ t.regex(output!, /"url":/);
274
+ });
275
+
276
+ test('MCP command: uses transport type getTransportIcon function correctly', t => {
277
+ // Test the helper function indirectly through component rendering
278
+ const testCases = [
279
+ {transport: 'stdio', expectedIcon: '💻'},
280
+ {transport: 'websocket', expectedIcon: '🔄'},
281
+ {transport: 'http', expectedIcon: '🌐'},
282
+ {transport: 'unknown', expectedIcon: '❓'},
283
+ ];
284
+
285
+ for (const testCase of testCases) {
286
+ const mockToolManager = {
287
+ getConnectedServers: () => ['test-server'],
288
+ getServerTools: () => [],
289
+ getServerInfo: () => ({
290
+ name: 'test-server',
291
+ transport: testCase.transport as any,
292
+ toolCount: 0,
293
+ connected: true,
294
+ }),
295
+ } as unknown as ToolManager;
296
+
297
+ const {lastFrame} = render(
298
+ <MockThemeProvider>
299
+ <MCP toolManager={mockToolManager} />
300
+ </MockThemeProvider>,
301
+ );
302
+
303
+ const output = lastFrame();
304
+ t.truthy(output);
305
+
306
+ // Should show the correct icon for the transport type
307
+ t.regex(
308
+ output!,
309
+ new RegExp(testCase.expectedIcon),
310
+ `Should show ${testCase.expectedIcon} for ${testCase.transport} transport`,
311
+ );
312
+ }
313
+ });