mirror of
https://github.com/bytedance/deer-flow.git
synced 2026-05-22 16:06:50 +00:00
fix(harness): wrap all async-only tools for sync clients (#2935)
This commit is contained in:
@@ -1,7 +1,9 @@
|
||||
import asyncio
|
||||
import contextvars
|
||||
from unittest.mock import AsyncMock, MagicMock, patch
|
||||
|
||||
import pytest
|
||||
from langchain_core.runnables import RunnableConfig
|
||||
from langchain_core.tools import StructuredTool
|
||||
from pydantic import BaseModel, Field
|
||||
|
||||
@@ -69,6 +71,58 @@ def test_mcp_tool_sync_wrapper_in_running_loop():
|
||||
assert result == "async_result: 100"
|
||||
|
||||
|
||||
def test_sync_wrapper_preserves_contextvars_in_running_loop():
|
||||
"""The executor branch preserves LangGraph-style contextvars."""
|
||||
current_value: contextvars.ContextVar[str | None] = contextvars.ContextVar("current_value", default=None)
|
||||
|
||||
async def mock_coro() -> str | None:
|
||||
return current_value.get()
|
||||
|
||||
sync_func = make_sync_tool_wrapper(mock_coro, "test_tool")
|
||||
|
||||
async def run_in_loop() -> str | None:
|
||||
token = current_value.set("from-parent-context")
|
||||
try:
|
||||
return sync_func()
|
||||
finally:
|
||||
current_value.reset(token)
|
||||
|
||||
assert asyncio.run(run_in_loop()) == "from-parent-context"
|
||||
|
||||
|
||||
def test_sync_wrapper_preserves_runnable_config_injection():
|
||||
"""LangChain can still inject RunnableConfig after an async tool is wrapped."""
|
||||
captured: dict[str, object] = {}
|
||||
|
||||
async def mock_coro(x: int, config: RunnableConfig = None):
|
||||
captured["thread_id"] = ((config or {}).get("configurable") or {}).get("thread_id")
|
||||
return f"result: {x}"
|
||||
|
||||
mock_tool = StructuredTool(
|
||||
name="test_tool",
|
||||
description="test description",
|
||||
args_schema=MockArgs,
|
||||
func=make_sync_tool_wrapper(mock_coro, "test_tool"),
|
||||
coroutine=mock_coro,
|
||||
)
|
||||
|
||||
result = mock_tool.invoke({"x": 42}, config={"configurable": {"thread_id": "thread-123"}})
|
||||
|
||||
assert result == "result: 42"
|
||||
assert captured["thread_id"] == "thread-123"
|
||||
|
||||
|
||||
def test_sync_wrapper_preserves_regular_config_argument():
|
||||
"""Only RunnableConfig-annotated coroutine params get special config injection."""
|
||||
|
||||
async def mock_coro(config: str):
|
||||
return config
|
||||
|
||||
sync_func = make_sync_tool_wrapper(mock_coro, "test_tool")
|
||||
|
||||
assert sync_func(config="user-config") == "user-config"
|
||||
|
||||
|
||||
def test_mcp_tool_sync_wrapper_exception_logging():
|
||||
"""Test the shared sync wrapper's error logging."""
|
||||
|
||||
|
||||
Reference in New Issue
Block a user