protocol_tools.py 32 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851
  1. """
  2. 协议工具集合
  3. 提供基于协议实现的工具接口:
  4. - MCP Tool: 基于 fastmcp 库,用于连接和调用 MCP 服务器
  5. - A2A Tool: 基于官方 a2a 库,用于 Agent 间通信(需要安装 a2a)
  6. - ANP Tool: 基于概念实现,用于服务发现和网络管理
  7. """
  8. from typing import Dict, Any, List, Optional
  9. from ..base import Tool, ToolParameter
  10. import os
  11. # MCP服务器环境变量映射表
  12. # 用于自动检测常见MCP服务器需要的环境变量
  13. MCP_SERVER_ENV_MAP = {
  14. "server-github": ["GITHUB_PERSONAL_ACCESS_TOKEN"],
  15. "server-slack": ["SLACK_BOT_TOKEN", "SLACK_TEAM_ID"],
  16. "server-google-drive": ["GOOGLE_CLIENT_ID", "GOOGLE_CLIENT_SECRET", "GOOGLE_REFRESH_TOKEN"],
  17. "server-postgres": ["POSTGRES_CONNECTION_STRING"],
  18. "server-sqlite": [], # 不需要环境变量
  19. "server-filesystem": [], # 不需要环境变量
  20. }
  21. class MCPTool(Tool):
  22. """MCP (Model Context Protocol) 工具
  23. 连接到 MCP 服务器并调用其提供的工具、资源和提示词。
  24. 功能:
  25. - 列出服务器提供的工具
  26. - 调用服务器工具
  27. - 读取服务器资源
  28. - 获取提示词模板
  29. 使用示例:
  30. >>> from hello_agents.tools.builtin import MCPTool
  31. >>>
  32. >>> # 方式1: 使用内置演示服务器
  33. >>> tool = MCPTool() # 自动创建内置服务器
  34. >>> result = tool.run({"action": "list_tools"})
  35. >>>
  36. >>> # 方式2: 连接到外部 MCP 服务器
  37. >>> tool = MCPTool(server_command=["python", "examples/mcp_example.py"])
  38. >>> result = tool.run({"action": "list_tools"})
  39. >>>
  40. >>> # 方式3: 使用自定义 FastMCP 服务器
  41. >>> from fastmcp import FastMCP
  42. >>> server = FastMCP("MyServer")
  43. >>> tool = MCPTool(server=server)
  44. 注意:使用 fastmcp 库,已包含在依赖中
  45. """
  46. def __init__(self,
  47. name: str = "mcp",
  48. description: Optional[str] = None,
  49. server_command: Optional[List[str]] = None,
  50. server_args: Optional[List[str]] = None,
  51. server: Optional[Any] = None,
  52. auto_expand: bool = True,
  53. env: Optional[Dict[str, str]] = None,
  54. env_keys: Optional[List[str]] = None):
  55. """
  56. 初始化 MCP 工具
  57. Args:
  58. name: 工具名称(默认为"mcp",建议为不同服务器指定不同名称)
  59. description: 工具描述(可选,默认为通用描述)
  60. server_command: 服务器启动命令(如 ["python", "server.py"])
  61. server_args: 服务器参数列表
  62. server: FastMCP 服务器实例(可选,用于内存传输)
  63. auto_expand: 是否自动展开为独立工具(默认True)
  64. env: 环境变量字典(优先级最高,直接传递给MCP服务器)
  65. env_keys: 要从系统环境变量加载的key列表(优先级中等)
  66. 环境变量优先级(从高到低):
  67. 1. 直接传递的env参数
  68. 2. env_keys指定的环境变量
  69. 3. 自动检测的环境变量(根据server_command)
  70. 注意:如果所有参数都为空,将创建内置演示服务器
  71. 示例:
  72. >>> # 方式1:直接传递环境变量(优先级最高)
  73. >>> github_tool = MCPTool(
  74. ... name="github",
  75. ... server_command=["npx", "-y", "@modelcontextprotocol/server-github"],
  76. ... env={"GITHUB_PERSONAL_ACCESS_TOKEN": "ghp_xxx"}
  77. ... )
  78. >>>
  79. >>> # 方式2:从.env文件加载指定的环境变量
  80. >>> github_tool = MCPTool(
  81. ... name="github",
  82. ... server_command=["npx", "-y", "@modelcontextprotocol/server-github"],
  83. ... env_keys=["GITHUB_PERSONAL_ACCESS_TOKEN"]
  84. ... )
  85. >>>
  86. >>> # 方式3:自动检测(最简单,推荐)
  87. >>> github_tool = MCPTool(
  88. ... name="github",
  89. ... server_command=["npx", "-y", "@modelcontextprotocol/server-github"]
  90. ... # 自动从环境变量加载GITHUB_PERSONAL_ACCESS_TOKEN
  91. ... )
  92. """
  93. self.server_command = server_command
  94. self.server_args = server_args or []
  95. self.server = server
  96. self._client = None
  97. self._available_tools = []
  98. self.auto_expand = auto_expand
  99. self.prefix = f"{name}_" if auto_expand else ""
  100. # 环境变量处理(优先级:env > env_keys > 自动检测)
  101. self.env = self._prepare_env(env, env_keys, server_command)
  102. # 如果没有指定任何服务器,创建内置演示服务器
  103. if not server_command and not server:
  104. self.server = self._create_builtin_server()
  105. # 自动发现工具
  106. self._discover_tools()
  107. # 设置默认描述或自动生成
  108. if description is None:
  109. description = self._generate_description()
  110. super().__init__(
  111. name=name,
  112. description=description
  113. )
  114. def _prepare_env(self,
  115. env: Optional[Dict[str, str]],
  116. env_keys: Optional[List[str]],
  117. server_command: Optional[List[str]]) -> Dict[str, str]:
  118. """
  119. 准备环境变量
  120. 优先级:env > env_keys > 自动检测
  121. Args:
  122. env: 直接传递的环境变量字典
  123. env_keys: 要从系统环境变量加载的key列表
  124. server_command: 服务器命令(用于自动检测)
  125. Returns:
  126. 合并后的环境变量字典
  127. """
  128. result_env = {}
  129. # 1. 自动检测(优先级最低)
  130. if server_command:
  131. # 从命令中提取服务器名称
  132. server_name = None
  133. for part in server_command:
  134. if "server-" in part:
  135. # 提取类似 "@modelcontextprotocol/server-github" 中的 "server-github"
  136. server_name = part.split("/")[-1] if "/" in part else part
  137. break
  138. # 查找映射表
  139. if server_name and server_name in MCP_SERVER_ENV_MAP:
  140. auto_keys = MCP_SERVER_ENV_MAP[server_name]
  141. for key in auto_keys:
  142. value = os.getenv(key)
  143. if value:
  144. result_env[key] = value
  145. print(f"🔑 自动加载环境变量: {key}")
  146. # 2. env_keys指定的环境变量(优先级中等)
  147. if env_keys:
  148. for key in env_keys:
  149. value = os.getenv(key)
  150. if value:
  151. result_env[key] = value
  152. print(f"🔑 从env_keys加载环境变量: {key}")
  153. else:
  154. print(f"⚠️ 警告: 环境变量 {key} 未设置")
  155. # 3. 直接传递的env(优先级最高)
  156. if env:
  157. result_env.update(env)
  158. for key in env.keys():
  159. print(f"🔑 使用直接传递的环境变量: {key}")
  160. return result_env
  161. def _create_builtin_server(self):
  162. """创建内置演示服务器"""
  163. try:
  164. from fastmcp import FastMCP
  165. server = FastMCP("HelloAgents-BuiltinServer")
  166. @server.tool()
  167. def add(a: float, b: float) -> float:
  168. """加法计算器"""
  169. return a + b
  170. @server.tool()
  171. def subtract(a: float, b: float) -> float:
  172. """减法计算器"""
  173. return a - b
  174. @server.tool()
  175. def multiply(a: float, b: float) -> float:
  176. """乘法计算器"""
  177. return a * b
  178. @server.tool()
  179. def divide(a: float, b: float) -> float:
  180. """除法计算器"""
  181. if b == 0:
  182. raise ValueError("除数不能为零")
  183. return a / b
  184. @server.tool()
  185. def greet(name: str = "World") -> str:
  186. """友好问候"""
  187. return f"Hello, {name}! 欢迎使用 HelloAgents MCP 工具!"
  188. @server.tool()
  189. def get_system_info() -> dict:
  190. """获取系统信息"""
  191. import platform
  192. import sys
  193. return {
  194. "platform": platform.system(),
  195. "python_version": sys.version,
  196. "server_name": "HelloAgents-BuiltinServer",
  197. "tools_count": 6
  198. }
  199. return server
  200. except ImportError:
  201. raise ImportError(
  202. "创建内置 MCP 服务器需要 fastmcp 库。请安装: pip install fastmcp"
  203. )
  204. def _discover_tools(self):
  205. """发现MCP服务器提供的所有工具"""
  206. try:
  207. from hello_agents.protocols.mcp.client import MCPClient
  208. import asyncio
  209. async def discover():
  210. client_source = self.server if self.server else self.server_command
  211. async with MCPClient(client_source, self.server_args, env=self.env) as client:
  212. tools = await client.list_tools()
  213. return tools
  214. # 运行异步发现
  215. try:
  216. loop = asyncio.get_running_loop()
  217. # 如果已有循环,在新线程中运行
  218. import concurrent.futures
  219. def run_in_thread():
  220. new_loop = asyncio.new_event_loop()
  221. asyncio.set_event_loop(new_loop)
  222. try:
  223. return new_loop.run_until_complete(discover())
  224. finally:
  225. new_loop.close()
  226. with concurrent.futures.ThreadPoolExecutor() as executor:
  227. future = executor.submit(run_in_thread)
  228. self._available_tools = future.result()
  229. except RuntimeError:
  230. # 没有运行中的循环
  231. self._available_tools = asyncio.run(discover())
  232. except Exception as e:
  233. # 工具发现失败不影响初始化
  234. self._available_tools = []
  235. def _generate_description(self) -> str:
  236. """生成增强的工具描述"""
  237. if not self._available_tools:
  238. return "连接到 MCP 服务器,调用工具、读取资源和获取提示词。支持内置服务器和外部服务器。"
  239. if self.auto_expand:
  240. # 展开模式:简单描述
  241. return f"MCP工具服务器,包含{len(self._available_tools)}个工具。这些工具会自动展开为独立的工具供Agent使用。"
  242. else:
  243. # 非展开模式:详细描述
  244. desc_parts = [
  245. f"MCP工具服务器,提供{len(self._available_tools)}个工具:"
  246. ]
  247. # 列出所有工具
  248. for tool in self._available_tools:
  249. tool_name = tool.get('name', 'unknown')
  250. tool_desc = tool.get('description', '无描述')
  251. # 简化描述,只取第一句
  252. short_desc = tool_desc.split('.')[0] if tool_desc else '无描述'
  253. desc_parts.append(f" • {tool_name}: {short_desc}")
  254. # 添加调用格式说明
  255. desc_parts.append("\n调用格式:返回JSON格式的参数")
  256. desc_parts.append('{"action": "call_tool", "tool_name": "工具名", "arguments": {...}}')
  257. # 添加示例
  258. if self._available_tools:
  259. first_tool = self._available_tools[0]
  260. tool_name = first_tool.get('name', 'example')
  261. desc_parts.append(f'\n示例:{{"action": "call_tool", "tool_name": "{tool_name}", "arguments": {{...}}}}')
  262. return "\n".join(desc_parts)
  263. def get_expanded_tools(self) -> List['Tool']: # type: ignore
  264. """
  265. 获取展开的工具列表
  266. 将MCP服务器的每个工具包装成独立的Tool对象
  267. Returns:
  268. Tool对象列表
  269. """
  270. if not self.auto_expand:
  271. return []
  272. from .mcp_wrapper_tool import MCPWrappedTool
  273. expanded_tools = []
  274. for tool_info in self._available_tools:
  275. wrapped_tool = MCPWrappedTool(
  276. mcp_tool=self,
  277. tool_info=tool_info,
  278. prefix=self.prefix
  279. )
  280. expanded_tools.append(wrapped_tool)
  281. return expanded_tools
  282. def run(self, parameters: Dict[str, Any]) -> str:
  283. """
  284. 执行 MCP 操作
  285. Args:
  286. parameters: 包含以下参数的字典
  287. - action: 操作类型 (list_tools, call_tool, list_resources, read_resource, list_prompts, get_prompt)
  288. 如果不指定action但指定了tool_name,会自动推断为call_tool
  289. - tool_name: 工具名称(call_tool 需要)
  290. - arguments: 工具参数(call_tool 需要)
  291. - uri: 资源 URI(read_resource 需要)
  292. - prompt_name: 提示词名称(get_prompt 需要)
  293. - prompt_arguments: 提示词参数(get_prompt 可选)
  294. Returns:
  295. 操作结果
  296. """
  297. from hello_agents.protocols.mcp.client import MCPClient
  298. # 智能推断action:如果没有action但有tool_name,自动设置为call_tool
  299. action = parameters.get("action", "").lower()
  300. if not action and "tool_name" in parameters:
  301. action = "call_tool"
  302. parameters["action"] = action
  303. if not action:
  304. return "错误:必须指定 action 参数或 tool_name 参数"
  305. try:
  306. # 使用增强的异步客户端
  307. import asyncio
  308. from hello_agents.protocols.mcp.client import MCPClient
  309. async def run_mcp_operation():
  310. # 根据配置选择客户端创建方式
  311. if self.server:
  312. # 使用内置服务器(内存传输)
  313. client_source = self.server
  314. else:
  315. # 使用外部服务器命令
  316. client_source = self.server_command
  317. async with MCPClient(client_source, self.server_args, env=self.env) as client:
  318. if action == "list_tools":
  319. tools = await client.list_tools()
  320. if not tools:
  321. return "没有找到可用的工具"
  322. result = f"找到 {len(tools)} 个工具:\n"
  323. for tool in tools:
  324. result += f"- {tool['name']}: {tool['description']}\n"
  325. return result
  326. elif action == "call_tool":
  327. tool_name = parameters.get("tool_name")
  328. arguments = parameters.get("arguments", {})
  329. if not tool_name:
  330. return "错误:必须指定 tool_name 参数"
  331. result = await client.call_tool(tool_name, arguments)
  332. return f"工具 '{tool_name}' 执行结果:\n{result}"
  333. elif action == "list_resources":
  334. resources = await client.list_resources()
  335. if not resources:
  336. return "没有找到可用的资源"
  337. result = f"找到 {len(resources)} 个资源:\n"
  338. for resource in resources:
  339. result += f"- {resource['uri']}: {resource['name']}\n"
  340. return result
  341. elif action == "read_resource":
  342. uri = parameters.get("uri")
  343. if not uri:
  344. return "错误:必须指定 uri 参数"
  345. content = await client.read_resource(uri)
  346. return f"资源 '{uri}' 内容:\n{content}"
  347. elif action == "list_prompts":
  348. prompts = await client.list_prompts()
  349. if not prompts:
  350. return "没有找到可用的提示词"
  351. result = f"找到 {len(prompts)} 个提示词:\n"
  352. for prompt in prompts:
  353. result += f"- {prompt['name']}: {prompt['description']}\n"
  354. return result
  355. elif action == "get_prompt":
  356. prompt_name = parameters.get("prompt_name")
  357. prompt_arguments = parameters.get("prompt_arguments", {})
  358. if not prompt_name:
  359. return "错误:必须指定 prompt_name 参数"
  360. messages = await client.get_prompt(prompt_name, prompt_arguments)
  361. result = f"提示词 '{prompt_name}':\n"
  362. for msg in messages:
  363. result += f"[{msg['role']}] {msg['content']}\n"
  364. return result
  365. else:
  366. return f"错误:不支持的操作 '{action}'"
  367. # 运行异步操作
  368. try:
  369. # 检查是否已有运行中的事件循环
  370. try:
  371. loop = asyncio.get_running_loop()
  372. # 如果有运行中的循环,在新线程中运行新的事件循环
  373. import concurrent.futures
  374. import threading
  375. def run_in_thread():
  376. # 在新线程中创建新的事件循环
  377. new_loop = asyncio.new_event_loop()
  378. asyncio.set_event_loop(new_loop)
  379. try:
  380. return new_loop.run_until_complete(run_mcp_operation())
  381. finally:
  382. new_loop.close()
  383. with concurrent.futures.ThreadPoolExecutor() as executor:
  384. future = executor.submit(run_in_thread)
  385. return future.result()
  386. except RuntimeError:
  387. # 没有运行中的循环,直接运行
  388. return asyncio.run(run_mcp_operation())
  389. except Exception as e:
  390. return f"异步操作失败: {str(e)}"
  391. except Exception as e:
  392. return f"MCP 操作失败: {str(e)}"
  393. def get_parameters(self) -> List[ToolParameter]:
  394. """获取工具参数定义"""
  395. return [
  396. ToolParameter(
  397. name="action",
  398. type="string",
  399. description="操作类型: list_tools, call_tool, list_resources, read_resource, list_prompts, get_prompt",
  400. required=True
  401. ),
  402. ToolParameter(
  403. name="tool_name",
  404. type="string",
  405. description="工具名称(call_tool 操作需要)",
  406. required=False
  407. ),
  408. ToolParameter(
  409. name="arguments",
  410. type="object",
  411. description="工具参数(call_tool 操作需要)",
  412. required=False
  413. ),
  414. ToolParameter(
  415. name="uri",
  416. type="string",
  417. description="资源 URI(read_resource 操作需要)",
  418. required=False
  419. ),
  420. ToolParameter(
  421. name="prompt_name",
  422. type="string",
  423. description="提示词名称(get_prompt 操作需要)",
  424. required=False
  425. ),
  426. ToolParameter(
  427. name="prompt_arguments",
  428. type="object",
  429. description="提示词参数(get_prompt 操作可选)",
  430. required=False
  431. )
  432. ]
  433. class A2ATool(Tool):
  434. """A2A (Agent-to-Agent Protocol) 工具
  435. 连接到 A2A Agent 并进行通信。
  436. 功能:
  437. - 向 Agent 提问
  438. - 获取 Agent 信息
  439. - 发送自定义消息
  440. 使用示例:
  441. >>> from hello_agents.tools.builtin import A2ATool
  442. >>> # 连接到 A2A Agent(使用默认名称)
  443. >>> tool = A2ATool(agent_url="http://localhost:5000")
  444. >>> # 连接到 A2A Agent(自定义名称和描述)
  445. >>> tool = A2ATool(
  446. ... agent_url="http://localhost:5000",
  447. ... name="tech_expert",
  448. ... description="技术专家,回答技术相关问题"
  449. ... )
  450. >>> # 提问
  451. >>> result = tool.run({"action": "ask", "question": "计算 2+2"})
  452. >>> # 获取信息
  453. >>> result = tool.run({"action": "get_info"})
  454. 注意:需要安装官方 a2a-sdk 库: pip install a2a-sdk
  455. 详见文档: docs/chapter10/A2A_GUIDE.md
  456. 官方仓库: https://github.com/a2aproject/a2a-python
  457. """
  458. def __init__(self, agent_url: str, name: str = "a2a", description: str = None):
  459. """
  460. 初始化 A2A 工具
  461. Args:
  462. agent_url: Agent URL
  463. name: 工具名称(可选,默认为 "a2a")
  464. description: 工具描述(可选)
  465. """
  466. if description is None:
  467. description = "连接到 A2A Agent,支持提问和获取信息。需要安装官方 a2a-sdk 库。"
  468. super().__init__(
  469. name=name,
  470. description=description
  471. )
  472. self.agent_url = agent_url
  473. def run(self, parameters: Dict[str, Any]) -> str:
  474. """
  475. 执行 A2A 操作
  476. Args:
  477. parameters: 包含以下参数的字典
  478. - action: 操作类型 (ask, get_info)
  479. - question: 问题文本(ask 需要)
  480. Returns:
  481. 操作结果
  482. """
  483. try:
  484. from hello_agents.protocols.a2a.implementation import A2AClient, A2A_AVAILABLE
  485. if not A2A_AVAILABLE:
  486. return ("错误:需要安装 a2a-sdk 库\n"
  487. "安装命令: pip install a2a-sdk\n"
  488. "详见文档: docs/chapter10/A2A_GUIDE.md\n"
  489. "官方仓库: https://github.com/a2aproject/a2a-python")
  490. except ImportError:
  491. return ("错误:无法导入 A2A 模块\n"
  492. "安装命令: pip install a2a-sdk\n"
  493. "详见文档: docs/chapter10/A2A_GUIDE.md\n"
  494. "官方仓库: https://github.com/a2aproject/a2a-python")
  495. action = parameters.get("action", "").lower()
  496. if not action:
  497. return "错误:必须指定 action 参数"
  498. try:
  499. client = A2AClient(self.agent_url)
  500. if action == "ask":
  501. question = parameters.get("question")
  502. if not question:
  503. return "错误:必须指定 question 参数"
  504. response = client.ask(question)
  505. return f"Agent 回答:\n{response}"
  506. elif action == "get_info":
  507. info = client.get_info()
  508. result = "Agent 信息:\n"
  509. for key, value in info.items():
  510. result += f"- {key}: {value}\n"
  511. return result
  512. else:
  513. return f"错误:不支持的操作 '{action}'"
  514. except Exception as e:
  515. return f"A2A 操作失败: {str(e)}"
  516. def get_parameters(self) -> List[ToolParameter]:
  517. """获取工具参数定义"""
  518. return [
  519. ToolParameter(
  520. name="action",
  521. type="string",
  522. description="操作类型: ask(提问), get_info(获取信息)",
  523. required=True
  524. ),
  525. ToolParameter(
  526. name="question",
  527. type="string",
  528. description="问题文本(ask 操作需要)",
  529. required=False
  530. )
  531. ]
  532. class ANPTool(Tool):
  533. """ANP (Agent Network Protocol) 工具
  534. 提供智能体网络管理功能,包括服务发现、节点管理和消息路由。
  535. 这是一个概念性实现,用于演示 Agent 网络管理的核心理念。
  536. 功能:
  537. - 注册和发现服务
  538. - 添加和管理网络节点
  539. - 消息路由
  540. - 网络统计
  541. 使用示例:
  542. >>> from hello_agents.tools.builtin import ANPTool
  543. >>> tool = ANPTool()
  544. >>> # 注册服务
  545. >>> result = tool.run({
  546. ... "action": "register_service",
  547. ... "service_id": "calc-1",
  548. ... "service_type": "calculator",
  549. ... "endpoint": "http://localhost:5001"
  550. ... })
  551. >>> # 发现服务
  552. >>> result = tool.run({
  553. ... "action": "discover_services",
  554. ... "service_type": "calculator"
  555. ... })
  556. >>> # 添加节点
  557. >>> result = tool.run({
  558. ... "action": "add_node",
  559. ... "node_id": "agent-1",
  560. ... "endpoint": "http://localhost:5001"
  561. ... })
  562. 注意:这是概念性实现,不需要额外依赖
  563. 详见文档: docs/chapter10/ANP_CONCEPTS.md
  564. """
  565. def __init__(self, name: str = "anp", description: str = None, discovery=None, network=None):
  566. """初始化 ANP 工具
  567. Args:
  568. name: 工具名称
  569. description: 工具描述
  570. discovery: 可选的 ANPDiscovery 实例,如果不提供则创建新实例
  571. network: 可选的 ANPNetwork 实例,如果不提供则创建新实例
  572. """
  573. if description is None:
  574. description = "智能体网络管理工具,支持服务发现、节点管理和消息路由。概念性实现。"
  575. super().__init__(
  576. name=name,
  577. description=description
  578. )
  579. from hello_agents.protocols.anp.implementation import ANPDiscovery, ANPNetwork
  580. self._discovery = discovery if discovery is not None else ANPDiscovery()
  581. self._network = network if network is not None else ANPNetwork()
  582. def run(self, parameters: Dict[str, Any]) -> str:
  583. """
  584. 执行 ANP 操作
  585. Args:
  586. parameters: 包含以下参数的字典
  587. - action: 操作类型 (register_service, discover_services, add_node, route_message, get_stats)
  588. - service_id, service_type, endpoint: 服务信息(register_service 需要)
  589. - node_id, endpoint: 节点信息(add_node 需要)
  590. - from_node, to_node, message: 路由信息(route_message 需要)
  591. Returns:
  592. 操作结果
  593. """
  594. from hello_agents.protocols.anp.implementation import ServiceInfo
  595. action = parameters.get("action", "").lower()
  596. if not action:
  597. return "错误:必须指定 action 参数"
  598. try:
  599. if action == "register_service":
  600. service_id = parameters.get("service_id")
  601. service_type = parameters.get("service_type")
  602. endpoint = parameters.get("endpoint")
  603. metadata = parameters.get("metadata", {})
  604. if not all([service_id, service_type, endpoint]):
  605. return "错误:必须指定 service_id, service_type 和 endpoint 参数"
  606. service = ServiceInfo(service_id, service_type, endpoint, metadata)
  607. self._discovery.register_service(service)
  608. return f"✅ 已注册服务 '{service_id}'"
  609. elif action == "unregister_service":
  610. service_id = parameters.get("service_id")
  611. if not service_id:
  612. return "错误:必须指定 service_id 参数"
  613. # 使用 ANPDiscovery 的 unregister_service 方法
  614. success = self._discovery.unregister_service(service_id)
  615. if success:
  616. return f"✅ 已注销服务 '{service_id}'"
  617. else:
  618. return f"错误:服务 '{service_id}' 不存在"
  619. elif action == "discover_services":
  620. service_type = parameters.get("service_type")
  621. services = self._discovery.discover_services(service_type)
  622. if not services:
  623. return "没有找到服务"
  624. result = f"找到 {len(services)} 个服务:\n\n"
  625. for service in services:
  626. result += f"服务ID: {service.service_id}\n"
  627. result += f" 名称: {service.service_name}\n"
  628. result += f" 类型: {service.service_type}\n"
  629. result += f" 端点: {service.endpoint}\n"
  630. if service.capabilities:
  631. result += f" 能力: {', '.join(service.capabilities)}\n"
  632. if service.metadata:
  633. result += f" 元数据: {service.metadata}\n"
  634. result += "\n"
  635. return result
  636. elif action == "add_node":
  637. node_id = parameters.get("node_id")
  638. endpoint = parameters.get("endpoint")
  639. metadata = parameters.get("metadata", {})
  640. if not all([node_id, endpoint]):
  641. return "错误:必须指定 node_id 和 endpoint 参数"
  642. self._network.add_node(node_id, endpoint, metadata)
  643. return f"✅ 已添加节点 '{node_id}'"
  644. elif action == "route_message":
  645. from_node = parameters.get("from_node")
  646. to_node = parameters.get("to_node")
  647. message = parameters.get("message", {})
  648. if not all([from_node, to_node]):
  649. return "错误:必须指定 from_node 和 to_node 参数"
  650. path = self._network.route_message(from_node, to_node, message)
  651. if path:
  652. return f"消息路由路径: {' -> '.join(path)}"
  653. else:
  654. return "无法找到路由路径"
  655. elif action == "get_stats":
  656. stats = self._network.get_network_stats()
  657. result = "网络统计:\n"
  658. for key, value in stats.items():
  659. result += f"- {key}: {value}\n"
  660. return result
  661. else:
  662. return f"错误:不支持的操作 '{action}'"
  663. except Exception as e:
  664. return f"ANP 操作失败: {str(e)}"
  665. def get_parameters(self) -> List[ToolParameter]:
  666. """获取工具参数定义"""
  667. return [
  668. ToolParameter(
  669. name="action",
  670. type="string",
  671. description="操作类型: register_service, unregister_service, discover_services, add_node, route_message, get_stats",
  672. required=True
  673. ),
  674. ToolParameter(
  675. name="service_id",
  676. type="string",
  677. description="服务 ID(register_service, unregister_service 需要)",
  678. required=False
  679. ),
  680. ToolParameter(
  681. name="service_type",
  682. type="string",
  683. description="服务类型(register_service 需要)",
  684. required=False
  685. ),
  686. ToolParameter(
  687. name="endpoint",
  688. type="string",
  689. description="端点地址(register_service, add_node 需要)",
  690. required=False
  691. ),
  692. ToolParameter(
  693. name="node_id",
  694. type="string",
  695. description="节点 ID(add_node 需要)",
  696. required=False
  697. ),
  698. ToolParameter(
  699. name="from_node",
  700. type="string",
  701. description="源节点 ID(route_message 需要)",
  702. required=False
  703. ),
  704. ToolParameter(
  705. name="to_node",
  706. type="string",
  707. description="目标节点 ID(route_message 需要)",
  708. required=False
  709. ),
  710. ToolParameter(
  711. name="message",
  712. type="object",
  713. description="消息内容(route_message 需要)",
  714. required=False
  715. ),
  716. ToolParameter(
  717. name="metadata",
  718. type="object",
  719. description="元数据(register_service, add_node 可选)",
  720. required=False
  721. )
  722. ]