aip-agents-binary 0.5.20__py3-none-any.whl

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.
Files changed (280) hide show
  1. aip_agents/__init__.py +65 -0
  2. aip_agents/a2a/__init__.py +19 -0
  3. aip_agents/a2a/server/__init__.py +10 -0
  4. aip_agents/a2a/server/base_executor.py +1086 -0
  5. aip_agents/a2a/server/google_adk_executor.py +198 -0
  6. aip_agents/a2a/server/langflow_executor.py +180 -0
  7. aip_agents/a2a/server/langgraph_executor.py +270 -0
  8. aip_agents/a2a/types.py +232 -0
  9. aip_agents/agent/__init__.py +27 -0
  10. aip_agents/agent/base_agent.py +970 -0
  11. aip_agents/agent/base_langgraph_agent.py +2942 -0
  12. aip_agents/agent/google_adk_agent.py +926 -0
  13. aip_agents/agent/google_adk_constants.py +6 -0
  14. aip_agents/agent/hitl/__init__.py +24 -0
  15. aip_agents/agent/hitl/config.py +28 -0
  16. aip_agents/agent/hitl/langgraph_hitl_mixin.py +515 -0
  17. aip_agents/agent/hitl/manager.py +532 -0
  18. aip_agents/agent/hitl/models.py +18 -0
  19. aip_agents/agent/hitl/prompt/__init__.py +9 -0
  20. aip_agents/agent/hitl/prompt/base.py +42 -0
  21. aip_agents/agent/hitl/prompt/deferred.py +73 -0
  22. aip_agents/agent/hitl/registry.py +149 -0
  23. aip_agents/agent/interface.py +138 -0
  24. aip_agents/agent/interfaces.py +65 -0
  25. aip_agents/agent/langflow_agent.py +464 -0
  26. aip_agents/agent/langgraph_memory_enhancer_agent.py +433 -0
  27. aip_agents/agent/langgraph_react_agent.py +2514 -0
  28. aip_agents/agent/system_instruction_context.py +34 -0
  29. aip_agents/clients/__init__.py +10 -0
  30. aip_agents/clients/langflow/__init__.py +10 -0
  31. aip_agents/clients/langflow/client.py +477 -0
  32. aip_agents/clients/langflow/types.py +18 -0
  33. aip_agents/constants.py +23 -0
  34. aip_agents/credentials/manager.py +132 -0
  35. aip_agents/examples/__init__.py +5 -0
  36. aip_agents/examples/compare_streaming_client.py +783 -0
  37. aip_agents/examples/compare_streaming_server.py +142 -0
  38. aip_agents/examples/demo_memory_recall.py +401 -0
  39. aip_agents/examples/hello_world_a2a_google_adk_client.py +49 -0
  40. aip_agents/examples/hello_world_a2a_google_adk_client_agent.py +48 -0
  41. aip_agents/examples/hello_world_a2a_google_adk_client_streaming.py +60 -0
  42. aip_agents/examples/hello_world_a2a_google_adk_server.py +79 -0
  43. aip_agents/examples/hello_world_a2a_langchain_client.py +39 -0
  44. aip_agents/examples/hello_world_a2a_langchain_client_agent.py +39 -0
  45. aip_agents/examples/hello_world_a2a_langchain_client_lm_invoker.py +37 -0
  46. aip_agents/examples/hello_world_a2a_langchain_client_streaming.py +41 -0
  47. aip_agents/examples/hello_world_a2a_langchain_reference_client_streaming.py +60 -0
  48. aip_agents/examples/hello_world_a2a_langchain_reference_server.py +105 -0
  49. aip_agents/examples/hello_world_a2a_langchain_server.py +79 -0
  50. aip_agents/examples/hello_world_a2a_langchain_server_lm_invoker.py +78 -0
  51. aip_agents/examples/hello_world_a2a_langflow_client.py +83 -0
  52. aip_agents/examples/hello_world_a2a_langflow_server.py +82 -0
  53. aip_agents/examples/hello_world_a2a_langgraph_artifact_client.py +73 -0
  54. aip_agents/examples/hello_world_a2a_langgraph_artifact_client_streaming.py +76 -0
  55. aip_agents/examples/hello_world_a2a_langgraph_artifact_server.py +92 -0
  56. aip_agents/examples/hello_world_a2a_langgraph_client.py +54 -0
  57. aip_agents/examples/hello_world_a2a_langgraph_client_agent.py +54 -0
  58. aip_agents/examples/hello_world_a2a_langgraph_client_agent_lm_invoker.py +32 -0
  59. aip_agents/examples/hello_world_a2a_langgraph_client_streaming.py +50 -0
  60. aip_agents/examples/hello_world_a2a_langgraph_client_streaming_lm_invoker.py +44 -0
  61. aip_agents/examples/hello_world_a2a_langgraph_client_streaming_tool_streaming.py +92 -0
  62. aip_agents/examples/hello_world_a2a_langgraph_server.py +84 -0
  63. aip_agents/examples/hello_world_a2a_langgraph_server_lm_invoker.py +79 -0
  64. aip_agents/examples/hello_world_a2a_langgraph_server_tool_streaming.py +132 -0
  65. aip_agents/examples/hello_world_a2a_mcp_langgraph.py +196 -0
  66. aip_agents/examples/hello_world_a2a_three_level_agent_hierarchy_client.py +244 -0
  67. aip_agents/examples/hello_world_a2a_three_level_agent_hierarchy_server.py +251 -0
  68. aip_agents/examples/hello_world_a2a_with_metadata_langchain_client.py +57 -0
  69. aip_agents/examples/hello_world_a2a_with_metadata_langchain_server_lm_invoker.py +80 -0
  70. aip_agents/examples/hello_world_google_adk.py +41 -0
  71. aip_agents/examples/hello_world_google_adk_mcp_http.py +34 -0
  72. aip_agents/examples/hello_world_google_adk_mcp_http_stream.py +40 -0
  73. aip_agents/examples/hello_world_google_adk_mcp_sse.py +44 -0
  74. aip_agents/examples/hello_world_google_adk_mcp_sse_stream.py +48 -0
  75. aip_agents/examples/hello_world_google_adk_mcp_stdio.py +44 -0
  76. aip_agents/examples/hello_world_google_adk_mcp_stdio_stream.py +48 -0
  77. aip_agents/examples/hello_world_google_adk_stream.py +44 -0
  78. aip_agents/examples/hello_world_langchain.py +28 -0
  79. aip_agents/examples/hello_world_langchain_lm_invoker.py +15 -0
  80. aip_agents/examples/hello_world_langchain_mcp_http.py +34 -0
  81. aip_agents/examples/hello_world_langchain_mcp_http_interactive.py +130 -0
  82. aip_agents/examples/hello_world_langchain_mcp_http_stream.py +42 -0
  83. aip_agents/examples/hello_world_langchain_mcp_multi_server.py +155 -0
  84. aip_agents/examples/hello_world_langchain_mcp_sse.py +34 -0
  85. aip_agents/examples/hello_world_langchain_mcp_sse_stream.py +40 -0
  86. aip_agents/examples/hello_world_langchain_mcp_stdio.py +30 -0
  87. aip_agents/examples/hello_world_langchain_mcp_stdio_stream.py +41 -0
  88. aip_agents/examples/hello_world_langchain_stream.py +36 -0
  89. aip_agents/examples/hello_world_langchain_stream_lm_invoker.py +39 -0
  90. aip_agents/examples/hello_world_langflow_agent.py +163 -0
  91. aip_agents/examples/hello_world_langgraph.py +39 -0
  92. aip_agents/examples/hello_world_langgraph_bosa_twitter.py +41 -0
  93. aip_agents/examples/hello_world_langgraph_mcp_http.py +31 -0
  94. aip_agents/examples/hello_world_langgraph_mcp_http_stream.py +34 -0
  95. aip_agents/examples/hello_world_langgraph_mcp_sse.py +35 -0
  96. aip_agents/examples/hello_world_langgraph_mcp_sse_stream.py +50 -0
  97. aip_agents/examples/hello_world_langgraph_mcp_stdio.py +35 -0
  98. aip_agents/examples/hello_world_langgraph_mcp_stdio_stream.py +50 -0
  99. aip_agents/examples/hello_world_langgraph_stream.py +43 -0
  100. aip_agents/examples/hello_world_langgraph_stream_lm_invoker.py +37 -0
  101. aip_agents/examples/hello_world_model_switch_cli.py +210 -0
  102. aip_agents/examples/hello_world_multi_agent_adk.py +75 -0
  103. aip_agents/examples/hello_world_multi_agent_langchain.py +54 -0
  104. aip_agents/examples/hello_world_multi_agent_langgraph.py +66 -0
  105. aip_agents/examples/hello_world_multi_agent_langgraph_lm_invoker.py +69 -0
  106. aip_agents/examples/hello_world_pii_logger.py +21 -0
  107. aip_agents/examples/hello_world_sentry.py +133 -0
  108. aip_agents/examples/hello_world_step_limits.py +273 -0
  109. aip_agents/examples/hello_world_stock_a2a_server.py +103 -0
  110. aip_agents/examples/hello_world_tool_output_client.py +46 -0
  111. aip_agents/examples/hello_world_tool_output_server.py +114 -0
  112. aip_agents/examples/hitl_demo.py +724 -0
  113. aip_agents/examples/mcp_configs/configs.py +63 -0
  114. aip_agents/examples/mcp_servers/common.py +76 -0
  115. aip_agents/examples/mcp_servers/mcp_name.py +29 -0
  116. aip_agents/examples/mcp_servers/mcp_server_http.py +19 -0
  117. aip_agents/examples/mcp_servers/mcp_server_sse.py +19 -0
  118. aip_agents/examples/mcp_servers/mcp_server_stdio.py +19 -0
  119. aip_agents/examples/mcp_servers/mcp_time.py +10 -0
  120. aip_agents/examples/pii_demo_langgraph_client.py +69 -0
  121. aip_agents/examples/pii_demo_langgraph_server.py +126 -0
  122. aip_agents/examples/pii_demo_multi_agent_client.py +80 -0
  123. aip_agents/examples/pii_demo_multi_agent_server.py +247 -0
  124. aip_agents/examples/todolist_planning_a2a_langchain_client.py +70 -0
  125. aip_agents/examples/todolist_planning_a2a_langgraph_server.py +88 -0
  126. aip_agents/examples/tools/__init__.py +27 -0
  127. aip_agents/examples/tools/adk_arithmetic_tools.py +36 -0
  128. aip_agents/examples/tools/adk_weather_tool.py +60 -0
  129. aip_agents/examples/tools/data_generator_tool.py +103 -0
  130. aip_agents/examples/tools/data_visualization_tool.py +312 -0
  131. aip_agents/examples/tools/image_artifact_tool.py +136 -0
  132. aip_agents/examples/tools/langchain_arithmetic_tools.py +26 -0
  133. aip_agents/examples/tools/langchain_currency_exchange_tool.py +88 -0
  134. aip_agents/examples/tools/langchain_graph_artifact_tool.py +172 -0
  135. aip_agents/examples/tools/langchain_weather_tool.py +48 -0
  136. aip_agents/examples/tools/langgraph_streaming_tool.py +130 -0
  137. aip_agents/examples/tools/mock_retrieval_tool.py +56 -0
  138. aip_agents/examples/tools/pii_demo_tools.py +189 -0
  139. aip_agents/examples/tools/random_chart_tool.py +142 -0
  140. aip_agents/examples/tools/serper_tool.py +202 -0
  141. aip_agents/examples/tools/stock_tools.py +82 -0
  142. aip_agents/examples/tools/table_generator_tool.py +167 -0
  143. aip_agents/examples/tools/time_tool.py +82 -0
  144. aip_agents/examples/tools/weather_forecast_tool.py +38 -0
  145. aip_agents/executor/agent_executor.py +473 -0
  146. aip_agents/executor/base.py +48 -0
  147. aip_agents/mcp/__init__.py +1 -0
  148. aip_agents/mcp/client/__init__.py +14 -0
  149. aip_agents/mcp/client/base_mcp_client.py +369 -0
  150. aip_agents/mcp/client/connection_manager.py +193 -0
  151. aip_agents/mcp/client/google_adk/__init__.py +11 -0
  152. aip_agents/mcp/client/google_adk/client.py +381 -0
  153. aip_agents/mcp/client/langchain/__init__.py +11 -0
  154. aip_agents/mcp/client/langchain/client.py +265 -0
  155. aip_agents/mcp/client/persistent_session.py +359 -0
  156. aip_agents/mcp/client/session_pool.py +351 -0
  157. aip_agents/mcp/client/transports.py +215 -0
  158. aip_agents/mcp/utils/__init__.py +7 -0
  159. aip_agents/mcp/utils/config_validator.py +139 -0
  160. aip_agents/memory/__init__.py +14 -0
  161. aip_agents/memory/adapters/__init__.py +10 -0
  162. aip_agents/memory/adapters/base_adapter.py +717 -0
  163. aip_agents/memory/adapters/mem0.py +84 -0
  164. aip_agents/memory/base.py +84 -0
  165. aip_agents/memory/constants.py +49 -0
  166. aip_agents/memory/factory.py +86 -0
  167. aip_agents/memory/guidance.py +20 -0
  168. aip_agents/memory/simple_memory.py +47 -0
  169. aip_agents/middleware/__init__.py +17 -0
  170. aip_agents/middleware/base.py +88 -0
  171. aip_agents/middleware/manager.py +128 -0
  172. aip_agents/middleware/todolist.py +274 -0
  173. aip_agents/schema/__init__.py +69 -0
  174. aip_agents/schema/a2a.py +56 -0
  175. aip_agents/schema/agent.py +111 -0
  176. aip_agents/schema/hitl.py +157 -0
  177. aip_agents/schema/langgraph.py +37 -0
  178. aip_agents/schema/model_id.py +97 -0
  179. aip_agents/schema/step_limit.py +108 -0
  180. aip_agents/schema/storage.py +40 -0
  181. aip_agents/sentry/__init__.py +11 -0
  182. aip_agents/sentry/sentry.py +151 -0
  183. aip_agents/storage/__init__.py +41 -0
  184. aip_agents/storage/base.py +85 -0
  185. aip_agents/storage/clients/__init__.py +12 -0
  186. aip_agents/storage/clients/minio_client.py +318 -0
  187. aip_agents/storage/config.py +62 -0
  188. aip_agents/storage/providers/__init__.py +15 -0
  189. aip_agents/storage/providers/base.py +106 -0
  190. aip_agents/storage/providers/memory.py +114 -0
  191. aip_agents/storage/providers/object_storage.py +214 -0
  192. aip_agents/tools/__init__.py +33 -0
  193. aip_agents/tools/bosa_tools.py +105 -0
  194. aip_agents/tools/browser_use/__init__.py +82 -0
  195. aip_agents/tools/browser_use/action_parser.py +103 -0
  196. aip_agents/tools/browser_use/browser_use_tool.py +1112 -0
  197. aip_agents/tools/browser_use/llm_config.py +120 -0
  198. aip_agents/tools/browser_use/minio_storage.py +198 -0
  199. aip_agents/tools/browser_use/schemas.py +119 -0
  200. aip_agents/tools/browser_use/session.py +76 -0
  201. aip_agents/tools/browser_use/session_errors.py +132 -0
  202. aip_agents/tools/browser_use/steel_session_recording.py +317 -0
  203. aip_agents/tools/browser_use/streaming.py +813 -0
  204. aip_agents/tools/browser_use/structured_data_parser.py +257 -0
  205. aip_agents/tools/browser_use/structured_data_recovery.py +204 -0
  206. aip_agents/tools/browser_use/types.py +78 -0
  207. aip_agents/tools/code_sandbox/__init__.py +26 -0
  208. aip_agents/tools/code_sandbox/constant.py +13 -0
  209. aip_agents/tools/code_sandbox/e2b_cloud_sandbox_extended.py +257 -0
  210. aip_agents/tools/code_sandbox/e2b_sandbox_tool.py +411 -0
  211. aip_agents/tools/constants.py +165 -0
  212. aip_agents/tools/document_loader/__init__.py +44 -0
  213. aip_agents/tools/document_loader/base_reader.py +302 -0
  214. aip_agents/tools/document_loader/docx_reader_tool.py +68 -0
  215. aip_agents/tools/document_loader/excel_reader_tool.py +171 -0
  216. aip_agents/tools/document_loader/pdf_reader_tool.py +79 -0
  217. aip_agents/tools/document_loader/pdf_splitter.py +169 -0
  218. aip_agents/tools/gl_connector/__init__.py +5 -0
  219. aip_agents/tools/gl_connector/tool.py +351 -0
  220. aip_agents/tools/memory_search/__init__.py +22 -0
  221. aip_agents/tools/memory_search/base.py +200 -0
  222. aip_agents/tools/memory_search/mem0.py +258 -0
  223. aip_agents/tools/memory_search/schema.py +48 -0
  224. aip_agents/tools/memory_search_tool.py +26 -0
  225. aip_agents/tools/time_tool.py +117 -0
  226. aip_agents/tools/tool_config_injector.py +300 -0
  227. aip_agents/tools/web_search/__init__.py +15 -0
  228. aip_agents/tools/web_search/serper_tool.py +187 -0
  229. aip_agents/types/__init__.py +70 -0
  230. aip_agents/types/a2a_events.py +13 -0
  231. aip_agents/utils/__init__.py +79 -0
  232. aip_agents/utils/a2a_connector.py +1757 -0
  233. aip_agents/utils/artifact_helpers.py +502 -0
  234. aip_agents/utils/constants.py +22 -0
  235. aip_agents/utils/datetime/__init__.py +34 -0
  236. aip_agents/utils/datetime/normalization.py +231 -0
  237. aip_agents/utils/datetime/timezone.py +206 -0
  238. aip_agents/utils/env_loader.py +27 -0
  239. aip_agents/utils/event_handler_registry.py +58 -0
  240. aip_agents/utils/file_prompt_utils.py +176 -0
  241. aip_agents/utils/final_response_builder.py +211 -0
  242. aip_agents/utils/formatter_llm_client.py +231 -0
  243. aip_agents/utils/langgraph/__init__.py +19 -0
  244. aip_agents/utils/langgraph/converter.py +128 -0
  245. aip_agents/utils/langgraph/tool_managers/__init__.py +15 -0
  246. aip_agents/utils/langgraph/tool_managers/a2a_tool_manager.py +99 -0
  247. aip_agents/utils/langgraph/tool_managers/base_tool_manager.py +66 -0
  248. aip_agents/utils/langgraph/tool_managers/delegation_tool_manager.py +1071 -0
  249. aip_agents/utils/langgraph/tool_output_management.py +967 -0
  250. aip_agents/utils/logger.py +195 -0
  251. aip_agents/utils/metadata/__init__.py +27 -0
  252. aip_agents/utils/metadata/activity_metadata_helper.py +407 -0
  253. aip_agents/utils/metadata/activity_narrative/__init__.py +35 -0
  254. aip_agents/utils/metadata/activity_narrative/builder.py +817 -0
  255. aip_agents/utils/metadata/activity_narrative/constants.py +51 -0
  256. aip_agents/utils/metadata/activity_narrative/context.py +49 -0
  257. aip_agents/utils/metadata/activity_narrative/formatters.py +230 -0
  258. aip_agents/utils/metadata/activity_narrative/utils.py +35 -0
  259. aip_agents/utils/metadata/schemas/__init__.py +16 -0
  260. aip_agents/utils/metadata/schemas/activity_schema.py +29 -0
  261. aip_agents/utils/metadata/schemas/thinking_schema.py +31 -0
  262. aip_agents/utils/metadata/thinking_metadata_helper.py +38 -0
  263. aip_agents/utils/metadata_helper.py +358 -0
  264. aip_agents/utils/name_preprocessor/__init__.py +17 -0
  265. aip_agents/utils/name_preprocessor/base_name_preprocessor.py +73 -0
  266. aip_agents/utils/name_preprocessor/google_name_preprocessor.py +100 -0
  267. aip_agents/utils/name_preprocessor/name_preprocessor.py +87 -0
  268. aip_agents/utils/name_preprocessor/openai_name_preprocessor.py +48 -0
  269. aip_agents/utils/pii/__init__.py +25 -0
  270. aip_agents/utils/pii/pii_handler.py +397 -0
  271. aip_agents/utils/pii/pii_helper.py +207 -0
  272. aip_agents/utils/pii/uuid_deanonymizer_mapping.py +195 -0
  273. aip_agents/utils/reference_helper.py +273 -0
  274. aip_agents/utils/sse_chunk_transformer.py +831 -0
  275. aip_agents/utils/step_limit_manager.py +265 -0
  276. aip_agents/utils/token_usage_helper.py +156 -0
  277. aip_agents_binary-0.5.20.dist-info/METADATA +681 -0
  278. aip_agents_binary-0.5.20.dist-info/RECORD +280 -0
  279. aip_agents_binary-0.5.20.dist-info/WHEEL +5 -0
  280. aip_agents_binary-0.5.20.dist-info/top_level.txt +1 -0
@@ -0,0 +1,381 @@
1
+ """Google ADK MCP Adapter for MCP Client with Session Persistence.
2
+
3
+ This module contains the GoogleADKMCPClient class, which extends the BaseMCPClient
4
+ to integrate persistent MCP tools with Google's Agent Development Kit (ADK).
5
+
6
+ The GoogleADKMCPClient adapts MCP tools into ADK FunctionTool instances that can
7
+ be used seamlessly with ADK agents while maintaining session persistence across
8
+ multiple tool calls.
9
+
10
+ Authors:
11
+ Fachriza Dian Adhiatma (fachriza.d.adhiatma@gdplabs.id)
12
+ Putu Ravindra Wiguna (putu.r.wiguna@gdplabs.id)
13
+ """
14
+
15
+ import base64
16
+ from typing import Any
17
+
18
+ from gllm_tools.mcp.client.config import MCPConfiguration
19
+ from gllm_tools.mcp.client.resource import MCPResource
20
+ from gllm_tools.mcp.client.tool import MCPTool
21
+ from mcp.types import (
22
+ BlobResourceContents,
23
+ CallToolResult,
24
+ EmbeddedResource,
25
+ ImageContent,
26
+ TextContent,
27
+ TextResourceContents,
28
+ )
29
+
30
+ from aip_agents.utils.logger import get_logger
31
+
32
+ try:
33
+ from google.adk.tools import FunctionTool
34
+ from google.genai.types import Part
35
+ except ImportError as e:
36
+ raise ImportError("Google ADK is required to use GoogleADKMCPClient. Install with: pip install google-adk") from e
37
+
38
+ from aip_agents.mcp.client.base_mcp_client import BaseMCPClient
39
+
40
+ NonTextContent = ImageContent | EmbeddedResource
41
+
42
+ logger = get_logger(__name__)
43
+
44
+
45
+ class GoogleADKMCPClient(BaseMCPClient):
46
+ """Google ADK MCP Client with Persistent Sessions.
47
+
48
+ This client extends BaseMCPClient to provide Google ADK-specific tool conversion
49
+ while maintaining persistent MCP sessions and connection reuse across tool calls.
50
+ It converts MCP tools into ADK FunctionTool instances for seamless integration.
51
+
52
+ The client handles:
53
+ - Converting MCP tools to ADK FunctionTool instances using persistent sessions
54
+ - Managing MCP server connections with automatic reconnection
55
+ - Converting MCP resources to ADK-compatible formats
56
+ - Handling tool execution with proper error formatting for ADK agents
57
+
58
+ Example:
59
+ ```python
60
+ from aip_agents.mcp.client.google_adk.client import GoogleADKMCPClient
61
+ from gllm_tools.mcp.client.config import MCPConfiguration
62
+
63
+ servers = {
64
+ "filesystem": MCPConfiguration(
65
+ command="npx",
66
+ args=["-y", "@modelcontextprotocol/server-filesystem", "/path/to/folder"]
67
+ )
68
+ }
69
+
70
+ client = GoogleADKMCPClient(servers)
71
+ await client.initialize() # Initialize persistent sessions
72
+ tools = await client.get_tools() # Returns list of ADK FunctionTool instances
73
+ ```
74
+ """
75
+
76
+ RESOURCE_FETCH_TIMEOUT = 10
77
+
78
+ def __init__(self, servers: dict[str, MCPConfiguration]):
79
+ """Initialize Google ADK MCP client.
80
+
81
+ Args:
82
+ servers (dict[str, MCPConfiguration]): Dictionary of MCP server configurations
83
+ """
84
+ super().__init__(servers)
85
+ # Cache converted ADK tools for consistent pattern with LangChain client
86
+ self._adk_tools_cache: dict[str | None, list[FunctionTool]] = {}
87
+
88
+ async def initialize(self) -> None:
89
+ """Initialize persistent MCP sessions for Google ADK integration.
90
+
91
+ This method ensures all MCP servers are connected with persistent sessions
92
+ and prepares the client for ADK tool conversion.
93
+
94
+ Raises:
95
+ Exception: If session initialization fails
96
+ """
97
+ await super().initialize()
98
+ logger.info(f"GoogleADKMCPClient initialized with {self.get_tools_count()} MCP tools ready for ADK conversion")
99
+
100
+ async def get_tools(self, server: str | None = None) -> list[FunctionTool]:
101
+ """Get ADK-compatible FunctionTool instances with smart caching.
102
+
103
+ Converts MCP tools to ADK format and caches them for better performance
104
+ on repeated access. Cache is keyed by server parameter.
105
+
106
+ Args:
107
+ server (str | None): Optional server name to filter tools from a specific server.
108
+ If None, returns tools from all configured servers.
109
+
110
+ Returns:
111
+ list[FunctionTool]: List of cached ADK FunctionTool instances.
112
+ """
113
+ if not self.is_initialized:
114
+ await self.initialize()
115
+
116
+ if server:
117
+ # Get tools from specific server with caching
118
+ server_tools = await self._get_server_tools_cached(server)
119
+ return server_tools.copy()
120
+ else:
121
+ # Get tools from all servers with efficient caching
122
+ all_tools = []
123
+ for server_name in self.servers.keys():
124
+ try:
125
+ server_tools = await self._get_server_tools_cached(server_name)
126
+ all_tools.extend(server_tools)
127
+ except Exception as e:
128
+ logger.warning(f"Failed to get tools from server '{server_name}': {e}")
129
+
130
+ logger.debug(f"Retrieved {len(all_tools)} total ADK FunctionTools from {len(self.servers)} servers")
131
+ return all_tools
132
+
133
+ async def _get_server_tools_cached(self, server_name: str) -> list[FunctionTool]:
134
+ """Get tools for a specific server with caching.
135
+
136
+ This method centralizes caching logic and ensures tools are only converted once per server.
137
+
138
+ Args:
139
+ server_name (str): Name of the MCP server
140
+
141
+ Returns:
142
+ list[FunctionTool]: List of cached FunctionTool instances for the server
143
+ """
144
+ # Check cache first
145
+ if server_name in self._adk_tools_cache:
146
+ logger.debug(f"Using cached ADK tools for server '{server_name}'")
147
+ return self._adk_tools_cache[server_name]
148
+
149
+ # Convert and cache tools for this server
150
+ logger.info(f"Converting and caching tools for server '{server_name}'")
151
+ mcp_tools = await self.get_raw_mcp_tools(server_name)
152
+ adk_tools = []
153
+
154
+ for mcp_tool in mcp_tools:
155
+ adk_tool = self._process_tool(mcp_tool, server_name)
156
+ adk_tools.append(adk_tool)
157
+ logger.info(f"Converted MCP tool '{mcp_tool.name}' to ADK FunctionTool for server '{server_name}'")
158
+
159
+ # Cache the converted tools
160
+ self._adk_tools_cache[server_name] = adk_tools
161
+ logger.debug(f"Cached {len(adk_tools)} ADK FunctionTools for server '{server_name}'")
162
+
163
+ return adk_tools
164
+
165
+ def _process_tool(self, tool: MCPTool, server_name: str | None = None) -> FunctionTool:
166
+ """Converts an MCP tool into an ADK FunctionTool using persistent session.
167
+
168
+ This method creates a dynamic function that wraps the MCP tool execution
169
+ using the base class's persistent session management, and converts the
170
+ response format to be compatible with ADK's expectations.
171
+
172
+ Args:
173
+ tool (MCPTool): The MCP tool to convert.
174
+ server_name (str | None): The server name for routing tool calls.
175
+
176
+ Returns:
177
+ FunctionTool: An ADK FunctionTool instance that wraps the MCP tool
178
+ with persistent session support.
179
+ """
180
+ # Store the original MCP tool name for the actual server call
181
+ original_tool_name = tool.name
182
+
183
+ # Create the dynamic function that will be wrapped by FunctionTool
184
+ async def mcp_tool_function(**arguments: dict[str, Any]) -> dict[str, Any]:
185
+ """Dynamic function that executes the MCP tool with persistent session.
186
+
187
+ This function uses the base class's call_tool method which handles
188
+ server routing and persistent session management automatically.
189
+
190
+ Args:
191
+ **arguments (dict[str, Any]): Keyword arguments for the MCP tool execution.
192
+
193
+ Returns:
194
+ dict[str, Any]: The tool execution result.
195
+ """
196
+ try:
197
+ # Determine server to route the call to
198
+ resolved_server = server_name or next(iter(self.servers.keys()), None)
199
+ if not resolved_server:
200
+ raise RuntimeError("No MCP servers configured for executing tool")
201
+
202
+ # Use the original MCP tool name for the server call, not the sanitized name
203
+ call_tool_result = await self.call_tool(resolved_server, original_tool_name, arguments)
204
+ return self._convert_call_tool_result(call_tool_result)
205
+ except Exception as e:
206
+ logger.error(f"MCP tool '{original_tool_name}' execution failed: {e}")
207
+ return {"status": "error", "message": str(e)}
208
+
209
+ # Set function metadata for ADK introspection (will be sanitized later by agent)
210
+ mcp_tool_function.__name__ = tool.name
211
+ mcp_tool_function.__doc__ = tool.description or f"MCP tool: {tool.name} (from server: {server_name})"
212
+
213
+ # Create and return the ADK FunctionTool
214
+ return FunctionTool(func=mcp_tool_function)
215
+
216
+ async def _process_resource(self, resource: MCPResource) -> dict[str, Any]:
217
+ """Converts an MCP resource into an ADK-compatible format using persistent session.
218
+
219
+ Args:
220
+ resource (MCPResource): The MCP resource to convert.
221
+
222
+ Returns:
223
+ dict[str, Any]: A dictionary containing resource metadata and accessor function.
224
+ """
225
+ # Determine server name from resource URI or use first available server
226
+ server_name = self._determine_server_name_for_resource(resource)
227
+
228
+ async def read_resource_content() -> Part:
229
+ """Reads the actual content of the MCP resource using persistent session."""
230
+ try:
231
+ # Use base class method for persistent session resource access
232
+ resource_result = await self.read_resource(server_name, str(resource.uri))
233
+
234
+ contents = resource_result.contents[0]
235
+ if isinstance(contents, TextResourceContents):
236
+ return Part.from_text(contents.text)
237
+ elif isinstance(contents, BlobResourceContents):
238
+ data = base64.b64decode(contents.blob)
239
+ return Part.from_bytes(data=data, mime_type=resource.mime_type)
240
+ else:
241
+ raise ValueError(f"Unsupported content type for URI {resource.uri}")
242
+ except Exception as e:
243
+ # Do not break callers; return a safe textual Part describing the error
244
+ logger.error(f"Failed to read MCP resource {resource.uri}: {e}")
245
+ try:
246
+ return Part.from_text(f"Error reading resource {resource.uri}: {e}")
247
+ except Exception as part_error:
248
+ logger.error(f"Failed to create error Part for resource {resource.uri}: {part_error}")
249
+ raise ValueError(f"Cannot create ADK Part for error response: {part_error}") from part_error
250
+
251
+ return {
252
+ "uri": str(resource.uri),
253
+ "name": resource.name,
254
+ "description": resource.description,
255
+ "mime_type": resource.mime_type,
256
+ "read_content": read_resource_content,
257
+ "metadata": {
258
+ "uri": resource.uri,
259
+ "annotations": resource.annotations.model_dump() if resource.annotations else None,
260
+ },
261
+ }
262
+
263
+ async def cleanup(self) -> None:
264
+ """Clean up Google ADK MCP client resources.
265
+
266
+ This method ensures all persistent sessions are properly closed and
267
+ ADK-specific resources are cleaned up.
268
+ """
269
+ logger.info("Cleaning up GoogleADKMCPClient resources")
270
+ try:
271
+ await super().cleanup()
272
+ except Exception as e:
273
+ logger.error(f"Error during base GoogleADKMCPClient cleanup: {e}", exc_info=True)
274
+ finally:
275
+ # Always clear the ADK tools cache
276
+ self._adk_tools_cache.clear()
277
+ logger.info("GoogleADKMCPClient cleanup completed")
278
+
279
+ def _convert_call_tool_result(self, call_tool_result: CallToolResult) -> dict[str, Any]:
280
+ """Converts an MCP call tool result into an ADK-compatible format.
281
+
282
+ ADK tools should return dictionaries with meaningful keys. This method
283
+ extracts text content and formats it appropriately for ADK agents.
284
+
285
+ Args:
286
+ call_tool_result (CallToolResult): The MCP tool execution result.
287
+
288
+ Returns:
289
+ dict[str, Any]: ADK-compatible result dictionary.
290
+
291
+ Raises:
292
+ Exception: If the tool execution resulted in an error.
293
+ """
294
+ text_contents, non_text_contents = self._separate_contents(call_tool_result.content)
295
+
296
+ if call_tool_result.isError:
297
+ error_message = self._format_error_message(text_contents)
298
+ raise RuntimeError(f"MCP tool execution failed: {error_message}")
299
+
300
+ result = {"status": "success"}
301
+ if text_contents:
302
+ result["result"] = (
303
+ text_contents[0].text if len(text_contents) == 1 else [content.text for content in text_contents]
304
+ )
305
+ else:
306
+ result["result"] = "Tool executed successfully"
307
+
308
+ artifacts = [a for a in (self._format_artifact(c) for c in non_text_contents) if a]
309
+ if artifacts:
310
+ result["artifacts"] = artifacts
311
+
312
+ return result
313
+
314
+ @staticmethod
315
+ def _separate_contents(contents) -> tuple[list[TextContent], list[NonTextContent]]:
316
+ """Separates a list of content objects into text and non-text content.
317
+
318
+ This helper method processes a list of content objects and categorizes them
319
+ into text content (TextContent) and other content types for further processing.
320
+
321
+ Args:
322
+ contents (list[Any]): List of content objects to be separated.
323
+
324
+ Returns:
325
+ tuple[list[TextContent], list[NonTextContent]]: A tuple containing two lists:
326
+ - First: TextContent objects
327
+ - Second: All other content types.
328
+ """
329
+ text_contents = []
330
+ non_text_contents = []
331
+ for content in contents:
332
+ if isinstance(content, TextContent):
333
+ text_contents.append(content)
334
+ else:
335
+ non_text_contents.append(content)
336
+ return text_contents, non_text_contents
337
+
338
+ @staticmethod
339
+ def _format_artifact(content: NonTextContent) -> dict[str, Any] | None:
340
+ """Formats non-text content into ADK-compatible artifact dictionaries.
341
+
342
+ Converts different types of content objects (images, embedded resources) into
343
+ a standardized dictionary format expected by ADK agents.
344
+
345
+ Args:
346
+ content (NonTextContent): The content object to be formatted. Can be either ImageContent
347
+ or EmbeddedResource.
348
+
349
+ Returns:
350
+ dict[str, Any] | None: A dictionary containing the formatted content with appropriate type-specific
351
+ fields, or None if the content type is not supported.
352
+ """
353
+ if isinstance(content, ImageContent):
354
+ return {
355
+ "type": "image",
356
+ "data": content.data,
357
+ "mime_type": content.mimeType,
358
+ }
359
+ if isinstance(content, EmbeddedResource):
360
+ return {
361
+ "type": "resource",
362
+ "uri": str(content.resource.uri),
363
+ "text": getattr(content.resource, "text", None),
364
+ }
365
+ return None
366
+
367
+ @staticmethod
368
+ def _format_error_message(text_contents: list[TextContent]) -> str:
369
+ """Formats a list of text contents into a single error message string.
370
+
371
+ Combines multiple text content objects into a single string, typically
372
+ used for creating human-readable error messages from tool execution results.
373
+
374
+ Args:
375
+ text_contents (list[TextContent]): List of TextContent objects containing error message parts.
376
+
377
+ Returns:
378
+ str: A single string containing all text contents joined by spaces,
379
+ or an empty string if the input list is empty.
380
+ """
381
+ return " ".join(content.text for content in text_contents) if text_contents else ""
@@ -0,0 +1,11 @@
1
+ """Langchain MCP Client.
2
+
3
+ This module provides an alias for Langchain MCP Client from GLLM Tools.
4
+
5
+ Authors:
6
+ Putu Ravindra Wiguna (putu.r.wiguna@gdplabs.id)
7
+ """
8
+
9
+ from aip_agents.mcp.client.langchain.client import LangchainMCPClient
10
+
11
+ __all__ = ["LangchainMCPClient"]
@@ -0,0 +1,265 @@
1
+ """Langchain MCP Adapter for MCP Client with Session Persistence.
2
+
3
+ Authors:
4
+ Samuel Lusandi (samuel.lusandi@gdplabs.id)
5
+ Putu Ravindra Wiguna (putu.r.wiguna@gdplabs.id)
6
+ """
7
+
8
+ import asyncio
9
+ import base64
10
+ from typing import Any
11
+
12
+ from gllm_tools.mcp.client.config import MCPConfiguration
13
+ from gllm_tools.mcp.client.resource import MCPResource
14
+ from gllm_tools.mcp.client.tool import MCPTool
15
+ from langchain_core.documents.base import Blob
16
+ from langchain_core.tools import StructuredTool, ToolException
17
+ from mcp.types import (
18
+ BlobResourceContents,
19
+ CallToolResult,
20
+ EmbeddedResource,
21
+ ImageContent,
22
+ TextContent,
23
+ TextResourceContents,
24
+ )
25
+ from pydantic import AnyUrl
26
+
27
+ from aip_agents.mcp.client.base_mcp_client import BaseMCPClient
28
+ from aip_agents.utils.logger import get_logger
29
+
30
+ NonTextContent = ImageContent | EmbeddedResource
31
+
32
+ logger = get_logger(__name__)
33
+
34
+
35
+ class LangchainMCPClient(BaseMCPClient):
36
+ """Langchain MCP Client with Session Persistence.
37
+
38
+ This client extends BaseMCPClient to provide LangChain-specific tool conversion
39
+ while maintaining persistent MCP sessions and connection reuse across tool calls.
40
+ """
41
+
42
+ RESOURCE_FETCH_TIMEOUT = 10
43
+
44
+ def __init__(self, servers: dict[str, MCPConfiguration]):
45
+ """Initialize LangChain MCP client.
46
+
47
+ Args:
48
+ servers (dict[str, MCPConfiguration]): Dictionary of MCP server configurations
49
+ """
50
+ super().__init__(servers)
51
+ # Cache converted LangChain tools for better performance on repeated access
52
+ self._langchain_tools_cache: dict[str | None, list[StructuredTool]] = {}
53
+
54
+ async def initialize(self) -> None:
55
+ """Initialize all sessions for LangChain client.
56
+
57
+ This method initializes the base MCP sessions and prepares for tool caching.
58
+
59
+ Raises:
60
+ Exception: If base initialization fails
61
+ """
62
+ # Call base class initialization - this caches raw MCP tools
63
+ await super().initialize()
64
+
65
+ logger.info(f"LangchainMCPClient initialized with {self.get_tools_count()} MCP tools available for conversion")
66
+
67
+ def _process_tool(self, tool: MCPTool, server_name: str) -> StructuredTool:
68
+ """Converts an MCP tool into a Langchain StructuredTool.
69
+
70
+ Args:
71
+ tool (MCPTool): The tool to convert.
72
+ server_name (str): The name of the MCP server.
73
+
74
+ Returns:
75
+ StructuredTool: The converted tool.
76
+ """
77
+ # Store the original MCP tool name for the actual server call
78
+ original_tool_name = tool.name
79
+
80
+ async def call_mcp_tool(
81
+ **arguments: dict[str, Any],
82
+ ) -> tuple[str | list[str], list[NonTextContent] | None]:
83
+ """Invoke the underlying MCP tool and normalize its response.
84
+
85
+ Args:
86
+ **arguments: Structured arguments expected by the MCP tool.
87
+
88
+ Returns:
89
+ tuple: Textual response (string or list of strings) and optional non-text artifacts.
90
+ """
91
+ # Use the original MCP tool name for the server call, not the sanitized name
92
+ call_tool_result = await self.call_tool(server_name, original_tool_name, arguments)
93
+ return self._convert_call_tool_result(call_tool_result)
94
+
95
+ return StructuredTool(
96
+ name=tool.name,
97
+ description=tool.description or "",
98
+ args_schema=tool.inputSchema,
99
+ coroutine=call_mcp_tool,
100
+ response_format="content_and_artifact",
101
+ metadata=tool.annotations.model_dump() if tool.annotations else None,
102
+ )
103
+
104
+ async def get_tools(self, server: str | None = None) -> list[StructuredTool]:
105
+ """Get LangChain StructuredTools with smart caching.
106
+
107
+ Converts MCP tools to LangChain format and caches them for better performance
108
+ on repeated access. Cache is keyed by server parameter.
109
+
110
+ Args:
111
+ server (str | None): Optional server name to filter tools. If None, returns all tools.
112
+
113
+ Returns:
114
+ list[StructuredTool]: List of cached LangChain StructuredTool instances
115
+ """
116
+ if not self.is_initialized:
117
+ await self.initialize()
118
+
119
+ if server:
120
+ # Get tools from specific server with caching
121
+ server_tools = await self._get_server_tools_cached(server)
122
+ return server_tools.copy()
123
+ else:
124
+ # Get tools from all servers with efficient caching
125
+ all_tools = []
126
+ for server_name in self.servers.keys():
127
+ try:
128
+ server_tools = await self._get_server_tools_cached(server_name)
129
+ all_tools.extend(server_tools)
130
+ except Exception as e:
131
+ logger.warning(f"Failed to get tools from server '{server_name}': {e}")
132
+
133
+ logger.debug(f"Retrieved {len(all_tools)} total LangChain tools from {len(self.servers)} servers")
134
+ return all_tools
135
+
136
+ async def _get_server_tools_cached(self, server_name: str) -> list[StructuredTool]:
137
+ """Get tools for a specific server with caching.
138
+
139
+ This method centralizes caching logic and ensures tools are only converted once per server.
140
+
141
+ Args:
142
+ server_name (str): Name of the MCP server
143
+
144
+ Returns:
145
+ list[StructuredTool]: List of cached StructuredTool instances for the server
146
+ """
147
+ # Check cache first
148
+ if server_name in self._langchain_tools_cache:
149
+ logger.debug(f"Using cached LangChain tools for server '{server_name}'")
150
+ return self._langchain_tools_cache[server_name]
151
+
152
+ # Convert and cache tools for this server
153
+ logger.info(f"Converting and caching tools for server '{server_name}'")
154
+ mcp_tools = await self.get_raw_mcp_tools(server_name)
155
+ langchain_tools = []
156
+
157
+ for mcp_tool in mcp_tools:
158
+ langchain_tool = self._process_tool(mcp_tool, server_name)
159
+ langchain_tools.append(langchain_tool)
160
+ logger.info(
161
+ f"Converted MCP tool '{mcp_tool.name}' "
162
+ f"to LangChain tool '{langchain_tool.name}' "
163
+ f"for server '{server_name}'"
164
+ )
165
+
166
+ # Cache the converted tools
167
+ self._langchain_tools_cache[server_name] = langchain_tools
168
+ logger.debug(f"Cached {len(langchain_tools)} LangChain tools for server '{server_name}'")
169
+
170
+ return langchain_tools
171
+
172
+ async def _process_resource(self, resource: MCPResource) -> Any:
173
+ """Converts an MCP resource into a Langchain Resource using persistent session.
174
+
175
+ Args:
176
+ resource (MCPResource): The resource to convert.
177
+
178
+ Returns:
179
+ Blob: The converted resource.
180
+ """
181
+ # Determine server name from resource URI or use first available server
182
+ server_name = self._determine_server_name_for_resource(resource)
183
+
184
+ async def read_resource(uri: AnyUrl) -> str:
185
+ """Read the actual content of the MCP resource using persistent session.
186
+
187
+ Note: This function is async for consistency with MCP operations, even though
188
+ the uri parameter is a synchronous AnyUrl object.
189
+
190
+ Args:
191
+ uri (AnyUrl): The URI of the resource to read.
192
+
193
+ Returns:
194
+ str: The content of the resource as a string.
195
+ """
196
+ try:
197
+ # Use base class method for persistent session resource access
198
+ resource_result = await self.read_resource(server_name, str(uri))
199
+
200
+ if not resource_result.contents:
201
+ raise ValueError(f"No contents found for resource {uri}")
202
+
203
+ contents = resource_result.contents[0]
204
+ if isinstance(contents, TextResourceContents):
205
+ return contents.text
206
+ elif isinstance(contents, BlobResourceContents):
207
+ return base64.b64decode(contents.blob)
208
+ else:
209
+ raise ValueError(f"Unsupported content type for URI {uri}")
210
+ except Exception as e:
211
+ logger.error(f"Failed to read MCP resource {uri}: {e}")
212
+ raise
213
+
214
+ return Blob.from_data(
215
+ await asyncio.wait_for(read_resource(resource.uri), timeout=self.RESOURCE_FETCH_TIMEOUT),
216
+ mime_type=resource.mime_type,
217
+ path=str(resource.uri),
218
+ metadata={"uri": resource.uri},
219
+ )
220
+
221
+ async def cleanup(self) -> None:
222
+ """Cleanup LangChain MCP resources.
223
+
224
+ This method extends base class cleanup and clears the LangChain tool cache.
225
+ """
226
+ logger.info("Cleaning up LangchainMCPClient resources")
227
+ try:
228
+ await super().cleanup()
229
+ except Exception as e:
230
+ logger.error(f"Error during base LangchainMCPClient cleanup: {e}", exc_info=True)
231
+ finally:
232
+ # Always clear the LangChain tools cache
233
+ self._langchain_tools_cache.clear()
234
+ logger.info("LangchainMCPClient cleanup complete")
235
+
236
+ def _convert_call_tool_result(
237
+ self,
238
+ call_tool_result: CallToolResult,
239
+ ) -> tuple[str | list[str], list[NonTextContent] | None]:
240
+ """Converts an MCP call tool result into a tuple of text and non-text contents.
241
+
242
+ Args:
243
+ call_tool_result (CallToolResult): The call tool result to convert.
244
+
245
+ Returns:
246
+ tuple[str | list[str], list[NonTextContent] | None]: The converted call tool result.
247
+ """
248
+ text_contents: list[TextContent] = []
249
+ non_text_contents = []
250
+ for content in call_tool_result.content:
251
+ if isinstance(content, TextContent):
252
+ text_contents.append(content)
253
+ else:
254
+ non_text_contents.append(content)
255
+
256
+ tool_content: str | list[str] = [content.text for content in text_contents]
257
+ if not text_contents:
258
+ tool_content = ""
259
+ elif len(text_contents) == 1:
260
+ tool_content = tool_content[0]
261
+
262
+ if call_tool_result.isError:
263
+ raise ToolException(tool_content)
264
+
265
+ return tool_content, non_text_contents if non_text_contents else None