xiaozhi-client 1.5.0-beta.6 → 1.5.0
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/adaptiveMCPPipe.js +0 -0
- package/dist/adaptiveMCPPipe.js.map +1 -1
- package/dist/autoCompletion.js.map +1 -1
- package/dist/cli.js +13 -13
- package/dist/cli.js.map +1 -1
- package/dist/configManager.js.map +1 -1
- package/dist/mcpCommands.js +1 -1
- package/dist/mcpCommands.js.map +1 -1
- package/dist/mcpServerProxy.js +7 -7
- package/dist/mcpServerProxy.js.map +1 -1
- package/dist/modelScopeMCPClient.d.ts +8 -0
- package/dist/modelScopeMCPClient.js +2 -2
- package/dist/modelScopeMCPClient.js.map +1 -1
- package/dist/multiEndpointMCPPipe.js +0 -0
- package/dist/multiEndpointMCPPipe.js.map +1 -1
- package/dist/services/mcpServer.d.ts +3 -0
- package/dist/services/mcpServer.js +8 -8
- package/dist/services/mcpServer.js.map +1 -1
- package/dist/streamableHttpMCPClient.d.ts +4 -0
- package/dist/streamableHttpMCPClient.js +2 -2
- package/dist/streamableHttpMCPClient.js.map +1 -1
- package/dist/webServer.js +13 -13
- package/dist/webServer.js.map +1 -1
- package/package.json +20 -20
- package/web/README.md +169 -0
- package/web/dist/assets/index-lAiGa4ms.js.map +1 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/mcpServerProxy.ts","../src/configManager.ts","../src/logger.ts","../src/modelScopeMCPClient.ts","../src/streamableHttpMCPClient.ts"],"sourcesContent":["#!/usr/bin/env node\n\n/**\n * MCP Server Proxy - JavaScript Implementation\n * Provides a proxy to aggregate multiple MCP servers dynamically from configuration\n * Version: 0.3.0 - Manual JSON-RPC 2.0 implementation (no SDK) with dynamic config\n */\n\nimport { type ChildProcess, spawn } from \"node:child_process\";\nimport { readFileSync } from \"node:fs\";\nimport { dirname, resolve } from \"node:path\";\nimport process from \"node:process\";\nimport { fileURLToPath } from \"node:url\";\nimport {\n type LocalMCPServerConfig,\n type MCPServerConfig,\n type MCPToolConfig,\n type SSEMCPServerConfig,\n type StreamableHTTPMCPServerConfig,\n configManager,\n} from \"./configManager\";\nimport { logger as globalLogger } from \"./logger\";\nimport { ModelScopeMCPClient } from \"./modelScopeMCPClient\";\nimport { StreamableHTTPMCPClient } from \"./streamableHttpMCPClient\";\n\n// ESM 兼容的 __dirname\nconst __dirname = dirname(fileURLToPath(import.meta.url));\n\n// 为 MCPProxy 创建带标签的 logger\nconst logger = globalLogger.withTag(\"MCPProxy\");\n\n// 如果在守护进程模式下运行,初始化日志文件\nif (process.env.XIAOZHI_DAEMON === \"true\" && process.env.XIAOZHI_CONFIG_DIR) {\n globalLogger.initLogFile(process.env.XIAOZHI_CONFIG_DIR);\n globalLogger.enableFileLogging(true);\n}\n\n// Type definitions\n\ninterface Tool {\n name: string;\n description?: string;\n inputSchema?: any;\n}\n\ninterface PendingRequest {\n resolve: (value: any) => void;\n reject: (reason: any) => void;\n method: string; // 添加方法名以便错误处理\n}\n\n// 定义 MCP 客户端接口\nexport interface IMCPClient {\n initialized: boolean;\n tools: Tool[];\n originalTools: Tool[];\n start(): Promise<void>;\n refreshTools(): Promise<void>;\n callTool(toolName: string, arguments_: any): Promise<any>;\n stop(): void | Promise<void>;\n getOriginalToolName(prefixedToolName: string): string | null;\n}\n\n/**\n * MCP Client for communicating with child MCP servers\n */\nexport class MCPClient implements IMCPClient {\n private name: string;\n private config: LocalMCPServerConfig;\n private process: ChildProcess | null;\n public initialized: boolean;\n public tools: Tool[];\n public originalTools: Tool[]; // 存储原始工具名称\n private requestId: number;\n private pendingRequests: Map<number, PendingRequest>;\n private messageBuffer: string;\n\n constructor(name: string, config: LocalMCPServerConfig) {\n this.name = name;\n this.config = config;\n this.process = null;\n this.initialized = false;\n this.tools = [];\n this.originalTools = [];\n this.requestId = 1;\n this.pendingRequests = new Map();\n this.messageBuffer = \"\";\n }\n\n /**\n * Resolve command for cross-platform execution\n * On Windows, npm/npx/uvx commands need special handling\n */\n public resolveCommand(\n command: string,\n args: string[]\n ): { resolvedCommand: string; resolvedArgs: string[] } {\n if (process.platform === \"win32\") {\n // On Windows, npm, npx, and uvx are .cmd files or .exe files\n if (command === \"npm\" || command === \"npx\") {\n return {\n resolvedCommand: `${command}.cmd`,\n resolvedArgs: args,\n };\n }\n // uvx is typically installed as uvx.bat on Windows (via pyenv)\n if (command === \"uvx\") {\n return {\n resolvedCommand: \"uvx.bat\",\n resolvedArgs: args,\n };\n }\n }\n\n return {\n resolvedCommand: command,\n resolvedArgs: args,\n };\n }\n\n /**\n * 生成带前缀的工具名称\n * 将服务器名称中的中划线替换为下划线,并添加 xzcli 前缀\n */\n private generatePrefixedToolName(originalToolName: string): string {\n const normalizedServerName = this.name.replace(/-/g, \"_\");\n return `${normalizedServerName}_xzcli_${originalToolName}`;\n }\n\n /**\n * 根据前缀工具名称获取原始工具名称\n */\n getOriginalToolName(prefixedToolName: string): string | null {\n const normalizedServerName = this.name.replace(/-/g, \"_\");\n const prefix = `${normalizedServerName}_xzcli_`;\n\n if (prefixedToolName.startsWith(prefix)) {\n return prefixedToolName.substring(prefix.length);\n }\n\n return null;\n }\n\n async start() {\n logger.info(`正在启动 MCP 客户端:${this.name}`);\n\n const { command, args, env } = this.config;\n\n // Handle cross-platform command execution\n const { resolvedCommand, resolvedArgs } = this.resolveCommand(\n command,\n args\n );\n\n const spawnOptions: any = {\n stdio: [\"pipe\", \"pipe\", \"pipe\"],\n };\n\n // Set working directory to user's current working directory\n // This ensures relative paths in MCP server configs are resolved correctly\n const userWorkingDir = process.env.XIAOZHI_CONFIG_DIR || process.cwd();\n spawnOptions.cwd = userWorkingDir;\n\n // Add environment variables if specified\n if (env) {\n spawnOptions.env = { ...process.env, ...env };\n } else {\n spawnOptions.env = { ...process.env };\n }\n\n // On Windows, we need to set shell: true for npm/npx/uvx commands\n if (\n process.platform === \"win32\" &&\n (command === \"npm\" || command === \"npx\" || command === \"uvx\")\n ) {\n spawnOptions.shell = true;\n }\n\n logger.debug(\n `${this.name} 正在生成进程:${resolvedCommand} ${resolvedArgs.join(\n \" \"\n )} 工作目录:${spawnOptions.cwd}`\n );\n logger.debug(\n `${this.name} 平台:${process.platform},shell 模式:${\n spawnOptions.shell || false\n }`\n );\n\n this.process = spawn(resolvedCommand, resolvedArgs, spawnOptions);\n\n // Handle process stdout - parse JSON-RPC messages\n this.process.stdout?.on(\"data\", (data: Buffer) => {\n this.handleStdoutData(data.toString());\n });\n\n // Handle process stderr - log errors\n this.process.stderr?.on(\"data\", (data: Buffer) => {\n logger.debug(`${this.name} 标准错误输出:${data.toString().trim()}`);\n });\n\n // Handle process exit\n this.process.on(\n \"exit\",\n (code: number | null, signal: NodeJS.Signals | null) => {\n logger.error(\n `${this.name} 进程已退出,退出码:${code},信号:${signal}`\n );\n this.initialized = false;\n }\n );\n\n // Handle process error\n this.process.on(\"error\", (error: Error) => {\n logger.error(`${this.name} 进程错误:${error.message}`);\n this.initialized = false;\n });\n\n // Initialize the MCP connection\n await this.initialize();\n }\n\n handleStdoutData(data: string): void {\n this.messageBuffer = `${this.messageBuffer}${data}`;\n\n // Split by newlines and process complete messages\n const lines = this.messageBuffer.split(\"\\n\");\n this.messageBuffer = lines.pop() || \"\"; // Keep incomplete line in buffer\n\n for (const line of lines) {\n if (line.trim()) {\n try {\n const message = JSON.parse(line.trim());\n this.handleMessage(message);\n } catch (error) {\n logger.error(`${this.name} 解析消息失败:${line.trim()}`);\n }\n }\n }\n }\n\n handleMessage(message: any): void {\n logger.debug(\n `${this.name} 收到消息:${JSON.stringify(message).substring(0, 200)}...`\n );\n\n if (message.id && this.pendingRequests.has(message.id)) {\n // This is a response to our request\n const pendingRequest = this.pendingRequests.get(message.id);\n if (pendingRequest) {\n const { resolve, reject } = pendingRequest;\n this.pendingRequests.delete(message.id);\n\n if (message.error) {\n // 对于 notifications/initialized 的 \"Method not found\" 错误,记录为调试信息而不是错误\n if (\n message.error.code === -32601 &&\n pendingRequest.method === \"notifications/initialized\"\n ) {\n logger.debug(\n `${this.name} 服务器不支持 notifications/initialized 通知,这是正常的`\n );\n resolve(null); // 将其视为成功\n } else {\n reject(\n new Error(\n `${message.error.message} (code: ${message.error.code})`\n )\n );\n }\n } else {\n resolve(message.result);\n }\n }\n }\n // Note: We don't handle notifications or requests from child servers in this proxy\n }\n\n async sendRequest(method: string, params: any = {}): Promise<any> {\n if (!this.process || !this.process.stdin) {\n throw new Error(`${this.name} 进程不可用`);\n }\n\n const id = this.requestId++;\n const request = {\n jsonrpc: \"2.0\",\n id: id,\n method: method,\n params: params,\n };\n\n return new Promise((resolve, reject) => {\n this.pendingRequests.set(id, { resolve, reject, method });\n\n // Set timeout for request\n setTimeout(() => {\n if (this.pendingRequests.has(id)) {\n this.pendingRequests.delete(id);\n reject(new Error(`请求超时:${method}`));\n }\n }, 30000); // 30 second timeout\n\n const message = `${JSON.stringify(request)}\\n`;\n logger.debug(`${this.name} 正在发送:${message.trim()}`);\n this.process?.stdin?.write(message);\n });\n }\n\n async initialize() {\n try {\n // Send initialize request\n const initResult = await this.sendRequest(\"initialize\", {\n protocolVersion: \"2024-11-05\",\n capabilities: {},\n clientInfo: {\n name: \"MCPProxy\",\n version: \"0.3.0\",\n },\n });\n\n logger.info(\n `${this.name} 已初始化,能力:${JSON.stringify(\n (initResult as any).capabilities\n )}`\n );\n\n // Send initialized notification (ignore errors as some servers don't support it)\n try {\n const notification = {\n jsonrpc: \"2.0\",\n method: \"notifications/initialized\",\n };\n this.process?.stdin?.write(`${JSON.stringify(notification)}\\n`);\n } catch (error) {\n logger.debug(\n `${\n this.name\n } notifications/initialized 发送失败(某些服务器不支持此通知):${\n error instanceof Error ? error.message : String(error)\n }`\n );\n }\n\n // Get tools list\n await this.refreshTools();\n\n this.initialized = true;\n logger.info(`${this.name} 客户端已就绪,共 ${this.tools.length} 个工具`);\n } catch (error) {\n logger.error(\n `初始化 ${this.name} 失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n throw error;\n }\n }\n\n async refreshTools(): Promise<void> {\n try {\n const result = await this.sendRequest(\"tools/list\");\n this.originalTools = (result as any).tools || [];\n\n // 为每个工具生成带前缀的名称,并应用过滤\n const allPrefixedTools = this.originalTools.map((tool) => ({\n ...tool,\n name: this.generatePrefixedToolName(tool.name),\n }));\n\n // 根据配置过滤工具\n this.tools = this.filterEnabledTools(allPrefixedTools);\n\n // 更新配置文件中的工具列表(如果需要)\n await this.updateToolsConfig();\n\n logger.info(\n `${this.name} 加载了 ${\n this.originalTools.length\n } 个工具:${this.originalTools.map((t) => t.name).join(\", \")}`\n );\n logger.info(\n `${this.name} 启用了 ${this.tools.length} 个工具:${this.tools\n .map((t) => t.name)\n .join(\", \")}`\n );\n } catch (error) {\n logger.error(\n `从 ${this.name} 获取工具失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n this.tools = [];\n this.originalTools = [];\n }\n }\n\n /**\n * 过滤启用的工具\n */\n private filterEnabledTools(allTools: Tool[]): Tool[] {\n return allTools.filter((tool) => {\n const originalName = this.getOriginalToolName(tool.name);\n if (!originalName) return true; // 如果无法解析原始名称,默认启用\n\n return configManager.isToolEnabled(this.name, originalName);\n });\n }\n\n /**\n * 更新配置文件中的工具列表\n */\n private async updateToolsConfig(): Promise<void> {\n try {\n const currentConfig = configManager.getServerToolsConfig(this.name);\n const toolsConfig: Record<string, MCPToolConfig> = {};\n\n // 为每个工具创建配置项\n for (const tool of this.originalTools) {\n const existingConfig = currentConfig[tool.name];\n toolsConfig[tool.name] = {\n description: tool.description || \"\",\n enable: existingConfig?.enable !== false, // 默认启用\n };\n }\n\n // 只有当配置发生变化时才更新\n const hasChanges = Object.keys(toolsConfig).some((toolName) => {\n const existing = currentConfig[toolName];\n const newConfig = toolsConfig[toolName];\n return (\n !existing ||\n existing.enable !== newConfig.enable ||\n existing.description !== newConfig.description\n );\n });\n\n if (hasChanges || Object.keys(currentConfig).length === 0) {\n configManager.updateServerToolsConfig(this.name, toolsConfig);\n logger.info(`${this.name} 已更新工具配置`);\n }\n } catch (error) {\n logger.error(\n `更新 ${this.name} 的工具配置失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n }\n }\n\n async callTool(prefixedName: string, arguments_: any): Promise<any> {\n try {\n // 将前缀名称转换回原始名称\n const originalName = this.getOriginalToolName(prefixedName);\n if (!originalName) {\n throw new Error(`无效的工具名称格式:${prefixedName}`);\n }\n\n const result = await this.sendRequest(\"tools/call\", {\n name: originalName,\n arguments: arguments_,\n });\n return result;\n } catch (error) {\n logger.error(\n `在 ${\n this.name\n } 上调用工具 ${prefixedName} (原始名称:${this.getOriginalToolName(\n prefixedName\n )}) 失败:${error instanceof Error ? error.message : String(error)}`\n );\n throw error;\n }\n }\n\n stop(): void {\n if (this.process) {\n logger.info(`正在停止 ${this.name} 客户端`);\n this.process.kill(\"SIGTERM\");\n this.process = null;\n }\n this.initialized = false;\n }\n}\n\n/**\n * Configuration loader for MCP servers\n */\nexport function loadMCPConfig(): Record<string, MCPServerConfig> {\n try {\n // 首先尝试从配置管理器读取\n if (configManager.configExists()) {\n const mcpServers = configManager.getMcpServers();\n logger.info(\n `从配置文件加载了 ${Object.keys(mcpServers).length} 个 MCP 服务`\n );\n return mcpServers as Record<string, MCPServerConfig>;\n }\n\n // 如果配置文件不存在,尝试从旧的 mcp_server.json 读取(向后兼容)\n const legacyConfigPath = resolve(__dirname, \"mcp_server.json\");\n if (readFileSync) {\n // 检查是否可以读取文件\n try {\n const configData = readFileSync(legacyConfigPath, \"utf8\");\n const config = JSON.parse(configData);\n\n if (!config.mcpServers || typeof config.mcpServers !== \"object\") {\n throw new Error(\"无效的配置:mcpServers 部分未找到或无效\");\n }\n\n logger.info(\n `从旧配置文件加载了 ${\n Object.keys(config.mcpServers).length\n } 个 MCP 服务(建议迁移到新配置格式)`\n );\n return config.mcpServers;\n } catch (legacyError) {\n // 旧配置文件也不存在或无效\n logger.error('配置文件不存在,请运行 \"xiaozhi init\" 初始化配置');\n throw new Error('配置文件不存在,请运行 \"xiaozhi init\" 初始化配置');\n }\n }\n\n throw new Error('配置文件不存在,请运行 \"xiaozhi init\" 初始化配置');\n } catch (error) {\n logger.error(\n `加载 MCP 配置失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n throw error;\n }\n}\n\n/**\n * MCP Server Proxy - Main proxy server implementation\n */\nexport class MCPServerProxy {\n private clients: Map<string, IMCPClient>;\n private toolMap: Map<string, string>; // Maps tool name to client name\n public initialized: boolean;\n private config: Record<string, MCPServerConfig> | null;\n\n constructor() {\n this.clients = new Map();\n this.toolMap = new Map(); // Maps tool name to client name\n this.initialized = false;\n this.config = null;\n }\n\n async start() {\n logger.info(\"正在启动 MCP 服务代理\");\n\n // Load configuration\n this.config = loadMCPConfig();\n\n // Initialize child MCP clients from configuration\n const clientPromises = [];\n\n for (const [serverName, serverConfig] of Object.entries(this.config)) {\n logger.info(`正在初始化 MCP 客户端:${serverName}`);\n\n let client: IMCPClient;\n\n // 判断服务类型\n if (\"url\" in serverConfig) {\n // URL 类型的配置\n const url = serverConfig.url;\n\n // 判断是 SSE 还是 Streamable HTTP\n const isSSE =\n // 1. 显式指定 type: \"sse\"\n (\"type\" in serverConfig && serverConfig.type === \"sse\") ||\n // 2. URL 以 /sse 结尾\n url.endsWith(\"/sse\") ||\n // 3. 域名包含 modelscope.net(向后兼容魔搭社区)\n url.includes(\"modelscope.net\");\n\n if (isSSE) {\n // SSE MCP 服务\n client = new ModelScopeMCPClient(\n serverName,\n serverConfig as SSEMCPServerConfig\n );\n } else {\n // Streamable HTTP MCP 服务\n client = new StreamableHTTPMCPClient(\n serverName,\n serverConfig as StreamableHTTPMCPServerConfig\n );\n }\n } else {\n // 本地 MCP 服务\n client = new MCPClient(\n serverName,\n serverConfig as LocalMCPServerConfig\n );\n }\n\n this.clients.set(serverName, client);\n clientPromises.push(client.start());\n }\n\n // Start all clients\n try {\n const results = await Promise.allSettled(clientPromises);\n\n // Check for failed clients\n let successCount = 0;\n for (let i = 0; i < results.length; i++) {\n const result = results[i];\n const serverName = Object.keys(this.config)[i];\n\n if (result.status === \"fulfilled\") {\n successCount++;\n logger.info(`成功启动 MCP 客户端:${serverName}`);\n } else {\n logger.error(\n `启动 MCP 客户端 ${serverName} 失败:${result.reason.message}`\n );\n // Remove failed client from clients map\n this.clients.delete(serverName);\n }\n }\n\n if (successCount === 0) {\n throw new Error(\"没有成功启动任何 MCP 客户端\");\n }\n\n // Build tool mapping\n this.buildToolMap();\n this.initialized = true;\n\n logger.info(\n `MCP 服务代理初始化成功,启动了 ${successCount}/${\n Object.keys(this.config).length\n } 个客户端`\n );\n } catch (error) {\n logger.error(\n `启动 MCP 客户端失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n throw error;\n }\n }\n\n buildToolMap() {\n this.toolMap.clear();\n\n for (const [clientName, client] of this.clients) {\n for (const tool of client.tools) {\n // 工具名称现在已经带有前缀,应该不会有重复\n // 但我们仍然检查以防万一\n if (this.toolMap.has(tool.name)) {\n logger.error(\n `重复的工具名称:${\n tool.name\n } (来自 ${clientName} 和 ${this.toolMap.get(tool.name)})`\n );\n } else {\n this.toolMap.set(tool.name, clientName);\n }\n }\n }\n\n logger.info(`已构建工具映射,共 ${this.toolMap.size} 个工具`);\n logger.debug(`工具映射:${Array.from(this.toolMap.keys()).join(\", \")}`);\n }\n\n getAllTools(): Tool[] {\n const allTools: Tool[] = [];\n for (const client of this.clients.values()) {\n allTools.push(...client.tools);\n }\n return allTools;\n }\n\n /**\n * 获取所有服务器的信息\n */\n getAllServers(): Array<{\n name: string;\n toolCount: number;\n enabledToolCount: number;\n }> {\n const servers: Array<{\n name: string;\n toolCount: number;\n enabledToolCount: number;\n }> = [];\n\n for (const [serverName, client] of this.clients) {\n servers.push({\n name: serverName,\n toolCount: client.originalTools.length,\n enabledToolCount: client.tools.length,\n });\n }\n\n return servers;\n }\n\n /**\n * 获取指定服务器的工具信息\n */\n getServerTools(\n serverName: string\n ): Array<{ name: string; description: string; enabled: boolean }> {\n const client = this.clients.get(serverName);\n if (!client) {\n throw new Error(`未找到服务器 ${serverName}`);\n }\n\n return client.originalTools.map((tool) => ({\n name: tool.name,\n description: tool.description || \"\",\n enabled: configManager.isToolEnabled(serverName, tool.name),\n }));\n }\n\n /**\n * 刷新指定服务器的工具列表\n */\n async refreshServerTools(serverName: string): Promise<void> {\n const client = this.clients.get(serverName);\n if (!client) {\n throw new Error(`未找到服务器 ${serverName}`);\n }\n\n await client.refreshTools();\n this.buildToolMap(); // 重新构建工具映射\n }\n\n /**\n * 刷新所有服务器的工具列表\n */\n async refreshAllTools(): Promise<void> {\n const refreshPromises = Array.from(this.clients.values()).map((client) =>\n client.refreshTools()\n );\n await Promise.allSettled(refreshPromises);\n this.buildToolMap(); // 重新构建工具映射\n }\n\n async callTool(toolName: string, arguments_: any): Promise<any> {\n const clientName = this.toolMap.get(toolName);\n if (!clientName) {\n throw new Error(`未知的工具:${toolName}`);\n }\n\n const client = this.clients.get(clientName);\n if (!client || !client.initialized) {\n throw new Error(`客户端 ${clientName} 不可用`);\n }\n\n return await client.callTool(toolName, arguments_);\n }\n\n stop() {\n logger.info(\"正在停止 MCP 服务代理\");\n for (const client of this.clients.values()) {\n client.stop();\n }\n this.initialized = false;\n }\n}\n\n/**\n * Manual JSON-RPC 2.0 Server Implementation\n */\nexport class JSONRPCServer {\n private proxy: MCPServerProxy;\n private requestId: number;\n\n constructor(proxy: MCPServerProxy) {\n this.proxy = proxy;\n this.requestId = 1;\n }\n\n async handleMessage(message: string): Promise<string | null> {\n try {\n const parsedMessage = JSON.parse(message);\n logger.debug(\n `收到请求:${JSON.stringify(parsedMessage).substring(0, 200)}...`\n );\n\n if (parsedMessage.method) {\n if (parsedMessage.id !== undefined) {\n // This is a request\n const response = await this.handleRequest(parsedMessage);\n return JSON.stringify(response);\n }\n // This is a notification\n await this.handleNotification(parsedMessage);\n return null; // No response for notifications\n }\n throw new Error(\"无效的 JSON-RPC 消息\");\n } catch (error) {\n logger.error(\n `处理消息时出错:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n return JSON.stringify({\n jsonrpc: \"2.0\",\n id: null,\n error: {\n code: -32700,\n message: \"解析错误\",\n },\n });\n }\n }\n\n async handleRequest(request: any): Promise<any> {\n const { id, method, params = {} } = request;\n\n try {\n let result: any;\n\n switch (method) {\n case \"initialize\":\n result = await this.handleInitialize(params);\n break;\n case \"tools/list\":\n result = await this.handleToolsList(params);\n break;\n case \"tools/call\":\n result = await this.handleToolsCall(params);\n break;\n case \"ping\":\n result = await this.handlePing(params);\n break;\n default:\n throw new Error(`未知的方法:${method}`);\n }\n\n return {\n jsonrpc: \"2.0\",\n id: id,\n result: result,\n };\n } catch (error) {\n logger.error(\n `处理请求 ${method} 时出错:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n return {\n jsonrpc: \"2.0\",\n id: id,\n error: {\n code: -32603,\n message: error instanceof Error ? error.message : String(error),\n },\n };\n }\n }\n\n async handleNotification(notification: any): Promise<void> {\n const { method } = notification;\n\n switch (method) {\n case \"notifications/initialized\":\n logger.info(\"客户端发送了初始化通知\");\n break;\n default:\n logger.debug(`收到通知:${method}`);\n break;\n }\n }\n\n async handleInitialize(params: any): Promise<any> {\n logger.info(`收到客户端的初始化请求:${JSON.stringify(params.clientInfo)}`);\n\n return {\n protocolVersion: \"2024-11-05\",\n capabilities: {\n tools: {\n listChanged: false,\n },\n },\n serverInfo: {\n name: \"MCPServerProxy\",\n version: \"0.3.0\",\n },\n };\n }\n\n async handleToolsList(_params: any): Promise<any> {\n if (!this.proxy.initialized) {\n throw new Error(\"代理未初始化\");\n }\n\n const tools = this.proxy.getAllTools();\n logger.info(`返回 ${tools.length} 个工具`);\n\n return {\n tools: tools,\n };\n }\n\n async handleToolsCall(params: any): Promise<any> {\n const { name, arguments: args } = params;\n\n if (!name) {\n throw new Error(\"工具名称是必需的\");\n }\n\n logger.info(`调用工具:${name},参数:${JSON.stringify(args)}`);\n\n const result = await this.proxy.callTool(name, args || {});\n\n // 添加调试日志\n logger.info(`工具调用结果类型: ${typeof result}`);\n logger.info(`工具调用结果: ${JSON.stringify(result).substring(0, 500)}...`);\n\n return result;\n }\n\n async handlePing(_params: any): Promise<any> {\n logger.debug(\"收到 ping 请求\");\n return {}; // Empty response for ping\n }\n}\n\n/**\n * Check if running in MCP Server mode\n */\nfunction isMCPServerMode(): boolean {\n return process.env.MCP_SERVER_MODE === \"true\";\n}\n\n/**\n * Main function to start the MCP Server Proxy\n */\nasync function main() {\n logger.info(\"正在启动 MCP 服务代理\");\n\n // Create and start the proxy\n const proxy = new MCPServerProxy();\n const jsonrpcServer = new JSONRPCServer(proxy);\n\n // Setup graceful shutdown\n const cleanup = () => {\n logger.info(\"正在关闭 MCP 服务代理\");\n proxy.stop();\n process.exit(0);\n };\n\n process.on(\"SIGINT\", cleanup);\n process.on(\"SIGTERM\", cleanup);\n\n try {\n // Start the proxy (initialize child clients)\n await proxy.start();\n\n // In MCP Server mode, announce readiness\n if (isMCPServerMode()) {\n logger.info(\"MCP proxy ready in server mode\");\n console.log(\"MCP proxy ready\"); // For parent process detection\n }\n\n // Handle stdin/stdout communication\n process.stdin.setEncoding(\"utf8\");\n\n let messageBuffer = \"\";\n\n process.stdin.on(\"data\", async (data) => {\n messageBuffer = `${messageBuffer}${data}`;\n\n // Split by newlines and process complete messages\n const lines = messageBuffer.split(\"\\n\");\n messageBuffer = lines.pop() || \"\"; // Keep incomplete line in buffer\n\n for (const line of lines) {\n if (line.trim()) {\n try {\n const response = await jsonrpcServer.handleMessage(line.trim());\n if (response) {\n process.stdout.write(`${response}\\n`);\n }\n } catch (error) {\n logger.error(\n `处理消息时出错:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n }\n }\n }\n });\n\n process.stdin.on(\"end\", () => {\n logger.info(\"标准输入已关闭,正在关闭\");\n cleanup();\n });\n\n logger.info(\"MCP 服务代理正在通过 stdio 运行\");\n } catch (error) {\n logger.error(\n `启动 MCP 服务代理失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n process.exit(1);\n }\n}\n\n// Run the server if this file is executed directly\n// Use fileURLToPath to properly handle Windows paths\nconst currentFileUrl = import.meta.url;\nconst scriptPath = fileURLToPath(currentFileUrl);\nconst argv1Path = process.argv[1];\n\nif (scriptPath === argv1Path) {\n main().catch((error) => {\n logger.error(\n `未处理的错误:${error instanceof Error ? error.message : String(error)}`\n );\n process.exit(1);\n });\n}\n","import { copyFileSync, existsSync, readFileSync, writeFileSync } from \"node:fs\";\nimport { dirname, resolve } from \"node:path\";\nimport { fileURLToPath } from \"node:url\";\n\n// 在 ESM 中,需要从 import.meta.url 获取当前文件目录\nconst __dirname = dirname(fileURLToPath(import.meta.url));\n\n// 默认连接配置\nconst DEFAULT_CONNECTION_CONFIG: Required<ConnectionConfig> = {\n heartbeatInterval: 30000, // 30秒心跳间隔\n heartbeatTimeout: 10000, // 10秒心跳超时\n reconnectInterval: 5000, // 5秒重连间隔\n};\n\n// 配置文件接口定义\n// 本地 MCP 服务配置\nexport interface LocalMCPServerConfig {\n command: string;\n args: string[];\n env?: Record<string, string>;\n}\n\n// ModelScope SSE MCP 服务配置\nexport interface SSEMCPServerConfig {\n type: \"sse\";\n url: string;\n}\n\n// Streamable HTTP MCP 服务配置\nexport interface StreamableHTTPMCPServerConfig {\n type?: \"streamable-http\"; // 可选,因为默认就是 streamable-http\n url: string;\n}\n\n// 统一的 MCP 服务配置\nexport type MCPServerConfig =\n | LocalMCPServerConfig\n | SSEMCPServerConfig\n | StreamableHTTPMCPServerConfig;\n\nexport interface MCPToolConfig {\n description?: string;\n enable: boolean;\n}\n\nexport interface MCPServerToolsConfig {\n tools: Record<string, MCPToolConfig>;\n}\n\nexport interface ConnectionConfig {\n heartbeatInterval?: number; // 心跳检测间隔(毫秒),默认30000\n heartbeatTimeout?: number; // 心跳超时时间(毫秒),默认10000\n reconnectInterval?: number; // 重连间隔(毫秒),默认5000\n}\n\nexport interface ModelScopeConfig {\n apiKey?: string; // ModelScope API 密钥\n}\n\nexport interface WebUIConfig {\n port?: number; // Web UI 端口号,默认 9999\n autoRestart?: boolean; // 是否在配置更新后自动重启服务,默认 true\n}\n\nexport interface AppConfig {\n mcpEndpoint: string | string[];\n mcpServers: Record<string, MCPServerConfig>;\n mcpServerConfig?: Record<string, MCPServerToolsConfig>;\n connection?: ConnectionConfig; // 连接配置(可选,用于向后兼容)\n modelscope?: ModelScopeConfig; // ModelScope 配置(可选)\n webUI?: WebUIConfig; // Web UI 配置(可选)\n}\n\n/**\n * 配置管理类\n * 负责管理应用配置,提供只读访问和安全的配置更新功能\n */\nexport class ConfigManager {\n private static instance: ConfigManager;\n private defaultConfigPath: string;\n private config: AppConfig | null = null;\n\n private constructor() {\n this.defaultConfigPath = resolve(__dirname, \"xiaozhi.config.default.json\");\n }\n\n /**\n * 获取配置文件路径(动态计算)\n */\n private getConfigFilePath(): string {\n // 配置文件路径 - 优先使用环境变量指定的目录,否则使用当前工作目录\n const configDir = process.env.XIAOZHI_CONFIG_DIR || process.cwd();\n return resolve(configDir, \"xiaozhi.config.json\");\n }\n\n /**\n * 获取配置管理器单例实例\n */\n public static getInstance(): ConfigManager {\n if (!ConfigManager.instance) {\n ConfigManager.instance = new ConfigManager();\n }\n return ConfigManager.instance;\n }\n\n /**\n * 检查配置文件是否存在\n */\n public configExists(): boolean {\n const configPath = this.getConfigFilePath();\n return existsSync(configPath);\n }\n\n /**\n * 初始化配置文件\n * 从 config.default.json 复制到 config.json\n */\n public initConfig(): void {\n if (!existsSync(this.defaultConfigPath)) {\n throw new Error(\"默认配置文件 xiaozhi.config.default.json 不存在\");\n }\n\n if (this.configExists()) {\n throw new Error(\"配置文件 xiaozhi.config.json 已存在,无需重复初始化\");\n }\n\n const configPath = this.getConfigFilePath();\n copyFileSync(this.defaultConfigPath, configPath);\n this.config = null; // 重置缓存\n }\n\n /**\n * 加载配置文件\n */\n private loadConfig(): AppConfig {\n if (!this.configExists()) {\n throw new Error(\n \"配置文件 xiaozhi.config.json 不存在,请先运行 xiaozhi init 初始化配置\"\n );\n }\n\n try {\n const configPath = this.getConfigFilePath();\n const configData = readFileSync(configPath, \"utf8\");\n const config = JSON.parse(configData) as AppConfig;\n\n // 验证配置结构\n this.validateConfig(config);\n\n return config;\n } catch (error) {\n if (error instanceof SyntaxError) {\n throw new Error(`配置文件格式错误: ${error.message}`);\n }\n throw error;\n }\n }\n\n /**\n * 验证配置文件结构\n */\n private validateConfig(config: unknown): void {\n if (!config || typeof config !== \"object\") {\n throw new Error(\"配置文件格式错误:根对象无效\");\n }\n\n const configObj = config as Record<string, unknown>;\n\n if (configObj.mcpEndpoint === undefined || configObj.mcpEndpoint === null) {\n throw new Error(\"配置文件格式错误:mcpEndpoint 字段无效\");\n }\n\n // 验证 mcpEndpoint 类型(字符串或字符串数组)\n if (typeof configObj.mcpEndpoint === \"string\") {\n // 空字符串是允许的,getMcpEndpoints 会返回空数组\n } else if (Array.isArray(configObj.mcpEndpoint)) {\n if (configObj.mcpEndpoint.length === 0) {\n throw new Error(\"配置文件格式错误:mcpEndpoint 数组不能为空\");\n }\n for (const endpoint of configObj.mcpEndpoint) {\n if (typeof endpoint !== \"string\" || endpoint.trim() === \"\") {\n throw new Error(\n \"配置文件格式错误:mcpEndpoint 数组中的每个元素必须是非空字符串\"\n );\n }\n }\n } else {\n throw new Error(\"配置文件格式错误:mcpEndpoint 必须是字符串或字符串数组\");\n }\n\n if (!configObj.mcpServers || typeof configObj.mcpServers !== \"object\") {\n throw new Error(\"配置文件格式错误:mcpServers 字段无效\");\n }\n\n // 验证每个 MCP 服务配置\n for (const [serverName, serverConfig] of Object.entries(\n configObj.mcpServers as Record<string, unknown>\n )) {\n if (!serverConfig || typeof serverConfig !== \"object\") {\n throw new Error(`配置文件格式错误:mcpServers.${serverName} 无效`);\n }\n\n const sc = serverConfig as Record<string, unknown>;\n\n // 检查服务类型\n if (sc.url && typeof sc.url === \"string\") {\n // URL 类型的服务(SSE 或 Streamable HTTP)\n // type 字段是可选的,可以是 \"sse\" 或 \"streamable-http\"\n if (sc.type && sc.type !== \"sse\" && sc.type !== \"streamable-http\") {\n throw new Error(\n `配置文件格式错误:mcpServers.${serverName}.type 必须是 \"sse\" 或 \"streamable-http\"`\n );\n }\n } else {\n // 本地类型的验证\n if (!sc.command || typeof sc.command !== \"string\") {\n throw new Error(\n `配置文件格式错误:mcpServers.${serverName}.command 无效`\n );\n }\n\n if (!Array.isArray(sc.args)) {\n throw new Error(\n `配置文件格式错误:mcpServers.${serverName}.args 必须是数组`\n );\n }\n\n if (sc.env && typeof sc.env !== \"object\") {\n throw new Error(\n `配置文件格式错误:mcpServers.${serverName}.env 必须是对象`\n );\n }\n }\n }\n }\n\n /**\n * 获取配置(只读)\n */\n public getConfig(): Readonly<AppConfig> {\n if (!this.config) {\n this.config = this.loadConfig();\n }\n\n // 返回深度只读副本\n return JSON.parse(JSON.stringify(this.config));\n }\n\n /**\n * 获取 MCP 端点(向后兼容)\n * @deprecated 使用 getMcpEndpoints() 获取所有端点\n */\n public getMcpEndpoint(): string {\n const config = this.getConfig();\n if (Array.isArray(config.mcpEndpoint)) {\n return config.mcpEndpoint[0] || \"\";\n }\n return config.mcpEndpoint;\n }\n\n /**\n * 获取所有 MCP 端点\n */\n public getMcpEndpoints(): string[] {\n const config = this.getConfig();\n if (Array.isArray(config.mcpEndpoint)) {\n return [...config.mcpEndpoint];\n }\n return config.mcpEndpoint ? [config.mcpEndpoint] : [];\n }\n\n /**\n * 获取 MCP 服务配置\n */\n public getMcpServers(): Readonly<Record<string, MCPServerConfig>> {\n const config = this.getConfig();\n return config.mcpServers;\n }\n\n /**\n * 获取 MCP 服务工具配置\n */\n public getMcpServerConfig(): Readonly<Record<string, MCPServerToolsConfig>> {\n const config = this.getConfig();\n return config.mcpServerConfig || {};\n }\n\n /**\n * 获取指定服务的工具配置\n */\n public getServerToolsConfig(\n serverName: string\n ): Readonly<Record<string, MCPToolConfig>> {\n const serverConfig = this.getMcpServerConfig();\n return serverConfig[serverName]?.tools || {};\n }\n\n /**\n * 检查工具是否启用\n */\n public isToolEnabled(serverName: string, toolName: string): boolean {\n const toolsConfig = this.getServerToolsConfig(serverName);\n const toolConfig = toolsConfig[toolName];\n return toolConfig?.enable !== false; // 默认启用\n }\n\n /**\n * 更新 MCP 端点(支持字符串或数组)\n */\n public updateMcpEndpoint(endpoint: string | string[]): void {\n if (Array.isArray(endpoint)) {\n if (endpoint.length === 0) {\n throw new Error(\"MCP 端点数组不能为空\");\n }\n for (const ep of endpoint) {\n if (!ep || typeof ep !== \"string\") {\n throw new Error(\"MCP 端点数组中的每个元素必须是非空字符串\");\n }\n }\n } else {\n if (!endpoint || typeof endpoint !== \"string\") {\n throw new Error(\"MCP 端点必须是非空字符串\");\n }\n }\n\n const config = this.getConfig();\n const newConfig = { ...config, mcpEndpoint: endpoint };\n this.saveConfig(newConfig);\n }\n\n /**\n * 添加 MCP 端点\n */\n public addMcpEndpoint(endpoint: string): void {\n if (!endpoint || typeof endpoint !== \"string\") {\n throw new Error(\"MCP 端点必须是非空字符串\");\n }\n\n const config = this.getConfig();\n const currentEndpoints = this.getMcpEndpoints();\n\n // 检查是否已存在\n if (currentEndpoints.includes(endpoint)) {\n throw new Error(`MCP 端点 ${endpoint} 已存在`);\n }\n\n const newEndpoints = [...currentEndpoints, endpoint];\n const newConfig = { ...config, mcpEndpoint: newEndpoints };\n this.saveConfig(newConfig);\n }\n\n /**\n * 移除 MCP 端点\n */\n public removeMcpEndpoint(endpoint: string): void {\n if (!endpoint || typeof endpoint !== \"string\") {\n throw new Error(\"MCP 端点必须是非空字符串\");\n }\n\n const config = this.getConfig();\n const currentEndpoints = this.getMcpEndpoints();\n\n // 检查是否存在\n const index = currentEndpoints.indexOf(endpoint);\n if (index === -1) {\n throw new Error(`MCP 端点 ${endpoint} 不存在`);\n }\n\n // 不允许删除最后一个端点\n if (currentEndpoints.length === 1) {\n throw new Error(\"不能删除最后一个 MCP 端点\");\n }\n\n const newEndpoints = currentEndpoints.filter((ep) => ep !== endpoint);\n const newConfig = { ...config, mcpEndpoint: newEndpoints };\n this.saveConfig(newConfig);\n }\n\n /**\n * 更新 MCP 服务配置\n */\n public updateMcpServer(\n serverName: string,\n serverConfig: MCPServerConfig\n ): void {\n if (!serverName || typeof serverName !== \"string\") {\n throw new Error(\"服务名称必须是非空字符串\");\n }\n\n // 验证服务配置\n if (\"type\" in serverConfig && serverConfig.type === \"sse\") {\n // SSE 类型的验证\n if (!serverConfig.url || typeof serverConfig.url !== \"string\") {\n throw new Error(\"SSE 服务配置的 url 字段必须是非空字符串\");\n }\n } else {\n // 本地类型的验证\n const localConfig = serverConfig as LocalMCPServerConfig;\n if (!localConfig.command || typeof localConfig.command !== \"string\") {\n throw new Error(\"服务配置的 command 字段必须是非空字符串\");\n }\n\n if (!Array.isArray(localConfig.args)) {\n throw new Error(\"服务配置的 args 字段必须是数组\");\n }\n\n if (localConfig.env && typeof localConfig.env !== \"object\") {\n throw new Error(\"服务配置的 env 字段必须是对象\");\n }\n }\n\n const config = this.getConfig();\n const newConfig = {\n ...config,\n mcpServers: {\n ...config.mcpServers,\n [serverName]: serverConfig,\n },\n };\n this.saveConfig(newConfig);\n }\n\n /**\n * 删除 MCP 服务配置\n */\n public removeMcpServer(serverName: string): void {\n if (!serverName || typeof serverName !== \"string\") {\n throw new Error(\"服务名称必须是非空字符串\");\n }\n\n const config = this.getConfig();\n if (!config.mcpServers[serverName]) {\n throw new Error(`服务 ${serverName} 不存在`);\n }\n\n const newMcpServers = { ...config.mcpServers };\n delete newMcpServers[serverName];\n\n const newConfig = {\n ...config,\n mcpServers: newMcpServers,\n };\n this.saveConfig(newConfig);\n }\n\n /**\n * 更新服务工具配置\n */\n public updateServerToolsConfig(\n serverName: string,\n toolsConfig: Record<string, MCPToolConfig>\n ): void {\n const config = this.getConfig();\n const newConfig = { ...config };\n\n // 确保 mcpServerConfig 存在\n if (!newConfig.mcpServerConfig) {\n newConfig.mcpServerConfig = {};\n }\n\n // 更新指定服务的工具配置\n newConfig.mcpServerConfig[serverName] = {\n tools: toolsConfig,\n };\n\n this.saveConfig(newConfig);\n }\n\n /**\n * 设置工具启用状态\n */\n public setToolEnabled(\n serverName: string,\n toolName: string,\n enabled: boolean,\n description?: string\n ): void {\n const config = this.getConfig();\n const newConfig = { ...config };\n\n // 确保 mcpServerConfig 存在\n if (!newConfig.mcpServerConfig) {\n newConfig.mcpServerConfig = {};\n }\n\n // 确保服务配置存在\n if (!newConfig.mcpServerConfig[serverName]) {\n newConfig.mcpServerConfig[serverName] = { tools: {} };\n }\n\n // 更新工具配置\n newConfig.mcpServerConfig[serverName].tools[toolName] = {\n enable: enabled,\n ...(description && { description }),\n };\n\n this.saveConfig(newConfig);\n }\n\n /**\n * 保存配置到文件\n */\n private saveConfig(config: AppConfig): void {\n try {\n // 验证配置\n this.validateConfig(config);\n\n // 格式化 JSON 并保存\n const configPath = this.getConfigFilePath();\n const configJson = JSON.stringify(config, null, 2);\n writeFileSync(configPath, configJson, \"utf8\");\n\n // 更新缓存\n this.config = config;\n } catch (error) {\n throw new Error(\n `保存配置失败: ${error instanceof Error ? error.message : String(error)}`\n );\n }\n }\n\n /**\n * 重新加载配置(清除缓存)\n */\n public reloadConfig(): void {\n this.config = null;\n }\n\n /**\n * 获取配置文件路径\n */\n public getConfigPath(): string {\n return this.getConfigFilePath();\n }\n\n /**\n * 获取默认配置文件路径\n */\n public getDefaultConfigPath(): string {\n return this.defaultConfigPath;\n }\n\n /**\n * 获取连接配置(包含默认值)\n */\n public getConnectionConfig(): Required<ConnectionConfig> {\n const config = this.getConfig();\n const connectionConfig = config.connection || {};\n\n return {\n heartbeatInterval:\n connectionConfig.heartbeatInterval ??\n DEFAULT_CONNECTION_CONFIG.heartbeatInterval,\n heartbeatTimeout:\n connectionConfig.heartbeatTimeout ??\n DEFAULT_CONNECTION_CONFIG.heartbeatTimeout,\n reconnectInterval:\n connectionConfig.reconnectInterval ??\n DEFAULT_CONNECTION_CONFIG.reconnectInterval,\n };\n }\n\n /**\n * 获取心跳检测间隔(毫秒)\n */\n public getHeartbeatInterval(): number {\n return this.getConnectionConfig().heartbeatInterval;\n }\n\n /**\n * 获取心跳超时时间(毫秒)\n */\n public getHeartbeatTimeout(): number {\n return this.getConnectionConfig().heartbeatTimeout;\n }\n\n /**\n * 获取重连间隔(毫秒)\n */\n public getReconnectInterval(): number {\n return this.getConnectionConfig().reconnectInterval;\n }\n\n /**\n * 更新连接配置\n */\n public updateConnectionConfig(\n connectionConfig: Partial<ConnectionConfig>\n ): void {\n const config = this.getConfig();\n const currentConnectionConfig = config.connection || {};\n\n const newConnectionConfig = {\n ...currentConnectionConfig,\n ...connectionConfig,\n };\n\n const newConfig = {\n ...config,\n connection: newConnectionConfig,\n };\n\n this.saveConfig(newConfig);\n }\n\n /**\n * 设置心跳检测间隔\n */\n public setHeartbeatInterval(interval: number): void {\n if (interval <= 0) {\n throw new Error(\"心跳检测间隔必须大于0\");\n }\n this.updateConnectionConfig({ heartbeatInterval: interval });\n }\n\n /**\n * 设置心跳超时时间\n */\n public setHeartbeatTimeout(timeout: number): void {\n if (timeout <= 0) {\n throw new Error(\"心跳超时时间必须大于0\");\n }\n this.updateConnectionConfig({ heartbeatTimeout: timeout });\n }\n\n /**\n * 设置重连间隔\n */\n public setReconnectInterval(interval: number): void {\n if (interval <= 0) {\n throw new Error(\"重连间隔必须大于0\");\n }\n this.updateConnectionConfig({ reconnectInterval: interval });\n }\n\n /**\n * 获取 ModelScope 配置\n */\n public getModelScopeConfig(): Readonly<ModelScopeConfig> {\n const config = this.getConfig();\n return config.modelscope || {};\n }\n\n /**\n * 获取 ModelScope API Key\n * 优先从配置文件读取,其次从环境变量读取\n */\n public getModelScopeApiKey(): string | undefined {\n const modelScopeConfig = this.getModelScopeConfig();\n return modelScopeConfig.apiKey || process.env.MODELSCOPE_API_TOKEN;\n }\n\n /**\n * 更新 ModelScope 配置\n */\n public updateModelScopeConfig(\n modelScopeConfig: Partial<ModelScopeConfig>\n ): void {\n const config = this.getConfig();\n const currentModelScopeConfig = config.modelscope || {};\n\n const newModelScopeConfig = {\n ...currentModelScopeConfig,\n ...modelScopeConfig,\n };\n\n const newConfig = {\n ...config,\n modelscope: newModelScopeConfig,\n };\n\n this.saveConfig(newConfig);\n }\n\n /**\n * 设置 ModelScope API Key\n */\n public setModelScopeApiKey(apiKey: string): void {\n if (!apiKey || typeof apiKey !== \"string\") {\n throw new Error(\"API Key 必须是非空字符串\");\n }\n this.updateModelScopeConfig({ apiKey });\n }\n\n /**\n * 获取 Web UI 配置\n */\n public getWebUIConfig(): Readonly<WebUIConfig> {\n const config = this.getConfig();\n return config.webUI || {};\n }\n\n /**\n * 获取 Web UI 端口号\n */\n public getWebUIPort(): number {\n const webUIConfig = this.getWebUIConfig();\n return webUIConfig.port ?? 9999; // 默认端口 9999\n }\n\n /**\n * 更新 Web UI 配置\n */\n public updateWebUIConfig(webUIConfig: Partial<WebUIConfig>): void {\n const config = this.getConfig();\n const currentWebUIConfig = config.webUI || {};\n\n const newWebUIConfig = {\n ...currentWebUIConfig,\n ...webUIConfig,\n };\n\n const newConfig = {\n ...config,\n webUI: newWebUIConfig,\n };\n\n this.saveConfig(newConfig);\n }\n\n /**\n * 设置 Web UI 端口号\n */\n public setWebUIPort(port: number): void {\n if (!Number.isInteger(port) || port <= 0 || port > 65535) {\n throw new Error(\"端口号必须是 1-65535 之间的整数\");\n }\n this.updateWebUIConfig({ port });\n }\n}\n\n// 导出单例实例\nexport const configManager = ConfigManager.getInstance();\n","import fs from \"node:fs\";\nimport path from \"node:path\";\nimport chalk from \"chalk\";\nimport { type consola, createConsola } from \"consola\";\n\nfunction formatDateTime(date: Date) {\n const year = date.getFullYear();\n const month = String(date.getMonth() + 1).padStart(2, \"0\");\n const day = String(date.getDate()).padStart(2, \"0\");\n const hours = String(date.getHours()).padStart(2, \"0\");\n const minutes = String(date.getMinutes()).padStart(2, \"0\");\n const seconds = String(date.getSeconds()).padStart(2, \"0\");\n\n return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`;\n}\n\nexport class Logger {\n private logFilePath: string | null = null;\n private writeStream: fs.WriteStream | null = null;\n private consolaInstance: typeof consola;\n private isDaemonMode: boolean;\n\n constructor() {\n // 检查是否为守护进程模式\n this.isDaemonMode = process.env.XIAOZHI_DAEMON === \"true\";\n // 创建自定义的 consola 实例,禁用图标并自定义格式\n this.consolaInstance = createConsola({\n formatOptions: {\n date: false,\n colors: true,\n compact: true,\n },\n fancy: false,\n });\n\n // 保存对当前实例的引用,以便在闭包中访问\n const isDaemonMode = this.isDaemonMode;\n\n // 自定义格式化器\n this.consolaInstance.setReporters([\n {\n log: (logObj) => {\n const levelMap: Record<string, string> = {\n info: \"INFO\",\n success: \"SUCCESS\",\n warn: \"WARN\",\n error: \"ERROR\",\n debug: \"DEBUG\",\n log: \"LOG\",\n };\n\n const colorMap: Record<string, (text: string) => string> = {\n info: chalk.blue,\n success: chalk.green,\n warn: chalk.yellow,\n error: chalk.red,\n debug: chalk.gray,\n log: (text: string) => text,\n };\n\n const level = levelMap[logObj.type] || logObj.type.toUpperCase();\n const colorFn = colorMap[logObj.type] || ((text: string) => text);\n const timestamp = formatDateTime(new Date());\n\n // 为级别添加颜色\n const coloredLevel = colorFn(`[${level}]`);\n const message = `[${timestamp}] ${coloredLevel} ${logObj.args.join(\n \" \"\n )}`;\n\n // 守护进程模式下不输出到控制台,只写入文件\n if (!isDaemonMode) {\n // 输出到 stderr(与原来保持一致)\n try {\n console.error(message);\n } catch (error) {\n // 忽略 EPIPE 错误\n if (error instanceof Error && error.message?.includes(\"EPIPE\")) {\n return;\n }\n throw error;\n }\n }\n },\n },\n ]);\n }\n\n /**\n * 初始化日志文件\n * @param projectDir 项目目录\n */\n initLogFile(projectDir: string): void {\n this.logFilePath = path.join(projectDir, \"xiaozhi.log\");\n\n // 确保日志文件存在\n if (!fs.existsSync(this.logFilePath)) {\n fs.writeFileSync(this.logFilePath, \"\");\n }\n\n // 创建写入流,追加模式\n this.writeStream = fs.createWriteStream(this.logFilePath, {\n flags: \"a\",\n encoding: \"utf8\",\n });\n }\n\n /**\n * 记录日志到文件\n * @param level 日志级别\n * @param message 日志消息\n * @param args 额外参数\n */\n private logToFile(level: string, message: string, ...args: any[]): void {\n if (this.writeStream) {\n const timestamp = new Date().toISOString();\n const formattedMessage = `[${timestamp}] [${level.toUpperCase()}] ${message}`;\n const fullMessage =\n args.length > 0\n ? `${formattedMessage} ${args\n .map((arg) =>\n typeof arg === \"object\" ? JSON.stringify(arg) : String(arg)\n )\n .join(\" \")}`\n : formattedMessage;\n\n this.writeStream.write(`${fullMessage}\\n`);\n }\n }\n\n /**\n * 设置是否启用文件日志\n * @param enable 是否启用\n */\n enableFileLogging(enable: boolean): void {\n if (enable && !this.writeStream && this.logFilePath) {\n this.writeStream = fs.createWriteStream(this.logFilePath, {\n flags: \"a\",\n encoding: \"utf8\",\n });\n } else if (!enable && this.writeStream) {\n this.writeStream.end();\n this.writeStream = null;\n }\n }\n\n /**\n * 日志方法\n */\n info(message: string, ...args: any[]): void {\n this.consolaInstance.info(message, ...args);\n this.logToFile(\"info\", message, ...args);\n }\n\n success(message: string, ...args: any[]): void {\n this.consolaInstance.success(message, ...args);\n this.logToFile(\"success\", message, ...args);\n }\n\n warn(message: string, ...args: any[]): void {\n this.consolaInstance.warn(message, ...args);\n this.logToFile(\"warn\", message, ...args);\n }\n\n error(message: string, ...args: any[]): void {\n this.consolaInstance.error(message, ...args);\n this.logToFile(\"error\", message, ...args);\n }\n\n debug(message: string, ...args: any[]): void {\n this.consolaInstance.debug(message, ...args);\n this.logToFile(\"debug\", message, ...args);\n }\n\n log(message: string, ...args: any[]): void {\n this.consolaInstance.log(message, ...args);\n this.logToFile(\"log\", message, ...args);\n }\n\n /**\n * 创建一个带标签的日志实例(已废弃,直接返回原实例)\n * @param tag 标签(不再使用)\n * @deprecated 标签功能已移除\n */\n withTag(tag: string): Logger {\n // 不再添加标签,直接返回共享实例\n return this;\n }\n\n /**\n * 关闭日志文件流\n */\n close(): void {\n if (this.writeStream) {\n this.writeStream.end();\n this.writeStream = null;\n }\n }\n}\n\n// 导出单例实例\nexport const logger = new Logger();\n","import { Client } from \"@modelcontextprotocol/sdk/client/index.js\";\nimport { SSEClientTransport } from \"@modelcontextprotocol/sdk/client/sse.js\";\nimport { EventSource } from \"eventsource\";\nimport { type SSEMCPServerConfig, configManager } from \"./configManager\";\nimport { logger as globalLogger } from \"./logger\";\nimport type { IMCPClient } from \"./mcpServerProxy\";\n\n// 全局 polyfill EventSource\n(global as any).EventSource = EventSource;\n\n// 为 ModelScope MCP 创建带标签的 logger\nconst logger = globalLogger.withTag(\"ModelScopeMCP\");\n\ninterface Tool {\n name: string;\n description?: string;\n inputSchema?: any;\n}\n\n/**\n * ModelScope MCP Client for SSE connections\n */\nexport class ModelScopeMCPClient implements IMCPClient {\n private name: string;\n private config: SSEMCPServerConfig;\n private client: Client | null = null;\n private transport: SSEClientTransport | null = null;\n public initialized = false;\n public tools: Tool[] = [];\n public originalTools: Tool[] = [];\n\n constructor(name: string, config: SSEMCPServerConfig) {\n this.name = name;\n this.config = config;\n }\n\n /**\n * 生成带前缀的工具名称\n */\n private generatePrefixedToolName(originalToolName: string): string {\n const normalizedServerName = this.name.replace(/-/g, \"_\");\n return `${normalizedServerName}_xzcli_${originalToolName}`;\n }\n\n /**\n * 根据前缀工具名称获取原始工具名称\n */\n getOriginalToolName(prefixedToolName: string): string | null {\n const normalizedServerName = this.name.replace(/-/g, \"_\");\n const prefix = `${normalizedServerName}_xzcli_`;\n\n if (prefixedToolName.startsWith(prefix)) {\n return prefixedToolName.substring(prefix.length);\n }\n\n return null;\n }\n\n async start() {\n logger.info(`正在启动 ModelScope MCP 客户端:${this.name}`);\n\n try {\n // 从配置或环境变量获取 API Token\n const token = configManager.getModelScopeApiKey();\n if (!token || token === \"\") {\n throw new Error(\n \"未设置 ModelScope API Key。请在配置文件中设置 modelscope.apiKey 或设置 MODELSCOPE_API_TOKEN 环境变量。\"\n );\n }\n\n // 创建 SSE 传输层\n const sseOptions = {\n eventSourceInit: {\n fetch: async (url: string | URL | Request, init?: RequestInit) => {\n // 添加认证头\n const headers = {\n ...init?.headers,\n Authorization: `Bearer ${token}`,\n };\n\n return fetch(url, { ...init, headers });\n },\n },\n requestInit: {\n headers: {\n Authorization: `Bearer ${token}`,\n },\n },\n };\n\n this.transport = new SSEClientTransport(\n new URL(this.config.url),\n sseOptions\n );\n\n // 创建 MCP 客户端\n this.client = new Client(\n {\n name: \"xiaozhi-modelscope-client\",\n version: \"1.0.0\",\n },\n {\n capabilities: {},\n }\n );\n\n // 连接到服务器\n logger.info(`正在连接到 ${this.config.url}`);\n await this.client.connect(this.transport);\n logger.info(`成功连接到 ModelScope MCP 服务器:${this.name}`);\n\n // 获取工具列表\n await this.refreshTools();\n\n this.initialized = true;\n logger.info(\n `${this.name} ModelScope 客户端已就绪,共 ${this.tools.length} 个工具`\n );\n } catch (error) {\n logger.error(\n `启动 ModelScope MCP 客户端 ${this.name} 失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n throw error;\n }\n }\n\n async refreshTools(): Promise<void> {\n try {\n if (!this.client) {\n throw new Error(\"客户端未初始化\");\n }\n\n const result = await this.client.listTools();\n this.originalTools = result.tools || [];\n\n // 为每个工具生成带前缀的名称\n this.tools = this.originalTools.map((tool) => ({\n ...tool,\n name: this.generatePrefixedToolName(tool.name),\n }));\n\n logger.info(\n `${this.name} 加载了 ${this.originalTools.length} 个工具:${this.originalTools\n .map((t) => t.name)\n .join(\", \")}`\n );\n } catch (error) {\n logger.error(\n `从 ${this.name} 获取工具失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n this.tools = [];\n this.originalTools = [];\n }\n }\n\n async callTool(prefixedName: string, arguments_: any): Promise<any> {\n try {\n if (!this.client) {\n throw new Error(\"客户端未初始化\");\n }\n\n // 将前缀名称转换回原始名称\n const originalName = this.getOriginalToolName(prefixedName);\n if (!originalName) {\n throw new Error(`无效的工具名称格式:${prefixedName}`);\n }\n\n logger.info(\n `调用 ModelScope 工具 ${originalName},参数:${JSON.stringify(\n arguments_\n )}`\n );\n\n const result = await this.client.callTool({\n name: originalName,\n arguments: arguments_,\n });\n\n logger.info(\n `ModelScope 工具调用返回: ${JSON.stringify(result).substring(0, 500)}...`\n );\n\n return result;\n } catch (error) {\n logger.error(\n `在 ${this.name} 上调用工具 ${prefixedName} 失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n throw error;\n }\n }\n\n async stop(): Promise<void> {\n if (this.client) {\n logger.info(`正在停止 ${this.name} ModelScope 客户端`);\n try {\n await this.client.close();\n } catch (error) {\n logger.error(\n `关闭客户端时出错:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n }\n this.client = null;\n this.transport = null;\n }\n this.initialized = false;\n }\n}\n","import {\n type StreamableHTTPMCPServerConfig,\n configManager,\n} from \"./configManager\";\nimport { logger as globalLogger } from \"./logger\";\nimport type { IMCPClient } from \"./mcpServerProxy\";\n\n// 为 StreamableHTTP MCP 创建带标签的 logger\nconst logger = globalLogger.withTag(\"StreamableHTTPMCP\");\n\ninterface Tool {\n name: string;\n description?: string;\n inputSchema?: any;\n}\n\ninterface JSONRPCRequest {\n jsonrpc: \"2.0\";\n method: string;\n params?: any;\n id: number | string;\n}\n\ninterface JSONRPCResponse {\n jsonrpc: \"2.0\";\n result?: any;\n error?: {\n code: number;\n message: string;\n data?: any;\n };\n id: number | string;\n}\n\n/**\n * Streamable HTTP MCP Client\n * 用于连接基于 HTTP 的 MCP 服务,如高德地图 MCP\n */\nexport class StreamableHTTPMCPClient implements IMCPClient {\n private name: string;\n private config: StreamableHTTPMCPServerConfig;\n private requestId = 1;\n public initialized = false;\n public tools: Tool[] = [];\n public originalTools: Tool[] = [];\n\n constructor(name: string, config: StreamableHTTPMCPServerConfig) {\n this.name = name;\n this.config = config;\n }\n\n /**\n * 生成带前缀的工具名称\n */\n private generatePrefixedToolName(originalToolName: string): string {\n const normalizedServerName = this.name.replace(/-/g, \"_\");\n return `${normalizedServerName}_xzcli_${originalToolName}`;\n }\n\n /**\n * 根据前缀工具名称获取原始工具名称\n */\n getOriginalToolName(prefixedToolName: string): string | null {\n const normalizedServerName = this.name.replace(/-/g, \"_\");\n const prefix = `${normalizedServerName}_xzcli_`;\n\n if (prefixedToolName.startsWith(prefix)) {\n return prefixedToolName.substring(prefix.length);\n }\n\n return null;\n }\n\n /**\n * 发送 JSON-RPC 请求到 HTTP 端点\n */\n private async sendRequest(method: string, params?: any): Promise<any> {\n const request: JSONRPCRequest = {\n jsonrpc: \"2.0\",\n method,\n params,\n id: this.requestId++,\n };\n\n logger.debug(`发送请求到 ${this.name}: ${JSON.stringify(request)}`);\n\n try {\n // 动态导入 node-fetch\n const fetch = (await import(\"node-fetch\")).default;\n\n const response = await fetch(this.config.url, {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/json\",\n Accept: \"application/json, text/event-stream\",\n },\n body: JSON.stringify(request),\n });\n\n if (!response.ok) {\n throw new Error(`HTTP error! status: ${response.status}`);\n }\n\n const result = (await response.json()) as JSONRPCResponse;\n\n if (result.error) {\n throw new Error(\n `JSON-RPC error: ${result.error.message} (code: ${result.error.code})`\n );\n }\n\n return result.result;\n } catch (error) {\n logger.error(\n `请求失败 (${method}): ${\n error instanceof Error ? error.message : String(error)\n }`\n );\n throw error;\n }\n }\n\n /**\n * 发送 JSON-RPC 通知到 HTTP 端点(不包含 id 字段,不期望响应)\n */\n private async sendNotification(method: string, params?: any): Promise<void> {\n const notification = {\n jsonrpc: \"2.0\",\n method,\n params,\n // 注意:通知不包含 id 字段\n };\n\n logger.debug(`发送通知到 ${this.name}: ${JSON.stringify(notification)}`);\n\n try {\n // 动态导入 node-fetch\n const fetch = (await import(\"node-fetch\")).default;\n\n const response = await fetch(this.config.url, {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/json\",\n Accept: \"application/json, text/event-stream\",\n },\n body: JSON.stringify(notification),\n });\n\n if (!response.ok) {\n throw new Error(`HTTP error! status: ${response.status}`);\n }\n\n // 对于通知,我们不期望有意义的响应,只要没有HTTP错误就认为成功\n logger.debug(`通知 ${method} 发送成功`);\n } catch (error) {\n logger.debug(\n `通知发送失败 (${method}): ${\n error instanceof Error ? error.message : String(error)\n }`\n );\n throw error;\n }\n }\n\n async start() {\n logger.info(`正在启动 Streamable HTTP MCP 客户端:${this.name}`);\n\n try {\n // 初始化连接\n await this.sendRequest(\"initialize\", {\n protocolVersion: \"2024-11-05\",\n capabilities: {\n tools: {},\n },\n clientInfo: {\n name: \"xiaozhi-streamable-http-client\",\n version: \"1.0.0\",\n },\n });\n\n // 发送初始化完成通知(作为通知发送,不是请求)\n try {\n await this.sendNotification(\"notifications/initialized\");\n } catch (error) {\n // 忽略此错误,因为某些服务(如高德地图)不支持此通知\n logger.debug(`${this.name} 不支持 notifications/initialized: ${error}`);\n }\n\n // 获取工具列表\n const listToolsResult = await this.sendRequest(\"tools/list\");\n\n if (listToolsResult?.tools) {\n this.originalTools = listToolsResult.tools;\n\n // 生成带前缀的工具名称\n this.tools = this.originalTools\n .filter((tool) => configManager.isToolEnabled(this.name, tool.name))\n .map((tool) => ({\n ...tool,\n name: this.generatePrefixedToolName(tool.name),\n }));\n\n logger.info(\n `Streamable HTTP MCP 客户端 ${this.name} 已初始化,包含 ${this.tools.length}/${this.originalTools.length} 个已启用的工具`\n );\n }\n\n this.initialized = true;\n } catch (error) {\n logger.error(\n `启动 Streamable HTTP MCP 客户端 ${this.name} 失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n throw error;\n }\n }\n\n async refreshTools() {\n try {\n const listToolsResult = await this.sendRequest(\"tools/list\");\n\n if (listToolsResult?.tools) {\n this.originalTools = listToolsResult.tools;\n\n // 重新生成带前缀的工具名称\n this.tools = this.originalTools\n .filter((tool) => configManager.isToolEnabled(this.name, tool.name))\n .map((tool) => ({\n ...tool,\n name: this.generatePrefixedToolName(tool.name),\n }));\n\n logger.info(\n `已刷新 ${this.name} 的工具列表,包含 ${this.tools.length}/${this.originalTools.length} 个已启用的工具`\n );\n }\n } catch (error) {\n logger.error(\n `刷新 ${this.name} 的工具列表失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n }\n }\n\n async callTool(prefixedName: string, arguments_: any): Promise<any> {\n try {\n // 将前缀名称转换回原始名称\n const originalName = this.getOriginalToolName(prefixedName);\n if (!originalName) {\n throw new Error(`无效的工具名称格式:${prefixedName}`);\n }\n\n const result = await this.sendRequest(\"tools/call\", {\n name: originalName,\n arguments: arguments_,\n });\n\n return result;\n } catch (error) {\n logger.error(\n `在 ${this.name} 上调用工具 ${prefixedName} 失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n throw error;\n }\n }\n\n async stop() {\n logger.info(`正在停止 ${this.name} 客户端`);\n this.initialized = false;\n }\n}\n"],"mappings":";+EAQA,OAA4B,SAAAA,MAAa,gBACzC,OAAS,gBAAAC,MAAoB,KAC7B,OAAS,WAAAC,EAAS,WAAAC,MAAe,OACjC,OAAOC,MAAa,UACpB,OAAS,iBAAAC,MAAqB,MCZ9B,OAAS,gBAAAC,EAAc,cAAAC,EAAY,gBAAAC,EAAc,iBAAAC,MAAqB,KACtE,OAAS,WAAAC,EAAS,WAAAC,MAAe,OACjC,OAAS,iBAAAC,MAAqB,MAG9B,IAAMC,EAAYC,EAAQC,EAAc,YAAY,GAAG,CAAC,EAGlDC,EAAwD,CAC5D,kBAAmB,IACnB,iBAAkB,IAClB,kBAAmB,GACrB,EAiEaC,EAAN,MAAMC,CAAc,CA7E3B,MA6E2B,CAAAC,EAAA,sBACzB,OAAe,SACP,kBACA,OAA2B,KAE3B,aAAc,CACpB,KAAK,kBAAoBC,EAAQP,EAAW,6BAA6B,CAC3E,CAKQ,mBAA4B,CAElC,IAAMQ,EAAY,QAAQ,IAAI,oBAAsB,QAAQ,IAAI,EAChE,OAAOD,EAAQC,EAAW,qBAAqB,CACjD,CAKA,OAAc,aAA6B,CACzC,OAAKH,EAAc,WACjBA,EAAc,SAAW,IAAIA,GAExBA,EAAc,QACvB,CAKO,cAAwB,CAC7B,IAAMI,EAAa,KAAK,kBAAkB,EAC1C,OAAOC,EAAWD,CAAU,CAC9B,CAMO,YAAmB,CACxB,GAAI,CAACC,EAAW,KAAK,iBAAiB,EACpC,MAAM,IAAI,MAAM,qFAAwC,EAG1D,GAAI,KAAK,aAAa,EACpB,MAAM,IAAI,MAAM,iHAAsC,EAGxD,IAAMD,EAAa,KAAK,kBAAkB,EAC1CE,EAAa,KAAK,kBAAmBF,CAAU,EAC/C,KAAK,OAAS,IAChB,CAKQ,YAAwB,CAC9B,GAAI,CAAC,KAAK,aAAa,EACrB,MAAM,IAAI,MACR,2IACF,EAGF,GAAI,CACF,IAAMA,EAAa,KAAK,kBAAkB,EACpCG,EAAaC,EAAaJ,EAAY,MAAM,EAC5CK,EAAS,KAAK,MAAMF,CAAU,EAGpC,YAAK,eAAeE,CAAM,EAEnBA,CACT,OAASC,EAAO,CACd,MAAIA,aAAiB,YACb,IAAI,MAAM,qDAAaA,EAAM,OAAO,EAAE,EAExCA,CACR,CACF,CAKQ,eAAeD,EAAuB,CAC5C,GAAI,CAACA,GAAU,OAAOA,GAAW,SAC/B,MAAM,IAAI,MAAM,sFAAgB,EAGlC,IAAME,EAAYF,EAElB,GAAIE,EAAU,cAAgB,QAAaA,EAAU,cAAgB,KACnE,MAAM,IAAI,MAAM,4FAA2B,EAI7C,GAAI,OAAOA,EAAU,aAAgB,SAE9B,GAAI,MAAM,QAAQA,EAAU,WAAW,EAAG,CAC/C,GAAIA,EAAU,YAAY,SAAW,EACnC,MAAM,IAAI,MAAM,wGAA6B,EAE/C,QAAWC,KAAYD,EAAU,YAC/B,GAAI,OAAOC,GAAa,UAAYA,EAAS,KAAK,IAAM,GACtD,MAAM,IAAI,MACR,oKACF,CAGN,KACE,OAAM,IAAI,MAAM,4IAAmC,EAGrD,GAAI,CAACD,EAAU,YAAc,OAAOA,EAAU,YAAe,SAC3D,MAAM,IAAI,MAAM,2FAA0B,EAI5C,OAAW,CAACE,EAAYC,CAAY,IAAK,OAAO,QAC9CH,EAAU,UACZ,EAAG,CACD,GAAI,CAACG,GAAgB,OAAOA,GAAiB,SAC3C,MAAM,IAAI,MAAM,oEAAuBD,CAAU,eAAK,EAGxD,IAAME,EAAKD,EAGX,GAAIC,EAAG,KAAO,OAAOA,EAAG,KAAQ,UAG9B,GAAIA,EAAG,MAAQA,EAAG,OAAS,OAASA,EAAG,OAAS,kBAC9C,MAAM,IAAI,MACR,oEAAuBF,CAAU,yDACnC,MAEG,CAEL,GAAI,CAACE,EAAG,SAAW,OAAOA,EAAG,SAAY,SACvC,MAAM,IAAI,MACR,oEAAuBF,CAAU,uBACnC,EAGF,GAAI,CAAC,MAAM,QAAQE,EAAG,IAAI,EACxB,MAAM,IAAI,MACR,oEAAuBF,CAAU,sCACnC,EAGF,GAAIE,EAAG,KAAO,OAAOA,EAAG,KAAQ,SAC9B,MAAM,IAAI,MACR,oEAAuBF,CAAU,qCACnC,CAEJ,CACF,CACF,CAKO,WAAiC,CACtC,OAAK,KAAK,SACR,KAAK,OAAS,KAAK,WAAW,GAIzB,KAAK,MAAM,KAAK,UAAU,KAAK,MAAM,CAAC,CAC/C,CAMO,gBAAyB,CAC9B,IAAMJ,EAAS,KAAK,UAAU,EAC9B,OAAI,MAAM,QAAQA,EAAO,WAAW,EAC3BA,EAAO,YAAY,CAAC,GAAK,GAE3BA,EAAO,WAChB,CAKO,iBAA4B,CACjC,IAAMA,EAAS,KAAK,UAAU,EAC9B,OAAI,MAAM,QAAQA,EAAO,WAAW,EAC3B,CAAC,GAAGA,EAAO,WAAW,EAExBA,EAAO,YAAc,CAACA,EAAO,WAAW,EAAI,CAAC,CACtD,CAKO,eAA2D,CAEhE,OADe,KAAK,UAAU,EAChB,UAChB,CAKO,oBAAqE,CAE1E,OADe,KAAK,UAAU,EAChB,iBAAmB,CAAC,CACpC,CAKO,qBACLI,EACyC,CAEzC,OADqB,KAAK,mBAAmB,EACzBA,CAAU,GAAG,OAAS,CAAC,CAC7C,CAKO,cAAcA,EAAoBG,EAA2B,CAGlE,OAFoB,KAAK,qBAAqBH,CAAU,EACzBG,CAAQ,GACpB,SAAW,EAChC,CAKO,kBAAkBJ,EAAmC,CAC1D,GAAI,MAAM,QAAQA,CAAQ,EAAG,CAC3B,GAAIA,EAAS,SAAW,EACtB,MAAM,IAAI,MAAM,sDAAc,EAEhC,QAAWK,KAAML,EACf,GAAI,CAACK,GAAM,OAAOA,GAAO,SACvB,MAAM,IAAI,MAAM,kHAAwB,CAG9C,SACM,CAACL,GAAY,OAAOA,GAAa,SACnC,MAAM,IAAI,MAAM,kEAAgB,EAKpC,IAAMM,EAAY,CAAE,GADL,KAAK,UAAU,EACC,YAAaN,CAAS,EACrD,KAAK,WAAWM,CAAS,CAC3B,CAKO,eAAeN,EAAwB,CAC5C,GAAI,CAACA,GAAY,OAAOA,GAAa,SACnC,MAAM,IAAI,MAAM,kEAAgB,EAGlC,IAAMH,EAAS,KAAK,UAAU,EACxBU,EAAmB,KAAK,gBAAgB,EAG9C,GAAIA,EAAiB,SAASP,CAAQ,EACpC,MAAM,IAAI,MAAM,oBAAUA,CAAQ,qBAAM,EAG1C,IAAMQ,EAAe,CAAC,GAAGD,EAAkBP,CAAQ,EAC7CM,EAAY,CAAE,GAAGT,EAAQ,YAAaW,CAAa,EACzD,KAAK,WAAWF,CAAS,CAC3B,CAKO,kBAAkBN,EAAwB,CAC/C,GAAI,CAACA,GAAY,OAAOA,GAAa,SACnC,MAAM,IAAI,MAAM,kEAAgB,EAGlC,IAAMH,EAAS,KAAK,UAAU,EACxBU,EAAmB,KAAK,gBAAgB,EAI9C,GADcA,EAAiB,QAAQP,CAAQ,IACjC,GACZ,MAAM,IAAI,MAAM,oBAAUA,CAAQ,qBAAM,EAI1C,GAAIO,EAAiB,SAAW,EAC9B,MAAM,IAAI,MAAM,mEAAiB,EAGnC,IAAMC,EAAeD,EAAiB,OAAQF,GAAOA,IAAOL,CAAQ,EAC9DM,EAAY,CAAE,GAAGT,EAAQ,YAAaW,CAAa,EACzD,KAAK,WAAWF,CAAS,CAC3B,CAKO,gBACLL,EACAC,EACM,CACN,GAAI,CAACD,GAAc,OAAOA,GAAe,SACvC,MAAM,IAAI,MAAM,0EAAc,EAIhC,GAAI,SAAUC,GAAgBA,EAAa,OAAS,OAElD,GAAI,CAACA,EAAa,KAAO,OAAOA,EAAa,KAAQ,SACnD,MAAM,IAAI,MAAM,qGAA0B,MAEvC,CAEL,IAAMO,EAAcP,EACpB,GAAI,CAACO,EAAY,SAAW,OAAOA,EAAY,SAAY,SACzD,MAAM,IAAI,MAAM,qGAA0B,EAG5C,GAAI,CAAC,MAAM,QAAQA,EAAY,IAAI,EACjC,MAAM,IAAI,MAAM,gFAAoB,EAGtC,GAAIA,EAAY,KAAO,OAAOA,EAAY,KAAQ,SAChD,MAAM,IAAI,MAAM,+EAAmB,CAEvC,CAEA,IAAMZ,EAAS,KAAK,UAAU,EACxBS,EAAY,CAChB,GAAGT,EACH,WAAY,CACV,GAAGA,EAAO,WACV,CAACI,CAAU,EAAGC,CAChB,CACF,EACA,KAAK,WAAWI,CAAS,CAC3B,CAKO,gBAAgBL,EAA0B,CAC/C,GAAI,CAACA,GAAc,OAAOA,GAAe,SACvC,MAAM,IAAI,MAAM,0EAAc,EAGhC,IAAMJ,EAAS,KAAK,UAAU,EAC9B,GAAI,CAACA,EAAO,WAAWI,CAAU,EAC/B,MAAM,IAAI,MAAM,gBAAMA,CAAU,qBAAM,EAGxC,IAAMS,EAAgB,CAAE,GAAGb,EAAO,UAAW,EAC7C,OAAOa,EAAcT,CAAU,EAE/B,IAAMK,EAAY,CAChB,GAAGT,EACH,WAAYa,CACd,EACA,KAAK,WAAWJ,CAAS,CAC3B,CAKO,wBACLL,EACAU,EACM,CAEN,IAAML,EAAY,CAAE,GADL,KAAK,UAAU,CACA,EAGzBA,EAAU,kBACbA,EAAU,gBAAkB,CAAC,GAI/BA,EAAU,gBAAgBL,CAAU,EAAI,CACtC,MAAOU,CACT,EAEA,KAAK,WAAWL,CAAS,CAC3B,CAKO,eACLL,EACAG,EACAQ,EACAC,EACM,CAEN,IAAMP,EAAY,CAAE,GADL,KAAK,UAAU,CACA,EAGzBA,EAAU,kBACbA,EAAU,gBAAkB,CAAC,GAI1BA,EAAU,gBAAgBL,CAAU,IACvCK,EAAU,gBAAgBL,CAAU,EAAI,CAAE,MAAO,CAAC,CAAE,GAItDK,EAAU,gBAAgBL,CAAU,EAAE,MAAMG,CAAQ,EAAI,CACtD,OAAQQ,EACR,GAAIC,GAAe,CAAE,YAAAA,CAAY,CACnC,EAEA,KAAK,WAAWP,CAAS,CAC3B,CAKQ,WAAWT,EAAyB,CAC1C,GAAI,CAEF,KAAK,eAAeA,CAAM,EAG1B,IAAML,EAAa,KAAK,kBAAkB,EACpCsB,EAAa,KAAK,UAAUjB,EAAQ,KAAM,CAAC,EACjDkB,EAAcvB,EAAYsB,EAAY,MAAM,EAG5C,KAAK,OAASjB,CAChB,OAASC,EAAO,CACd,MAAM,IAAI,MACR,yCAAWA,aAAiB,MAAQA,EAAM,QAAU,OAAOA,CAAK,CAAC,EACnE,CACF,CACF,CAKO,cAAqB,CAC1B,KAAK,OAAS,IAChB,CAKO,eAAwB,CAC7B,OAAO,KAAK,kBAAkB,CAChC,CAKO,sBAA+B,CACpC,OAAO,KAAK,iBACd,CAKO,qBAAkD,CAEvD,IAAMkB,EADS,KAAK,UAAU,EACE,YAAc,CAAC,EAE/C,MAAO,CACL,kBACEA,EAAiB,mBACjB9B,EAA0B,kBAC5B,iBACE8B,EAAiB,kBACjB9B,EAA0B,iBAC5B,kBACE8B,EAAiB,mBACjB9B,EAA0B,iBAC9B,CACF,CAKO,sBAA+B,CACpC,OAAO,KAAK,oBAAoB,EAAE,iBACpC,CAKO,qBAA8B,CACnC,OAAO,KAAK,oBAAoB,EAAE,gBACpC,CAKO,sBAA+B,CACpC,OAAO,KAAK,oBAAoB,EAAE,iBACpC,CAKO,uBACL8B,EACM,CACN,IAAMnB,EAAS,KAAK,UAAU,EAGxBoB,EAAsB,CAC1B,GAH8BpB,EAAO,YAAc,CAAC,EAIpD,GAAGmB,CACL,EAEMV,EAAY,CAChB,GAAGT,EACH,WAAYoB,CACd,EAEA,KAAK,WAAWX,CAAS,CAC3B,CAKO,qBAAqBY,EAAwB,CAClD,GAAIA,GAAY,EACd,MAAM,IAAI,MAAM,+DAAa,EAE/B,KAAK,uBAAuB,CAAE,kBAAmBA,CAAS,CAAC,CAC7D,CAKO,oBAAoBC,EAAuB,CAChD,GAAIA,GAAW,EACb,MAAM,IAAI,MAAM,+DAAa,EAE/B,KAAK,uBAAuB,CAAE,iBAAkBA,CAAQ,CAAC,CAC3D,CAKO,qBAAqBD,EAAwB,CAClD,GAAIA,GAAY,EACd,MAAM,IAAI,MAAM,mDAAW,EAE7B,KAAK,uBAAuB,CAAE,kBAAmBA,CAAS,CAAC,CAC7D,CAKO,qBAAkD,CAEvD,OADe,KAAK,UAAU,EAChB,YAAc,CAAC,CAC/B,CAMO,qBAA0C,CAE/C,OADyB,KAAK,oBAAoB,EAC1B,QAAU,QAAQ,IAAI,oBAChD,CAKO,uBACLE,EACM,CACN,IAAMvB,EAAS,KAAK,UAAU,EAGxBwB,EAAsB,CAC1B,GAH8BxB,EAAO,YAAc,CAAC,EAIpD,GAAGuB,CACL,EAEMd,EAAY,CAChB,GAAGT,EACH,WAAYwB,CACd,EAEA,KAAK,WAAWf,CAAS,CAC3B,CAKO,oBAAoBgB,EAAsB,CAC/C,GAAI,CAACA,GAAU,OAAOA,GAAW,SAC/B,MAAM,IAAI,MAAM,0DAAkB,EAEpC,KAAK,uBAAuB,CAAE,OAAAA,CAAO,CAAC,CACxC,CAKO,gBAAwC,CAE7C,OADe,KAAK,UAAU,EAChB,OAAS,CAAC,CAC1B,CAKO,cAAuB,CAE5B,OADoB,KAAK,eAAe,EACrB,MAAQ,IAC7B,CAKO,kBAAkBC,EAAyC,CAChE,IAAM1B,EAAS,KAAK,UAAU,EAGxB2B,EAAiB,CACrB,GAHyB3B,EAAO,OAAS,CAAC,EAI1C,GAAG0B,CACL,EAEMjB,EAAY,CAChB,GAAGT,EACH,MAAO2B,CACT,EAEA,KAAK,WAAWlB,CAAS,CAC3B,CAKO,aAAamB,EAAoB,CACtC,GAAI,CAAC,OAAO,UAAUA,CAAI,GAAKA,GAAQ,GAAKA,EAAO,MACjD,MAAM,IAAI,MAAM,6EAAsB,EAExC,KAAK,kBAAkB,CAAE,KAAAA,CAAK,CAAC,CACjC,CACF,EAGaC,EAAgBvC,EAAc,YAAY,EC5tBvD,OAAOwC,MAAQ,KACf,OAAOC,MAAU,OACjB,OAAOC,MAAW,QAClB,OAAuB,iBAAAC,MAAqB,UAE5C,SAASC,EAAeC,EAAY,CAClC,IAAMC,EAAOD,EAAK,YAAY,EACxBE,EAAQ,OAAOF,EAAK,SAAS,EAAI,CAAC,EAAE,SAAS,EAAG,GAAG,EACnDG,EAAM,OAAOH,EAAK,QAAQ,CAAC,EAAE,SAAS,EAAG,GAAG,EAC5CI,EAAQ,OAAOJ,EAAK,SAAS,CAAC,EAAE,SAAS,EAAG,GAAG,EAC/CK,EAAU,OAAOL,EAAK,WAAW,CAAC,EAAE,SAAS,EAAG,GAAG,EACnDM,EAAU,OAAON,EAAK,WAAW,CAAC,EAAE,SAAS,EAAG,GAAG,EAEzD,MAAO,GAAGC,CAAI,IAAIC,CAAK,IAAIC,CAAG,IAAIC,CAAK,IAAIC,CAAO,IAAIC,CAAO,EAC/D,CATSC,EAAAR,EAAA,kBAWF,IAAMS,EAAN,KAAa,CAhBpB,MAgBoB,CAAAD,EAAA,eACV,YAA6B,KAC7B,YAAqC,KACrC,gBACA,aAER,aAAc,CAEZ,KAAK,aAAe,QAAQ,IAAI,iBAAmB,OAEnD,KAAK,gBAAkBE,EAAc,CACnC,cAAe,CACb,KAAM,GACN,OAAQ,GACR,QAAS,EACX,EACA,MAAO,EACT,CAAC,EAGD,IAAMC,EAAe,KAAK,aAG1B,KAAK,gBAAgB,aAAa,CAChC,CACE,IAAKH,EAACI,GAAW,CACf,IAAMC,EAAmC,CACvC,KAAM,OACN,QAAS,UACT,KAAM,OACN,MAAO,QACP,MAAO,QACP,IAAK,KACP,EAEMC,EAAqD,CACzD,KAAMC,EAAM,KACZ,QAASA,EAAM,MACf,KAAMA,EAAM,OACZ,MAAOA,EAAM,IACb,MAAOA,EAAM,KACb,IAAKP,EAACQ,GAAiBA,EAAlB,MACP,EAEMC,EAAQJ,EAASD,EAAO,IAAI,GAAKA,EAAO,KAAK,YAAY,EACzDM,EAAUJ,EAASF,EAAO,IAAI,IAAOI,GAAiBA,GACtDG,EAAYnB,EAAe,IAAI,IAAM,EAGrCoB,EAAeF,EAAQ,IAAID,CAAK,GAAG,EACnCI,EAAU,IAAIF,CAAS,KAAKC,CAAY,IAAIR,EAAO,KAAK,KAC5D,GACF,CAAC,GAGD,GAAI,CAACD,EAEH,GAAI,CACF,QAAQ,MAAMU,CAAO,CACvB,OAASC,EAAO,CAEd,GAAIA,aAAiB,OAASA,EAAM,SAAS,SAAS,OAAO,EAC3D,OAEF,MAAMA,CACR,CAEJ,EA1CK,MA2CP,CACF,CAAC,CACH,CAMA,YAAYC,EAA0B,CACpC,KAAK,YAAcC,EAAK,KAAKD,EAAY,aAAa,EAGjDE,EAAG,WAAW,KAAK,WAAW,GACjCA,EAAG,cAAc,KAAK,YAAa,EAAE,EAIvC,KAAK,YAAcA,EAAG,kBAAkB,KAAK,YAAa,CACxD,MAAO,IACP,SAAU,MACZ,CAAC,CACH,CAQQ,UAAUR,EAAeI,KAAoBK,EAAmB,CACtE,GAAI,KAAK,YAAa,CAEpB,IAAMC,EAAmB,IADP,IAAI,KAAK,EAAE,YAAY,CACH,MAAMV,EAAM,YAAY,CAAC,KAAKI,CAAO,GACrEO,EACJF,EAAK,OAAS,EACV,GAAGC,CAAgB,IAAID,EACpB,IAAKG,GACJ,OAAOA,GAAQ,SAAW,KAAK,UAAUA,CAAG,EAAI,OAAOA,CAAG,CAC5D,EACC,KAAK,GAAG,CAAC,GACZF,EAEN,KAAK,YAAY,MAAM,GAAGC,CAAW;AAAA,CAAI,CAC3C,CACF,CAMA,kBAAkBE,EAAuB,CACnCA,GAAU,CAAC,KAAK,aAAe,KAAK,YACtC,KAAK,YAAcL,EAAG,kBAAkB,KAAK,YAAa,CACxD,MAAO,IACP,SAAU,MACZ,CAAC,EACQ,CAACK,GAAU,KAAK,cACzB,KAAK,YAAY,IAAI,EACrB,KAAK,YAAc,KAEvB,CAKA,KAAKT,KAAoBK,EAAmB,CAC1C,KAAK,gBAAgB,KAAKL,EAAS,GAAGK,CAAI,EAC1C,KAAK,UAAU,OAAQL,EAAS,GAAGK,CAAI,CACzC,CAEA,QAAQL,KAAoBK,EAAmB,CAC7C,KAAK,gBAAgB,QAAQL,EAAS,GAAGK,CAAI,EAC7C,KAAK,UAAU,UAAWL,EAAS,GAAGK,CAAI,CAC5C,CAEA,KAAKL,KAAoBK,EAAmB,CAC1C,KAAK,gBAAgB,KAAKL,EAAS,GAAGK,CAAI,EAC1C,KAAK,UAAU,OAAQL,EAAS,GAAGK,CAAI,CACzC,CAEA,MAAML,KAAoBK,EAAmB,CAC3C,KAAK,gBAAgB,MAAML,EAAS,GAAGK,CAAI,EAC3C,KAAK,UAAU,QAASL,EAAS,GAAGK,CAAI,CAC1C,CAEA,MAAML,KAAoBK,EAAmB,CAC3C,KAAK,gBAAgB,MAAML,EAAS,GAAGK,CAAI,EAC3C,KAAK,UAAU,QAASL,EAAS,GAAGK,CAAI,CAC1C,CAEA,IAAIL,KAAoBK,EAAmB,CACzC,KAAK,gBAAgB,IAAIL,EAAS,GAAGK,CAAI,EACzC,KAAK,UAAU,MAAOL,EAAS,GAAGK,CAAI,CACxC,CAOA,QAAQK,EAAqB,CAE3B,OAAO,IACT,CAKA,OAAc,CACR,KAAK,cACP,KAAK,YAAY,IAAI,EACrB,KAAK,YAAc,KAEvB,CACF,EAGaC,EAAS,IAAIvB,ECzM1B,OAAS,UAAAwB,MAAc,4CACvB,OAAS,sBAAAC,MAA0B,0CACnC,OAAS,eAAAC,MAAmB,cAM3B,OAAe,YAAcC,EAG9B,IAAMC,EAASA,EAAa,QAAQ,eAAe,EAWtCC,EAAN,KAAgD,CAtBvD,MAsBuD,CAAAC,EAAA,4BAC7C,KACA,OACA,OAAwB,KACxB,UAAuC,KACxC,YAAc,GACd,MAAgB,CAAC,EACjB,cAAwB,CAAC,EAEhC,YAAYC,EAAcC,EAA4B,CACpD,KAAK,KAAOD,EACZ,KAAK,OAASC,CAChB,CAKQ,yBAAyBC,EAAkC,CAEjE,MAAO,GADsB,KAAK,KAAK,QAAQ,KAAM,GAAG,CAC1B,UAAUA,CAAgB,EAC1D,CAKA,oBAAoBC,EAAyC,CAE3D,IAAMC,EAAS,GADc,KAAK,KAAK,QAAQ,KAAM,GAAG,CAClB,UAEtC,OAAID,EAAiB,WAAWC,CAAM,EAC7BD,EAAiB,UAAUC,EAAO,MAAM,EAG1C,IACT,CAEA,MAAM,OAAQ,CACZP,EAAO,KAAK,mEAA2B,KAAK,IAAI,EAAE,EAElD,GAAI,CAEF,IAAMQ,EAAQC,EAAc,oBAAoB,EAChD,GAAI,CAACD,GAASA,IAAU,GACtB,MAAM,IAAI,MACR,4LACF,EAIF,IAAME,EAAa,CACjB,gBAAiB,CACf,MAAOR,EAAA,MAAOS,EAA6BC,IAAuB,CAEhE,IAAMC,EAAU,CACd,GAAGD,GAAM,QACT,cAAe,UAAUJ,CAAK,EAChC,EAEA,OAAO,MAAMG,EAAK,CAAE,GAAGC,EAAM,QAAAC,CAAQ,CAAC,CACxC,EARO,QAST,EACA,YAAa,CACX,QAAS,CACP,cAAe,UAAUL,CAAK,EAChC,CACF,CACF,EAEA,KAAK,UAAY,IAAIM,EACnB,IAAI,IAAI,KAAK,OAAO,GAAG,EACvBJ,CACF,EAGA,KAAK,OAAS,IAAIK,EAChB,CACE,KAAM,4BACN,QAAS,OACX,EACA,CACE,aAAc,CAAC,CACjB,CACF,EAGAf,EAAO,KAAK,kCAAS,KAAK,OAAO,GAAG,EAAE,EACtC,MAAM,KAAK,OAAO,QAAQ,KAAK,SAAS,EACxCA,EAAO,KAAK,yEAA4B,KAAK,IAAI,EAAE,EAGnD,MAAM,KAAK,aAAa,EAExB,KAAK,YAAc,GACnBA,EAAO,KACL,GAAG,KAAK,IAAI,gEAAwB,KAAK,MAAM,MAAM,qBACvD,CACF,OAASgB,EAAO,CACd,MAAAhB,EAAO,MACL,kDAAyB,KAAK,IAAI,sBAChCgB,aAAiB,MAAQA,EAAM,QAAU,OAAOA,CAAK,CACvD,EACF,EACMA,CACR,CACF,CAEA,MAAM,cAA8B,CAClC,GAAI,CACF,GAAI,CAAC,KAAK,OACR,MAAM,IAAI,MAAM,4CAAS,EAG3B,IAAMC,EAAS,MAAM,KAAK,OAAO,UAAU,EAC3C,KAAK,cAAgBA,EAAO,OAAS,CAAC,EAGtC,KAAK,MAAQ,KAAK,cAAc,IAAKC,IAAU,CAC7C,GAAGA,EACH,KAAM,KAAK,yBAAyBA,EAAK,IAAI,CAC/C,EAAE,EAEFlB,EAAO,KACL,GAAG,KAAK,IAAI,uBAAQ,KAAK,cAAc,MAAM,4BAAQ,KAAK,cACvD,IAAK,GAAM,EAAE,IAAI,EACjB,KAAK,IAAI,CAAC,EACf,CACF,OAASgB,EAAO,CACdhB,EAAO,MACL,UAAK,KAAK,IAAI,8CACZgB,aAAiB,MAAQA,EAAM,QAAU,OAAOA,CAAK,CACvD,EACF,EACA,KAAK,MAAQ,CAAC,EACd,KAAK,cAAgB,CAAC,CACxB,CACF,CAEA,MAAM,SAASG,EAAsBC,EAA+B,CAClE,GAAI,CACF,GAAI,CAAC,KAAK,OACR,MAAM,IAAI,MAAM,4CAAS,EAI3B,IAAMC,EAAe,KAAK,oBAAoBF,CAAY,EAC1D,GAAI,CAACE,EACH,MAAM,IAAI,MAAM,+DAAaF,CAAY,EAAE,EAG7CnB,EAAO,KACL,wCAAoBqB,CAAY,2BAAO,KAAK,UAC1CD,CACF,CAAC,EACH,EAEA,IAAMH,EAAS,MAAM,KAAK,OAAO,SAAS,CACxC,KAAMI,EACN,UAAWD,CACb,CAAC,EAED,OAAApB,EAAO,KACL,oDAAsB,KAAK,UAAUiB,CAAM,EAAE,UAAU,EAAG,GAAG,CAAC,KAChE,EAEOA,CACT,OAASD,EAAO,CACd,MAAAhB,EAAO,MACL,UAAK,KAAK,IAAI,mCAAUmB,CAAY,sBAClCH,aAAiB,MAAQA,EAAM,QAAU,OAAOA,CAAK,CACvD,EACF,EACMA,CACR,CACF,CAEA,MAAM,MAAsB,CAC1B,GAAI,KAAK,OAAQ,CACfhB,EAAO,KAAK,4BAAQ,KAAK,IAAI,gCAAiB,EAC9C,GAAI,CACF,MAAM,KAAK,OAAO,MAAM,CAC1B,OAASgB,EAAO,CACdhB,EAAO,MACL,yDACEgB,aAAiB,MAAQA,EAAM,QAAU,OAAOA,CAAK,CACvD,EACF,CACF,CACA,KAAK,OAAS,KACd,KAAK,UAAY,IACnB,CACA,KAAK,YAAc,EACrB,CACF,EC9MA,IAAMM,EAASA,EAAa,QAAQ,mBAAmB,EA8B1CC,EAAN,KAAoD,CAtC3D,MAsC2D,CAAAC,EAAA,gCACjD,KACA,OACA,UAAY,EACb,YAAc,GACd,MAAgB,CAAC,EACjB,cAAwB,CAAC,EAEhC,YAAYC,EAAcC,EAAuC,CAC/D,KAAK,KAAOD,EACZ,KAAK,OAASC,CAChB,CAKQ,yBAAyBC,EAAkC,CAEjE,MAAO,GADsB,KAAK,KAAK,QAAQ,KAAM,GAAG,CAC1B,UAAUA,CAAgB,EAC1D,CAKA,oBAAoBC,EAAyC,CAE3D,IAAMC,EAAS,GADc,KAAK,KAAK,QAAQ,KAAM,GAAG,CAClB,UAEtC,OAAID,EAAiB,WAAWC,CAAM,EAC7BD,EAAiB,UAAUC,EAAO,MAAM,EAG1C,IACT,CAKA,MAAc,YAAYC,EAAgBC,EAA4B,CACpE,IAAMC,EAA0B,CAC9B,QAAS,MACT,OAAAF,EACA,OAAAC,EACA,GAAI,KAAK,WACX,EAEAT,EAAO,MAAM,kCAAS,KAAK,IAAI,KAAK,KAAK,UAAUU,CAAO,CAAC,EAAE,EAE7D,GAAI,CAEF,IAAMC,GAAS,KAAM,QAAO,YAAY,GAAG,QAErCC,EAAW,MAAMD,EAAM,KAAK,OAAO,IAAK,CAC5C,OAAQ,OACR,QAAS,CACP,eAAgB,mBAChB,OAAQ,qCACV,EACA,KAAM,KAAK,UAAUD,CAAO,CAC9B,CAAC,EAED,GAAI,CAACE,EAAS,GACZ,MAAM,IAAI,MAAM,uBAAuBA,EAAS,MAAM,EAAE,EAG1D,IAAMC,EAAU,MAAMD,EAAS,KAAK,EAEpC,GAAIC,EAAO,MACT,MAAM,IAAI,MACR,mBAAmBA,EAAO,MAAM,OAAO,WAAWA,EAAO,MAAM,IAAI,GACrE,EAGF,OAAOA,EAAO,MAChB,OAASC,EAAO,CACd,MAAAd,EAAO,MACL,6BAASQ,CAAM,MACbM,aAAiB,MAAQA,EAAM,QAAU,OAAOA,CAAK,CACvD,EACF,EACMA,CACR,CACF,CAKA,MAAc,iBAAiBN,EAAgBC,EAA6B,CAC1E,IAAMM,EAAe,CACnB,QAAS,MACT,OAAAP,EACA,OAAAC,CAEF,EAEAT,EAAO,MAAM,kCAAS,KAAK,IAAI,KAAK,KAAK,UAAUe,CAAY,CAAC,EAAE,EAElE,GAAI,CAEF,IAAMJ,GAAS,KAAM,QAAO,YAAY,GAAG,QAErCC,EAAW,MAAMD,EAAM,KAAK,OAAO,IAAK,CAC5C,OAAQ,OACR,QAAS,CACP,eAAgB,mBAChB,OAAQ,qCACV,EACA,KAAM,KAAK,UAAUI,CAAY,CACnC,CAAC,EAED,GAAI,CAACH,EAAS,GACZ,MAAM,IAAI,MAAM,uBAAuBA,EAAS,MAAM,EAAE,EAI1DZ,EAAO,MAAM,gBAAMQ,CAAM,2BAAO,CAClC,OAASM,EAAO,CACd,MAAAd,EAAO,MACL,yCAAWQ,CAAM,MACfM,aAAiB,MAAQA,EAAM,QAAU,OAAOA,CAAK,CACvD,EACF,EACMA,CACR,CACF,CAEA,MAAM,OAAQ,CACZd,EAAO,KAAK,wEAAgC,KAAK,IAAI,EAAE,EAEvD,GAAI,CAEF,MAAM,KAAK,YAAY,aAAc,CACnC,gBAAiB,aACjB,aAAc,CACZ,MAAO,CAAC,CACV,EACA,WAAY,CACV,KAAM,iCACN,QAAS,OACX,CACF,CAAC,EAGD,GAAI,CACF,MAAM,KAAK,iBAAiB,2BAA2B,CACzD,OAASc,EAAO,CAEdd,EAAO,MAAM,GAAG,KAAK,IAAI,kDAAmCc,CAAK,EAAE,CACrE,CAGA,IAAME,EAAkB,MAAM,KAAK,YAAY,YAAY,EAEvDA,GAAiB,QACnB,KAAK,cAAgBA,EAAgB,MAGrC,KAAK,MAAQ,KAAK,cACf,OAAQC,GAASC,EAAc,cAAc,KAAK,KAAMD,EAAK,IAAI,CAAC,EAClE,IAAKA,IAAU,CACd,GAAGA,EACH,KAAM,KAAK,yBAAyBA,EAAK,IAAI,CAC/C,EAAE,EAEJjB,EAAO,KACL,0CAA2B,KAAK,IAAI,+CAAY,KAAK,MAAM,MAAM,IAAI,KAAK,cAAc,MAAM,6CAChG,GAGF,KAAK,YAAc,EACrB,OAASc,EAAO,CACd,MAAAd,EAAO,MACL,uDAA8B,KAAK,IAAI,sBACrCc,aAAiB,MAAQA,EAAM,QAAU,OAAOA,CAAK,CACvD,EACF,EACMA,CACR,CACF,CAEA,MAAM,cAAe,CACnB,GAAI,CACF,IAAME,EAAkB,MAAM,KAAK,YAAY,YAAY,EAEvDA,GAAiB,QACnB,KAAK,cAAgBA,EAAgB,MAGrC,KAAK,MAAQ,KAAK,cACf,OAAQC,GAASC,EAAc,cAAc,KAAK,KAAMD,EAAK,IAAI,CAAC,EAClE,IAAKA,IAAU,CACd,GAAGA,EACH,KAAM,KAAK,yBAAyBA,EAAK,IAAI,CAC/C,EAAE,EAEJjB,EAAO,KACL,sBAAO,KAAK,IAAI,qDAAa,KAAK,MAAM,MAAM,IAAI,KAAK,cAAc,MAAM,6CAC7E,EAEJ,OAASc,EAAO,CACdd,EAAO,MACL,gBAAM,KAAK,IAAI,oDACbc,aAAiB,MAAQA,EAAM,QAAU,OAAOA,CAAK,CACvD,EACF,CACF,CACF,CAEA,MAAM,SAASK,EAAsBC,EAA+B,CAClE,GAAI,CAEF,IAAMC,EAAe,KAAK,oBAAoBF,CAAY,EAC1D,GAAI,CAACE,EACH,MAAM,IAAI,MAAM,+DAAaF,CAAY,EAAE,EAQ7C,OALe,MAAM,KAAK,YAAY,aAAc,CAClD,KAAME,EACN,UAAWD,CACb,CAAC,CAGH,OAASN,EAAO,CACd,MAAAd,EAAO,MACL,UAAK,KAAK,IAAI,mCAAUmB,CAAY,sBAClCL,aAAiB,MAAQA,EAAM,QAAU,OAAOA,CAAK,CACvD,EACF,EACMA,CACR,CACF,CAEA,MAAM,MAAO,CACXd,EAAO,KAAK,4BAAQ,KAAK,IAAI,qBAAM,EACnC,KAAK,YAAc,EACrB,CACF,EJxPA,IAAMsB,EAAYC,EAAQC,EAAc,YAAY,GAAG,CAAC,EAGlDC,EAASA,EAAa,QAAQ,UAAU,EAG1CC,EAAQ,IAAI,iBAAmB,QAAUA,EAAQ,IAAI,qBACvDD,EAAa,YAAYC,EAAQ,IAAI,kBAAkB,EACvDD,EAAa,kBAAkB,EAAI,GAgC9B,IAAME,EAAN,KAAsC,CAlE7C,MAkE6C,CAAAC,EAAA,kBACnC,KACA,OACA,QACD,YACA,MACA,cACC,UACA,gBACA,cAER,YAAYC,EAAcC,EAA8B,CACtD,KAAK,KAAOD,EACZ,KAAK,OAASC,EACd,KAAK,QAAU,KACf,KAAK,YAAc,GACnB,KAAK,MAAQ,CAAC,EACd,KAAK,cAAgB,CAAC,EACtB,KAAK,UAAY,EACjB,KAAK,gBAAkB,IAAI,IAC3B,KAAK,cAAgB,EACvB,CAMO,eACLC,EACAC,EACqD,CACrD,GAAIN,EAAQ,WAAa,QAAS,CAEhC,GAAIK,IAAY,OAASA,IAAY,MACnC,MAAO,CACL,gBAAiB,GAAGA,CAAO,OAC3B,aAAcC,CAChB,EAGF,GAAID,IAAY,MACd,MAAO,CACL,gBAAiB,UACjB,aAAcC,CAChB,CAEJ,CAEA,MAAO,CACL,gBAAiBD,EACjB,aAAcC,CAChB,CACF,CAMQ,yBAAyBC,EAAkC,CAEjE,MAAO,GADsB,KAAK,KAAK,QAAQ,KAAM,GAAG,CAC1B,UAAUA,CAAgB,EAC1D,CAKA,oBAAoBC,EAAyC,CAE3D,IAAMC,EAAS,GADc,KAAK,KAAK,QAAQ,KAAM,GAAG,CAClB,UAEtC,OAAID,EAAiB,WAAWC,CAAM,EAC7BD,EAAiB,UAAUC,EAAO,MAAM,EAG1C,IACT,CAEA,MAAM,OAAQ,CACZV,EAAO,KAAK,wDAAgB,KAAK,IAAI,EAAE,EAEvC,GAAM,CAAE,QAAAM,EAAS,KAAAC,EAAM,IAAAI,CAAI,EAAI,KAAK,OAG9B,CAAE,gBAAAC,EAAiB,aAAAC,CAAa,EAAI,KAAK,eAC7CP,EACAC,CACF,EAEMO,EAAoB,CACxB,MAAO,CAAC,OAAQ,OAAQ,MAAM,CAChC,EAIMC,EAAiBd,EAAQ,IAAI,oBAAsBA,EAAQ,IAAI,EACrEa,EAAa,IAAMC,EAGfJ,EACFG,EAAa,IAAM,CAAE,GAAGb,EAAQ,IAAK,GAAGU,CAAI,EAE5CG,EAAa,IAAM,CAAE,GAAGb,EAAQ,GAAI,EAKpCA,EAAQ,WAAa,UACpBK,IAAY,OAASA,IAAY,OAASA,IAAY,SAEvDQ,EAAa,MAAQ,IAGvBd,EAAO,MACL,GAAG,KAAK,IAAI,8CAAWY,CAAe,IAAIC,EAAa,KACrD,GACF,CAAC,kCAASC,EAAa,GAAG,EAC5B,EACAd,EAAO,MACL,GAAG,KAAK,IAAI,sBAAOC,EAAQ,QAAQ,iCACjCa,EAAa,OAAS,EACxB,EACF,EAEA,KAAK,QAAUE,EAAMJ,EAAiBC,EAAcC,CAAY,EAGhE,KAAK,QAAQ,QAAQ,GAAG,OAASG,GAAiB,CAChD,KAAK,iBAAiBA,EAAK,SAAS,CAAC,CACvC,CAAC,EAGD,KAAK,QAAQ,QAAQ,GAAG,OAASA,GAAiB,CAChDjB,EAAO,MAAM,GAAG,KAAK,IAAI,8CAAWiB,EAAK,SAAS,EAAE,KAAK,CAAC,EAAE,CAC9D,CAAC,EAGD,KAAK,QAAQ,GACX,OACA,CAACC,EAAqBC,IAAkC,CACtDnB,EAAO,MACL,GAAG,KAAK,IAAI,gEAAckB,CAAI,2BAAOC,CAAM,EAC7C,EACA,KAAK,YAAc,EACrB,CACF,EAGA,KAAK,QAAQ,GAAG,QAAUC,GAAiB,CACzCpB,EAAO,MAAM,GAAG,KAAK,IAAI,kCAASoB,EAAM,OAAO,EAAE,EACjD,KAAK,YAAc,EACrB,CAAC,EAGD,MAAM,KAAK,WAAW,CACxB,CAEA,iBAAiBH,EAAoB,CACnC,KAAK,cAAgB,GAAG,KAAK,aAAa,GAAGA,CAAI,GAGjD,IAAMI,EAAQ,KAAK,cAAc,MAAM;AAAA,CAAI,EAC3C,KAAK,cAAgBA,EAAM,IAAI,GAAK,GAEpC,QAAWC,KAAQD,EACjB,GAAIC,EAAK,KAAK,EACZ,GAAI,CACF,IAAMC,EAAU,KAAK,MAAMD,EAAK,KAAK,CAAC,EACtC,KAAK,cAAcC,CAAO,CAC5B,MAAgB,CACdvB,EAAO,MAAM,GAAG,KAAK,IAAI,8CAAWsB,EAAK,KAAK,CAAC,EAAE,CACnD,CAGN,CAEA,cAAcC,EAAoB,CAKhC,GAJAvB,EAAO,MACL,GAAG,KAAK,IAAI,kCAAS,KAAK,UAAUuB,CAAO,EAAE,UAAU,EAAG,GAAG,CAAC,KAChE,EAEIA,EAAQ,IAAM,KAAK,gBAAgB,IAAIA,EAAQ,EAAE,EAAG,CAEtD,IAAMC,EAAiB,KAAK,gBAAgB,IAAID,EAAQ,EAAE,EAC1D,GAAIC,EAAgB,CAClB,GAAM,CAAE,QAAAC,EAAS,OAAAC,CAAO,EAAIF,EAC5B,KAAK,gBAAgB,OAAOD,EAAQ,EAAE,EAElCA,EAAQ,MAGRA,EAAQ,MAAM,OAAS,QACvBC,EAAe,SAAW,6BAE1BxB,EAAO,MACL,GAAG,KAAK,IAAI,kHACd,EACAyB,EAAQ,IAAI,GAEZC,EACE,IAAI,MACF,GAAGH,EAAQ,MAAM,OAAO,WAAWA,EAAQ,MAAM,IAAI,GACvD,CACF,EAGFE,EAAQF,EAAQ,MAAM,CAE1B,CACF,CAEF,CAEA,MAAM,YAAYI,EAAgBC,EAAc,CAAC,EAAiB,CAChE,GAAI,CAAC,KAAK,SAAW,CAAC,KAAK,QAAQ,MACjC,MAAM,IAAI,MAAM,GAAG,KAAK,IAAI,iCAAQ,EAGtC,IAAMC,EAAK,KAAK,YACVC,EAAU,CACd,QAAS,MACT,GAAID,EACJ,OAAQF,EACR,OAAQC,CACV,EAEA,OAAO,IAAI,QAAQ,CAACH,EAASC,IAAW,CACtC,KAAK,gBAAgB,IAAIG,EAAI,CAAE,QAAAJ,EAAS,OAAAC,EAAQ,OAAAC,CAAO,CAAC,EAGxD,WAAW,IAAM,CACX,KAAK,gBAAgB,IAAIE,CAAE,IAC7B,KAAK,gBAAgB,OAAOA,CAAE,EAC9BH,EAAO,IAAI,MAAM,iCAAQC,CAAM,EAAE,CAAC,EAEtC,EAAG,GAAK,EAER,IAAMJ,EAAU,GAAG,KAAK,UAAUO,CAAO,CAAC;AAAA,EAC1C9B,EAAO,MAAM,GAAG,KAAK,IAAI,kCAASuB,EAAQ,KAAK,CAAC,EAAE,EAClD,KAAK,SAAS,OAAO,MAAMA,CAAO,CACpC,CAAC,CACH,CAEA,MAAM,YAAa,CACjB,GAAI,CAEF,IAAMQ,EAAa,MAAM,KAAK,YAAY,aAAc,CACtD,gBAAiB,aACjB,aAAc,CAAC,EACf,WAAY,CACV,KAAM,WACN,QAAS,OACX,CACF,CAAC,EAED/B,EAAO,KACL,GAAG,KAAK,IAAI,oDAAY,KAAK,UAC1B+B,EAAmB,YACtB,CAAC,EACH,EAGA,GAAI,CACF,IAAMC,EAAe,CACnB,QAAS,MACT,OAAQ,2BACV,EACA,KAAK,SAAS,OAAO,MAAM,GAAG,KAAK,UAAUA,CAAY,CAAC;AAAA,CAAI,CAChE,OAASZ,EAAO,CACdpB,EAAO,MACL,GACE,KAAK,IACP,0IACEoB,aAAiB,MAAQA,EAAM,QAAU,OAAOA,CAAK,CACvD,EACF,CACF,CAGA,MAAM,KAAK,aAAa,EAExB,KAAK,YAAc,GACnBpB,EAAO,KAAK,GAAG,KAAK,IAAI,qDAAa,KAAK,MAAM,MAAM,qBAAM,CAC9D,OAASoB,EAAO,CACd,MAAApB,EAAO,MACL,sBAAO,KAAK,IAAI,sBACdoB,aAAiB,MAAQA,EAAM,QAAU,OAAOA,CAAK,CACvD,EACF,EACMA,CACR,CACF,CAEA,MAAM,cAA8B,CAClC,GAAI,CACF,IAAMa,EAAS,MAAM,KAAK,YAAY,YAAY,EAClD,KAAK,cAAiBA,EAAe,OAAS,CAAC,EAG/C,IAAMC,EAAmB,KAAK,cAAc,IAAKC,IAAU,CACzD,GAAGA,EACH,KAAM,KAAK,yBAAyBA,EAAK,IAAI,CAC/C,EAAE,EAGF,KAAK,MAAQ,KAAK,mBAAmBD,CAAgB,EAGrD,MAAM,KAAK,kBAAkB,EAE7BlC,EAAO,KACL,GAAG,KAAK,IAAI,uBACV,KAAK,cAAc,MACrB,4BAAQ,KAAK,cAAc,IAAKoC,GAAMA,EAAE,IAAI,EAAE,KAAK,IAAI,CAAC,EAC1D,EACApC,EAAO,KACL,GAAG,KAAK,IAAI,uBAAQ,KAAK,MAAM,MAAM,4BAAQ,KAAK,MAC/C,IAAKoC,GAAMA,EAAE,IAAI,EACjB,KAAK,IAAI,CAAC,EACf,CACF,OAAShB,EAAO,CACdpB,EAAO,MACL,UAAK,KAAK,IAAI,8CACZoB,aAAiB,MAAQA,EAAM,QAAU,OAAOA,CAAK,CACvD,EACF,EACA,KAAK,MAAQ,CAAC,EACd,KAAK,cAAgB,CAAC,CACxB,CACF,CAKQ,mBAAmBiB,EAA0B,CACnD,OAAOA,EAAS,OAAQF,GAAS,CAC/B,IAAMG,EAAe,KAAK,oBAAoBH,EAAK,IAAI,EACvD,OAAKG,EAEEC,EAAc,cAAc,KAAK,KAAMD,CAAY,EAFhC,EAG5B,CAAC,CACH,CAKA,MAAc,mBAAmC,CAC/C,GAAI,CACF,IAAME,EAAgBD,EAAc,qBAAqB,KAAK,IAAI,EAC5DE,EAA6C,CAAC,EAGpD,QAAWN,KAAQ,KAAK,cAAe,CACrC,IAAMO,EAAiBF,EAAcL,EAAK,IAAI,EAC9CM,EAAYN,EAAK,IAAI,EAAI,CACvB,YAAaA,EAAK,aAAe,GACjC,OAAQO,GAAgB,SAAW,EACrC,CACF,EAGmB,OAAO,KAAKD,CAAW,EAAE,KAAME,GAAa,CAC7D,IAAMC,EAAWJ,EAAcG,CAAQ,EACjCE,EAAYJ,EAAYE,CAAQ,EACtC,MACE,CAACC,GACDA,EAAS,SAAWC,EAAU,QAC9BD,EAAS,cAAgBC,EAAU,WAEvC,CAAC,GAEiB,OAAO,KAAKL,CAAa,EAAE,SAAW,KACtDD,EAAc,wBAAwB,KAAK,KAAME,CAAW,EAC5DzC,EAAO,KAAK,GAAG,KAAK,IAAI,6CAAU,EAEtC,OAASoB,EAAO,CACdpB,EAAO,MACL,gBAAM,KAAK,IAAI,oDACboB,aAAiB,MAAQA,EAAM,QAAU,OAAOA,CAAK,CACvD,EACF,CACF,CACF,CAEA,MAAM,SAAS0B,EAAsBC,EAA+B,CAClE,GAAI,CAEF,IAAMT,EAAe,KAAK,oBAAoBQ,CAAY,EAC1D,GAAI,CAACR,EACH,MAAM,IAAI,MAAM,+DAAaQ,CAAY,EAAE,EAO7C,OAJe,MAAM,KAAK,YAAY,aAAc,CAClD,KAAMR,EACN,UAAWS,CACb,CAAC,CAEH,OAAS3B,EAAO,CACd,MAAApB,EAAO,MACL,UACE,KAAK,IACP,mCAAU8C,CAAY,mCAAU,KAAK,oBACnCA,CACF,CAAC,uBAAQ1B,aAAiB,MAAQA,EAAM,QAAU,OAAOA,CAAK,CAAC,EACjE,EACMA,CACR,CACF,CAEA,MAAa,CACP,KAAK,UACPpB,EAAO,KAAK,4BAAQ,KAAK,IAAI,qBAAM,EACnC,KAAK,QAAQ,KAAK,SAAS,EAC3B,KAAK,QAAU,MAEjB,KAAK,YAAc,EACrB,CACF,EAKO,SAASgD,GAAiD,CAC/D,GAAI,CAEF,GAAIT,EAAc,aAAa,EAAG,CAChC,IAAMU,EAAaV,EAAc,cAAc,EAC/C,OAAAvC,EAAO,KACL,oDAAY,OAAO,KAAKiD,CAAU,EAAE,MAAM,0BAC5C,EACOA,CACT,CAGA,IAAMC,EAAmBzB,EAAQ5B,EAAW,iBAAiB,EAC7D,GAAIsD,EAEF,GAAI,CACF,IAAMC,EAAaD,EAAaD,EAAkB,MAAM,EAClD7C,EAAS,KAAK,MAAM+C,CAAU,EAEpC,GAAI,CAAC/C,EAAO,YAAc,OAAOA,EAAO,YAAe,SACrD,MAAM,IAAI,MAAM,iGAA2B,EAG7C,OAAAL,EAAO,KACL,0DACE,OAAO,KAAKK,EAAO,UAAU,EAAE,MACjC,kGACF,EACOA,EAAO,UAChB,MAAsB,CAEpB,MAAAL,EAAO,MAAM,kHAAkC,EACzC,IAAI,MAAM,kHAAkC,CACpD,CAGF,MAAM,IAAI,MAAM,kHAAkC,CACpD,OAASoB,EAAO,CACd,MAAApB,EAAO,MACL,kDACEoB,aAAiB,MAAQA,EAAM,QAAU,OAAOA,CAAK,CACvD,EACF,EACMA,CACR,CACF,CA7CgBjB,EAAA6C,EAAA,iBAkDT,IAAMK,EAAN,KAAqB,CAzhB5B,MAyhB4B,CAAAlD,EAAA,uBAClB,QACA,QACD,YACC,OAER,aAAc,CACZ,KAAK,QAAU,IAAI,IACnB,KAAK,QAAU,IAAI,IACnB,KAAK,YAAc,GACnB,KAAK,OAAS,IAChB,CAEA,MAAM,OAAQ,CACZH,EAAO,KAAK,uDAAe,EAG3B,KAAK,OAASgD,EAAc,EAG5B,IAAMM,EAAiB,CAAC,EAExB,OAAW,CAACC,EAAYC,CAAY,IAAK,OAAO,QAAQ,KAAK,MAAM,EAAG,CACpExD,EAAO,KAAK,8DAAiBuD,CAAU,EAAE,EAEzC,IAAIE,EAGJ,GAAI,QAASD,EAAc,CAEzB,IAAME,EAAMF,EAAa,IAKtB,SAAUA,GAAgBA,EAAa,OAAS,OAEjDE,EAAI,SAAS,MAAM,GAEnBA,EAAI,SAAS,gBAAgB,EAI7BD,EAAS,IAAIE,EACXJ,EACAC,CACF,EAGAC,EAAS,IAAIG,EACXL,EACAC,CACF,CAEJ,MAEEC,EAAS,IAAIvD,EACXqD,EACAC,CACF,EAGF,KAAK,QAAQ,IAAID,EAAYE,CAAM,EACnCH,EAAe,KAAKG,EAAO,MAAM,CAAC,CACpC,CAGA,GAAI,CACF,IAAMI,EAAU,MAAM,QAAQ,WAAWP,CAAc,EAGnDQ,EAAe,EACnB,QAAS,EAAI,EAAG,EAAID,EAAQ,OAAQ,IAAK,CACvC,IAAM5B,EAAS4B,EAAQ,CAAC,EAClBN,EAAa,OAAO,KAAK,KAAK,MAAM,EAAE,CAAC,EAEzCtB,EAAO,SAAW,aACpB6B,IACA9D,EAAO,KAAK,wDAAgBuD,CAAU,EAAE,IAExCvD,EAAO,MACL,uCAAcuD,CAAU,sBAAOtB,EAAO,OAAO,OAAO,EACtD,EAEA,KAAK,QAAQ,OAAOsB,CAAU,EAElC,CAEA,GAAIO,IAAiB,EACnB,MAAM,IAAI,MAAM,yEAAkB,EAIpC,KAAK,aAAa,EAClB,KAAK,YAAc,GAEnB9D,EAAO,KACL,sFAAqB8D,CAAY,IAC/B,OAAO,KAAK,KAAK,MAAM,EAAE,MAC3B,2BACF,CACF,OAAS1C,EAAO,CACd,MAAApB,EAAO,MACL,wDACEoB,aAAiB,MAAQA,EAAM,QAAU,OAAOA,CAAK,CACvD,EACF,EACMA,CACR,CACF,CAEA,cAAe,CACb,KAAK,QAAQ,MAAM,EAEnB,OAAW,CAAC2C,EAAYN,CAAM,IAAK,KAAK,QACtC,QAAWtB,KAAQsB,EAAO,MAGpB,KAAK,QAAQ,IAAItB,EAAK,IAAI,EAC5BnC,EAAO,MACL,mDACEmC,EAAK,IACP,kBAAQ4B,CAAU,WAAM,KAAK,QAAQ,IAAI5B,EAAK,IAAI,CAAC,GACrD,EAEA,KAAK,QAAQ,IAAIA,EAAK,KAAM4B,CAAU,EAK5C/D,EAAO,KAAK,0DAAa,KAAK,QAAQ,IAAI,qBAAM,EAChDA,EAAO,MAAM,iCAAQ,MAAM,KAAK,KAAK,QAAQ,KAAK,CAAC,EAAE,KAAK,IAAI,CAAC,EAAE,CACnE,CAEA,aAAsB,CACpB,IAAMqC,EAAmB,CAAC,EAC1B,QAAWoB,KAAU,KAAK,QAAQ,OAAO,EACvCpB,EAAS,KAAK,GAAGoB,EAAO,KAAK,EAE/B,OAAOpB,CACT,CAKA,eAIG,CACD,IAAM2B,EAID,CAAC,EAEN,OAAW,CAACT,EAAYE,CAAM,IAAK,KAAK,QACtCO,EAAQ,KAAK,CACX,KAAMT,EACN,UAAWE,EAAO,cAAc,OAChC,iBAAkBA,EAAO,MAAM,MACjC,CAAC,EAGH,OAAOO,CACT,CAKA,eACET,EACgE,CAChE,IAAME,EAAS,KAAK,QAAQ,IAAIF,CAAU,EAC1C,GAAI,CAACE,EACH,MAAM,IAAI,MAAM,wCAAUF,CAAU,EAAE,EAGxC,OAAOE,EAAO,cAAc,IAAKtB,IAAU,CACzC,KAAMA,EAAK,KACX,YAAaA,EAAK,aAAe,GACjC,QAASI,EAAc,cAAcgB,EAAYpB,EAAK,IAAI,CAC5D,EAAE,CACJ,CAKA,MAAM,mBAAmBoB,EAAmC,CAC1D,IAAME,EAAS,KAAK,QAAQ,IAAIF,CAAU,EAC1C,GAAI,CAACE,EACH,MAAM,IAAI,MAAM,wCAAUF,CAAU,EAAE,EAGxC,MAAME,EAAO,aAAa,EAC1B,KAAK,aAAa,CACpB,CAKA,MAAM,iBAAiC,CACrC,IAAMQ,EAAkB,MAAM,KAAK,KAAK,QAAQ,OAAO,CAAC,EAAE,IAAKR,GAC7DA,EAAO,aAAa,CACtB,EACA,MAAM,QAAQ,WAAWQ,CAAe,EACxC,KAAK,aAAa,CACpB,CAEA,MAAM,SAAStB,EAAkBI,EAA+B,CAC9D,IAAMgB,EAAa,KAAK,QAAQ,IAAIpB,CAAQ,EAC5C,GAAI,CAACoB,EACH,MAAM,IAAI,MAAM,uCAASpB,CAAQ,EAAE,EAGrC,IAAMc,EAAS,KAAK,QAAQ,IAAIM,CAAU,EAC1C,GAAI,CAACN,GAAU,CAACA,EAAO,YACrB,MAAM,IAAI,MAAM,sBAAOM,CAAU,qBAAM,EAGzC,OAAO,MAAMN,EAAO,SAASd,EAAUI,CAAU,CACnD,CAEA,MAAO,CACL/C,EAAO,KAAK,uDAAe,EAC3B,QAAWyD,KAAU,KAAK,QAAQ,OAAO,EACvCA,EAAO,KAAK,EAEd,KAAK,YAAc,EACrB,CACF,EAKaS,EAAN,KAAoB,CApwB3B,MAowB2B,CAAA/D,EAAA,sBACjB,MACA,UAER,YAAYgE,EAAuB,CACjC,KAAK,MAAQA,EACb,KAAK,UAAY,CACnB,CAEA,MAAM,cAAc5C,EAAyC,CAC3D,GAAI,CACF,IAAM6C,EAAgB,KAAK,MAAM7C,CAAO,EAKxC,GAJAvB,EAAO,MACL,iCAAQ,KAAK,UAAUoE,CAAa,EAAE,UAAU,EAAG,GAAG,CAAC,KACzD,EAEIA,EAAc,OAAQ,CACxB,GAAIA,EAAc,KAAO,OAAW,CAElC,IAAMC,EAAW,MAAM,KAAK,cAAcD,CAAa,EACvD,OAAO,KAAK,UAAUC,CAAQ,CAChC,CAEA,aAAM,KAAK,mBAAmBD,CAAa,EACpC,IACT,CACA,MAAM,IAAI,MAAM,0CAAiB,CACnC,OAAShD,EAAO,CACd,OAAApB,EAAO,MACL,mDACEoB,aAAiB,MAAQA,EAAM,QAAU,OAAOA,CAAK,CACvD,EACF,EACO,KAAK,UAAU,CACpB,QAAS,MACT,GAAI,KACJ,MAAO,CACL,KAAM,OACN,QAAS,0BACX,CACF,CAAC,CACH,CACF,CAEA,MAAM,cAAcU,EAA4B,CAC9C,GAAM,CAAE,GAAAD,EAAI,OAAAF,EAAQ,OAAAC,EAAS,CAAC,CAAE,EAAIE,EAEpC,GAAI,CACF,IAAIG,EAEJ,OAAQN,EAAQ,CACd,IAAK,aACHM,EAAS,MAAM,KAAK,iBAAiBL,CAAM,EAC3C,MACF,IAAK,aACHK,EAAS,MAAM,KAAK,gBAAgBL,CAAM,EAC1C,MACF,IAAK,aACHK,EAAS,MAAM,KAAK,gBAAgBL,CAAM,EAC1C,MACF,IAAK,OACHK,EAAS,MAAM,KAAK,WAAWL,CAAM,EACrC,MACF,QACE,MAAM,IAAI,MAAM,uCAASD,CAAM,EAAE,CACrC,CAEA,MAAO,CACL,QAAS,MACT,GAAIE,EACJ,OAAQI,CACV,CACF,OAASb,EAAO,CACd,OAAApB,EAAO,MACL,4BAAQ2B,CAAM,4BACZP,aAAiB,MAAQA,EAAM,QAAU,OAAOA,CAAK,CACvD,EACF,EACO,CACL,QAAS,MACT,GAAIS,EACJ,MAAO,CACL,KAAM,OACN,QAAST,aAAiB,MAAQA,EAAM,QAAU,OAAOA,CAAK,CAChE,CACF,CACF,CACF,CAEA,MAAM,mBAAmBY,EAAkC,CACzD,GAAM,CAAE,OAAAL,CAAO,EAAIK,EAEnB,OAAQL,EAAQ,CACd,IAAK,4BACH3B,EAAO,KAAK,oEAAa,EACzB,MACF,QACEA,EAAO,MAAM,iCAAQ2B,CAAM,EAAE,EAC7B,KACJ,CACF,CAEA,MAAM,iBAAiBC,EAA2B,CAChD,OAAA5B,EAAO,KAAK,2EAAe,KAAK,UAAU4B,EAAO,UAAU,CAAC,EAAE,EAEvD,CACL,gBAAiB,aACjB,aAAc,CACZ,MAAO,CACL,YAAa,EACf,CACF,EACA,WAAY,CACV,KAAM,iBACN,QAAS,OACX,CACF,CACF,CAEA,MAAM,gBAAgB0C,EAA4B,CAChD,GAAI,CAAC,KAAK,MAAM,YACd,MAAM,IAAI,MAAM,sCAAQ,EAG1B,IAAMC,EAAQ,KAAK,MAAM,YAAY,EACrC,OAAAvE,EAAO,KAAK,gBAAMuE,EAAM,MAAM,qBAAM,EAE7B,CACL,MAAOA,CACT,CACF,CAEA,MAAM,gBAAgB3C,EAA2B,CAC/C,GAAM,CAAE,KAAAxB,EAAM,UAAWG,CAAK,EAAIqB,EAElC,GAAI,CAACxB,EACH,MAAM,IAAI,MAAM,kDAAU,EAG5BJ,EAAO,KAAK,iCAAQI,CAAI,2BAAO,KAAK,UAAUG,CAAI,CAAC,EAAE,EAErD,IAAM0B,EAAS,MAAM,KAAK,MAAM,SAAS7B,EAAMG,GAAQ,CAAC,CAAC,EAGzD,OAAAP,EAAO,KAAK,qDAAa,OAAOiC,CAAM,EAAE,EACxCjC,EAAO,KAAK,yCAAW,KAAK,UAAUiC,CAAM,EAAE,UAAU,EAAG,GAAG,CAAC,KAAK,EAE7DA,CACT,CAEA,MAAM,WAAWqC,EAA4B,CAC3C,OAAAtE,EAAO,MAAM,gCAAY,EAClB,CAAC,CACV,CACF,EAKA,SAASwE,GAA2B,CAClC,OAAOvE,EAAQ,IAAI,kBAAoB,MACzC,CAFSE,EAAAqE,EAAA,mBAOT,eAAeC,GAAO,CACpBzE,EAAO,KAAK,uDAAe,EAG3B,IAAMmE,EAAQ,IAAId,EACZqB,EAAgB,IAAIR,EAAcC,CAAK,EAGvCQ,EAAUxE,EAAA,IAAM,CACpBH,EAAO,KAAK,uDAAe,EAC3BmE,EAAM,KAAK,EACXlE,EAAQ,KAAK,CAAC,CAChB,EAJgB,WAMhBA,EAAQ,GAAG,SAAU0E,CAAO,EAC5B1E,EAAQ,GAAG,UAAW0E,CAAO,EAE7B,GAAI,CAEF,MAAMR,EAAM,MAAM,EAGdK,EAAgB,IAClBxE,EAAO,KAAK,gCAAgC,EAC5C,QAAQ,IAAI,iBAAiB,GAI/BC,EAAQ,MAAM,YAAY,MAAM,EAEhC,IAAI2E,EAAgB,GAEpB3E,EAAQ,MAAM,GAAG,OAAQ,MAAOgB,GAAS,CACvC2D,EAAgB,GAAGA,CAAa,GAAG3D,CAAI,GAGvC,IAAMI,EAAQuD,EAAc,MAAM;AAAA,CAAI,EACtCA,EAAgBvD,EAAM,IAAI,GAAK,GAE/B,QAAWC,KAAQD,EACjB,GAAIC,EAAK,KAAK,EACZ,GAAI,CACF,IAAM+C,EAAW,MAAMK,EAAc,cAAcpD,EAAK,KAAK,CAAC,EAC1D+C,GACFpE,EAAQ,OAAO,MAAM,GAAGoE,CAAQ;AAAA,CAAI,CAExC,OAASjD,EAAO,CACdpB,EAAO,MACL,mDACEoB,aAAiB,MAAQA,EAAM,QAAU,OAAOA,CAAK,CACvD,EACF,CACF,CAGN,CAAC,EAEDnB,EAAQ,MAAM,GAAG,MAAO,IAAM,CAC5BD,EAAO,KAAK,0EAAc,EAC1B2E,EAAQ,CACV,CAAC,EAED3E,EAAO,KAAK,yEAAuB,CACrC,OAASoB,EAAO,CACdpB,EAAO,MACL,8DACEoB,aAAiB,MAAQA,EAAM,QAAU,OAAOA,CAAK,CACvD,EACF,EACAnB,EAAQ,KAAK,CAAC,CAChB,CACF,CAvEeE,EAAAsE,EAAA,QA2Ef,IAAMI,EAAiB,YAAY,IAC7BC,EAAa/E,EAAc8E,CAAc,EACzCE,GAAY9E,EAAQ,KAAK,CAAC,EAE5B6E,IAAeC,IACjBN,EAAK,EAAE,MAAOrD,GAAU,CACtBpB,EAAO,MACL,6CAAUoB,aAAiB,MAAQA,EAAM,QAAU,OAAOA,CAAK,CAAC,EAClE,EACAnB,EAAQ,KAAK,CAAC,CAChB,CAAC","names":["spawn","readFileSync","dirname","resolve","process","fileURLToPath","copyFileSync","existsSync","readFileSync","writeFileSync","dirname","resolve","fileURLToPath","__dirname","dirname","fileURLToPath","DEFAULT_CONNECTION_CONFIG","ConfigManager","_ConfigManager","__name","resolve","configDir","configPath","existsSync","copyFileSync","configData","readFileSync","config","error","configObj","endpoint","serverName","serverConfig","sc","toolName","ep","newConfig","currentEndpoints","newEndpoints","localConfig","newMcpServers","toolsConfig","enabled","description","configJson","writeFileSync","connectionConfig","newConnectionConfig","interval","timeout","modelScopeConfig","newModelScopeConfig","apiKey","webUIConfig","newWebUIConfig","port","configManager","fs","path","chalk","createConsola","formatDateTime","date","year","month","day","hours","minutes","seconds","__name","Logger","createConsola","isDaemonMode","logObj","levelMap","colorMap","chalk","text","level","colorFn","timestamp","coloredLevel","message","error","projectDir","path","fs","args","formattedMessage","fullMessage","arg","enable","tag","logger","Client","SSEClientTransport","EventSource","EventSource","logger","ModelScopeMCPClient","__name","name","config","originalToolName","prefixedToolName","prefix","token","configManager","sseOptions","url","init","headers","SSEClientTransport","Client","error","result","tool","prefixedName","arguments_","originalName","logger","StreamableHTTPMCPClient","__name","name","config","originalToolName","prefixedToolName","prefix","method","params","request","fetch","response","result","error","notification","listToolsResult","tool","configManager","prefixedName","arguments_","originalName","__dirname","dirname","fileURLToPath","logger","process","MCPClient","__name","name","config","command","args","originalToolName","prefixedToolName","prefix","env","resolvedCommand","resolvedArgs","spawnOptions","userWorkingDir","spawn","data","code","signal","error","lines","line","message","pendingRequest","resolve","reject","method","params","id","request","initResult","notification","result","allPrefixedTools","tool","t","allTools","originalName","configManager","currentConfig","toolsConfig","existingConfig","toolName","existing","newConfig","prefixedName","arguments_","loadMCPConfig","mcpServers","legacyConfigPath","readFileSync","configData","MCPServerProxy","clientPromises","serverName","serverConfig","client","url","ModelScopeMCPClient","StreamableHTTPMCPClient","results","successCount","clientName","servers","refreshPromises","JSONRPCServer","proxy","parsedMessage","response","_params","tools","isMCPServerMode","main","jsonrpcServer","cleanup","messageBuffer","currentFileUrl","scriptPath","argv1Path"]}
|
|
1
|
+
{"version":3,"sources":["../src/mcpServerProxy.ts","../src/configManager.ts","../src/logger.ts","../src/modelScopeMCPClient.ts","../src/sseMCPClient.ts","../src/streamableHttpMCPClient.ts"],"sourcesContent":["#!/usr/bin/env node\n\n/**\n * MCP Server Proxy - JavaScript Implementation\n * Provides a proxy to aggregate multiple MCP servers dynamically from configuration\n * Version: 0.3.0 - Manual JSON-RPC 2.0 implementation (no SDK) with dynamic config\n */\n\nimport { type ChildProcess, spawn } from \"node:child_process\";\nimport { readFileSync } from \"node:fs\";\nimport { dirname, resolve } from \"node:path\";\nimport process from \"node:process\";\nimport { fileURLToPath } from \"node:url\";\nimport {\n type LocalMCPServerConfig,\n type MCPServerConfig,\n type MCPToolConfig,\n type SSEMCPServerConfig,\n type StreamableHTTPMCPServerConfig,\n configManager,\n} from \"./configManager\";\nimport { logger as globalLogger } from \"./logger\";\nimport { ModelScopeMCPClient } from \"./modelScopeMCPClient\";\nimport { SSEMCPClient } from \"./sseMCPClient\";\nimport { StreamableHTTPMCPClient } from \"./streamableHttpMCPClient\";\n\n// ESM 兼容的 __dirname\nconst __dirname = dirname(fileURLToPath(import.meta.url));\n\n// 为 MCPProxy 创建带标签的 logger\nconst logger = globalLogger.withTag(\"MCPProxy\");\n\n// 初始化日志文件\n// 优先使用配置目录,如果没有则使用当前工作目录\nconst logDir = process.env.XIAOZHI_CONFIG_DIR || process.cwd();\nglobalLogger.initLogFile(logDir);\nglobalLogger.enableFileLogging(true);\nlogger.info(`日志文件已初始化: ${logDir}/xiaozhi.log`);\n\n// Type definitions\n\ninterface Tool {\n name: string;\n description?: string;\n inputSchema?: any;\n}\n\ninterface PendingRequest {\n resolve: (value: any) => void;\n reject: (reason: any) => void;\n method: string; // 添加方法名以便错误处理\n}\n\n// 定义 MCP 客户端接口\nexport interface IMCPClient {\n initialized: boolean;\n tools: Tool[];\n originalTools: Tool[];\n start(): Promise<void>;\n refreshTools(): Promise<void>;\n callTool(toolName: string, arguments_: any): Promise<any>;\n stop(): void | Promise<void>;\n getOriginalToolName(prefixedToolName: string): string | null;\n}\n\n/**\n * MCP Client for communicating with child MCP servers\n */\nexport class MCPClient implements IMCPClient {\n private name: string;\n private config: LocalMCPServerConfig;\n private process: ChildProcess | null;\n public initialized: boolean;\n public tools: Tool[];\n public originalTools: Tool[]; // 存储原始工具名称\n private requestId: number;\n private pendingRequests: Map<number, PendingRequest>;\n private messageBuffer: string;\n\n constructor(name: string, config: LocalMCPServerConfig) {\n this.name = name;\n this.config = config;\n this.process = null;\n this.initialized = false;\n this.tools = [];\n this.originalTools = [];\n this.requestId = 1;\n this.pendingRequests = new Map();\n this.messageBuffer = \"\";\n }\n\n /**\n * Resolve command for cross-platform execution\n * On Windows, npm/npx/uvx commands need special handling\n */\n public resolveCommand(\n command: string,\n args: string[]\n ): { resolvedCommand: string; resolvedArgs: string[] } {\n if (process.platform === \"win32\") {\n // On Windows, npm, npx, and uvx are .cmd files or .exe files\n if (command === \"npm\" || command === \"npx\") {\n return {\n resolvedCommand: `${command}.cmd`,\n resolvedArgs: args,\n };\n }\n // uvx is typically installed as uvx.bat on Windows (via pyenv)\n if (command === \"uvx\") {\n return {\n resolvedCommand: \"uvx.bat\",\n resolvedArgs: args,\n };\n }\n }\n\n return {\n resolvedCommand: command,\n resolvedArgs: args,\n };\n }\n\n /**\n * 生成带前缀的工具名称\n * 将服务器名称中的中划线替换为下划线,并添加 xzcli 前缀\n */\n private generatePrefixedToolName(originalToolName: string): string {\n const normalizedServerName = this.name.replace(/-/g, \"_\");\n return `${normalizedServerName}_xzcli_${originalToolName}`;\n }\n\n /**\n * 根据前缀工具名称获取原始工具名称\n */\n getOriginalToolName(prefixedToolName: string): string | null {\n const normalizedServerName = this.name.replace(/-/g, \"_\");\n const prefix = `${normalizedServerName}_xzcli_`;\n\n if (prefixedToolName.startsWith(prefix)) {\n return prefixedToolName.substring(prefix.length);\n }\n\n return null;\n }\n\n async start() {\n logger.info(`正在启动 MCP 客户端:${this.name}`);\n\n const { command, args, env } = this.config;\n\n // Handle cross-platform command execution\n const { resolvedCommand, resolvedArgs } = this.resolveCommand(\n command,\n args\n );\n\n const spawnOptions: any = {\n stdio: [\"pipe\", \"pipe\", \"pipe\"],\n };\n\n // Set working directory to user's current working directory\n // This ensures relative paths in MCP server configs are resolved correctly\n const userWorkingDir = process.env.XIAOZHI_CONFIG_DIR || process.cwd();\n spawnOptions.cwd = userWorkingDir;\n\n // Add environment variables if specified\n if (env) {\n spawnOptions.env = { ...process.env, ...env };\n } else {\n spawnOptions.env = { ...process.env };\n }\n\n // On Windows, we need to set shell: true for npm/npx/uvx commands\n if (\n process.platform === \"win32\" &&\n (command === \"npm\" || command === \"npx\" || command === \"uvx\")\n ) {\n spawnOptions.shell = true;\n }\n\n logger.debug(\n `${this.name} 正在生成进程:${resolvedCommand} ${resolvedArgs.join(\n \" \"\n )} 工作目录:${spawnOptions.cwd}`\n );\n logger.debug(\n `${this.name} 平台:${process.platform},shell 模式:${\n spawnOptions.shell || false\n }`\n );\n\n this.process = spawn(resolvedCommand, resolvedArgs, spawnOptions);\n\n // Handle process stdout - parse JSON-RPC messages\n this.process.stdout?.on(\"data\", (data: Buffer) => {\n this.handleStdoutData(data.toString());\n });\n\n // Handle process stderr - log errors\n this.process.stderr?.on(\"data\", (data: Buffer) => {\n logger.debug(`${this.name} 标准错误输出:${data.toString().trim()}`);\n });\n\n // Handle process exit\n this.process.on(\n \"exit\",\n (code: number | null, signal: NodeJS.Signals | null) => {\n logger.error(\n `${this.name} 进程已退出,退出码:${code},信号:${signal}`\n );\n this.initialized = false;\n }\n );\n\n // Handle process error\n this.process.on(\"error\", (error: Error) => {\n logger.error(`${this.name} 进程错误:${error.message}`);\n this.initialized = false;\n });\n\n // Initialize the MCP connection\n await this.initialize();\n }\n\n handleStdoutData(data: string): void {\n this.messageBuffer = `${this.messageBuffer}${data}`;\n\n // Split by newlines and process complete messages\n const lines = this.messageBuffer.split(\"\\n\");\n this.messageBuffer = lines.pop() || \"\"; // Keep incomplete line in buffer\n\n for (const line of lines) {\n if (line.trim()) {\n try {\n const message = JSON.parse(line.trim());\n this.handleMessage(message);\n } catch (error) {\n logger.error(`${this.name} 解析消息失败:${line.trim()}`);\n }\n }\n }\n }\n\n handleMessage(message: any): void {\n logger.debug(\n `${this.name} 收到消息:${JSON.stringify(message).substring(0, 200)}...`\n );\n\n if (message.id && this.pendingRequests.has(message.id)) {\n // This is a response to our request\n const pendingRequest = this.pendingRequests.get(message.id);\n if (pendingRequest) {\n const { resolve, reject } = pendingRequest;\n this.pendingRequests.delete(message.id);\n\n if (message.error) {\n // 对于 notifications/initialized 的 \"Method not found\" 错误,记录为调试信息而不是错误\n if (\n message.error.code === -32601 &&\n pendingRequest.method === \"notifications/initialized\"\n ) {\n logger.debug(\n `${this.name} 服务器不支持 notifications/initialized 通知,这是正常的`\n );\n resolve(null); // 将其视为成功\n } else {\n reject(\n new Error(\n `${message.error.message} (code: ${message.error.code})`\n )\n );\n }\n } else {\n resolve(message.result);\n }\n }\n }\n // Note: We don't handle notifications or requests from child servers in this proxy\n }\n\n async sendRequest(method: string, params: any = {}): Promise<any> {\n if (!this.process || !this.process.stdin) {\n throw new Error(`${this.name} 进程不可用`);\n }\n\n const id = this.requestId++;\n const request = {\n jsonrpc: \"2.0\",\n id: id,\n method: method,\n params: params,\n };\n\n return new Promise((resolve, reject) => {\n this.pendingRequests.set(id, { resolve, reject, method });\n\n // Set timeout for request\n setTimeout(() => {\n if (this.pendingRequests.has(id)) {\n this.pendingRequests.delete(id);\n reject(new Error(`请求超时:${method}`));\n }\n }, 30000); // 30 second timeout\n\n const message = `${JSON.stringify(request)}\\n`;\n logger.debug(`${this.name} 正在发送:${message.trim()}`);\n this.process?.stdin?.write(message);\n });\n }\n\n async initialize() {\n try {\n // Send initialize request\n const initResult = await this.sendRequest(\"initialize\", {\n protocolVersion: \"2024-11-05\",\n capabilities: {},\n clientInfo: {\n name: \"MCPProxy\",\n version: \"0.3.0\",\n },\n });\n\n logger.info(\n `${this.name} 已初始化,能力:${JSON.stringify(\n (initResult as any).capabilities\n )}`\n );\n\n // Send initialized notification (ignore errors as some servers don't support it)\n try {\n const notification = {\n jsonrpc: \"2.0\",\n method: \"notifications/initialized\",\n };\n this.process?.stdin?.write(`${JSON.stringify(notification)}\\n`);\n } catch (error) {\n logger.debug(\n `${\n this.name\n } notifications/initialized 发送失败(某些服务器不支持此通知):${\n error instanceof Error ? error.message : String(error)\n }`\n );\n }\n\n // Get tools list\n await this.refreshTools();\n\n this.initialized = true;\n logger.info(`${this.name} 客户端已就绪,共 ${this.tools.length} 个工具`);\n } catch (error) {\n logger.error(\n `初始化 ${this.name} 失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n throw error;\n }\n }\n\n async refreshTools(): Promise<void> {\n try {\n const result = await this.sendRequest(\"tools/list\");\n this.originalTools = (result as any).tools || [];\n\n // 为每个工具生成带前缀的名称,并应用过滤\n const allPrefixedTools = this.originalTools.map((tool) => ({\n ...tool,\n name: this.generatePrefixedToolName(tool.name),\n }));\n\n // 根据配置过滤工具\n this.tools = this.filterEnabledTools(allPrefixedTools);\n\n // 更新配置文件中的工具列表(如果需要)\n await this.updateToolsConfig();\n\n logger.info(\n `${this.name} 加载了 ${\n this.originalTools.length\n } 个工具:${this.originalTools.map((t) => t.name).join(\", \")}`\n );\n logger.info(\n `${this.name} 启用了 ${this.tools.length} 个工具:${this.tools\n .map((t) => t.name)\n .join(\", \")}`\n );\n } catch (error) {\n logger.error(\n `从 ${this.name} 获取工具失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n this.tools = [];\n this.originalTools = [];\n }\n }\n\n /**\n * 过滤启用的工具\n */\n private filterEnabledTools(allTools: Tool[]): Tool[] {\n return allTools.filter((tool) => {\n const originalName = this.getOriginalToolName(tool.name);\n if (!originalName) return true; // 如果无法解析原始名称,默认启用\n\n return configManager.isToolEnabled(this.name, originalName);\n });\n }\n\n /**\n * 更新配置文件中的工具列表\n */\n private async updateToolsConfig(): Promise<void> {\n try {\n const currentConfig = configManager.getServerToolsConfig(this.name);\n const toolsConfig: Record<string, MCPToolConfig> = {};\n\n // 为每个工具创建配置项\n for (const tool of this.originalTools) {\n const existingConfig = currentConfig[tool.name];\n toolsConfig[tool.name] = {\n description: tool.description || \"\",\n enable: existingConfig?.enable !== false, // 默认启用\n };\n }\n\n // 只有当配置发生变化时才更新\n const hasChanges = Object.keys(toolsConfig).some((toolName) => {\n const existing = currentConfig[toolName];\n const newConfig = toolsConfig[toolName];\n return (\n !existing ||\n existing.enable !== newConfig.enable ||\n existing.description !== newConfig.description\n );\n });\n\n if (hasChanges || Object.keys(currentConfig).length === 0) {\n configManager.updateServerToolsConfig(this.name, toolsConfig);\n logger.info(`${this.name} 已更新工具配置`);\n }\n } catch (error) {\n logger.error(\n `更新 ${this.name} 的工具配置失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n }\n }\n\n async callTool(prefixedName: string, arguments_: any): Promise<any> {\n const startTime = Date.now();\n\n try {\n // 将前缀名称转换回原始名称\n const originalName = this.getOriginalToolName(prefixedName);\n if (!originalName) {\n throw new Error(`无效的工具名称格式:${prefixedName}`);\n }\n\n // 记录MCP客户端级别的工具调用请求\n logger.info(`[MCP客户端 ${this.name}] 工具调用开始`);\n logger.info(\n `[MCP客户端 ${this.name}] 工具名称: ${originalName} (前缀: ${prefixedName})`\n );\n logger.info(\n `[MCP客户端 ${this.name}] 请求参数: ${JSON.stringify(arguments_, null, 2)}`\n );\n\n const result = await this.sendRequest(\"tools/call\", {\n name: originalName,\n arguments: arguments_,\n });\n\n const duration = Date.now() - startTime;\n\n // 记录MCP客户端级别的工具调用结果\n logger.info(`[MCP客户端 ${this.name}] 工具调用成功`);\n logger.info(`[MCP客户端 ${this.name}] 工具名称: ${originalName}`);\n logger.info(`[MCP客户端 ${this.name}] 执行耗时: ${duration}ms`);\n logger.info(\n `[MCP客户端 ${this.name}] 完整结果: ${JSON.stringify(result, null, 2)}`\n );\n\n return result;\n } catch (error) {\n const duration = Date.now() - startTime;\n\n logger.error(\n `[MCP客户端 ${this.name}] 工具调用失败` +\n `工具: ${prefixedName} -> ${this.getOriginalToolName(prefixedName) || \"未知\"}` +\n `耗时: ${duration}ms` +\n `错误: ${error instanceof Error ? error.message : String(error)}`\n );\n logger.error(\n `[MCP客户端 ${this.name}] 错误堆栈: ${error instanceof Error ? error.stack : \"无堆栈信息\"}`\n );\n throw error;\n }\n }\n\n stop(): void {\n if (this.process) {\n logger.info(`正在停止 ${this.name} 客户端`);\n this.process.kill(\"SIGTERM\");\n this.process = null;\n }\n this.initialized = false;\n }\n}\n\n/**\n * Configuration loader for MCP servers\n */\nexport function loadMCPConfig(): Record<string, MCPServerConfig> {\n try {\n // 首先尝试从配置管理器读取\n if (configManager.configExists()) {\n const mcpServers = configManager.getMcpServers();\n logger.info(\n `从配置文件加载了 ${Object.keys(mcpServers).length} 个 MCP 服务`\n );\n return mcpServers as Record<string, MCPServerConfig>;\n }\n\n // 如果配置文件不存在,尝试从旧的 mcp_server.json 读取(向后兼容)\n const legacyConfigPath = resolve(__dirname, \"mcp_server.json\");\n if (readFileSync) {\n // 检查是否可以读取文件\n try {\n const configData = readFileSync(legacyConfigPath, \"utf8\");\n const config = JSON.parse(configData);\n\n if (!config.mcpServers || typeof config.mcpServers !== \"object\") {\n throw new Error(\"无效的配置:mcpServers 部分未找到或无效\");\n }\n\n logger.info(\n `从旧配置文件加载了 ${\n Object.keys(config.mcpServers).length\n } 个 MCP 服务(建议迁移到新配置格式)`\n );\n return config.mcpServers;\n } catch (legacyError) {\n // 旧配置文件也不存在或无效\n logger.error('配置文件不存在,请运行 \"xiaozhi init\" 初始化配置');\n throw new Error('配置文件不存在,请运行 \"xiaozhi init\" 初始化配置');\n }\n }\n\n throw new Error('配置文件不存在,请运行 \"xiaozhi init\" 初始化配置');\n } catch (error) {\n logger.error(\n `加载 MCP 配置失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n throw error;\n }\n}\n\n/**\n * MCP Server Proxy - Main proxy server implementation\n */\nexport class MCPServerProxy {\n private clients: Map<string, IMCPClient>;\n private toolMap: Map<string, string>; // Maps tool name to client name\n public initialized: boolean;\n private config: Record<string, MCPServerConfig> | null;\n\n constructor() {\n this.clients = new Map();\n this.toolMap = new Map(); // Maps tool name to client name\n this.initialized = false;\n this.config = null;\n }\n\n async start() {\n logger.info(\"正在启动 MCP 服务代理\");\n\n // Load configuration\n this.config = loadMCPConfig();\n\n // Initialize child MCP clients from configuration\n const clientPromises = [];\n\n for (const [serverName, serverConfig] of Object.entries(this.config)) {\n logger.info(`正在初始化 MCP 客户端:${serverName}`);\n\n let client: IMCPClient;\n\n // 判断服务类型\n if (\"url\" in serverConfig) {\n // URL 类型的配置\n const url = serverConfig.url;\n\n // 判断是 SSE 还是 Streamable HTTP\n const isSSE =\n // 1. 显式指定 type: \"sse\"\n (\"type\" in serverConfig && serverConfig.type === \"sse\") ||\n // 2. URL 路径以 /sse 结尾(忽略查询参数)\n (() => {\n try {\n const urlObj = new URL(url);\n return urlObj.pathname.endsWith(\"/sse\");\n } catch {\n // 如果 URL 解析失败,回退到简单的字符串匹配\n return url.includes(\"/sse\");\n }\n })() ||\n // 3. 域名包含 modelscope.net(向后兼容魔搭社区)\n url.includes(\"modelscope.net\");\n\n if (isSSE) {\n // SSE MCP 服务 - 区分 ModelScope 和通用 SSE\n if (url.includes(\"modelscope.net\")) {\n // ModelScope SSE 服务(需要特殊认证)\n client = new ModelScopeMCPClient(\n serverName,\n serverConfig as SSEMCPServerConfig\n );\n } else {\n // 通用 SSE 服务(如高德地图等)\n client = new SSEMCPClient(\n serverName,\n serverConfig as SSEMCPServerConfig\n );\n }\n } else {\n // Streamable HTTP MCP 服务\n client = new StreamableHTTPMCPClient(\n serverName,\n serverConfig as StreamableHTTPMCPServerConfig\n );\n }\n } else {\n // 本地 MCP 服务\n client = new MCPClient(\n serverName,\n serverConfig as LocalMCPServerConfig\n );\n }\n\n this.clients.set(serverName, client);\n clientPromises.push(client.start());\n }\n\n // Start all clients\n try {\n const results = await Promise.allSettled(clientPromises);\n\n // Check for failed clients\n let successCount = 0;\n for (let i = 0; i < results.length; i++) {\n const result = results[i];\n const serverName = Object.keys(this.config)[i];\n\n if (result.status === \"fulfilled\") {\n successCount++;\n logger.info(`成功启动 MCP 客户端:${serverName}`);\n } else {\n logger.error(\n `启动 MCP 客户端 ${serverName} 失败:${result.reason.message}`\n );\n // Remove failed client from clients map\n this.clients.delete(serverName);\n }\n }\n\n if (successCount === 0) {\n throw new Error(\"没有成功启动任何 MCP 客户端\");\n }\n\n // Build tool mapping\n this.buildToolMap();\n this.initialized = true;\n\n logger.info(\n `MCP 服务代理初始化成功,启动了 ${successCount}/${\n Object.keys(this.config).length\n } 个客户端`\n );\n } catch (error) {\n logger.error(\n `启动 MCP 客户端失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n throw error;\n }\n }\n\n buildToolMap() {\n this.toolMap.clear();\n\n for (const [clientName, client] of this.clients) {\n for (const tool of client.tools) {\n // 工具名称现在已经带有前缀,应该不会有重复\n // 但我们仍然检查以防万一\n if (this.toolMap.has(tool.name)) {\n logger.error(\n `重复的工具名称:${\n tool.name\n } (来自 ${clientName} 和 ${this.toolMap.get(tool.name)})`\n );\n } else {\n this.toolMap.set(tool.name, clientName);\n }\n }\n }\n\n logger.info(`已构建工具映射,共 ${this.toolMap.size} 个工具`);\n logger.debug(`工具映射:${Array.from(this.toolMap.keys()).join(\", \")}`);\n }\n\n getAllTools(): Tool[] {\n const allTools: Tool[] = [];\n for (const client of this.clients.values()) {\n allTools.push(...client.tools);\n }\n return allTools;\n }\n\n /**\n * 获取所有服务器的信息\n */\n getAllServers(): Array<{\n name: string;\n toolCount: number;\n enabledToolCount: number;\n }> {\n const servers: Array<{\n name: string;\n toolCount: number;\n enabledToolCount: number;\n }> = [];\n\n for (const [serverName, client] of this.clients) {\n servers.push({\n name: serverName,\n toolCount: client.originalTools.length,\n enabledToolCount: client.tools.length,\n });\n }\n\n return servers;\n }\n\n /**\n * 获取指定服务器的工具信息\n */\n getServerTools(\n serverName: string\n ): Array<{ name: string; description: string; enabled: boolean }> {\n const client = this.clients.get(serverName);\n if (!client) {\n throw new Error(`未找到服务器 ${serverName}`);\n }\n\n return client.originalTools.map((tool) => ({\n name: tool.name,\n description: tool.description || \"\",\n enabled: configManager.isToolEnabled(serverName, tool.name),\n }));\n }\n\n /**\n * 刷新指定服务器的工具列表\n */\n async refreshServerTools(serverName: string): Promise<void> {\n const client = this.clients.get(serverName);\n if (!client) {\n throw new Error(`未找到服务器 ${serverName}`);\n }\n\n await client.refreshTools();\n this.buildToolMap(); // 重新构建工具映射\n }\n\n /**\n * 刷新所有服务器的工具列表\n */\n async refreshAllTools(): Promise<void> {\n const refreshPromises = Array.from(this.clients.values()).map((client) =>\n client.refreshTools()\n );\n await Promise.allSettled(refreshPromises);\n this.buildToolMap(); // 重新构建工具映射\n }\n\n async callTool(toolName: string, arguments_: any): Promise<any> {\n const startTime = Date.now();\n const clientName = this.toolMap.get(toolName);\n\n if (!clientName) {\n throw new Error(`未知的工具:${toolName}`);\n }\n\n const client = this.clients.get(clientName);\n if (!client || !client.initialized) {\n throw new Error(`客户端 ${clientName} 不可用`);\n }\n\n // 记录在代理层的工具调用请求\n logger.info(\"[代理层] 工具调用开始\");\n logger.info(`[代理层] 工具名称: ${toolName}`);\n logger.info(`[代理层] 所属客户端: ${clientName}`);\n logger.info(`[代理层] 请求参数: ${JSON.stringify(arguments_, null, 2)}`);\n\n try {\n const result = await client.callTool(toolName, arguments_);\n const duration = Date.now() - startTime;\n\n // 记录在代理层的工具调用结果\n logger.info(\"[代理层] 工具调用成功\");\n logger.info(`[代理层] 工具名称: ${toolName}`);\n logger.info(`[代理层] 所属客户端: ${clientName}`);\n logger.info(`[代理层] 执行耗时: ${duration}ms`);\n logger.info(`[代理层] 完整结果: ${JSON.stringify(result, null, 2)}`);\n\n return result;\n } catch (error) {\n const duration = Date.now() - startTime;\n\n logger.error(\"[代理层] 工具调用失败\");\n logger.error(`[代理层] 工具名称: ${toolName}`);\n logger.error(`[代理层] 所属客户端: ${clientName}`);\n logger.error(`[代理层] 执行耗时: ${duration}ms`);\n logger.error(\n `[代理层] 错误信息: ${error instanceof Error ? error.message : String(error)}`\n );\n\n throw error;\n }\n }\n\n stop() {\n logger.info(\"正在停止 MCP 服务代理\");\n for (const client of this.clients.values()) {\n client.stop();\n }\n this.initialized = false;\n }\n}\n\n/**\n * Manual JSON-RPC 2.0 Server Implementation\n */\nexport class JSONRPCServer {\n private proxy: MCPServerProxy;\n private requestId: number;\n\n constructor(proxy: MCPServerProxy) {\n this.proxy = proxy;\n this.requestId = 1;\n }\n\n async handleMessage(message: string): Promise<string | null> {\n try {\n const parsedMessage = JSON.parse(message);\n logger.debug(\n `收到请求:${JSON.stringify(parsedMessage).substring(0, 200)}...`\n );\n\n if (parsedMessage.method) {\n if (parsedMessage.id !== undefined) {\n // This is a request\n const response = await this.handleRequest(parsedMessage);\n return JSON.stringify(response);\n }\n // This is a notification\n await this.handleNotification(parsedMessage);\n return null; // No response for notifications\n }\n throw new Error(\"无效的 JSON-RPC 消息\");\n } catch (error) {\n logger.error(\n `处理消息时出错:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n return JSON.stringify({\n jsonrpc: \"2.0\",\n id: null,\n error: {\n code: -32700,\n message: \"解析错误\",\n },\n });\n }\n }\n\n async handleRequest(request: any): Promise<any> {\n const { id, method, params = {} } = request;\n\n try {\n let result: any;\n\n switch (method) {\n case \"initialize\":\n result = await this.handleInitialize(params);\n break;\n case \"tools/list\":\n result = await this.handleToolsList(params);\n break;\n case \"tools/call\":\n result = await this.handleToolsCall(params);\n break;\n case \"ping\":\n result = await this.handlePing(params);\n break;\n default:\n throw new Error(`未知的方法:${method}`);\n }\n\n return {\n jsonrpc: \"2.0\",\n id: id,\n result: result,\n };\n } catch (error) {\n logger.error(\n `处理请求 ${method} 时出错:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n return {\n jsonrpc: \"2.0\",\n id: id,\n error: {\n code: -32603,\n message: error instanceof Error ? error.message : String(error),\n },\n };\n }\n }\n\n async handleNotification(notification: any): Promise<void> {\n const { method } = notification;\n\n switch (method) {\n case \"notifications/initialized\":\n logger.info(\"客户端发送了初始化通知\");\n break;\n default:\n logger.debug(`收到通知:${method}`);\n break;\n }\n }\n\n async handleInitialize(params: any): Promise<any> {\n logger.info(`收到客户端的初始化请求:${JSON.stringify(params.clientInfo)}`);\n\n return {\n protocolVersion: \"2024-11-05\",\n capabilities: {\n tools: {\n listChanged: false,\n },\n },\n serverInfo: {\n name: \"MCPServerProxy\",\n version: \"0.3.0\",\n },\n };\n }\n\n async handleToolsList(_params: any): Promise<any> {\n if (!this.proxy.initialized) {\n throw new Error(\"代理未初始化\");\n }\n\n const tools = this.proxy.getAllTools();\n logger.info(`返回 ${tools.length} 个工具`);\n\n return {\n tools: tools,\n };\n }\n\n async handleToolsCall(params: any): Promise<any> {\n const { name, arguments: args } = params;\n const startTime = Date.now();\n\n if (!name) {\n throw new Error(\"工具名称是必需的\");\n }\n\n // 记录完整的工具调用请求\n logger.info(\"=== 工具调用开始 ===\");\n logger.info(`工具名称: ${name}`);\n logger.info(`请求参数: ${JSON.stringify(args, null, 2)}`);\n logger.info(`调用时间: ${new Date().toISOString()}`);\n\n try {\n const result = await this.proxy.callTool(name, args || {});\n const duration = Date.now() - startTime;\n\n // 记录完整的工具调用结果\n logger.info(\"=== 工具调用成功 ===\");\n logger.info(`工具名称: ${name}`);\n logger.info(`执行耗时: ${duration}ms`);\n logger.info(`完整结果: ${JSON.stringify(result, null, 2)}`);\n logger.info(`结果类型: ${typeof result}`);\n if (result && typeof result === \"object\") {\n logger.info(\n `结果长度: ${Array.isArray(result) ? result.length : Object.keys(result).length} 项`\n );\n }\n\n return result;\n } catch (error) {\n const duration = Date.now() - startTime;\n\n // 记录工具调用错误\n logger.error(\"=== 工具调用失败 ===\");\n logger.error(`工具名称: ${name}`);\n logger.error(`执行耗时: ${duration}ms`);\n logger.error(\n `错误信息: ${error instanceof Error ? error.message : String(error)}`\n );\n logger.error(\n `错误堆栈: ${error instanceof Error ? error.stack : \"无堆栈信息\"}`\n );\n\n throw error;\n }\n }\n\n async handlePing(_params: any): Promise<any> {\n logger.debug(\"收到 ping 请求\");\n return {}; // Empty response for ping\n }\n}\n\n/**\n * Check if running in MCP Server mode\n */\nfunction isMCPServerMode(): boolean {\n return process.env.MCP_SERVER_MODE === \"true\";\n}\n\n/**\n * Main function to start the MCP Server Proxy\n */\nasync function main() {\n logger.info(\"正在启动 MCP 服务代理\");\n\n // Create and start the proxy\n const proxy = new MCPServerProxy();\n const jsonrpcServer = new JSONRPCServer(proxy);\n\n // Setup graceful shutdown\n const cleanup = () => {\n logger.info(\"正在关闭 MCP 服务代理\");\n proxy.stop();\n process.exit(0);\n };\n\n process.on(\"SIGINT\", cleanup);\n process.on(\"SIGTERM\", cleanup);\n\n try {\n // Start the proxy (initialize child clients)\n await proxy.start();\n\n // In MCP Server mode, announce readiness\n if (isMCPServerMode()) {\n logger.info(\"MCP proxy ready in server mode\");\n console.log(\"MCP proxy ready\"); // For parent process detection\n }\n\n // Handle stdin/stdout communication\n process.stdin.setEncoding(\"utf8\");\n\n let messageBuffer = \"\";\n\n process.stdin.on(\"data\", async (data) => {\n messageBuffer = `${messageBuffer}${data}`;\n\n // Split by newlines and process complete messages\n const lines = messageBuffer.split(\"\\n\");\n messageBuffer = lines.pop() || \"\"; // Keep incomplete line in buffer\n\n for (const line of lines) {\n if (line.trim()) {\n try {\n const response = await jsonrpcServer.handleMessage(line.trim());\n if (response) {\n process.stdout.write(`${response}\\n`);\n }\n } catch (error) {\n logger.error(\n `处理消息时出错:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n }\n }\n }\n });\n\n process.stdin.on(\"end\", () => {\n logger.info(\"标准输入已关闭,正在关闭\");\n cleanup();\n });\n\n logger.info(\"MCP 服务代理正在通过 stdio 运行\");\n } catch (error) {\n logger.error(\n `启动 MCP 服务代理失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n process.exit(1);\n }\n}\n\n// Run the server if this file is executed directly\n// Use fileURLToPath to properly handle Windows paths\nconst currentFileUrl = import.meta.url;\nconst scriptPath = fileURLToPath(currentFileUrl);\nconst argv1Path = process.argv[1];\n\nif (scriptPath === argv1Path) {\n main().catch((error) => {\n logger.error(\n `未处理的错误:${error instanceof Error ? error.message : String(error)}`\n );\n process.exit(1);\n });\n}\n","import { copyFileSync, existsSync, readFileSync, writeFileSync } from \"node:fs\";\nimport { dirname, resolve } from \"node:path\";\nimport { fileURLToPath } from \"node:url\";\n\n// 在 ESM 中,需要从 import.meta.url 获取当前文件目录\nconst __dirname = dirname(fileURLToPath(import.meta.url));\n\n// 默认连接配置\nconst DEFAULT_CONNECTION_CONFIG: Required<ConnectionConfig> = {\n heartbeatInterval: 30000, // 30秒心跳间隔\n heartbeatTimeout: 10000, // 10秒心跳超时\n reconnectInterval: 5000, // 5秒重连间隔\n};\n\n// 配置文件接口定义\n// 本地 MCP 服务配置\nexport interface LocalMCPServerConfig {\n command: string;\n args: string[];\n env?: Record<string, string>;\n}\n\n// SSE MCP 服务配置\nexport interface SSEMCPServerConfig {\n type: \"sse\";\n url: string;\n}\n\n// Streamable HTTP MCP 服务配置\nexport interface StreamableHTTPMCPServerConfig {\n type?: \"streamable-http\"; // 可选,因为默认就是 streamable-http\n url: string;\n}\n\n// 统一的 MCP 服务配置\nexport type MCPServerConfig =\n | LocalMCPServerConfig\n | SSEMCPServerConfig\n | StreamableHTTPMCPServerConfig;\n\nexport interface MCPToolConfig {\n description?: string;\n enable: boolean;\n}\n\nexport interface MCPServerToolsConfig {\n tools: Record<string, MCPToolConfig>;\n}\n\nexport interface ConnectionConfig {\n heartbeatInterval?: number; // 心跳检测间隔(毫秒),默认30000\n heartbeatTimeout?: number; // 心跳超时时间(毫秒),默认10000\n reconnectInterval?: number; // 重连间隔(毫秒),默认5000\n}\n\nexport interface ModelScopeConfig {\n apiKey?: string; // ModelScope API 密钥\n}\n\nexport interface WebUIConfig {\n port?: number; // Web UI 端口号,默认 9999\n autoRestart?: boolean; // 是否在配置更新后自动重启服务,默认 true\n}\n\nexport interface AppConfig {\n mcpEndpoint: string | string[];\n mcpServers: Record<string, MCPServerConfig>;\n mcpServerConfig?: Record<string, MCPServerToolsConfig>;\n connection?: ConnectionConfig; // 连接配置(可选,用于向后兼容)\n modelscope?: ModelScopeConfig; // ModelScope 配置(可选)\n webUI?: WebUIConfig; // Web UI 配置(可选)\n}\n\n/**\n * 配置管理类\n * 负责管理应用配置,提供只读访问和安全的配置更新功能\n */\nexport class ConfigManager {\n private static instance: ConfigManager;\n private defaultConfigPath: string;\n private config: AppConfig | null = null;\n\n private constructor() {\n this.defaultConfigPath = resolve(__dirname, \"xiaozhi.config.default.json\");\n }\n\n /**\n * 获取配置文件路径(动态计算)\n */\n private getConfigFilePath(): string {\n // 配置文件路径 - 优先使用环境变量指定的目录,否则使用当前工作目录\n const configDir = process.env.XIAOZHI_CONFIG_DIR || process.cwd();\n return resolve(configDir, \"xiaozhi.config.json\");\n }\n\n /**\n * 获取配置管理器单例实例\n */\n public static getInstance(): ConfigManager {\n if (!ConfigManager.instance) {\n ConfigManager.instance = new ConfigManager();\n }\n return ConfigManager.instance;\n }\n\n /**\n * 检查配置文件是否存在\n */\n public configExists(): boolean {\n const configPath = this.getConfigFilePath();\n return existsSync(configPath);\n }\n\n /**\n * 初始化配置文件\n * 从 config.default.json 复制到 config.json\n */\n public initConfig(): void {\n if (!existsSync(this.defaultConfigPath)) {\n throw new Error(\"默认配置文件 xiaozhi.config.default.json 不存在\");\n }\n\n if (this.configExists()) {\n throw new Error(\"配置文件 xiaozhi.config.json 已存在,无需重复初始化\");\n }\n\n const configPath = this.getConfigFilePath();\n copyFileSync(this.defaultConfigPath, configPath);\n this.config = null; // 重置缓存\n }\n\n /**\n * 加载配置文件\n */\n private loadConfig(): AppConfig {\n if (!this.configExists()) {\n throw new Error(\n \"配置文件 xiaozhi.config.json 不存在,请先运行 xiaozhi init 初始化配置\"\n );\n }\n\n try {\n const configPath = this.getConfigFilePath();\n const configData = readFileSync(configPath, \"utf8\");\n const config = JSON.parse(configData) as AppConfig;\n\n // 验证配置结构\n this.validateConfig(config);\n\n return config;\n } catch (error) {\n if (error instanceof SyntaxError) {\n throw new Error(`配置文件格式错误: ${error.message}`);\n }\n throw error;\n }\n }\n\n /**\n * 验证配置文件结构\n */\n private validateConfig(config: unknown): void {\n if (!config || typeof config !== \"object\") {\n throw new Error(\"配置文件格式错误:根对象无效\");\n }\n\n const configObj = config as Record<string, unknown>;\n\n if (configObj.mcpEndpoint === undefined || configObj.mcpEndpoint === null) {\n throw new Error(\"配置文件格式错误:mcpEndpoint 字段无效\");\n }\n\n // 验证 mcpEndpoint 类型(字符串或字符串数组)\n if (typeof configObj.mcpEndpoint === \"string\") {\n // 空字符串是允许的,getMcpEndpoints 会返回空数组\n } else if (Array.isArray(configObj.mcpEndpoint)) {\n if (configObj.mcpEndpoint.length === 0) {\n throw new Error(\"配置文件格式错误:mcpEndpoint 数组不能为空\");\n }\n for (const endpoint of configObj.mcpEndpoint) {\n if (typeof endpoint !== \"string\" || endpoint.trim() === \"\") {\n throw new Error(\n \"配置文件格式错误:mcpEndpoint 数组中的每个元素必须是非空字符串\"\n );\n }\n }\n } else {\n throw new Error(\"配置文件格式错误:mcpEndpoint 必须是字符串或字符串数组\");\n }\n\n if (!configObj.mcpServers || typeof configObj.mcpServers !== \"object\") {\n throw new Error(\"配置文件格式错误:mcpServers 字段无效\");\n }\n\n // 验证每个 MCP 服务配置\n for (const [serverName, serverConfig] of Object.entries(\n configObj.mcpServers as Record<string, unknown>\n )) {\n if (!serverConfig || typeof serverConfig !== \"object\") {\n throw new Error(`配置文件格式错误:mcpServers.${serverName} 无效`);\n }\n\n const sc = serverConfig as Record<string, unknown>;\n\n // 检查服务类型\n if (sc.url && typeof sc.url === \"string\") {\n // URL 类型的服务(SSE 或 Streamable HTTP)\n // type 字段是可选的,可以是 \"sse\" 或 \"streamable-http\"\n if (sc.type && sc.type !== \"sse\" && sc.type !== \"streamable-http\") {\n throw new Error(\n `配置文件格式错误:mcpServers.${serverName}.type 必须是 \"sse\" 或 \"streamable-http\"`\n );\n }\n } else {\n // 本地类型的验证\n if (!sc.command || typeof sc.command !== \"string\") {\n throw new Error(\n `配置文件格式错误:mcpServers.${serverName}.command 无效`\n );\n }\n\n if (!Array.isArray(sc.args)) {\n throw new Error(\n `配置文件格式错误:mcpServers.${serverName}.args 必须是数组`\n );\n }\n\n if (sc.env && typeof sc.env !== \"object\") {\n throw new Error(\n `配置文件格式错误:mcpServers.${serverName}.env 必须是对象`\n );\n }\n }\n }\n }\n\n /**\n * 获取配置(只读)\n */\n public getConfig(): Readonly<AppConfig> {\n if (!this.config) {\n this.config = this.loadConfig();\n }\n\n // 返回深度只读副本\n return JSON.parse(JSON.stringify(this.config));\n }\n\n /**\n * 获取 MCP 端点(向后兼容)\n * @deprecated 使用 getMcpEndpoints() 获取所有端点\n */\n public getMcpEndpoint(): string {\n const config = this.getConfig();\n if (Array.isArray(config.mcpEndpoint)) {\n return config.mcpEndpoint[0] || \"\";\n }\n return config.mcpEndpoint;\n }\n\n /**\n * 获取所有 MCP 端点\n */\n public getMcpEndpoints(): string[] {\n const config = this.getConfig();\n if (Array.isArray(config.mcpEndpoint)) {\n return [...config.mcpEndpoint];\n }\n return config.mcpEndpoint ? [config.mcpEndpoint] : [];\n }\n\n /**\n * 获取 MCP 服务配置\n */\n public getMcpServers(): Readonly<Record<string, MCPServerConfig>> {\n const config = this.getConfig();\n return config.mcpServers;\n }\n\n /**\n * 获取 MCP 服务工具配置\n */\n public getMcpServerConfig(): Readonly<Record<string, MCPServerToolsConfig>> {\n const config = this.getConfig();\n return config.mcpServerConfig || {};\n }\n\n /**\n * 获取指定服务的工具配置\n */\n public getServerToolsConfig(\n serverName: string\n ): Readonly<Record<string, MCPToolConfig>> {\n const serverConfig = this.getMcpServerConfig();\n return serverConfig[serverName]?.tools || {};\n }\n\n /**\n * 检查工具是否启用\n */\n public isToolEnabled(serverName: string, toolName: string): boolean {\n const toolsConfig = this.getServerToolsConfig(serverName);\n const toolConfig = toolsConfig[toolName];\n return toolConfig?.enable !== false; // 默认启用\n }\n\n /**\n * 更新 MCP 端点(支持字符串或数组)\n */\n public updateMcpEndpoint(endpoint: string | string[]): void {\n if (Array.isArray(endpoint)) {\n if (endpoint.length === 0) {\n throw new Error(\"MCP 端点数组不能为空\");\n }\n for (const ep of endpoint) {\n if (!ep || typeof ep !== \"string\") {\n throw new Error(\"MCP 端点数组中的每个元素必须是非空字符串\");\n }\n }\n } else {\n if (!endpoint || typeof endpoint !== \"string\") {\n throw new Error(\"MCP 端点必须是非空字符串\");\n }\n }\n\n const config = this.getConfig();\n const newConfig = { ...config, mcpEndpoint: endpoint };\n this.saveConfig(newConfig);\n }\n\n /**\n * 添加 MCP 端点\n */\n public addMcpEndpoint(endpoint: string): void {\n if (!endpoint || typeof endpoint !== \"string\") {\n throw new Error(\"MCP 端点必须是非空字符串\");\n }\n\n const config = this.getConfig();\n const currentEndpoints = this.getMcpEndpoints();\n\n // 检查是否已存在\n if (currentEndpoints.includes(endpoint)) {\n throw new Error(`MCP 端点 ${endpoint} 已存在`);\n }\n\n const newEndpoints = [...currentEndpoints, endpoint];\n const newConfig = { ...config, mcpEndpoint: newEndpoints };\n this.saveConfig(newConfig);\n }\n\n /**\n * 移除 MCP 端点\n */\n public removeMcpEndpoint(endpoint: string): void {\n if (!endpoint || typeof endpoint !== \"string\") {\n throw new Error(\"MCP 端点必须是非空字符串\");\n }\n\n const config = this.getConfig();\n const currentEndpoints = this.getMcpEndpoints();\n\n // 检查是否存在\n const index = currentEndpoints.indexOf(endpoint);\n if (index === -1) {\n throw new Error(`MCP 端点 ${endpoint} 不存在`);\n }\n\n // 不允许删除最后一个端点\n if (currentEndpoints.length === 1) {\n throw new Error(\"不能删除最后一个 MCP 端点\");\n }\n\n const newEndpoints = currentEndpoints.filter((ep) => ep !== endpoint);\n const newConfig = { ...config, mcpEndpoint: newEndpoints };\n this.saveConfig(newConfig);\n }\n\n /**\n * 更新 MCP 服务配置\n */\n public updateMcpServer(\n serverName: string,\n serverConfig: MCPServerConfig\n ): void {\n if (!serverName || typeof serverName !== \"string\") {\n throw new Error(\"服务名称必须是非空字符串\");\n }\n\n // 验证服务配置\n if (\"type\" in serverConfig && serverConfig.type === \"sse\") {\n // SSE 类型的验证\n if (!serverConfig.url || typeof serverConfig.url !== \"string\") {\n throw new Error(\"SSE 服务配置的 url 字段必须是非空字符串\");\n }\n } else {\n // 本地类型的验证\n const localConfig = serverConfig as LocalMCPServerConfig;\n if (!localConfig.command || typeof localConfig.command !== \"string\") {\n throw new Error(\"服务配置的 command 字段必须是非空字符串\");\n }\n\n if (!Array.isArray(localConfig.args)) {\n throw new Error(\"服务配置的 args 字段必须是数组\");\n }\n\n if (localConfig.env && typeof localConfig.env !== \"object\") {\n throw new Error(\"服务配置的 env 字段必须是对象\");\n }\n }\n\n const config = this.getConfig();\n const newConfig = {\n ...config,\n mcpServers: {\n ...config.mcpServers,\n [serverName]: serverConfig,\n },\n };\n this.saveConfig(newConfig);\n }\n\n /**\n * 删除 MCP 服务配置\n */\n public removeMcpServer(serverName: string): void {\n if (!serverName || typeof serverName !== \"string\") {\n throw new Error(\"服务名称必须是非空字符串\");\n }\n\n const config = this.getConfig();\n if (!config.mcpServers[serverName]) {\n throw new Error(`服务 ${serverName} 不存在`);\n }\n\n const newMcpServers = { ...config.mcpServers };\n delete newMcpServers[serverName];\n\n const newConfig = {\n ...config,\n mcpServers: newMcpServers,\n };\n this.saveConfig(newConfig);\n }\n\n /**\n * 更新服务工具配置\n */\n public updateServerToolsConfig(\n serverName: string,\n toolsConfig: Record<string, MCPToolConfig>\n ): void {\n const config = this.getConfig();\n const newConfig = { ...config };\n\n // 确保 mcpServerConfig 存在\n if (!newConfig.mcpServerConfig) {\n newConfig.mcpServerConfig = {};\n }\n\n // 更新指定服务的工具配置\n newConfig.mcpServerConfig[serverName] = {\n tools: toolsConfig,\n };\n\n this.saveConfig(newConfig);\n }\n\n /**\n * 设置工具启用状态\n */\n public setToolEnabled(\n serverName: string,\n toolName: string,\n enabled: boolean,\n description?: string\n ): void {\n const config = this.getConfig();\n const newConfig = { ...config };\n\n // 确保 mcpServerConfig 存在\n if (!newConfig.mcpServerConfig) {\n newConfig.mcpServerConfig = {};\n }\n\n // 确保服务配置存在\n if (!newConfig.mcpServerConfig[serverName]) {\n newConfig.mcpServerConfig[serverName] = { tools: {} };\n }\n\n // 更新工具配置\n newConfig.mcpServerConfig[serverName].tools[toolName] = {\n enable: enabled,\n ...(description && { description }),\n };\n\n this.saveConfig(newConfig);\n }\n\n /**\n * 保存配置到文件\n */\n private saveConfig(config: AppConfig): void {\n try {\n // 验证配置\n this.validateConfig(config);\n\n // 格式化 JSON 并保存\n const configPath = this.getConfigFilePath();\n const configJson = JSON.stringify(config, null, 2);\n writeFileSync(configPath, configJson, \"utf8\");\n\n // 更新缓存\n this.config = config;\n } catch (error) {\n throw new Error(\n `保存配置失败: ${error instanceof Error ? error.message : String(error)}`\n );\n }\n }\n\n /**\n * 重新加载配置(清除缓存)\n */\n public reloadConfig(): void {\n this.config = null;\n }\n\n /**\n * 获取配置文件路径\n */\n public getConfigPath(): string {\n return this.getConfigFilePath();\n }\n\n /**\n * 获取默认配置文件路径\n */\n public getDefaultConfigPath(): string {\n return this.defaultConfigPath;\n }\n\n /**\n * 获取连接配置(包含默认值)\n */\n public getConnectionConfig(): Required<ConnectionConfig> {\n const config = this.getConfig();\n const connectionConfig = config.connection || {};\n\n return {\n heartbeatInterval:\n connectionConfig.heartbeatInterval ??\n DEFAULT_CONNECTION_CONFIG.heartbeatInterval,\n heartbeatTimeout:\n connectionConfig.heartbeatTimeout ??\n DEFAULT_CONNECTION_CONFIG.heartbeatTimeout,\n reconnectInterval:\n connectionConfig.reconnectInterval ??\n DEFAULT_CONNECTION_CONFIG.reconnectInterval,\n };\n }\n\n /**\n * 获取心跳检测间隔(毫秒)\n */\n public getHeartbeatInterval(): number {\n return this.getConnectionConfig().heartbeatInterval;\n }\n\n /**\n * 获取心跳超时时间(毫秒)\n */\n public getHeartbeatTimeout(): number {\n return this.getConnectionConfig().heartbeatTimeout;\n }\n\n /**\n * 获取重连间隔(毫秒)\n */\n public getReconnectInterval(): number {\n return this.getConnectionConfig().reconnectInterval;\n }\n\n /**\n * 更新连接配置\n */\n public updateConnectionConfig(\n connectionConfig: Partial<ConnectionConfig>\n ): void {\n const config = this.getConfig();\n const currentConnectionConfig = config.connection || {};\n\n const newConnectionConfig = {\n ...currentConnectionConfig,\n ...connectionConfig,\n };\n\n const newConfig = {\n ...config,\n connection: newConnectionConfig,\n };\n\n this.saveConfig(newConfig);\n }\n\n /**\n * 设置心跳检测间隔\n */\n public setHeartbeatInterval(interval: number): void {\n if (interval <= 0) {\n throw new Error(\"心跳检测间隔必须大于0\");\n }\n this.updateConnectionConfig({ heartbeatInterval: interval });\n }\n\n /**\n * 设置心跳超时时间\n */\n public setHeartbeatTimeout(timeout: number): void {\n if (timeout <= 0) {\n throw new Error(\"心跳超时时间必须大于0\");\n }\n this.updateConnectionConfig({ heartbeatTimeout: timeout });\n }\n\n /**\n * 设置重连间隔\n */\n public setReconnectInterval(interval: number): void {\n if (interval <= 0) {\n throw new Error(\"重连间隔必须大于0\");\n }\n this.updateConnectionConfig({ reconnectInterval: interval });\n }\n\n /**\n * 获取 ModelScope 配置\n */\n public getModelScopeConfig(): Readonly<ModelScopeConfig> {\n const config = this.getConfig();\n return config.modelscope || {};\n }\n\n /**\n * 获取 ModelScope API Key\n * 优先从配置文件读取,其次从环境变量读取\n */\n public getModelScopeApiKey(): string | undefined {\n const modelScopeConfig = this.getModelScopeConfig();\n return modelScopeConfig.apiKey || process.env.MODELSCOPE_API_TOKEN;\n }\n\n /**\n * 更新 ModelScope 配置\n */\n public updateModelScopeConfig(\n modelScopeConfig: Partial<ModelScopeConfig>\n ): void {\n const config = this.getConfig();\n const currentModelScopeConfig = config.modelscope || {};\n\n const newModelScopeConfig = {\n ...currentModelScopeConfig,\n ...modelScopeConfig,\n };\n\n const newConfig = {\n ...config,\n modelscope: newModelScopeConfig,\n };\n\n this.saveConfig(newConfig);\n }\n\n /**\n * 设置 ModelScope API Key\n */\n public setModelScopeApiKey(apiKey: string): void {\n if (!apiKey || typeof apiKey !== \"string\") {\n throw new Error(\"API Key 必须是非空字符串\");\n }\n this.updateModelScopeConfig({ apiKey });\n }\n\n /**\n * 获取 Web UI 配置\n */\n public getWebUIConfig(): Readonly<WebUIConfig> {\n const config = this.getConfig();\n return config.webUI || {};\n }\n\n /**\n * 获取 Web UI 端口号\n */\n public getWebUIPort(): number {\n const webUIConfig = this.getWebUIConfig();\n return webUIConfig.port ?? 9999; // 默认端口 9999\n }\n\n /**\n * 更新 Web UI 配置\n */\n public updateWebUIConfig(webUIConfig: Partial<WebUIConfig>): void {\n const config = this.getConfig();\n const currentWebUIConfig = config.webUI || {};\n\n const newWebUIConfig = {\n ...currentWebUIConfig,\n ...webUIConfig,\n };\n\n const newConfig = {\n ...config,\n webUI: newWebUIConfig,\n };\n\n this.saveConfig(newConfig);\n }\n\n /**\n * 设置 Web UI 端口号\n */\n public setWebUIPort(port: number): void {\n if (!Number.isInteger(port) || port <= 0 || port > 65535) {\n throw new Error(\"端口号必须是 1-65535 之间的整数\");\n }\n this.updateWebUIConfig({ port });\n }\n}\n\n// 导出单例实例\nexport const configManager = ConfigManager.getInstance();\n","import fs from \"node:fs\";\nimport path from \"node:path\";\nimport chalk from \"chalk\";\nimport { type consola, createConsola } from \"consola\";\n\nfunction formatDateTime(date: Date) {\n const year = date.getFullYear();\n const month = String(date.getMonth() + 1).padStart(2, \"0\");\n const day = String(date.getDate()).padStart(2, \"0\");\n const hours = String(date.getHours()).padStart(2, \"0\");\n const minutes = String(date.getMinutes()).padStart(2, \"0\");\n const seconds = String(date.getSeconds()).padStart(2, \"0\");\n\n return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`;\n}\n\nexport class Logger {\n private logFilePath: string | null = null;\n private writeStream: fs.WriteStream | null = null;\n private consolaInstance: typeof consola;\n private isDaemonMode: boolean;\n\n constructor() {\n // 检查是否为守护进程模式\n this.isDaemonMode = process.env.XIAOZHI_DAEMON === \"true\";\n // 创建自定义的 consola 实例,禁用图标并自定义格式\n this.consolaInstance = createConsola({\n formatOptions: {\n date: false,\n colors: true,\n compact: true,\n },\n fancy: false,\n });\n\n // 保存对当前实例的引用,以便在闭包中访问\n const isDaemonMode = this.isDaemonMode;\n\n // 自定义格式化器\n this.consolaInstance.setReporters([\n {\n log: (logObj) => {\n const levelMap: Record<string, string> = {\n info: \"INFO\",\n success: \"SUCCESS\",\n warn: \"WARN\",\n error: \"ERROR\",\n debug: \"DEBUG\",\n log: \"LOG\",\n };\n\n const colorMap: Record<string, (text: string) => string> = {\n info: chalk.blue,\n success: chalk.green,\n warn: chalk.yellow,\n error: chalk.red,\n debug: chalk.gray,\n log: (text: string) => text,\n };\n\n const level = levelMap[logObj.type] || logObj.type.toUpperCase();\n const colorFn = colorMap[logObj.type] || ((text: string) => text);\n const timestamp = formatDateTime(new Date());\n\n // 为级别添加颜色\n const coloredLevel = colorFn(`[${level}]`);\n const message = `[${timestamp}] ${coloredLevel} ${logObj.args.join(\n \" \"\n )}`;\n\n // 守护进程模式下不输出到控制台,只写入文件\n if (!isDaemonMode) {\n // 输出到 stderr(与原来保持一致)\n try {\n console.error(message);\n } catch (error) {\n // 忽略 EPIPE 错误\n if (error instanceof Error && error.message?.includes(\"EPIPE\")) {\n return;\n }\n throw error;\n }\n }\n },\n },\n ]);\n }\n\n /**\n * 初始化日志文件\n * @param projectDir 项目目录\n */\n initLogFile(projectDir: string): void {\n this.logFilePath = path.join(projectDir, \"xiaozhi.log\");\n\n // 确保日志文件存在\n if (!fs.existsSync(this.logFilePath)) {\n fs.writeFileSync(this.logFilePath, \"\");\n }\n\n // 创建写入流,追加模式\n this.writeStream = fs.createWriteStream(this.logFilePath, {\n flags: \"a\",\n encoding: \"utf8\",\n });\n }\n\n /**\n * 记录日志到文件\n * @param level 日志级别\n * @param message 日志消息\n * @param args 额外参数\n */\n private logToFile(level: string, message: string, ...args: any[]): void {\n if (this.writeStream) {\n const timestamp = new Date().toISOString();\n const formattedMessage = `[${timestamp}] [${level.toUpperCase()}] ${message}`;\n const fullMessage =\n args.length > 0\n ? `${formattedMessage} ${args\n .map((arg) =>\n typeof arg === \"object\" ? JSON.stringify(arg) : String(arg)\n )\n .join(\" \")}`\n : formattedMessage;\n\n this.writeStream.write(`${fullMessage}\\n`);\n }\n }\n\n /**\n * 设置是否启用文件日志\n * @param enable 是否启用\n */\n enableFileLogging(enable: boolean): void {\n if (enable && !this.writeStream && this.logFilePath) {\n this.writeStream = fs.createWriteStream(this.logFilePath, {\n flags: \"a\",\n encoding: \"utf8\",\n });\n } else if (!enable && this.writeStream) {\n this.writeStream.end();\n this.writeStream = null;\n }\n }\n\n /**\n * 日志方法\n */\n info(message: string, ...args: any[]): void {\n this.consolaInstance.info(message, ...args);\n this.logToFile(\"info\", message, ...args);\n }\n\n success(message: string, ...args: any[]): void {\n this.consolaInstance.success(message, ...args);\n this.logToFile(\"success\", message, ...args);\n }\n\n warn(message: string, ...args: any[]): void {\n this.consolaInstance.warn(message, ...args);\n this.logToFile(\"warn\", message, ...args);\n }\n\n error(message: string, ...args: any[]): void {\n this.consolaInstance.error(message, ...args);\n this.logToFile(\"error\", message, ...args);\n }\n\n debug(message: string, ...args: any[]): void {\n this.consolaInstance.debug(message, ...args);\n this.logToFile(\"debug\", message, ...args);\n }\n\n log(message: string, ...args: any[]): void {\n this.consolaInstance.log(message, ...args);\n this.logToFile(\"log\", message, ...args);\n }\n\n /**\n * 创建一个带标签的日志实例(已废弃,直接返回原实例)\n * @param tag 标签(不再使用)\n * @deprecated 标签功能已移除\n */\n withTag(tag: string): Logger {\n // 不再添加标签,直接返回共享实例\n return this;\n }\n\n /**\n * 关闭日志文件流\n */\n close(): void {\n if (this.writeStream) {\n this.writeStream.end();\n this.writeStream = null;\n }\n }\n}\n\n// 导出单例实例\nexport const logger = new Logger();\n","import { Client } from \"@modelcontextprotocol/sdk/client/index.js\";\nimport { SSEClientTransport } from \"@modelcontextprotocol/sdk/client/sse.js\";\nimport { EventSource } from \"eventsource\";\nimport {\n type MCPToolConfig,\n type SSEMCPServerConfig,\n configManager,\n} from \"./configManager\";\nimport { logger as globalLogger } from \"./logger\";\nimport type { IMCPClient } from \"./mcpServerProxy\";\n\n// 全局 polyfill EventSource\n(global as any).EventSource = EventSource;\n\n// 为 ModelScope MCP 创建带标签的 logger\nconst logger = globalLogger.withTag(\"ModelScopeMCP\");\n\ninterface Tool {\n name: string;\n description?: string;\n inputSchema?: any;\n}\n\n/**\n * ModelScope MCP Client for SSE connections\n */\nexport class ModelScopeMCPClient implements IMCPClient {\n private name: string;\n private config: SSEMCPServerConfig;\n private client: Client | null = null;\n private transport: SSEClientTransport | null = null;\n public initialized = false;\n public tools: Tool[] = [];\n public originalTools: Tool[] = [];\n\n constructor(name: string, config: SSEMCPServerConfig) {\n this.name = name;\n this.config = config;\n }\n\n /**\n * 生成带前缀的工具名称\n */\n private generatePrefixedToolName(originalToolName: string): string {\n const normalizedServerName = this.name.replace(/-/g, \"_\");\n return `${normalizedServerName}_xzcli_${originalToolName}`;\n }\n\n /**\n * 根据前缀工具名称获取原始工具名称\n */\n getOriginalToolName(prefixedToolName: string): string | null {\n const normalizedServerName = this.name.replace(/-/g, \"_\");\n const prefix = `${normalizedServerName}_xzcli_`;\n\n if (prefixedToolName.startsWith(prefix)) {\n return prefixedToolName.substring(prefix.length);\n }\n\n return null;\n }\n\n async start() {\n logger.info(`正在启动 ModelScope MCP 客户端:${this.name}`);\n\n try {\n // 从配置或环境变量获取 API Token\n const token = configManager.getModelScopeApiKey();\n if (!token || token === \"\") {\n throw new Error(\n \"未设置 ModelScope API Key。请在配置文件中设置 modelscope.apiKey 或设置 MODELSCOPE_API_TOKEN 环境变量。\"\n );\n }\n\n // 创建 SSE 传输层\n const sseOptions = {\n eventSourceInit: {\n fetch: async (url: string | URL | Request, init?: RequestInit) => {\n // 添加认证头\n const headers = {\n ...init?.headers,\n Authorization: `Bearer ${token}`,\n };\n\n return fetch(url, { ...init, headers });\n },\n },\n requestInit: {\n headers: {\n Authorization: `Bearer ${token}`,\n },\n },\n };\n\n this.transport = new SSEClientTransport(\n new URL(this.config.url),\n sseOptions\n );\n\n // 创建 MCP 客户端\n this.client = new Client(\n {\n name: \"xiaozhi-modelscope-client\",\n version: \"1.0.0\",\n },\n {\n capabilities: {},\n }\n );\n\n // 连接到服务器\n logger.info(`正在连接到 ${this.config.url}`);\n await this.client.connect(this.transport);\n logger.info(`成功连接到 ModelScope MCP 服务器:${this.name}`);\n\n // 获取工具列表\n await this.refreshTools();\n\n this.initialized = true;\n logger.info(\n `${this.name} ModelScope 客户端已就绪,共 ${this.tools.length} 个工具`\n );\n } catch (error) {\n logger.error(\n `启动 ModelScope MCP 客户端 ${this.name} 失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n throw error;\n }\n }\n\n async refreshTools(): Promise<void> {\n try {\n if (!this.client) {\n throw new Error(\"客户端未初始化\");\n }\n\n const result = await this.client.listTools();\n this.originalTools = result.tools || [];\n\n // 为每个工具生成带前缀的名称\n const allPrefixedTools = this.originalTools.map((tool) => ({\n ...tool,\n name: this.generatePrefixedToolName(tool.name),\n }));\n\n // 根据配置过滤工具\n this.tools = this.filterEnabledTools(allPrefixedTools);\n\n // 更新配置文件中的工具列表(如果需要)\n await this.updateToolsConfig();\n\n logger.info(\n `${this.name} 加载了 ${\n this.originalTools.length\n } 个工具:${this.originalTools.map((t) => t.name).join(\", \")}`\n );\n logger.info(\n `${this.name} 启用了 ${this.tools.length} 个工具:${this.tools\n .map((t) => t.name)\n .join(\", \")}`\n );\n } catch (error) {\n logger.error(\n `从 ${this.name} 获取工具失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n this.tools = [];\n this.originalTools = [];\n }\n }\n\n /**\n * 过滤启用的工具\n */\n private filterEnabledTools(allTools: Tool[]): Tool[] {\n return allTools.filter((tool) => {\n const originalName = this.getOriginalToolName(tool.name);\n if (!originalName) return true; // 如果无法解析原始名称,默认启用\n\n return configManager.isToolEnabled(this.name, originalName);\n });\n }\n\n /**\n * 更新配置文件中的工具列表\n */\n private async updateToolsConfig(): Promise<void> {\n try {\n const currentConfig = configManager.getServerToolsConfig(this.name);\n const toolsConfig: Record<string, MCPToolConfig> = {};\n\n // 为每个工具创建配置项\n for (const tool of this.originalTools) {\n const existingConfig = currentConfig[tool.name];\n toolsConfig[tool.name] = {\n description: tool.description || \"\",\n enable: existingConfig?.enable !== false, // 默认启用\n };\n }\n\n // 只有当配置发生变化时才更新\n const hasChanges = Object.keys(toolsConfig).some((toolName) => {\n const existing = currentConfig[toolName];\n const newConfig = toolsConfig[toolName];\n return (\n !existing ||\n existing.enable !== newConfig.enable ||\n existing.description !== newConfig.description\n );\n });\n\n if (hasChanges || Object.keys(currentConfig).length === 0) {\n configManager.updateServerToolsConfig(this.name, toolsConfig);\n logger.info(`${this.name} 已更新工具配置`);\n }\n } catch (error) {\n logger.error(\n `更新 ${this.name} 的工具配置失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n }\n }\n\n async callTool(prefixedName: string, arguments_: any): Promise<any> {\n try {\n if (!this.client) {\n throw new Error(\"客户端未初始化\");\n }\n\n // 将前缀名称转换回原始名称\n const originalName = this.getOriginalToolName(prefixedName);\n if (!originalName) {\n throw new Error(`无效的工具名称格式:${prefixedName}`);\n }\n\n logger.info(\n `调用 ModelScope 工具 ${originalName},参数:${JSON.stringify(\n arguments_\n )}`\n );\n\n const result = await this.client.callTool({\n name: originalName,\n arguments: arguments_,\n });\n\n logger.info(\n `ModelScope 工具调用返回: ${JSON.stringify(result).substring(0, 500)}...`\n );\n\n return result;\n } catch (error) {\n logger.error(\n `在 ${this.name} 上调用工具 ${prefixedName} 失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n throw error;\n }\n }\n\n async stop(): Promise<void> {\n if (this.client) {\n logger.info(`正在停止 ${this.name} ModelScope 客户端`);\n try {\n await this.client.close();\n } catch (error) {\n logger.error(\n `关闭客户端时出错:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n }\n this.client = null;\n this.transport = null;\n }\n this.initialized = false;\n }\n}\n","import { Client } from \"@modelcontextprotocol/sdk/client/index.js\";\nimport { SSEClientTransport } from \"@modelcontextprotocol/sdk/client/sse.js\";\nimport { EventSource } from \"eventsource\";\nimport {\n type MCPToolConfig,\n type SSEMCPServerConfig,\n configManager,\n} from \"./configManager\";\nimport { logger as globalLogger } from \"./logger\";\nimport type { IMCPClient } from \"./mcpServerProxy\";\n\n// 全局 polyfill EventSource\n(global as any).EventSource = EventSource;\n\n// 为通用 SSE MCP 创建带标签的 logger\nconst logger = globalLogger.withTag(\"SSEMCP\");\n\ninterface Tool {\n name: string;\n description?: string;\n inputSchema?: any;\n}\n\n/**\n * 通用 SSE MCP Client\n * 用于连接基于 SSE 的 MCP 服务,支持不需要特殊认证的服务\n */\nexport class SSEMCPClient implements IMCPClient {\n private name: string;\n private config: SSEMCPServerConfig;\n private client: Client | null = null;\n private transport: SSEClientTransport | null = null;\n public initialized = false;\n public tools: Tool[] = [];\n public originalTools: Tool[] = [];\n\n constructor(name: string, config: SSEMCPServerConfig) {\n this.name = name;\n this.config = config;\n }\n\n /**\n * 生成带前缀的工具名称\n */\n private generatePrefixedToolName(originalToolName: string): string {\n const normalizedServerName = this.name.replace(/-/g, \"_\");\n return `${normalizedServerName}_xzcli_${originalToolName}`;\n }\n\n /**\n * 过滤启用的工具\n */\n private filterEnabledTools(allTools: Tool[]): Tool[] {\n return allTools.filter((tool) => {\n // 从前缀名称中提取原始名称\n const originalName = this.getOriginalToolName(tool.name);\n if (!originalName) return false;\n\n return configManager.isToolEnabled(this.name, originalName);\n });\n }\n\n /**\n * 更新配置文件中的工具列表\n */\n private async updateToolsConfig(): Promise<void> {\n try {\n const currentConfig = configManager.getServerToolsConfig(this.name);\n const newToolsConfig: Record<string, MCPToolConfig> = {};\n\n // 为每个工具创建配置项(如果不存在)\n for (const tool of this.originalTools) {\n const existingConfig = currentConfig[tool.name];\n newToolsConfig[tool.name] = {\n description: tool.description || existingConfig?.description || \"\",\n enable: existingConfig?.enable !== false, // 默认启用\n };\n }\n\n // 保留现有工具的配置\n for (const [toolName, toolConfig] of Object.entries(currentConfig)) {\n if (!newToolsConfig[toolName]) {\n newToolsConfig[toolName] = toolConfig;\n }\n }\n\n configManager.updateServerToolsConfig(this.name, newToolsConfig);\n } catch (error) {\n logger.debug(\n `更新 ${this.name} 工具配置失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n }\n }\n\n async start(): Promise<void> {\n logger.info(`正在启动 SSE MCP 客户端:${this.name}`);\n\n try {\n // 创建 SSE 传输层(不需要特殊认证)\n this.transport = new SSEClientTransport(new URL(this.config.url));\n\n // 创建 MCP 客户端\n this.client = new Client(\n {\n name: \"xiaozhi-sse-client\",\n version: \"1.0.0\",\n },\n {\n capabilities: {},\n }\n );\n\n // 连接到服务器\n logger.info(`正在连接到 ${this.config.url}`);\n await this.client.connect(this.transport);\n logger.info(`成功连接到 SSE MCP 服务器:${this.name}`);\n\n // 获取工具列表\n await this.refreshTools();\n\n this.initialized = true;\n logger.info(\n `${this.name} SSE 客户端已就绪,共 ${this.tools.length} 个工具`\n );\n } catch (error) {\n logger.error(\n `启动 SSE MCP 客户端 ${this.name} 失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n throw error;\n }\n }\n\n async refreshTools(): Promise<void> {\n try {\n if (!this.client) {\n throw new Error(\"客户端未初始化\");\n }\n\n const result = await this.client.listTools();\n this.originalTools = result.tools || [];\n\n // 为每个工具生成带前缀的名称\n const allPrefixedTools = this.originalTools.map((tool) => ({\n ...tool,\n name: this.generatePrefixedToolName(tool.name),\n }));\n\n // 根据配置过滤工具\n this.tools = this.filterEnabledTools(allPrefixedTools);\n\n // 更新配置文件中的工具列表(如果需要)\n await this.updateToolsConfig();\n\n logger.info(\n `${this.name} 加载了 ${\n this.originalTools.length\n } 个工具:${this.originalTools.map((t) => t.name).join(\", \")}`\n );\n logger.info(\n `${this.name} 启用了 ${this.tools.length} 个工具:${this.tools\n .map((t) => t.name)\n .join(\", \")}`\n );\n } catch (error) {\n logger.error(\n `刷新 ${this.name} 工具列表失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n throw error;\n }\n }\n\n /**\n * 从带前缀的工具名称中提取原始工具名称\n */\n getOriginalToolName(prefixedName: string): string | null {\n const normalizedServerName = this.name.replace(/-/g, \"_\");\n const prefix = `${normalizedServerName}_xzcli_`;\n\n if (prefixedName.startsWith(prefix)) {\n return prefixedName.substring(prefix.length);\n }\n\n return null;\n }\n\n async callTool(prefixedName: string, arguments_: any): Promise<any> {\n try {\n if (!this.client) {\n throw new Error(\"客户端未初始化\");\n }\n\n // 将前缀名称转换回原始名称\n const originalName = this.getOriginalToolName(prefixedName);\n if (!originalName) {\n throw new Error(`无效的工具名称格式:${prefixedName}`);\n }\n\n logger.info(\n `调用 SSE 工具 ${originalName},参数:${JSON.stringify(arguments_)}`\n );\n\n const result = await this.client.callTool({\n name: originalName,\n arguments: arguments_,\n });\n\n logger.info(\n `SSE 工具调用返回: ${JSON.stringify(result).substring(0, 500)}...`\n );\n\n return result;\n } catch (error) {\n logger.error(\n `在 ${this.name} 上调用工具 ${prefixedName} 失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n throw error;\n }\n }\n\n async stop(): Promise<void> {\n logger.info(`正在停止 SSE MCP 客户端:${this.name}`);\n\n try {\n if (this.client) {\n await this.client.close();\n this.client = null;\n }\n\n if (this.transport) {\n this.transport = null;\n }\n\n logger.info(`SSE MCP 客户端 ${this.name} 已停止`);\n } catch (error) {\n logger.error(\n `停止 SSE MCP 客户端 ${this.name} 失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n } finally {\n // 无论是否出错,都要将状态设置为未初始化\n this.initialized = false;\n }\n }\n}\n","import {\n type MCPToolConfig,\n type StreamableHTTPMCPServerConfig,\n configManager,\n} from \"./configManager\";\nimport { logger as globalLogger } from \"./logger\";\nimport type { IMCPClient } from \"./mcpServerProxy\";\n\n// 为 StreamableHTTP MCP 创建带标签的 logger\nconst logger = globalLogger.withTag(\"StreamableHTTPMCP\");\n\ninterface Tool {\n name: string;\n description?: string;\n inputSchema?: any;\n}\n\ninterface JSONRPCRequest {\n jsonrpc: \"2.0\";\n method: string;\n params?: any;\n id: number | string;\n}\n\ninterface JSONRPCResponse {\n jsonrpc: \"2.0\";\n result?: any;\n error?: {\n code: number;\n message: string;\n data?: any;\n };\n id: number | string;\n}\n\n/**\n * Streamable HTTP MCP Client\n * 用于连接基于 HTTP 的 MCP 服务,如高德地图 MCP\n */\nexport class StreamableHTTPMCPClient implements IMCPClient {\n private name: string;\n private config: StreamableHTTPMCPServerConfig;\n private requestId = 1;\n public initialized = false;\n public tools: Tool[] = [];\n public originalTools: Tool[] = [];\n\n constructor(name: string, config: StreamableHTTPMCPServerConfig) {\n this.name = name;\n this.config = config;\n }\n\n /**\n * 生成带前缀的工具名称\n */\n private generatePrefixedToolName(originalToolName: string): string {\n const normalizedServerName = this.name.replace(/-/g, \"_\");\n return `${normalizedServerName}_xzcli_${originalToolName}`;\n }\n\n /**\n * 根据前缀工具名称获取原始工具名称\n */\n getOriginalToolName(prefixedToolName: string): string | null {\n const normalizedServerName = this.name.replace(/-/g, \"_\");\n const prefix = `${normalizedServerName}_xzcli_`;\n\n if (prefixedToolName.startsWith(prefix)) {\n return prefixedToolName.substring(prefix.length);\n }\n\n return null;\n }\n\n /**\n * 发送 JSON-RPC 请求到 HTTP 端点\n */\n private async sendRequest(method: string, params?: any): Promise<any> {\n const request: JSONRPCRequest = {\n jsonrpc: \"2.0\",\n method,\n params,\n id: this.requestId++,\n };\n\n logger.debug(`发送请求到 ${this.name}: ${JSON.stringify(request)}`);\n\n try {\n // 动态导入 node-fetch\n const fetch = (await import(\"node-fetch\")).default;\n\n const response = await fetch(this.config.url, {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/json\",\n Accept: \"application/json, text/event-stream\",\n },\n body: JSON.stringify(request),\n });\n\n if (!response.ok) {\n throw new Error(`HTTP error! status: ${response.status}`);\n }\n\n const result = (await response.json()) as JSONRPCResponse;\n\n if (result.error) {\n throw new Error(\n `JSON-RPC error: ${result.error.message} (code: ${result.error.code})`\n );\n }\n\n return result.result;\n } catch (error) {\n logger.error(\n `请求失败 (${method}): ${\n error instanceof Error ? error.message : String(error)\n }`\n );\n throw error;\n }\n }\n\n /**\n * 发送 JSON-RPC 通知到 HTTP 端点(不包含 id 字段,不期望响应)\n */\n private async sendNotification(method: string, params?: any): Promise<void> {\n const notification = {\n jsonrpc: \"2.0\",\n method,\n params,\n // 注意:通知不包含 id 字段\n };\n\n logger.debug(`发送通知到 ${this.name}: ${JSON.stringify(notification)}`);\n\n try {\n // 动态导入 node-fetch\n const fetch = (await import(\"node-fetch\")).default;\n\n const response = await fetch(this.config.url, {\n method: \"POST\",\n headers: {\n \"Content-Type\": \"application/json\",\n Accept: \"application/json, text/event-stream\",\n },\n body: JSON.stringify(notification),\n });\n\n if (!response.ok) {\n throw new Error(`HTTP error! status: ${response.status}`);\n }\n\n // 对于通知,我们不期望有意义的响应,只要没有HTTP错误就认为成功\n logger.debug(`通知 ${method} 发送成功`);\n } catch (error) {\n logger.debug(\n `通知发送失败 (${method}): ${\n error instanceof Error ? error.message : String(error)\n }`\n );\n throw error;\n }\n }\n\n async start() {\n logger.info(`正在启动 Streamable HTTP MCP 客户端:${this.name}`);\n\n try {\n // 初始化连接\n await this.sendRequest(\"initialize\", {\n protocolVersion: \"2024-11-05\",\n capabilities: {\n tools: {},\n },\n clientInfo: {\n name: \"xiaozhi-streamable-http-client\",\n version: \"1.0.0\",\n },\n });\n\n // 发送初始化完成通知(作为通知发送,不是请求)\n try {\n await this.sendNotification(\"notifications/initialized\");\n } catch (error) {\n // 忽略此错误,因为某些服务(如高德地图)不支持此通知\n logger.debug(`${this.name} 不支持 notifications/initialized: ${error}`);\n }\n\n // 获取工具列表\n const listToolsResult = await this.sendRequest(\"tools/list\");\n\n if (listToolsResult?.tools) {\n this.originalTools = listToolsResult.tools;\n\n // 更新配置文件中的工具列表\n await this.updateToolsConfig();\n\n // 生成带前缀的工具名称\n this.tools = this.originalTools\n .filter((tool) => configManager.isToolEnabled(this.name, tool.name))\n .map((tool) => ({\n ...tool,\n name: this.generatePrefixedToolName(tool.name),\n }));\n\n logger.info(\n `Streamable HTTP MCP 客户端 ${this.name} 已初始化,包含 ${this.tools.length}/${this.originalTools.length} 个已启用的工具`\n );\n }\n\n this.initialized = true;\n } catch (error) {\n logger.error(\n `启动 Streamable HTTP MCP 客户端 ${this.name} 失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n throw error;\n }\n }\n\n async refreshTools() {\n try {\n const listToolsResult = await this.sendRequest(\"tools/list\");\n\n if (listToolsResult?.tools) {\n this.originalTools = listToolsResult.tools;\n\n // 更新配置文件中的工具列表\n await this.updateToolsConfig();\n\n // 重新生成带前缀的工具名称\n this.tools = this.originalTools\n .filter((tool) => configManager.isToolEnabled(this.name, tool.name))\n .map((tool) => ({\n ...tool,\n name: this.generatePrefixedToolName(tool.name),\n }));\n\n logger.info(\n `已刷新 ${this.name} 的工具列表,包含 ${this.tools.length}/${this.originalTools.length} 个已启用的工具`\n );\n }\n } catch (error) {\n logger.error(\n `刷新 ${this.name} 的工具列表失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n }\n }\n\n async callTool(prefixedName: string, arguments_: any): Promise<any> {\n try {\n // 将前缀名称转换回原始名称\n const originalName = this.getOriginalToolName(prefixedName);\n if (!originalName) {\n throw new Error(`无效的工具名称格式:${prefixedName}`);\n }\n\n const result = await this.sendRequest(\"tools/call\", {\n name: originalName,\n arguments: arguments_,\n });\n\n return result;\n } catch (error) {\n logger.error(\n `在 ${this.name} 上调用工具 ${prefixedName} 失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n throw error;\n }\n }\n\n /**\n * 更新配置文件中的工具列表\n */\n private async updateToolsConfig(): Promise<void> {\n try {\n const currentConfig = configManager.getServerToolsConfig(this.name);\n const toolsConfig: Record<string, MCPToolConfig> = {};\n\n // 为每个工具创建配置项\n for (const tool of this.originalTools) {\n const existingConfig = currentConfig[tool.name];\n toolsConfig[tool.name] = {\n description: tool.description || \"\",\n enable: existingConfig?.enable !== false, // 默认启用\n };\n }\n\n // 只有当配置发生变化时才更新\n const hasChanges = Object.keys(toolsConfig).some((toolName) => {\n const existing = currentConfig[toolName];\n const newConfig = toolsConfig[toolName];\n return (\n !existing ||\n existing.enable !== newConfig.enable ||\n existing.description !== newConfig.description\n );\n });\n\n if (hasChanges || Object.keys(currentConfig).length === 0) {\n configManager.updateServerToolsConfig(this.name, toolsConfig);\n logger.info(`${this.name} 已更新工具配置`);\n }\n } catch (error) {\n logger.error(\n `更新 ${this.name} 的工具配置失败:${\n error instanceof Error ? error.message : String(error)\n }`\n );\n }\n }\n\n async stop() {\n logger.info(`正在停止 ${this.name} 客户端`);\n this.initialized = false;\n }\n}\n"],"mappings":"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","names":["spawn","readFileSync","dirname","resolve","process","fileURLToPath","copyFileSync","existsSync","readFileSync","writeFileSync","dirname","resolve","fileURLToPath","__dirname","dirname","fileURLToPath","DEFAULT_CONNECTION_CONFIG","ConfigManager","_ConfigManager","__name","resolve","configDir","configPath","existsSync","copyFileSync","configData","readFileSync","config","error","configObj","endpoint","serverName","serverConfig","sc","toolName","ep","newConfig","currentEndpoints","newEndpoints","localConfig","newMcpServers","toolsConfig","enabled","description","configJson","writeFileSync","connectionConfig","newConnectionConfig","interval","timeout","modelScopeConfig","newModelScopeConfig","apiKey","webUIConfig","newWebUIConfig","port","configManager","fs","path","chalk","createConsola","formatDateTime","date","year","month","day","hours","minutes","seconds","__name","Logger","createConsola","isDaemonMode","logObj","levelMap","colorMap","chalk","text","level","colorFn","timestamp","coloredLevel","message","error","projectDir","path","fs","args","formattedMessage","fullMessage","arg","enable","tag","logger","Client","SSEClientTransport","EventSource","EventSource","logger","ModelScopeMCPClient","__name","name","config","originalToolName","prefixedToolName","prefix","token","configManager","sseOptions","url","init","headers","SSEClientTransport","Client","error","result","allPrefixedTools","tool","t","allTools","originalName","currentConfig","toolsConfig","existingConfig","toolName","existing","newConfig","prefixedName","arguments_","Client","SSEClientTransport","EventSource","EventSource","logger","SSEMCPClient","__name","name","config","originalToolName","allTools","tool","originalName","configManager","currentConfig","newToolsConfig","existingConfig","toolName","toolConfig","error","SSEClientTransport","Client","result","allPrefixedTools","t","prefixedName","prefix","arguments_","logger","StreamableHTTPMCPClient","__name","name","config","originalToolName","prefixedToolName","prefix","method","params","request","fetch","response","result","error","notification","listToolsResult","tool","configManager","prefixedName","arguments_","originalName","currentConfig","toolsConfig","existingConfig","toolName","existing","newConfig","__dirname","dirname","fileURLToPath","logger","logDir","process","MCPClient","__name","name","config","command","args","originalToolName","prefixedToolName","prefix","env","resolvedCommand","resolvedArgs","spawnOptions","userWorkingDir","spawn","data","code","signal","error","lines","line","message","pendingRequest","resolve","reject","method","params","id","request","initResult","notification","result","allPrefixedTools","tool","t","allTools","originalName","configManager","currentConfig","toolsConfig","existingConfig","toolName","existing","newConfig","prefixedName","arguments_","startTime","duration","loadMCPConfig","mcpServers","legacyConfigPath","readFileSync","configData","MCPServerProxy","clientPromises","serverName","serverConfig","client","url","ModelScopeMCPClient","SSEMCPClient","StreamableHTTPMCPClient","results","successCount","clientName","servers","refreshPromises","JSONRPCServer","proxy","parsedMessage","response","_params","tools","isMCPServerMode","main","jsonrpcServer","cleanup","messageBuffer","currentFileUrl","scriptPath","argv1Path"]}
|
|
@@ -28,6 +28,14 @@ declare class ModelScopeMCPClient implements IMCPClient {
|
|
|
28
28
|
getOriginalToolName(prefixedToolName: string): string | null;
|
|
29
29
|
start(): Promise<void>;
|
|
30
30
|
refreshTools(): Promise<void>;
|
|
31
|
+
/**
|
|
32
|
+
* 过滤启用的工具
|
|
33
|
+
*/
|
|
34
|
+
private filterEnabledTools;
|
|
35
|
+
/**
|
|
36
|
+
* 更新配置文件中的工具列表
|
|
37
|
+
*/
|
|
38
|
+
private updateToolsConfig;
|
|
31
39
|
callTool(prefixedName: string, arguments_: any): Promise<any>;
|
|
32
40
|
stop(): Promise<void>;
|
|
33
41
|
}
|