@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.
- package/.editorconfig +16 -0
- package/.env.example +63 -0
- package/.gitattributes +1 -0
- package/.semgrepignore +19 -0
- package/coder-dummy-file.ts +52 -0
- package/coder.config.example.json +59 -0
- package/coder.config.json +13 -0
- package/color_picker.html +36 -0
- package/package.json +2 -14
- package/scripts/extract-changelog.js +73 -0
- package/scripts/fetch-models.js +143 -0
- package/scripts/test.sh +40 -0
- package/scripts/update-homebrew-formula.sh +125 -0
- package/scripts/update-nix-version.sh +157 -0
- package/source/ai-sdk-client/AISDKClient.spec.ts +117 -0
- package/source/ai-sdk-client/AISDKClient.ts +155 -0
- package/source/ai-sdk-client/chat/chat-handler.spec.ts +121 -0
- package/source/ai-sdk-client/chat/chat-handler.ts +276 -0
- package/source/ai-sdk-client/chat/streaming-handler.spec.ts +173 -0
- package/source/ai-sdk-client/chat/streaming-handler.ts +110 -0
- package/source/ai-sdk-client/chat/tool-processor.spec.ts +92 -0
- package/source/ai-sdk-client/chat/tool-processor.ts +70 -0
- package/source/ai-sdk-client/converters/message-converter.spec.ts +220 -0
- package/source/ai-sdk-client/converters/message-converter.ts +113 -0
- package/source/ai-sdk-client/converters/tool-converter.spec.ts +90 -0
- package/source/ai-sdk-client/converters/tool-converter.ts +46 -0
- package/source/ai-sdk-client/error-handling/error-extractor.spec.ts +55 -0
- package/source/ai-sdk-client/error-handling/error-extractor.ts +15 -0
- package/source/ai-sdk-client/error-handling/error-parser.spec.ts +169 -0
- package/source/ai-sdk-client/error-handling/error-parser.ts +161 -0
- package/source/ai-sdk-client/index.ts +7 -0
- package/source/ai-sdk-client/providers/provider-factory.spec.ts +71 -0
- package/source/ai-sdk-client/providers/provider-factory.ts +41 -0
- package/source/ai-sdk-client/types.ts +9 -0
- package/source/ai-sdk-client-empty-message.spec.ts +141 -0
- package/source/ai-sdk-client-error-handling.spec.ts +186 -0
- package/source/ai-sdk-client-maxretries.spec.ts +114 -0
- package/source/ai-sdk-client-preparestep.spec.ts +279 -0
- package/source/app/App.spec.tsx +32 -0
- package/source/app/App.tsx +480 -0
- package/source/app/components/AppContainer.spec.tsx +96 -0
- package/source/app/components/AppContainer.tsx +56 -0
- package/source/app/components/ChatInterface.spec.tsx +163 -0
- package/source/app/components/ChatInterface.tsx +144 -0
- package/source/app/components/ModalSelectors.spec.tsx +141 -0
- package/source/app/components/ModalSelectors.tsx +135 -0
- package/source/app/helpers.spec.ts +97 -0
- package/source/app/helpers.ts +63 -0
- package/source/app/index.ts +4 -0
- package/source/app/types.ts +39 -0
- package/source/app/utils/appUtils.ts +294 -0
- package/source/app/utils/conversationState.ts +310 -0
- package/source/app.spec.tsx +244 -0
- package/source/cli.spec.ts +73 -0
- package/source/cli.tsx +51 -0
- package/source/client-factory.spec.ts +48 -0
- package/source/client-factory.ts +178 -0
- package/source/command-parser.spec.ts +127 -0
- package/source/command-parser.ts +36 -0
- package/source/commands/checkpoint.spec.tsx +277 -0
- package/source/commands/checkpoint.tsx +366 -0
- package/source/commands/clear.tsx +22 -0
- package/source/commands/custom-commands.tsx +121 -0
- package/source/commands/exit.ts +21 -0
- package/source/commands/export.spec.tsx +131 -0
- package/source/commands/export.tsx +79 -0
- package/source/commands/help.tsx +120 -0
- package/source/commands/index.ts +17 -0
- package/source/commands/init.tsx +339 -0
- package/source/commands/lsp-command.spec.tsx +281 -0
- package/source/commands/lsp.tsx +120 -0
- package/source/commands/mcp-command.spec.tsx +313 -0
- package/source/commands/mcp.tsx +162 -0
- package/source/commands/model-database.spec.tsx +758 -0
- package/source/commands/model-database.tsx +418 -0
- package/source/commands/model.ts +12 -0
- package/source/commands/provider.ts +12 -0
- package/source/commands/setup-config.tsx +16 -0
- package/source/commands/simple-commands.spec.tsx +175 -0
- package/source/commands/status.ts +12 -0
- package/source/commands/theme.ts +12 -0
- package/source/commands/update.spec.tsx +261 -0
- package/source/commands/update.tsx +201 -0
- package/source/commands/usage.spec.tsx +495 -0
- package/source/commands/usage.tsx +100 -0
- package/source/commands.spec.ts +436 -0
- package/source/commands.ts +83 -0
- package/source/components/assistant-message.spec.tsx +796 -0
- package/source/components/assistant-message.tsx +34 -0
- package/source/components/bash-execution-indicator.tsx +21 -0
- package/source/components/cancelling-indicator.tsx +16 -0
- package/source/components/chat-queue.spec.tsx +83 -0
- package/source/components/chat-queue.tsx +36 -0
- package/source/components/checkpoint-display.spec.tsx +219 -0
- package/source/components/checkpoint-display.tsx +126 -0
- package/source/components/checkpoint-selector.spec.tsx +173 -0
- package/source/components/checkpoint-selector.tsx +173 -0
- package/source/components/development-mode-indicator.spec.tsx +268 -0
- package/source/components/development-mode-indicator.tsx +38 -0
- package/source/components/message-box.spec.tsx +427 -0
- package/source/components/message-box.tsx +87 -0
- package/source/components/model-selector.tsx +132 -0
- package/source/components/provider-selector.tsx +75 -0
- package/source/components/random-spinner.tsx +19 -0
- package/source/components/security-disclaimer.tsx +73 -0
- package/source/components/status-connection-display.spec.tsx +133 -0
- package/source/components/status.tsx +267 -0
- package/source/components/theme-selector.tsx +126 -0
- package/source/components/tool-confirmation.tsx +190 -0
- package/source/components/tool-execution-indicator.tsx +33 -0
- package/source/components/tool-message.tsx +85 -0
- package/source/components/ui/titled-box.spec.tsx +207 -0
- package/source/components/ui/titled-box.tsx +57 -0
- package/source/components/usage/progress-bar.spec.tsx +398 -0
- package/source/components/usage/progress-bar.tsx +30 -0
- package/source/components/usage/usage-display.spec.tsx +780 -0
- package/source/components/usage/usage-display.tsx +291 -0
- package/source/components/user-input.spec.tsx +327 -0
- package/source/components/user-input.tsx +533 -0
- package/source/components/user-message.spec.tsx +230 -0
- package/source/components/user-message.tsx +84 -0
- package/source/components/welcome-message.tsx +76 -0
- package/source/config/env-substitution.ts +65 -0
- package/source/config/index.spec.ts +171 -0
- package/source/config/index.ts +154 -0
- package/source/config/paths.spec.ts +241 -0
- package/source/config/paths.ts +55 -0
- package/source/config/preferences.ts +51 -0
- package/source/config/themes.ts +315 -0
- package/source/constants.ts +130 -0
- package/source/context/mode-context.spec.ts +79 -0
- package/source/context/mode-context.ts +24 -0
- package/source/custom-commands/executor.spec.ts +142 -0
- package/source/custom-commands/executor.ts +64 -0
- package/source/custom-commands/loader.spec.ts +314 -0
- package/source/custom-commands/loader.ts +153 -0
- package/source/custom-commands/parser.ts +196 -0
- package/source/hooks/chat-handler/conversation/conversation-loop.spec.ts +39 -0
- package/source/hooks/chat-handler/conversation/conversation-loop.tsx +511 -0
- package/source/hooks/chat-handler/conversation/tool-executor.spec.ts +50 -0
- package/source/hooks/chat-handler/conversation/tool-executor.tsx +109 -0
- package/source/hooks/chat-handler/index.ts +12 -0
- package/source/hooks/chat-handler/state/streaming-state.spec.ts +26 -0
- package/source/hooks/chat-handler/state/streaming-state.ts +19 -0
- package/source/hooks/chat-handler/types.ts +38 -0
- package/source/hooks/chat-handler/useChatHandler.spec.tsx +321 -0
- package/source/hooks/chat-handler/useChatHandler.tsx +194 -0
- package/source/hooks/chat-handler/utils/context-checker.spec.ts +60 -0
- package/source/hooks/chat-handler/utils/context-checker.tsx +73 -0
- package/source/hooks/chat-handler/utils/message-helpers.spec.ts +42 -0
- package/source/hooks/chat-handler/utils/message-helpers.tsx +36 -0
- package/source/hooks/chat-handler/utils/tool-filters.spec.ts +109 -0
- package/source/hooks/chat-handler/utils/tool-filters.ts +64 -0
- package/source/hooks/useAppHandlers.tsx +291 -0
- package/source/hooks/useAppInitialization.tsx +422 -0
- package/source/hooks/useAppState.tsx +311 -0
- package/source/hooks/useDirectoryTrust.tsx +98 -0
- package/source/hooks/useInputState.ts +414 -0
- package/source/hooks/useModeHandlers.tsx +302 -0
- package/source/hooks/useNonInteractiveMode.ts +140 -0
- package/source/hooks/useTerminalWidth.tsx +81 -0
- package/source/hooks/useTheme.ts +18 -0
- package/source/hooks/useToolHandler.tsx +349 -0
- package/source/hooks/useUIState.ts +61 -0
- package/source/init/agents-template-generator.ts +421 -0
- package/source/init/existing-rules-extractor.ts +319 -0
- package/source/init/file-scanner.spec.ts +227 -0
- package/source/init/file-scanner.ts +238 -0
- package/source/init/framework-detector.ts +382 -0
- package/source/init/language-detector.ts +269 -0
- package/source/init/project-analyzer.spec.ts +231 -0
- package/source/init/project-analyzer.ts +458 -0
- package/source/lsp/index.ts +31 -0
- package/source/lsp/lsp-client.spec.ts +508 -0
- package/source/lsp/lsp-client.ts +487 -0
- package/source/lsp/lsp-manager.spec.ts +477 -0
- package/source/lsp/lsp-manager.ts +419 -0
- package/source/lsp/protocol.spec.ts +502 -0
- package/source/lsp/protocol.ts +360 -0
- package/source/lsp/server-discovery.spec.ts +654 -0
- package/source/lsp/server-discovery.ts +515 -0
- package/source/markdown-parser/html-entities.spec.ts +88 -0
- package/source/markdown-parser/html-entities.ts +45 -0
- package/source/markdown-parser/index.spec.ts +281 -0
- package/source/markdown-parser/index.ts +126 -0
- package/source/markdown-parser/table-parser.spec.ts +133 -0
- package/source/markdown-parser/table-parser.ts +114 -0
- package/source/markdown-parser/utils.spec.ts +70 -0
- package/source/markdown-parser/utils.ts +13 -0
- package/source/mcp/mcp-client.spec.ts +81 -0
- package/source/mcp/mcp-client.ts +625 -0
- package/source/mcp/transport-factory.spec.ts +406 -0
- package/source/mcp/transport-factory.ts +312 -0
- package/source/message-handler.ts +67 -0
- package/source/model-database/database-engine.spec.ts +494 -0
- package/source/model-database/database-engine.ts +50 -0
- package/source/model-database/model-database.spec.ts +363 -0
- package/source/model-database/model-database.ts +91 -0
- package/source/model-database/model-engine.spec.ts +447 -0
- package/source/model-database/model-engine.ts +65 -0
- package/source/model-database/model-fetcher.spec.ts +583 -0
- package/source/model-database/model-fetcher.ts +330 -0
- package/source/models/index.ts +1 -0
- package/source/models/models-cache.spec.ts +214 -0
- package/source/models/models-cache.ts +78 -0
- package/source/models/models-dev-client.spec.ts +379 -0
- package/source/models/models-dev-client.ts +329 -0
- package/source/models/models-types.ts +68 -0
- package/source/prompt-history.ts +155 -0
- package/source/security/command-injection.spec.ts +240 -0
- package/source/services/checkpoint-manager.spec.ts +523 -0
- package/source/services/checkpoint-manager.ts +466 -0
- package/source/services/file-snapshot.spec.ts +569 -0
- package/source/services/file-snapshot.ts +220 -0
- package/source/test-utils/render-with-theme.tsx +48 -0
- package/source/tokenization/index.ts +1 -0
- package/source/tokenization/tokenizer-factory.spec.ts +170 -0
- package/source/tokenization/tokenizer-factory.ts +125 -0
- package/source/tokenization/tokenizers/anthropic-tokenizer.spec.ts +200 -0
- package/source/tokenization/tokenizers/anthropic-tokenizer.ts +43 -0
- package/source/tokenization/tokenizers/fallback-tokenizer.spec.ts +236 -0
- package/source/tokenization/tokenizers/fallback-tokenizer.ts +26 -0
- package/source/tokenization/tokenizers/llama-tokenizer.spec.ts +224 -0
- package/source/tokenization/tokenizers/llama-tokenizer.ts +41 -0
- package/source/tokenization/tokenizers/openai-tokenizer.spec.ts +184 -0
- package/source/tokenization/tokenizers/openai-tokenizer.ts +57 -0
- package/source/tool-calling/index.ts +5 -0
- package/source/tool-calling/json-parser.spec.ts +639 -0
- package/source/tool-calling/json-parser.ts +247 -0
- package/source/tool-calling/tool-parser.spec.ts +395 -0
- package/source/tool-calling/tool-parser.ts +120 -0
- package/source/tool-calling/xml-parser.spec.ts +662 -0
- package/source/tool-calling/xml-parser.ts +289 -0
- package/source/tools/execute-bash.spec.tsx +353 -0
- package/source/tools/execute-bash.tsx +219 -0
- package/source/tools/execute-function.spec.ts +130 -0
- package/source/tools/fetch-url.spec.tsx +342 -0
- package/source/tools/fetch-url.tsx +172 -0
- package/source/tools/find-files.spec.tsx +924 -0
- package/source/tools/find-files.tsx +293 -0
- package/source/tools/index.ts +102 -0
- package/source/tools/lsp-get-diagnostics.tsx +192 -0
- package/source/tools/needs-approval.spec.ts +282 -0
- package/source/tools/read-file.spec.tsx +801 -0
- package/source/tools/read-file.tsx +387 -0
- package/source/tools/search-file-contents.spec.tsx +1273 -0
- package/source/tools/search-file-contents.tsx +293 -0
- package/source/tools/string-replace.spec.tsx +730 -0
- package/source/tools/string-replace.tsx +548 -0
- package/source/tools/tool-manager.ts +210 -0
- package/source/tools/tool-registry.spec.ts +415 -0
- package/source/tools/tool-registry.ts +228 -0
- package/source/tools/web-search.tsx +223 -0
- package/source/tools/write-file.spec.tsx +559 -0
- package/source/tools/write-file.tsx +228 -0
- package/source/types/app.ts +37 -0
- package/source/types/checkpoint.ts +48 -0
- package/source/types/commands.ts +46 -0
- package/source/types/components.ts +27 -0
- package/source/types/config.ts +103 -0
- package/source/types/core-connection-status.spec.ts +67 -0
- package/source/types/core.ts +181 -0
- package/source/types/hooks.ts +50 -0
- package/source/types/index.ts +12 -0
- package/source/types/markdown-parser.ts +11 -0
- package/source/types/mcp.ts +52 -0
- package/source/types/system.ts +16 -0
- package/source/types/tokenization.ts +41 -0
- package/source/types/ui.ts +40 -0
- package/source/types/usage.ts +58 -0
- package/source/types/utils.ts +16 -0
- package/source/usage/calculator.spec.ts +385 -0
- package/source/usage/calculator.ts +104 -0
- package/source/usage/storage.spec.ts +703 -0
- package/source/usage/storage.ts +238 -0
- package/source/usage/tracker.spec.ts +456 -0
- package/source/usage/tracker.ts +102 -0
- package/source/utils/atomic-deletion.spec.ts +194 -0
- package/source/utils/atomic-deletion.ts +127 -0
- package/source/utils/bounded-map.spec.ts +300 -0
- package/source/utils/bounded-map.ts +193 -0
- package/source/utils/checkpoint-utils.spec.ts +222 -0
- package/source/utils/checkpoint-utils.ts +92 -0
- package/source/utils/error-formatter.spec.ts +169 -0
- package/source/utils/error-formatter.ts +194 -0
- package/source/utils/file-autocomplete.spec.ts +173 -0
- package/source/utils/file-autocomplete.ts +196 -0
- package/source/utils/file-cache.spec.ts +309 -0
- package/source/utils/file-cache.ts +195 -0
- package/source/utils/file-content-loader.spec.ts +180 -0
- package/source/utils/file-content-loader.ts +179 -0
- package/source/utils/file-mention-handler.spec.ts +261 -0
- package/source/utils/file-mention-handler.ts +84 -0
- package/source/utils/file-mention-parser.spec.ts +182 -0
- package/source/utils/file-mention-parser.ts +170 -0
- package/source/utils/fuzzy-matching.spec.ts +149 -0
- package/source/utils/fuzzy-matching.ts +146 -0
- package/source/utils/indentation-normalizer.spec.ts +216 -0
- package/source/utils/indentation-normalizer.ts +76 -0
- package/source/utils/installation-detector.spec.ts +178 -0
- package/source/utils/installation-detector.ts +153 -0
- package/source/utils/logging/config.spec.ts +311 -0
- package/source/utils/logging/config.ts +210 -0
- package/source/utils/logging/console-facade.spec.ts +184 -0
- package/source/utils/logging/console-facade.ts +384 -0
- package/source/utils/logging/correlation.spec.ts +679 -0
- package/source/utils/logging/correlation.ts +474 -0
- package/source/utils/logging/formatters.spec.ts +464 -0
- package/source/utils/logging/formatters.ts +207 -0
- package/source/utils/logging/health-monitor/alerts/alert-manager.spec.ts +93 -0
- package/source/utils/logging/health-monitor/alerts/alert-manager.ts +79 -0
- package/source/utils/logging/health-monitor/checks/configuration-check.spec.ts +56 -0
- package/source/utils/logging/health-monitor/checks/configuration-check.ts +43 -0
- package/source/utils/logging/health-monitor/checks/logging-check.spec.ts +56 -0
- package/source/utils/logging/health-monitor/checks/logging-check.ts +58 -0
- package/source/utils/logging/health-monitor/checks/memory-check.spec.ts +100 -0
- package/source/utils/logging/health-monitor/checks/memory-check.ts +78 -0
- package/source/utils/logging/health-monitor/checks/performance-check.spec.ts +56 -0
- package/source/utils/logging/health-monitor/checks/performance-check.ts +56 -0
- package/source/utils/logging/health-monitor/checks/request-check.spec.ts +56 -0
- package/source/utils/logging/health-monitor/checks/request-check.ts +76 -0
- package/source/utils/logging/health-monitor/core/health-check-runner.spec.ts +70 -0
- package/source/utils/logging/health-monitor/core/health-check-runner.ts +138 -0
- package/source/utils/logging/health-monitor/core/health-monitor.spec.ts +58 -0
- package/source/utils/logging/health-monitor/core/health-monitor.ts +344 -0
- package/source/utils/logging/health-monitor/core/scoring.spec.ts +65 -0
- package/source/utils/logging/health-monitor/core/scoring.ts +91 -0
- package/source/utils/logging/health-monitor/index.ts +15 -0
- package/source/utils/logging/health-monitor/instances.ts +48 -0
- package/source/utils/logging/health-monitor/middleware/http-middleware.spec.ts +141 -0
- package/source/utils/logging/health-monitor/middleware/http-middleware.ts +75 -0
- package/source/utils/logging/health-monitor/types.ts +126 -0
- package/source/utils/logging/index.spec.ts +284 -0
- package/source/utils/logging/index.ts +236 -0
- package/source/utils/logging/integration.spec.ts +441 -0
- package/source/utils/logging/log-method-factory.spec.ts +573 -0
- package/source/utils/logging/log-method-factory.ts +233 -0
- package/source/utils/logging/log-query/aggregation/aggregator.spec.ts +277 -0
- package/source/utils/logging/log-query/aggregation/aggregator.ts +159 -0
- package/source/utils/logging/log-query/aggregation/facet-generator.spec.ts +159 -0
- package/source/utils/logging/log-query/aggregation/facet-generator.ts +47 -0
- package/source/utils/logging/log-query/index.ts +23 -0
- package/source/utils/logging/log-query/query/filter-predicates.spec.ts +247 -0
- package/source/utils/logging/log-query/query/filter-predicates.ts +154 -0
- package/source/utils/logging/log-query/query/query-builder.spec.ts +182 -0
- package/source/utils/logging/log-query/query/query-builder.ts +151 -0
- package/source/utils/logging/log-query/query/query-engine.spec.ts +214 -0
- package/source/utils/logging/log-query/query/query-engine.ts +45 -0
- package/source/utils/logging/log-query/storage/circular-buffer.spec.ts +143 -0
- package/source/utils/logging/log-query/storage/circular-buffer.ts +75 -0
- package/source/utils/logging/log-query/storage/index-manager.spec.ts +150 -0
- package/source/utils/logging/log-query/storage/index-manager.ts +71 -0
- package/source/utils/logging/log-query/storage/log-storage.spec.ts +257 -0
- package/source/utils/logging/log-query/storage/log-storage.ts +80 -0
- package/source/utils/logging/log-query/types.ts +163 -0
- package/source/utils/logging/log-query/utils/helpers.spec.ts +263 -0
- package/source/utils/logging/log-query/utils/helpers.ts +72 -0
- package/source/utils/logging/log-query/utils/sorting.spec.ts +182 -0
- package/source/utils/logging/log-query/utils/sorting.ts +61 -0
- package/source/utils/logging/logger-provider.spec.ts +262 -0
- package/source/utils/logging/logger-provider.ts +362 -0
- package/source/utils/logging/performance.spec.ts +209 -0
- package/source/utils/logging/performance.ts +757 -0
- package/source/utils/logging/pino-logger.spec.ts +425 -0
- package/source/utils/logging/pino-logger.ts +514 -0
- package/source/utils/logging/redaction.spec.ts +490 -0
- package/source/utils/logging/redaction.ts +267 -0
- package/source/utils/logging/request-tracker.spec.ts +1198 -0
- package/source/utils/logging/request-tracker.ts +803 -0
- package/source/utils/logging/transports.spec.ts +505 -0
- package/source/utils/logging/transports.ts +305 -0
- package/source/utils/logging/types.ts +216 -0
- package/source/utils/message-builder.spec.ts +179 -0
- package/source/utils/message-builder.ts +101 -0
- package/source/utils/message-queue.tsx +486 -0
- package/source/utils/paste-detection.spec.ts +69 -0
- package/source/utils/paste-detection.ts +124 -0
- package/source/utils/paste-roundtrip.spec.ts +442 -0
- package/source/utils/paste-utils.spec.ts +128 -0
- package/source/utils/paste-utils.ts +52 -0
- package/source/utils/programming-language-helper.spec.ts +74 -0
- package/source/utils/programming-language-helper.ts +32 -0
- package/source/utils/prompt-assembly.spec.ts +221 -0
- package/source/utils/prompt-processor.ts +173 -0
- package/source/utils/tool-args-parser.spec.ts +136 -0
- package/source/utils/tool-args-parser.ts +54 -0
- package/source/utils/tool-cancellation.spec.ts +230 -0
- package/source/utils/tool-cancellation.ts +28 -0
- package/source/utils/tool-result-display.spec.tsx +469 -0
- package/source/utils/tool-result-display.tsx +90 -0
- package/source/utils/update-checker.spec.ts +383 -0
- package/source/utils/update-checker.ts +183 -0
- package/source/wizard/config-wizard.spec.tsx +103 -0
- package/source/wizard/config-wizard.tsx +382 -0
- package/source/wizard/steps/location-step.spec.tsx +186 -0
- package/source/wizard/steps/location-step.tsx +147 -0
- package/source/wizard/steps/mcp-step.spec.tsx +607 -0
- package/source/wizard/steps/mcp-step.tsx +632 -0
- package/source/wizard/steps/provider-step.spec.tsx +342 -0
- package/source/wizard/steps/provider-step.tsx +957 -0
- package/source/wizard/steps/summary-step.spec.tsx +749 -0
- package/source/wizard/steps/summary-step.tsx +228 -0
- package/source/wizard/templates/mcp-templates.spec.ts +613 -0
- package/source/wizard/templates/mcp-templates.ts +570 -0
- package/source/wizard/templates/provider-templates.spec.ts +152 -0
- package/source/wizard/templates/provider-templates.ts +485 -0
- package/source/wizard/utils/fetch-cloud-models.spec.ts +428 -0
- package/source/wizard/utils/fetch-cloud-models.ts +223 -0
- package/source/wizard/utils/fetch-local-models.spec.ts +297 -0
- package/source/wizard/utils/fetch-local-models.ts +192 -0
- package/source/wizard/validation-array.spec.ts +264 -0
- package/source/wizard/validation.spec.ts +373 -0
- package/source/wizard/validation.ts +232 -0
- 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
|
+
});
|