2025-08-19 21:03:19 +02:00
|
|
|
/**
|
|
|
|
|
* @license
|
|
|
|
|
* Copyright 2025 Google LLC
|
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
*/
|
|
|
|
|
|
2025-11-04 07:51:18 -08:00
|
|
|
import {
|
|
|
|
|
afterEach,
|
|
|
|
|
beforeEach,
|
|
|
|
|
describe,
|
|
|
|
|
expect,
|
|
|
|
|
it,
|
|
|
|
|
vi,
|
|
|
|
|
type MockedObject,
|
|
|
|
|
} from 'vitest';
|
2025-08-19 21:03:19 +02:00
|
|
|
import { McpClientManager } from './mcp-client-manager.js';
|
|
|
|
|
import { McpClient } from './mcp-client.js';
|
2025-08-26 00:04:53 +02:00
|
|
|
import type { ToolRegistry } from './tool-registry.js';
|
2025-08-28 15:46:27 -07:00
|
|
|
import type { Config } from '../config/config.js';
|
2025-08-19 21:03:19 +02:00
|
|
|
|
|
|
|
|
vi.mock('./mcp-client.js', async () => {
|
|
|
|
|
const originalModule = await vi.importActual('./mcp-client.js');
|
|
|
|
|
return {
|
|
|
|
|
...originalModule,
|
|
|
|
|
McpClient: vi.fn(),
|
|
|
|
|
};
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
describe('McpClientManager', () => {
|
2025-11-04 07:51:18 -08:00
|
|
|
let mockedMcpClient: MockedObject<McpClient>;
|
|
|
|
|
let mockConfig: MockedObject<Config>;
|
2025-08-19 21:03:19 +02:00
|
|
|
|
2025-11-04 07:51:18 -08:00
|
|
|
beforeEach(() => {
|
|
|
|
|
mockedMcpClient = vi.mockObject({
|
2025-08-19 21:03:19 +02:00
|
|
|
connect: vi.fn(),
|
|
|
|
|
discover: vi.fn(),
|
|
|
|
|
disconnect: vi.fn(),
|
|
|
|
|
getStatus: vi.fn(),
|
2025-11-04 07:51:18 -08:00
|
|
|
getServerConfig: vi.fn(),
|
|
|
|
|
} as unknown as McpClient);
|
|
|
|
|
vi.mocked(McpClient).mockReturnValue(mockedMcpClient);
|
|
|
|
|
mockConfig = vi.mockObject({
|
|
|
|
|
isTrustedFolder: vi.fn().mockReturnValue(true),
|
|
|
|
|
getMcpServers: vi.fn().mockReturnValue({}),
|
|
|
|
|
getPromptRegistry: () => {},
|
|
|
|
|
getDebugMode: () => false,
|
|
|
|
|
getWorkspaceContext: () => {},
|
|
|
|
|
getAllowedMcpServers: vi.fn().mockReturnValue([]),
|
|
|
|
|
getBlockedMcpServers: vi.fn().mockReturnValue([]),
|
|
|
|
|
getMcpServerCommand: vi.fn().mockReturnValue(''),
|
|
|
|
|
getGeminiClient: vi.fn().mockReturnValue({
|
|
|
|
|
isInitialized: vi.fn(),
|
|
|
|
|
}),
|
|
|
|
|
} as unknown as Config);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
afterEach(() => {
|
|
|
|
|
vi.restoreAllMocks();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('should discover tools from all configured', async () => {
|
|
|
|
|
mockConfig.getMcpServers.mockReturnValue({
|
|
|
|
|
'test-server': {},
|
|
|
|
|
});
|
|
|
|
|
const manager = new McpClientManager({} as ToolRegistry, mockConfig);
|
|
|
|
|
await manager.startConfiguredMcpServers();
|
2025-08-19 21:03:19 +02:00
|
|
|
expect(mockedMcpClient.connect).toHaveBeenCalledOnce();
|
|
|
|
|
expect(mockedMcpClient.discover).toHaveBeenCalledOnce();
|
|
|
|
|
});
|
2025-08-28 15:46:27 -07:00
|
|
|
|
|
|
|
|
it('should not discover tools if folder is not trusted', async () => {
|
2025-11-04 07:51:18 -08:00
|
|
|
mockConfig.getMcpServers.mockReturnValue({
|
|
|
|
|
'test-server': {},
|
|
|
|
|
});
|
|
|
|
|
mockConfig.isTrustedFolder.mockReturnValue(false);
|
|
|
|
|
const manager = new McpClientManager({} as ToolRegistry, mockConfig);
|
|
|
|
|
await manager.startConfiguredMcpServers();
|
|
|
|
|
expect(mockedMcpClient.connect).not.toHaveBeenCalled();
|
|
|
|
|
expect(mockedMcpClient.discover).not.toHaveBeenCalled();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('should not start blocked servers', async () => {
|
|
|
|
|
mockConfig.getMcpServers.mockReturnValue({
|
|
|
|
|
'test-server': {},
|
|
|
|
|
});
|
|
|
|
|
mockConfig.getBlockedMcpServers.mockReturnValue(['test-server']);
|
|
|
|
|
const manager = new McpClientManager({} as ToolRegistry, mockConfig);
|
|
|
|
|
await manager.startConfiguredMcpServers();
|
2025-08-28 15:46:27 -07:00
|
|
|
expect(mockedMcpClient.connect).not.toHaveBeenCalled();
|
|
|
|
|
expect(mockedMcpClient.discover).not.toHaveBeenCalled();
|
|
|
|
|
});
|
2025-11-04 07:51:18 -08:00
|
|
|
|
|
|
|
|
it('should only start allowed servers if allow list is not empty', async () => {
|
|
|
|
|
mockConfig.getMcpServers.mockReturnValue({
|
|
|
|
|
'test-server': {},
|
|
|
|
|
'another-server': {},
|
|
|
|
|
});
|
|
|
|
|
mockConfig.getAllowedMcpServers.mockReturnValue(['another-server']);
|
|
|
|
|
const manager = new McpClientManager({} as ToolRegistry, mockConfig);
|
|
|
|
|
await manager.startConfiguredMcpServers();
|
|
|
|
|
expect(mockedMcpClient.connect).toHaveBeenCalledOnce();
|
|
|
|
|
expect(mockedMcpClient.discover).toHaveBeenCalledOnce();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('should start servers from extensions', async () => {
|
|
|
|
|
const manager = new McpClientManager({} as ToolRegistry, mockConfig);
|
|
|
|
|
await manager.startExtension({
|
|
|
|
|
name: 'test-extension',
|
|
|
|
|
mcpServers: {
|
|
|
|
|
'test-server': {},
|
|
|
|
|
},
|
|
|
|
|
isActive: true,
|
|
|
|
|
version: '1.0.0',
|
|
|
|
|
path: '/some-path',
|
|
|
|
|
contextFiles: [],
|
|
|
|
|
id: '123',
|
|
|
|
|
});
|
|
|
|
|
expect(mockedMcpClient.connect).toHaveBeenCalledOnce();
|
|
|
|
|
expect(mockedMcpClient.discover).toHaveBeenCalledOnce();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('should not start servers from disabled extensions', async () => {
|
|
|
|
|
const manager = new McpClientManager({} as ToolRegistry, mockConfig);
|
|
|
|
|
await manager.startExtension({
|
|
|
|
|
name: 'test-extension',
|
|
|
|
|
mcpServers: {
|
|
|
|
|
'test-server': {},
|
|
|
|
|
},
|
|
|
|
|
isActive: false,
|
|
|
|
|
version: '1.0.0',
|
|
|
|
|
path: '/some-path',
|
|
|
|
|
contextFiles: [],
|
|
|
|
|
id: '123',
|
|
|
|
|
});
|
|
|
|
|
expect(mockedMcpClient.connect).not.toHaveBeenCalled();
|
|
|
|
|
expect(mockedMcpClient.discover).not.toHaveBeenCalled();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('should add blocked servers to the blockedMcpServers list', async () => {
|
|
|
|
|
mockConfig.getMcpServers.mockReturnValue({
|
|
|
|
|
'test-server': {},
|
|
|
|
|
});
|
|
|
|
|
mockConfig.getBlockedMcpServers.mockReturnValue(['test-server']);
|
|
|
|
|
const manager = new McpClientManager({} as ToolRegistry, mockConfig);
|
|
|
|
|
await manager.startConfiguredMcpServers();
|
|
|
|
|
expect(manager.getBlockedMcpServers()).toEqual([
|
|
|
|
|
{ name: 'test-server', extensionName: '' },
|
|
|
|
|
]);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
describe('restart', () => {
|
|
|
|
|
it('should restart all running servers', async () => {
|
|
|
|
|
mockConfig.getMcpServers.mockReturnValue({
|
|
|
|
|
'test-server': {},
|
|
|
|
|
});
|
|
|
|
|
mockedMcpClient.getServerConfig.mockReturnValue({});
|
|
|
|
|
const manager = new McpClientManager({} as ToolRegistry, mockConfig);
|
|
|
|
|
await manager.startConfiguredMcpServers();
|
|
|
|
|
|
|
|
|
|
expect(mockedMcpClient.connect).toHaveBeenCalledTimes(1);
|
|
|
|
|
expect(mockedMcpClient.discover).toHaveBeenCalledTimes(1);
|
|
|
|
|
await manager.restart();
|
|
|
|
|
|
|
|
|
|
expect(mockedMcpClient.disconnect).toHaveBeenCalledTimes(1);
|
|
|
|
|
expect(mockedMcpClient.connect).toHaveBeenCalledTimes(2);
|
|
|
|
|
expect(mockedMcpClient.discover).toHaveBeenCalledTimes(2);
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
describe('restartServer', () => {
|
|
|
|
|
it('should restart the specified server', async () => {
|
|
|
|
|
mockConfig.getMcpServers.mockReturnValue({
|
|
|
|
|
'test-server': {},
|
|
|
|
|
});
|
|
|
|
|
mockedMcpClient.getServerConfig.mockReturnValue({});
|
|
|
|
|
const manager = new McpClientManager({} as ToolRegistry, mockConfig);
|
|
|
|
|
await manager.startConfiguredMcpServers();
|
|
|
|
|
|
|
|
|
|
expect(mockedMcpClient.connect).toHaveBeenCalledTimes(1);
|
|
|
|
|
expect(mockedMcpClient.discover).toHaveBeenCalledTimes(1);
|
|
|
|
|
|
|
|
|
|
await manager.restartServer('test-server');
|
|
|
|
|
|
|
|
|
|
expect(mockedMcpClient.disconnect).toHaveBeenCalledTimes(1);
|
|
|
|
|
expect(mockedMcpClient.connect).toHaveBeenCalledTimes(2);
|
|
|
|
|
expect(mockedMcpClient.discover).toHaveBeenCalledTimes(2);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('should throw an error if the server does not exist', async () => {
|
|
|
|
|
const manager = new McpClientManager({} as ToolRegistry, mockConfig);
|
|
|
|
|
await expect(manager.restartServer('non-existent')).rejects.toThrow(
|
|
|
|
|
'No MCP server registered with the name "non-existent"',
|
|
|
|
|
);
|
|
|
|
|
});
|
|
|
|
|
});
|
2025-08-19 21:03:19 +02:00
|
|
|
});
|