fss-link 1.0.49 → 1.0.51
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/dist/index.js +0 -0
- package/dist/package.json +2 -2
- package/dist/src/config/auth.js +8 -5
- package/dist/src/config/auth.js.map +1 -1
- package/dist/src/config/database.d.ts +103 -11
- package/dist/src/config/database.js +301 -59
- package/dist/src/config/database.js.map +1 -1
- package/dist/src/config/databaseBackup.d.ts +114 -0
- package/dist/src/config/databaseBackup.js +334 -0
- package/dist/src/config/databaseBackup.js.map +1 -0
- package/dist/src/config/databaseMigrations.d.ts +63 -0
- package/dist/src/config/databaseMigrations.js +379 -0
- package/dist/src/config/databaseMigrations.js.map +1 -0
- package/dist/src/config/databasePool.d.ts +70 -0
- package/dist/src/config/databasePool.js +193 -0
- package/dist/src/config/databasePool.js.map +1 -0
- package/dist/src/config/queryOptimizer.d.ts +127 -0
- package/dist/src/config/queryOptimizer.js +309 -0
- package/dist/src/config/queryOptimizer.js.map +1 -0
- package/dist/src/utils/sandbox.js +2 -8
- package/dist/src/utils/sandbox.js.map +1 -1
- package/dist/src/validateNonInterActiveAuth.js +3 -7
- package/dist/src/validateNonInterActiveAuth.js.map +1 -1
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +2 -2
- package/dist/commands/mcp/add.test.ts +0 -122
- package/dist/commands/mcp/add.ts +0 -222
- package/dist/commands/mcp/list.test.ts +0 -154
- package/dist/commands/mcp/list.ts +0 -139
- package/dist/commands/mcp/remove.test.ts +0 -69
- package/dist/commands/mcp/remove.ts +0 -60
- package/dist/commands/mcp.test.ts +0 -55
- package/dist/commands/mcp.ts +0 -27
- package/dist/config/apiValidation.test.ts +0 -118
- package/dist/config/auth.test.ts +0 -79
- package/dist/config/auth.ts +0 -100
- package/dist/config/config.integration.test.ts +0 -407
- package/dist/config/config.test.ts +0 -1952
- package/dist/config/config.ts +0 -690
- package/dist/config/database.test.ts +0 -96
- package/dist/config/database.ts +0 -824
- package/dist/config/extension.test.ts +0 -236
- package/dist/config/extension.ts +0 -180
- package/dist/config/keyBindings.test.ts +0 -62
- package/dist/config/keyBindings.ts +0 -184
- package/dist/config/modelManager.ts +0 -326
- package/dist/config/providerManager.ts +0 -244
- package/dist/config/providerPersistence.test.ts +0 -377
- package/dist/config/providerPersistence.ts +0 -105
- package/dist/config/sandboxConfig.ts +0 -107
- package/dist/config/settings.test.ts +0 -1424
- package/dist/config/settings.ts +0 -517
- package/dist/config/settingsSchema.test.ts +0 -252
- package/dist/config/settingsSchema.ts +0 -728
- package/dist/config/trustedFolders.test.ts +0 -208
- package/dist/config/trustedFolders.ts +0 -167
- package/dist/gemini.test.tsx +0 -252
- package/dist/gemini.tsx +0 -357
- package/dist/generated/git-commit.ts +0 -10
- package/dist/index.ts +0 -21
- package/dist/nonInteractiveCli.test.ts +0 -276
- package/dist/nonInteractiveCli.ts +0 -143
- package/dist/patches/is-in-ci.ts +0 -17
- package/dist/services/BuiltinCommandLoader.test.ts +0 -127
- package/dist/services/BuiltinCommandLoader.ts +0 -95
- package/dist/services/CommandService.test.ts +0 -352
- package/dist/services/CommandService.ts +0 -103
- package/dist/services/FileCommandLoader.test.ts +0 -1002
- package/dist/services/FileCommandLoader.ts +0 -289
- package/dist/services/McpPromptLoader.ts +0 -231
- package/dist/services/SearchEngineConfigProvider.ts +0 -100
- package/dist/services/prompt-processors/argumentProcessor.test.ts +0 -41
- package/dist/services/prompt-processors/argumentProcessor.ts +0 -23
- package/dist/services/prompt-processors/shellProcessor.test.ts +0 -709
- package/dist/services/prompt-processors/shellProcessor.ts +0 -248
- package/dist/services/prompt-processors/types.ts +0 -44
- package/dist/services/types.ts +0 -24
- package/dist/src/config/apiValidation.test.d.ts +0 -6
- package/dist/src/config/apiValidation.test.js +0 -99
- package/dist/src/config/apiValidation.test.js.map +0 -1
- package/dist/src/config/database.test.d.ts +0 -6
- package/dist/src/config/database.test.js +0 -80
- package/dist/src/config/database.test.js.map +0 -1
- package/dist/src/config/providerManager.d.ts +0 -74
- package/dist/src/config/providerManager.js +0 -203
- package/dist/src/config/providerManager.js.map +0 -1
- package/dist/src/config/providerPersistence.test.d.ts +0 -6
- package/dist/src/config/providerPersistence.test.js +0 -283
- package/dist/src/config/providerPersistence.test.js.map +0 -1
- package/dist/src/ui/components/GeminiKeyDialog.d.ts +0 -11
- package/dist/src/ui/components/GeminiKeyDialog.js +0 -156
- package/dist/src/ui/components/GeminiKeyDialog.js.map +0 -1
- package/dist/src/ui/components/OpenAIEndpointDialog.d.ts +0 -19
- package/dist/src/ui/components/OpenAIEndpointDialog.js +0 -163
- package/dist/src/ui/components/OpenAIEndpointDialog.js.map +0 -1
- package/dist/test-setup.ts +0 -12
- package/dist/test-utils/customMatchers.ts +0 -65
- package/dist/test-utils/mockCommandContext.test.ts +0 -62
- package/dist/test-utils/mockCommandContext.ts +0 -105
- package/dist/test-utils/render.tsx +0 -18
- package/dist/ui/App.test.tsx +0 -2181
- package/dist/ui/App.tsx +0 -1344
- package/dist/ui/IdeIntegrationNudge.tsx +0 -98
- package/dist/ui/__snapshots__/App.test.tsx.snap +0 -124
- package/dist/ui/colors.ts +0 -56
- package/dist/ui/commands/aboutCommand.test.ts +0 -153
- package/dist/ui/commands/aboutCommand.ts +0 -49
- package/dist/ui/commands/authCommand.test.ts +0 -36
- package/dist/ui/commands/authCommand.ts +0 -17
- package/dist/ui/commands/bugCommand.test.ts +0 -114
- package/dist/ui/commands/bugCommand.ts +0 -92
- package/dist/ui/commands/chatCommand.test.ts +0 -414
- package/dist/ui/commands/chatCommand.ts +0 -280
- package/dist/ui/commands/clearCommand.test.ts +0 -100
- package/dist/ui/commands/clearCommand.ts +0 -29
- package/dist/ui/commands/compressCommand.test.ts +0 -129
- package/dist/ui/commands/compressCommand.ts +0 -78
- package/dist/ui/commands/contextCommand.ts +0 -132
- package/dist/ui/commands/copyCommand.test.ts +0 -296
- package/dist/ui/commands/copyCommand.ts +0 -67
- package/dist/ui/commands/corgiCommand.test.ts +0 -34
- package/dist/ui/commands/corgiCommand.ts +0 -16
- package/dist/ui/commands/directoryCommand.test.tsx +0 -185
- package/dist/ui/commands/directoryCommand.tsx +0 -179
- package/dist/ui/commands/docsCommand.test.ts +0 -99
- package/dist/ui/commands/docsCommand.ts +0 -42
- package/dist/ui/commands/editorCommand.test.ts +0 -30
- package/dist/ui/commands/editorCommand.ts +0 -21
- package/dist/ui/commands/extensionsCommand.test.ts +0 -67
- package/dist/ui/commands/extensionsCommand.ts +0 -46
- package/dist/ui/commands/helpCommand.test.ts +0 -52
- package/dist/ui/commands/helpCommand.ts +0 -23
- package/dist/ui/commands/ideCommand.test.ts +0 -255
- package/dist/ui/commands/ideCommand.ts +0 -283
- package/dist/ui/commands/initCommand.test.ts +0 -127
- package/dist/ui/commands/initCommand.ts +0 -117
- package/dist/ui/commands/mcpCommand.test.ts +0 -1057
- package/dist/ui/commands/mcpCommand.ts +0 -531
- package/dist/ui/commands/memoryCommand.test.ts +0 -344
- package/dist/ui/commands/memoryCommand.ts +0 -305
- package/dist/ui/commands/privacyCommand.test.ts +0 -38
- package/dist/ui/commands/privacyCommand.ts +0 -17
- package/dist/ui/commands/quitCommand.test.ts +0 -55
- package/dist/ui/commands/quitCommand.ts +0 -36
- package/dist/ui/commands/restoreCommand.test.ts +0 -250
- package/dist/ui/commands/restoreCommand.ts +0 -157
- package/dist/ui/commands/searchEngineSetupCommand.ts +0 -18
- package/dist/ui/commands/settingsCommand.test.ts +0 -36
- package/dist/ui/commands/settingsCommand.ts +0 -17
- package/dist/ui/commands/setupGithubCommand.test.ts +0 -238
- package/dist/ui/commands/setupGithubCommand.ts +0 -212
- package/dist/ui/commands/speakCommand.ts +0 -175
- package/dist/ui/commands/statsCommand.test.ts +0 -78
- package/dist/ui/commands/statsCommand.ts +0 -70
- package/dist/ui/commands/terminalSetupCommand.test.ts +0 -85
- package/dist/ui/commands/terminalSetupCommand.ts +0 -45
- package/dist/ui/commands/themeCommand.test.ts +0 -38
- package/dist/ui/commands/themeCommand.ts +0 -17
- package/dist/ui/commands/toolsCommand.test.ts +0 -105
- package/dist/ui/commands/toolsCommand.ts +0 -71
- package/dist/ui/commands/ttsCommand.ts +0 -143
- package/dist/ui/commands/types.ts +0 -204
- package/dist/ui/commands/vimCommand.ts +0 -25
- package/dist/ui/commands/voiceCommand.ts +0 -125
- package/dist/ui/components/AboutBox.tsx +0 -133
- package/dist/ui/components/AsciiArt.ts +0 -54
- package/dist/ui/components/AuthDialog.test.tsx +0 -334
- package/dist/ui/components/AuthDialog.tsx +0 -289
- package/dist/ui/components/AuthInProgress.tsx +0 -62
- package/dist/ui/components/AutoAcceptIndicator.tsx +0 -47
- package/dist/ui/components/ConsoleSummaryDisplay.tsx +0 -35
- package/dist/ui/components/ContextSummaryDisplay.test.tsx +0 -85
- package/dist/ui/components/ContextSummaryDisplay.tsx +0 -120
- package/dist/ui/components/ContextUsageDisplay.tsx +0 -77
- package/dist/ui/components/DebugProfiler.tsx +0 -36
- package/dist/ui/components/DetailedMessagesDisplay.tsx +0 -82
- package/dist/ui/components/EditorSettingsDialog.tsx +0 -172
- package/dist/ui/components/FolderTrustDialog.test.tsx +0 -36
- package/dist/ui/components/FolderTrustDialog.tsx +0 -74
- package/dist/ui/components/Footer.test.tsx +0 -159
- package/dist/ui/components/Footer.tsx +0 -158
- package/dist/ui/components/GeminiKeyDialog.tsx +0 -252
- package/dist/ui/components/GeminiRespondingSpinner.tsx +0 -34
- package/dist/ui/components/Header.test.tsx +0 -44
- package/dist/ui/components/Header.tsx +0 -70
- package/dist/ui/components/Help.tsx +0 -174
- package/dist/ui/components/HistoryItemDisplay.test.tsx +0 -125
- package/dist/ui/components/HistoryItemDisplay.tsx +0 -98
- package/dist/ui/components/InputPrompt.test.tsx +0 -1467
- package/dist/ui/components/InputPrompt.tsx +0 -641
- package/dist/ui/components/LMStudioModelPrompt.tsx +0 -215
- package/dist/ui/components/LoadingIndicator.test.tsx +0 -296
- package/dist/ui/components/LoadingIndicator.tsx +0 -82
- package/dist/ui/components/MemoryUsageDisplay.tsx +0 -36
- package/dist/ui/components/ModelStatsDisplay.test.tsx +0 -252
- package/dist/ui/components/ModelStatsDisplay.tsx +0 -197
- package/dist/ui/components/OllamaModelPrompt.tsx +0 -206
- package/dist/ui/components/OpenAIEndpointDialog.tsx +0 -261
- package/dist/ui/components/OpenAIKeyPrompt.test.tsx +0 -64
- package/dist/ui/components/OpenAIKeyPrompt.tsx +0 -197
- package/dist/ui/components/PrepareLabel.tsx +0 -48
- package/dist/ui/components/SearchEngineConfigDialog.tsx +0 -280
- package/dist/ui/components/SessionSummaryDisplay.test.tsx +0 -75
- package/dist/ui/components/SessionSummaryDisplay.tsx +0 -18
- package/dist/ui/components/SettingsDialog.test.tsx +0 -865
- package/dist/ui/components/SettingsDialog.tsx +0 -753
- package/dist/ui/components/ShellConfirmationDialog.test.tsx +0 -53
- package/dist/ui/components/ShellConfirmationDialog.tsx +0 -103
- package/dist/ui/components/ShellModeIndicator.tsx +0 -18
- package/dist/ui/components/ShowMoreLines.tsx +0 -40
- package/dist/ui/components/StatsDisplay.test.tsx +0 -401
- package/dist/ui/components/StatsDisplay.tsx +0 -273
- package/dist/ui/components/SuggestionsDisplay.tsx +0 -102
- package/dist/ui/components/ThemeDialog.tsx +0 -310
- package/dist/ui/components/Tips.tsx +0 -45
- package/dist/ui/components/TodoDisplay.test.tsx +0 -97
- package/dist/ui/components/TodoDisplay.tsx +0 -72
- package/dist/ui/components/ToolStatsDisplay.test.tsx +0 -180
- package/dist/ui/components/ToolStatsDisplay.tsx +0 -208
- package/dist/ui/components/UpdateNotification.tsx +0 -23
- package/dist/ui/components/WelcomeBackDialog.tsx +0 -290
- package/dist/ui/components/__snapshots__/IDEContextDetailDisplay.test.tsx.snap +0 -24
- package/dist/ui/components/__snapshots__/ModelStatsDisplay.test.tsx.snap +0 -121
- package/dist/ui/components/__snapshots__/SessionSummaryDisplay.test.tsx.snap +0 -30
- package/dist/ui/components/__snapshots__/ShellConfirmationDialog.test.tsx.snap +0 -21
- package/dist/ui/components/__snapshots__/StatsDisplay.test.tsx.snap +0 -264
- package/dist/ui/components/__snapshots__/ToolStatsDisplay.test.tsx.snap +0 -91
- package/dist/ui/components/messages/CompressionMessage.tsx +0 -49
- package/dist/ui/components/messages/DiffRenderer.test.tsx +0 -365
- package/dist/ui/components/messages/DiffRenderer.tsx +0 -358
- package/dist/ui/components/messages/ErrorMessage.tsx +0 -31
- package/dist/ui/components/messages/GeminiMessage.tsx +0 -43
- package/dist/ui/components/messages/GeminiMessageContent.tsx +0 -43
- package/dist/ui/components/messages/InfoMessage.tsx +0 -32
- package/dist/ui/components/messages/ToolConfirmationMessage.test.tsx +0 -58
- package/dist/ui/components/messages/ToolConfirmationMessage.tsx +0 -297
- package/dist/ui/components/messages/ToolGroupMessage.tsx +0 -126
- package/dist/ui/components/messages/ToolMessage.test.tsx +0 -183
- package/dist/ui/components/messages/ToolMessage.tsx +0 -296
- package/dist/ui/components/messages/UserMessage.tsx +0 -43
- package/dist/ui/components/messages/UserShellMessage.tsx +0 -25
- package/dist/ui/components/shared/MaxSizedBox.test.tsx +0 -425
- package/dist/ui/components/shared/MaxSizedBox.tsx +0 -624
- package/dist/ui/components/shared/RadioButtonSelect.test.tsx +0 -181
- package/dist/ui/components/shared/RadioButtonSelect.tsx +0 -234
- package/dist/ui/components/shared/__snapshots__/RadioButtonSelect.test.tsx.snap +0 -47
- package/dist/ui/components/shared/text-buffer.test.ts +0 -1728
- package/dist/ui/components/shared/text-buffer.ts +0 -2227
- package/dist/ui/components/shared/vim-buffer-actions.test.ts +0 -1119
- package/dist/ui/components/shared/vim-buffer-actions.ts +0 -814
- package/dist/ui/constants.ts +0 -17
- package/dist/ui/contexts/KeypressContext.test.tsx +0 -391
- package/dist/ui/contexts/KeypressContext.tsx +0 -440
- package/dist/ui/contexts/OverflowContext.tsx +0 -87
- package/dist/ui/contexts/SessionContext.test.tsx +0 -132
- package/dist/ui/contexts/SessionContext.tsx +0 -143
- package/dist/ui/contexts/SettingsContext.tsx +0 -20
- package/dist/ui/contexts/StreamingContext.tsx +0 -22
- package/dist/ui/contexts/VimModeContext.tsx +0 -79
- package/dist/ui/editors/editorSettingsManager.ts +0 -66
- package/dist/ui/hooks/atCommandProcessor.test.ts +0 -1102
- package/dist/ui/hooks/atCommandProcessor.ts +0 -485
- package/dist/ui/hooks/shellCommandProcessor.test.ts +0 -481
- package/dist/ui/hooks/shellCommandProcessor.ts +0 -314
- package/dist/ui/hooks/slashCommandProcessor.test.ts +0 -1044
- package/dist/ui/hooks/slashCommandProcessor.ts +0 -595
- package/dist/ui/hooks/useAtCompletion.test.ts +0 -497
- package/dist/ui/hooks/useAtCompletion.ts +0 -244
- package/dist/ui/hooks/useAuthCommand.ts +0 -129
- package/dist/ui/hooks/useAutoAcceptIndicator.test.ts +0 -300
- package/dist/ui/hooks/useAutoAcceptIndicator.ts +0 -52
- package/dist/ui/hooks/useBracketedPaste.ts +0 -37
- package/dist/ui/hooks/useCommandCompletion.test.ts +0 -518
- package/dist/ui/hooks/useCommandCompletion.tsx +0 -238
- package/dist/ui/hooks/useCompletion.ts +0 -128
- package/dist/ui/hooks/useConsoleMessages.test.ts +0 -147
- package/dist/ui/hooks/useConsoleMessages.ts +0 -110
- package/dist/ui/hooks/useEditorSettings.test.ts +0 -283
- package/dist/ui/hooks/useEditorSettings.ts +0 -75
- package/dist/ui/hooks/useFocus.test.ts +0 -119
- package/dist/ui/hooks/useFocus.ts +0 -48
- package/dist/ui/hooks/useFolderTrust.test.ts +0 -159
- package/dist/ui/hooks/useFolderTrust.ts +0 -72
- package/dist/ui/hooks/useGeminiStream.test.tsx +0 -1998
- package/dist/ui/hooks/useGeminiStream.ts +0 -1017
- package/dist/ui/hooks/useGitBranchName.test.ts +0 -280
- package/dist/ui/hooks/useGitBranchName.ts +0 -79
- package/dist/ui/hooks/useHistoryManager.test.ts +0 -202
- package/dist/ui/hooks/useHistoryManager.ts +0 -111
- package/dist/ui/hooks/useInputHistory.test.ts +0 -261
- package/dist/ui/hooks/useInputHistory.ts +0 -111
- package/dist/ui/hooks/useKeypress.test.ts +0 -280
- package/dist/ui/hooks/useKeypress.ts +0 -39
- package/dist/ui/hooks/useKittyKeyboardProtocol.ts +0 -31
- package/dist/ui/hooks/useLoadingIndicator.test.ts +0 -139
- package/dist/ui/hooks/useLoadingIndicator.ts +0 -57
- package/dist/ui/hooks/useLogger.ts +0 -32
- package/dist/ui/hooks/useMessageQueue.test.ts +0 -226
- package/dist/ui/hooks/useMessageQueue.ts +0 -69
- package/dist/ui/hooks/usePhraseCycler.test.ts +0 -145
- package/dist/ui/hooks/usePhraseCycler.ts +0 -198
- package/dist/ui/hooks/usePrivacySettings.test.ts +0 -242
- package/dist/ui/hooks/usePrivacySettings.ts +0 -150
- package/dist/ui/hooks/useReactToolScheduler.ts +0 -309
- package/dist/ui/hooks/useRefreshMemoryCommand.ts +0 -7
- package/dist/ui/hooks/useReverseSearchCompletion.test.tsx +0 -260
- package/dist/ui/hooks/useReverseSearchCompletion.tsx +0 -95
- package/dist/ui/hooks/useSettingsCommand.ts +0 -25
- package/dist/ui/hooks/useShellHistory.test.ts +0 -219
- package/dist/ui/hooks/useShellHistory.ts +0 -133
- package/dist/ui/hooks/useShowMemoryCommand.ts +0 -75
- package/dist/ui/hooks/useSlashCompletion.test.ts +0 -434
- package/dist/ui/hooks/useSlashCompletion.ts +0 -187
- package/dist/ui/hooks/useStateAndRef.ts +0 -36
- package/dist/ui/hooks/useTerminalSize.ts +0 -32
- package/dist/ui/hooks/useThemeCommand.ts +0 -110
- package/dist/ui/hooks/useTimer.test.ts +0 -120
- package/dist/ui/hooks/useTimer.ts +0 -65
- package/dist/ui/hooks/useToolScheduler.test.ts +0 -1123
- package/dist/ui/hooks/useWelcomeBack.ts +0 -253
- package/dist/ui/hooks/vim.test.ts +0 -1691
- package/dist/ui/hooks/vim.ts +0 -784
- package/dist/ui/keyMatchers.test.ts +0 -337
- package/dist/ui/keyMatchers.ts +0 -105
- package/dist/ui/privacy/CloudFreePrivacyNotice.tsx +0 -117
- package/dist/ui/privacy/CloudPaidPrivacyNotice.tsx +0 -59
- package/dist/ui/privacy/GeminiPrivacyNotice.tsx +0 -62
- package/dist/ui/privacy/PrivacyNotice.tsx +0 -42
- package/dist/ui/semantic-colors.ts +0 -26
- package/dist/ui/themes/ansi-light.ts +0 -150
- package/dist/ui/themes/ansi.ts +0 -159
- package/dist/ui/themes/atom-one-dark.ts +0 -147
- package/dist/ui/themes/ayu-light.ts +0 -139
- package/dist/ui/themes/ayu.ts +0 -113
- package/dist/ui/themes/color-utils.test.ts +0 -221
- package/dist/ui/themes/color-utils.ts +0 -231
- package/dist/ui/themes/default-light.ts +0 -108
- package/dist/ui/themes/default.ts +0 -151
- package/dist/ui/themes/dracula.ts +0 -124
- package/dist/ui/themes/fss-code-dark.ts +0 -156
- package/dist/ui/themes/fss-dark.ts +0 -113
- package/dist/ui/themes/fss-light.ts +0 -139
- package/dist/ui/themes/github-dark.ts +0 -147
- package/dist/ui/themes/github-light.ts +0 -149
- package/dist/ui/themes/googlecode.ts +0 -146
- package/dist/ui/themes/no-color.ts +0 -125
- package/dist/ui/themes/qwen-dark.ts +0 -118
- package/dist/ui/themes/qwen-light.ts +0 -144
- package/dist/ui/themes/semantic-tokens.ts +0 -127
- package/dist/ui/themes/shades-of-purple.ts +0 -352
- package/dist/ui/themes/theme-manager.test.ts +0 -99
- package/dist/ui/themes/theme-manager.ts +0 -257
- package/dist/ui/themes/theme.test.ts +0 -97
- package/dist/ui/themes/theme.ts +0 -451
- package/dist/ui/themes/xcode.ts +0 -154
- package/dist/ui/types.ts +0 -255
- package/dist/ui/utils/CodeColorizer.tsx +0 -217
- package/dist/ui/utils/ConsolePatcher.ts +0 -71
- package/dist/ui/utils/InlineMarkdownRenderer.tsx +0 -173
- package/dist/ui/utils/MarkdownDisplay.test.tsx +0 -244
- package/dist/ui/utils/MarkdownDisplay.tsx +0 -415
- package/dist/ui/utils/TableRenderer.tsx +0 -159
- package/dist/ui/utils/__snapshots__/MarkdownDisplay.test.tsx.snap +0 -93
- package/dist/ui/utils/clipboardUtils.test.ts +0 -76
- package/dist/ui/utils/clipboardUtils.ts +0 -149
- package/dist/ui/utils/commandUtils.test.ts +0 -384
- package/dist/ui/utils/commandUtils.ts +0 -106
- package/dist/ui/utils/computeStats.test.ts +0 -292
- package/dist/ui/utils/computeStats.ts +0 -86
- package/dist/ui/utils/displayUtils.test.ts +0 -58
- package/dist/ui/utils/displayUtils.ts +0 -32
- package/dist/ui/utils/formatters.test.ts +0 -72
- package/dist/ui/utils/formatters.ts +0 -63
- package/dist/ui/utils/isNarrowWidth.ts +0 -9
- package/dist/ui/utils/kittyProtocolDetector.ts +0 -105
- package/dist/ui/utils/markdownUtilities.test.ts +0 -50
- package/dist/ui/utils/markdownUtilities.ts +0 -125
- package/dist/ui/utils/platformConstants.ts +0 -52
- package/dist/ui/utils/terminalSetup.ts +0 -342
- package/dist/ui/utils/textUtils.ts +0 -40
- package/dist/ui/utils/updateCheck.test.ts +0 -163
- package/dist/ui/utils/updateCheck.ts +0 -100
- package/dist/utils/checks.ts +0 -28
- package/dist/utils/cleanup.test.ts +0 -68
- package/dist/utils/cleanup.ts +0 -36
- package/dist/utils/dialogScopeUtils.ts +0 -64
- package/dist/utils/events.ts +0 -14
- package/dist/utils/gitUtils.test.ts +0 -149
- package/dist/utils/gitUtils.ts +0 -116
- package/dist/utils/handleAutoUpdate.test.ts +0 -272
- package/dist/utils/handleAutoUpdate.ts +0 -145
- package/dist/utils/installationInfo.test.ts +0 -315
- package/dist/utils/installationInfo.ts +0 -176
- package/dist/utils/package.ts +0 -38
- package/dist/utils/readStdin.ts +0 -51
- package/dist/utils/resolvePath.ts +0 -21
- package/dist/utils/sandbox-macos-permissive-closed.sb +0 -32
- package/dist/utils/sandbox-macos-permissive-open.sb +0 -25
- package/dist/utils/sandbox-macos-permissive-proxied.sb +0 -37
- package/dist/utils/sandbox-macos-restrictive-closed.sb +0 -93
- package/dist/utils/sandbox-macos-restrictive-open.sb +0 -96
- package/dist/utils/sandbox-macos-restrictive-proxied.sb +0 -98
- package/dist/utils/sandbox.ts +0 -962
- package/dist/utils/settingsUtils.test.ts +0 -797
- package/dist/utils/settingsUtils.ts +0 -489
- package/dist/utils/spawnWrapper.ts +0 -9
- package/dist/utils/startupWarnings.test.ts +0 -83
- package/dist/utils/startupWarnings.ts +0 -40
- package/dist/utils/updateEventEmitter.ts +0 -13
- package/dist/utils/userStartupWarnings.test.ts +0 -87
- package/dist/utils/userStartupWarnings.ts +0 -69
- package/dist/utils/version.ts +0 -12
- package/dist/validateNonInterActiveAuth.test.ts +0 -260
- package/dist/validateNonInterActiveAuth.ts +0 -51
- package/dist/vitest.config.ts +0 -37
- package/dist/zed-integration/acp.ts +0 -366
- package/dist/zed-integration/fileSystemService.ts +0 -47
- package/dist/zed-integration/schema.ts +0 -466
- package/dist/zed-integration/zedIntegration.ts +0 -944
|
@@ -1,1057 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright 2025 Google LLC
|
|
4
|
-
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
-
*/
|
|
6
|
-
|
|
7
|
-
import { vi, describe, it, expect, beforeEach } from 'vitest';
|
|
8
|
-
import { mcpCommand } from './mcpCommand.js';
|
|
9
|
-
import { createMockCommandContext } from '../../test-utils/mockCommandContext.js';
|
|
10
|
-
import {
|
|
11
|
-
MCPServerStatus,
|
|
12
|
-
MCPDiscoveryState,
|
|
13
|
-
getMCPServerStatus,
|
|
14
|
-
getMCPDiscoveryState,
|
|
15
|
-
DiscoveredMCPTool,
|
|
16
|
-
} from 'fss-link-core';
|
|
17
|
-
|
|
18
|
-
import { MessageActionReturn } from './types.js';
|
|
19
|
-
import { Type, CallableTool } from '@google/genai';
|
|
20
|
-
|
|
21
|
-
vi.mock('fss-link-core', async (importOriginal) => {
|
|
22
|
-
const actual =
|
|
23
|
-
await importOriginal<typeof import('fss-link-core')>();
|
|
24
|
-
return {
|
|
25
|
-
...actual,
|
|
26
|
-
getMCPServerStatus: vi.fn(),
|
|
27
|
-
getMCPDiscoveryState: vi.fn(),
|
|
28
|
-
MCPOAuthProvider: {
|
|
29
|
-
authenticate: vi.fn(),
|
|
30
|
-
},
|
|
31
|
-
MCPOAuthTokenStorage: {
|
|
32
|
-
getToken: vi.fn(),
|
|
33
|
-
isTokenExpired: vi.fn(),
|
|
34
|
-
},
|
|
35
|
-
};
|
|
36
|
-
});
|
|
37
|
-
|
|
38
|
-
// Helper function to check if result is a message action
|
|
39
|
-
const isMessageAction = (result: unknown): result is MessageActionReturn =>
|
|
40
|
-
result !== null &&
|
|
41
|
-
typeof result === 'object' &&
|
|
42
|
-
'type' in result &&
|
|
43
|
-
result.type === 'message';
|
|
44
|
-
|
|
45
|
-
// Helper function to create a mock DiscoveredMCPTool
|
|
46
|
-
const createMockMCPTool = (
|
|
47
|
-
name: string,
|
|
48
|
-
serverName: string,
|
|
49
|
-
description?: string,
|
|
50
|
-
) =>
|
|
51
|
-
new DiscoveredMCPTool(
|
|
52
|
-
{
|
|
53
|
-
callTool: vi.fn(),
|
|
54
|
-
tool: vi.fn(),
|
|
55
|
-
} as unknown as CallableTool,
|
|
56
|
-
serverName,
|
|
57
|
-
name,
|
|
58
|
-
description || `Description for ${name}`,
|
|
59
|
-
{ type: Type.OBJECT, properties: {} },
|
|
60
|
-
name, // serverToolName same as name for simplicity
|
|
61
|
-
);
|
|
62
|
-
|
|
63
|
-
describe('mcpCommand', () => {
|
|
64
|
-
let mockContext: ReturnType<typeof createMockCommandContext>;
|
|
65
|
-
let mockConfig: {
|
|
66
|
-
getToolRegistry: ReturnType<typeof vi.fn>;
|
|
67
|
-
getMcpServers: ReturnType<typeof vi.fn>;
|
|
68
|
-
getBlockedMcpServers: ReturnType<typeof vi.fn>;
|
|
69
|
-
getPromptRegistry: ReturnType<typeof vi.fn>;
|
|
70
|
-
};
|
|
71
|
-
|
|
72
|
-
beforeEach(() => {
|
|
73
|
-
vi.clearAllMocks();
|
|
74
|
-
|
|
75
|
-
// Set up default mock environment
|
|
76
|
-
vi.unstubAllEnvs();
|
|
77
|
-
|
|
78
|
-
// Default mock implementations
|
|
79
|
-
vi.mocked(getMCPServerStatus).mockReturnValue(MCPServerStatus.CONNECTED);
|
|
80
|
-
vi.mocked(getMCPDiscoveryState).mockReturnValue(
|
|
81
|
-
MCPDiscoveryState.COMPLETED,
|
|
82
|
-
);
|
|
83
|
-
|
|
84
|
-
// Create mock config with all necessary methods
|
|
85
|
-
mockConfig = {
|
|
86
|
-
getToolRegistry: vi.fn().mockReturnValue({
|
|
87
|
-
getAllTools: vi.fn().mockReturnValue([]),
|
|
88
|
-
}),
|
|
89
|
-
getMcpServers: vi.fn().mockReturnValue({}),
|
|
90
|
-
getBlockedMcpServers: vi.fn().mockReturnValue([]),
|
|
91
|
-
getPromptRegistry: vi.fn().mockResolvedValue({
|
|
92
|
-
getAllPrompts: vi.fn().mockReturnValue([]),
|
|
93
|
-
getPromptsByServer: vi.fn().mockReturnValue([]),
|
|
94
|
-
}),
|
|
95
|
-
};
|
|
96
|
-
|
|
97
|
-
mockContext = createMockCommandContext({
|
|
98
|
-
services: {
|
|
99
|
-
config: mockConfig,
|
|
100
|
-
},
|
|
101
|
-
});
|
|
102
|
-
});
|
|
103
|
-
|
|
104
|
-
describe('basic functionality', () => {
|
|
105
|
-
it('should show an error if config is not available', async () => {
|
|
106
|
-
const contextWithoutConfig = createMockCommandContext({
|
|
107
|
-
services: {
|
|
108
|
-
config: null,
|
|
109
|
-
},
|
|
110
|
-
});
|
|
111
|
-
|
|
112
|
-
const result = await mcpCommand.action!(contextWithoutConfig, '');
|
|
113
|
-
|
|
114
|
-
expect(result).toEqual({
|
|
115
|
-
type: 'message',
|
|
116
|
-
messageType: 'error',
|
|
117
|
-
content: 'Config not loaded.',
|
|
118
|
-
});
|
|
119
|
-
});
|
|
120
|
-
|
|
121
|
-
it('should show an error if tool registry is not available', async () => {
|
|
122
|
-
mockConfig.getToolRegistry = vi.fn().mockReturnValue(undefined);
|
|
123
|
-
|
|
124
|
-
const result = await mcpCommand.action!(mockContext, '');
|
|
125
|
-
|
|
126
|
-
expect(result).toEqual({
|
|
127
|
-
type: 'message',
|
|
128
|
-
messageType: 'error',
|
|
129
|
-
content: 'Could not retrieve tool registry.',
|
|
130
|
-
});
|
|
131
|
-
});
|
|
132
|
-
});
|
|
133
|
-
|
|
134
|
-
describe('no MCP servers configured', () => {
|
|
135
|
-
beforeEach(() => {
|
|
136
|
-
mockConfig.getToolRegistry = vi.fn().mockReturnValue({
|
|
137
|
-
getAllTools: vi.fn().mockReturnValue([]),
|
|
138
|
-
});
|
|
139
|
-
mockConfig.getMcpServers = vi.fn().mockReturnValue({});
|
|
140
|
-
});
|
|
141
|
-
|
|
142
|
-
it('should display a message with a URL when no MCP servers are configured', async () => {
|
|
143
|
-
const result = await mcpCommand.action!(mockContext, '');
|
|
144
|
-
|
|
145
|
-
expect(result).toEqual({
|
|
146
|
-
type: 'message',
|
|
147
|
-
messageType: 'info',
|
|
148
|
-
content:
|
|
149
|
-
'No MCP servers configured. Please view MCP documentation in your browser: https://docs.anthropic.com/en/docs/claude-code/tools/mcp-server/#how-to-set-up-your-mcp-server or use the cli /docs command',
|
|
150
|
-
});
|
|
151
|
-
});
|
|
152
|
-
});
|
|
153
|
-
|
|
154
|
-
describe('with configured MCP servers', () => {
|
|
155
|
-
beforeEach(() => {
|
|
156
|
-
const mockMcpServers = {
|
|
157
|
-
server1: { command: 'cmd1' },
|
|
158
|
-
server2: { command: 'cmd2' },
|
|
159
|
-
server3: { command: 'cmd3' },
|
|
160
|
-
};
|
|
161
|
-
|
|
162
|
-
mockConfig.getMcpServers = vi.fn().mockReturnValue(mockMcpServers);
|
|
163
|
-
});
|
|
164
|
-
|
|
165
|
-
it('should display configured MCP servers with status indicators and their tools', async () => {
|
|
166
|
-
// Setup getMCPServerStatus mock implementation
|
|
167
|
-
vi.mocked(getMCPServerStatus).mockImplementation((serverName) => {
|
|
168
|
-
if (serverName === 'server1') return MCPServerStatus.CONNECTED;
|
|
169
|
-
if (serverName === 'server2') return MCPServerStatus.CONNECTED;
|
|
170
|
-
return MCPServerStatus.DISCONNECTED; // server3
|
|
171
|
-
});
|
|
172
|
-
|
|
173
|
-
// Mock tools from each server using actual DiscoveredMCPTool instances
|
|
174
|
-
const mockServer1Tools = [
|
|
175
|
-
createMockMCPTool('server1_tool1', 'server1'),
|
|
176
|
-
createMockMCPTool('server1_tool2', 'server1'),
|
|
177
|
-
];
|
|
178
|
-
const mockServer2Tools = [createMockMCPTool('server2_tool1', 'server2')];
|
|
179
|
-
const mockServer3Tools = [createMockMCPTool('server3_tool1', 'server3')];
|
|
180
|
-
|
|
181
|
-
const allTools = [
|
|
182
|
-
...mockServer1Tools,
|
|
183
|
-
...mockServer2Tools,
|
|
184
|
-
...mockServer3Tools,
|
|
185
|
-
];
|
|
186
|
-
|
|
187
|
-
mockConfig.getToolRegistry = vi.fn().mockReturnValue({
|
|
188
|
-
getAllTools: vi.fn().mockReturnValue(allTools),
|
|
189
|
-
});
|
|
190
|
-
|
|
191
|
-
const result = await mcpCommand.action!(mockContext, '');
|
|
192
|
-
|
|
193
|
-
expect(result).toEqual({
|
|
194
|
-
type: 'message',
|
|
195
|
-
messageType: 'info',
|
|
196
|
-
content: expect.stringContaining('Configured MCP servers:'),
|
|
197
|
-
});
|
|
198
|
-
|
|
199
|
-
expect(isMessageAction(result)).toBe(true);
|
|
200
|
-
if (isMessageAction(result)) {
|
|
201
|
-
const message = result.content;
|
|
202
|
-
// Server 1 - Connected
|
|
203
|
-
expect(message).toContain(
|
|
204
|
-
'🟢 \u001b[1mserver1\u001b[0m - Ready (2 tools)',
|
|
205
|
-
);
|
|
206
|
-
expect(message).toContain('server1_tool1');
|
|
207
|
-
expect(message).toContain('server1_tool2');
|
|
208
|
-
|
|
209
|
-
// Server 2 - Connected
|
|
210
|
-
expect(message).toContain(
|
|
211
|
-
'🟢 \u001b[1mserver2\u001b[0m - Ready (1 tool)',
|
|
212
|
-
);
|
|
213
|
-
expect(message).toContain('server2_tool1');
|
|
214
|
-
|
|
215
|
-
// Server 3 - Disconnected but with cached tools, so shows as Ready
|
|
216
|
-
expect(message).toContain(
|
|
217
|
-
'🟢 \u001b[1mserver3\u001b[0m - Ready (1 tool)',
|
|
218
|
-
);
|
|
219
|
-
expect(message).toContain('server3_tool1');
|
|
220
|
-
|
|
221
|
-
// Check that helpful tips are displayed when no arguments are provided
|
|
222
|
-
expect(message).toContain('💡 Tips:');
|
|
223
|
-
expect(message).toContain('/mcp desc');
|
|
224
|
-
expect(message).toContain('/mcp schema');
|
|
225
|
-
expect(message).toContain('/mcp nodesc');
|
|
226
|
-
expect(message).toContain('Ctrl+T');
|
|
227
|
-
}
|
|
228
|
-
});
|
|
229
|
-
|
|
230
|
-
it('should display tool descriptions when desc argument is used', async () => {
|
|
231
|
-
const mockMcpServers = {
|
|
232
|
-
server1: {
|
|
233
|
-
command: 'cmd1',
|
|
234
|
-
description: 'This is a server description',
|
|
235
|
-
},
|
|
236
|
-
};
|
|
237
|
-
|
|
238
|
-
mockConfig.getMcpServers = vi.fn().mockReturnValue(mockMcpServers);
|
|
239
|
-
|
|
240
|
-
// Mock tools with descriptions using actual DiscoveredMCPTool instances
|
|
241
|
-
const mockServerTools = [
|
|
242
|
-
createMockMCPTool('tool1', 'server1', 'This is tool 1 description'),
|
|
243
|
-
createMockMCPTool('tool2', 'server1', 'This is tool 2 description'),
|
|
244
|
-
];
|
|
245
|
-
|
|
246
|
-
mockConfig.getToolRegistry = vi.fn().mockReturnValue({
|
|
247
|
-
getAllTools: vi.fn().mockReturnValue(mockServerTools),
|
|
248
|
-
});
|
|
249
|
-
|
|
250
|
-
const result = await mcpCommand.action!(mockContext, 'desc');
|
|
251
|
-
|
|
252
|
-
expect(result).toEqual({
|
|
253
|
-
type: 'message',
|
|
254
|
-
messageType: 'info',
|
|
255
|
-
content: expect.stringContaining('Configured MCP servers:'),
|
|
256
|
-
});
|
|
257
|
-
|
|
258
|
-
expect(isMessageAction(result)).toBe(true);
|
|
259
|
-
if (isMessageAction(result)) {
|
|
260
|
-
const message = result.content;
|
|
261
|
-
|
|
262
|
-
// Check that server description is included
|
|
263
|
-
expect(message).toContain(
|
|
264
|
-
'\u001b[1mserver1\u001b[0m - Ready (2 tools)',
|
|
265
|
-
);
|
|
266
|
-
expect(message).toContain(
|
|
267
|
-
'\u001b[32mThis is a server description\u001b[0m',
|
|
268
|
-
);
|
|
269
|
-
|
|
270
|
-
// Check that tool descriptions are included
|
|
271
|
-
expect(message).toContain('\u001b[36mtool1\u001b[0m');
|
|
272
|
-
expect(message).toContain(
|
|
273
|
-
'\u001b[32mThis is tool 1 description\u001b[0m',
|
|
274
|
-
);
|
|
275
|
-
expect(message).toContain('\u001b[36mtool2\u001b[0m');
|
|
276
|
-
expect(message).toContain(
|
|
277
|
-
'\u001b[32mThis is tool 2 description\u001b[0m',
|
|
278
|
-
);
|
|
279
|
-
|
|
280
|
-
// Check that tips are NOT displayed when arguments are provided
|
|
281
|
-
expect(message).not.toContain('💡 Tips:');
|
|
282
|
-
}
|
|
283
|
-
});
|
|
284
|
-
|
|
285
|
-
it('should not display descriptions when nodesc argument is used', async () => {
|
|
286
|
-
const mockMcpServers = {
|
|
287
|
-
server1: {
|
|
288
|
-
command: 'cmd1',
|
|
289
|
-
description: 'This is a server description',
|
|
290
|
-
},
|
|
291
|
-
};
|
|
292
|
-
|
|
293
|
-
mockConfig.getMcpServers = vi.fn().mockReturnValue(mockMcpServers);
|
|
294
|
-
|
|
295
|
-
const mockServerTools = [
|
|
296
|
-
createMockMCPTool('tool1', 'server1', 'This is tool 1 description'),
|
|
297
|
-
];
|
|
298
|
-
|
|
299
|
-
mockConfig.getToolRegistry = vi.fn().mockReturnValue({
|
|
300
|
-
getAllTools: vi.fn().mockReturnValue(mockServerTools),
|
|
301
|
-
});
|
|
302
|
-
|
|
303
|
-
const result = await mcpCommand.action!(mockContext, 'nodesc');
|
|
304
|
-
|
|
305
|
-
expect(result).toEqual({
|
|
306
|
-
type: 'message',
|
|
307
|
-
messageType: 'info',
|
|
308
|
-
content: expect.stringContaining('Configured MCP servers:'),
|
|
309
|
-
});
|
|
310
|
-
|
|
311
|
-
expect(isMessageAction(result)).toBe(true);
|
|
312
|
-
if (isMessageAction(result)) {
|
|
313
|
-
const message = result.content;
|
|
314
|
-
|
|
315
|
-
// Check that descriptions are not included
|
|
316
|
-
expect(message).not.toContain('This is a server description');
|
|
317
|
-
expect(message).not.toContain('This is tool 1 description');
|
|
318
|
-
expect(message).toContain('\u001b[36mtool1\u001b[0m');
|
|
319
|
-
|
|
320
|
-
// Check that tips are NOT displayed when arguments are provided
|
|
321
|
-
expect(message).not.toContain('💡 Tips:');
|
|
322
|
-
}
|
|
323
|
-
});
|
|
324
|
-
|
|
325
|
-
it('should indicate when a server has no tools', async () => {
|
|
326
|
-
const mockMcpServers = {
|
|
327
|
-
server1: { command: 'cmd1' },
|
|
328
|
-
server2: { command: 'cmd2' },
|
|
329
|
-
};
|
|
330
|
-
|
|
331
|
-
mockConfig.getMcpServers = vi.fn().mockReturnValue(mockMcpServers);
|
|
332
|
-
|
|
333
|
-
// Setup server statuses
|
|
334
|
-
vi.mocked(getMCPServerStatus).mockImplementation((serverName) => {
|
|
335
|
-
if (serverName === 'server1') return MCPServerStatus.CONNECTED;
|
|
336
|
-
if (serverName === 'server2') return MCPServerStatus.DISCONNECTED;
|
|
337
|
-
return MCPServerStatus.DISCONNECTED;
|
|
338
|
-
});
|
|
339
|
-
|
|
340
|
-
// Mock tools - only server1 has tools
|
|
341
|
-
const mockServerTools = [createMockMCPTool('server1_tool1', 'server1')];
|
|
342
|
-
|
|
343
|
-
mockConfig.getToolRegistry = vi.fn().mockReturnValue({
|
|
344
|
-
getAllTools: vi.fn().mockReturnValue(mockServerTools),
|
|
345
|
-
});
|
|
346
|
-
|
|
347
|
-
const result = await mcpCommand.action!(mockContext, '');
|
|
348
|
-
|
|
349
|
-
expect(isMessageAction(result)).toBe(true);
|
|
350
|
-
if (isMessageAction(result)) {
|
|
351
|
-
const message = result.content;
|
|
352
|
-
expect(message).toContain(
|
|
353
|
-
'🟢 \u001b[1mserver1\u001b[0m - Ready (1 tool)',
|
|
354
|
-
);
|
|
355
|
-
expect(message).toContain('\u001b[36mserver1_tool1\u001b[0m');
|
|
356
|
-
expect(message).toContain(
|
|
357
|
-
'🔴 \u001b[1mserver2\u001b[0m - Disconnected (0 tools cached)',
|
|
358
|
-
);
|
|
359
|
-
expect(message).toContain('No tools or prompts available');
|
|
360
|
-
}
|
|
361
|
-
});
|
|
362
|
-
|
|
363
|
-
it('should show startup indicator when servers are connecting', async () => {
|
|
364
|
-
const mockMcpServers = {
|
|
365
|
-
server1: { command: 'cmd1' },
|
|
366
|
-
server2: { command: 'cmd2' },
|
|
367
|
-
};
|
|
368
|
-
|
|
369
|
-
mockConfig.getMcpServers = vi.fn().mockReturnValue(mockMcpServers);
|
|
370
|
-
|
|
371
|
-
// Setup server statuses with one connecting
|
|
372
|
-
vi.mocked(getMCPServerStatus).mockImplementation((serverName) => {
|
|
373
|
-
if (serverName === 'server1') return MCPServerStatus.CONNECTED;
|
|
374
|
-
if (serverName === 'server2') return MCPServerStatus.CONNECTING;
|
|
375
|
-
return MCPServerStatus.DISCONNECTED;
|
|
376
|
-
});
|
|
377
|
-
|
|
378
|
-
// Setup discovery state as in progress
|
|
379
|
-
vi.mocked(getMCPDiscoveryState).mockReturnValue(
|
|
380
|
-
MCPDiscoveryState.IN_PROGRESS,
|
|
381
|
-
);
|
|
382
|
-
|
|
383
|
-
// Mock tools
|
|
384
|
-
const mockServerTools = [
|
|
385
|
-
createMockMCPTool('server1_tool1', 'server1'),
|
|
386
|
-
createMockMCPTool('server2_tool1', 'server2'),
|
|
387
|
-
];
|
|
388
|
-
|
|
389
|
-
mockConfig.getToolRegistry = vi.fn().mockReturnValue({
|
|
390
|
-
getAllTools: vi.fn().mockReturnValue(mockServerTools),
|
|
391
|
-
});
|
|
392
|
-
|
|
393
|
-
const result = await mcpCommand.action!(mockContext, '');
|
|
394
|
-
|
|
395
|
-
expect(isMessageAction(result)).toBe(true);
|
|
396
|
-
if (isMessageAction(result)) {
|
|
397
|
-
const message = result.content;
|
|
398
|
-
|
|
399
|
-
// Check that startup indicator is shown
|
|
400
|
-
expect(message).toContain(
|
|
401
|
-
'⏳ MCP servers are starting up (1 initializing)...',
|
|
402
|
-
);
|
|
403
|
-
expect(message).toContain(
|
|
404
|
-
'Note: First startup may take longer. Tool availability will update automatically.',
|
|
405
|
-
);
|
|
406
|
-
|
|
407
|
-
// Check server statuses
|
|
408
|
-
expect(message).toContain(
|
|
409
|
-
'🟢 \u001b[1mserver1\u001b[0m - Ready (1 tool)',
|
|
410
|
-
);
|
|
411
|
-
expect(message).toContain(
|
|
412
|
-
'🔄 \u001b[1mserver2\u001b[0m - Starting... (first startup may take longer) (tools and prompts will appear when ready)',
|
|
413
|
-
);
|
|
414
|
-
}
|
|
415
|
-
});
|
|
416
|
-
|
|
417
|
-
it('should display the extension name for servers from extensions', async () => {
|
|
418
|
-
const mockMcpServers = {
|
|
419
|
-
server1: { command: 'cmd1', extensionName: 'my-extension' },
|
|
420
|
-
};
|
|
421
|
-
mockConfig.getMcpServers = vi.fn().mockReturnValue(mockMcpServers);
|
|
422
|
-
|
|
423
|
-
const result = await mcpCommand.action!(mockContext, '');
|
|
424
|
-
|
|
425
|
-
expect(isMessageAction(result)).toBe(true);
|
|
426
|
-
if (isMessageAction(result)) {
|
|
427
|
-
const message = result.content;
|
|
428
|
-
expect(message).toContain('server1 (from my-extension)');
|
|
429
|
-
}
|
|
430
|
-
});
|
|
431
|
-
|
|
432
|
-
it('should display blocked MCP servers', async () => {
|
|
433
|
-
mockConfig.getMcpServers = vi.fn().mockReturnValue({});
|
|
434
|
-
const blockedServers = [
|
|
435
|
-
{ name: 'blocked-server', extensionName: 'my-extension' },
|
|
436
|
-
];
|
|
437
|
-
mockConfig.getBlockedMcpServers = vi.fn().mockReturnValue(blockedServers);
|
|
438
|
-
|
|
439
|
-
const result = await mcpCommand.action!(mockContext, '');
|
|
440
|
-
|
|
441
|
-
expect(isMessageAction(result)).toBe(true);
|
|
442
|
-
if (isMessageAction(result)) {
|
|
443
|
-
const message = result.content;
|
|
444
|
-
expect(message).toContain(
|
|
445
|
-
'🔴 \u001b[1mblocked-server (from my-extension)\u001b[0m - Blocked',
|
|
446
|
-
);
|
|
447
|
-
}
|
|
448
|
-
});
|
|
449
|
-
|
|
450
|
-
it('should display both active and blocked servers correctly', async () => {
|
|
451
|
-
const mockMcpServers = {
|
|
452
|
-
server1: { command: 'cmd1', extensionName: 'my-extension' },
|
|
453
|
-
};
|
|
454
|
-
mockConfig.getMcpServers = vi.fn().mockReturnValue(mockMcpServers);
|
|
455
|
-
const blockedServers = [
|
|
456
|
-
{ name: 'blocked-server', extensionName: 'another-extension' },
|
|
457
|
-
];
|
|
458
|
-
mockConfig.getBlockedMcpServers = vi.fn().mockReturnValue(blockedServers);
|
|
459
|
-
|
|
460
|
-
const result = await mcpCommand.action!(mockContext, '');
|
|
461
|
-
|
|
462
|
-
expect(isMessageAction(result)).toBe(true);
|
|
463
|
-
if (isMessageAction(result)) {
|
|
464
|
-
const message = result.content;
|
|
465
|
-
expect(message).toContain('server1 (from my-extension)');
|
|
466
|
-
expect(message).toContain(
|
|
467
|
-
'🔴 \u001b[1mblocked-server (from another-extension)\u001b[0m - Blocked',
|
|
468
|
-
);
|
|
469
|
-
}
|
|
470
|
-
});
|
|
471
|
-
});
|
|
472
|
-
|
|
473
|
-
describe('schema functionality', () => {
|
|
474
|
-
it('should display tool schemas when schema argument is used', async () => {
|
|
475
|
-
const mockMcpServers = {
|
|
476
|
-
server1: {
|
|
477
|
-
command: 'cmd1',
|
|
478
|
-
description: 'This is a server description',
|
|
479
|
-
},
|
|
480
|
-
};
|
|
481
|
-
|
|
482
|
-
mockConfig.getMcpServers = vi.fn().mockReturnValue(mockMcpServers);
|
|
483
|
-
|
|
484
|
-
// Create tools with parameter schemas
|
|
485
|
-
const mockCallableTool1: CallableTool = {
|
|
486
|
-
callTool: vi.fn(),
|
|
487
|
-
tool: vi.fn(),
|
|
488
|
-
} as unknown as CallableTool;
|
|
489
|
-
const mockCallableTool2: CallableTool = {
|
|
490
|
-
callTool: vi.fn(),
|
|
491
|
-
tool: vi.fn(),
|
|
492
|
-
} as unknown as CallableTool;
|
|
493
|
-
|
|
494
|
-
const tool1 = new DiscoveredMCPTool(
|
|
495
|
-
mockCallableTool1,
|
|
496
|
-
'server1',
|
|
497
|
-
'tool1',
|
|
498
|
-
'This is tool 1 description',
|
|
499
|
-
{
|
|
500
|
-
type: Type.OBJECT,
|
|
501
|
-
properties: {
|
|
502
|
-
param1: { type: Type.STRING, description: 'First parameter' },
|
|
503
|
-
},
|
|
504
|
-
required: ['param1'],
|
|
505
|
-
},
|
|
506
|
-
'tool1',
|
|
507
|
-
);
|
|
508
|
-
|
|
509
|
-
const tool2 = new DiscoveredMCPTool(
|
|
510
|
-
mockCallableTool2,
|
|
511
|
-
'server1',
|
|
512
|
-
'tool2',
|
|
513
|
-
'This is tool 2 description',
|
|
514
|
-
{
|
|
515
|
-
type: Type.OBJECT,
|
|
516
|
-
properties: {
|
|
517
|
-
param2: { type: Type.NUMBER, description: 'Second parameter' },
|
|
518
|
-
},
|
|
519
|
-
required: ['param2'],
|
|
520
|
-
},
|
|
521
|
-
'tool2',
|
|
522
|
-
);
|
|
523
|
-
|
|
524
|
-
const mockServerTools = [tool1, tool2];
|
|
525
|
-
|
|
526
|
-
mockConfig.getToolRegistry = vi.fn().mockReturnValue({
|
|
527
|
-
getAllTools: vi.fn().mockReturnValue(mockServerTools),
|
|
528
|
-
});
|
|
529
|
-
|
|
530
|
-
const result = await mcpCommand.action!(mockContext, 'schema');
|
|
531
|
-
|
|
532
|
-
expect(result).toEqual({
|
|
533
|
-
type: 'message',
|
|
534
|
-
messageType: 'info',
|
|
535
|
-
content: expect.stringContaining('Configured MCP servers:'),
|
|
536
|
-
});
|
|
537
|
-
|
|
538
|
-
expect(isMessageAction(result)).toBe(true);
|
|
539
|
-
if (isMessageAction(result)) {
|
|
540
|
-
const message = result.content;
|
|
541
|
-
|
|
542
|
-
// Check that server description is included
|
|
543
|
-
expect(message).toContain('Ready (2 tools)');
|
|
544
|
-
expect(message).toContain('This is a server description');
|
|
545
|
-
|
|
546
|
-
// Check that tool descriptions and schemas are included
|
|
547
|
-
expect(message).toContain('This is tool 1 description');
|
|
548
|
-
expect(message).toContain('Parameters:');
|
|
549
|
-
expect(message).toContain('param1');
|
|
550
|
-
expect(message).toContain('STRING');
|
|
551
|
-
expect(message).toContain('This is tool 2 description');
|
|
552
|
-
expect(message).toContain('param2');
|
|
553
|
-
expect(message).toContain('NUMBER');
|
|
554
|
-
}
|
|
555
|
-
});
|
|
556
|
-
|
|
557
|
-
it('should handle tools without parameter schemas gracefully', async () => {
|
|
558
|
-
const mockMcpServers = {
|
|
559
|
-
server1: { command: 'cmd1' },
|
|
560
|
-
};
|
|
561
|
-
|
|
562
|
-
mockConfig.getMcpServers = vi.fn().mockReturnValue(mockMcpServers);
|
|
563
|
-
|
|
564
|
-
// Mock tools without parameter schemas
|
|
565
|
-
const mockServerTools = [
|
|
566
|
-
createMockMCPTool('tool1', 'server1', 'Tool without schema'),
|
|
567
|
-
];
|
|
568
|
-
|
|
569
|
-
mockConfig.getToolRegistry = vi.fn().mockReturnValue({
|
|
570
|
-
getAllTools: vi.fn().mockReturnValue(mockServerTools),
|
|
571
|
-
});
|
|
572
|
-
|
|
573
|
-
const result = await mcpCommand.action!(mockContext, 'schema');
|
|
574
|
-
|
|
575
|
-
expect(result).toEqual({
|
|
576
|
-
type: 'message',
|
|
577
|
-
messageType: 'info',
|
|
578
|
-
content: expect.stringContaining('Configured MCP servers:'),
|
|
579
|
-
});
|
|
580
|
-
|
|
581
|
-
expect(isMessageAction(result)).toBe(true);
|
|
582
|
-
if (isMessageAction(result)) {
|
|
583
|
-
const message = result.content;
|
|
584
|
-
expect(message).toContain('tool1');
|
|
585
|
-
expect(message).toContain('Tool without schema');
|
|
586
|
-
// Should not crash when parameterSchema is undefined
|
|
587
|
-
}
|
|
588
|
-
});
|
|
589
|
-
});
|
|
590
|
-
|
|
591
|
-
describe('argument parsing', () => {
|
|
592
|
-
beforeEach(() => {
|
|
593
|
-
const mockMcpServers = {
|
|
594
|
-
server1: {
|
|
595
|
-
command: 'cmd1',
|
|
596
|
-
description: 'Server description',
|
|
597
|
-
},
|
|
598
|
-
};
|
|
599
|
-
|
|
600
|
-
mockConfig.getMcpServers = vi.fn().mockReturnValue(mockMcpServers);
|
|
601
|
-
|
|
602
|
-
const mockServerTools = [
|
|
603
|
-
createMockMCPTool('tool1', 'server1', 'Test tool'),
|
|
604
|
-
];
|
|
605
|
-
|
|
606
|
-
mockConfig.getToolRegistry = vi.fn().mockReturnValue({
|
|
607
|
-
getAllTools: vi.fn().mockReturnValue(mockServerTools),
|
|
608
|
-
});
|
|
609
|
-
});
|
|
610
|
-
|
|
611
|
-
it('should handle "descriptions" as alias for "desc"', async () => {
|
|
612
|
-
const result = await mcpCommand.action!(mockContext, 'descriptions');
|
|
613
|
-
|
|
614
|
-
expect(isMessageAction(result)).toBe(true);
|
|
615
|
-
if (isMessageAction(result)) {
|
|
616
|
-
const message = result.content;
|
|
617
|
-
expect(message).toContain('Test tool');
|
|
618
|
-
expect(message).toContain('Server description');
|
|
619
|
-
}
|
|
620
|
-
});
|
|
621
|
-
|
|
622
|
-
it('should handle "nodescriptions" as alias for "nodesc"', async () => {
|
|
623
|
-
const result = await mcpCommand.action!(mockContext, 'nodescriptions');
|
|
624
|
-
|
|
625
|
-
expect(isMessageAction(result)).toBe(true);
|
|
626
|
-
if (isMessageAction(result)) {
|
|
627
|
-
const message = result.content;
|
|
628
|
-
expect(message).not.toContain('Test tool');
|
|
629
|
-
expect(message).not.toContain('Server description');
|
|
630
|
-
expect(message).toContain('\u001b[36mtool1\u001b[0m');
|
|
631
|
-
}
|
|
632
|
-
});
|
|
633
|
-
|
|
634
|
-
it('should handle mixed case arguments', async () => {
|
|
635
|
-
const result = await mcpCommand.action!(mockContext, 'DESC');
|
|
636
|
-
|
|
637
|
-
expect(isMessageAction(result)).toBe(true);
|
|
638
|
-
if (isMessageAction(result)) {
|
|
639
|
-
const message = result.content;
|
|
640
|
-
expect(message).toContain('Test tool');
|
|
641
|
-
expect(message).toContain('Server description');
|
|
642
|
-
}
|
|
643
|
-
});
|
|
644
|
-
|
|
645
|
-
it('should handle multiple arguments - "schema desc"', async () => {
|
|
646
|
-
const result = await mcpCommand.action!(mockContext, 'schema desc');
|
|
647
|
-
|
|
648
|
-
expect(isMessageAction(result)).toBe(true);
|
|
649
|
-
if (isMessageAction(result)) {
|
|
650
|
-
const message = result.content;
|
|
651
|
-
expect(message).toContain('Test tool');
|
|
652
|
-
expect(message).toContain('Server description');
|
|
653
|
-
expect(message).toContain('Parameters:');
|
|
654
|
-
}
|
|
655
|
-
});
|
|
656
|
-
|
|
657
|
-
it('should handle multiple arguments - "desc schema"', async () => {
|
|
658
|
-
const result = await mcpCommand.action!(mockContext, 'desc schema');
|
|
659
|
-
|
|
660
|
-
expect(isMessageAction(result)).toBe(true);
|
|
661
|
-
if (isMessageAction(result)) {
|
|
662
|
-
const message = result.content;
|
|
663
|
-
expect(message).toContain('Test tool');
|
|
664
|
-
expect(message).toContain('Server description');
|
|
665
|
-
expect(message).toContain('Parameters:');
|
|
666
|
-
}
|
|
667
|
-
});
|
|
668
|
-
|
|
669
|
-
it('should handle "schema" alone showing descriptions', async () => {
|
|
670
|
-
const result = await mcpCommand.action!(mockContext, 'schema');
|
|
671
|
-
|
|
672
|
-
expect(isMessageAction(result)).toBe(true);
|
|
673
|
-
if (isMessageAction(result)) {
|
|
674
|
-
const message = result.content;
|
|
675
|
-
expect(message).toContain('Test tool');
|
|
676
|
-
expect(message).toContain('Server description');
|
|
677
|
-
expect(message).toContain('Parameters:');
|
|
678
|
-
}
|
|
679
|
-
});
|
|
680
|
-
|
|
681
|
-
it('should handle "nodesc" overriding "schema" - "schema nodesc"', async () => {
|
|
682
|
-
const result = await mcpCommand.action!(mockContext, 'schema nodesc');
|
|
683
|
-
|
|
684
|
-
expect(isMessageAction(result)).toBe(true);
|
|
685
|
-
if (isMessageAction(result)) {
|
|
686
|
-
const message = result.content;
|
|
687
|
-
expect(message).not.toContain('Test tool');
|
|
688
|
-
expect(message).not.toContain('Server description');
|
|
689
|
-
expect(message).toContain('Parameters:'); // Schema should still show
|
|
690
|
-
expect(message).toContain('\u001b[36mtool1\u001b[0m');
|
|
691
|
-
}
|
|
692
|
-
});
|
|
693
|
-
|
|
694
|
-
it('should handle "nodesc" overriding "desc" - "desc nodesc"', async () => {
|
|
695
|
-
const result = await mcpCommand.action!(mockContext, 'desc nodesc');
|
|
696
|
-
|
|
697
|
-
expect(isMessageAction(result)).toBe(true);
|
|
698
|
-
if (isMessageAction(result)) {
|
|
699
|
-
const message = result.content;
|
|
700
|
-
expect(message).not.toContain('Test tool');
|
|
701
|
-
expect(message).not.toContain('Server description');
|
|
702
|
-
expect(message).not.toContain('Parameters:');
|
|
703
|
-
expect(message).toContain('\u001b[36mtool1\u001b[0m');
|
|
704
|
-
}
|
|
705
|
-
});
|
|
706
|
-
|
|
707
|
-
it('should handle "nodesc" overriding both "desc" and "schema" - "desc schema nodesc"', async () => {
|
|
708
|
-
const result = await mcpCommand.action!(
|
|
709
|
-
mockContext,
|
|
710
|
-
'desc schema nodesc',
|
|
711
|
-
);
|
|
712
|
-
|
|
713
|
-
expect(isMessageAction(result)).toBe(true);
|
|
714
|
-
if (isMessageAction(result)) {
|
|
715
|
-
const message = result.content;
|
|
716
|
-
expect(message).not.toContain('Test tool');
|
|
717
|
-
expect(message).not.toContain('Server description');
|
|
718
|
-
expect(message).toContain('Parameters:'); // Schema should still show
|
|
719
|
-
expect(message).toContain('\u001b[36mtool1\u001b[0m');
|
|
720
|
-
}
|
|
721
|
-
});
|
|
722
|
-
|
|
723
|
-
it('should handle extra whitespace in arguments', async () => {
|
|
724
|
-
const result = await mcpCommand.action!(mockContext, ' desc schema ');
|
|
725
|
-
|
|
726
|
-
expect(isMessageAction(result)).toBe(true);
|
|
727
|
-
if (isMessageAction(result)) {
|
|
728
|
-
const message = result.content;
|
|
729
|
-
expect(message).toContain('Test tool');
|
|
730
|
-
expect(message).toContain('Server description');
|
|
731
|
-
expect(message).toContain('Parameters:');
|
|
732
|
-
}
|
|
733
|
-
});
|
|
734
|
-
|
|
735
|
-
it('should handle empty arguments gracefully', async () => {
|
|
736
|
-
const result = await mcpCommand.action!(mockContext, '');
|
|
737
|
-
|
|
738
|
-
expect(isMessageAction(result)).toBe(true);
|
|
739
|
-
if (isMessageAction(result)) {
|
|
740
|
-
const message = result.content;
|
|
741
|
-
expect(message).not.toContain('Test tool');
|
|
742
|
-
expect(message).not.toContain('Server description');
|
|
743
|
-
expect(message).not.toContain('Parameters:');
|
|
744
|
-
expect(message).toContain('\u001b[36mtool1\u001b[0m');
|
|
745
|
-
}
|
|
746
|
-
});
|
|
747
|
-
|
|
748
|
-
it('should handle unknown arguments gracefully', async () => {
|
|
749
|
-
const result = await mcpCommand.action!(mockContext, 'unknown arg');
|
|
750
|
-
|
|
751
|
-
expect(isMessageAction(result)).toBe(true);
|
|
752
|
-
if (isMessageAction(result)) {
|
|
753
|
-
const message = result.content;
|
|
754
|
-
expect(message).not.toContain('Test tool');
|
|
755
|
-
expect(message).not.toContain('Server description');
|
|
756
|
-
expect(message).not.toContain('Parameters:');
|
|
757
|
-
expect(message).toContain('\u001b[36mtool1\u001b[0m');
|
|
758
|
-
}
|
|
759
|
-
});
|
|
760
|
-
});
|
|
761
|
-
|
|
762
|
-
describe('edge cases', () => {
|
|
763
|
-
it('should handle empty server names gracefully', async () => {
|
|
764
|
-
const mockMcpServers = {
|
|
765
|
-
'': { command: 'cmd1' }, // Empty server name
|
|
766
|
-
};
|
|
767
|
-
|
|
768
|
-
mockConfig.getMcpServers = vi.fn().mockReturnValue(mockMcpServers);
|
|
769
|
-
mockConfig.getToolRegistry = vi.fn().mockReturnValue({
|
|
770
|
-
getAllTools: vi.fn().mockReturnValue([]),
|
|
771
|
-
});
|
|
772
|
-
|
|
773
|
-
const result = await mcpCommand.action!(mockContext, '');
|
|
774
|
-
|
|
775
|
-
expect(result).toEqual({
|
|
776
|
-
type: 'message',
|
|
777
|
-
messageType: 'info',
|
|
778
|
-
content: expect.stringContaining('Configured MCP servers:'),
|
|
779
|
-
});
|
|
780
|
-
});
|
|
781
|
-
|
|
782
|
-
it('should handle servers with special characters in names', async () => {
|
|
783
|
-
const mockMcpServers = {
|
|
784
|
-
'server-with-dashes': { command: 'cmd1' },
|
|
785
|
-
server_with_underscores: { command: 'cmd2' },
|
|
786
|
-
'server.with.dots': { command: 'cmd3' },
|
|
787
|
-
};
|
|
788
|
-
|
|
789
|
-
mockConfig.getMcpServers = vi.fn().mockReturnValue(mockMcpServers);
|
|
790
|
-
mockConfig.getToolRegistry = vi.fn().mockReturnValue({
|
|
791
|
-
getAllTools: vi.fn().mockReturnValue([]),
|
|
792
|
-
});
|
|
793
|
-
|
|
794
|
-
const result = await mcpCommand.action!(mockContext, '');
|
|
795
|
-
|
|
796
|
-
expect(isMessageAction(result)).toBe(true);
|
|
797
|
-
if (isMessageAction(result)) {
|
|
798
|
-
const message = result.content;
|
|
799
|
-
expect(message).toContain('server-with-dashes');
|
|
800
|
-
expect(message).toContain('server_with_underscores');
|
|
801
|
-
expect(message).toContain('server.with.dots');
|
|
802
|
-
}
|
|
803
|
-
});
|
|
804
|
-
});
|
|
805
|
-
|
|
806
|
-
describe('auth subcommand', () => {
|
|
807
|
-
beforeEach(() => {
|
|
808
|
-
vi.clearAllMocks();
|
|
809
|
-
});
|
|
810
|
-
|
|
811
|
-
it('should list OAuth-enabled servers when no server name is provided', async () => {
|
|
812
|
-
const context = createMockCommandContext({
|
|
813
|
-
services: {
|
|
814
|
-
config: {
|
|
815
|
-
getMcpServers: vi.fn().mockReturnValue({
|
|
816
|
-
'oauth-server': { oauth: { enabled: true } },
|
|
817
|
-
'regular-server': {},
|
|
818
|
-
'another-oauth': { oauth: { enabled: true } },
|
|
819
|
-
}),
|
|
820
|
-
},
|
|
821
|
-
},
|
|
822
|
-
});
|
|
823
|
-
|
|
824
|
-
const authCommand = mcpCommand.subCommands?.find(
|
|
825
|
-
(cmd) => cmd.name === 'auth',
|
|
826
|
-
);
|
|
827
|
-
expect(authCommand).toBeDefined();
|
|
828
|
-
|
|
829
|
-
const result = await authCommand!.action!(context, '');
|
|
830
|
-
expect(isMessageAction(result)).toBe(true);
|
|
831
|
-
if (isMessageAction(result)) {
|
|
832
|
-
expect(result.messageType).toBe('info');
|
|
833
|
-
expect(result.content).toContain('oauth-server');
|
|
834
|
-
expect(result.content).toContain('another-oauth');
|
|
835
|
-
expect(result.content).not.toContain('regular-server');
|
|
836
|
-
expect(result.content).toContain('/mcp auth <server-name>');
|
|
837
|
-
}
|
|
838
|
-
});
|
|
839
|
-
|
|
840
|
-
it('should show message when no OAuth servers are configured', async () => {
|
|
841
|
-
const context = createMockCommandContext({
|
|
842
|
-
services: {
|
|
843
|
-
config: {
|
|
844
|
-
getMcpServers: vi.fn().mockReturnValue({
|
|
845
|
-
'regular-server': {},
|
|
846
|
-
}),
|
|
847
|
-
},
|
|
848
|
-
},
|
|
849
|
-
});
|
|
850
|
-
|
|
851
|
-
const authCommand = mcpCommand.subCommands?.find(
|
|
852
|
-
(cmd) => cmd.name === 'auth',
|
|
853
|
-
);
|
|
854
|
-
const result = await authCommand!.action!(context, '');
|
|
855
|
-
|
|
856
|
-
expect(isMessageAction(result)).toBe(true);
|
|
857
|
-
if (isMessageAction(result)) {
|
|
858
|
-
expect(result.messageType).toBe('info');
|
|
859
|
-
expect(result.content).toBe(
|
|
860
|
-
'No MCP servers configured with OAuth authentication.',
|
|
861
|
-
);
|
|
862
|
-
}
|
|
863
|
-
});
|
|
864
|
-
|
|
865
|
-
it('should authenticate with a specific server', async () => {
|
|
866
|
-
const mockToolRegistry = {
|
|
867
|
-
discoverToolsForServer: vi.fn(),
|
|
868
|
-
};
|
|
869
|
-
const mockGeminiClient = {
|
|
870
|
-
setTools: vi.fn(),
|
|
871
|
-
};
|
|
872
|
-
|
|
873
|
-
const context = createMockCommandContext({
|
|
874
|
-
services: {
|
|
875
|
-
config: {
|
|
876
|
-
getMcpServers: vi.fn().mockReturnValue({
|
|
877
|
-
'test-server': {
|
|
878
|
-
url: 'http://localhost:3000',
|
|
879
|
-
oauth: { enabled: true },
|
|
880
|
-
},
|
|
881
|
-
}),
|
|
882
|
-
getToolRegistry: vi.fn().mockReturnValue(mockToolRegistry),
|
|
883
|
-
getGeminiClient: vi.fn().mockReturnValue(mockGeminiClient),
|
|
884
|
-
getPromptRegistry: vi.fn().mockResolvedValue({
|
|
885
|
-
removePromptsByServer: vi.fn(),
|
|
886
|
-
}),
|
|
887
|
-
},
|
|
888
|
-
},
|
|
889
|
-
});
|
|
890
|
-
// Mock the reloadCommands function
|
|
891
|
-
context.ui.reloadCommands = vi.fn();
|
|
892
|
-
|
|
893
|
-
const { MCPOAuthProvider } = await import('fss-link-core');
|
|
894
|
-
|
|
895
|
-
const authCommand = mcpCommand.subCommands?.find(
|
|
896
|
-
(cmd) => cmd.name === 'auth',
|
|
897
|
-
);
|
|
898
|
-
const result = await authCommand!.action!(context, 'test-server');
|
|
899
|
-
|
|
900
|
-
expect(MCPOAuthProvider.authenticate).toHaveBeenCalledWith(
|
|
901
|
-
'test-server',
|
|
902
|
-
{ enabled: true },
|
|
903
|
-
'http://localhost:3000',
|
|
904
|
-
);
|
|
905
|
-
expect(mockToolRegistry.discoverToolsForServer).toHaveBeenCalledWith(
|
|
906
|
-
'test-server',
|
|
907
|
-
);
|
|
908
|
-
expect(mockGeminiClient.setTools).toHaveBeenCalled();
|
|
909
|
-
expect(context.ui.reloadCommands).toHaveBeenCalledTimes(1);
|
|
910
|
-
|
|
911
|
-
expect(isMessageAction(result)).toBe(true);
|
|
912
|
-
if (isMessageAction(result)) {
|
|
913
|
-
expect(result.messageType).toBe('info');
|
|
914
|
-
expect(result.content).toContain('Successfully authenticated');
|
|
915
|
-
}
|
|
916
|
-
});
|
|
917
|
-
|
|
918
|
-
it('should handle authentication errors', async () => {
|
|
919
|
-
const context = createMockCommandContext({
|
|
920
|
-
services: {
|
|
921
|
-
config: {
|
|
922
|
-
getMcpServers: vi.fn().mockReturnValue({
|
|
923
|
-
'test-server': { oauth: { enabled: true } },
|
|
924
|
-
}),
|
|
925
|
-
},
|
|
926
|
-
},
|
|
927
|
-
});
|
|
928
|
-
|
|
929
|
-
const { MCPOAuthProvider } = await import('fss-link-core');
|
|
930
|
-
(
|
|
931
|
-
MCPOAuthProvider.authenticate as ReturnType<typeof vi.fn>
|
|
932
|
-
).mockRejectedValue(new Error('Auth failed'));
|
|
933
|
-
|
|
934
|
-
const authCommand = mcpCommand.subCommands?.find(
|
|
935
|
-
(cmd) => cmd.name === 'auth',
|
|
936
|
-
);
|
|
937
|
-
const result = await authCommand!.action!(context, 'test-server');
|
|
938
|
-
|
|
939
|
-
expect(isMessageAction(result)).toBe(true);
|
|
940
|
-
if (isMessageAction(result)) {
|
|
941
|
-
expect(result.messageType).toBe('error');
|
|
942
|
-
expect(result.content).toContain('Failed to authenticate');
|
|
943
|
-
expect(result.content).toContain('Auth failed');
|
|
944
|
-
}
|
|
945
|
-
});
|
|
946
|
-
|
|
947
|
-
it('should handle non-existent server', async () => {
|
|
948
|
-
const context = createMockCommandContext({
|
|
949
|
-
services: {
|
|
950
|
-
config: {
|
|
951
|
-
getMcpServers: vi.fn().mockReturnValue({
|
|
952
|
-
'existing-server': {},
|
|
953
|
-
}),
|
|
954
|
-
},
|
|
955
|
-
},
|
|
956
|
-
});
|
|
957
|
-
|
|
958
|
-
const authCommand = mcpCommand.subCommands?.find(
|
|
959
|
-
(cmd) => cmd.name === 'auth',
|
|
960
|
-
);
|
|
961
|
-
const result = await authCommand!.action!(context, 'non-existent');
|
|
962
|
-
|
|
963
|
-
expect(isMessageAction(result)).toBe(true);
|
|
964
|
-
if (isMessageAction(result)) {
|
|
965
|
-
expect(result.messageType).toBe('error');
|
|
966
|
-
expect(result.content).toContain("MCP server 'non-existent' not found");
|
|
967
|
-
}
|
|
968
|
-
});
|
|
969
|
-
});
|
|
970
|
-
|
|
971
|
-
describe('refresh subcommand', () => {
|
|
972
|
-
it('should refresh the list of tools and display the status', async () => {
|
|
973
|
-
const mockToolRegistry = {
|
|
974
|
-
discoverMcpTools: vi.fn(),
|
|
975
|
-
restartMcpServers: vi.fn(),
|
|
976
|
-
getAllTools: vi.fn().mockReturnValue([]),
|
|
977
|
-
};
|
|
978
|
-
const mockGeminiClient = {
|
|
979
|
-
setTools: vi.fn(),
|
|
980
|
-
};
|
|
981
|
-
|
|
982
|
-
const context = createMockCommandContext({
|
|
983
|
-
services: {
|
|
984
|
-
config: {
|
|
985
|
-
getMcpServers: vi.fn().mockReturnValue({ server1: {} }),
|
|
986
|
-
getBlockedMcpServers: vi.fn().mockReturnValue([]),
|
|
987
|
-
getToolRegistry: vi.fn().mockReturnValue(mockToolRegistry),
|
|
988
|
-
getGeminiClient: vi.fn().mockReturnValue(mockGeminiClient),
|
|
989
|
-
getPromptRegistry: vi.fn().mockResolvedValue({
|
|
990
|
-
getPromptsByServer: vi.fn().mockReturnValue([]),
|
|
991
|
-
}),
|
|
992
|
-
},
|
|
993
|
-
},
|
|
994
|
-
});
|
|
995
|
-
// Mock the reloadCommands function, which is new logic.
|
|
996
|
-
context.ui.reloadCommands = vi.fn();
|
|
997
|
-
|
|
998
|
-
const refreshCommand = mcpCommand.subCommands?.find(
|
|
999
|
-
(cmd) => cmd.name === 'refresh',
|
|
1000
|
-
);
|
|
1001
|
-
expect(refreshCommand).toBeDefined();
|
|
1002
|
-
|
|
1003
|
-
const result = await refreshCommand!.action!(context, '');
|
|
1004
|
-
|
|
1005
|
-
expect(context.ui.addItem).toHaveBeenCalledWith(
|
|
1006
|
-
{
|
|
1007
|
-
type: 'info',
|
|
1008
|
-
text: 'Restarting MCP servers...',
|
|
1009
|
-
},
|
|
1010
|
-
expect.any(Number),
|
|
1011
|
-
);
|
|
1012
|
-
expect(mockToolRegistry.restartMcpServers).toHaveBeenCalled();
|
|
1013
|
-
expect(mockGeminiClient.setTools).toHaveBeenCalled();
|
|
1014
|
-
expect(context.ui.reloadCommands).toHaveBeenCalledTimes(1);
|
|
1015
|
-
|
|
1016
|
-
expect(isMessageAction(result)).toBe(true);
|
|
1017
|
-
if (isMessageAction(result)) {
|
|
1018
|
-
expect(result.messageType).toBe('info');
|
|
1019
|
-
expect(result.content).toContain('Configured MCP servers:');
|
|
1020
|
-
}
|
|
1021
|
-
});
|
|
1022
|
-
|
|
1023
|
-
it('should show an error if config is not available', async () => {
|
|
1024
|
-
const contextWithoutConfig = createMockCommandContext({
|
|
1025
|
-
services: {
|
|
1026
|
-
config: null,
|
|
1027
|
-
},
|
|
1028
|
-
});
|
|
1029
|
-
|
|
1030
|
-
const refreshCommand = mcpCommand.subCommands?.find(
|
|
1031
|
-
(cmd) => cmd.name === 'refresh',
|
|
1032
|
-
);
|
|
1033
|
-
const result = await refreshCommand!.action!(contextWithoutConfig, '');
|
|
1034
|
-
|
|
1035
|
-
expect(result).toEqual({
|
|
1036
|
-
type: 'message',
|
|
1037
|
-
messageType: 'error',
|
|
1038
|
-
content: 'Config not loaded.',
|
|
1039
|
-
});
|
|
1040
|
-
});
|
|
1041
|
-
|
|
1042
|
-
it('should show an error if tool registry is not available', async () => {
|
|
1043
|
-
mockConfig.getToolRegistry = vi.fn().mockReturnValue(undefined);
|
|
1044
|
-
|
|
1045
|
-
const refreshCommand = mcpCommand.subCommands?.find(
|
|
1046
|
-
(cmd) => cmd.name === 'refresh',
|
|
1047
|
-
);
|
|
1048
|
-
const result = await refreshCommand!.action!(mockContext, '');
|
|
1049
|
-
|
|
1050
|
-
expect(result).toEqual({
|
|
1051
|
-
type: 'message',
|
|
1052
|
-
messageType: 'error',
|
|
1053
|
-
content: 'Could not retrieve tool registry.',
|
|
1054
|
-
});
|
|
1055
|
-
});
|
|
1056
|
-
});
|
|
1057
|
-
});
|