test(workspace): add 10-case coverage for enrich_peer_metadata
Covers all paths of the sync peer-metadata enrichment helper: - Cache hit within TTL (returns cached, no HTTP) - Cache expiry triggers refetch + cache update - Network exception → None + negative cache - HTTP non-200 → None + negative cache - Non-JSON body → None + negative cache - Non-dict JSON → None + negative cache - Invalid peer ID rejected at trust boundary (no HTTP) - Happy path returns data + caches + indexes peer name - GET URL and X-Workspace-ID header correctness - source_workspace_id parameter overrides default workspace header
This commit is contained in:
@@ -1061,3 +1061,238 @@ class TestGetWorkspaceInfo:
|
||||
|
||||
url = mock_client.get.call_args.args[0]
|
||||
assert "/workspaces/" in url
|
||||
|
||||
|
||||
# ---------------------------------------------------------------------------
|
||||
# enrich_peer_metadata — sync helper, separate from the async path.
|
||||
# ---------------------------------------------------------------------------
|
||||
|
||||
|
||||
def _make_sync_mock_client(*, get_resp=None, get_exc=None):
|
||||
"""Build a synchronous httpx.Client context-manager mock for enrich_peer_metadata."""
|
||||
mock_get = MagicMock()
|
||||
if get_exc is not None:
|
||||
mock_get.side_effect = get_exc
|
||||
elif get_resp is not None:
|
||||
mock_get.return_value = get_resp
|
||||
mock_client = MagicMock()
|
||||
mock_client.get = mock_get
|
||||
mock_client.__enter__ = MagicMock(return_value=mock_client)
|
||||
mock_client.__exit__ = MagicMock(return_value=False)
|
||||
return mock_client
|
||||
|
||||
|
||||
def _make_sync_response(status_code: int, data) -> MagicMock:
|
||||
"""Build a sync httpx.Response mock."""
|
||||
resp = MagicMock()
|
||||
resp.status_code = status_code
|
||||
resp.json = MagicMock(return_value=data)
|
||||
return resp
|
||||
|
||||
|
||||
class TestEnrichPeerMetadata:
|
||||
"""Tests for a2a_client.enrich_peer_metadata.
|
||||
|
||||
Uses the same test-ID constant and cache-isolation pattern as the
|
||||
async tests above.
|
||||
"""
|
||||
|
||||
def _call(self, peer_id, *, source_workspace_id=None, now=None):
|
||||
import a2a_client
|
||||
|
||||
return a2a_client.enrich_peer_metadata(
|
||||
peer_id,
|
||||
source_workspace_id=source_workspace_id,
|
||||
now=now,
|
||||
)
|
||||
|
||||
def test_cache_hit_within_ttl_returns_cached(self):
|
||||
"""Fresh cache entry → no HTTP call, returns the cached record."""
|
||||
import a2a_client
|
||||
|
||||
peer_data = {"id": _TEST_PEER_ID, "name": "Cached Peer", "url": "http://cached"}
|
||||
now = 1000.0
|
||||
# Seed cache with a fresh entry (TTL = 300s, so 1000+100 = 1100 < 1300).
|
||||
a2a_client._peer_metadata_set(_TEST_PEER_ID, (now, peer_data))
|
||||
|
||||
try:
|
||||
result = self._call(_TEST_PEER_ID, now=now + 100)
|
||||
assert result == peer_data
|
||||
finally:
|
||||
# Clean up so other tests are not polluted.
|
||||
a2a_client._peer_metadata.clear()
|
||||
a2a_client._peer_names.clear()
|
||||
|
||||
def test_cache_expired_causes_refetch(self):
|
||||
"""Stale cache entry (TTL exceeded) → HTTP GET issued, cache updated."""
|
||||
import a2a_client
|
||||
|
||||
old_data = {"id": _TEST_PEER_ID, "name": "Old"}
|
||||
fresh_data = {"id": _TEST_PEER_ID, "name": "Fresh", "url": "http://fresh"}
|
||||
now = 1000.0
|
||||
|
||||
# Seed cache with an expired entry (> 300s ago).
|
||||
a2a_client._peer_metadata_set(_TEST_PEER_ID, (now - 1000, old_data))
|
||||
resp = _make_sync_response(200, fresh_data)
|
||||
mock_client = _make_sync_mock_client(get_resp=resp)
|
||||
|
||||
with patch("a2a_client.httpx.Client", return_value=mock_client):
|
||||
result = self._call(_TEST_PEER_ID, now=now)
|
||||
|
||||
assert result == fresh_data
|
||||
# Cache should now hold the fresh data.
|
||||
cached = a2a_client._peer_metadata_get(_TEST_PEER_ID)
|
||||
assert cached is not None
|
||||
assert cached[1] == fresh_data
|
||||
a2a_client._peer_metadata.clear()
|
||||
a2a_client._peer_names.clear()
|
||||
|
||||
def test_network_exception_returns_none_negative_cache_set(self):
|
||||
"""Network failure → returns None, failure cached (negative cache)."""
|
||||
import a2a_client
|
||||
|
||||
now = 1000.0
|
||||
mock_client = _make_sync_mock_client(get_exc=ConnectionError("unreachable"))
|
||||
|
||||
with patch("a2a_client.httpx.Client", return_value=mock_client):
|
||||
result = self._call(_TEST_PEER_ID, now=now)
|
||||
|
||||
assert result is None
|
||||
# Negative cache: failure stored so we don't re-fetch on every call.
|
||||
cached = a2a_client._peer_metadata_get(_TEST_PEER_ID)
|
||||
assert cached is not None
|
||||
assert cached[1] is None # None sentinel = negative cache
|
||||
a2a_client._peer_metadata.clear()
|
||||
a2a_client._peer_names.clear()
|
||||
|
||||
def test_non_200_returns_none_negative_cache_set(self):
|
||||
"""HTTP 404/403/500 → returns None, failure cached."""
|
||||
import a2a_client
|
||||
|
||||
now = 1000.0
|
||||
resp = _make_sync_response(404, {"detail": "not found"})
|
||||
mock_client = _make_sync_mock_client(get_resp=resp)
|
||||
|
||||
with patch("a2a_client.httpx.Client", return_value=mock_client):
|
||||
result = self._call(_TEST_PEER_ID, now=now)
|
||||
|
||||
assert result is None
|
||||
cached = a2a_client._peer_metadata_get(_TEST_PEER_ID)
|
||||
assert cached is not None
|
||||
assert cached[1] is None
|
||||
a2a_client._peer_metadata.clear()
|
||||
a2a_client._peer_names.clear()
|
||||
|
||||
def test_non_json_response_returns_none_negative_cache_set(self):
|
||||
"""Server returns non-JSON body → returns None, failure cached."""
|
||||
import a2a_client
|
||||
|
||||
now = 1000.0
|
||||
resp = MagicMock()
|
||||
resp.status_code = 200
|
||||
resp.json.side_effect = ValueError("invalid json")
|
||||
mock_client = _make_sync_mock_client(get_resp=resp)
|
||||
|
||||
with patch("a2a_client.httpx.Client", return_value=mock_client):
|
||||
result = self._call(_TEST_PEER_ID, now=now)
|
||||
|
||||
assert result is None
|
||||
cached = a2a_client._peer_metadata_get(_TEST_PEER_ID)
|
||||
assert cached is not None
|
||||
assert cached[1] is None
|
||||
a2a_client._peer_metadata.clear()
|
||||
a2a_client._peer_names.clear()
|
||||
|
||||
def test_non_dict_json_returns_none_negative_cache_set(self):
|
||||
"""Server returns a JSON array or scalar → returns None, failure cached."""
|
||||
import a2a_client
|
||||
|
||||
now = 1000.0
|
||||
resp = _make_sync_response(200, ["peer-a", "peer-b"])
|
||||
mock_client = _make_sync_mock_client(get_resp=resp)
|
||||
|
||||
with patch("a2a_client.httpx.Client", return_value=mock_client):
|
||||
result = self._call(_TEST_PEER_ID, now=now)
|
||||
|
||||
assert result is None
|
||||
cached = a2a_client._peer_metadata_get(_TEST_PEER_ID)
|
||||
assert cached is not None
|
||||
assert cached[1] is None
|
||||
a2a_client._peer_metadata.clear()
|
||||
a2a_client._peer_names.clear()
|
||||
|
||||
def test_invalid_peer_id_returns_none_without_http(self):
|
||||
"""Path-traversal / malformed peer IDs are rejected at the trust boundary."""
|
||||
import a2a_client
|
||||
|
||||
mock_client = _make_sync_mock_client(get_resp=_make_sync_response(200, {}))
|
||||
with patch("a2a_client.httpx.Client", return_value=mock_client):
|
||||
for bad in ("", "ws-abc", "../admin", "not-a-uuid", "8dad3e29"):
|
||||
assert self._call(bad) is None
|
||||
# No GET should have been issued for any invalid ID.
|
||||
mock_client.get.assert_not_called()
|
||||
|
||||
def test_happy_path_returns_data_and_caches(self):
|
||||
"""200 + dict JSON → returns data, cache updated, peer name stored."""
|
||||
import a2a_client
|
||||
|
||||
now = 1000.0
|
||||
peer_data = {
|
||||
"id": _TEST_PEER_ID,
|
||||
"name": "Happy Peer",
|
||||
"role": "sre",
|
||||
"url": "http://happy-peer:8080",
|
||||
}
|
||||
resp = _make_sync_response(200, peer_data)
|
||||
mock_client = _make_sync_mock_client(get_resp=resp)
|
||||
|
||||
with patch("a2a_client.httpx.Client", return_value=mock_client):
|
||||
result = self._call(_TEST_PEER_ID, now=now)
|
||||
|
||||
assert result == peer_data
|
||||
# Cache updated.
|
||||
cached = a2a_client._peer_metadata_get(_TEST_PEER_ID)
|
||||
assert cached is not None
|
||||
assert cached[1] == peer_data
|
||||
# Peer name indexed.
|
||||
assert a2a_client._peer_names.get(_TEST_PEER_ID) == "Happy Peer"
|
||||
a2a_client._peer_metadata.clear()
|
||||
a2a_client._peer_names.clear()
|
||||
a2a_client._peer_names.clear()
|
||||
|
||||
def test_get_url_includes_peer_id_and_workspace_header(self):
|
||||
"""GET is issued to /registry/discover/<peer_id> with X-Workspace-ID."""
|
||||
import a2a_client
|
||||
|
||||
now = 1000.0
|
||||
resp = _make_sync_response(200, {"id": _TEST_PEER_ID})
|
||||
mock_client = _make_sync_mock_client(get_resp=resp)
|
||||
|
||||
with patch("a2a_client.httpx.Client", return_value=mock_client):
|
||||
self._call(_TEST_PEER_ID, now=now)
|
||||
|
||||
mock_client.get.assert_called_once()
|
||||
positional_url = mock_client.get.call_args.args[0]
|
||||
assert _TEST_PEER_ID in positional_url
|
||||
assert "/registry/discover/" in positional_url
|
||||
headers_sent = mock_client.get.call_args.kwargs.get("headers", {})
|
||||
assert "X-Workspace-ID" in headers_sent
|
||||
a2a_client._peer_metadata.clear()
|
||||
a2a_client._peer_names.clear()
|
||||
|
||||
def test_source_workspace_id_header_overrides_default(self):
|
||||
"""Caller can pass source_workspace_id to set X-Workspace-ID header."""
|
||||
import a2a_client
|
||||
|
||||
now = 1000.0
|
||||
src_id = "22222222-2222-2222-2222-222222222222"
|
||||
resp = _make_sync_response(200, {"id": _TEST_PEER_ID})
|
||||
mock_client = _make_sync_mock_client(get_resp=resp)
|
||||
|
||||
with patch("a2a_client.httpx.Client", return_value=mock_client):
|
||||
self._call(_TEST_PEER_ID, source_workspace_id=src_id, now=now)
|
||||
|
||||
headers_sent = mock_client.get.call_args.kwargs.get("headers", {})
|
||||
assert headers_sent.get("X-Workspace-ID") == src_id
|
||||
a2a_client._peer_metadata.clear()
|
||||
a2a_client._peer_names.clear()
|
||||
|
||||
Reference in New Issue
Block a user