@xiaozhi-client/endpoint 1.10.9 → 2.0.0-beta.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/index.d.ts +17 -22
- package/dist/index.js +3 -19
- package/dist/index.js.map +1 -1
- package/package.json +3 -3
package/dist/index.d.ts
CHANGED
|
@@ -1,6 +1,20 @@
|
|
|
1
|
+
import { JSONSchema } from '@xiaozhi-client/mcp-core';
|
|
2
|
+
export { JSONSchema, ensureToolJSONSchema } from '@xiaozhi-client/mcp-core';
|
|
1
3
|
import { Tool } from '@modelcontextprotocol/sdk/types.js';
|
|
2
4
|
import { EventEmitter } from 'node:events';
|
|
3
5
|
|
|
6
|
+
/**
|
|
7
|
+
* Endpoint 包核心类型定义
|
|
8
|
+
*
|
|
9
|
+
* 定义小智接入点相关的所有核心类型,包括:
|
|
10
|
+
* - 工具调用相关类型(ToolCallResult、ToolCallParams 等)
|
|
11
|
+
* - JSON Schema 类型定义
|
|
12
|
+
* - 工具信息类型(EnhancedToolInfo 等)
|
|
13
|
+
* - MCP 服务配置类型
|
|
14
|
+
* - 连接状态类型
|
|
15
|
+
*
|
|
16
|
+
* @module types
|
|
17
|
+
*/
|
|
4
18
|
/**
|
|
5
19
|
* 工具调用结果接口
|
|
6
20
|
* 使用更宽松的类型定义以兼容不同来源的 ToolCallResult
|
|
@@ -49,26 +63,7 @@ declare class ToolCallError extends Error {
|
|
|
49
63
|
data?: unknown | undefined;
|
|
50
64
|
constructor(code: ToolCallErrorCode, message: string, data?: unknown | undefined);
|
|
51
65
|
}
|
|
52
|
-
|
|
53
|
-
* JSON Schema 类型定义
|
|
54
|
-
* 兼容 MCP SDK 的 JSON Schema 格式
|
|
55
|
-
*/
|
|
56
|
-
type JSONSchema = (Record<string, unknown> & {
|
|
57
|
-
type: "object";
|
|
58
|
-
properties?: Record<string, unknown>;
|
|
59
|
-
required?: string[];
|
|
60
|
-
additionalProperties?: boolean;
|
|
61
|
-
}) | Record<string, unknown>;
|
|
62
|
-
/**
|
|
63
|
-
* 确保对象符合 MCP Tool JSON Schema 格式
|
|
64
|
-
* 返回类型兼容 MCP SDK 的 Tool 类型
|
|
65
|
-
*/
|
|
66
|
-
declare function ensureToolJSONSchema(schema: JSONSchema): {
|
|
67
|
-
type: "object";
|
|
68
|
-
properties?: Record<string, object>;
|
|
69
|
-
required?: string[];
|
|
70
|
-
additionalProperties?: boolean;
|
|
71
|
-
};
|
|
66
|
+
|
|
72
67
|
/**
|
|
73
68
|
* 增强的工具信息接口
|
|
74
69
|
* 包含工具的启用状态和使用统计信息
|
|
@@ -502,7 +497,7 @@ declare class EndpointManager extends EventEmitter {
|
|
|
502
497
|
*
|
|
503
498
|
* @param endpoint - Endpoint 实例
|
|
504
499
|
*/
|
|
505
|
-
removeEndpoint(endpoint: Endpoint): void
|
|
500
|
+
removeEndpoint(endpoint: Endpoint): Promise<void>;
|
|
506
501
|
/**
|
|
507
502
|
* 连接 Endpoint
|
|
508
503
|
*
|
|
@@ -800,4 +795,4 @@ interface ExtendedMCPMessage {
|
|
|
800
795
|
};
|
|
801
796
|
}
|
|
802
797
|
|
|
803
|
-
export { type ConfigChangeEvent, type ConnectionOptions, ConnectionState, type ConnectionStatus, Endpoint, type EndpointConfig, type EndpointConnectionStatus, EndpointManager, type EndpointManagerConfig, type ExtendedMCPMessage, type IMCPServiceManager, type
|
|
798
|
+
export { type ConfigChangeEvent, type ConnectionOptions, ConnectionState, type ConnectionStatus, Endpoint, type EndpointConfig, type EndpointConnectionStatus, EndpointManager, type EndpointManagerConfig, type ExtendedMCPMessage, type IMCPServiceManager, type LocalMCPServerConfig, type MCPMessage, type MCPServerConfig, type ParsedEndpointInfo, type ReconnectResult, type SSEMCPServerConfig, SharedMCPAdapter, type SimpleConnectionStatus, type StreamableHTTPMCPServerConfig, ToolCallError, ToolCallErrorCode, type ToolCallParams, type ToolCallResult, type ValidatedToolCallParams, type XiaozhiTokenPayload, decodeJWTToken, deepMerge, extractTokenFromUrl, formatErrorMessage, isValidEndpointUrl, parseEndpointUrl, sleep, sliceEndpoint, validateToolCallParams };
|
package/dist/index.js
CHANGED
|
@@ -121,6 +121,7 @@ var init_internal_mcp_manager = __esm({
|
|
|
121
121
|
import WebSocket from "ws";
|
|
122
122
|
|
|
123
123
|
// src/types.ts
|
|
124
|
+
import { ensureToolJSONSchema } from "@xiaozhi-client/mcp-core";
|
|
124
125
|
var ToolCallErrorCode = /* @__PURE__ */ ((ToolCallErrorCode2) => {
|
|
125
126
|
ToolCallErrorCode2[ToolCallErrorCode2["INVALID_PARAMS"] = -32602] = "INVALID_PARAMS";
|
|
126
127
|
ToolCallErrorCode2[ToolCallErrorCode2["TOOL_NOT_FOUND"] = -32601] = "TOOL_NOT_FOUND";
|
|
@@ -140,18 +141,6 @@ var ToolCallError = class extends Error {
|
|
|
140
141
|
__name(this, "ToolCallError");
|
|
141
142
|
}
|
|
142
143
|
};
|
|
143
|
-
function ensureToolJSONSchema(schema) {
|
|
144
|
-
if (typeof schema === "object" && schema !== null && "type" in schema && schema.type === "object") {
|
|
145
|
-
return schema;
|
|
146
|
-
}
|
|
147
|
-
return {
|
|
148
|
-
type: "object",
|
|
149
|
-
properties: {},
|
|
150
|
-
required: [],
|
|
151
|
-
additionalProperties: true
|
|
152
|
-
};
|
|
153
|
-
}
|
|
154
|
-
__name(ensureToolJSONSchema, "ensureToolJSONSchema");
|
|
155
144
|
var ConnectionState = /* @__PURE__ */ ((ConnectionState2) => {
|
|
156
145
|
ConnectionState2["DISCONNECTED"] = "disconnected";
|
|
157
146
|
ConnectionState2["CONNECTING"] = "connecting";
|
|
@@ -522,7 +511,6 @@ var Endpoint = class _Endpoint {
|
|
|
522
511
|
* 处理 MCP 消息
|
|
523
512
|
*/
|
|
524
513
|
handleMessage(message) {
|
|
525
|
-
console.debug("\u6536\u5230 MCP \u6D88\u606F:", JSON.stringify(message, null, 2));
|
|
526
514
|
if (!message.method) {
|
|
527
515
|
console.debug("\u6536\u5230\u6CA1\u6709 method \u5B57\u6BB5\u7684\u6D88\u606F\uFF0C\u5FFD\u7565");
|
|
528
516
|
return;
|
|
@@ -558,7 +546,6 @@ var Endpoint = class _Endpoint {
|
|
|
558
546
|
}
|
|
559
547
|
case "ping":
|
|
560
548
|
this.sendResponse(message.id, {});
|
|
561
|
-
console.debug("\u56DE\u5E94 MCP ping \u6D88\u606F");
|
|
562
549
|
break;
|
|
563
550
|
default:
|
|
564
551
|
console.warn(`\u672A\u77E5\u7684 MCP \u8BF7\u6C42: ${message.method}`);
|
|
@@ -568,9 +555,6 @@ var Endpoint = class _Endpoint {
|
|
|
568
555
|
* 发送响应消息
|
|
569
556
|
*/
|
|
570
557
|
sendResponse(id, result) {
|
|
571
|
-
console.debug(
|
|
572
|
-
`\u5C1D\u8BD5\u53D1\u9001\u54CD\u5E94: id=${id}, isConnected=${this.connectionStatus}, wsReadyState=${this.ws?.readyState}`
|
|
573
|
-
);
|
|
574
558
|
if (this.connectionStatus && this.ws?.readyState === WebSocket.OPEN) {
|
|
575
559
|
const response = {
|
|
576
560
|
jsonrpc: "2.0",
|
|
@@ -931,7 +915,7 @@ var EndpointManager = class extends EventEmitter {
|
|
|
931
915
|
*
|
|
932
916
|
* @param endpoint - Endpoint 实例
|
|
933
917
|
*/
|
|
934
|
-
removeEndpoint(endpoint) {
|
|
918
|
+
async removeEndpoint(endpoint) {
|
|
935
919
|
const url = endpoint.getUrl();
|
|
936
920
|
if (!this.endpoints.has(url)) {
|
|
937
921
|
console.debug(
|
|
@@ -940,7 +924,7 @@ var EndpointManager = class extends EventEmitter {
|
|
|
940
924
|
return;
|
|
941
925
|
}
|
|
942
926
|
console.debug(`[EndpointManager] \u79FB\u9664\u63A5\u5165\u70B9: ${sliceEndpoint(url)}`);
|
|
943
|
-
endpoint.disconnect();
|
|
927
|
+
await endpoint.disconnect();
|
|
944
928
|
this.endpoints.delete(url);
|
|
945
929
|
this.connectionStates.delete(url);
|
|
946
930
|
this.emit("endpointRemoved", { endpoint: url });
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/internal-mcp-manager.ts","../src/endpoint.ts","../src/types.ts","../src/utils.ts","../src/manager.ts","../src/shared-mcp-adapter.ts"],"sourcesContent":["/**\n * 内部 MCP 服务管理器适配器\n * 将 mcpServers 配置转换为 IMCPServiceManager 接口\n * 使用 @xiaozhi-client/mcp-core 的 MCPManager 实现真实的 MCP 功能\n */\n\nimport { MCPManager } from \"@xiaozhi-client/mcp-core\";\nimport type { EnhancedToolInfo, ToolCallResult } from \"./types.js\";\nimport type { IMCPServiceManager } from \"./types.js\";\nimport type { EndpointConfig } from \"./types.js\";\nimport { normalizeServiceConfig } from \"@xiaozhi-client/config\";\n\n/**\n * 内部 MCP 服务管理器适配器\n * 实现 IMCPServiceManager 接口,使用真正的 MCPManager\n */\nexport class InternalMCPManagerAdapter implements IMCPServiceManager {\n private mcpManager: MCPManager;\n private tools: Map<string, EnhancedToolInfo> = new Map();\n private isInitialized = false;\n\n constructor(private config: EndpointConfig) {\n this.mcpManager = new MCPManager();\n\n // 转换配置并添加到 MCPManager\n for (const [serviceName, serverConfig] of Object.entries(\n config.mcpServers\n )) {\n const mcpConfig = normalizeServiceConfig(serverConfig);\n this.mcpManager.addServer(serviceName, mcpConfig);\n }\n\n // 设置事件监听\n this.mcpManager.on(\"connected\", (data) => {\n console.info(\n `MCP 服务 ${data.serverName} 已连接,工具数: ${data.tools.length}`\n );\n });\n\n this.mcpManager.on(\"error\", (data) => {\n console.error(`MCP 服务 ${data.serverName} 出错:`, data.error);\n });\n }\n\n /**\n * 初始化并启动所有 MCP 服务\n */\n async initialize(): Promise<void> {\n if (this.isInitialized) {\n return;\n }\n\n // 连接所有 MCP 服务\n await this.mcpManager.connect();\n\n // 刷新工具列表\n await this.refreshTools();\n\n this.isInitialized = true;\n }\n\n /**\n * 获取所有工具列表\n */\n getAllTools(): EnhancedToolInfo[] {\n return Array.from(this.tools.values());\n }\n\n /**\n * 调用工具(真实实现)\n */\n async callTool(\n toolName: string,\n arguments_: Record<string, unknown>\n ): Promise<ToolCallResult> {\n // 解析工具名称:serviceName__toolName\n const [serviceName, actualToolName] = this.parseToolName(toolName);\n\n // 调用真实的 MCP 工具\n return (await this.mcpManager.callTool(\n serviceName,\n actualToolName,\n arguments_\n )) as ToolCallResult;\n }\n\n /**\n * 清理资源\n */\n async cleanup(): Promise<void> {\n await this.mcpManager.disconnect();\n this.tools.clear();\n this.isInitialized = false;\n }\n\n /**\n * 刷新工具列表\n */\n private async refreshTools(): Promise<void> {\n this.tools.clear();\n\n const mcpTools = this.mcpManager.listTools();\n\n for (const mcpTool of mcpTools) {\n const enhancedTool: EnhancedToolInfo = {\n name: `${mcpTool.serverName}__${mcpTool.name}`,\n description: mcpTool.description,\n inputSchema: mcpTool.inputSchema as any,\n serviceName: mcpTool.serverName,\n originalName: mcpTool.name,\n enabled: true,\n usageCount: 0,\n lastUsedTime: new Date().toISOString(),\n };\n this.tools.set(enhancedTool.name, enhancedTool);\n }\n }\n\n /**\n * 解析工具名称\n */\n private parseToolName(toolName: string): [string, string] {\n const parts = toolName.split(\"__\");\n if (parts.length < 2) {\n throw new Error(`无效的工具名称格式: ${toolName}`);\n }\n const serviceName = parts[0];\n const actualToolName = parts.slice(1).join(\"__\"); // 支持工具名中包含 __\n return [serviceName, actualToolName];\n }\n}\n","/**\n * Endpoint 类\n * 管理单个小智接入点的 WebSocket 连接\n * 实现 MCP (Model Context Protocol) 协议通信\n *\n * 使用方式:\n * ```typescript\n * const mcpManager = new SharedMCPAdapter(globalMCPManager);\n * const endpoint = new Endpoint(\"ws://...\", mcpManager);\n * await endpoint.connect();\n * ```\n */\n\nimport type { Tool } from \"@modelcontextprotocol/sdk/types.js\";\nimport WebSocket from \"ws\";\nimport type { ExtendedMCPMessage, MCPMessage } from \"./mcp.js\";\nimport type {\n IMCPServiceManager,\n EndpointConnectionStatus,\n ToolCallResult,\n} from \"./types.js\";\nimport {\n ConnectionState,\n ToolCallError as ToolCallErrorClass,\n ToolCallErrorCode as ToolCallErrorCodeEnum,\n ensureToolJSONSchema as ensureToolJSONSchemaFn,\n} from \"./types.js\";\nimport { validateToolCallParams } from \"./utils.js\";\nimport { sliceEndpoint } from \"./utils.js\";\n\n// 导出错误类型供外部使用\nexport {\n ToolCallErrorCodeEnum as ToolCallErrorCode,\n ToolCallErrorClass as ToolCallError,\n};\n\n/**\n * Endpoint.create() 工厂方法配置接口\n */\nexport interface EndpointCreateConfig {\n /** 小智接入点 URL */\n endpointUrl: string;\n /** MCP 服务器配置(声明式) */\n mcpServers: Record<string, import(\"./types.js\").MCPServerConfig>;\n /** 可选:重连延迟(毫秒),默认 2000 */\n reconnectDelay?: number;\n}\n\n/**\n * Endpoint 类\n * 负责管理单个小智接入点的 WebSocket 连接\n *\n * 使用方式 1:直接使用构造函数(需要先创建 MCPManager)\n * ```typescript\n * const mcpManager = new MCPManager();\n * mcpManager.addServer(\"calculator\", { command: \"npx\", args: [\"-y\", \"@xiaozhi-client/calculator-mcp\"] });\n * await mcpManager.connect();\n * const mcpAdapter = new SharedMCPAdapter(mcpManager);\n * const endpoint = new Endpoint(\"ws://...\", mcpAdapter, 2000);\n * ```\n *\n * 使用方式 2:使用工厂方法(推荐,更简洁)\n * ```typescript\n * const endpoint = await Endpoint.create({\n * endpointUrl: \"ws://...\",\n * mcpServers: {\n * calculator: { command: \"npx\", args: [\"-y\", \"@xiaozhi-client/calculator-mcp\"] }\n * },\n * reconnectDelay: 2000,\n * });\n * ```\n */\nexport class Endpoint {\n private endpointUrl: string;\n private ws: WebSocket | null = null;\n private connectionStatus = false;\n private serverInitialized = false;\n private mcpAdapter: IMCPServiceManager;\n\n // 连接状态管理\n private connectionState: ConnectionState = ConnectionState.DISCONNECTED;\n\n // 最后一次错误信息\n private lastError: string | null = null;\n\n // 连接超时定时器\n private connectionTimeout: NodeJS.Timeout | null = null;\n\n // 工具调用超时配置\n private toolCallTimeout = 30000;\n private reconnectDelay: number; // 重连延迟(毫秒)\n\n /**\n * 构造函数\n *\n * @param endpointUrl - 小智接入点 URL\n * @param mcpManager - MCP 服务管理器(依赖注入)\n * @param reconnectDelay - 可选的重连延迟(毫秒)\n */\n constructor(\n endpointUrl: string,\n mcpManager: IMCPServiceManager,\n reconnectDelay?: number\n ) {\n this.endpointUrl = endpointUrl;\n this.reconnectDelay = reconnectDelay ?? 2000;\n\n // 使用注入的 MCP 管理器\n this.mcpAdapter = mcpManager;\n }\n\n /**\n * 工厂方法:使用声明式配置创建 Endpoint 实例\n *\n * 这是一个便捷方法,用于简化 Endpoint 的创建流程。\n * 内部会自动创建并配置 MCPManager,然后创建 Endpoint 实例。\n *\n * @param config - 配置对象\n * @returns Promise<Endpoint> - 已创建的 Endpoint 实例\n *\n * @example\n * ```typescript\n * const endpoint = await Endpoint.create({\n * endpointUrl: \"wss://api.xiaozhi.me/mcp/?token=...\",\n * mcpServers: {\n * calculator: {\n * command: \"npx\",\n * args: [\"-y\", \"@xiaozhi-client/calculator-mcp\"]\n * }\n * },\n * reconnectDelay: 2000\n * });\n * ```\n */\n static async create(config: EndpointCreateConfig): Promise<Endpoint> {\n // 动态导入相关模块\n const { InternalMCPManagerAdapter } = await import(\"./internal-mcp-manager.js\");\n\n // 创建内部 MCP 管理器适配器配置\n const endpointConfig = {\n mcpServers: config.mcpServers,\n reconnectDelay: config.reconnectDelay,\n };\n\n // 使用 InternalMCPManagerAdapter 创建 MCP 管理器\n const internalMCPManager = new InternalMCPManagerAdapter(endpointConfig);\n\n try {\n // 初始化 MCP 管理器(这会连接所有 MCP 服务)\n await internalMCPManager.initialize();\n } catch (error) {\n // 清理已启动的资源\n await internalMCPManager.cleanup();\n throw error;\n }\n\n // 创建并返回 Endpoint 实例\n return new Endpoint(\n config.endpointUrl,\n internalMCPManager,\n config.reconnectDelay\n );\n }\n\n /**\n * 获取 Endpoint URL\n */\n getUrl(): string {\n return this.endpointUrl;\n }\n\n /**\n * 获取当前所有工具列表\n */\n getTools(): Tool[] {\n try {\n const allTools = this.mcpAdapter.getAllTools();\n\n return allTools.map((toolInfo) => ({\n name: toolInfo.name,\n description: toolInfo.description,\n inputSchema: ensureToolJSONSchemaFn(toolInfo.inputSchema),\n }));\n } catch (error) {\n console.error(\n `获取工具列表失败: ${error instanceof Error ? error.message : String(error)}`\n );\n return [];\n }\n }\n\n /**\n * 连接小智接入点\n */\n public async connect(): Promise<void> {\n // 初始化 MCP 适配器\n await this.mcpAdapter.initialize();\n\n // 如果正在连接中,等待当前连接完成\n if (this.connectionState === ConnectionState.CONNECTING) {\n throw new Error(\"连接正在进行中,请等待连接完成\");\n }\n\n // 清理之前的连接\n this.cleanupConnection();\n\n return this.attemptConnection();\n }\n\n /**\n * 尝试建立连接\n */\n private async attemptConnection(): Promise<void> {\n this.connectionState = ConnectionState.CONNECTING;\n console.debug(`正在连接小智接入点: ${sliceEndpoint(this.endpointUrl)}`);\n\n return new Promise((resolve, reject) => {\n // 设置连接超时\n this.connectionTimeout = setTimeout(() => {\n const error = new Error(\"连接超时 (10000ms)\");\n this.handleConnectionError(error);\n reject(error);\n }, 10000);\n\n this.ws = new WebSocket(this.endpointUrl);\n\n this.ws.on(\"open\", () => {\n this.handleConnectionSuccess();\n resolve();\n });\n\n this.ws.on(\"message\", (data) => {\n try {\n const message: MCPMessage = JSON.parse(data.toString());\n this.handleMessage(message);\n } catch (error) {\n console.error(\"MCP 消息解析错误:\", error);\n }\n });\n\n this.ws.on(\"close\", (code, reason) => {\n this.handleConnectionClose(code, reason.toString());\n });\n\n this.ws.on(\"error\", (error) => {\n this.handleConnectionError(error);\n reject(error);\n });\n });\n }\n\n /**\n * 处理连接成功\n */\n private handleConnectionSuccess(): void {\n // 清理连接超时定时器\n if (this.connectionTimeout) {\n clearTimeout(this.connectionTimeout);\n this.connectionTimeout = null;\n }\n\n this.connectionStatus = true;\n this.connectionState = ConnectionState.CONNECTED;\n\n console.debug(\"MCP WebSocket 连接已建立\");\n }\n\n /**\n * 处理连接错误\n */\n private handleConnectionError(error: Error): void {\n // 记录最后一次错误信息\n this.lastError = error.message;\n\n // 清理连接超时定时器\n if (this.connectionTimeout) {\n clearTimeout(this.connectionTimeout);\n this.connectionTimeout = null;\n }\n\n console.error(\"MCP WebSocket 错误:\", error.message);\n\n // 清理当前连接\n this.cleanupConnection();\n }\n\n /**\n * 处理连接关闭\n */\n private handleConnectionClose(code: number, reason: string): void {\n this.connectionStatus = false;\n this.serverInitialized = false;\n this.connectionState = ConnectionState.DISCONNECTED;\n console.info(`小智连接已关闭 (代码: ${code}, 原因: ${reason})`);\n }\n\n /**\n * 清理连接资源\n */\n private cleanupConnection(): void {\n // 清理 WebSocket\n if (this.ws) {\n this.ws.removeAllListeners();\n\n try {\n if (this.ws.readyState === WebSocket.OPEN) {\n this.ws.close(1000, \"Cleaning up connection\");\n } else if (this.ws.readyState === WebSocket.CONNECTING) {\n this.ws.terminate();\n }\n } catch (error) {\n console.debug(\"WebSocket 关闭时出现错误(已忽略):\", error);\n }\n\n this.ws = null;\n }\n\n // 清理连接超时定时器\n if (this.connectionTimeout) {\n clearTimeout(this.connectionTimeout);\n this.connectionTimeout = null;\n }\n\n // 重置连接状态\n this.connectionStatus = false;\n this.serverInitialized = false;\n\n // 重置连接状态为已断开\n this.connectionState = ConnectionState.DISCONNECTED;\n }\n\n /**\n * 处理 MCP 消息\n */\n private handleMessage(message: MCPMessage): void {\n console.debug(\"收到 MCP 消息:\", JSON.stringify(message, null, 2));\n\n if (!message.method) {\n console.debug(\"收到没有 method 字段的消息,忽略\");\n return;\n }\n\n switch (message.method) {\n case \"initialize\":\n case \"notifications/initialized\":\n this.sendResponse(message.id, {\n protocolVersion: \"2024-11-05\",\n capabilities: {\n tools: { listChanged: true },\n logging: {},\n },\n serverInfo: {\n name: \"xiaozhi-mcp-server\",\n version: \"1.0.0\",\n },\n });\n this.serverInitialized = true;\n console.debug(\"MCP 服务器初始化完成\");\n break;\n\n case \"tools/list\": {\n const toolsList = this.getTools();\n this.sendResponse(message.id, { tools: toolsList });\n console.debug(`MCP 工具列表已发送 (${toolsList.length}个工具)`);\n break;\n }\n\n case \"tools/call\": {\n this.handleToolCall(message).catch((error) => {\n console.error(\"处理工具调用时发生未捕获错误:\", error);\n });\n break;\n }\n\n case \"ping\":\n this.sendResponse(message.id, {});\n console.debug(\"回应 MCP ping 消息\");\n break;\n\n default:\n console.warn(`未知的 MCP 请求: ${message.method}`);\n }\n }\n\n /**\n * 发送响应消息\n */\n private sendResponse(id: number | string, result: unknown): void {\n console.debug(\n `尝试发送响应: id=${id}, isConnected=${this.connectionStatus}, wsReadyState=${this.ws?.readyState}`\n );\n\n if (this.connectionStatus && this.ws?.readyState === WebSocket.OPEN) {\n const response: ExtendedMCPMessage = {\n jsonrpc: \"2.0\",\n id,\n result,\n };\n\n try {\n this.ws.send(JSON.stringify(response));\n console.debug(\"响应已发送\", {\n id,\n responseSize: JSON.stringify(response).length,\n });\n } catch (error) {\n console.error(\"发送响应失败\", {\n id,\n error,\n });\n }\n } else {\n console.error(\"无法发送响应\", {\n id,\n isConnected: this.connectionStatus,\n wsReadyState: this.ws?.readyState,\n });\n }\n }\n\n /**\n * 获取服务器状态\n */\n public getStatus(): EndpointConnectionStatus {\n const availableTools = this.mcpAdapter.getAllTools().length;\n\n return {\n connected: this.connectionStatus,\n initialized: this.serverInitialized,\n url: this.endpointUrl,\n availableTools,\n connectionState: this.connectionState,\n lastError: this.lastError,\n };\n }\n\n /**\n * 检查连接状态\n */\n public isConnected(): boolean {\n return this.connectionStatus;\n }\n\n /**\n * 主动断开小智连接\n */\n public async disconnect(): Promise<void> {\n console.info(\"主动断开小智连接\");\n\n // 清理 MCP 适配器\n await this.mcpAdapter.cleanup();\n\n // 清理 WebSocket 连接\n this.cleanupConnection();\n }\n\n /**\n * 重连小智接入点\n */\n public async reconnect(): Promise<void> {\n console.info(`重连小智接入点: ${sliceEndpoint(this.endpointUrl)}`);\n\n // 先断开连接\n this.disconnect();\n\n // 等待可配置的时间确保连接完全断开\n await new Promise((resolve) => setTimeout(resolve, this.reconnectDelay));\n\n // 重新连接\n await this.connect();\n }\n\n /**\n * 处理工具调用请求\n */\n private async handleToolCall(request: MCPMessage): Promise<void> {\n if (request.id === undefined || request.id === null) {\n throw new ToolCallErrorClass(\n ToolCallErrorCodeEnum.INVALID_PARAMS,\n \"请求 ID 不能为空\"\n );\n }\n\n const requestId = request.id;\n const startTime = Date.now();\n\n try {\n const params = validateToolCallParams(request.params);\n\n console.info(\"开始处理工具调用\", {\n requestId,\n toolName: params.name,\n hasArguments: !!params.arguments,\n });\n\n const result = await this.executeToolWithTimeout(\n params.name,\n params.arguments || {},\n this.toolCallTimeout\n );\n\n this.sendResponse(requestId, {\n content: result.content || [\n { type: \"text\", text: JSON.stringify(result) },\n ],\n isError: result.isError || false,\n });\n\n console.info(\"工具调用成功\", {\n requestId,\n toolName: params.name,\n duration: `${Date.now() - startTime}ms`,\n });\n } catch (error) {\n this.handleToolCallError(error, requestId, Date.now() - startTime);\n }\n }\n\n /**\n * 带超时控制的工具执行\n */\n private async executeToolWithTimeout(\n toolName: string,\n arguments_: Record<string, unknown>,\n timeoutMs = 30000\n ): Promise<ToolCallResult> {\n return new Promise((resolve, reject) => {\n const timeoutId = setTimeout(() => {\n reject(\n new ToolCallErrorClass(\n ToolCallErrorCodeEnum.TIMEOUT,\n `工具调用超时 (${timeoutMs}ms): ${toolName}`\n )\n );\n }, timeoutMs);\n\n this.mcpAdapter\n .callTool(toolName, arguments_)\n .then((result: ToolCallResult) => {\n clearTimeout(timeoutId);\n resolve(result);\n })\n .catch((error: unknown) => {\n clearTimeout(timeoutId);\n\n const errorMessage =\n error instanceof Error ? error.message : String(error);\n\n if (errorMessage.includes(\"未找到工具\")) {\n reject(\n new ToolCallErrorClass(\n ToolCallErrorCodeEnum.TOOL_NOT_FOUND,\n `工具不存在: ${toolName}`\n )\n );\n } else {\n reject(\n new ToolCallErrorClass(\n ToolCallErrorCodeEnum.TOOL_EXECUTION_ERROR,\n `工具执行失败: ${errorMessage}`\n )\n );\n }\n });\n });\n }\n\n /**\n * 处理工具调用错误\n */\n private handleToolCallError(\n error: unknown,\n requestId: string | number | undefined,\n duration: number\n ): void {\n let errorResponse: {\n code: number;\n message: string;\n data?: unknown;\n };\n\n if (error instanceof ToolCallErrorClass) {\n errorResponse = {\n code: error.code,\n message: error.message,\n data: error.data,\n };\n } else {\n const errorMessage = error instanceof Error ? error.message : \"未知错误\";\n errorResponse = {\n code: ToolCallErrorCodeEnum.TOOL_EXECUTION_ERROR,\n message: errorMessage,\n data: { originalError: String(error) || \"null\" },\n };\n }\n\n this.sendErrorResponse(requestId, errorResponse);\n\n console.error(\"工具调用失败\", {\n requestId,\n duration: `${duration}ms`,\n error: errorResponse,\n });\n }\n\n /**\n * 发送错误响应\n */\n private sendErrorResponse(\n id: string | number | undefined,\n error: { code: number; message: string; data?: unknown }\n ): void {\n if (this.connectionStatus && this.ws?.readyState === WebSocket.OPEN) {\n const response = {\n jsonrpc: \"2.0\",\n id,\n error,\n };\n try {\n this.ws.send(JSON.stringify(response));\n console.debug(\"已发送错误响应:\", response);\n } catch (sendError) {\n // 发送错误响应失败,记录日志但不抛出异常\n console.error(\"发送错误响应失败:\", {\n id,\n errorResponse: error,\n sendError: sendError instanceof Error\n ? { message: sendError.message, stack: sendError.stack }\n : sendError,\n });\n }\n } else {\n console.error(\"无法发送错误响应\", {\n id,\n isConnected: this.connectionStatus,\n wsReadyState: this.ws?.readyState,\n });\n }\n }\n}\n","// =========================\n// 1. 工具调用相关类型\n// =========================\n\n/**\n * 工具调用结果接口\n * 使用更宽松的类型定义以兼容不同来源的 ToolCallResult\n */\nexport interface ToolCallResult {\n content: Array<Record<string, unknown>>;\n isError?: boolean;\n _meta?: Record<string, unknown>;\n toolResult?: unknown; // 支持旧协议版本\n [key: string]: unknown; // 支持其他未知字段\n}\n\n/**\n * 工具调用参数接口\n */\nexport interface ToolCallParams {\n name: string;\n arguments?: Record<string, unknown>;\n}\n\n/**\n * 验证后的工具调用参数\n */\nexport interface ValidatedToolCallParams {\n name: string;\n arguments?: Record<string, unknown>;\n}\n\n/**\n * 工具调用错误码枚举\n */\nexport enum ToolCallErrorCode {\n /** 无效参数 */\n INVALID_PARAMS = -32602,\n /** 工具不存在 */\n TOOL_NOT_FOUND = -32601,\n /** 服务不可用 */\n SERVICE_UNAVAILABLE = -32001,\n /** 调用超时 */\n TIMEOUT = -32002,\n /** 工具执行错误 */\n TOOL_EXECUTION_ERROR = -32000,\n}\n\n/**\n * 工具调用错误类\n */\nexport class ToolCallError extends Error {\n constructor(\n public code: ToolCallErrorCode,\n message: string,\n public data?: unknown\n ) {\n super(message);\n this.name = \"ToolCallError\";\n }\n}\n\n// =========================\n// 2. JSON Schema 类型\n// =========================\n\n/**\n * JSON Schema 类型定义\n * 兼容 MCP SDK 的 JSON Schema 格式\n */\nexport type JSONSchema =\n | (Record<string, unknown> & {\n type: \"object\";\n properties?: Record<string, unknown>;\n required?: string[];\n additionalProperties?: boolean;\n })\n | Record<string, unknown>;\n\n/**\n * 确保对象符合 MCP Tool JSON Schema 格式\n * 返回类型兼容 MCP SDK 的 Tool 类型\n */\nexport function ensureToolJSONSchema(schema: JSONSchema): {\n type: \"object\";\n properties?: Record<string, object>;\n required?: string[];\n additionalProperties?: boolean;\n} {\n if (\n typeof schema === \"object\" &&\n schema !== null &&\n \"type\" in schema &&\n schema.type === \"object\"\n ) {\n return schema as {\n type: \"object\";\n properties?: Record<string, object>;\n required?: string[];\n additionalProperties?: boolean;\n };\n }\n\n // 如果不符合标准格式,返回默认的空对象 schema\n return {\n type: \"object\",\n properties: {},\n required: [],\n additionalProperties: true,\n };\n}\n\n// =========================\n// 3. 工具信息类型\n// =========================\n\n/**\n * 增强的工具信息接口\n * 包含工具的启用状态和使用统计信息\n */\nexport interface EnhancedToolInfo {\n /** 工具唯一标识符,格式为 \"{serviceName}__{originalName}\" */\n name: string;\n\n /** 工具描述信息 */\n description: string;\n\n /** 工具输入参数的 JSON Schema 定义 */\n inputSchema: JSONSchema;\n\n /** 工具所属的 MCP 服务名称 */\n serviceName: string;\n\n /** 工具在 MCP 服务中的原始名称 */\n originalName: string;\n\n /** 工具是否启用 */\n enabled: boolean;\n\n /** 工具使用次数统计 */\n usageCount: number;\n\n /** 工具最后使用时间 (ISO 8601 格式字符串) */\n lastUsedTime: string;\n}\n\n/**\n * MCP 服务管理器接口\n * 用于工具调用,避免循环依赖\n */\nexport interface IMCPServiceManager {\n /** 获取所有工具列表 */\n getAllTools(): EnhancedToolInfo[];\n\n /** 调用工具 */\n callTool(\n toolName: string,\n arguments_: Record<string, unknown>\n ): Promise<ToolCallResult>;\n\n /** 初始化 */\n initialize(): Promise<void>;\n\n /** 清理资源 */\n cleanup(): Promise<void>;\n}\n\n// =========================\n// 4. 连接状态类型\n// =========================\n\n/**\n * 连接状态枚举\n */\nexport enum ConnectionState {\n DISCONNECTED = \"disconnected\",\n CONNECTING = \"connecting\",\n CONNECTED = \"connected\",\n FAILED = \"failed\",\n}\n\n/**\n * 连接选项接口\n */\nexport interface ConnectionOptions {\n /** 连接超时时间(毫秒),默认 10000 */\n connectionTimeout?: number;\n /** 重连延迟时间(毫秒),默认 2000 */\n reconnectDelay?: number;\n}\n\n// =========================\n// 5. EndpointConnection 状态类型\n// =========================\n\n/**\n * EndpointConnection 状态接口\n */\nexport interface EndpointConnectionStatus {\n /** 是否已连接 */\n connected: boolean;\n /** 是否已初始化 */\n initialized: boolean;\n /** 接入点 URL */\n url: string;\n /** 可用工具数量 */\n availableTools: number;\n /** 连接状态 */\n connectionState: ConnectionState;\n /** 最后一次错误信息 */\n lastError: string | null;\n}\n\n// =========================\n// 6. EndpointManager 状态类型\n// =========================\n\n/**\n * 简单连接状态接口\n */\nexport interface SimpleConnectionStatus {\n /** 接入点地址 */\n endpoint: string;\n /** 是否已连接 */\n connected: boolean;\n /** 是否已初始化 */\n initialized: boolean;\n /** 最后连接时间 */\n lastConnected?: Date;\n /** 最后错误信息 */\n lastError?: string;\n}\n\n/**\n * 完整连接状态接口(扩展 SimpleConnectionStatus)\n */\nexport interface ConnectionStatus extends SimpleConnectionStatus {\n // 扩展字段可以在这里添加\n}\n\n/**\n * 配置变更事件类型\n */\nexport interface ConfigChangeEvent {\n type:\n | \"endpoints_added\"\n | \"endpoints_removed\"\n | \"endpoints_updated\"\n | \"options_updated\";\n data: {\n added?: string[];\n removed?: string[];\n updated?: string[];\n oldOptions?: Partial<ConnectionOptions>;\n newOptions?: Partial<ConnectionOptions>;\n };\n timestamp: Date;\n}\n\n/**\n * 重连结果接口\n */\nexport interface ReconnectResult {\n successCount: number;\n failureCount: number;\n results: Array<{\n endpoint: string;\n success: boolean;\n error?: string;\n }>;\n}\n\n// =========================\n// 7. 新 API 配置类型\n// =========================\n\n/**\n * MCP 服务器配置类型\n * 支持三种配置方式:\n * 1. 本地命令 (stdio): { command: string; args: string[]; env?: Record<string, string> }\n * 2. SSE: { type: \"sse\"; url: string; headers?: Record<string, string> }\n * 3. HTTP: { type?: \"http\"; url: string; headers?: Record<string, string> }\n *\n * 向后兼容:自动将 streamable-http/streamable_http/streamableHttp 转换为 http\n */\nexport type MCPServerConfig =\n | LocalMCPServerConfig\n | SSEMCPServerConfig\n | HTTPMCPServerConfig;\n\n/**\n * 本地 MCP 服务器配置\n */\nexport interface LocalMCPServerConfig {\n command: string;\n args: string[];\n env?: Record<string, string>;\n}\n\n/**\n * SSE MCP 服务器配置\n */\nexport interface SSEMCPServerConfig {\n type: \"sse\";\n url: string;\n headers?: Record<string, string>;\n}\n\n/**\n * HTTP MCP 服务器配置\n * 使用 type: \"http\"\n * 向后兼容 streamable-http 写法\n */\nexport interface HTTPMCPServerConfig {\n type?: \"http\" | \"streamable-http\"; // 可选,默认就是 http\n url: string;\n headers?: Record<string, string>;\n}\n\n// 向后兼容的别名\n/** @deprecated 使用 HTTPMCPServerConfig 代替 */\nexport type StreamableHTTPMCPServerConfig = HTTPMCPServerConfig;\n\n/**\n * Endpoint 配置接口\n * @deprecated 不再使用,Endpoint 构造函数改为接收 IMCPServiceManager\n */\nexport interface EndpointConfig {\n /** MCP 服务器配置(声明式) */\n mcpServers: Record<string, MCPServerConfig>;\n /** 可选:重连延迟(毫秒),默认 2000 */\n reconnectDelay?: number;\n /** 可选:ModelScope API Key(全局) */\n modelscopeApiKey?: string;\n}\n\n/**\n * EndpointManager 配置接口\n */\nexport interface EndpointManagerConfig {\n /** 可选:默认重连延迟(毫秒) */\n defaultReconnectDelay?: number;\n}\n\n// =========================\n// 8. JWT Token 类型\n// =========================\n\n/**\n * 小智平台 JWT Token Payload 接口\n *\n * @example\n * ```typescript\n * // 从 endpoint URL 解码得到的 payload\n * const payload: XiaozhiTokenPayload = {\n * userId: 302720,\n * agentId: 1324149,\n * endpointId: \"agent_1324149\",\n * purpose: \"mcp-endpoint\",\n * iat: 1768480930,\n * exp: 1800038530\n * };\n * ```\n */\nexport interface XiaozhiTokenPayload {\n /** 用户 ID */\n userId: number;\n /** 代理 ID */\n agentId: number;\n /** 接入点 ID,格式为 \"agent_{agentId}\" */\n endpointId: string;\n /** Token 用途 */\n purpose: string;\n /** 签发时间(Unix 时间戳) */\n iat: number;\n /** 过期时间(Unix 时间戳) */\n exp: number;\n}\n\n/**\n * 解析后的 Endpoint URL 信息\n */\nexport interface ParsedEndpointInfo {\n /** 完整的 endpoint URL */\n url: string;\n /** 提取的 JWT Token */\n token: string;\n /** 解码后的 Token Payload */\n payload: XiaozhiTokenPayload;\n /** WebSocket 服务器地址(不含 token 参数) */\n wsUrl: string;\n}\n","/**\n * 工具函数模块\n *\n * 内联的必要工具函数,确保包的独立性\n */\n\nimport type {\n ParsedEndpointInfo,\n ValidatedToolCallParams,\n XiaozhiTokenPayload,\n} from \"./types.js\";\n\n/**\n * 截断端点 URL 用于日志显示\n *\n * @param endpoint - 完整的端点 URL\n * @returns 截断后的 URL\n *\n * @example\n * ```typescript\n * sliceEndpoint(\"ws://very-long-endpoint-url-here.example.com/endpoint\")\n * // 返回: \"ws://very-long-endpoint-u...e.com/endpoint\"\n * ```\n */\nexport function sliceEndpoint(endpoint: string): string {\n return `${endpoint.slice(0, 30)}...${endpoint.slice(-10)}`;\n}\n\n/**\n * 验证工具调用参数\n *\n * @param params - 待验证的参数\n * @returns 验证后的参数\n * @throws {Error} 如果参数无效\n *\n * @example\n * ```typescript\n * const params = validateToolCallParams({\n * name: \"test_tool\",\n * arguments: { foo: \"bar\" }\n * });\n * ```\n */\nexport function validateToolCallParams(\n params: unknown\n): ValidatedToolCallParams {\n // 基础类型检查\n if (!params || typeof params !== \"object\") {\n throw new Error(\"工具调用参数必须是对象\");\n }\n\n const p = params as Record<string, unknown>;\n\n // 验证工具名称\n if (!p.name || typeof p.name !== \"string\") {\n throw new Error(\"工具名称必须是字符串\");\n }\n\n // 构建验证后的参数\n const validated: ValidatedToolCallParams = {\n name: p.name,\n };\n\n // 验证参数(可选)\n if (p.arguments !== undefined) {\n if (typeof p.arguments !== \"object\" || p.arguments === null) {\n throw new Error(\"工具参数必须是对象\");\n }\n validated.arguments = p.arguments as Record<string, unknown>;\n }\n\n return validated;\n}\n\n/**\n * 验证端点 URL 格式\n *\n * @param endpoint - 待验证的端点 URL\n * @returns 是否为有效的 WebSocket URL\n */\nexport function isValidEndpointUrl(endpoint: string): boolean {\n if (!endpoint || typeof endpoint !== \"string\") {\n return false;\n }\n\n if (!endpoint.startsWith(\"ws://\") && !endpoint.startsWith(\"wss://\")) {\n return false;\n }\n\n try {\n new URL(endpoint);\n return true;\n } catch {\n return false;\n }\n}\n\n/**\n * 检查属性名是否安全,防止原型污染攻击\n *\n * @param key - 待检查的属性名\n * @returns 是否为安全的属性名\n *\n * @example\n * ```typescript\n * isSafeKey(\"name\") // true\n * isSafeKey(\"__proto__\") // false\n * isSafeKey(\"constructor\") // false\n * ```\n */\nfunction isSafeKey(key: string): boolean {\n // 拒绝已知的危险属性名\n const dangerousKeys = [\"__proto__\", \"constructor\", \"prototype\"];\n if (dangerousKeys.includes(key)) {\n return false;\n }\n\n // 确保属性名是普通的字符串(不是 Symbol 等)\n return typeof key === \"string\" && key.length > 0;\n}\n\n/**\n * 深度合并对象\n *\n * @param target - 目标对象\n * @param sources - 源对象\n * @returns 合并后的对象\n *\n * @example\n * ```typescript\n * const result = deepMerge(\n * { a: 1, b: { x: 1 } },\n * { b: { y: 2 }, c: 3 }\n * );\n * // 返回: { a: 1, b: { x: 1, y: 2 }, c: 3 }\n * ```\n */\nexport function deepMerge<T>(\n target: Partial<T>,\n ...sources: Array<Partial<T>>\n): T {\n if (sources.length === 0) {\n return target as T;\n }\n\n const source = sources.shift();\n\n if (source === undefined) {\n return target as T;\n }\n\n // 使用 Object.keys() 替代 for...in,只遍历对象自身的可枚举属性\n const keys = Object.keys(source as Record<string, unknown>);\n\n for (const key of keys) {\n // 跳过不安全的 key,防止原型污染攻击\n if (!isSafeKey(key)) {\n continue;\n }\n\n const sourceValue = (source as Record<string, unknown>)[key];\n const targetValue = (target as Record<string, unknown>)[key];\n\n if (\n typeof sourceValue === \"object\" &&\n sourceValue !== null &&\n !Array.isArray(sourceValue) &&\n typeof targetValue === \"object\" &&\n targetValue !== null &&\n !Array.isArray(targetValue)\n ) {\n (target as Record<string, unknown>)[key] = deepMerge(\n targetValue as Record<string, unknown>,\n sourceValue as Record<string, unknown>\n );\n } else {\n (target as Record<string, unknown>)[key] = sourceValue;\n }\n }\n\n return deepMerge(target, ...sources);\n}\n\n/**\n * 延迟执行\n *\n * @param ms - 延迟时间(毫秒)\n * @returns Promise\n */\nexport function sleep(ms: number): Promise<void> {\n return new Promise((resolve) => setTimeout(resolve, ms));\n}\n\n/**\n * 格式化错误消息\n *\n * @param error - 错误对象\n * @returns 格式化后的错误消息\n */\nexport function formatErrorMessage(error: unknown): string {\n if (error instanceof Error) {\n return error.message;\n }\n\n if (typeof error === \"string\") {\n return error;\n }\n\n return String(error);\n}\n\n// =========================\n// JWT Token 解码相关函数\n// =========================\n\n/**\n * Base64URL 解码\n *\n * @param input - Base64URL 编码的字符串\n * @returns 解码后的字符串\n *\n * @example\n * ```typescript\n * base64UrlDecode(\"SGVsbG8gV29ybGQ\") // 返回: \"Hello World\"\n * ```\n */\nfunction base64UrlDecode(input: string): string {\n // 将 Base64URL 格式转换为标准 Base64 格式\n let base64 = input.replace(/-/g, \"+\").replace(/_/g, \"/\");\n\n // 补全 padding\n while (base64.length % 4) {\n base64 += \"=\";\n }\n\n // 使用 Node.js Buffer 进行解码\n return Buffer.from(base64, \"base64\").toString(\"utf-8\");\n}\n\n/**\n * 解码 JWT Token(仅解析 payload,不验证签名)\n *\n * @param token - JWT Token 字符串\n * @returns 解码后的 Token Payload,解码失败返回 null\n *\n * @example\n * ```typescript\n * const payload = decodeJWTToken(\"eyJ...token...\");\n * if (payload) {\n * console.log(payload.endpointId); // \"agent_1324149\"\n * }\n * ```\n */\nexport function decodeJWTToken(token: string): XiaozhiTokenPayload | null {\n if (!token || typeof token !== \"string\") {\n return null;\n }\n\n try {\n // JWT 格式: header.payload.signature\n const parts = token.split(\".\");\n if (parts.length !== 3) {\n return null;\n }\n\n // 解码 payload 部分(第二部分)\n const payloadStr = base64UrlDecode(parts[1]);\n const payload = JSON.parse(payloadStr) as unknown;\n\n // 验证 payload 结构\n if (\n typeof payload !== \"object\" ||\n payload === null ||\n !(\"userId\" in payload) ||\n !(\"agentId\" in payload) ||\n !(\"endpointId\" in payload) ||\n !(\"purpose\" in payload) ||\n !(\"iat\" in payload) ||\n !(\"exp\" in payload)\n ) {\n return null;\n }\n\n return payload as XiaozhiTokenPayload;\n } catch {\n return null;\n }\n}\n\n/**\n * 从 endpoint URL 中提取 token 参数\n *\n * @param url - 完整的 endpoint URL\n * @returns 提取的 token 字符串,未找到返回 null\n *\n * @example\n * ```typescript\n * const token = extractTokenFromUrl(\n * \"wss://api.xiaozhi.me/mcp/?token=eyJ...\"\n * );\n * ```\n */\nexport function extractTokenFromUrl(url: string): string | null {\n if (!url || typeof url !== \"string\") {\n return null;\n }\n\n try {\n const urlObj = new URL(url);\n return urlObj.searchParams.get(\"token\");\n } catch {\n return null;\n }\n}\n\n/**\n * 解析 endpoint URL 获取完整信息\n *\n * @param url - 完整的 endpoint URL\n * @returns 解析后的 endpoint 信息,解析失败返回 null\n *\n * @example\n * ```typescript\n * const info = parseEndpointUrl(\n * \"wss://api.xiaozhi.me/mcp/?token=eyJ...\"\n * );\n * if (info) {\n * console.log(info.payload.endpointId); // \"agent_1324149\"\n * console.log(info.wsUrl); // \"wss://api.xiaozhi.me/mcp/\"\n * }\n * ```\n */\nexport function parseEndpointUrl(url: string): ParsedEndpointInfo | null {\n if (!url || typeof url !== \"string\") {\n return null;\n }\n\n try {\n // 提取 token\n const token = extractTokenFromUrl(url);\n if (!token) {\n return null;\n }\n\n // 解码 token\n const payload = decodeJWTToken(token);\n if (!payload) {\n return null;\n }\n\n // 移除 token 参数得到纯净的 WebSocket URL\n const urlObj = new URL(url);\n urlObj.searchParams.delete(\"token\");\n const wsUrl = urlObj.toString();\n\n return {\n url,\n token,\n payload,\n wsUrl,\n };\n } catch {\n return null;\n }\n}\n","/**\n * EndpointManager\n * 管理多个小智接入点的连接,共享外部传入的 MCPManager\n *\n * 使用方式:\n * ```typescript\n * // 1. 先创建并配置 MCPManager\n * const mcpManager = new MCPManager();\n * mcpManager.addServer(\"calculator\", { command: \"npx\", args: [\"-y\", \"calculator-mcp\"] });\n * await mcpManager.connect();\n *\n * // 2. 创建 EndpointManager 并设置 MCPManager\n * const manager = new EndpointManager();\n * manager.setMcpManager(mcpManager);\n *\n * // 3. 添加接入点\n * manager.addEndpoint(\"ws://endpoint1\");\n * await manager.connect();\n * ```\n */\n\nimport { EventEmitter } from \"node:events\";\nimport type { Endpoint } from \"./endpoint.js\";\nimport { Endpoint as EndpointClass } from \"./endpoint.js\";\nimport { SharedMCPAdapter } from \"./shared-mcp-adapter.js\";\nimport type {\n EndpointManagerConfig,\n IMCPServiceManager,\n SimpleConnectionStatus,\n} from \"./types.js\";\nimport { sliceEndpoint } from \"./utils.js\";\n\n/**\n * MCP 服务连接事件数据\n */\ninterface MCPConnectedEvent {\n serverName: string;\n tools?: Array<unknown>;\n}\n\n/**\n * MCP 服务错误事件数据\n */\ninterface MCPErrorEvent {\n serverName: string;\n error: Error | unknown;\n}\n\n/**\n * 小智接入点管理器\n * 负责管理多个小智接入点的连接,共享 MCP 服务\n *\n * 注意:MCPManager 必须通过 setMcpManager() 方法设置,且由外部管理其生命周期\n */\nexport class EndpointManager extends EventEmitter {\n private endpoints: Map<string, Endpoint> = new Map();\n private connectionStates: Map<string, SimpleConnectionStatus> = new Map();\n private mcpManager: IMCPServiceManager | null = null;\n private sharedMCPAdapter: SharedMCPAdapter | null = null;\n private mcpEventListeners: Array<(...args: unknown[]) => void> = [];\n\n /**\n * 构造函数\n *\n * @param config - 可选的配置\n */\n constructor(private config?: EndpointManagerConfig) {\n super();\n console.debug(\"[EndpointManager] 实例已创建\");\n }\n\n /**\n * 设置 MCPManager 实例\n *\n * 注意:MCPManager 的生命周期由外部管理,EndpointManager 不负责连接和断开\n *\n * @param mcpManager - 外部创建并已连接的 MCPManager 实例\n */\n setMcpManager(mcpManager: IMCPServiceManager): void {\n if (this.sharedMCPAdapter) {\n throw new Error(\"MCPManager 已经设置,不能重复设置\");\n }\n\n this.mcpManager = mcpManager;\n this.sharedMCPAdapter = new SharedMCPAdapter(mcpManager);\n\n // 监听 MCP 服务连接事件(如果支持 EventEmitter)\n if (\"on\" in mcpManager && typeof mcpManager.on === \"function\") {\n const connectedHandler = (...args: unknown[]) => {\n const data = args[0] as MCPConnectedEvent;\n console.info(\n `[EndpointManager] MCP 服务已连接: ${data.serverName}, 工具数: ${data.tools?.length || 0}`\n );\n };\n const errorHandler = (...args: unknown[]) => {\n const data = args[0] as MCPErrorEvent;\n console.error(\n `[EndpointManager] MCP 服务连接失败: ${data.serverName}`,\n data.error\n );\n };\n\n mcpManager.on(\"connected\", connectedHandler);\n mcpManager.on(\"error\", errorHandler);\n\n // 保存监听器引用以便后续清理\n this.mcpEventListeners.push(connectedHandler, errorHandler);\n }\n\n console.info(\"[EndpointManager] MCPManager 已设置\");\n }\n\n /**\n * 添加 Endpoint(支持 URL 字符串或 Endpoint 实例)\n *\n * @param endpoint - Endpoint URL 字符串或 Endpoint 实例\n */\n addEndpoint(endpoint: string | Endpoint): void {\n // 如果是字符串,创建新的 Endpoint 实例\n if (typeof endpoint === \"string\") {\n if (!this.sharedMCPAdapter) {\n throw new Error(\n \"MCPManager 未设置,请先调用 setMcpManager() 方法设置 MCPManager\"\n );\n }\n\n const endpointInstance = new EndpointClass(\n endpoint,\n this.sharedMCPAdapter,\n this.config?.defaultReconnectDelay\n );\n\n this.addEndpointInternal(endpointInstance);\n return;\n }\n\n // 原有的 Endpoint 实例处理逻辑\n // 当 EndpointManager 已配置共享 MCPManager 时,不允许再传入外部构造的 Endpoint 实例,\n // 以避免不同 Endpoint 使用不同的 MCP 管理器,破坏共享 MCPManager 的设计。\n if (this.sharedMCPAdapter) {\n throw new Error(\n \"[EndpointManager] 当使用共享 MCPManager 时,不支持传入自定义 Endpoint 实例,请传入 Endpoint URL 字符串以便由 EndpointManager 创建实例\"\n );\n }\n\n this.addEndpointInternal(endpoint);\n }\n\n /**\n * 内部添加 Endpoint 方法\n */\n private addEndpointInternal(endpoint: Endpoint): void {\n const url = endpoint.getUrl();\n\n if (this.endpoints.has(url)) {\n console.debug(\n `[EndpointManager] 接入点 ${sliceEndpoint(url)} 已存在,跳过添加`\n );\n return;\n }\n\n console.debug(`[EndpointManager] 添加接入点: ${sliceEndpoint(url)}`);\n\n this.endpoints.set(url, endpoint);\n this.connectionStates.set(url, {\n endpoint: url,\n connected: false,\n initialized: false,\n });\n\n // 发射事件\n this.emit(\"endpointAdded\", { endpoint: url });\n }\n\n /**\n * 移除 Endpoint 实例\n *\n * @param endpoint - Endpoint 实例\n */\n removeEndpoint(endpoint: Endpoint): void {\n const url = endpoint.getUrl();\n\n if (!this.endpoints.has(url)) {\n console.debug(\n `[EndpointManager] 接入点 ${sliceEndpoint(url)} 不存在,跳过移除`\n );\n return;\n }\n\n console.debug(`[EndpointManager] 移除接入点: ${sliceEndpoint(url)}`);\n\n // 断开连接\n endpoint.disconnect();\n\n // 清理状态\n this.endpoints.delete(url);\n this.connectionStates.delete(url);\n\n // 发射事件\n this.emit(\"endpointRemoved\", { endpoint: url });\n }\n\n /**\n * 连接 Endpoint\n *\n * 注意:此方法不负责连接 MCPManager,MCPManager 必须在调用此方法前已连接\n *\n * @param endpoint - 可选,指定要连接的端点 URL。如果不传入,则连接所有端点\n */\n async connect(endpoint?: string): Promise<void> {\n // 如果指定了端点,只连接该端点\n if (endpoint) {\n const endpointInstance = this.endpoints.get(endpoint);\n if (!endpointInstance) {\n throw new Error(`接入点不存在: ${sliceEndpoint(endpoint)}`);\n }\n\n const status = this.connectionStates.get(endpoint);\n if (status?.connected) {\n console.debug(\n `[EndpointManager] 接入点已连接,跳过: ${sliceEndpoint(endpoint)}`\n );\n return;\n }\n\n await this.connectSingleEndpoint(endpoint, endpointInstance);\n return;\n }\n\n // 连接所有未连接的 Endpoint\n console.debug(\n `[EndpointManager] 开始连接接入点,总数: ${this.endpoints.size}`\n );\n\n const promises: Promise<void>[] = [];\n\n for (const [url, endpoint] of this.endpoints) {\n const status = this.connectionStates.get(url);\n\n // 跳过已连接的端点\n if (status?.connected) {\n console.debug(\n `[EndpointManager] 接入点已连接,跳过: ${sliceEndpoint(url)}`\n );\n continue;\n }\n\n promises.push(\n this.connectSingleEndpoint(url, endpoint).catch((error) => {\n console.error(\n `[EndpointManager] 连接失败: ${sliceEndpoint(url)}`,\n error\n );\n // 更新失败状态\n const status = this.connectionStates.get(url);\n if (status) {\n status.connected = false;\n status.initialized = false;\n status.lastError =\n error instanceof Error ? error.message : String(error);\n }\n })\n );\n }\n\n await Promise.allSettled(promises);\n\n // 统计连接结果\n const connectedCount = Array.from(this.connectionStates.values()).filter(\n (s) => s.connected\n ).length;\n\n console.info(\n `[EndpointManager] 连接完成: 成功 ${connectedCount}/${this.endpoints.size}`\n );\n }\n\n /**\n * 断开连接\n *\n * 注意:此方法不断开 MCPManager,MCPManager 的生命周期由外部管理\n *\n * @param endpoint - 可选,指定要断开的端点 URL。如果不传入,则断开所有端点\n */\n async disconnect(endpoint?: string): Promise<void> {\n // 如果指定了端点,只断开该端点\n if (endpoint) {\n const endpointInstance = this.endpoints.get(endpoint);\n if (!endpointInstance) {\n throw new Error(`接入点不存在: ${sliceEndpoint(endpoint)}`);\n }\n\n endpointInstance.disconnect();\n\n const status = this.connectionStates.get(endpoint);\n if (status) {\n status.connected = false;\n status.initialized = false;\n }\n\n console.debug(\n `[EndpointManager] 接入点已断开: ${sliceEndpoint(endpoint)}`\n );\n return;\n }\n\n // 断开所有连接\n console.debug(\"[EndpointManager] 开始断开所有连接\");\n\n const promises: Promise<void>[] = [];\n\n for (const endpoint of this.endpoints.values()) {\n promises.push(\n Promise.resolve().then(() => {\n endpoint.disconnect();\n })\n );\n }\n\n await Promise.allSettled(promises);\n\n // 重置所有状态\n for (const status of this.connectionStates.values()) {\n status.connected = false;\n status.initialized = false;\n }\n\n console.debug(\"[EndpointManager] 所有接入点已断开连接\");\n }\n\n /**\n * 获取所有 Endpoint URL\n */\n getEndpoints(): string[] {\n return Array.from(this.endpoints.keys());\n }\n\n /**\n * 获取指定 Endpoint 实例\n *\n * @param url - Endpoint URL\n */\n getEndpoint(url: string): Endpoint | undefined {\n return this.endpoints.get(url);\n }\n\n /**\n * 获取所有连接状态\n */\n getConnectionStatus(): SimpleConnectionStatus[] {\n return Array.from(this.connectionStates.values());\n }\n\n /**\n * 检查是否有任何连接处于连接状态\n */\n isAnyConnected(): boolean {\n for (const status of this.connectionStates.values()) {\n if (status.connected) {\n return true;\n }\n }\n return false;\n }\n\n /**\n * 检查指定端点是否已连接\n *\n * @param url - 端点 URL\n */\n isEndpointConnected(url: string): boolean {\n const status = this.connectionStates.get(url);\n return status?.connected ?? false;\n }\n\n /**\n * 获取指定端点的状态\n *\n * @param url - 端点 URL\n */\n getEndpointStatus(url: string): SimpleConnectionStatus | undefined {\n return this.connectionStates.get(url);\n }\n\n /**\n * 重连\n *\n * @param endpoint - 可选,指定要重连的端点 URL。如果不传入,则重连所有端点\n * @param delay - 可选,disconnect 和 connect 之间的等待时间(毫秒),默认为 1000ms。\n * 注意:此参数只控制断开和重新连接之间的等待时间,不影响底层 Endpoint 实例的重连延迟\n */\n async reconnect(endpoint?: string, delay = 1000): Promise<void> {\n console.info(\"[EndpointManager] 开始重连\");\n\n // 先断开连接\n await this.disconnect(endpoint);\n\n // 等待一段时间\n console.debug(`[EndpointManager] 等待 ${delay}ms 后重连`);\n await new Promise((resolve) => setTimeout(resolve, delay));\n\n // 再重新连接\n await this.connect(endpoint);\n\n console.info(\"[EndpointManager] 重连完成\");\n }\n\n /**\n * 重连所有端点\n */\n async reconnectAll(): Promise<void> {\n console.info(\"[EndpointManager] 开始重连所有接入点\");\n\n const promises: Promise<void>[] = [];\n\n for (const [url, endpoint] of this.endpoints) {\n promises.push(\n this.reconnectSingleEndpoint(url, endpoint).catch((error) => {\n console.error(\n `[EndpointManager] 重连失败: ${sliceEndpoint(url)}`,\n error\n );\n })\n );\n }\n\n await Promise.allSettled(promises);\n }\n\n /**\n * 重连指定的端点\n *\n * @param url - 要重连的端点 URL\n */\n async reconnectEndpoint(url: string): Promise<void> {\n const endpoint = this.endpoints.get(url);\n if (!endpoint) {\n throw new Error(`接入点不存在: ${sliceEndpoint(url)}`);\n }\n\n await this.reconnectSingleEndpoint(url, endpoint);\n }\n\n /**\n * 清除所有端点\n *\n * 注意:此方法不会清理 MCPManager,MCPManager 的生命周期由外部管理\n */\n async clearEndpoints(): Promise<void> {\n console.debug(\"[EndpointManager] 清除所有接入点\");\n\n await this.disconnect();\n\n this.endpoints.clear();\n this.connectionStates.clear();\n\n // 注意:不清理 MCPManager,由外部管理\n // this.mcpManager = null;\n // this.sharedMCPAdapter = null;\n\n console.info(\"[EndpointManager] 所有接入点已清除\");\n }\n\n /**\n * 清理资源\n *\n * 注意:此方法不会清理 MCPManager,MCPManager 的生命周期由外部管理\n */\n async cleanup(): Promise<void> {\n console.debug(\"[EndpointManager] 开始清理资源\");\n\n // 移除 MCP 服务事件监听器\n if (\n this.mcpManager &&\n \"removeListener\" in this.mcpManager &&\n typeof this.mcpManager.removeListener === \"function\"\n ) {\n for (const listener of this.mcpEventListeners) {\n this.mcpManager.removeListener(\"connected\", listener);\n this.mcpManager.removeListener(\"error\", listener);\n }\n }\n this.mcpEventListeners = [];\n\n await this.clearEndpoints();\n\n console.debug(\"[EndpointManager] 资源清理完成\");\n }\n\n // ==================== 私有方法 ====================\n\n /**\n * 连接单个端点\n */\n private async connectSingleEndpoint(\n url: string,\n endpoint: Endpoint\n ): Promise<void> {\n const status = this.connectionStates.get(url);\n if (!status) {\n throw new Error(`端点状态不存在: ${sliceEndpoint(url)}`);\n }\n\n console.debug(`[EndpointManager] 连接端点: ${sliceEndpoint(url)}`);\n\n // 更新状态为连接中\n status.connected = false;\n status.initialized = false;\n\n // 执行连接\n await endpoint.connect();\n\n // 更新连接成功状态\n status.connected = true;\n status.initialized = true;\n status.lastConnected = new Date();\n status.lastError = undefined;\n\n console.info(`[EndpointManager] 端点连接成功: ${sliceEndpoint(url)}`);\n }\n\n /**\n * 重连单个端点\n */\n private async reconnectSingleEndpoint(\n url: string,\n endpoint: Endpoint\n ): Promise<void> {\n const status = this.connectionStates.get(url);\n if (!status) {\n throw new Error(`端点状态不存在: ${sliceEndpoint(url)}`);\n }\n\n console.debug(`[EndpointManager] 重连端点: ${sliceEndpoint(url)}`);\n\n // 执行重连\n await endpoint.reconnect();\n\n // 更新连接成功状态\n status.connected = true;\n status.initialized = true;\n status.lastConnected = new Date();\n status.lastError = undefined;\n\n console.info(`[EndpointManager] 端点重连成功: ${sliceEndpoint(url)}`);\n }\n}\n","/**\n * 共享 MCP 管理器适配器\n *\n * 接收全局 MCPManager 实例的引用,不创建独立连接\n * 用于多个 Endpoint 共享同一个 MCPManager 实例\n *\n * @example\n * ```typescript\n * const globalMCPManager = new MCPManager();\n * await globalMCPManager.connect();\n *\n * const adapter = new SharedMCPAdapter(globalMCPManager);\n * const endpoint = new Endpoint(\"ws://...\", adapter);\n * ```\n */\n\nimport type {\n IMCPServiceManager,\n EnhancedToolInfo,\n ToolCallResult,\n} from \"./types.js\";\n\n/**\n * 共享 MCP 管理器适配器\n * 实现 IMCPServiceManager 接口,将操作委托给全局 MCPManager\n */\nexport class SharedMCPAdapter implements IMCPServiceManager {\n private isInitialized = false;\n\n /**\n * 构造函数\n *\n * @param globalMCPManager - 全局 MCPManager 实例\n */\n constructor(private globalMCPManager: IMCPServiceManager) {\n if (!globalMCPManager) {\n throw new Error(\"全局 MCPManager 不能为空\");\n }\n }\n\n /**\n * 初始化适配器\n *\n * 注意:此方法不执行任何连接操作,仅标记初始化状态,因为全局 MCPManager 的生命周期由外部管理\n */\n async initialize(): Promise<void> {\n if (this.isInitialized) {\n return;\n }\n\n // 全局实例已经在外部连接,这里只标记状态\n this.isInitialized = true;\n }\n\n /**\n * 获取所有工具列表\n *\n * 从全局 MCPManager 获取工具列表,并转换为增强格式\n */\n getAllTools(): EnhancedToolInfo[] {\n // 直接使用 IMCPServiceManager 接口的 getAllTools() 方法\n return this.globalMCPManager.getAllTools();\n }\n\n /**\n * 调用工具\n *\n * 将工具调用委托给全局 MCPManager\n *\n * @param toolName - 工具名称(格式:serviceName__toolName)\n * @param arguments_ - 工具调用参数\n */\n async callTool(\n toolName: string,\n arguments_: Record<string, unknown>\n ): Promise<ToolCallResult> {\n // 直接使用 IMCPServiceManager 接口的 callTool() 方法\n // toolName 已经是完整格式(serviceName__toolName)\n return this.globalMCPManager.callTool(toolName, arguments_);\n }\n\n /**\n * 清理资源\n *\n * 注意:此方法不会断开全局 MCPManager,由创建者负责管理\n */\n async cleanup(): Promise<void> {\n this.isInitialized = false;\n // 不断开全局 MCPManager\n }\n}\n"],"mappings":"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","names":["ToolCallErrorCode","ConnectionState","InternalMCPManagerAdapter","endpoint","status"]}
|
|
1
|
+
{"version":3,"sources":["../src/internal-mcp-manager.ts","../src/endpoint.ts","../src/types.ts","../src/utils.ts","../src/manager.ts","../src/shared-mcp-adapter.ts"],"sourcesContent":["/**\n * 内部 MCP 服务管理器适配器\n * 将 mcpServers 配置转换为 IMCPServiceManager 接口\n * 使用 @xiaozhi-client/mcp-core 的 MCPManager 实现真实的 MCP 功能\n */\n\nimport { MCPManager } from \"@xiaozhi-client/mcp-core\";\nimport type { EnhancedToolInfo, ToolCallResult } from \"./types.js\";\nimport type { IMCPServiceManager } from \"./types.js\";\nimport type { EndpointConfig } from \"./types.js\";\nimport { normalizeServiceConfig } from \"@xiaozhi-client/config\";\n\n/**\n * 内部 MCP 服务管理器适配器\n * 实现 IMCPServiceManager 接口,使用真正的 MCPManager\n */\nexport class InternalMCPManagerAdapter implements IMCPServiceManager {\n private mcpManager: MCPManager;\n private tools: Map<string, EnhancedToolInfo> = new Map();\n private isInitialized = false;\n\n constructor(private config: EndpointConfig) {\n this.mcpManager = new MCPManager();\n\n // 转换配置并添加到 MCPManager\n for (const [serviceName, serverConfig] of Object.entries(\n config.mcpServers\n )) {\n const mcpConfig = normalizeServiceConfig(serverConfig);\n this.mcpManager.addServer(serviceName, mcpConfig);\n }\n\n // 设置事件监听\n this.mcpManager.on(\"connected\", (data) => {\n console.info(\n `MCP 服务 ${data.serverName} 已连接,工具数: ${data.tools.length}`\n );\n });\n\n this.mcpManager.on(\"error\", (data) => {\n console.error(`MCP 服务 ${data.serverName} 出错:`, data.error);\n });\n }\n\n /**\n * 初始化并启动所有 MCP 服务\n */\n async initialize(): Promise<void> {\n if (this.isInitialized) {\n return;\n }\n\n // 连接所有 MCP 服务\n await this.mcpManager.connect();\n\n // 刷新工具列表\n await this.refreshTools();\n\n this.isInitialized = true;\n }\n\n /**\n * 获取所有工具列表\n */\n getAllTools(): EnhancedToolInfo[] {\n return Array.from(this.tools.values());\n }\n\n /**\n * 调用工具(真实实现)\n */\n async callTool(\n toolName: string,\n arguments_: Record<string, unknown>\n ): Promise<ToolCallResult> {\n // 解析工具名称:serviceName__toolName\n const [serviceName, actualToolName] = this.parseToolName(toolName);\n\n // 调用真实的 MCP 工具\n return (await this.mcpManager.callTool(\n serviceName,\n actualToolName,\n arguments_\n )) as ToolCallResult;\n }\n\n /**\n * 清理资源\n */\n async cleanup(): Promise<void> {\n await this.mcpManager.disconnect();\n this.tools.clear();\n this.isInitialized = false;\n }\n\n /**\n * 刷新工具列表\n */\n private async refreshTools(): Promise<void> {\n this.tools.clear();\n\n const mcpTools = this.mcpManager.listTools();\n\n for (const mcpTool of mcpTools) {\n const enhancedTool: EnhancedToolInfo = {\n name: `${mcpTool.serverName}__${mcpTool.name}`,\n description: mcpTool.description,\n inputSchema: mcpTool.inputSchema as any,\n serviceName: mcpTool.serverName,\n originalName: mcpTool.name,\n enabled: true,\n usageCount: 0,\n lastUsedTime: new Date().toISOString(),\n };\n this.tools.set(enhancedTool.name, enhancedTool);\n }\n }\n\n /**\n * 解析工具名称\n */\n private parseToolName(toolName: string): [string, string] {\n const parts = toolName.split(\"__\");\n if (parts.length < 2) {\n throw new Error(`无效的工具名称格式: ${toolName}`);\n }\n const serviceName = parts[0];\n const actualToolName = parts.slice(1).join(\"__\"); // 支持工具名中包含 __\n return [serviceName, actualToolName];\n }\n}\n","/**\n * Endpoint 类\n * 管理单个小智接入点的 WebSocket 连接\n * 实现 MCP (Model Context Protocol) 协议通信\n *\n * 使用方式:\n * ```typescript\n * const mcpManager = new SharedMCPAdapter(globalMCPManager);\n * const endpoint = new Endpoint(\"ws://...\", mcpManager);\n * await endpoint.connect();\n * ```\n */\n\nimport type { Tool } from \"@modelcontextprotocol/sdk/types.js\";\nimport WebSocket from \"ws\";\nimport type { ExtendedMCPMessage, MCPMessage } from \"./mcp.js\";\nimport type {\n IMCPServiceManager,\n EndpointConnectionStatus,\n ToolCallResult,\n} from \"./types.js\";\nimport {\n ConnectionState,\n ToolCallError as ToolCallErrorClass,\n ToolCallErrorCode as ToolCallErrorCodeEnum,\n ensureToolJSONSchema as ensureToolJSONSchemaFn,\n} from \"./types.js\";\nimport { validateToolCallParams } from \"./utils.js\";\nimport { sliceEndpoint } from \"./utils.js\";\n\n// 导出错误类型供外部使用\nexport {\n ToolCallErrorCodeEnum as ToolCallErrorCode,\n ToolCallErrorClass as ToolCallError,\n};\n\n/**\n * Endpoint.create() 工厂方法配置接口\n */\nexport interface EndpointCreateConfig {\n /** 小智接入点 URL */\n endpointUrl: string;\n /** MCP 服务器配置(声明式) */\n mcpServers: Record<string, import(\"./types.js\").MCPServerConfig>;\n /** 可选:重连延迟(毫秒),默认 2000 */\n reconnectDelay?: number;\n}\n\n/**\n * Endpoint 类\n * 负责管理单个小智接入点的 WebSocket 连接\n *\n * 使用方式 1:直接使用构造函数(需要先创建 MCPManager)\n * ```typescript\n * const mcpManager = new MCPManager();\n * mcpManager.addServer(\"calculator\", { command: \"npx\", args: [\"-y\", \"@xiaozhi-client/calculator-mcp\"] });\n * await mcpManager.connect();\n * const mcpAdapter = new SharedMCPAdapter(mcpManager);\n * const endpoint = new Endpoint(\"ws://...\", mcpAdapter, 2000);\n * ```\n *\n * 使用方式 2:使用工厂方法(推荐,更简洁)\n * ```typescript\n * const endpoint = await Endpoint.create({\n * endpointUrl: \"ws://...\",\n * mcpServers: {\n * calculator: { command: \"npx\", args: [\"-y\", \"@xiaozhi-client/calculator-mcp\"] }\n * },\n * reconnectDelay: 2000,\n * });\n * ```\n */\nexport class Endpoint {\n private endpointUrl: string;\n private ws: WebSocket | null = null;\n private connectionStatus = false;\n private serverInitialized = false;\n private mcpAdapter: IMCPServiceManager;\n\n // 连接状态管理\n private connectionState: ConnectionState = ConnectionState.DISCONNECTED;\n\n // 最后一次错误信息\n private lastError: string | null = null;\n\n // 连接超时定时器\n private connectionTimeout: NodeJS.Timeout | null = null;\n\n // 工具调用超时配置\n private toolCallTimeout = 30000;\n private reconnectDelay: number; // 重连延迟(毫秒)\n\n /**\n * 构造函数\n *\n * @param endpointUrl - 小智接入点 URL\n * @param mcpManager - MCP 服务管理器(依赖注入)\n * @param reconnectDelay - 可选的重连延迟(毫秒)\n */\n constructor(\n endpointUrl: string,\n mcpManager: IMCPServiceManager,\n reconnectDelay?: number\n ) {\n this.endpointUrl = endpointUrl;\n this.reconnectDelay = reconnectDelay ?? 2000;\n\n // 使用注入的 MCP 管理器\n this.mcpAdapter = mcpManager;\n }\n\n /**\n * 工厂方法:使用声明式配置创建 Endpoint 实例\n *\n * 这是一个便捷方法,用于简化 Endpoint 的创建流程。\n * 内部会自动创建并配置 MCPManager,然后创建 Endpoint 实例。\n *\n * @param config - 配置对象\n * @returns Promise<Endpoint> - 已创建的 Endpoint 实例\n *\n * @example\n * ```typescript\n * const endpoint = await Endpoint.create({\n * endpointUrl: \"wss://api.xiaozhi.me/mcp/?token=...\",\n * mcpServers: {\n * calculator: {\n * command: \"npx\",\n * args: [\"-y\", \"@xiaozhi-client/calculator-mcp\"]\n * }\n * },\n * reconnectDelay: 2000\n * });\n * ```\n */\n static async create(config: EndpointCreateConfig): Promise<Endpoint> {\n // 动态导入相关模块\n const { InternalMCPManagerAdapter } = await import(\"./internal-mcp-manager.js\");\n\n // 创建内部 MCP 管理器适配器配置\n const endpointConfig = {\n mcpServers: config.mcpServers,\n reconnectDelay: config.reconnectDelay,\n };\n\n // 使用 InternalMCPManagerAdapter 创建 MCP 管理器\n const internalMCPManager = new InternalMCPManagerAdapter(endpointConfig);\n\n try {\n // 初始化 MCP 管理器(这会连接所有 MCP 服务)\n await internalMCPManager.initialize();\n } catch (error) {\n // 清理已启动的资源\n await internalMCPManager.cleanup();\n throw error;\n }\n\n // 创建并返回 Endpoint 实例\n return new Endpoint(\n config.endpointUrl,\n internalMCPManager,\n config.reconnectDelay\n );\n }\n\n /**\n * 获取 Endpoint URL\n */\n getUrl(): string {\n return this.endpointUrl;\n }\n\n /**\n * 获取当前所有工具列表\n */\n getTools(): Tool[] {\n try {\n const allTools = this.mcpAdapter.getAllTools();\n\n return allTools.map((toolInfo) => ({\n name: toolInfo.name,\n description: toolInfo.description,\n inputSchema: ensureToolJSONSchemaFn(toolInfo.inputSchema),\n }));\n } catch (error) {\n console.error(\n `获取工具列表失败: ${error instanceof Error ? error.message : String(error)}`\n );\n return [];\n }\n }\n\n /**\n * 连接小智接入点\n */\n public async connect(): Promise<void> {\n // 初始化 MCP 适配器\n await this.mcpAdapter.initialize();\n\n // 如果正在连接中,等待当前连接完成\n if (this.connectionState === ConnectionState.CONNECTING) {\n throw new Error(\"连接正在进行中,请等待连接完成\");\n }\n\n // 清理之前的连接\n this.cleanupConnection();\n\n return this.attemptConnection();\n }\n\n /**\n * 尝试建立连接\n */\n private async attemptConnection(): Promise<void> {\n this.connectionState = ConnectionState.CONNECTING;\n console.debug(`正在连接小智接入点: ${sliceEndpoint(this.endpointUrl)}`);\n\n return new Promise((resolve, reject) => {\n // 设置连接超时\n this.connectionTimeout = setTimeout(() => {\n const error = new Error(\"连接超时 (10000ms)\");\n this.handleConnectionError(error);\n reject(error);\n }, 10000);\n\n this.ws = new WebSocket(this.endpointUrl);\n\n this.ws.on(\"open\", () => {\n this.handleConnectionSuccess();\n resolve();\n });\n\n this.ws.on(\"message\", (data) => {\n try {\n const message: MCPMessage = JSON.parse(data.toString());\n this.handleMessage(message);\n } catch (error) {\n console.error(\"MCP 消息解析错误:\", error);\n }\n });\n\n this.ws.on(\"close\", (code, reason) => {\n this.handleConnectionClose(code, reason.toString());\n });\n\n this.ws.on(\"error\", (error) => {\n this.handleConnectionError(error);\n reject(error);\n });\n });\n }\n\n /**\n * 处理连接成功\n */\n private handleConnectionSuccess(): void {\n // 清理连接超时定时器\n if (this.connectionTimeout) {\n clearTimeout(this.connectionTimeout);\n this.connectionTimeout = null;\n }\n\n this.connectionStatus = true;\n this.connectionState = ConnectionState.CONNECTED;\n\n console.debug(\"MCP WebSocket 连接已建立\");\n }\n\n /**\n * 处理连接错误\n */\n private handleConnectionError(error: Error): void {\n // 记录最后一次错误信息\n this.lastError = error.message;\n\n // 清理连接超时定时器\n if (this.connectionTimeout) {\n clearTimeout(this.connectionTimeout);\n this.connectionTimeout = null;\n }\n\n console.error(\"MCP WebSocket 错误:\", error.message);\n\n // 清理当前连接\n this.cleanupConnection();\n }\n\n /**\n * 处理连接关闭\n */\n private handleConnectionClose(code: number, reason: string): void {\n this.connectionStatus = false;\n this.serverInitialized = false;\n this.connectionState = ConnectionState.DISCONNECTED;\n console.info(`小智连接已关闭 (代码: ${code}, 原因: ${reason})`);\n }\n\n /**\n * 清理连接资源\n */\n private cleanupConnection(): void {\n // 清理 WebSocket\n if (this.ws) {\n this.ws.removeAllListeners();\n\n try {\n if (this.ws.readyState === WebSocket.OPEN) {\n this.ws.close(1000, \"Cleaning up connection\");\n } else if (this.ws.readyState === WebSocket.CONNECTING) {\n this.ws.terminate();\n }\n } catch (error) {\n console.debug(\"WebSocket 关闭时出现错误(已忽略):\", error);\n }\n\n this.ws = null;\n }\n\n // 清理连接超时定时器\n if (this.connectionTimeout) {\n clearTimeout(this.connectionTimeout);\n this.connectionTimeout = null;\n }\n\n // 重置连接状态\n this.connectionStatus = false;\n this.serverInitialized = false;\n\n // 重置连接状态为已断开\n this.connectionState = ConnectionState.DISCONNECTED;\n }\n\n /**\n * 处理 MCP 消息\n */\n private handleMessage(message: MCPMessage): void {\n // console.debug(\"收到 MCP 消息:\", JSON.stringify(message, null, 2));\n\n if (!message.method) {\n console.debug(\"收到没有 method 字段的消息,忽略\");\n return;\n }\n\n switch (message.method) {\n case \"initialize\":\n case \"notifications/initialized\":\n this.sendResponse(message.id, {\n protocolVersion: \"2024-11-05\",\n capabilities: {\n tools: { listChanged: true },\n logging: {},\n },\n serverInfo: {\n name: \"xiaozhi-mcp-server\",\n version: \"1.0.0\",\n },\n });\n this.serverInitialized = true;\n console.debug(\"MCP 服务器初始化完成\");\n break;\n\n case \"tools/list\": {\n const toolsList = this.getTools();\n this.sendResponse(message.id, { tools: toolsList });\n console.debug(`MCP 工具列表已发送 (${toolsList.length}个工具)`);\n break;\n }\n\n case \"tools/call\": {\n this.handleToolCall(message).catch((error) => {\n console.error(\"处理工具调用时发生未捕获错误:\", error);\n });\n break;\n }\n\n case \"ping\":\n this.sendResponse(message.id, {});\n // console.debug(\"回应 MCP ping 消息\");\n break;\n\n default:\n console.warn(`未知的 MCP 请求: ${message.method}`);\n }\n }\n\n /**\n * 发送响应消息\n */\n private sendResponse(id: number | string, result: unknown): void {\n // console.debug(\n // `尝试发送响应: id=${id}, isConnected=${this.connectionStatus}, wsReadyState=${this.ws?.readyState}`\n // );\n\n if (this.connectionStatus && this.ws?.readyState === WebSocket.OPEN) {\n const response: ExtendedMCPMessage = {\n jsonrpc: \"2.0\",\n id,\n result,\n };\n\n try {\n this.ws.send(JSON.stringify(response));\n console.debug(\"响应已发送\", {\n id,\n responseSize: JSON.stringify(response).length,\n });\n } catch (error) {\n console.error(\"发送响应失败\", {\n id,\n error,\n });\n }\n } else {\n console.error(\"无法发送响应\", {\n id,\n isConnected: this.connectionStatus,\n wsReadyState: this.ws?.readyState,\n });\n }\n }\n\n /**\n * 获取服务器状态\n */\n public getStatus(): EndpointConnectionStatus {\n const availableTools = this.mcpAdapter.getAllTools().length;\n\n return {\n connected: this.connectionStatus,\n initialized: this.serverInitialized,\n url: this.endpointUrl,\n availableTools,\n connectionState: this.connectionState,\n lastError: this.lastError,\n };\n }\n\n /**\n * 检查连接状态\n */\n public isConnected(): boolean {\n return this.connectionStatus;\n }\n\n /**\n * 主动断开小智连接\n */\n public async disconnect(): Promise<void> {\n console.info(\"主动断开小智连接\");\n\n // 清理 MCP 适配器\n await this.mcpAdapter.cleanup();\n\n // 清理 WebSocket 连接\n this.cleanupConnection();\n }\n\n /**\n * 重连小智接入点\n */\n public async reconnect(): Promise<void> {\n console.info(`重连小智接入点: ${sliceEndpoint(this.endpointUrl)}`);\n\n // 先断开连接\n this.disconnect();\n\n // 等待可配置的时间确保连接完全断开\n await new Promise((resolve) => setTimeout(resolve, this.reconnectDelay));\n\n // 重新连接\n await this.connect();\n }\n\n /**\n * 处理工具调用请求\n */\n private async handleToolCall(request: MCPMessage): Promise<void> {\n if (request.id === undefined || request.id === null) {\n throw new ToolCallErrorClass(\n ToolCallErrorCodeEnum.INVALID_PARAMS,\n \"请求 ID 不能为空\"\n );\n }\n\n const requestId = request.id;\n const startTime = Date.now();\n\n try {\n const params = validateToolCallParams(request.params);\n\n console.info(\"开始处理工具调用\", {\n requestId,\n toolName: params.name,\n hasArguments: !!params.arguments,\n });\n\n const result = await this.executeToolWithTimeout(\n params.name,\n params.arguments || {},\n this.toolCallTimeout\n );\n\n this.sendResponse(requestId, {\n content: result.content || [\n { type: \"text\", text: JSON.stringify(result) },\n ],\n isError: result.isError || false,\n });\n\n console.info(\"工具调用成功\", {\n requestId,\n toolName: params.name,\n duration: `${Date.now() - startTime}ms`,\n });\n } catch (error) {\n this.handleToolCallError(error, requestId, Date.now() - startTime);\n }\n }\n\n /**\n * 带超时控制的工具执行\n */\n private async executeToolWithTimeout(\n toolName: string,\n arguments_: Record<string, unknown>,\n timeoutMs = 30000\n ): Promise<ToolCallResult> {\n return new Promise((resolve, reject) => {\n const timeoutId = setTimeout(() => {\n reject(\n new ToolCallErrorClass(\n ToolCallErrorCodeEnum.TIMEOUT,\n `工具调用超时 (${timeoutMs}ms): ${toolName}`\n )\n );\n }, timeoutMs);\n\n this.mcpAdapter\n .callTool(toolName, arguments_)\n .then((result: ToolCallResult) => {\n clearTimeout(timeoutId);\n resolve(result);\n })\n .catch((error: unknown) => {\n clearTimeout(timeoutId);\n\n const errorMessage =\n error instanceof Error ? error.message : String(error);\n\n if (errorMessage.includes(\"未找到工具\")) {\n reject(\n new ToolCallErrorClass(\n ToolCallErrorCodeEnum.TOOL_NOT_FOUND,\n `工具不存在: ${toolName}`\n )\n );\n } else {\n reject(\n new ToolCallErrorClass(\n ToolCallErrorCodeEnum.TOOL_EXECUTION_ERROR,\n `工具执行失败: ${errorMessage}`\n )\n );\n }\n });\n });\n }\n\n /**\n * 处理工具调用错误\n */\n private handleToolCallError(\n error: unknown,\n requestId: string | number | undefined,\n duration: number\n ): void {\n let errorResponse: {\n code: number;\n message: string;\n data?: unknown;\n };\n\n if (error instanceof ToolCallErrorClass) {\n errorResponse = {\n code: error.code,\n message: error.message,\n data: error.data,\n };\n } else {\n const errorMessage = error instanceof Error ? error.message : \"未知错误\";\n errorResponse = {\n code: ToolCallErrorCodeEnum.TOOL_EXECUTION_ERROR,\n message: errorMessage,\n data: { originalError: String(error) || \"null\" },\n };\n }\n\n this.sendErrorResponse(requestId, errorResponse);\n\n console.error(\"工具调用失败\", {\n requestId,\n duration: `${duration}ms`,\n error: errorResponse,\n });\n }\n\n /**\n * 发送错误响应\n */\n private sendErrorResponse(\n id: string | number | undefined,\n error: { code: number; message: string; data?: unknown }\n ): void {\n if (this.connectionStatus && this.ws?.readyState === WebSocket.OPEN) {\n const response = {\n jsonrpc: \"2.0\",\n id,\n error,\n };\n try {\n this.ws.send(JSON.stringify(response));\n console.debug(\"已发送错误响应:\", response);\n } catch (sendError) {\n // 发送错误响应失败,记录日志但不抛出异常\n console.error(\"发送错误响应失败:\", {\n id,\n errorResponse: error,\n sendError: sendError instanceof Error\n ? { message: sendError.message, stack: sendError.stack }\n : sendError,\n });\n }\n } else {\n console.error(\"无法发送错误响应\", {\n id,\n isConnected: this.connectionStatus,\n wsReadyState: this.ws?.readyState,\n });\n }\n }\n}\n","/**\n * Endpoint 包核心类型定义\n *\n * 定义小智接入点相关的所有核心类型,包括:\n * - 工具调用相关类型(ToolCallResult、ToolCallParams 等)\n * - JSON Schema 类型定义\n * - 工具信息类型(EnhancedToolInfo 等)\n * - MCP 服务配置类型\n * - 连接状态类型\n *\n * @module types\n */\n\n// =========================\n// 1. 工具调用相关类型\n// =========================\n\n/**\n * 工具调用结果接口\n * 使用更宽松的类型定义以兼容不同来源的 ToolCallResult\n */\nexport interface ToolCallResult {\n content: Array<Record<string, unknown>>;\n isError?: boolean;\n _meta?: Record<string, unknown>;\n toolResult?: unknown; // 支持旧协议版本\n [key: string]: unknown; // 支持其他未知字段\n}\n\n/**\n * 工具调用参数接口\n */\nexport interface ToolCallParams {\n name: string;\n arguments?: Record<string, unknown>;\n}\n\n/**\n * 验证后的工具调用参数\n */\nexport interface ValidatedToolCallParams {\n name: string;\n arguments?: Record<string, unknown>;\n}\n\n/**\n * 工具调用错误码枚举\n */\nexport enum ToolCallErrorCode {\n /** 无效参数 */\n INVALID_PARAMS = -32602,\n /** 工具不存在 */\n TOOL_NOT_FOUND = -32601,\n /** 服务不可用 */\n SERVICE_UNAVAILABLE = -32001,\n /** 调用超时 */\n TIMEOUT = -32002,\n /** 工具执行错误 */\n TOOL_EXECUTION_ERROR = -32000,\n}\n\n/**\n * 工具调用错误类\n */\nexport class ToolCallError extends Error {\n constructor(\n public code: ToolCallErrorCode,\n message: string,\n public data?: unknown\n ) {\n super(message);\n this.name = \"ToolCallError\";\n }\n}\n\n// =========================\n// 2. JSON Schema 类型\n// =========================\n\n// 从 mcp-core 重新导出 JSONSchema 类型和相关函数,避免重复定义\nimport type { JSONSchema } from \"@xiaozhi-client/mcp-core\";\nimport { ensureToolJSONSchema } from \"@xiaozhi-client/mcp-core\";\n\n// 重新导出供外部使用\nexport type { JSONSchema };\nexport { ensureToolJSONSchema };\n\n// =========================\n// 3. 工具信息类型\n// =========================\n\n/**\n * 增强的工具信息接口\n * 包含工具的启用状态和使用统计信息\n */\nexport interface EnhancedToolInfo {\n /** 工具唯一标识符,格式为 \"{serviceName}__{originalName}\" */\n name: string;\n\n /** 工具描述信息 */\n description: string;\n\n /** 工具输入参数的 JSON Schema 定义 */\n inputSchema: JSONSchema;\n\n /** 工具所属的 MCP 服务名称 */\n serviceName: string;\n\n /** 工具在 MCP 服务中的原始名称 */\n originalName: string;\n\n /** 工具是否启用 */\n enabled: boolean;\n\n /** 工具使用次数统计 */\n usageCount: number;\n\n /** 工具最后使用时间 (ISO 8601 格式字符串) */\n lastUsedTime: string;\n}\n\n/**\n * MCP 服务管理器接口\n * 用于工具调用,避免循环依赖\n */\nexport interface IMCPServiceManager {\n /** 获取所有工具列表 */\n getAllTools(): EnhancedToolInfo[];\n\n /** 调用工具 */\n callTool(\n toolName: string,\n arguments_: Record<string, unknown>\n ): Promise<ToolCallResult>;\n\n /** 初始化 */\n initialize(): Promise<void>;\n\n /** 清理资源 */\n cleanup(): Promise<void>;\n}\n\n// =========================\n// 4. 连接状态类型\n// =========================\n\n/**\n * 连接状态枚举\n */\nexport enum ConnectionState {\n DISCONNECTED = \"disconnected\",\n CONNECTING = \"connecting\",\n CONNECTED = \"connected\",\n FAILED = \"failed\",\n}\n\n/**\n * 连接选项接口\n */\nexport interface ConnectionOptions {\n /** 连接超时时间(毫秒),默认 10000 */\n connectionTimeout?: number;\n /** 重连延迟时间(毫秒),默认 2000 */\n reconnectDelay?: number;\n}\n\n// =========================\n// 5. EndpointConnection 状态类型\n// =========================\n\n/**\n * EndpointConnection 状态接口\n */\nexport interface EndpointConnectionStatus {\n /** 是否已连接 */\n connected: boolean;\n /** 是否已初始化 */\n initialized: boolean;\n /** 接入点 URL */\n url: string;\n /** 可用工具数量 */\n availableTools: number;\n /** 连接状态 */\n connectionState: ConnectionState;\n /** 最后一次错误信息 */\n lastError: string | null;\n}\n\n// =========================\n// 6. EndpointManager 状态类型\n// =========================\n\n/**\n * 简单连接状态接口\n */\nexport interface SimpleConnectionStatus {\n /** 接入点地址 */\n endpoint: string;\n /** 是否已连接 */\n connected: boolean;\n /** 是否已初始化 */\n initialized: boolean;\n /** 最后连接时间 */\n lastConnected?: Date;\n /** 最后错误信息 */\n lastError?: string;\n}\n\n/**\n * 完整连接状态接口(扩展 SimpleConnectionStatus)\n */\nexport interface ConnectionStatus extends SimpleConnectionStatus {\n // 扩展字段可以在这里添加\n}\n\n/**\n * 配置变更事件类型\n */\nexport interface ConfigChangeEvent {\n type:\n | \"endpoints_added\"\n | \"endpoints_removed\"\n | \"endpoints_updated\"\n | \"options_updated\";\n data: {\n added?: string[];\n removed?: string[];\n updated?: string[];\n oldOptions?: Partial<ConnectionOptions>;\n newOptions?: Partial<ConnectionOptions>;\n };\n timestamp: Date;\n}\n\n/**\n * 重连结果接口\n */\nexport interface ReconnectResult {\n successCount: number;\n failureCount: number;\n results: Array<{\n endpoint: string;\n success: boolean;\n error?: string;\n }>;\n}\n\n// =========================\n// 7. 新 API 配置类型\n// =========================\n\n/**\n * MCP 服务器配置类型\n * 支持三种配置方式:\n * 1. 本地命令 (stdio): { command: string; args: string[]; env?: Record<string, string> }\n * 2. SSE: { type: \"sse\"; url: string; headers?: Record<string, string> }\n * 3. HTTP: { type?: \"http\"; url: string; headers?: Record<string, string> }\n *\n * 向后兼容:自动将 streamable-http/streamable_http/streamableHttp 转换为 http\n */\nexport type MCPServerConfig =\n | LocalMCPServerConfig\n | SSEMCPServerConfig\n | HTTPMCPServerConfig;\n\n/**\n * 本地 MCP 服务器配置\n */\nexport interface LocalMCPServerConfig {\n command: string;\n args: string[];\n env?: Record<string, string>;\n}\n\n/**\n * SSE MCP 服务器配置\n */\nexport interface SSEMCPServerConfig {\n type: \"sse\";\n url: string;\n headers?: Record<string, string>;\n}\n\n/**\n * HTTP MCP 服务器配置\n * 使用 type: \"http\"\n * 向后兼容 streamable-http 写法\n */\nexport interface HTTPMCPServerConfig {\n type?: \"http\" | \"streamable-http\"; // 可选,默认就是 http\n url: string;\n headers?: Record<string, string>;\n}\n\n// 向后兼容的别名\n/** @deprecated 使用 HTTPMCPServerConfig 代替 */\nexport type StreamableHTTPMCPServerConfig = HTTPMCPServerConfig;\n\n/**\n * Endpoint 配置接口\n * @deprecated 不再使用,Endpoint 构造函数改为接收 IMCPServiceManager\n */\nexport interface EndpointConfig {\n /** MCP 服务器配置(声明式) */\n mcpServers: Record<string, MCPServerConfig>;\n /** 可选:重连延迟(毫秒),默认 2000 */\n reconnectDelay?: number;\n /** 可选:ModelScope API Key(全局) */\n modelscopeApiKey?: string;\n}\n\n/**\n * EndpointManager 配置接口\n */\nexport interface EndpointManagerConfig {\n /** 可选:默认重连延迟(毫秒) */\n defaultReconnectDelay?: number;\n}\n\n// =========================\n// 8. JWT Token 类型\n// =========================\n\n/**\n * 小智平台 JWT Token Payload 接口\n *\n * @example\n * ```typescript\n * // 从 endpoint URL 解码得到的 payload\n * const payload: XiaozhiTokenPayload = {\n * userId: 302720,\n * agentId: 1324149,\n * endpointId: \"agent_1324149\",\n * purpose: \"mcp-endpoint\",\n * iat: 1768480930,\n * exp: 1800038530\n * };\n * ```\n */\nexport interface XiaozhiTokenPayload {\n /** 用户 ID */\n userId: number;\n /** 代理 ID */\n agentId: number;\n /** 接入点 ID,格式为 \"agent_{agentId}\" */\n endpointId: string;\n /** Token 用途 */\n purpose: string;\n /** 签发时间(Unix 时间戳) */\n iat: number;\n /** 过期时间(Unix 时间戳) */\n exp: number;\n}\n\n/**\n * 解析后的 Endpoint URL 信息\n */\nexport interface ParsedEndpointInfo {\n /** 完整的 endpoint URL */\n url: string;\n /** 提取的 JWT Token */\n token: string;\n /** 解码后的 Token Payload */\n payload: XiaozhiTokenPayload;\n /** WebSocket 服务器地址(不含 token 参数) */\n wsUrl: string;\n}\n","/**\n * 工具函数模块\n *\n * 内联的必要工具函数,确保包的独立性\n */\n\nimport type {\n ParsedEndpointInfo,\n ValidatedToolCallParams,\n XiaozhiTokenPayload,\n} from \"./types.js\";\n\n/**\n * 截断端点 URL 用于日志显示\n *\n * @param endpoint - 完整的端点 URL\n * @returns 截断后的 URL\n *\n * @example\n * ```typescript\n * sliceEndpoint(\"ws://very-long-endpoint-url-here.example.com/endpoint\")\n * // 返回: \"ws://very-long-endpoint-u...e.com/endpoint\"\n * ```\n */\nexport function sliceEndpoint(endpoint: string): string {\n return `${endpoint.slice(0, 30)}...${endpoint.slice(-10)}`;\n}\n\n/**\n * 验证工具调用参数\n *\n * @param params - 待验证的参数\n * @returns 验证后的参数\n * @throws {Error} 如果参数无效\n *\n * @example\n * ```typescript\n * const params = validateToolCallParams({\n * name: \"test_tool\",\n * arguments: { foo: \"bar\" }\n * });\n * ```\n */\nexport function validateToolCallParams(\n params: unknown\n): ValidatedToolCallParams {\n // 基础类型检查\n if (!params || typeof params !== \"object\") {\n throw new Error(\"工具调用参数必须是对象\");\n }\n\n const p = params as Record<string, unknown>;\n\n // 验证工具名称\n if (!p.name || typeof p.name !== \"string\") {\n throw new Error(\"工具名称必须是字符串\");\n }\n\n // 构建验证后的参数\n const validated: ValidatedToolCallParams = {\n name: p.name,\n };\n\n // 验证参数(可选)\n if (p.arguments !== undefined) {\n if (typeof p.arguments !== \"object\" || p.arguments === null) {\n throw new Error(\"工具参数必须是对象\");\n }\n validated.arguments = p.arguments as Record<string, unknown>;\n }\n\n return validated;\n}\n\n/**\n * 验证端点 URL 格式\n *\n * @param endpoint - 待验证的端点 URL\n * @returns 是否为有效的 WebSocket URL\n */\nexport function isValidEndpointUrl(endpoint: string): boolean {\n if (!endpoint || typeof endpoint !== \"string\") {\n return false;\n }\n\n if (!endpoint.startsWith(\"ws://\") && !endpoint.startsWith(\"wss://\")) {\n return false;\n }\n\n try {\n new URL(endpoint);\n return true;\n } catch {\n return false;\n }\n}\n\n/**\n * 检查属性名是否安全,防止原型污染攻击\n *\n * @param key - 待检查的属性名\n * @returns 是否为安全的属性名\n *\n * @example\n * ```typescript\n * isSafeKey(\"name\") // true\n * isSafeKey(\"__proto__\") // false\n * isSafeKey(\"constructor\") // false\n * ```\n */\nfunction isSafeKey(key: string): boolean {\n // 拒绝已知的危险属性名\n const dangerousKeys = [\"__proto__\", \"constructor\", \"prototype\"];\n if (dangerousKeys.includes(key)) {\n return false;\n }\n\n // 确保属性名是普通的字符串(不是 Symbol 等)\n return typeof key === \"string\" && key.length > 0;\n}\n\n/**\n * 深度合并对象\n *\n * @param target - 目标对象\n * @param sources - 源对象\n * @returns 合并后的对象\n *\n * @example\n * ```typescript\n * const result = deepMerge(\n * { a: 1, b: { x: 1 } },\n * { b: { y: 2 }, c: 3 }\n * );\n * // 返回: { a: 1, b: { x: 1, y: 2 }, c: 3 }\n * ```\n */\nexport function deepMerge<T>(\n target: Partial<T>,\n ...sources: Array<Partial<T>>\n): T {\n if (sources.length === 0) {\n return target as T;\n }\n\n const source = sources.shift();\n\n if (source === undefined) {\n return target as T;\n }\n\n // 使用 Object.keys() 替代 for...in,只遍历对象自身的可枚举属性\n const keys = Object.keys(source as Record<string, unknown>);\n\n for (const key of keys) {\n // 跳过不安全的 key,防止原型污染攻击\n if (!isSafeKey(key)) {\n continue;\n }\n\n const sourceValue = (source as Record<string, unknown>)[key];\n const targetValue = (target as Record<string, unknown>)[key];\n\n if (\n typeof sourceValue === \"object\" &&\n sourceValue !== null &&\n !Array.isArray(sourceValue) &&\n typeof targetValue === \"object\" &&\n targetValue !== null &&\n !Array.isArray(targetValue)\n ) {\n (target as Record<string, unknown>)[key] = deepMerge(\n targetValue as Record<string, unknown>,\n sourceValue as Record<string, unknown>\n );\n } else {\n (target as Record<string, unknown>)[key] = sourceValue;\n }\n }\n\n return deepMerge(target, ...sources);\n}\n\n/**\n * 延迟执行\n *\n * @param ms - 延迟时间(毫秒)\n * @returns Promise\n */\nexport function sleep(ms: number): Promise<void> {\n return new Promise((resolve) => setTimeout(resolve, ms));\n}\n\n/**\n * 格式化错误消息\n *\n * @param error - 错误对象\n * @returns 格式化后的错误消息\n */\nexport function formatErrorMessage(error: unknown): string {\n if (error instanceof Error) {\n return error.message;\n }\n\n if (typeof error === \"string\") {\n return error;\n }\n\n return String(error);\n}\n\n// =========================\n// JWT Token 解码相关函数\n// =========================\n\n/**\n * Base64URL 解码\n *\n * @param input - Base64URL 编码的字符串\n * @returns 解码后的字符串\n *\n * @example\n * ```typescript\n * base64UrlDecode(\"SGVsbG8gV29ybGQ\") // 返回: \"Hello World\"\n * ```\n */\nfunction base64UrlDecode(input: string): string {\n // 将 Base64URL 格式转换为标准 Base64 格式\n let base64 = input.replace(/-/g, \"+\").replace(/_/g, \"/\");\n\n // 补全 padding\n while (base64.length % 4) {\n base64 += \"=\";\n }\n\n // 使用 Node.js Buffer 进行解码\n return Buffer.from(base64, \"base64\").toString(\"utf-8\");\n}\n\n/**\n * 解码 JWT Token(仅解析 payload,不验证签名)\n *\n * @param token - JWT Token 字符串\n * @returns 解码后的 Token Payload,解码失败返回 null\n *\n * @example\n * ```typescript\n * const payload = decodeJWTToken(\"eyJ...token...\");\n * if (payload) {\n * console.log(payload.endpointId); // \"agent_1324149\"\n * }\n * ```\n */\nexport function decodeJWTToken(token: string): XiaozhiTokenPayload | null {\n if (!token || typeof token !== \"string\") {\n return null;\n }\n\n try {\n // JWT 格式: header.payload.signature\n const parts = token.split(\".\");\n if (parts.length !== 3) {\n return null;\n }\n\n // 解码 payload 部分(第二部分)\n const payloadStr = base64UrlDecode(parts[1]);\n const payload = JSON.parse(payloadStr) as unknown;\n\n // 验证 payload 结构\n if (\n typeof payload !== \"object\" ||\n payload === null ||\n !(\"userId\" in payload) ||\n !(\"agentId\" in payload) ||\n !(\"endpointId\" in payload) ||\n !(\"purpose\" in payload) ||\n !(\"iat\" in payload) ||\n !(\"exp\" in payload)\n ) {\n return null;\n }\n\n return payload as XiaozhiTokenPayload;\n } catch {\n return null;\n }\n}\n\n/**\n * 从 endpoint URL 中提取 token 参数\n *\n * @param url - 完整的 endpoint URL\n * @returns 提取的 token 字符串,未找到返回 null\n *\n * @example\n * ```typescript\n * const token = extractTokenFromUrl(\n * \"wss://api.xiaozhi.me/mcp/?token=eyJ...\"\n * );\n * ```\n */\nexport function extractTokenFromUrl(url: string): string | null {\n if (!url || typeof url !== \"string\") {\n return null;\n }\n\n try {\n const urlObj = new URL(url);\n return urlObj.searchParams.get(\"token\");\n } catch {\n return null;\n }\n}\n\n/**\n * 解析 endpoint URL 获取完整信息\n *\n * @param url - 完整的 endpoint URL\n * @returns 解析后的 endpoint 信息,解析失败返回 null\n *\n * @example\n * ```typescript\n * const info = parseEndpointUrl(\n * \"wss://api.xiaozhi.me/mcp/?token=eyJ...\"\n * );\n * if (info) {\n * console.log(info.payload.endpointId); // \"agent_1324149\"\n * console.log(info.wsUrl); // \"wss://api.xiaozhi.me/mcp/\"\n * }\n * ```\n */\nexport function parseEndpointUrl(url: string): ParsedEndpointInfo | null {\n if (!url || typeof url !== \"string\") {\n return null;\n }\n\n try {\n // 提取 token\n const token = extractTokenFromUrl(url);\n if (!token) {\n return null;\n }\n\n // 解码 token\n const payload = decodeJWTToken(token);\n if (!payload) {\n return null;\n }\n\n // 移除 token 参数得到纯净的 WebSocket URL\n const urlObj = new URL(url);\n urlObj.searchParams.delete(\"token\");\n const wsUrl = urlObj.toString();\n\n return {\n url,\n token,\n payload,\n wsUrl,\n };\n } catch {\n return null;\n }\n}\n","/**\n * EndpointManager\n * 管理多个小智接入点的连接,共享外部传入的 MCPManager\n *\n * 使用方式:\n * ```typescript\n * // 1. 先创建并配置 MCPManager\n * const mcpManager = new MCPManager();\n * mcpManager.addServer(\"calculator\", { command: \"npx\", args: [\"-y\", \"calculator-mcp\"] });\n * await mcpManager.connect();\n *\n * // 2. 创建 EndpointManager 并设置 MCPManager\n * const manager = new EndpointManager();\n * manager.setMcpManager(mcpManager);\n *\n * // 3. 添加接入点\n * manager.addEndpoint(\"ws://endpoint1\");\n * await manager.connect();\n * ```\n */\n\nimport { EventEmitter } from \"node:events\";\nimport type { Endpoint } from \"./endpoint.js\";\nimport { Endpoint as EndpointClass } from \"./endpoint.js\";\nimport { SharedMCPAdapter } from \"./shared-mcp-adapter.js\";\nimport type {\n EndpointManagerConfig,\n IMCPServiceManager,\n SimpleConnectionStatus,\n} from \"./types.js\";\nimport { sliceEndpoint } from \"./utils.js\";\n\n/**\n * MCP 服务连接事件数据\n */\ninterface MCPConnectedEvent {\n serverName: string;\n tools?: Array<unknown>;\n}\n\n/**\n * MCP 服务错误事件数据\n */\ninterface MCPErrorEvent {\n serverName: string;\n error: Error | unknown;\n}\n\n/**\n * 小智接入点管理器\n * 负责管理多个小智接入点的连接,共享 MCP 服务\n *\n * 注意:MCPManager 必须通过 setMcpManager() 方法设置,且由外部管理其生命周期\n */\nexport class EndpointManager extends EventEmitter {\n private endpoints: Map<string, Endpoint> = new Map();\n private connectionStates: Map<string, SimpleConnectionStatus> = new Map();\n private mcpManager: IMCPServiceManager | null = null;\n private sharedMCPAdapter: SharedMCPAdapter | null = null;\n private mcpEventListeners: Array<(...args: unknown[]) => void> = [];\n\n /**\n * 构造函数\n *\n * @param config - 可选的配置\n */\n constructor(private config?: EndpointManagerConfig) {\n super();\n console.debug(\"[EndpointManager] 实例已创建\");\n }\n\n /**\n * 设置 MCPManager 实例\n *\n * 注意:MCPManager 的生命周期由外部管理,EndpointManager 不负责连接和断开\n *\n * @param mcpManager - 外部创建并已连接的 MCPManager 实例\n */\n setMcpManager(mcpManager: IMCPServiceManager): void {\n if (this.sharedMCPAdapter) {\n throw new Error(\"MCPManager 已经设置,不能重复设置\");\n }\n\n this.mcpManager = mcpManager;\n this.sharedMCPAdapter = new SharedMCPAdapter(mcpManager);\n\n // 监听 MCP 服务连接事件(如果支持 EventEmitter)\n if (\"on\" in mcpManager && typeof mcpManager.on === \"function\") {\n const connectedHandler = (...args: unknown[]) => {\n const data = args[0] as MCPConnectedEvent;\n console.info(\n `[EndpointManager] MCP 服务已连接: ${data.serverName}, 工具数: ${data.tools?.length || 0}`\n );\n };\n const errorHandler = (...args: unknown[]) => {\n const data = args[0] as MCPErrorEvent;\n console.error(\n `[EndpointManager] MCP 服务连接失败: ${data.serverName}`,\n data.error\n );\n };\n\n mcpManager.on(\"connected\", connectedHandler);\n mcpManager.on(\"error\", errorHandler);\n\n // 保存监听器引用以便后续清理\n this.mcpEventListeners.push(connectedHandler, errorHandler);\n }\n\n console.info(\"[EndpointManager] MCPManager 已设置\");\n }\n\n /**\n * 添加 Endpoint(支持 URL 字符串或 Endpoint 实例)\n *\n * @param endpoint - Endpoint URL 字符串或 Endpoint 实例\n */\n addEndpoint(endpoint: string | Endpoint): void {\n // 如果是字符串,创建新的 Endpoint 实例\n if (typeof endpoint === \"string\") {\n if (!this.sharedMCPAdapter) {\n throw new Error(\n \"MCPManager 未设置,请先调用 setMcpManager() 方法设置 MCPManager\"\n );\n }\n\n const endpointInstance = new EndpointClass(\n endpoint,\n this.sharedMCPAdapter,\n this.config?.defaultReconnectDelay\n );\n\n this.addEndpointInternal(endpointInstance);\n return;\n }\n\n // 原有的 Endpoint 实例处理逻辑\n // 当 EndpointManager 已配置共享 MCPManager 时,不允许再传入外部构造的 Endpoint 实例,\n // 以避免不同 Endpoint 使用不同的 MCP 管理器,破坏共享 MCPManager 的设计。\n if (this.sharedMCPAdapter) {\n throw new Error(\n \"[EndpointManager] 当使用共享 MCPManager 时,不支持传入自定义 Endpoint 实例,请传入 Endpoint URL 字符串以便由 EndpointManager 创建实例\"\n );\n }\n\n this.addEndpointInternal(endpoint);\n }\n\n /**\n * 内部添加 Endpoint 方法\n */\n private addEndpointInternal(endpoint: Endpoint): void {\n const url = endpoint.getUrl();\n\n if (this.endpoints.has(url)) {\n console.debug(\n `[EndpointManager] 接入点 ${sliceEndpoint(url)} 已存在,跳过添加`\n );\n return;\n }\n\n console.debug(`[EndpointManager] 添加接入点: ${sliceEndpoint(url)}`);\n\n this.endpoints.set(url, endpoint);\n this.connectionStates.set(url, {\n endpoint: url,\n connected: false,\n initialized: false,\n });\n\n // 发射事件\n this.emit(\"endpointAdded\", { endpoint: url });\n }\n\n /**\n * 移除 Endpoint 实例\n *\n * @param endpoint - Endpoint 实例\n */\n async removeEndpoint(endpoint: Endpoint): Promise<void> {\n const url = endpoint.getUrl();\n\n if (!this.endpoints.has(url)) {\n console.debug(\n `[EndpointManager] 接入点 ${sliceEndpoint(url)} 不存在,跳过移除`\n );\n return;\n }\n\n console.debug(`[EndpointManager] 移除接入点: ${sliceEndpoint(url)}`);\n\n // 断开连接\n await endpoint.disconnect();\n\n // 清理状态\n this.endpoints.delete(url);\n this.connectionStates.delete(url);\n\n // 发射事件\n this.emit(\"endpointRemoved\", { endpoint: url });\n }\n\n /**\n * 连接 Endpoint\n *\n * 注意:此方法不负责连接 MCPManager,MCPManager 必须在调用此方法前已连接\n *\n * @param endpoint - 可选,指定要连接的端点 URL。如果不传入,则连接所有端点\n */\n async connect(endpoint?: string): Promise<void> {\n // 如果指定了端点,只连接该端点\n if (endpoint) {\n const endpointInstance = this.endpoints.get(endpoint);\n if (!endpointInstance) {\n throw new Error(`接入点不存在: ${sliceEndpoint(endpoint)}`);\n }\n\n const status = this.connectionStates.get(endpoint);\n if (status?.connected) {\n console.debug(\n `[EndpointManager] 接入点已连接,跳过: ${sliceEndpoint(endpoint)}`\n );\n return;\n }\n\n await this.connectSingleEndpoint(endpoint, endpointInstance);\n return;\n }\n\n // 连接所有未连接的 Endpoint\n console.debug(\n `[EndpointManager] 开始连接接入点,总数: ${this.endpoints.size}`\n );\n\n const promises: Promise<void>[] = [];\n\n for (const [url, endpoint] of this.endpoints) {\n const status = this.connectionStates.get(url);\n\n // 跳过已连接的端点\n if (status?.connected) {\n console.debug(\n `[EndpointManager] 接入点已连接,跳过: ${sliceEndpoint(url)}`\n );\n continue;\n }\n\n promises.push(\n this.connectSingleEndpoint(url, endpoint).catch((error) => {\n console.error(\n `[EndpointManager] 连接失败: ${sliceEndpoint(url)}`,\n error\n );\n // 更新失败状态\n const status = this.connectionStates.get(url);\n if (status) {\n status.connected = false;\n status.initialized = false;\n status.lastError =\n error instanceof Error ? error.message : String(error);\n }\n })\n );\n }\n\n await Promise.allSettled(promises);\n\n // 统计连接结果\n const connectedCount = Array.from(this.connectionStates.values()).filter(\n (s) => s.connected\n ).length;\n\n console.info(\n `[EndpointManager] 连接完成: 成功 ${connectedCount}/${this.endpoints.size}`\n );\n }\n\n /**\n * 断开连接\n *\n * 注意:此方法不断开 MCPManager,MCPManager 的生命周期由外部管理\n *\n * @param endpoint - 可选,指定要断开的端点 URL。如果不传入,则断开所有端点\n */\n async disconnect(endpoint?: string): Promise<void> {\n // 如果指定了端点,只断开该端点\n if (endpoint) {\n const endpointInstance = this.endpoints.get(endpoint);\n if (!endpointInstance) {\n throw new Error(`接入点不存在: ${sliceEndpoint(endpoint)}`);\n }\n\n endpointInstance.disconnect();\n\n const status = this.connectionStates.get(endpoint);\n if (status) {\n status.connected = false;\n status.initialized = false;\n }\n\n console.debug(\n `[EndpointManager] 接入点已断开: ${sliceEndpoint(endpoint)}`\n );\n return;\n }\n\n // 断开所有连接\n console.debug(\"[EndpointManager] 开始断开所有连接\");\n\n const promises: Promise<void>[] = [];\n\n for (const endpoint of this.endpoints.values()) {\n promises.push(\n Promise.resolve().then(() => {\n endpoint.disconnect();\n })\n );\n }\n\n await Promise.allSettled(promises);\n\n // 重置所有状态\n for (const status of this.connectionStates.values()) {\n status.connected = false;\n status.initialized = false;\n }\n\n console.debug(\"[EndpointManager] 所有接入点已断开连接\");\n }\n\n /**\n * 获取所有 Endpoint URL\n */\n getEndpoints(): string[] {\n return Array.from(this.endpoints.keys());\n }\n\n /**\n * 获取指定 Endpoint 实例\n *\n * @param url - Endpoint URL\n */\n getEndpoint(url: string): Endpoint | undefined {\n return this.endpoints.get(url);\n }\n\n /**\n * 获取所有连接状态\n */\n getConnectionStatus(): SimpleConnectionStatus[] {\n return Array.from(this.connectionStates.values());\n }\n\n /**\n * 检查是否有任何连接处于连接状态\n */\n isAnyConnected(): boolean {\n for (const status of this.connectionStates.values()) {\n if (status.connected) {\n return true;\n }\n }\n return false;\n }\n\n /**\n * 检查指定端点是否已连接\n *\n * @param url - 端点 URL\n */\n isEndpointConnected(url: string): boolean {\n const status = this.connectionStates.get(url);\n return status?.connected ?? false;\n }\n\n /**\n * 获取指定端点的状态\n *\n * @param url - 端点 URL\n */\n getEndpointStatus(url: string): SimpleConnectionStatus | undefined {\n return this.connectionStates.get(url);\n }\n\n /**\n * 重连\n *\n * @param endpoint - 可选,指定要重连的端点 URL。如果不传入,则重连所有端点\n * @param delay - 可选,disconnect 和 connect 之间的等待时间(毫秒),默认为 1000ms。\n * 注意:此参数只控制断开和重新连接之间的等待时间,不影响底层 Endpoint 实例的重连延迟\n */\n async reconnect(endpoint?: string, delay = 1000): Promise<void> {\n console.info(\"[EndpointManager] 开始重连\");\n\n // 先断开连接\n await this.disconnect(endpoint);\n\n // 等待一段时间\n console.debug(`[EndpointManager] 等待 ${delay}ms 后重连`);\n await new Promise((resolve) => setTimeout(resolve, delay));\n\n // 再重新连接\n await this.connect(endpoint);\n\n console.info(\"[EndpointManager] 重连完成\");\n }\n\n /**\n * 重连所有端点\n */\n async reconnectAll(): Promise<void> {\n console.info(\"[EndpointManager] 开始重连所有接入点\");\n\n const promises: Promise<void>[] = [];\n\n for (const [url, endpoint] of this.endpoints) {\n promises.push(\n this.reconnectSingleEndpoint(url, endpoint).catch((error) => {\n console.error(\n `[EndpointManager] 重连失败: ${sliceEndpoint(url)}`,\n error\n );\n })\n );\n }\n\n await Promise.allSettled(promises);\n }\n\n /**\n * 重连指定的端点\n *\n * @param url - 要重连的端点 URL\n */\n async reconnectEndpoint(url: string): Promise<void> {\n const endpoint = this.endpoints.get(url);\n if (!endpoint) {\n throw new Error(`接入点不存在: ${sliceEndpoint(url)}`);\n }\n\n await this.reconnectSingleEndpoint(url, endpoint);\n }\n\n /**\n * 清除所有端点\n *\n * 注意:此方法不会清理 MCPManager,MCPManager 的生命周期由外部管理\n */\n async clearEndpoints(): Promise<void> {\n console.debug(\"[EndpointManager] 清除所有接入点\");\n\n await this.disconnect();\n\n this.endpoints.clear();\n this.connectionStates.clear();\n\n // 注意:不清理 MCPManager,由外部管理\n // this.mcpManager = null;\n // this.sharedMCPAdapter = null;\n\n console.info(\"[EndpointManager] 所有接入点已清除\");\n }\n\n /**\n * 清理资源\n *\n * 注意:此方法不会清理 MCPManager,MCPManager 的生命周期由外部管理\n */\n async cleanup(): Promise<void> {\n console.debug(\"[EndpointManager] 开始清理资源\");\n\n // 移除 MCP 服务事件监听器\n if (\n this.mcpManager &&\n \"removeListener\" in this.mcpManager &&\n typeof this.mcpManager.removeListener === \"function\"\n ) {\n for (const listener of this.mcpEventListeners) {\n this.mcpManager.removeListener(\"connected\", listener);\n this.mcpManager.removeListener(\"error\", listener);\n }\n }\n this.mcpEventListeners = [];\n\n await this.clearEndpoints();\n\n console.debug(\"[EndpointManager] 资源清理完成\");\n }\n\n // ==================== 私有方法 ====================\n\n /**\n * 连接单个端点\n */\n private async connectSingleEndpoint(\n url: string,\n endpoint: Endpoint\n ): Promise<void> {\n const status = this.connectionStates.get(url);\n if (!status) {\n throw new Error(`端点状态不存在: ${sliceEndpoint(url)}`);\n }\n\n console.debug(`[EndpointManager] 连接端点: ${sliceEndpoint(url)}`);\n\n // 更新状态为连接中\n status.connected = false;\n status.initialized = false;\n\n // 执行连接\n await endpoint.connect();\n\n // 更新连接成功状态\n status.connected = true;\n status.initialized = true;\n status.lastConnected = new Date();\n status.lastError = undefined;\n\n console.info(`[EndpointManager] 端点连接成功: ${sliceEndpoint(url)}`);\n }\n\n /**\n * 重连单个端点\n */\n private async reconnectSingleEndpoint(\n url: string,\n endpoint: Endpoint\n ): Promise<void> {\n const status = this.connectionStates.get(url);\n if (!status) {\n throw new Error(`端点状态不存在: ${sliceEndpoint(url)}`);\n }\n\n console.debug(`[EndpointManager] 重连端点: ${sliceEndpoint(url)}`);\n\n // 执行重连\n await endpoint.reconnect();\n\n // 更新连接成功状态\n status.connected = true;\n status.initialized = true;\n status.lastConnected = new Date();\n status.lastError = undefined;\n\n console.info(`[EndpointManager] 端点重连成功: ${sliceEndpoint(url)}`);\n }\n}\n","/**\n * 共享 MCP 管理器适配器\n *\n * 接收全局 MCPManager 实例的引用,不创建独立连接\n * 用于多个 Endpoint 共享同一个 MCPManager 实例\n *\n * @example\n * ```typescript\n * const globalMCPManager = new MCPManager();\n * await globalMCPManager.connect();\n *\n * const adapter = new SharedMCPAdapter(globalMCPManager);\n * const endpoint = new Endpoint(\"ws://...\", adapter);\n * ```\n */\n\nimport type {\n IMCPServiceManager,\n EnhancedToolInfo,\n ToolCallResult,\n} from \"./types.js\";\n\n/**\n * 共享 MCP 管理器适配器\n * 实现 IMCPServiceManager 接口,将操作委托给全局 MCPManager\n */\nexport class SharedMCPAdapter implements IMCPServiceManager {\n private isInitialized = false;\n\n /**\n * 构造函数\n *\n * @param globalMCPManager - 全局 MCPManager 实例\n */\n constructor(private globalMCPManager: IMCPServiceManager) {\n if (!globalMCPManager) {\n throw new Error(\"全局 MCPManager 不能为空\");\n }\n }\n\n /**\n * 初始化适配器\n *\n * 注意:此方法不执行任何连接操作,仅标记初始化状态,因为全局 MCPManager 的生命周期由外部管理\n */\n async initialize(): Promise<void> {\n if (this.isInitialized) {\n return;\n }\n\n // 全局实例已经在外部连接,这里只标记状态\n this.isInitialized = true;\n }\n\n /**\n * 获取所有工具列表\n *\n * 从全局 MCPManager 获取工具列表,并转换为增强格式\n */\n getAllTools(): EnhancedToolInfo[] {\n // 直接使用 IMCPServiceManager 接口的 getAllTools() 方法\n return this.globalMCPManager.getAllTools();\n }\n\n /**\n * 调用工具\n *\n * 将工具调用委托给全局 MCPManager\n *\n * @param toolName - 工具名称(格式:serviceName__toolName)\n * @param arguments_ - 工具调用参数\n */\n async callTool(\n toolName: string,\n arguments_: Record<string, unknown>\n ): Promise<ToolCallResult> {\n // 直接使用 IMCPServiceManager 接口的 callTool() 方法\n // toolName 已经是完整格式(serviceName__toolName)\n return this.globalMCPManager.callTool(toolName, arguments_);\n }\n\n /**\n * 清理资源\n *\n * 注意:此方法不会断开全局 MCPManager,由创建者负责管理\n */\n async cleanup(): Promise<void> {\n this.isInitialized = false;\n // 不断开全局 MCPManager\n }\n}\n"],"mappings":";;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAMA,SAAS,kBAAkB;AAI3B,SAAS,8BAA8B;AAVvC,IAgBa;AAhBb;AAAA;AAAA;AAgBO,IAAM,4BAAN,MAA8D;AAAA,MAKnE,YAAoB,QAAwB;AAAxB;AAClB,aAAK,aAAa,IAAI,WAAW;AAGjC,mBAAW,CAAC,aAAa,YAAY,KAAK,OAAO;AAAA,UAC/C,OAAO;AAAA,QACT,GAAG;AACD,gBAAM,YAAY,uBAAuB,YAAY;AACrD,eAAK,WAAW,UAAU,aAAa,SAAS;AAAA,QAClD;AAGA,aAAK,WAAW,GAAG,aAAa,CAAC,SAAS;AACxC,kBAAQ;AAAA,YACN,oBAAU,KAAK,UAAU,gDAAa,KAAK,MAAM,MAAM;AAAA,UACzD;AAAA,QACF,CAAC;AAED,aAAK,WAAW,GAAG,SAAS,CAAC,SAAS;AACpC,kBAAQ,MAAM,oBAAU,KAAK,UAAU,kBAAQ,KAAK,KAAK;AAAA,QAC3D,CAAC;AAAA,MACH;AAAA,MA1CF,OAgBqE;AAAA;AAAA;AAAA,MAC3D;AAAA,MACA,QAAuC,oBAAI,IAAI;AAAA,MAC/C,gBAAgB;AAAA;AAAA;AAAA;AAAA,MA4BxB,MAAM,aAA4B;AAChC,YAAI,KAAK,eAAe;AACtB;AAAA,QACF;AAGA,cAAM,KAAK,WAAW,QAAQ;AAG9B,cAAM,KAAK,aAAa;AAExB,aAAK,gBAAgB;AAAA,MACvB;AAAA;AAAA;AAAA;AAAA,MAKA,cAAkC;AAChC,eAAO,MAAM,KAAK,KAAK,MAAM,OAAO,CAAC;AAAA,MACvC;AAAA;AAAA;AAAA;AAAA,MAKA,MAAM,SACJ,UACA,YACyB;AAEzB,cAAM,CAAC,aAAa,cAAc,IAAI,KAAK,cAAc,QAAQ;AAGjE,eAAQ,MAAM,KAAK,WAAW;AAAA,UAC5B;AAAA,UACA;AAAA,UACA;AAAA,QACF;AAAA,MACF;AAAA;AAAA;AAAA;AAAA,MAKA,MAAM,UAAyB;AAC7B,cAAM,KAAK,WAAW,WAAW;AACjC,aAAK,MAAM,MAAM;AACjB,aAAK,gBAAgB;AAAA,MACvB;AAAA;AAAA;AAAA;AAAA,MAKA,MAAc,eAA8B;AAC1C,aAAK,MAAM,MAAM;AAEjB,cAAM,WAAW,KAAK,WAAW,UAAU;AAE3C,mBAAW,WAAW,UAAU;AAC9B,gBAAM,eAAiC;AAAA,YACrC,MAAM,GAAG,QAAQ,UAAU,KAAK,QAAQ,IAAI;AAAA,YAC5C,aAAa,QAAQ;AAAA,YACrB,aAAa,QAAQ;AAAA,YACrB,aAAa,QAAQ;AAAA,YACrB,cAAc,QAAQ;AAAA,YACtB,SAAS;AAAA,YACT,YAAY;AAAA,YACZ,eAAc,oBAAI,KAAK,GAAE,YAAY;AAAA,UACvC;AACA,eAAK,MAAM,IAAI,aAAa,MAAM,YAAY;AAAA,QAChD;AAAA,MACF;AAAA;AAAA;AAAA;AAAA,MAKQ,cAAc,UAAoC;AACxD,cAAM,QAAQ,SAAS,MAAM,IAAI;AACjC,YAAI,MAAM,SAAS,GAAG;AACpB,gBAAM,IAAI,MAAM,2DAAc,QAAQ,EAAE;AAAA,QAC1C;AACA,cAAM,cAAc,MAAM,CAAC;AAC3B,cAAM,iBAAiB,MAAM,MAAM,CAAC,EAAE,KAAK,IAAI;AAC/C,eAAO,CAAC,aAAa,cAAc;AAAA,MACrC;AAAA,IACF;AAAA;AAAA;;;ACpHA,OAAO,eAAe;;;ACmEtB,SAAS,4BAA4B;AAjC9B,IAAK,oBAAL,kBAAKA,uBAAL;AAEL,EAAAA,sCAAA,oBAAiB,UAAjB;AAEA,EAAAA,sCAAA,oBAAiB,UAAjB;AAEA,EAAAA,sCAAA,yBAAsB,UAAtB;AAEA,EAAAA,sCAAA,aAAU,UAAV;AAEA,EAAAA,sCAAA,0BAAuB,SAAvB;AAVU,SAAAA;AAAA,GAAA;AAgBL,IAAM,gBAAN,cAA4B,MAAM;AAAA,EACvC,YACS,MACP,SACO,MACP;AACA,UAAM,OAAO;AAJN;AAEA;AAGP,SAAK,OAAO;AAAA,EACd;AAAA,EAxEF,OAgEyC;AAAA;AAAA;AASzC;AA4EO,IAAK,kBAAL,kBAAKC,qBAAL;AACL,EAAAA,iBAAA,kBAAe;AACf,EAAAA,iBAAA,gBAAa;AACb,EAAAA,iBAAA,eAAY;AACZ,EAAAA,iBAAA,YAAS;AAJC,SAAAA;AAAA,GAAA;;;AC7HL,SAAS,cAAc,UAA0B;AACtD,SAAO,GAAG,SAAS,MAAM,GAAG,EAAE,CAAC,MAAM,SAAS,MAAM,GAAG,CAAC;AAC1D;AAFgB;AAmBT,SAAS,uBACd,QACyB;AAEzB,MAAI,CAAC,UAAU,OAAO,WAAW,UAAU;AACzC,UAAM,IAAI,MAAM,oEAAa;AAAA,EAC/B;AAEA,QAAM,IAAI;AAGV,MAAI,CAAC,EAAE,QAAQ,OAAO,EAAE,SAAS,UAAU;AACzC,UAAM,IAAI,MAAM,8DAAY;AAAA,EAC9B;AAGA,QAAM,YAAqC;AAAA,IACzC,MAAM,EAAE;AAAA,EACV;AAGA,MAAI,EAAE,cAAc,QAAW;AAC7B,QAAI,OAAO,EAAE,cAAc,YAAY,EAAE,cAAc,MAAM;AAC3D,YAAM,IAAI,MAAM,wDAAW;AAAA,IAC7B;AACA,cAAU,YAAY,EAAE;AAAA,EAC1B;AAEA,SAAO;AACT;AA7BgB;AAqCT,SAAS,mBAAmB,UAA2B;AAC5D,MAAI,CAAC,YAAY,OAAO,aAAa,UAAU;AAC7C,WAAO;AAAA,EACT;AAEA,MAAI,CAAC,SAAS,WAAW,OAAO,KAAK,CAAC,SAAS,WAAW,QAAQ,GAAG;AACnE,WAAO;AAAA,EACT;AAEA,MAAI;AACF,QAAI,IAAI,QAAQ;AAChB,WAAO;AAAA,EACT,QAAQ;AACN,WAAO;AAAA,EACT;AACF;AAfgB;AA8BhB,SAAS,UAAU,KAAsB;AAEvC,QAAM,gBAAgB,CAAC,aAAa,eAAe,WAAW;AAC9D,MAAI,cAAc,SAAS,GAAG,GAAG;AAC/B,WAAO;AAAA,EACT;AAGA,SAAO,OAAO,QAAQ,YAAY,IAAI,SAAS;AACjD;AATS;AA2BF,SAAS,UACd,WACG,SACA;AACH,MAAI,QAAQ,WAAW,GAAG;AACxB,WAAO;AAAA,EACT;AAEA,QAAM,SAAS,QAAQ,MAAM;AAE7B,MAAI,WAAW,QAAW;AACxB,WAAO;AAAA,EACT;AAGA,QAAM,OAAO,OAAO,KAAK,MAAiC;AAE1D,aAAW,OAAO,MAAM;AAEtB,QAAI,CAAC,UAAU,GAAG,GAAG;AACnB;AAAA,IACF;AAEA,UAAM,cAAe,OAAmC,GAAG;AAC3D,UAAM,cAAe,OAAmC,GAAG;AAE3D,QACE,OAAO,gBAAgB,YACvB,gBAAgB,QAChB,CAAC,MAAM,QAAQ,WAAW,KAC1B,OAAO,gBAAgB,YACvB,gBAAgB,QAChB,CAAC,MAAM,QAAQ,WAAW,GAC1B;AACA,MAAC,OAAmC,GAAG,IAAI;AAAA,QACzC;AAAA,QACA;AAAA,MACF;AAAA,IACF,OAAO;AACL,MAAC,OAAmC,GAAG,IAAI;AAAA,IAC7C;AAAA,EACF;AAEA,SAAO,UAAU,QAAQ,GAAG,OAAO;AACrC;AA5CgB;AAoDT,SAAS,MAAM,IAA2B;AAC/C,SAAO,IAAI,QAAQ,CAAC,YAAY,WAAW,SAAS,EAAE,CAAC;AACzD;AAFgB;AAUT,SAAS,mBAAmB,OAAwB;AACzD,MAAI,iBAAiB,OAAO;AAC1B,WAAO,MAAM;AAAA,EACf;AAEA,MAAI,OAAO,UAAU,UAAU;AAC7B,WAAO;AAAA,EACT;AAEA,SAAO,OAAO,KAAK;AACrB;AAVgB;AA2BhB,SAAS,gBAAgB,OAAuB;AAE9C,MAAI,SAAS,MAAM,QAAQ,MAAM,GAAG,EAAE,QAAQ,MAAM,GAAG;AAGvD,SAAO,OAAO,SAAS,GAAG;AACxB,cAAU;AAAA,EACZ;AAGA,SAAO,OAAO,KAAK,QAAQ,QAAQ,EAAE,SAAS,OAAO;AACvD;AAXS;AA2BF,SAAS,eAAe,OAA2C;AACxE,MAAI,CAAC,SAAS,OAAO,UAAU,UAAU;AACvC,WAAO;AAAA,EACT;AAEA,MAAI;AAEF,UAAM,QAAQ,MAAM,MAAM,GAAG;AAC7B,QAAI,MAAM,WAAW,GAAG;AACtB,aAAO;AAAA,IACT;AAGA,UAAM,aAAa,gBAAgB,MAAM,CAAC,CAAC;AAC3C,UAAM,UAAU,KAAK,MAAM,UAAU;AAGrC,QACE,OAAO,YAAY,YACnB,YAAY,QACZ,EAAE,YAAY,YACd,EAAE,aAAa,YACf,EAAE,gBAAgB,YAClB,EAAE,aAAa,YACf,EAAE,SAAS,YACX,EAAE,SAAS,UACX;AACA,aAAO;AAAA,IACT;AAEA,WAAO;AAAA,EACT,QAAQ;AACN,WAAO;AAAA,EACT;AACF;AAlCgB;AAiDT,SAAS,oBAAoB,KAA4B;AAC9D,MAAI,CAAC,OAAO,OAAO,QAAQ,UAAU;AACnC,WAAO;AAAA,EACT;AAEA,MAAI;AACF,UAAM,SAAS,IAAI,IAAI,GAAG;AAC1B,WAAO,OAAO,aAAa,IAAI,OAAO;AAAA,EACxC,QAAQ;AACN,WAAO;AAAA,EACT;AACF;AAXgB;AA8BT,SAAS,iBAAiB,KAAwC;AACvE,MAAI,CAAC,OAAO,OAAO,QAAQ,UAAU;AACnC,WAAO;AAAA,EACT;AAEA,MAAI;AAEF,UAAM,QAAQ,oBAAoB,GAAG;AACrC,QAAI,CAAC,OAAO;AACV,aAAO;AAAA,IACT;AAGA,UAAM,UAAU,eAAe,KAAK;AACpC,QAAI,CAAC,SAAS;AACZ,aAAO;AAAA,IACT;AAGA,UAAM,SAAS,IAAI,IAAI,GAAG;AAC1B,WAAO,aAAa,OAAO,OAAO;AAClC,UAAM,QAAQ,OAAO,SAAS;AAE9B,WAAO;AAAA,MACL;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,IACF;AAAA,EACF,QAAQ;AACN,WAAO;AAAA,EACT;AACF;AAhCgB;;;AFpQT,IAAM,WAAN,MAAM,UAAS;AAAA,EAxEtB,OAwEsB;AAAA;AAAA;AAAA,EACZ;AAAA,EACA,KAAuB;AAAA,EACvB,mBAAmB;AAAA,EACnB,oBAAoB;AAAA,EACpB;AAAA;AAAA,EAGA;AAAA;AAAA,EAGA,YAA2B;AAAA;AAAA,EAG3B,oBAA2C;AAAA;AAAA,EAG3C,kBAAkB;AAAA,EAClB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASR,YACE,aACA,YACA,gBACA;AACA,SAAK,cAAc;AACnB,SAAK,iBAAiB,kBAAkB;AAGxC,SAAK,aAAa;AAAA,EACpB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAyBA,aAAa,OAAO,QAAiD;AAEnE,UAAM,EAAE,2BAAAC,2BAA0B,IAAI,MAAM;AAG5C,UAAM,iBAAiB;AAAA,MACrB,YAAY,OAAO;AAAA,MACnB,gBAAgB,OAAO;AAAA,IACzB;AAGA,UAAM,qBAAqB,IAAIA,2BAA0B,cAAc;AAEvE,QAAI;AAEF,YAAM,mBAAmB,WAAW;AAAA,IACtC,SAAS,OAAO;AAEd,YAAM,mBAAmB,QAAQ;AACjC,YAAM;AAAA,IACR;AAGA,WAAO,IAAI;AAAA,MACT,OAAO;AAAA,MACP;AAAA,MACA,OAAO;AAAA,IACT;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,SAAiB;AACf,WAAO,KAAK;AAAA,EACd;AAAA;AAAA;AAAA;AAAA,EAKA,WAAmB;AACjB,QAAI;AACF,YAAM,WAAW,KAAK,WAAW,YAAY;AAE7C,aAAO,SAAS,IAAI,CAAC,cAAc;AAAA,QACjC,MAAM,SAAS;AAAA,QACf,aAAa,SAAS;AAAA,QACtB,aAAa,qBAAuB,SAAS,WAAW;AAAA,MAC1D,EAAE;AAAA,IACJ,SAAS,OAAO;AACd,cAAQ;AAAA,QACN,qDAAa,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK,CAAC;AAAA,MACrE;AACA,aAAO,CAAC;AAAA,IACV;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,MAAa,UAAyB;AAEpC,UAAM,KAAK,WAAW,WAAW;AAGjC,QAAI,KAAK,mDAAgD;AACvD,YAAM,IAAI,MAAM,4FAAiB;AAAA,IACnC;AAGA,SAAK,kBAAkB;AAEvB,WAAO,KAAK,kBAAkB;AAAA,EAChC;AAAA;AAAA;AAAA;AAAA,EAKA,MAAc,oBAAmC;AAC/C,SAAK;AACL,YAAQ,MAAM,2DAAc,cAAc,KAAK,WAAW,CAAC,EAAE;AAE7D,WAAO,IAAI,QAAQ,CAAC,SAAS,WAAW;AAEtC,WAAK,oBAAoB,WAAW,MAAM;AACxC,cAAM,QAAQ,IAAI,MAAM,oCAAgB;AACxC,aAAK,sBAAsB,KAAK;AAChC,eAAO,KAAK;AAAA,MACd,GAAG,GAAK;AAER,WAAK,KAAK,IAAI,UAAU,KAAK,WAAW;AAExC,WAAK,GAAG,GAAG,QAAQ,MAAM;AACvB,aAAK,wBAAwB;AAC7B,gBAAQ;AAAA,MACV,CAAC;AAED,WAAK,GAAG,GAAG,WAAW,CAAC,SAAS;AAC9B,YAAI;AACF,gBAAM,UAAsB,KAAK,MAAM,KAAK,SAAS,CAAC;AACtD,eAAK,cAAc,OAAO;AAAA,QAC5B,SAAS,OAAO;AACd,kBAAQ,MAAM,6CAAe,KAAK;AAAA,QACpC;AAAA,MACF,CAAC;AAED,WAAK,GAAG,GAAG,SAAS,CAAC,MAAM,WAAW;AACpC,aAAK,sBAAsB,MAAM,OAAO,SAAS,CAAC;AAAA,MACpD,CAAC;AAED,WAAK,GAAG,GAAG,SAAS,CAAC,UAAU;AAC7B,aAAK,sBAAsB,KAAK;AAChC,eAAO,KAAK;AAAA,MACd,CAAC;AAAA,IACH,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKQ,0BAAgC;AAEtC,QAAI,KAAK,mBAAmB;AAC1B,mBAAa,KAAK,iBAAiB;AACnC,WAAK,oBAAoB;AAAA,IAC3B;AAEA,SAAK,mBAAmB;AACxB,SAAK;AAEL,YAAQ,MAAM,8CAAqB;AAAA,EACrC;AAAA;AAAA;AAAA;AAAA,EAKQ,sBAAsB,OAAoB;AAEhD,SAAK,YAAY,MAAM;AAGvB,QAAI,KAAK,mBAAmB;AAC1B,mBAAa,KAAK,iBAAiB;AACnC,WAAK,oBAAoB;AAAA,IAC3B;AAEA,YAAQ,MAAM,+BAAqB,MAAM,OAAO;AAGhD,SAAK,kBAAkB;AAAA,EACzB;AAAA;AAAA;AAAA;AAAA,EAKQ,sBAAsB,MAAc,QAAsB;AAChE,SAAK,mBAAmB;AACxB,SAAK,oBAAoB;AACzB,SAAK;AACL,YAAQ,KAAK,6DAAgB,IAAI,mBAAS,MAAM,GAAG;AAAA,EACrD;AAAA;AAAA;AAAA;AAAA,EAKQ,oBAA0B;AAEhC,QAAI,KAAK,IAAI;AACX,WAAK,GAAG,mBAAmB;AAE3B,UAAI;AACF,YAAI,KAAK,GAAG,eAAe,UAAU,MAAM;AACzC,eAAK,GAAG,MAAM,KAAM,wBAAwB;AAAA,QAC9C,WAAW,KAAK,GAAG,eAAe,UAAU,YAAY;AACtD,eAAK,GAAG,UAAU;AAAA,QACpB;AAAA,MACF,SAAS,OAAO;AACd,gBAAQ,MAAM,uFAA2B,KAAK;AAAA,MAChD;AAEA,WAAK,KAAK;AAAA,IACZ;AAGA,QAAI,KAAK,mBAAmB;AAC1B,mBAAa,KAAK,iBAAiB;AACnC,WAAK,oBAAoB;AAAA,IAC3B;AAGA,SAAK,mBAAmB;AACxB,SAAK,oBAAoB;AAGzB,SAAK;AAAA,EACP;AAAA;AAAA;AAAA;AAAA,EAKQ,cAAc,SAA2B;AAG/C,QAAI,CAAC,QAAQ,QAAQ;AACnB,cAAQ,MAAM,kFAAsB;AACpC;AAAA,IACF;AAEA,YAAQ,QAAQ,QAAQ;AAAA,MACtB,KAAK;AAAA,MACL,KAAK;AACH,aAAK,aAAa,QAAQ,IAAI;AAAA,UAC5B,iBAAiB;AAAA,UACjB,cAAc;AAAA,YACZ,OAAO,EAAE,aAAa,KAAK;AAAA,YAC3B,SAAS,CAAC;AAAA,UACZ;AAAA,UACA,YAAY;AAAA,YACV,MAAM;AAAA,YACN,SAAS;AAAA,UACX;AAAA,QACF,CAAC;AACD,aAAK,oBAAoB;AACzB,gBAAQ,MAAM,sDAAc;AAC5B;AAAA,MAEF,KAAK,cAAc;AACjB,cAAM,YAAY,KAAK,SAAS;AAChC,aAAK,aAAa,QAAQ,IAAI,EAAE,OAAO,UAAU,CAAC;AAClD,gBAAQ,MAAM,mDAAgB,UAAU,MAAM,qBAAM;AACpD;AAAA,MACF;AAAA,MAEA,KAAK,cAAc;AACjB,aAAK,eAAe,OAAO,EAAE,MAAM,CAAC,UAAU;AAC5C,kBAAQ,MAAM,yFAAmB,KAAK;AAAA,QACxC,CAAC;AACD;AAAA,MACF;AAAA,MAEA,KAAK;AACH,aAAK,aAAa,QAAQ,IAAI,CAAC,CAAC;AAEhC;AAAA,MAEF;AACE,gBAAQ,KAAK,wCAAe,QAAQ,MAAM,EAAE;AAAA,IAChD;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKQ,aAAa,IAAqB,QAAuB;AAK/D,QAAI,KAAK,oBAAoB,KAAK,IAAI,eAAe,UAAU,MAAM;AACnE,YAAM,WAA+B;AAAA,QACnC,SAAS;AAAA,QACT;AAAA,QACA;AAAA,MACF;AAEA,UAAI;AACF,aAAK,GAAG,KAAK,KAAK,UAAU,QAAQ,CAAC;AACrC,gBAAQ,MAAM,kCAAS;AAAA,UACrB;AAAA,UACA,cAAc,KAAK,UAAU,QAAQ,EAAE;AAAA,QACzC,CAAC;AAAA,MACH,SAAS,OAAO;AACd,gBAAQ,MAAM,wCAAU;AAAA,UACtB;AAAA,UACA;AAAA,QACF,CAAC;AAAA,MACH;AAAA,IACF,OAAO;AACL,cAAQ,MAAM,wCAAU;AAAA,QACtB;AAAA,QACA,aAAa,KAAK;AAAA,QAClB,cAAc,KAAK,IAAI;AAAA,MACzB,CAAC;AAAA,IACH;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKO,YAAsC;AAC3C,UAAM,iBAAiB,KAAK,WAAW,YAAY,EAAE;AAErD,WAAO;AAAA,MACL,WAAW,KAAK;AAAA,MAChB,aAAa,KAAK;AAAA,MAClB,KAAK,KAAK;AAAA,MACV;AAAA,MACA,iBAAiB,KAAK;AAAA,MACtB,WAAW,KAAK;AAAA,IAClB;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKO,cAAuB;AAC5B,WAAO,KAAK;AAAA,EACd;AAAA;AAAA;AAAA;AAAA,EAKA,MAAa,aAA4B;AACvC,YAAQ,KAAK,kDAAU;AAGvB,UAAM,KAAK,WAAW,QAAQ;AAG9B,SAAK,kBAAkB;AAAA,EACzB;AAAA;AAAA;AAAA;AAAA,EAKA,MAAa,YAA2B;AACtC,YAAQ,KAAK,+CAAY,cAAc,KAAK,WAAW,CAAC,EAAE;AAG1D,SAAK,WAAW;AAGhB,UAAM,IAAI,QAAQ,CAAC,YAAY,WAAW,SAAS,KAAK,cAAc,CAAC;AAGvE,UAAM,KAAK,QAAQ;AAAA,EACrB;AAAA;AAAA;AAAA;AAAA,EAKA,MAAc,eAAe,SAAoC;AAC/D,QAAI,QAAQ,OAAO,UAAa,QAAQ,OAAO,MAAM;AACnD,YAAM,IAAI;AAAA;AAAA,QAER;AAAA,MACF;AAAA,IACF;AAEA,UAAM,YAAY,QAAQ;AAC1B,UAAM,YAAY,KAAK,IAAI;AAE3B,QAAI;AACF,YAAM,SAAS,uBAAuB,QAAQ,MAAM;AAEpD,cAAQ,KAAK,oDAAY;AAAA,QACvB;AAAA,QACA,UAAU,OAAO;AAAA,QACjB,cAAc,CAAC,CAAC,OAAO;AAAA,MACzB,CAAC;AAED,YAAM,SAAS,MAAM,KAAK;AAAA,QACxB,OAAO;AAAA,QACP,OAAO,aAAa,CAAC;AAAA,QACrB,KAAK;AAAA,MACP;AAEA,WAAK,aAAa,WAAW;AAAA,QAC3B,SAAS,OAAO,WAAW;AAAA,UACzB,EAAE,MAAM,QAAQ,MAAM,KAAK,UAAU,MAAM,EAAE;AAAA,QAC/C;AAAA,QACA,SAAS,OAAO,WAAW;AAAA,MAC7B,CAAC;AAED,cAAQ,KAAK,wCAAU;AAAA,QACrB;AAAA,QACA,UAAU,OAAO;AAAA,QACjB,UAAU,GAAG,KAAK,IAAI,IAAI,SAAS;AAAA,MACrC,CAAC;AAAA,IACH,SAAS,OAAO;AACd,WAAK,oBAAoB,OAAO,WAAW,KAAK,IAAI,IAAI,SAAS;AAAA,IACnE;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,MAAc,uBACZ,UACA,YACA,YAAY,KACa;AACzB,WAAO,IAAI,QAAQ,CAAC,SAAS,WAAW;AACtC,YAAM,YAAY,WAAW,MAAM;AACjC;AAAA,UACE,IAAI;AAAA;AAAA,YAEF,yCAAW,SAAS,QAAQ,QAAQ;AAAA,UACtC;AAAA,QACF;AAAA,MACF,GAAG,SAAS;AAEZ,WAAK,WACF,SAAS,UAAU,UAAU,EAC7B,KAAK,CAAC,WAA2B;AAChC,qBAAa,SAAS;AACtB,gBAAQ,MAAM;AAAA,MAChB,CAAC,EACA,MAAM,CAAC,UAAmB;AACzB,qBAAa,SAAS;AAEtB,cAAM,eACJ,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK;AAEvD,YAAI,aAAa,SAAS,gCAAO,GAAG;AAClC;AAAA,YACE,IAAI;AAAA;AAAA,cAEF,mCAAU,QAAQ;AAAA,YACpB;AAAA,UACF;AAAA,QACF,OAAO;AACL;AAAA,YACE,IAAI;AAAA;AAAA,cAEF,yCAAW,YAAY;AAAA,YACzB;AAAA,UACF;AAAA,QACF;AAAA,MACF,CAAC;AAAA,IACL,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKQ,oBACN,OACA,WACA,UACM;AACN,QAAI;AAMJ,QAAI,iBAAiB,eAAoB;AACvC,sBAAgB;AAAA,QACd,MAAM,MAAM;AAAA,QACZ,SAAS,MAAM;AAAA,QACf,MAAM,MAAM;AAAA,MACd;AAAA,IACF,OAAO;AACL,YAAM,eAAe,iBAAiB,QAAQ,MAAM,UAAU;AAC9D,sBAAgB;AAAA,QACd;AAAA,QACA,SAAS;AAAA,QACT,MAAM,EAAE,eAAe,OAAO,KAAK,KAAK,OAAO;AAAA,MACjD;AAAA,IACF;AAEA,SAAK,kBAAkB,WAAW,aAAa;AAE/C,YAAQ,MAAM,wCAAU;AAAA,MACtB;AAAA,MACA,UAAU,GAAG,QAAQ;AAAA,MACrB,OAAO;AAAA,IACT,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKQ,kBACN,IACA,OACM;AACN,QAAI,KAAK,oBAAoB,KAAK,IAAI,eAAe,UAAU,MAAM;AACnE,YAAM,WAAW;AAAA,QACf,SAAS;AAAA,QACT;AAAA,QACA;AAAA,MACF;AACA,UAAI;AACF,aAAK,GAAG,KAAK,KAAK,UAAU,QAAQ,CAAC;AACrC,gBAAQ,MAAM,+CAAY,QAAQ;AAAA,MACpC,SAAS,WAAW;AAElB,gBAAQ,MAAM,qDAAa;AAAA,UACzB;AAAA,UACA,eAAe;AAAA,UACf,WAAW,qBAAqB,QAC5B,EAAE,SAAS,UAAU,SAAS,OAAO,UAAU,MAAM,IACrD;AAAA,QACN,CAAC;AAAA,MACH;AAAA,IACF,OAAO;AACL,cAAQ,MAAM,oDAAY;AAAA,QACxB;AAAA,QACA,aAAa,KAAK;AAAA,QAClB,cAAc,KAAK,IAAI;AAAA,MACzB,CAAC;AAAA,IACH;AAAA,EACF;AACF;;;AG1mBA,SAAS,oBAAoB;;;ACKtB,IAAM,mBAAN,MAAqD;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQ1D,YAAoB,kBAAsC;AAAtC;AAClB,QAAI,CAAC,kBAAkB;AACrB,YAAM,IAAI,MAAM,kDAAoB;AAAA,IACtC;AAAA,EACF;AAAA,EAtCF,OA0B4D;AAAA;AAAA;AAAA,EAClD,gBAAgB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAkBxB,MAAM,aAA4B;AAChC,QAAI,KAAK,eAAe;AACtB;AAAA,IACF;AAGA,SAAK,gBAAgB;AAAA,EACvB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,cAAkC;AAEhC,WAAO,KAAK,iBAAiB,YAAY;AAAA,EAC3C;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAUA,MAAM,SACJ,UACA,YACyB;AAGzB,WAAO,KAAK,iBAAiB,SAAS,UAAU,UAAU;AAAA,EAC5D;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAM,UAAyB;AAC7B,SAAK,gBAAgB;AAAA,EAEvB;AACF;;;ADpCO,IAAM,kBAAN,cAA8B,aAAa;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAYhD,YAAoB,QAAgC;AAClD,UAAM;AADY;AAElB,YAAQ,MAAM,kDAAyB;AAAA,EACzC;AAAA,EArEF,OAsDkD;AAAA;AAAA;AAAA,EACxC,YAAmC,oBAAI,IAAI;AAAA,EAC3C,mBAAwD,oBAAI,IAAI;AAAA,EAChE,aAAwC;AAAA,EACxC,mBAA4C;AAAA,EAC5C,oBAAyD,CAAC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAmBlE,cAAc,YAAsC;AAClD,QAAI,KAAK,kBAAkB;AACzB,YAAM,IAAI,MAAM,+EAAwB;AAAA,IAC1C;AAEA,SAAK,aAAa;AAClB,SAAK,mBAAmB,IAAI,iBAAiB,UAAU;AAGvD,QAAI,QAAQ,cAAc,OAAO,WAAW,OAAO,YAAY;AAC7D,YAAM,mBAAmB,2BAAI,SAAoB;AAC/C,cAAM,OAAO,KAAK,CAAC;AACnB,gBAAQ;AAAA,UACN,yDAAgC,KAAK,UAAU,yBAAU,KAAK,OAAO,UAAU,CAAC;AAAA,QAClF;AAAA,MACF,GALyB;AAMzB,YAAM,eAAe,2BAAI,SAAoB;AAC3C,cAAM,OAAO,KAAK,CAAC;AACnB,gBAAQ;AAAA,UACN,+DAAiC,KAAK,UAAU;AAAA,UAChD,KAAK;AAAA,QACP;AAAA,MACF,GANqB;AAQrB,iBAAW,GAAG,aAAa,gBAAgB;AAC3C,iBAAW,GAAG,SAAS,YAAY;AAGnC,WAAK,kBAAkB,KAAK,kBAAkB,YAAY;AAAA,IAC5D;AAEA,YAAQ,KAAK,iDAAkC;AAAA,EACjD;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,YAAY,UAAmC;AAE7C,QAAI,OAAO,aAAa,UAAU;AAChC,UAAI,CAAC,KAAK,kBAAkB;AAC1B,cAAM,IAAI;AAAA,UACR;AAAA,QACF;AAAA,MACF;AAEA,YAAM,mBAAmB,IAAI;AAAA,QAC3B;AAAA,QACA,KAAK;AAAA,QACL,KAAK,QAAQ;AAAA,MACf;AAEA,WAAK,oBAAoB,gBAAgB;AACzC;AAAA,IACF;AAKA,QAAI,KAAK,kBAAkB;AACzB,YAAM,IAAI;AAAA,QACR;AAAA,MACF;AAAA,IACF;AAEA,SAAK,oBAAoB,QAAQ;AAAA,EACnC;AAAA;AAAA;AAAA;AAAA,EAKQ,oBAAoB,UAA0B;AACpD,UAAM,MAAM,SAAS,OAAO;AAE5B,QAAI,KAAK,UAAU,IAAI,GAAG,GAAG;AAC3B,cAAQ;AAAA,QACN,wCAAyB,cAAc,GAAG,CAAC;AAAA,MAC7C;AACA;AAAA,IACF;AAEA,YAAQ,MAAM,qDAA4B,cAAc,GAAG,CAAC,EAAE;AAE9D,SAAK,UAAU,IAAI,KAAK,QAAQ;AAChC,SAAK,iBAAiB,IAAI,KAAK;AAAA,MAC7B,UAAU;AAAA,MACV,WAAW;AAAA,MACX,aAAa;AAAA,IACf,CAAC;AAGD,SAAK,KAAK,iBAAiB,EAAE,UAAU,IAAI,CAAC;AAAA,EAC9C;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAM,eAAe,UAAmC;AACtD,UAAM,MAAM,SAAS,OAAO;AAE5B,QAAI,CAAC,KAAK,UAAU,IAAI,GAAG,GAAG;AAC5B,cAAQ;AAAA,QACN,wCAAyB,cAAc,GAAG,CAAC;AAAA,MAC7C;AACA;AAAA,IACF;AAEA,YAAQ,MAAM,qDAA4B,cAAc,GAAG,CAAC,EAAE;AAG9D,UAAM,SAAS,WAAW;AAG1B,SAAK,UAAU,OAAO,GAAG;AACzB,SAAK,iBAAiB,OAAO,GAAG;AAGhC,SAAK,KAAK,mBAAmB,EAAE,UAAU,IAAI,CAAC;AAAA,EAChD;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASA,MAAM,QAAQ,UAAkC;AAE9C,QAAI,UAAU;AACZ,YAAM,mBAAmB,KAAK,UAAU,IAAI,QAAQ;AACpD,UAAI,CAAC,kBAAkB;AACrB,cAAM,IAAI,MAAM,yCAAW,cAAc,QAAQ,CAAC,EAAE;AAAA,MACtD;AAEA,YAAM,SAAS,KAAK,iBAAiB,IAAI,QAAQ;AACjD,UAAI,QAAQ,WAAW;AACrB,gBAAQ;AAAA,UACN,6EAAgC,cAAc,QAAQ,CAAC;AAAA,QACzD;AACA;AAAA,MACF;AAEA,YAAM,KAAK,sBAAsB,UAAU,gBAAgB;AAC3D;AAAA,IACF;AAGA,YAAQ;AAAA,MACN,mFAAiC,KAAK,UAAU,IAAI;AAAA,IACtD;AAEA,UAAM,WAA4B,CAAC;AAEnC,eAAW,CAAC,KAAKC,SAAQ,KAAK,KAAK,WAAW;AAC5C,YAAM,SAAS,KAAK,iBAAiB,IAAI,GAAG;AAG5C,UAAI,QAAQ,WAAW;AACrB,gBAAQ;AAAA,UACN,6EAAgC,cAAc,GAAG,CAAC;AAAA,QACpD;AACA;AAAA,MACF;AAEA,eAAS;AAAA,QACP,KAAK,sBAAsB,KAAKA,SAAQ,EAAE,MAAM,CAAC,UAAU;AACzD,kBAAQ;AAAA,YACN,+CAA2B,cAAc,GAAG,CAAC;AAAA,YAC7C;AAAA,UACF;AAEA,gBAAMC,UAAS,KAAK,iBAAiB,IAAI,GAAG;AAC5C,cAAIA,SAAQ;AACV,YAAAA,QAAO,YAAY;AACnB,YAAAA,QAAO,cAAc;AACrB,YAAAA,QAAO,YACL,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK;AAAA,UACzD;AAAA,QACF,CAAC;AAAA,MACH;AAAA,IACF;AAEA,UAAM,QAAQ,WAAW,QAAQ;AAGjC,UAAM,iBAAiB,MAAM,KAAK,KAAK,iBAAiB,OAAO,CAAC,EAAE;AAAA,MAChE,CAAC,MAAM,EAAE;AAAA,IACX,EAAE;AAEF,YAAQ;AAAA,MACN,4DAA8B,cAAc,IAAI,KAAK,UAAU,IAAI;AAAA,IACrE;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASA,MAAM,WAAW,UAAkC;AAEjD,QAAI,UAAU;AACZ,YAAM,mBAAmB,KAAK,UAAU,IAAI,QAAQ;AACpD,UAAI,CAAC,kBAAkB;AACrB,cAAM,IAAI,MAAM,yCAAW,cAAc,QAAQ,CAAC,EAAE;AAAA,MACtD;AAEA,uBAAiB,WAAW;AAE5B,YAAM,SAAS,KAAK,iBAAiB,IAAI,QAAQ;AACjD,UAAI,QAAQ;AACV,eAAO,YAAY;AACnB,eAAO,cAAc;AAAA,MACvB;AAEA,cAAQ;AAAA,QACN,2DAA6B,cAAc,QAAQ,CAAC;AAAA,MACtD;AACA;AAAA,IACF;AAGA,YAAQ,MAAM,oEAA4B;AAE1C,UAAM,WAA4B,CAAC;AAEnC,eAAWD,aAAY,KAAK,UAAU,OAAO,GAAG;AAC9C,eAAS;AAAA,QACP,QAAQ,QAAQ,EAAE,KAAK,MAAM;AAC3B,UAAAA,UAAS,WAAW;AAAA,QACtB,CAAC;AAAA,MACH;AAAA,IACF;AAEA,UAAM,QAAQ,WAAW,QAAQ;AAGjC,eAAW,UAAU,KAAK,iBAAiB,OAAO,GAAG;AACnD,aAAO,YAAY;AACnB,aAAO,cAAc;AAAA,IACvB;AAEA,YAAQ,MAAM,gFAA8B;AAAA,EAC9C;AAAA;AAAA;AAAA;AAAA,EAKA,eAAyB;AACvB,WAAO,MAAM,KAAK,KAAK,UAAU,KAAK,CAAC;AAAA,EACzC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,YAAY,KAAmC;AAC7C,WAAO,KAAK,UAAU,IAAI,GAAG;AAAA,EAC/B;AAAA;AAAA;AAAA;AAAA,EAKA,sBAAgD;AAC9C,WAAO,MAAM,KAAK,KAAK,iBAAiB,OAAO,CAAC;AAAA,EAClD;AAAA;AAAA;AAAA;AAAA,EAKA,iBAA0B;AACxB,eAAW,UAAU,KAAK,iBAAiB,OAAO,GAAG;AACnD,UAAI,OAAO,WAAW;AACpB,eAAO;AAAA,MACT;AAAA,IACF;AACA,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,oBAAoB,KAAsB;AACxC,UAAM,SAAS,KAAK,iBAAiB,IAAI,GAAG;AAC5C,WAAO,QAAQ,aAAa;AAAA,EAC9B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,kBAAkB,KAAiD;AACjE,WAAO,KAAK,iBAAiB,IAAI,GAAG;AAAA,EACtC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASA,MAAM,UAAU,UAAmB,QAAQ,KAAqB;AAC9D,YAAQ,KAAK,4CAAwB;AAGrC,UAAM,KAAK,WAAW,QAAQ;AAG9B,YAAQ,MAAM,kCAAwB,KAAK,uBAAQ;AACnD,UAAM,IAAI,QAAQ,CAAC,YAAY,WAAW,SAAS,KAAK,CAAC;AAGzD,UAAM,KAAK,QAAQ,QAAQ;AAE3B,YAAQ,KAAK,4CAAwB;AAAA,EACvC;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,eAA8B;AAClC,YAAQ,KAAK,0EAA6B;AAE1C,UAAM,WAA4B,CAAC;AAEnC,eAAW,CAAC,KAAK,QAAQ,KAAK,KAAK,WAAW;AAC5C,eAAS;AAAA,QACP,KAAK,wBAAwB,KAAK,QAAQ,EAAE,MAAM,CAAC,UAAU;AAC3D,kBAAQ;AAAA,YACN,+CAA2B,cAAc,GAAG,CAAC;AAAA,YAC7C;AAAA,UACF;AAAA,QACF,CAAC;AAAA,MACH;AAAA,IACF;AAEA,UAAM,QAAQ,WAAW,QAAQ;AAAA,EACnC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAM,kBAAkB,KAA4B;AAClD,UAAM,WAAW,KAAK,UAAU,IAAI,GAAG;AACvC,QAAI,CAAC,UAAU;AACb,YAAM,IAAI,MAAM,yCAAW,cAAc,GAAG,CAAC,EAAE;AAAA,IACjD;AAEA,UAAM,KAAK,wBAAwB,KAAK,QAAQ;AAAA,EAClD;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAM,iBAAgC;AACpC,YAAQ,MAAM,8DAA2B;AAEzC,UAAM,KAAK,WAAW;AAEtB,SAAK,UAAU,MAAM;AACrB,SAAK,iBAAiB,MAAM;AAM5B,YAAQ,KAAK,oEAA4B;AAAA,EAC3C;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAM,UAAyB;AAC7B,YAAQ,MAAM,wDAA0B;AAGxC,QACE,KAAK,cACL,oBAAoB,KAAK,cACzB,OAAO,KAAK,WAAW,mBAAmB,YAC1C;AACA,iBAAW,YAAY,KAAK,mBAAmB;AAC7C,aAAK,WAAW,eAAe,aAAa,QAAQ;AACpD,aAAK,WAAW,eAAe,SAAS,QAAQ;AAAA,MAClD;AAAA,IACF;AACA,SAAK,oBAAoB,CAAC;AAE1B,UAAM,KAAK,eAAe;AAE1B,YAAQ,MAAM,wDAA0B;AAAA,EAC1C;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAc,sBACZ,KACA,UACe;AACf,UAAM,SAAS,KAAK,iBAAiB,IAAI,GAAG;AAC5C,QAAI,CAAC,QAAQ;AACX,YAAM,IAAI,MAAM,+CAAY,cAAc,GAAG,CAAC,EAAE;AAAA,IAClD;AAEA,YAAQ,MAAM,+CAA2B,cAAc,GAAG,CAAC,EAAE;AAG7D,WAAO,YAAY;AACnB,WAAO,cAAc;AAGrB,UAAM,SAAS,QAAQ;AAGvB,WAAO,YAAY;AACnB,WAAO,cAAc;AACrB,WAAO,gBAAgB,oBAAI,KAAK;AAChC,WAAO,YAAY;AAEnB,YAAQ,KAAK,2DAA6B,cAAc,GAAG,CAAC,EAAE;AAAA,EAChE;AAAA;AAAA;AAAA;AAAA,EAKA,MAAc,wBACZ,KACA,UACe;AACf,UAAM,SAAS,KAAK,iBAAiB,IAAI,GAAG;AAC5C,QAAI,CAAC,QAAQ;AACX,YAAM,IAAI,MAAM,+CAAY,cAAc,GAAG,CAAC,EAAE;AAAA,IAClD;AAEA,YAAQ,MAAM,+CAA2B,cAAc,GAAG,CAAC,EAAE;AAG7D,UAAM,SAAS,UAAU;AAGzB,WAAO,YAAY;AACnB,WAAO,cAAc;AACrB,WAAO,gBAAgB,oBAAI,KAAK;AAChC,WAAO,YAAY;AAEnB,YAAQ,KAAK,2DAA6B,cAAc,GAAG,CAAC,EAAE;AAAA,EAChE;AACF;","names":["ToolCallErrorCode","ConnectionState","InternalMCPManagerAdapter","endpoint","status"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@xiaozhi-client/endpoint",
|
|
3
|
-
"version": "
|
|
3
|
+
"version": "2.0.0-beta.0",
|
|
4
4
|
"description": "小智接入点 WebSocket 连接管理库",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"main": "./dist/index.js",
|
|
@@ -21,8 +21,8 @@
|
|
|
21
21
|
],
|
|
22
22
|
"dependencies": {
|
|
23
23
|
"ws": "^8.14.2",
|
|
24
|
-
"@xiaozhi-client/mcp-core": "
|
|
25
|
-
"@xiaozhi-client/config": "
|
|
24
|
+
"@xiaozhi-client/mcp-core": "2.0.0-beta.0",
|
|
25
|
+
"@xiaozhi-client/config": "2.0.0-beta.0"
|
|
26
26
|
},
|
|
27
27
|
"peerDependencies": {
|
|
28
28
|
"@modelcontextprotocol/sdk": "^1.26.0"
|