@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,406 @@
|
|
|
1
|
+
import test from 'ava';
|
|
2
|
+
import type {MCPServer, MCPTransportType} from '../types/mcp';
|
|
3
|
+
import {TransportFactory} from './transport-factory';
|
|
4
|
+
|
|
5
|
+
console.log(`\ntransport-factory.spec.ts`);
|
|
6
|
+
|
|
7
|
+
// ============================================================================
|
|
8
|
+
// Tests for TransportFactory
|
|
9
|
+
// ============================================================================
|
|
10
|
+
|
|
11
|
+
test('TransportFactory.createTransport: creates stdio transport', t => {
|
|
12
|
+
const server: MCPServer = {
|
|
13
|
+
name: 'test-stdio',
|
|
14
|
+
transport: 'stdio',
|
|
15
|
+
command: 'node',
|
|
16
|
+
args: ['server.js'],
|
|
17
|
+
env: {NODE_ENV: 'test'},
|
|
18
|
+
};
|
|
19
|
+
|
|
20
|
+
const transport = TransportFactory.createTransport(server);
|
|
21
|
+
|
|
22
|
+
t.truthy(transport);
|
|
23
|
+
});
|
|
24
|
+
|
|
25
|
+
test('TransportFactory.createTransport: creates websocket transport', t => {
|
|
26
|
+
const server: MCPServer = {
|
|
27
|
+
name: 'test-websocket',
|
|
28
|
+
transport: 'websocket',
|
|
29
|
+
url: 'ws://localhost:3000/mcp',
|
|
30
|
+
};
|
|
31
|
+
|
|
32
|
+
const transport = TransportFactory.createTransport(server);
|
|
33
|
+
|
|
34
|
+
t.truthy(transport);
|
|
35
|
+
});
|
|
36
|
+
|
|
37
|
+
test('TransportFactory.createTransport: creates http transport', t => {
|
|
38
|
+
const server: MCPServer = {
|
|
39
|
+
name: 'test-http',
|
|
40
|
+
transport: 'http',
|
|
41
|
+
url: 'https://example.com/mcp',
|
|
42
|
+
};
|
|
43
|
+
|
|
44
|
+
const transport = TransportFactory.createTransport(server);
|
|
45
|
+
|
|
46
|
+
t.truthy(transport);
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
test('TransportFactory.createTransport: throws error for unsupported transport', t => {
|
|
50
|
+
const server = {
|
|
51
|
+
name: 'test-unsupported',
|
|
52
|
+
transport: 'unsupported' as MCPTransportType,
|
|
53
|
+
};
|
|
54
|
+
|
|
55
|
+
t.throws(
|
|
56
|
+
() => {
|
|
57
|
+
TransportFactory.createTransport(server);
|
|
58
|
+
},
|
|
59
|
+
{
|
|
60
|
+
message: /Unsupported transport type: unsupported/,
|
|
61
|
+
},
|
|
62
|
+
);
|
|
63
|
+
});
|
|
64
|
+
|
|
65
|
+
test('TransportFactory.createTransport: throws error for stdio missing command', t => {
|
|
66
|
+
const server: MCPServer = {
|
|
67
|
+
name: 'test-stdio-no-command',
|
|
68
|
+
transport: 'stdio',
|
|
69
|
+
};
|
|
70
|
+
|
|
71
|
+
t.throws(
|
|
72
|
+
() => {
|
|
73
|
+
TransportFactory.createTransport(server);
|
|
74
|
+
},
|
|
75
|
+
{
|
|
76
|
+
message: /missing command for stdio transport/,
|
|
77
|
+
},
|
|
78
|
+
);
|
|
79
|
+
});
|
|
80
|
+
|
|
81
|
+
test('TransportFactory.createTransport: throws error for websocket missing URL', t => {
|
|
82
|
+
const server: MCPServer = {
|
|
83
|
+
name: 'test-websocket-no-url',
|
|
84
|
+
transport: 'websocket',
|
|
85
|
+
};
|
|
86
|
+
|
|
87
|
+
t.throws(
|
|
88
|
+
() => {
|
|
89
|
+
TransportFactory.createTransport(server);
|
|
90
|
+
},
|
|
91
|
+
{
|
|
92
|
+
message: /missing URL for websocket transport/,
|
|
93
|
+
},
|
|
94
|
+
);
|
|
95
|
+
});
|
|
96
|
+
|
|
97
|
+
test('TransportFactory.createTransport: throws error for http missing URL', t => {
|
|
98
|
+
const server: MCPServer = {
|
|
99
|
+
name: 'test-http-no-url',
|
|
100
|
+
transport: 'http',
|
|
101
|
+
};
|
|
102
|
+
|
|
103
|
+
t.throws(
|
|
104
|
+
() => {
|
|
105
|
+
TransportFactory.createTransport(server);
|
|
106
|
+
},
|
|
107
|
+
{
|
|
108
|
+
message: /missing URL for http transport/,
|
|
109
|
+
},
|
|
110
|
+
);
|
|
111
|
+
});
|
|
112
|
+
|
|
113
|
+
test('TransportFactory.createTransport: throws error for invalid websocket URL', t => {
|
|
114
|
+
const server: MCPServer = {
|
|
115
|
+
name: 'test-websocket-invalid-url',
|
|
116
|
+
transport: 'websocket',
|
|
117
|
+
url: 'http://example.com/mcp', // Should be ws:// or wss://
|
|
118
|
+
};
|
|
119
|
+
|
|
120
|
+
t.throws(
|
|
121
|
+
() => {
|
|
122
|
+
TransportFactory.createTransport(server);
|
|
123
|
+
},
|
|
124
|
+
{
|
|
125
|
+
message: /Invalid WebSocket URL protocol/,
|
|
126
|
+
},
|
|
127
|
+
);
|
|
128
|
+
});
|
|
129
|
+
|
|
130
|
+
test('TransportFactory.createTransport: throws error for invalid http URL', t => {
|
|
131
|
+
const server: MCPServer = {
|
|
132
|
+
name: 'test-http-invalid-url',
|
|
133
|
+
transport: 'http',
|
|
134
|
+
url: 'ws://example.com/mcp', // Should be http:// or https://
|
|
135
|
+
};
|
|
136
|
+
|
|
137
|
+
t.throws(
|
|
138
|
+
() => {
|
|
139
|
+
TransportFactory.createTransport(server);
|
|
140
|
+
},
|
|
141
|
+
{
|
|
142
|
+
message: /Invalid HTTP URL protocol/,
|
|
143
|
+
},
|
|
144
|
+
);
|
|
145
|
+
});
|
|
146
|
+
|
|
147
|
+
// ============================================================================
|
|
148
|
+
// Tests for TransportFactory.validateServerConfig
|
|
149
|
+
// ============================================================================
|
|
150
|
+
|
|
151
|
+
test('TransportFactory.validateServerConfig: validates stdio config', t => {
|
|
152
|
+
const server: MCPServer = {
|
|
153
|
+
name: 'test-stdio',
|
|
154
|
+
transport: 'stdio',
|
|
155
|
+
command: 'node',
|
|
156
|
+
args: ['server.js'],
|
|
157
|
+
};
|
|
158
|
+
|
|
159
|
+
const result = TransportFactory.validateServerConfig(server);
|
|
160
|
+
|
|
161
|
+
t.true(result.valid);
|
|
162
|
+
t.is(result.errors.length, 0);
|
|
163
|
+
});
|
|
164
|
+
|
|
165
|
+
test('TransportFactory.validateServerConfig: validates websocket config', t => {
|
|
166
|
+
const server: MCPServer = {
|
|
167
|
+
name: 'test-websocket',
|
|
168
|
+
transport: 'websocket',
|
|
169
|
+
url: 'wss://example.com/mcp',
|
|
170
|
+
};
|
|
171
|
+
|
|
172
|
+
const result = TransportFactory.validateServerConfig(server);
|
|
173
|
+
|
|
174
|
+
t.true(result.valid);
|
|
175
|
+
t.is(result.errors.length, 0);
|
|
176
|
+
});
|
|
177
|
+
|
|
178
|
+
test('TransportFactory.validateServerConfig: validates http config', t => {
|
|
179
|
+
const server: MCPServer = {
|
|
180
|
+
name: 'test-http',
|
|
181
|
+
transport: 'http',
|
|
182
|
+
url: 'https://example.com/mcp',
|
|
183
|
+
};
|
|
184
|
+
|
|
185
|
+
const result = TransportFactory.validateServerConfig(server);
|
|
186
|
+
|
|
187
|
+
t.true(result.valid);
|
|
188
|
+
t.is(result.errors.length, 0);
|
|
189
|
+
});
|
|
190
|
+
|
|
191
|
+
test('TransportFactory.validateServerConfig: detects invalid stdio config', t => {
|
|
192
|
+
const server: MCPServer = {
|
|
193
|
+
name: 'test-stdio-invalid',
|
|
194
|
+
transport: 'stdio',
|
|
195
|
+
// Missing command
|
|
196
|
+
};
|
|
197
|
+
|
|
198
|
+
const result = TransportFactory.validateServerConfig(server);
|
|
199
|
+
|
|
200
|
+
t.false(result.valid);
|
|
201
|
+
t.true(
|
|
202
|
+
result.errors.some((error: string) =>
|
|
203
|
+
error.includes('stdio transport requires a command'),
|
|
204
|
+
),
|
|
205
|
+
);
|
|
206
|
+
});
|
|
207
|
+
|
|
208
|
+
test('TransportFactory.validateServerConfig: detects invalid websocket config', t => {
|
|
209
|
+
const server: MCPServer = {
|
|
210
|
+
name: 'test-websocket-invalid',
|
|
211
|
+
transport: 'websocket',
|
|
212
|
+
url: 'http://example.com/mcp', // Invalid protocol
|
|
213
|
+
};
|
|
214
|
+
|
|
215
|
+
const result = TransportFactory.validateServerConfig(server);
|
|
216
|
+
|
|
217
|
+
t.false(result.valid);
|
|
218
|
+
t.true(
|
|
219
|
+
result.errors.some((error: string) =>
|
|
220
|
+
error.includes('websocket URL must use ws:// or wss:// protocol'),
|
|
221
|
+
),
|
|
222
|
+
);
|
|
223
|
+
});
|
|
224
|
+
|
|
225
|
+
test('TransportFactory.validateServerConfig: detects invalid http config', t => {
|
|
226
|
+
const server: MCPServer = {
|
|
227
|
+
name: 'test-http-invalid',
|
|
228
|
+
transport: 'http',
|
|
229
|
+
url: 'ws://example.com/mcp', // Invalid protocol
|
|
230
|
+
};
|
|
231
|
+
|
|
232
|
+
const result = TransportFactory.validateServerConfig(server);
|
|
233
|
+
|
|
234
|
+
t.false(result.valid);
|
|
235
|
+
t.true(
|
|
236
|
+
result.errors.some((error: string) =>
|
|
237
|
+
error.includes('http URL must use http:// or https:// protocol'),
|
|
238
|
+
),
|
|
239
|
+
);
|
|
240
|
+
});
|
|
241
|
+
|
|
242
|
+
// ============================================================================
|
|
243
|
+
// Tests for TransportFactory.getTransportTips
|
|
244
|
+
// ============================================================================
|
|
245
|
+
|
|
246
|
+
test('TransportFactory.getTransportTips: returns stdio tips', t => {
|
|
247
|
+
const tips = TransportFactory.getTransportTips('stdio');
|
|
248
|
+
|
|
249
|
+
t.true(tips.length > 0);
|
|
250
|
+
t.true(tips.some((tip: string) => tip.includes('Stdio transport')));
|
|
251
|
+
});
|
|
252
|
+
|
|
253
|
+
test('TransportFactory.getTransportTips: returns websocket tips', t => {
|
|
254
|
+
const tips = TransportFactory.getTransportTips('websocket');
|
|
255
|
+
|
|
256
|
+
t.true(tips.length > 0);
|
|
257
|
+
t.true(tips.some((tip: string) => tip.includes('WebSocket')));
|
|
258
|
+
});
|
|
259
|
+
|
|
260
|
+
test('TransportFactory.getTransportTips: returns http tips', t => {
|
|
261
|
+
const tips = TransportFactory.getTransportTips('http');
|
|
262
|
+
|
|
263
|
+
t.true(tips.length > 0);
|
|
264
|
+
t.true(tips.some((tip: string) => tip.includes('HTTP')));
|
|
265
|
+
});
|
|
266
|
+
|
|
267
|
+
test('TransportFactory.getTransportTips: handles unknown transport', t => {
|
|
268
|
+
const tips = TransportFactory.getTransportTips('unknown' as MCPTransportType);
|
|
269
|
+
|
|
270
|
+
t.is(tips.length, 1);
|
|
271
|
+
t.is(tips[0], 'Unknown transport type');
|
|
272
|
+
});
|
|
273
|
+
|
|
274
|
+
// ============================================================================
|
|
275
|
+
// Tests for Header Warning Functionality (New Feature)
|
|
276
|
+
// ============================================================================
|
|
277
|
+
|
|
278
|
+
test('TransportFactory.createTransport: warns about auth config for websocket transport', t => {
|
|
279
|
+
// Test that transport is created despite auth config warning
|
|
280
|
+
// Note: Warnings are now logged via the logging system (logWarning),
|
|
281
|
+
// not console.warn directly
|
|
282
|
+
const server: MCPServer = {
|
|
283
|
+
name: 'test-websocket-with-auth',
|
|
284
|
+
transport: 'websocket',
|
|
285
|
+
url: 'ws://localhost:3000/mcp',
|
|
286
|
+
auth: {
|
|
287
|
+
type: 'bearer',
|
|
288
|
+
token: 'token123',
|
|
289
|
+
},
|
|
290
|
+
};
|
|
291
|
+
|
|
292
|
+
const transport = TransportFactory.createTransport(server);
|
|
293
|
+
|
|
294
|
+
// Transport should still be created even with unsupported auth config
|
|
295
|
+
t.truthy(transport);
|
|
296
|
+
// Warning is logged via logWarning() to the logging system
|
|
297
|
+
});
|
|
298
|
+
|
|
299
|
+
test('TransportFactory.createTransport: creates http transport with headers', t => {
|
|
300
|
+
const server: MCPServer = {
|
|
301
|
+
name: 'test-http-with-headers',
|
|
302
|
+
transport: 'http',
|
|
303
|
+
url: 'https://example.com/mcp',
|
|
304
|
+
headers: {
|
|
305
|
+
Authorization: 'Bearer token123',
|
|
306
|
+
},
|
|
307
|
+
};
|
|
308
|
+
|
|
309
|
+
const transport = TransportFactory.createTransport(server);
|
|
310
|
+
|
|
311
|
+
t.truthy(transport);
|
|
312
|
+
});
|
|
313
|
+
|
|
314
|
+
test('TransportFactory.createTransport: warns about auth config for http transport', t => {
|
|
315
|
+
// Test that transport is created despite auth config warning
|
|
316
|
+
// Note: Warnings are now logged via the logging system (logWarning),
|
|
317
|
+
// not console.warn directly
|
|
318
|
+
const server: MCPServer = {
|
|
319
|
+
name: 'test-http-with-auth',
|
|
320
|
+
transport: 'http',
|
|
321
|
+
url: 'https://example.com/mcp',
|
|
322
|
+
auth: {
|
|
323
|
+
type: 'bearer',
|
|
324
|
+
token: 'token123',
|
|
325
|
+
},
|
|
326
|
+
};
|
|
327
|
+
|
|
328
|
+
const transport = TransportFactory.createTransport(server);
|
|
329
|
+
|
|
330
|
+
// Transport should still be created even with unsupported auth config
|
|
331
|
+
t.truthy(transport);
|
|
332
|
+
// Warning is logged via logWarning() to the logging system
|
|
333
|
+
});
|
|
334
|
+
|
|
335
|
+
test('TransportFactory.validateServerConfig: validates http config with headers', t => {
|
|
336
|
+
const server: MCPServer = {
|
|
337
|
+
name: 'test-http-with-headers-validation',
|
|
338
|
+
transport: 'http',
|
|
339
|
+
url: 'https://example.com/mcp',
|
|
340
|
+
headers: {
|
|
341
|
+
'Custom-Header': 'value',
|
|
342
|
+
},
|
|
343
|
+
};
|
|
344
|
+
|
|
345
|
+
const result = TransportFactory.validateServerConfig(server);
|
|
346
|
+
|
|
347
|
+
t.true(result.valid);
|
|
348
|
+
t.is(result.errors.length, 0);
|
|
349
|
+
});
|
|
350
|
+
|
|
351
|
+
// ============================================================================
|
|
352
|
+
// Tests for Command Existence Validation (Issue #148 Fix)
|
|
353
|
+
// ============================================================================
|
|
354
|
+
|
|
355
|
+
test('TransportFactory.validateServerConfig: detects missing command', t => {
|
|
356
|
+
const server: MCPServer = {
|
|
357
|
+
name: 'test-stdio-missing-command',
|
|
358
|
+
transport: 'stdio',
|
|
359
|
+
command: 'this-command-definitely-does-not-exist-xyz123',
|
|
360
|
+
};
|
|
361
|
+
|
|
362
|
+
const result = TransportFactory.validateServerConfig(server);
|
|
363
|
+
|
|
364
|
+
t.false(result.valid);
|
|
365
|
+
t.true(
|
|
366
|
+
result.errors.some((error: string) =>
|
|
367
|
+
error.includes("Command 'this-command-definitely-does-not-exist-xyz123' not found"),
|
|
368
|
+
),
|
|
369
|
+
);
|
|
370
|
+
});
|
|
371
|
+
|
|
372
|
+
test('TransportFactory.validateServerConfig: returns uvx-specific installation hint', t => {
|
|
373
|
+
const server: MCPServer = {
|
|
374
|
+
name: 'test-stdio-uvx-hint',
|
|
375
|
+
transport: 'stdio',
|
|
376
|
+
command: 'uvx',
|
|
377
|
+
};
|
|
378
|
+
|
|
379
|
+
const result = TransportFactory.validateServerConfig(server);
|
|
380
|
+
|
|
381
|
+
// Note: uvx may or may not be installed in different test environments
|
|
382
|
+
// If uvx is installed and valid, the test passes
|
|
383
|
+
// If uvx is not installed, check for uvx-specific installation instructions
|
|
384
|
+
if (result.valid) {
|
|
385
|
+
t.pass('uvx is installed in this environment');
|
|
386
|
+
} else {
|
|
387
|
+
t.true(result.errors.length > 0);
|
|
388
|
+
t.true(result.errors.some((error: string) => error.includes("'uv' Python package manager")));
|
|
389
|
+
t.true(result.errors.some((error: string) => error.includes('astral.sh/uv/install.sh')));
|
|
390
|
+
}
|
|
391
|
+
});
|
|
392
|
+
|
|
393
|
+
test('TransportFactory.validateServerConfig: validates existing command (node)', t => {
|
|
394
|
+
// node should exist since we're running tests with it
|
|
395
|
+
const server: MCPServer = {
|
|
396
|
+
name: 'test-stdio-node-exists',
|
|
397
|
+
transport: 'stdio',
|
|
398
|
+
command: 'node',
|
|
399
|
+
args: ['--version'],
|
|
400
|
+
};
|
|
401
|
+
|
|
402
|
+
const result = TransportFactory.validateServerConfig(server);
|
|
403
|
+
|
|
404
|
+
t.true(result.valid);
|
|
405
|
+
t.is(result.errors.length, 0);
|
|
406
|
+
});
|
|
@@ -0,0 +1,312 @@
|
|
|
1
|
+
import {execFileSync} from 'child_process';
|
|
2
|
+
import {accessSync, constants as fsConstants} from 'fs';
|
|
3
|
+
import {logWarning} from '@/utils/message-queue';
|
|
4
|
+
import {StdioClientTransport} from '@modelcontextprotocol/sdk/client/stdio.js';
|
|
5
|
+
import {StreamableHTTPClientTransport} from '@modelcontextprotocol/sdk/client/streamableHttp.js';
|
|
6
|
+
import {WebSocketClientTransport} from '@modelcontextprotocol/sdk/client/websocket.js';
|
|
7
|
+
import type {MCPServer, MCPTransportType} from '../types/mcp.js';
|
|
8
|
+
|
|
9
|
+
/**
|
|
10
|
+
* Installation instructions for common MCP server dependencies
|
|
11
|
+
*/
|
|
12
|
+
const COMMAND_INSTALL_HINTS: Record<string, string> = {
|
|
13
|
+
uvx: `'uvx' is part of the 'uv' Python package manager.
|
|
14
|
+
|
|
15
|
+
Install uv:
|
|
16
|
+
• macOS/Linux: curl -LsSf https://astral.sh/uv/install.sh | sh
|
|
17
|
+
• Windows: powershell -c "irm https://astral.sh/uv/install.ps1 | iex"
|
|
18
|
+
• pip: pip install uv
|
|
19
|
+
• Homebrew: brew install uv
|
|
20
|
+
|
|
21
|
+
After installation, restart your terminal and try again.`,
|
|
22
|
+
npx: `'npx' is part of Node.js.
|
|
23
|
+
|
|
24
|
+
Install Node.js from: https://nodejs.org/
|
|
25
|
+
Or use a version manager like nvm, fnm, or volta.`,
|
|
26
|
+
node: `'node' is not installed.
|
|
27
|
+
|
|
28
|
+
Install Node.js from: https://nodejs.org/
|
|
29
|
+
Or use a version manager like nvm, fnm, or volta.`,
|
|
30
|
+
python: `'python' is not installed.
|
|
31
|
+
|
|
32
|
+
Install Python from: https://python.org/downloads/
|
|
33
|
+
Or use a version manager like pyenv.`,
|
|
34
|
+
python3: `'python3' is not installed.
|
|
35
|
+
|
|
36
|
+
Install Python from: https://python.org/downloads/
|
|
37
|
+
Or use a version manager like pyenv.`,
|
|
38
|
+
};
|
|
39
|
+
|
|
40
|
+
/**
|
|
41
|
+
* Checks if a command exists in the system PATH or as an executable path.
|
|
42
|
+
* Uses execFileSync with separate arguments to prevent shell injection.
|
|
43
|
+
* Handles both PATH lookups and direct path references (./bin/cmd, /usr/bin/cmd).
|
|
44
|
+
*/
|
|
45
|
+
function commandExists(command: string): boolean {
|
|
46
|
+
// Check if command is a path (contains path separators)
|
|
47
|
+
if (command.includes('/') || command.includes('\\')) {
|
|
48
|
+
try {
|
|
49
|
+
// Check if file exists and is executable
|
|
50
|
+
accessSync(command, fsConstants.X_OK);
|
|
51
|
+
return true;
|
|
52
|
+
} catch {
|
|
53
|
+
return false;
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
// PATH lookup using which/where
|
|
58
|
+
try {
|
|
59
|
+
const checkCmd = process.platform === 'win32' ? 'where' : 'which';
|
|
60
|
+
execFileSync(checkCmd, [command], {stdio: 'ignore'});
|
|
61
|
+
return true;
|
|
62
|
+
} catch {
|
|
63
|
+
return false;
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
/**
|
|
68
|
+
* Gets installation hint for a missing command
|
|
69
|
+
*/
|
|
70
|
+
function getInstallHint(command: string): string {
|
|
71
|
+
return (
|
|
72
|
+
COMMAND_INSTALL_HINTS[command] ||
|
|
73
|
+
`'${command}' is not installed or not in your PATH.`
|
|
74
|
+
);
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
// Union type for all supported client transports
|
|
78
|
+
type ClientTransport =
|
|
79
|
+
| StdioClientTransport
|
|
80
|
+
| WebSocketClientTransport
|
|
81
|
+
| StreamableHTTPClientTransport;
|
|
82
|
+
|
|
83
|
+
/**
|
|
84
|
+
* Factory for creating MCP client transports based on server configuration
|
|
85
|
+
*/
|
|
86
|
+
export class TransportFactory {
|
|
87
|
+
/**
|
|
88
|
+
* Creates a transport instance for the given MCP server configuration
|
|
89
|
+
*/
|
|
90
|
+
static createTransport(server: MCPServer): ClientTransport {
|
|
91
|
+
switch (server.transport) {
|
|
92
|
+
case 'stdio':
|
|
93
|
+
return this.createStdioTransport(server);
|
|
94
|
+
|
|
95
|
+
case 'websocket':
|
|
96
|
+
return this.createWebSocketTransport(server);
|
|
97
|
+
|
|
98
|
+
case 'http':
|
|
99
|
+
return this.createHTTPTransport(server);
|
|
100
|
+
|
|
101
|
+
default: {
|
|
102
|
+
const _exhaustiveCheck: never = server.transport;
|
|
103
|
+
throw new Error(
|
|
104
|
+
`Unsupported transport type: ${_exhaustiveCheck as string}`,
|
|
105
|
+
);
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
/**
|
|
111
|
+
* Creates a stdio transport for local MCP servers
|
|
112
|
+
*/
|
|
113
|
+
private static createStdioTransport(server: MCPServer): StdioClientTransport {
|
|
114
|
+
if (!server.command) {
|
|
115
|
+
throw new Error(
|
|
116
|
+
`MCP server "${server.name}" missing command for stdio transport`,
|
|
117
|
+
);
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
return new StdioClientTransport({
|
|
121
|
+
command: server.command,
|
|
122
|
+
args: server.args || [],
|
|
123
|
+
env: server.env
|
|
124
|
+
? ({...process.env, ...server.env} as Record<string, string>)
|
|
125
|
+
: undefined,
|
|
126
|
+
});
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
/**
|
|
130
|
+
* Creates a WebSocket transport for remote MCP servers
|
|
131
|
+
*/
|
|
132
|
+
private static createWebSocketTransport(
|
|
133
|
+
server: MCPServer,
|
|
134
|
+
): WebSocketClientTransport {
|
|
135
|
+
if (!server.url) {
|
|
136
|
+
throw new Error(
|
|
137
|
+
`MCP server "${server.name}" missing URL for websocket transport`,
|
|
138
|
+
);
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
const url = new URL(server.url);
|
|
142
|
+
|
|
143
|
+
// Validate WebSocket URL
|
|
144
|
+
if (!url.protocol.startsWith('ws')) {
|
|
145
|
+
throw new Error(
|
|
146
|
+
`Invalid WebSocket URL protocol: ${url.protocol}. Expected ws:// or wss://`, // nosemgrep
|
|
147
|
+
);
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
const transport = new WebSocketClientTransport(url);
|
|
151
|
+
|
|
152
|
+
// Note: The WebSocketClientTransport doesn't directly support headers in the current SDK
|
|
153
|
+
// Authentication would need to be handled at the protocol level or via URL parameters
|
|
154
|
+
if (server.auth) {
|
|
155
|
+
logWarning('WebSocket transport has unsupported auth config', true, {
|
|
156
|
+
context: {
|
|
157
|
+
serverName: server.name,
|
|
158
|
+
transportType: 'websocket',
|
|
159
|
+
reason:
|
|
160
|
+
'Current SDK does not support headers for WebSocket transport',
|
|
161
|
+
},
|
|
162
|
+
});
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
return transport;
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
/**
|
|
169
|
+
* Creates an HTTP transport for remote MCP servers
|
|
170
|
+
*/
|
|
171
|
+
private static createHTTPTransport(
|
|
172
|
+
server: MCPServer,
|
|
173
|
+
): StreamableHTTPClientTransport {
|
|
174
|
+
if (!server.url) {
|
|
175
|
+
throw new Error(
|
|
176
|
+
`MCP server "${server.name}" missing URL for http transport`,
|
|
177
|
+
);
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
const url = new URL(server.url);
|
|
181
|
+
|
|
182
|
+
// Validate HTTP URL
|
|
183
|
+
if (!url.protocol.startsWith('http')) {
|
|
184
|
+
throw new Error(
|
|
185
|
+
`Invalid HTTP URL protocol: ${url.protocol}. Expected http:// or https://`,
|
|
186
|
+
);
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
// Create transport with headers if provided
|
|
190
|
+
const transportOptions = server.headers
|
|
191
|
+
? {requestInit: {headers: server.headers}}
|
|
192
|
+
: undefined;
|
|
193
|
+
const transport = new StreamableHTTPClientTransport(url, transportOptions);
|
|
194
|
+
|
|
195
|
+
if (server.auth) {
|
|
196
|
+
logWarning('HTTP transport has unsupported auth config', true, {
|
|
197
|
+
context: {
|
|
198
|
+
serverName: server.name,
|
|
199
|
+
transportType: 'http',
|
|
200
|
+
reason:
|
|
201
|
+
'Current SDK does not support custom headers for HTTP transport',
|
|
202
|
+
},
|
|
203
|
+
});
|
|
204
|
+
}
|
|
205
|
+
|
|
206
|
+
// Check if headers are specified but cannot be used
|
|
207
|
+
if (server.headers) {
|
|
208
|
+
const headerKeys = Object.keys(server.headers);
|
|
209
|
+
if (headerKeys.length > 0) {
|
|
210
|
+
// Headers are now being used, so don't show the warning anymore
|
|
211
|
+
// console.warn(...) - Commented out since headers are now supported
|
|
212
|
+
}
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
return transport;
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
/**
|
|
219
|
+
* Validates the server configuration for the given transport type
|
|
220
|
+
*/
|
|
221
|
+
static validateServerConfig(server: MCPServer): {
|
|
222
|
+
valid: boolean;
|
|
223
|
+
errors: string[];
|
|
224
|
+
} {
|
|
225
|
+
const errors: string[] = [];
|
|
226
|
+
|
|
227
|
+
switch (server.transport) {
|
|
228
|
+
case 'stdio':
|
|
229
|
+
if (!server.command) {
|
|
230
|
+
errors.push('stdio transport requires a command');
|
|
231
|
+
} else if (!commandExists(server.command)) {
|
|
232
|
+
const hint = getInstallHint(server.command);
|
|
233
|
+
errors.push(`Command '${server.command}' not found.\n\n${hint}`);
|
|
234
|
+
}
|
|
235
|
+
break;
|
|
236
|
+
|
|
237
|
+
case 'websocket':
|
|
238
|
+
if (!server.url) {
|
|
239
|
+
errors.push('websocket transport requires a URL');
|
|
240
|
+
} else {
|
|
241
|
+
try {
|
|
242
|
+
const url = new URL(server.url);
|
|
243
|
+
if (!url.protocol.startsWith('ws')) {
|
|
244
|
+
errors.push('websocket URL must use ws:// or wss:// protocol'); // nosemgrep
|
|
245
|
+
}
|
|
246
|
+
} catch {
|
|
247
|
+
errors.push('websocket URL is invalid');
|
|
248
|
+
}
|
|
249
|
+
}
|
|
250
|
+
break;
|
|
251
|
+
|
|
252
|
+
case 'http':
|
|
253
|
+
if (!server.url) {
|
|
254
|
+
errors.push('http transport requires a URL');
|
|
255
|
+
} else {
|
|
256
|
+
try {
|
|
257
|
+
const url = new URL(server.url);
|
|
258
|
+
if (!url.protocol.startsWith('http')) {
|
|
259
|
+
errors.push('http URL must use http:// or https:// protocol');
|
|
260
|
+
}
|
|
261
|
+
} catch {
|
|
262
|
+
errors.push('http URL is invalid');
|
|
263
|
+
}
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
// Headers are now supported, so we don't need to warn about them being ignored
|
|
267
|
+
// The actual warning logic has been moved to the createHTTPTransport method
|
|
268
|
+
break;
|
|
269
|
+
}
|
|
270
|
+
|
|
271
|
+
return {
|
|
272
|
+
valid: errors.length === 0,
|
|
273
|
+
errors,
|
|
274
|
+
};
|
|
275
|
+
}
|
|
276
|
+
|
|
277
|
+
/**
|
|
278
|
+
* Gets transport-specific configuration tips for users
|
|
279
|
+
*/
|
|
280
|
+
static getTransportTips(transportType: MCPTransportType): string[] {
|
|
281
|
+
switch (transportType) {
|
|
282
|
+
case 'stdio':
|
|
283
|
+
return [
|
|
284
|
+
'Stdio transport spawns a local process',
|
|
285
|
+
'Requires a command and optional arguments',
|
|
286
|
+
'Environment variables can be passed to the process',
|
|
287
|
+
'Best for local MCP servers and tools',
|
|
288
|
+
];
|
|
289
|
+
|
|
290
|
+
case 'websocket':
|
|
291
|
+
return [
|
|
292
|
+
'WebSocket transport connects to remote MCP servers',
|
|
293
|
+
'Requires a ws:// or wss:// URL', // nosemgrep
|
|
294
|
+
'Supports real-time bidirectional communication',
|
|
295
|
+
'Best for interactive remote services',
|
|
296
|
+
'Note: Custom headers are not currently supported by the SDK and will be ignored',
|
|
297
|
+
];
|
|
298
|
+
|
|
299
|
+
case 'http':
|
|
300
|
+
return [
|
|
301
|
+
'HTTP transport connects to remote MCP servers via REST API',
|
|
302
|
+
'Requires an http:// or https:// URL',
|
|
303
|
+
'Uses the StreamableHTTP protocol from MCP specification',
|
|
304
|
+
'Best for stateless remote services and APIs',
|
|
305
|
+
'Custom headers are now supported for authentication',
|
|
306
|
+
];
|
|
307
|
+
|
|
308
|
+
default:
|
|
309
|
+
return ['Unknown transport type'];
|
|
310
|
+
}
|
|
311
|
+
}
|
|
312
|
+
}
|