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.
- aip_agents/__init__.py +65 -0
- aip_agents/a2a/__init__.py +19 -0
- aip_agents/a2a/server/__init__.py +10 -0
- aip_agents/a2a/server/base_executor.py +1086 -0
- aip_agents/a2a/server/google_adk_executor.py +198 -0
- aip_agents/a2a/server/langflow_executor.py +180 -0
- aip_agents/a2a/server/langgraph_executor.py +270 -0
- aip_agents/a2a/types.py +232 -0
- aip_agents/agent/__init__.py +27 -0
- aip_agents/agent/base_agent.py +970 -0
- aip_agents/agent/base_langgraph_agent.py +2942 -0
- aip_agents/agent/google_adk_agent.py +926 -0
- aip_agents/agent/google_adk_constants.py +6 -0
- aip_agents/agent/hitl/__init__.py +24 -0
- aip_agents/agent/hitl/config.py +28 -0
- aip_agents/agent/hitl/langgraph_hitl_mixin.py +515 -0
- aip_agents/agent/hitl/manager.py +532 -0
- aip_agents/agent/hitl/models.py +18 -0
- aip_agents/agent/hitl/prompt/__init__.py +9 -0
- aip_agents/agent/hitl/prompt/base.py +42 -0
- aip_agents/agent/hitl/prompt/deferred.py +73 -0
- aip_agents/agent/hitl/registry.py +149 -0
- aip_agents/agent/interface.py +138 -0
- aip_agents/agent/interfaces.py +65 -0
- aip_agents/agent/langflow_agent.py +464 -0
- aip_agents/agent/langgraph_memory_enhancer_agent.py +433 -0
- aip_agents/agent/langgraph_react_agent.py +2514 -0
- aip_agents/agent/system_instruction_context.py +34 -0
- aip_agents/clients/__init__.py +10 -0
- aip_agents/clients/langflow/__init__.py +10 -0
- aip_agents/clients/langflow/client.py +477 -0
- aip_agents/clients/langflow/types.py +18 -0
- aip_agents/constants.py +23 -0
- aip_agents/credentials/manager.py +132 -0
- aip_agents/examples/__init__.py +5 -0
- aip_agents/examples/compare_streaming_client.py +783 -0
- aip_agents/examples/compare_streaming_server.py +142 -0
- aip_agents/examples/demo_memory_recall.py +401 -0
- aip_agents/examples/hello_world_a2a_google_adk_client.py +49 -0
- aip_agents/examples/hello_world_a2a_google_adk_client_agent.py +48 -0
- aip_agents/examples/hello_world_a2a_google_adk_client_streaming.py +60 -0
- aip_agents/examples/hello_world_a2a_google_adk_server.py +79 -0
- aip_agents/examples/hello_world_a2a_langchain_client.py +39 -0
- aip_agents/examples/hello_world_a2a_langchain_client_agent.py +39 -0
- aip_agents/examples/hello_world_a2a_langchain_client_lm_invoker.py +37 -0
- aip_agents/examples/hello_world_a2a_langchain_client_streaming.py +41 -0
- aip_agents/examples/hello_world_a2a_langchain_reference_client_streaming.py +60 -0
- aip_agents/examples/hello_world_a2a_langchain_reference_server.py +105 -0
- aip_agents/examples/hello_world_a2a_langchain_server.py +79 -0
- aip_agents/examples/hello_world_a2a_langchain_server_lm_invoker.py +78 -0
- aip_agents/examples/hello_world_a2a_langflow_client.py +83 -0
- aip_agents/examples/hello_world_a2a_langflow_server.py +82 -0
- aip_agents/examples/hello_world_a2a_langgraph_artifact_client.py +73 -0
- aip_agents/examples/hello_world_a2a_langgraph_artifact_client_streaming.py +76 -0
- aip_agents/examples/hello_world_a2a_langgraph_artifact_server.py +92 -0
- aip_agents/examples/hello_world_a2a_langgraph_client.py +54 -0
- aip_agents/examples/hello_world_a2a_langgraph_client_agent.py +54 -0
- aip_agents/examples/hello_world_a2a_langgraph_client_agent_lm_invoker.py +32 -0
- aip_agents/examples/hello_world_a2a_langgraph_client_streaming.py +50 -0
- aip_agents/examples/hello_world_a2a_langgraph_client_streaming_lm_invoker.py +44 -0
- aip_agents/examples/hello_world_a2a_langgraph_client_streaming_tool_streaming.py +92 -0
- aip_agents/examples/hello_world_a2a_langgraph_server.py +84 -0
- aip_agents/examples/hello_world_a2a_langgraph_server_lm_invoker.py +79 -0
- aip_agents/examples/hello_world_a2a_langgraph_server_tool_streaming.py +132 -0
- aip_agents/examples/hello_world_a2a_mcp_langgraph.py +196 -0
- aip_agents/examples/hello_world_a2a_three_level_agent_hierarchy_client.py +244 -0
- aip_agents/examples/hello_world_a2a_three_level_agent_hierarchy_server.py +251 -0
- aip_agents/examples/hello_world_a2a_with_metadata_langchain_client.py +57 -0
- aip_agents/examples/hello_world_a2a_with_metadata_langchain_server_lm_invoker.py +80 -0
- aip_agents/examples/hello_world_google_adk.py +41 -0
- aip_agents/examples/hello_world_google_adk_mcp_http.py +34 -0
- aip_agents/examples/hello_world_google_adk_mcp_http_stream.py +40 -0
- aip_agents/examples/hello_world_google_adk_mcp_sse.py +44 -0
- aip_agents/examples/hello_world_google_adk_mcp_sse_stream.py +48 -0
- aip_agents/examples/hello_world_google_adk_mcp_stdio.py +44 -0
- aip_agents/examples/hello_world_google_adk_mcp_stdio_stream.py +48 -0
- aip_agents/examples/hello_world_google_adk_stream.py +44 -0
- aip_agents/examples/hello_world_langchain.py +28 -0
- aip_agents/examples/hello_world_langchain_lm_invoker.py +15 -0
- aip_agents/examples/hello_world_langchain_mcp_http.py +34 -0
- aip_agents/examples/hello_world_langchain_mcp_http_interactive.py +130 -0
- aip_agents/examples/hello_world_langchain_mcp_http_stream.py +42 -0
- aip_agents/examples/hello_world_langchain_mcp_multi_server.py +155 -0
- aip_agents/examples/hello_world_langchain_mcp_sse.py +34 -0
- aip_agents/examples/hello_world_langchain_mcp_sse_stream.py +40 -0
- aip_agents/examples/hello_world_langchain_mcp_stdio.py +30 -0
- aip_agents/examples/hello_world_langchain_mcp_stdio_stream.py +41 -0
- aip_agents/examples/hello_world_langchain_stream.py +36 -0
- aip_agents/examples/hello_world_langchain_stream_lm_invoker.py +39 -0
- aip_agents/examples/hello_world_langflow_agent.py +163 -0
- aip_agents/examples/hello_world_langgraph.py +39 -0
- aip_agents/examples/hello_world_langgraph_bosa_twitter.py +41 -0
- aip_agents/examples/hello_world_langgraph_mcp_http.py +31 -0
- aip_agents/examples/hello_world_langgraph_mcp_http_stream.py +34 -0
- aip_agents/examples/hello_world_langgraph_mcp_sse.py +35 -0
- aip_agents/examples/hello_world_langgraph_mcp_sse_stream.py +50 -0
- aip_agents/examples/hello_world_langgraph_mcp_stdio.py +35 -0
- aip_agents/examples/hello_world_langgraph_mcp_stdio_stream.py +50 -0
- aip_agents/examples/hello_world_langgraph_stream.py +43 -0
- aip_agents/examples/hello_world_langgraph_stream_lm_invoker.py +37 -0
- aip_agents/examples/hello_world_model_switch_cli.py +210 -0
- aip_agents/examples/hello_world_multi_agent_adk.py +75 -0
- aip_agents/examples/hello_world_multi_agent_langchain.py +54 -0
- aip_agents/examples/hello_world_multi_agent_langgraph.py +66 -0
- aip_agents/examples/hello_world_multi_agent_langgraph_lm_invoker.py +69 -0
- aip_agents/examples/hello_world_pii_logger.py +21 -0
- aip_agents/examples/hello_world_sentry.py +133 -0
- aip_agents/examples/hello_world_step_limits.py +273 -0
- aip_agents/examples/hello_world_stock_a2a_server.py +103 -0
- aip_agents/examples/hello_world_tool_output_client.py +46 -0
- aip_agents/examples/hello_world_tool_output_server.py +114 -0
- aip_agents/examples/hitl_demo.py +724 -0
- aip_agents/examples/mcp_configs/configs.py +63 -0
- aip_agents/examples/mcp_servers/common.py +76 -0
- aip_agents/examples/mcp_servers/mcp_name.py +29 -0
- aip_agents/examples/mcp_servers/mcp_server_http.py +19 -0
- aip_agents/examples/mcp_servers/mcp_server_sse.py +19 -0
- aip_agents/examples/mcp_servers/mcp_server_stdio.py +19 -0
- aip_agents/examples/mcp_servers/mcp_time.py +10 -0
- aip_agents/examples/pii_demo_langgraph_client.py +69 -0
- aip_agents/examples/pii_demo_langgraph_server.py +126 -0
- aip_agents/examples/pii_demo_multi_agent_client.py +80 -0
- aip_agents/examples/pii_demo_multi_agent_server.py +247 -0
- aip_agents/examples/todolist_planning_a2a_langchain_client.py +70 -0
- aip_agents/examples/todolist_planning_a2a_langgraph_server.py +88 -0
- aip_agents/examples/tools/__init__.py +27 -0
- aip_agents/examples/tools/adk_arithmetic_tools.py +36 -0
- aip_agents/examples/tools/adk_weather_tool.py +60 -0
- aip_agents/examples/tools/data_generator_tool.py +103 -0
- aip_agents/examples/tools/data_visualization_tool.py +312 -0
- aip_agents/examples/tools/image_artifact_tool.py +136 -0
- aip_agents/examples/tools/langchain_arithmetic_tools.py +26 -0
- aip_agents/examples/tools/langchain_currency_exchange_tool.py +88 -0
- aip_agents/examples/tools/langchain_graph_artifact_tool.py +172 -0
- aip_agents/examples/tools/langchain_weather_tool.py +48 -0
- aip_agents/examples/tools/langgraph_streaming_tool.py +130 -0
- aip_agents/examples/tools/mock_retrieval_tool.py +56 -0
- aip_agents/examples/tools/pii_demo_tools.py +189 -0
- aip_agents/examples/tools/random_chart_tool.py +142 -0
- aip_agents/examples/tools/serper_tool.py +202 -0
- aip_agents/examples/tools/stock_tools.py +82 -0
- aip_agents/examples/tools/table_generator_tool.py +167 -0
- aip_agents/examples/tools/time_tool.py +82 -0
- aip_agents/examples/tools/weather_forecast_tool.py +38 -0
- aip_agents/executor/agent_executor.py +473 -0
- aip_agents/executor/base.py +48 -0
- aip_agents/mcp/__init__.py +1 -0
- aip_agents/mcp/client/__init__.py +14 -0
- aip_agents/mcp/client/base_mcp_client.py +369 -0
- aip_agents/mcp/client/connection_manager.py +193 -0
- aip_agents/mcp/client/google_adk/__init__.py +11 -0
- aip_agents/mcp/client/google_adk/client.py +381 -0
- aip_agents/mcp/client/langchain/__init__.py +11 -0
- aip_agents/mcp/client/langchain/client.py +265 -0
- aip_agents/mcp/client/persistent_session.py +359 -0
- aip_agents/mcp/client/session_pool.py +351 -0
- aip_agents/mcp/client/transports.py +215 -0
- aip_agents/mcp/utils/__init__.py +7 -0
- aip_agents/mcp/utils/config_validator.py +139 -0
- aip_agents/memory/__init__.py +14 -0
- aip_agents/memory/adapters/__init__.py +10 -0
- aip_agents/memory/adapters/base_adapter.py +717 -0
- aip_agents/memory/adapters/mem0.py +84 -0
- aip_agents/memory/base.py +84 -0
- aip_agents/memory/constants.py +49 -0
- aip_agents/memory/factory.py +86 -0
- aip_agents/memory/guidance.py +20 -0
- aip_agents/memory/simple_memory.py +47 -0
- aip_agents/middleware/__init__.py +17 -0
- aip_agents/middleware/base.py +88 -0
- aip_agents/middleware/manager.py +128 -0
- aip_agents/middleware/todolist.py +274 -0
- aip_agents/schema/__init__.py +69 -0
- aip_agents/schema/a2a.py +56 -0
- aip_agents/schema/agent.py +111 -0
- aip_agents/schema/hitl.py +157 -0
- aip_agents/schema/langgraph.py +37 -0
- aip_agents/schema/model_id.py +97 -0
- aip_agents/schema/step_limit.py +108 -0
- aip_agents/schema/storage.py +40 -0
- aip_agents/sentry/__init__.py +11 -0
- aip_agents/sentry/sentry.py +151 -0
- aip_agents/storage/__init__.py +41 -0
- aip_agents/storage/base.py +85 -0
- aip_agents/storage/clients/__init__.py +12 -0
- aip_agents/storage/clients/minio_client.py +318 -0
- aip_agents/storage/config.py +62 -0
- aip_agents/storage/providers/__init__.py +15 -0
- aip_agents/storage/providers/base.py +106 -0
- aip_agents/storage/providers/memory.py +114 -0
- aip_agents/storage/providers/object_storage.py +214 -0
- aip_agents/tools/__init__.py +33 -0
- aip_agents/tools/bosa_tools.py +105 -0
- aip_agents/tools/browser_use/__init__.py +82 -0
- aip_agents/tools/browser_use/action_parser.py +103 -0
- aip_agents/tools/browser_use/browser_use_tool.py +1112 -0
- aip_agents/tools/browser_use/llm_config.py +120 -0
- aip_agents/tools/browser_use/minio_storage.py +198 -0
- aip_agents/tools/browser_use/schemas.py +119 -0
- aip_agents/tools/browser_use/session.py +76 -0
- aip_agents/tools/browser_use/session_errors.py +132 -0
- aip_agents/tools/browser_use/steel_session_recording.py +317 -0
- aip_agents/tools/browser_use/streaming.py +813 -0
- aip_agents/tools/browser_use/structured_data_parser.py +257 -0
- aip_agents/tools/browser_use/structured_data_recovery.py +204 -0
- aip_agents/tools/browser_use/types.py +78 -0
- aip_agents/tools/code_sandbox/__init__.py +26 -0
- aip_agents/tools/code_sandbox/constant.py +13 -0
- aip_agents/tools/code_sandbox/e2b_cloud_sandbox_extended.py +257 -0
- aip_agents/tools/code_sandbox/e2b_sandbox_tool.py +411 -0
- aip_agents/tools/constants.py +165 -0
- aip_agents/tools/document_loader/__init__.py +44 -0
- aip_agents/tools/document_loader/base_reader.py +302 -0
- aip_agents/tools/document_loader/docx_reader_tool.py +68 -0
- aip_agents/tools/document_loader/excel_reader_tool.py +171 -0
- aip_agents/tools/document_loader/pdf_reader_tool.py +79 -0
- aip_agents/tools/document_loader/pdf_splitter.py +169 -0
- aip_agents/tools/gl_connector/__init__.py +5 -0
- aip_agents/tools/gl_connector/tool.py +351 -0
- aip_agents/tools/memory_search/__init__.py +22 -0
- aip_agents/tools/memory_search/base.py +200 -0
- aip_agents/tools/memory_search/mem0.py +258 -0
- aip_agents/tools/memory_search/schema.py +48 -0
- aip_agents/tools/memory_search_tool.py +26 -0
- aip_agents/tools/time_tool.py +117 -0
- aip_agents/tools/tool_config_injector.py +300 -0
- aip_agents/tools/web_search/__init__.py +15 -0
- aip_agents/tools/web_search/serper_tool.py +187 -0
- aip_agents/types/__init__.py +70 -0
- aip_agents/types/a2a_events.py +13 -0
- aip_agents/utils/__init__.py +79 -0
- aip_agents/utils/a2a_connector.py +1757 -0
- aip_agents/utils/artifact_helpers.py +502 -0
- aip_agents/utils/constants.py +22 -0
- aip_agents/utils/datetime/__init__.py +34 -0
- aip_agents/utils/datetime/normalization.py +231 -0
- aip_agents/utils/datetime/timezone.py +206 -0
- aip_agents/utils/env_loader.py +27 -0
- aip_agents/utils/event_handler_registry.py +58 -0
- aip_agents/utils/file_prompt_utils.py +176 -0
- aip_agents/utils/final_response_builder.py +211 -0
- aip_agents/utils/formatter_llm_client.py +231 -0
- aip_agents/utils/langgraph/__init__.py +19 -0
- aip_agents/utils/langgraph/converter.py +128 -0
- aip_agents/utils/langgraph/tool_managers/__init__.py +15 -0
- aip_agents/utils/langgraph/tool_managers/a2a_tool_manager.py +99 -0
- aip_agents/utils/langgraph/tool_managers/base_tool_manager.py +66 -0
- aip_agents/utils/langgraph/tool_managers/delegation_tool_manager.py +1071 -0
- aip_agents/utils/langgraph/tool_output_management.py +967 -0
- aip_agents/utils/logger.py +195 -0
- aip_agents/utils/metadata/__init__.py +27 -0
- aip_agents/utils/metadata/activity_metadata_helper.py +407 -0
- aip_agents/utils/metadata/activity_narrative/__init__.py +35 -0
- aip_agents/utils/metadata/activity_narrative/builder.py +817 -0
- aip_agents/utils/metadata/activity_narrative/constants.py +51 -0
- aip_agents/utils/metadata/activity_narrative/context.py +49 -0
- aip_agents/utils/metadata/activity_narrative/formatters.py +230 -0
- aip_agents/utils/metadata/activity_narrative/utils.py +35 -0
- aip_agents/utils/metadata/schemas/__init__.py +16 -0
- aip_agents/utils/metadata/schemas/activity_schema.py +29 -0
- aip_agents/utils/metadata/schemas/thinking_schema.py +31 -0
- aip_agents/utils/metadata/thinking_metadata_helper.py +38 -0
- aip_agents/utils/metadata_helper.py +358 -0
- aip_agents/utils/name_preprocessor/__init__.py +17 -0
- aip_agents/utils/name_preprocessor/base_name_preprocessor.py +73 -0
- aip_agents/utils/name_preprocessor/google_name_preprocessor.py +100 -0
- aip_agents/utils/name_preprocessor/name_preprocessor.py +87 -0
- aip_agents/utils/name_preprocessor/openai_name_preprocessor.py +48 -0
- aip_agents/utils/pii/__init__.py +25 -0
- aip_agents/utils/pii/pii_handler.py +397 -0
- aip_agents/utils/pii/pii_helper.py +207 -0
- aip_agents/utils/pii/uuid_deanonymizer_mapping.py +195 -0
- aip_agents/utils/reference_helper.py +273 -0
- aip_agents/utils/sse_chunk_transformer.py +831 -0
- aip_agents/utils/step_limit_manager.py +265 -0
- aip_agents/utils/token_usage_helper.py +156 -0
- aip_agents_binary-0.5.20.dist-info/METADATA +681 -0
- aip_agents_binary-0.5.20.dist-info/RECORD +280 -0
- aip_agents_binary-0.5.20.dist-info/WHEEL +5 -0
- 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
|