2025-03-28 15:21:10 -04:00
|
|
|
import { CallToolResultSchema, ErrorCode, McpError } from '@modelcontextprotocol/sdk/types.js';
|
2025-03-20 22:56:57 -04:00
|
|
|
import type { RequestOptions } from '@modelcontextprotocol/sdk/shared/protocol.js';
|
2025-03-18 23:16:45 -04:00
|
|
|
import type { JsonSchemaType, MCPOptions } from 'librechat-data-provider';
|
2024-12-17 13:12:57 -05:00
|
|
|
import type { Logger } from 'winston';
|
|
|
|
|
import type * as t from './types/mcp';
|
|
|
|
|
import { formatToolContent } from './parsers';
|
|
|
|
|
import { MCPConnection } from './connection';
|
|
|
|
|
import { CONSTANTS } from './enum';
|
|
|
|
|
|
2025-03-28 15:21:10 -04:00
|
|
|
export interface CallToolOptions extends RequestOptions {
|
|
|
|
|
userId?: string;
|
|
|
|
|
}
|
|
|
|
|
|
2024-12-17 13:12:57 -05:00
|
|
|
export class MCPManager {
|
|
|
|
|
private static instance: MCPManager | null = null;
|
2025-03-28 15:21:10 -04:00
|
|
|
/** App-level connections initialized at startup */
|
2024-12-17 13:12:57 -05:00
|
|
|
private connections: Map<string, MCPConnection> = new Map();
|
2025-03-28 15:21:10 -04:00
|
|
|
/** User-specific connections initialized on demand */
|
|
|
|
|
private userConnections: Map<string, Map<string, MCPConnection>> = new Map();
|
|
|
|
|
/** Last activity timestamp for users (not per server) */
|
|
|
|
|
private userLastActivity: Map<string, number> = new Map();
|
|
|
|
|
private readonly USER_CONNECTION_IDLE_TIMEOUT = 15 * 60 * 1000; // 15 minutes (TODO: make configurable)
|
|
|
|
|
private mcpConfigs: t.MCPServers = {};
|
|
|
|
|
private processMCPEnv?: (obj: MCPOptions, userId?: string) => MCPOptions; // Store the processing function
|
2024-12-17 13:12:57 -05:00
|
|
|
private logger: Logger;
|
|
|
|
|
|
|
|
|
|
private static getDefaultLogger(): Logger {
|
|
|
|
|
return {
|
|
|
|
|
error: console.error,
|
|
|
|
|
warn: console.warn,
|
|
|
|
|
info: console.info,
|
|
|
|
|
debug: console.debug,
|
|
|
|
|
} as Logger;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private constructor(logger?: Logger) {
|
|
|
|
|
this.logger = logger || MCPManager.getDefaultLogger();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static getInstance(logger?: Logger): MCPManager {
|
|
|
|
|
if (!MCPManager.instance) {
|
|
|
|
|
MCPManager.instance = new MCPManager(logger);
|
|
|
|
|
}
|
2025-03-28 15:21:10 -04:00
|
|
|
// Check for idle connections when getInstance is called
|
|
|
|
|
MCPManager.instance.checkIdleConnections();
|
2024-12-17 13:12:57 -05:00
|
|
|
return MCPManager.instance;
|
|
|
|
|
}
|
|
|
|
|
|
2025-03-28 15:21:10 -04:00
|
|
|
/** Stores configs and initializes app-level connections */
|
2025-03-18 23:16:45 -04:00
|
|
|
public async initializeMCP(
|
|
|
|
|
mcpServers: t.MCPServers,
|
|
|
|
|
processMCPEnv?: (obj: MCPOptions) => MCPOptions,
|
|
|
|
|
): Promise<void> {
|
2025-03-28 15:21:10 -04:00
|
|
|
this.logger.info('[MCP] Initializing app-level servers');
|
|
|
|
|
this.processMCPEnv = processMCPEnv; // Store the function
|
|
|
|
|
this.mcpConfigs = mcpServers;
|
2024-12-17 13:12:57 -05:00
|
|
|
|
|
|
|
|
const entries = Object.entries(mcpServers);
|
|
|
|
|
const initializedServers = new Set();
|
|
|
|
|
const connectionResults = await Promise.allSettled(
|
2025-03-18 23:16:45 -04:00
|
|
|
entries.map(async ([serverName, _config], i) => {
|
2025-03-28 15:21:10 -04:00
|
|
|
/** Process env for app-level connections */
|
|
|
|
|
const config = this.processMCPEnv ? this.processMCPEnv(_config) : _config;
|
2024-12-17 13:12:57 -05:00
|
|
|
const connection = new MCPConnection(serverName, config, this.logger);
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
const connectionTimeout = new Promise<void>((_, reject) =>
|
|
|
|
|
setTimeout(() => reject(new Error('Connection timeout')), 30000),
|
|
|
|
|
);
|
|
|
|
|
|
2025-03-28 15:21:10 -04:00
|
|
|
const connectionAttempt = this.initializeServer(connection, `[MCP][${serverName}]`);
|
2024-12-17 13:12:57 -05:00
|
|
|
await Promise.race([connectionAttempt, connectionTimeout]);
|
|
|
|
|
|
|
|
|
|
if (connection.isConnected()) {
|
|
|
|
|
initializedServers.add(i);
|
2025-03-28 15:21:10 -04:00
|
|
|
this.connections.set(serverName, connection); // Store in app-level map
|
2024-12-17 13:12:57 -05:00
|
|
|
|
|
|
|
|
const serverCapabilities = connection.client.getServerCapabilities();
|
|
|
|
|
this.logger.info(
|
|
|
|
|
`[MCP][${serverName}] Capabilities: ${JSON.stringify(serverCapabilities)}`,
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
if (serverCapabilities?.tools) {
|
|
|
|
|
const tools = await connection.client.listTools();
|
|
|
|
|
if (tools.tools.length) {
|
|
|
|
|
this.logger.info(
|
|
|
|
|
`[MCP][${serverName}] Available tools: ${tools.tools
|
|
|
|
|
.map((tool) => tool.name)
|
|
|
|
|
.join(', ')}`,
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} catch (error) {
|
|
|
|
|
this.logger.error(`[MCP][${serverName}] Initialization failed`, error);
|
|
|
|
|
throw error;
|
|
|
|
|
}
|
|
|
|
|
}),
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
const failedConnections = connectionResults.filter(
|
|
|
|
|
(result): result is PromiseRejectedResult => result.status === 'rejected',
|
|
|
|
|
);
|
|
|
|
|
|
2025-03-28 15:21:10 -04:00
|
|
|
this.logger.info(
|
|
|
|
|
`[MCP] Initialized ${initializedServers.size}/${entries.length} app-level server(s)`,
|
|
|
|
|
);
|
2024-12-17 13:12:57 -05:00
|
|
|
|
|
|
|
|
if (failedConnections.length > 0) {
|
|
|
|
|
this.logger.warn(
|
2025-03-28 15:21:10 -04:00
|
|
|
`[MCP] ${failedConnections.length}/${entries.length} app-level server(s) failed to initialize`,
|
2024-12-17 13:12:57 -05:00
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
entries.forEach(([serverName], index) => {
|
|
|
|
|
if (initializedServers.has(index)) {
|
|
|
|
|
this.logger.info(`[MCP][${serverName}] ✓ Initialized`);
|
|
|
|
|
} else {
|
|
|
|
|
this.logger.info(`[MCP][${serverName}] ✗ Failed`);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
if (initializedServers.size === entries.length) {
|
2025-03-28 15:21:10 -04:00
|
|
|
this.logger.info('[MCP] All app-level servers initialized successfully');
|
2024-12-17 13:12:57 -05:00
|
|
|
} else if (initializedServers.size === 0) {
|
2025-03-28 15:21:10 -04:00
|
|
|
this.logger.warn('[MCP] No app-level servers initialized');
|
2024-12-17 13:12:57 -05:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-03-28 15:21:10 -04:00
|
|
|
/** Generic server initialization logic */
|
|
|
|
|
private async initializeServer(connection: MCPConnection, logPrefix: string): Promise<void> {
|
2024-12-17 13:12:57 -05:00
|
|
|
const maxAttempts = 3;
|
|
|
|
|
let attempts = 0;
|
|
|
|
|
|
|
|
|
|
while (attempts < maxAttempts) {
|
|
|
|
|
try {
|
|
|
|
|
await connection.connect();
|
|
|
|
|
if (connection.isConnected()) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2025-03-28 15:21:10 -04:00
|
|
|
throw new Error('Connection attempt succeeded but status is not connected');
|
2024-12-17 13:12:57 -05:00
|
|
|
} catch (error) {
|
|
|
|
|
attempts++;
|
|
|
|
|
if (attempts === maxAttempts) {
|
2025-03-28 15:21:10 -04:00
|
|
|
this.logger.error(`${logPrefix} Failed to connect after ${maxAttempts} attempts`, error);
|
|
|
|
|
throw error; // Re-throw the last error
|
2024-12-17 13:12:57 -05:00
|
|
|
}
|
|
|
|
|
await new Promise((resolve) => setTimeout(resolve, 2000 * attempts));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-03-28 15:21:10 -04:00
|
|
|
/** Check for and disconnect idle connections */
|
2025-04-12 18:46:36 -04:00
|
|
|
private checkIdleConnections(currentUserId?: string): void {
|
2025-03-28 15:21:10 -04:00
|
|
|
const now = Date.now();
|
|
|
|
|
|
|
|
|
|
// Iterate through all users to check for idle ones
|
|
|
|
|
for (const [userId, lastActivity] of this.userLastActivity.entries()) {
|
2025-04-12 18:46:36 -04:00
|
|
|
if (currentUserId && currentUserId === userId) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2025-03-28 15:21:10 -04:00
|
|
|
if (now - lastActivity > this.USER_CONNECTION_IDLE_TIMEOUT) {
|
|
|
|
|
this.logger.info(
|
|
|
|
|
`[MCP][User: ${userId}] User idle for too long. Disconnecting all connections...`,
|
|
|
|
|
);
|
|
|
|
|
// Disconnect all user connections asynchronously (fire and forget)
|
|
|
|
|
this.disconnectUserConnections(userId).catch((err) =>
|
|
|
|
|
this.logger.error(`[MCP][User: ${userId}] Error disconnecting idle connections:`, err),
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** Updates the last activity timestamp for a user */
|
|
|
|
|
private updateUserLastActivity(userId: string): void {
|
|
|
|
|
const now = Date.now();
|
|
|
|
|
this.userLastActivity.set(userId, now);
|
|
|
|
|
this.logger.debug(
|
|
|
|
|
`[MCP][User: ${userId}] Updated last activity timestamp: ${new Date(now).toISOString()}`,
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** Gets or creates a connection for a specific user */
|
|
|
|
|
public async getUserConnection(userId: string, serverName: string): Promise<MCPConnection> {
|
|
|
|
|
const userServerMap = this.userConnections.get(userId);
|
|
|
|
|
let connection = userServerMap?.get(serverName);
|
|
|
|
|
const now = Date.now();
|
|
|
|
|
|
|
|
|
|
// Check if user is idle
|
|
|
|
|
const lastActivity = this.userLastActivity.get(userId);
|
|
|
|
|
if (lastActivity && now - lastActivity > this.USER_CONNECTION_IDLE_TIMEOUT) {
|
|
|
|
|
this.logger.info(
|
|
|
|
|
`[MCP][User: ${userId}] User idle for too long. Disconnecting all connections.`,
|
|
|
|
|
);
|
|
|
|
|
// Disconnect all user connections
|
2025-04-23 18:56:06 -04:00
|
|
|
try {
|
|
|
|
|
await this.disconnectUserConnections(userId);
|
|
|
|
|
} catch (err) {
|
|
|
|
|
this.logger.error(`[MCP][User: ${userId}] Error disconnecting idle connections:`, err);
|
|
|
|
|
}
|
2025-03-28 15:21:10 -04:00
|
|
|
connection = undefined; // Force creation of a new connection
|
|
|
|
|
} else if (connection) {
|
|
|
|
|
if (connection.isConnected()) {
|
|
|
|
|
this.logger.debug(`[MCP][User: ${userId}][${serverName}] Reusing active connection`);
|
|
|
|
|
// Update timestamp on reuse
|
|
|
|
|
this.updateUserLastActivity(userId);
|
|
|
|
|
return connection;
|
|
|
|
|
} else {
|
|
|
|
|
// Connection exists but is not connected, attempt to remove potentially stale entry
|
|
|
|
|
this.logger.warn(
|
|
|
|
|
`[MCP][User: ${userId}][${serverName}] Found existing but disconnected connection object. Cleaning up.`,
|
|
|
|
|
);
|
|
|
|
|
this.removeUserConnection(userId, serverName); // Clean up maps
|
|
|
|
|
connection = undefined;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// If no valid connection exists, create a new one
|
|
|
|
|
if (!connection) {
|
|
|
|
|
this.logger.info(`[MCP][User: ${userId}][${serverName}] Establishing new connection`);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let config = this.mcpConfigs[serverName];
|
|
|
|
|
if (!config) {
|
|
|
|
|
throw new McpError(
|
|
|
|
|
ErrorCode.InvalidRequest,
|
|
|
|
|
`[MCP][User: ${userId}] Configuration for server "${serverName}" not found.`,
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (this.processMCPEnv) {
|
|
|
|
|
config = { ...(this.processMCPEnv(config, userId) ?? {}) };
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
connection = new MCPConnection(serverName, config, this.logger, userId);
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
const connectionTimeout = new Promise<void>((_, reject) =>
|
|
|
|
|
setTimeout(() => reject(new Error('Connection timeout')), 30000),
|
|
|
|
|
);
|
|
|
|
|
const connectionAttempt = this.initializeServer(
|
|
|
|
|
connection,
|
|
|
|
|
`[MCP][User: ${userId}][${serverName}]`,
|
|
|
|
|
);
|
|
|
|
|
await Promise.race([connectionAttempt, connectionTimeout]);
|
|
|
|
|
|
|
|
|
|
if (!connection.isConnected()) {
|
|
|
|
|
throw new Error('Failed to establish connection after initialization attempt.');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!this.userConnections.has(userId)) {
|
|
|
|
|
this.userConnections.set(userId, new Map());
|
|
|
|
|
}
|
|
|
|
|
this.userConnections.get(userId)?.set(serverName, connection);
|
|
|
|
|
this.logger.info(`[MCP][User: ${userId}][${serverName}] Connection successfully established`);
|
|
|
|
|
// Update timestamp on creation
|
|
|
|
|
this.updateUserLastActivity(userId);
|
|
|
|
|
return connection;
|
|
|
|
|
} catch (error) {
|
|
|
|
|
this.logger.error(
|
|
|
|
|
`[MCP][User: ${userId}][${serverName}] Failed to establish connection`,
|
|
|
|
|
error,
|
|
|
|
|
);
|
|
|
|
|
// Ensure partial connection state is cleaned up if initialization fails
|
|
|
|
|
await connection.disconnect().catch((disconnectError) => {
|
|
|
|
|
this.logger.error(
|
|
|
|
|
`[MCP][User: ${userId}][${serverName}] Error during cleanup after failed connection`,
|
|
|
|
|
disconnectError,
|
|
|
|
|
);
|
|
|
|
|
});
|
|
|
|
|
// Ensure cleanup even if connection attempt fails
|
|
|
|
|
this.removeUserConnection(userId, serverName);
|
|
|
|
|
throw error; // Re-throw the error to the caller
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** Removes a specific user connection entry */
|
|
|
|
|
private removeUserConnection(userId: string, serverName: string): void {
|
|
|
|
|
// Remove connection object
|
|
|
|
|
const userMap = this.userConnections.get(userId);
|
|
|
|
|
if (userMap) {
|
|
|
|
|
userMap.delete(serverName);
|
|
|
|
|
if (userMap.size === 0) {
|
|
|
|
|
this.userConnections.delete(userId);
|
|
|
|
|
// Only remove user activity timestamp if all connections are gone
|
|
|
|
|
this.userLastActivity.delete(userId);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this.logger.debug(`[MCP][User: ${userId}][${serverName}] Removed connection entry.`);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** Disconnects and removes a specific user connection */
|
|
|
|
|
public async disconnectUserConnection(userId: string, serverName: string): Promise<void> {
|
|
|
|
|
const userMap = this.userConnections.get(userId);
|
|
|
|
|
const connection = userMap?.get(serverName);
|
|
|
|
|
if (connection) {
|
|
|
|
|
this.logger.info(`[MCP][User: ${userId}][${serverName}] Disconnecting...`);
|
|
|
|
|
await connection.disconnect();
|
|
|
|
|
this.removeUserConnection(userId, serverName);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** Disconnects and removes all connections for a specific user */
|
|
|
|
|
public async disconnectUserConnections(userId: string): Promise<void> {
|
|
|
|
|
const userMap = this.userConnections.get(userId);
|
2025-04-23 18:56:06 -04:00
|
|
|
const disconnectPromises: Promise<void>[] = [];
|
2025-03-28 15:21:10 -04:00
|
|
|
if (userMap) {
|
|
|
|
|
this.logger.info(`[MCP][User: ${userId}] Disconnecting all servers...`);
|
2025-04-23 18:56:06 -04:00
|
|
|
const userServers = Array.from(userMap.keys());
|
|
|
|
|
for (const serverName of userServers) {
|
|
|
|
|
disconnectPromises.push(
|
|
|
|
|
this.disconnectUserConnection(userId, serverName).catch((error) => {
|
|
|
|
|
this.logger.error(
|
|
|
|
|
`[MCP][User: ${userId}][${serverName}] Error during disconnection:`,
|
|
|
|
|
error,
|
|
|
|
|
);
|
|
|
|
|
}),
|
|
|
|
|
);
|
|
|
|
|
}
|
2025-03-28 15:21:10 -04:00
|
|
|
await Promise.allSettled(disconnectPromises);
|
|
|
|
|
// Ensure user activity timestamp is removed
|
|
|
|
|
this.userLastActivity.delete(userId);
|
|
|
|
|
this.logger.info(`[MCP][User: ${userId}] All connections processed for disconnection.`);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** Returns the app-level connection (used for mapping tools, etc.) */
|
2024-12-17 13:12:57 -05:00
|
|
|
public getConnection(serverName: string): MCPConnection | undefined {
|
|
|
|
|
return this.connections.get(serverName);
|
|
|
|
|
}
|
|
|
|
|
|
2025-03-28 15:21:10 -04:00
|
|
|
/** Returns all app-level connections */
|
2024-12-17 13:12:57 -05:00
|
|
|
public getAllConnections(): Map<string, MCPConnection> {
|
|
|
|
|
return this.connections;
|
|
|
|
|
}
|
|
|
|
|
|
2025-03-28 15:21:10 -04:00
|
|
|
/**
|
|
|
|
|
* Maps available tools from all app-level connections into the provided object.
|
|
|
|
|
* The object is modified in place.
|
|
|
|
|
*/
|
2024-12-17 13:12:57 -05:00
|
|
|
public async mapAvailableTools(availableTools: t.LCAvailableTools): Promise<void> {
|
|
|
|
|
for (const [serverName, connection] of this.connections.entries()) {
|
|
|
|
|
try {
|
|
|
|
|
if (connection.isConnected() !== true) {
|
2025-03-28 15:21:10 -04:00
|
|
|
this.logger.warn(
|
|
|
|
|
`[MCP][${serverName}] Connection not established. Skipping tool mapping.`,
|
|
|
|
|
);
|
2024-12-17 13:12:57 -05:00
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const tools = await connection.fetchTools();
|
|
|
|
|
for (const tool of tools) {
|
|
|
|
|
const name = `${tool.name}${CONSTANTS.mcp_delimiter}${serverName}`;
|
|
|
|
|
availableTools[name] = {
|
|
|
|
|
type: 'function',
|
|
|
|
|
['function']: {
|
|
|
|
|
name,
|
|
|
|
|
description: tool.description,
|
|
|
|
|
parameters: tool.inputSchema as JsonSchemaType,
|
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
} catch (error) {
|
2025-03-28 15:21:10 -04:00
|
|
|
this.logger.warn(`[MCP][${serverName}] Error fetching tools for mapping:`, error);
|
2024-12-17 13:12:57 -05:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-03-28 15:21:10 -04:00
|
|
|
/**
|
|
|
|
|
* Loads tools from all app-level connections into the manifest.
|
|
|
|
|
*/
|
2025-05-08 12:12:36 -04:00
|
|
|
public async loadManifestTools(manifestTools: t.LCToolManifest): Promise<t.LCToolManifest> {
|
|
|
|
|
const mcpTools: t.LCManifestTool[] = [];
|
|
|
|
|
|
2024-12-17 13:12:57 -05:00
|
|
|
for (const [serverName, connection] of this.connections.entries()) {
|
|
|
|
|
try {
|
|
|
|
|
if (connection.isConnected() !== true) {
|
2025-03-28 15:21:10 -04:00
|
|
|
this.logger.warn(
|
|
|
|
|
`[MCP][${serverName}] Connection not established. Skipping manifest loading.`,
|
|
|
|
|
);
|
2024-12-17 13:12:57 -05:00
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const tools = await connection.fetchTools();
|
|
|
|
|
for (const tool of tools) {
|
|
|
|
|
const pluginKey = `${tool.name}${CONSTANTS.mcp_delimiter}${serverName}`;
|
2025-05-08 12:12:36 -04:00
|
|
|
const manifestTool: t.LCManifestTool = {
|
2024-12-17 13:12:57 -05:00
|
|
|
name: tool.name,
|
|
|
|
|
pluginKey,
|
|
|
|
|
description: tool.description ?? '',
|
|
|
|
|
icon: connection.iconPath,
|
2025-05-08 12:12:36 -04:00
|
|
|
};
|
|
|
|
|
const config = this.mcpConfigs[serverName];
|
|
|
|
|
if (config?.chatMenu === false) {
|
|
|
|
|
manifestTool.chatMenu = false;
|
|
|
|
|
}
|
|
|
|
|
mcpTools.push(manifestTool);
|
2024-12-17 13:12:57 -05:00
|
|
|
}
|
|
|
|
|
} catch (error) {
|
2025-03-28 15:21:10 -04:00
|
|
|
this.logger.error(`[MCP][${serverName}] Error fetching tools for manifest:`, error);
|
2024-12-17 13:12:57 -05:00
|
|
|
}
|
|
|
|
|
}
|
2025-05-08 12:12:36 -04:00
|
|
|
|
|
|
|
|
return [...mcpTools, ...manifestTools];
|
2024-12-17 13:12:57 -05:00
|
|
|
}
|
|
|
|
|
|
2025-03-28 15:21:10 -04:00
|
|
|
/**
|
|
|
|
|
* Calls a tool on an MCP server, using either a user-specific connection
|
|
|
|
|
* (if userId is provided) or an app-level connection. Updates the last activity timestamp
|
|
|
|
|
* for user-specific connections upon successful call initiation.
|
|
|
|
|
*/
|
2025-03-20 22:56:57 -04:00
|
|
|
async callTool({
|
|
|
|
|
serverName,
|
|
|
|
|
toolName,
|
|
|
|
|
provider,
|
|
|
|
|
toolArguments,
|
|
|
|
|
options,
|
|
|
|
|
}: {
|
|
|
|
|
serverName: string;
|
|
|
|
|
toolName: string;
|
|
|
|
|
provider: t.Provider;
|
|
|
|
|
toolArguments?: Record<string, unknown>;
|
2025-03-28 15:21:10 -04:00
|
|
|
options?: CallToolOptions;
|
2025-03-20 22:56:57 -04:00
|
|
|
}): Promise<t.FormattedToolResponse> {
|
2025-03-28 15:21:10 -04:00
|
|
|
let connection: MCPConnection | undefined;
|
|
|
|
|
const { userId, ...callOptions } = options ?? {};
|
|
|
|
|
const logPrefix = userId ? `[MCP][User: ${userId}][${serverName}]` : `[MCP][${serverName}]`;
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
if (userId) {
|
|
|
|
|
this.updateUserLastActivity(userId);
|
|
|
|
|
// Get or create user-specific connection
|
|
|
|
|
connection = await this.getUserConnection(userId, serverName);
|
|
|
|
|
} else {
|
|
|
|
|
// Use app-level connection
|
|
|
|
|
connection = this.connections.get(serverName);
|
|
|
|
|
if (!connection) {
|
|
|
|
|
throw new McpError(
|
|
|
|
|
ErrorCode.InvalidRequest,
|
|
|
|
|
`${logPrefix} No app-level connection found. Cannot execute tool ${toolName}.`,
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!connection.isConnected()) {
|
|
|
|
|
// This might happen if getUserConnection failed silently or app connection dropped
|
|
|
|
|
throw new McpError(
|
|
|
|
|
ErrorCode.InternalError, // Use InternalError for connection issues
|
|
|
|
|
`${logPrefix} Connection is not active. Cannot execute tool ${toolName}.`,
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const result = await connection.client.request(
|
|
|
|
|
{
|
|
|
|
|
method: 'tools/call',
|
|
|
|
|
params: {
|
|
|
|
|
name: toolName,
|
|
|
|
|
arguments: toolArguments,
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
CallToolResultSchema,
|
|
|
|
|
{
|
|
|
|
|
timeout: connection.timeout,
|
|
|
|
|
...callOptions,
|
|
|
|
|
},
|
2024-12-17 13:12:57 -05:00
|
|
|
);
|
2025-03-28 15:21:10 -04:00
|
|
|
if (userId) {
|
|
|
|
|
this.updateUserLastActivity(userId);
|
|
|
|
|
}
|
|
|
|
|
this.checkIdleConnections();
|
|
|
|
|
return formatToolContent(result, provider);
|
|
|
|
|
} catch (error) {
|
|
|
|
|
// Log with context and re-throw or handle as needed
|
|
|
|
|
this.logger.error(`${logPrefix}[${toolName}] Tool call failed`, error);
|
|
|
|
|
// Rethrowing allows the caller (createMCPTool) to handle the final user message
|
|
|
|
|
throw error;
|
2024-12-17 13:12:57 -05:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-03-28 15:21:10 -04:00
|
|
|
/** Disconnects a specific app-level server */
|
2024-12-17 13:12:57 -05:00
|
|
|
public async disconnectServer(serverName: string): Promise<void> {
|
|
|
|
|
const connection = this.connections.get(serverName);
|
|
|
|
|
if (connection) {
|
2025-03-28 15:21:10 -04:00
|
|
|
this.logger.info(`[MCP][${serverName}] Disconnecting...`);
|
2024-12-17 13:12:57 -05:00
|
|
|
await connection.disconnect();
|
|
|
|
|
this.connections.delete(serverName);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-03-28 15:21:10 -04:00
|
|
|
/** Disconnects all app-level and user-level connections */
|
2024-12-17 13:12:57 -05:00
|
|
|
public async disconnectAll(): Promise<void> {
|
2025-03-28 15:21:10 -04:00
|
|
|
this.logger.info('[MCP] Disconnecting all app-level and user-level connections...');
|
|
|
|
|
|
|
|
|
|
const userDisconnectPromises = Array.from(this.userConnections.keys()).map((userId) =>
|
|
|
|
|
this.disconnectUserConnections(userId),
|
2024-12-17 13:12:57 -05:00
|
|
|
);
|
2025-03-28 15:21:10 -04:00
|
|
|
await Promise.allSettled(userDisconnectPromises);
|
|
|
|
|
this.userLastActivity.clear();
|
|
|
|
|
|
|
|
|
|
// Disconnect all app-level connections
|
|
|
|
|
const appDisconnectPromises = Array.from(this.connections.values()).map((connection) =>
|
|
|
|
|
connection.disconnect().catch((error) => {
|
|
|
|
|
this.logger.error(`[MCP][${connection.serverName}] Error during disconnectAll:`, error);
|
|
|
|
|
}),
|
|
|
|
|
);
|
|
|
|
|
await Promise.allSettled(appDisconnectPromises);
|
2024-12-17 13:12:57 -05:00
|
|
|
this.connections.clear();
|
2025-03-28 15:21:10 -04:00
|
|
|
|
|
|
|
|
this.logger.info('[MCP] All connections processed for disconnection.');
|
2024-12-17 13:12:57 -05:00
|
|
|
}
|
|
|
|
|
|
2025-03-28 15:21:10 -04:00
|
|
|
/** Destroys the singleton instance and disconnects all connections */
|
2024-12-17 13:12:57 -05:00
|
|
|
public static async destroyInstance(): Promise<void> {
|
|
|
|
|
if (MCPManager.instance) {
|
|
|
|
|
await MCPManager.instance.disconnectAll();
|
|
|
|
|
MCPManager.instance = null;
|
2025-03-28 15:21:10 -04:00
|
|
|
const logger = MCPManager.getDefaultLogger();
|
|
|
|
|
logger.info('[MCP] Manager instance destroyed.');
|
2024-12-17 13:12:57 -05:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|