mirror of
https://github.com/QwenLM/qwen-code.git
synced 2026-05-02 21:50:52 +00:00
rebase and resolve conflict
This commit is contained in:
parent
bd900d3668
commit
781bd7880b
4 changed files with 284 additions and 24 deletions
|
|
@ -9,15 +9,16 @@ import { McpClientManager } from './mcp-client-manager.js';
|
|||
import { McpClient } from './mcp-client.js';
|
||||
import type { ToolRegistry } from './tool-registry.js';
|
||||
import type { Config } from '../config/config.js';
|
||||
import type { PromptRegistry } from '../prompts/prompt-registry.js';
|
||||
import type { WorkspaceContext } from '../utils/workspaceContext.js';
|
||||
|
||||
vi.mock('./mcp-client.js', async () => {
|
||||
const originalModule = await vi.importActual('./mcp-client.js');
|
||||
return {
|
||||
...originalModule,
|
||||
McpClient: vi.fn(),
|
||||
populateMcpServerCommand: vi.fn(() => ({
|
||||
'test-server': {},
|
||||
})),
|
||||
// Return the input servers unchanged (identity function)
|
||||
populateMcpServerCommand: vi.fn((servers) => servers),
|
||||
};
|
||||
});
|
||||
|
||||
|
|
@ -73,4 +74,178 @@ describe('McpClientManager', () => {
|
|||
expect(mockedMcpClient.connect).not.toHaveBeenCalled();
|
||||
expect(mockedMcpClient.discover).not.toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it('should disconnect all clients when stop is called', async () => {
|
||||
// Track disconnect calls across all instances
|
||||
const disconnectCalls: string[] = [];
|
||||
vi.mocked(McpClient).mockImplementation(
|
||||
(name: string) =>
|
||||
({
|
||||
connect: vi.fn(),
|
||||
discover: vi.fn(),
|
||||
disconnect: vi.fn().mockImplementation(() => {
|
||||
disconnectCalls.push(name);
|
||||
return Promise.resolve();
|
||||
}),
|
||||
getStatus: vi.fn(),
|
||||
}) as unknown as McpClient,
|
||||
);
|
||||
const mockConfig = {
|
||||
isTrustedFolder: () => true,
|
||||
getMcpServers: () => ({ 'test-server': {}, 'another-server': {} }),
|
||||
getMcpServerCommand: () => undefined,
|
||||
getPromptRegistry: () => ({}) as PromptRegistry,
|
||||
getWorkspaceContext: () => ({}) as WorkspaceContext,
|
||||
getDebugMode: () => false,
|
||||
} as unknown as Config;
|
||||
const manager = new McpClientManager(mockConfig, {} as ToolRegistry);
|
||||
// First connect to create the clients
|
||||
await manager.discoverAllMcpTools({
|
||||
isTrustedFolder: () => true,
|
||||
} as unknown as Config);
|
||||
|
||||
// Clear the disconnect calls from initial stop() in discoverAllMcpTools
|
||||
disconnectCalls.length = 0;
|
||||
|
||||
// Then stop
|
||||
await manager.stop();
|
||||
expect(disconnectCalls).toHaveLength(2);
|
||||
expect(disconnectCalls).toContain('test-server');
|
||||
expect(disconnectCalls).toContain('another-server');
|
||||
});
|
||||
|
||||
it('should be idempotent - stop can be called multiple times safely', async () => {
|
||||
const mockedMcpClient = {
|
||||
connect: vi.fn(),
|
||||
discover: vi.fn(),
|
||||
disconnect: vi.fn().mockResolvedValue(undefined),
|
||||
getStatus: vi.fn(),
|
||||
};
|
||||
vi.mocked(McpClient).mockReturnValue(
|
||||
mockedMcpClient as unknown as McpClient,
|
||||
);
|
||||
const mockConfig = {
|
||||
isTrustedFolder: () => true,
|
||||
getMcpServers: () => ({ 'test-server': {} }),
|
||||
getMcpServerCommand: () => undefined,
|
||||
getPromptRegistry: () => ({}) as PromptRegistry,
|
||||
getWorkspaceContext: () => ({}) as WorkspaceContext,
|
||||
getDebugMode: () => false,
|
||||
} as unknown as Config;
|
||||
const manager = new McpClientManager(mockConfig, {} as ToolRegistry);
|
||||
await manager.discoverAllMcpTools({
|
||||
isTrustedFolder: () => true,
|
||||
} as unknown as Config);
|
||||
|
||||
// Call stop multiple times - should not throw
|
||||
await manager.stop();
|
||||
await manager.stop();
|
||||
await manager.stop();
|
||||
});
|
||||
|
||||
it('should discover tools for a single server and track the client for stop', async () => {
|
||||
const mockedMcpClient = {
|
||||
connect: vi.fn(),
|
||||
discover: vi.fn(),
|
||||
disconnect: vi.fn().mockResolvedValue(undefined),
|
||||
getStatus: vi.fn(),
|
||||
};
|
||||
vi.mocked(McpClient).mockReturnValue(
|
||||
mockedMcpClient as unknown as McpClient,
|
||||
);
|
||||
|
||||
const mockConfig = {
|
||||
isTrustedFolder: () => true,
|
||||
getMcpServers: () => ({ 'test-server': {} }),
|
||||
getMcpServerCommand: () => undefined,
|
||||
getPromptRegistry: () => ({}) as PromptRegistry,
|
||||
getWorkspaceContext: () => ({}) as WorkspaceContext,
|
||||
getDebugMode: () => false,
|
||||
} as unknown as Config;
|
||||
const manager = new McpClientManager(mockConfig, {} as ToolRegistry);
|
||||
|
||||
await manager.discoverMcpToolsForServer(
|
||||
'test-server',
|
||||
{} as unknown as Config,
|
||||
);
|
||||
|
||||
expect(mockedMcpClient.connect).toHaveBeenCalledOnce();
|
||||
expect(mockedMcpClient.discover).toHaveBeenCalledOnce();
|
||||
|
||||
await manager.stop();
|
||||
expect(mockedMcpClient.disconnect).toHaveBeenCalledOnce();
|
||||
});
|
||||
|
||||
it('should replace an existing client when re-discovering a server', async () => {
|
||||
const firstClient = {
|
||||
connect: vi.fn(),
|
||||
discover: vi.fn(),
|
||||
disconnect: vi.fn().mockResolvedValue(undefined),
|
||||
getStatus: vi.fn(),
|
||||
};
|
||||
const secondClient = {
|
||||
connect: vi.fn(),
|
||||
discover: vi.fn(),
|
||||
disconnect: vi.fn().mockResolvedValue(undefined),
|
||||
getStatus: vi.fn(),
|
||||
};
|
||||
|
||||
vi.mocked(McpClient)
|
||||
.mockReturnValueOnce(firstClient as unknown as McpClient)
|
||||
.mockReturnValueOnce(secondClient as unknown as McpClient);
|
||||
|
||||
const mockConfig = {
|
||||
isTrustedFolder: () => true,
|
||||
getMcpServers: () => ({ 'test-server': {} }),
|
||||
getMcpServerCommand: () => undefined,
|
||||
getPromptRegistry: () => ({}) as PromptRegistry,
|
||||
getWorkspaceContext: () => ({}) as WorkspaceContext,
|
||||
getDebugMode: () => false,
|
||||
} as unknown as Config;
|
||||
const manager = new McpClientManager(mockConfig, {} as ToolRegistry);
|
||||
|
||||
await manager.discoverMcpToolsForServer(
|
||||
'test-server',
|
||||
{} as unknown as Config,
|
||||
);
|
||||
await manager.discoverMcpToolsForServer(
|
||||
'test-server',
|
||||
{} as unknown as Config,
|
||||
);
|
||||
|
||||
expect(firstClient.disconnect).toHaveBeenCalledOnce();
|
||||
expect(secondClient.connect).toHaveBeenCalledOnce();
|
||||
expect(secondClient.discover).toHaveBeenCalledOnce();
|
||||
|
||||
await manager.stop();
|
||||
expect(secondClient.disconnect).toHaveBeenCalledOnce();
|
||||
});
|
||||
|
||||
it('should no-op when discovering an unknown server', async () => {
|
||||
const mockedMcpClient = {
|
||||
connect: vi.fn(),
|
||||
discover: vi.fn(),
|
||||
disconnect: vi.fn().mockResolvedValue(undefined),
|
||||
getStatus: vi.fn(),
|
||||
};
|
||||
vi.mocked(McpClient).mockReturnValue(
|
||||
mockedMcpClient as unknown as McpClient,
|
||||
);
|
||||
|
||||
const mockConfig = {
|
||||
isTrustedFolder: () => true,
|
||||
getMcpServers: () => ({}),
|
||||
getMcpServerCommand: () => undefined,
|
||||
getPromptRegistry: () => ({}) as PromptRegistry,
|
||||
getWorkspaceContext: () => ({}) as WorkspaceContext,
|
||||
getDebugMode: () => false,
|
||||
} as unknown as Config;
|
||||
const manager = new McpClientManager(mockConfig, {} as ToolRegistry);
|
||||
|
||||
await manager.discoverMcpToolsForServer('unknown-server', {
|
||||
isTrustedFolder: () => true,
|
||||
} as unknown as Config);
|
||||
|
||||
expect(vi.mocked(McpClient)).not.toHaveBeenCalled();
|
||||
});
|
||||
});
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue