fix(mcp): wire source_workspace_id through dispatcher for memory + chat_history + workspace_info
Self-review of merged PR #2766 (multi-workspace MCP routing) revealed a silent gap: PR #2766 added the ``source_workspace_id`` parameter to ``tool_commit_memory`` / ``tool_recall_memory`` / ``tool_chat_history`` / ``tool_get_workspace_info`` AND advertised it in the registry's input schemas, but the MCP server's dispatch arms in ``a2a_mcp_server.py`` were never updated to forward ``arguments["source_workspace_id"]`` to those four tools. Result: the schema lied. The LLM saw ``source_workspace_id`` as a valid tool parameter, could correctly populate it from the inbound message's ``arrival_workspace_id``, but the dispatcher dropped it on the floor and every memory commit / recall / chat-history fetch silently fell back to the module-level ``WORKSPACE_ID``. The cross-tenant leak that PR #2766 was meant to prevent is NOT prevented for these four tools without this follow-up. Why the existing dispatcher tests didn't catch it: the tests asserted return-value strings (``"working" in result``) but never asserted what arguments the inner tool was called with. So the dispatcher could ignore any kwarg and the tests would still pass. Fix: 1. Wire ``source_workspace_id=arguments.get("source_workspace_id") or None`` into the four dispatch arms, mirroring the pattern already used for ``delegate_task`` / ``delegate_task_async`` / ``check_task_status`` / ``list_peers``. 2. Add five tests in ``test_a2a_mcp_server.py`` that assert the inner tool was awaited with the exact source_workspace_id kwarg (``assert_awaited_once_with(..., source_workspace_id="ws-X")``) — substring-on-result tests can't catch this class of bug. 3. Add a fallback test ensuring single-workspace operators (no source_workspace_id key) get ``source_workspace_id=None`` — pinning the documented None contract over an accidental empty-string forward. Suite: 1705 passed (was 1700 + 5 new), 3 skipped, 2 xfailed. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
parent
02960209a0
commit
41ae4ec50b
@ -123,16 +123,20 @@ async def handle_tool_call(name: str, arguments: dict) -> str:
|
||||
source_workspace_id=arguments.get("source_workspace_id") or None,
|
||||
)
|
||||
elif name == "get_workspace_info":
|
||||
return await tool_get_workspace_info()
|
||||
return await tool_get_workspace_info(
|
||||
source_workspace_id=arguments.get("source_workspace_id") or None,
|
||||
)
|
||||
elif name == "commit_memory":
|
||||
return await tool_commit_memory(
|
||||
arguments.get("content", ""),
|
||||
arguments.get("scope", "LOCAL"),
|
||||
source_workspace_id=arguments.get("source_workspace_id") or None,
|
||||
)
|
||||
elif name == "recall_memory":
|
||||
return await tool_recall_memory(
|
||||
arguments.get("query", ""),
|
||||
arguments.get("scope", ""),
|
||||
source_workspace_id=arguments.get("source_workspace_id") or None,
|
||||
)
|
||||
elif name == "wait_for_message":
|
||||
return await tool_wait_for_message(
|
||||
@ -151,6 +155,7 @@ async def handle_tool_call(name: str, arguments: dict) -> str:
|
||||
arguments.get("peer_id", ""),
|
||||
arguments.get("limit", 20),
|
||||
arguments.get("before_ts", ""),
|
||||
source_workspace_id=arguments.get("source_workspace_id") or None,
|
||||
)
|
||||
return f"Unknown tool: {name}"
|
||||
|
||||
|
||||
@ -71,6 +71,105 @@ async def test_handle_tool_call_unknown_tool():
|
||||
assert "Unknown tool" in result
|
||||
|
||||
|
||||
# ---------------------------------------------------------------------------
|
||||
# source_workspace_id propagation — every workspace-scoped tool's schema
|
||||
# advertises this parameter (PR #2766) so the LLM can route a memory commit
|
||||
# or chat-history query through the workspace the inbound message arrived
|
||||
# on. The dispatch path itself MUST forward the kwarg — otherwise the
|
||||
# schema lies and every call silently falls back to the module-level
|
||||
# WORKSPACE_ID, defeating multi-workspace isolation. These tests pin
|
||||
# end-to-end argument flow on the four tools that ship in PR #2766.
|
||||
# ---------------------------------------------------------------------------
|
||||
|
||||
|
||||
async def test_dispatch_get_workspace_info_forwards_source_workspace_id():
|
||||
from a2a_mcp_server import handle_tool_call
|
||||
mock = AsyncMock(return_value='{"id":"ws-X"}')
|
||||
with patch("a2a_mcp_server.tool_get_workspace_info", new=mock):
|
||||
await handle_tool_call(
|
||||
"get_workspace_info",
|
||||
{"source_workspace_id": "ws-X"},
|
||||
)
|
||||
mock.assert_awaited_once_with(source_workspace_id="ws-X")
|
||||
|
||||
|
||||
async def test_dispatch_commit_memory_forwards_source_workspace_id():
|
||||
from a2a_mcp_server import handle_tool_call
|
||||
mock = AsyncMock(return_value='{"success":true}')
|
||||
with patch("a2a_mcp_server.tool_commit_memory", new=mock):
|
||||
await handle_tool_call(
|
||||
"commit_memory",
|
||||
{
|
||||
"content": "remember this",
|
||||
"scope": "LOCAL",
|
||||
"source_workspace_id": "ws-Y",
|
||||
},
|
||||
)
|
||||
mock.assert_awaited_once_with(
|
||||
"remember this",
|
||||
"LOCAL",
|
||||
source_workspace_id="ws-Y",
|
||||
)
|
||||
|
||||
|
||||
async def test_dispatch_recall_memory_forwards_source_workspace_id():
|
||||
from a2a_mcp_server import handle_tool_call
|
||||
mock = AsyncMock(return_value="[LOCAL] remember this")
|
||||
with patch("a2a_mcp_server.tool_recall_memory", new=mock):
|
||||
await handle_tool_call(
|
||||
"recall_memory",
|
||||
{
|
||||
"query": "remember",
|
||||
"scope": "LOCAL",
|
||||
"source_workspace_id": "ws-Z",
|
||||
},
|
||||
)
|
||||
mock.assert_awaited_once_with(
|
||||
"remember",
|
||||
"LOCAL",
|
||||
source_workspace_id="ws-Z",
|
||||
)
|
||||
|
||||
|
||||
async def test_dispatch_chat_history_forwards_source_workspace_id():
|
||||
from a2a_mcp_server import handle_tool_call
|
||||
mock = AsyncMock(return_value="[]")
|
||||
with patch("a2a_mcp_server.tool_chat_history", new=mock):
|
||||
await handle_tool_call(
|
||||
"chat_history",
|
||||
{
|
||||
"peer_id": "peer-A",
|
||||
"limit": 10,
|
||||
"source_workspace_id": "ws-W",
|
||||
},
|
||||
)
|
||||
mock.assert_awaited_once_with(
|
||||
"peer-A",
|
||||
10,
|
||||
"",
|
||||
source_workspace_id="ws-W",
|
||||
)
|
||||
|
||||
|
||||
async def test_dispatch_omits_source_workspace_id_when_unset():
|
||||
"""Single-workspace operators (no source_workspace_id key in args) must
|
||||
forward None — preserving the legacy fallback to module-level WORKSPACE_ID
|
||||
inside the tool. An accidental empty-string forward would also fall back,
|
||||
but None is the documented contract."""
|
||||
from a2a_mcp_server import handle_tool_call
|
||||
mock = AsyncMock(return_value='{"success":true}')
|
||||
with patch("a2a_mcp_server.tool_commit_memory", new=mock):
|
||||
await handle_tool_call(
|
||||
"commit_memory",
|
||||
{"content": "x", "scope": "LOCAL"},
|
||||
)
|
||||
mock.assert_awaited_once_with(
|
||||
"x",
|
||||
"LOCAL",
|
||||
source_workspace_id=None,
|
||||
)
|
||||
|
||||
|
||||
async def test_handle_tool_call_missing_args_defaults():
|
||||
"""Test that missing args default to empty strings (defensive)."""
|
||||
from a2a_mcp_server import handle_tool_call
|
||||
|
||||
Loading…
Reference in New Issue
Block a user