Files
nonebot-plugin-marshoai/nonebot_plugin_marshoai/extensions/mcp_extension/client.py
2025-09-05 20:37:15 +08:00

121 lines
3.4 KiB
Python

import asyncio
from typing import Any, Optional
from nonebot import logger
from .config import get_mcp_server_config
from .server import Server, Tool
_servers: list[Server] = list()
async def initialize_servers() -> None:
"""
初始化全部 MCP 实例
"""
server_config = get_mcp_server_config()
_servers.extend(
[Server(name, srv_config) for name, srv_config in server_config.items()]
)
for server in _servers:
logger.info(f"正在初始化 MCP 服务器: {server.name}...")
try:
await server.initialize()
except Exception as e:
logger.error(f"初始化 MCP 服务器实例时出现问题: {e}")
await cleanup_servers()
raise
async def handle_mcp_tool(
tool: str, arguments: Optional[dict[str, Any]] = None
) -> Optional[str]:
"""
处理 MCP Tool 调用
"""
logger.info(f"执行 MCP 工具: {tool} (参数: {arguments})")
for server in _servers:
server_tools = await server.list_tools()
if not any(server_tool.name == tool for server_tool in server_tools):
continue
try:
result = await server.execute_tool(tool, arguments)
if isinstance(result, dict) and "progress" in result:
progress = result["progress"]
total = result["total"]
percentage = (progress / total) * 100
logger.info(
f"工具 {tool} 执行进度: {progress}/{total} ({percentage:.1f}%)"
)
return f"Tool execution result: {result}"
except Exception as e:
error_msg = f"Error executing tool: {str(e)}"
logger.error(error_msg)
return error_msg
return None # Not found.
async def cleanup_servers() -> None:
"""
清理 MCP 实例
"""
cleanup_tasks = [asyncio.create_task(server.cleanup()) for server in _servers]
if cleanup_tasks:
try:
await asyncio.gather(*cleanup_tasks, return_exceptions=True)
except Exception as e:
logger.warning(f"清理 MCP 实例时出现错误: {e}")
async def transform_json(tool: Tool) -> dict[str, Any]:
"""
将 MCP Tool 转换为 OpenAI 所需的 parameters 格式,并删除多余字段
"""
func_desc = {
"name": tool.name,
"description": tool.description,
"parameters": {},
"required": [],
}
if tool.input_schema:
parameters = {
"type": tool.input_schema.get("type", "object"),
"properties": tool.input_schema.get("properties", {}),
"required": tool.input_schema.get("required", []),
}
func_desc["parameters"] = parameters
output = {"type": "function", "function": func_desc}
return output
async def get_mcp_list() -> list[dict[str, dict]]:
"""
获得适用于 OpenAI Tool Call 输入格式的 MCP 工具列表
"""
all_tools: list[dict[str, dict]] = []
for server in _servers:
tools = await server.list_tools()
all_tools.extend([await transform_json(tool) for tool in tools])
return all_tools
async def is_mcp_tool(tool_name: str) -> bool:
"""
检查工具是否为 MCP 工具
"""
mcp_list = await get_mcp_list()
for tool in mcp_list:
if tool["function"]["name"] == tool_name:
return True
return False