mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-06-30 11:52:04 +00:00
A DM reply carries no guild_id, so the connector's egress guard cannot resolve the owning tenant from metadata.guild_id and declines the send with "discord egress declined: target not routed to an onboarded tenant" — the bug behind "the bot never replies in DMs". Guild replies are unaffected (they carry guild_id), which is why the guild path worked end-to-end while DMs looked broken. The connector now resolves a DM reply's tenant from the recipient's author binding (gateway-gateway #67, resolveByUser keyed on metadata.user_id) — the outbound counterpart to inbound Phase 7a author-first resolution. But it needs the recipient user_id ON the outbound action, and the adapter only re-attached guild_id (_capture_scope/_with_scope), no-op for DMs (the docstring even said so). This extends the adapter's inbound-scope capture: for a DM (no guild_id) remember chat_id -> the authentic author user_id we observed, and re-attach it as metadata.user_id on outbound. Guild capture is unchanged and wins when present; user_id is the DM-only fallback. The id is the one the connector observed inbound (never gateway-asserted), so the trust invariant holds. +4 unit tests (DM reply re-attaches user_id + no guild_id; unknown chat invents nothing; explicit user_id preserved; guild reply never carries user_id). Proved load-bearing (reverting the re-attach fails the DM test). 144 relay tests pass, ruff clean. Pairs with gateway-gateway #67 (the connector-side resolver). Together they close the DM-reply egress gap end-to-end.
266 lines
9.3 KiB
Python
266 lines
9.3 KiB
Python
"""RelayAdapter capability-advertisement tests (relay Phase 1, Task 1.1)."""
|
|
|
|
import pytest
|
|
|
|
from gateway.config import Platform, PlatformConfig
|
|
from gateway.relay.adapter import RelayAdapter
|
|
from gateway.relay.descriptor import CONTRACT_VERSION, CapabilityDescriptor
|
|
|
|
|
|
def make_desc(**kw) -> CapabilityDescriptor:
|
|
base = dict(
|
|
contract_version=CONTRACT_VERSION,
|
|
platform="telegram",
|
|
label="Telegram",
|
|
max_message_length=4096,
|
|
supports_draft_streaming=False,
|
|
supports_edit=True,
|
|
supports_threads=True,
|
|
markdown_dialect="markdown_v2",
|
|
len_unit="utf16",
|
|
emoji="\u2708\ufe0f",
|
|
platform_hint="",
|
|
pii_safe=False,
|
|
)
|
|
base.update(kw)
|
|
return CapabilityDescriptor(**base)
|
|
|
|
|
|
def _adapter(**desc_kw) -> RelayAdapter:
|
|
return RelayAdapter(PlatformConfig(), make_desc(**desc_kw))
|
|
|
|
|
|
def test_relay_platform_member_exists():
|
|
assert Platform("relay") is Platform.RELAY
|
|
|
|
|
|
def test_advertises_descriptor_max_length():
|
|
a = _adapter(max_message_length=2000)
|
|
assert a.MAX_MESSAGE_LENGTH == 2000
|
|
|
|
|
|
def test_supports_draft_streaming_follows_descriptor():
|
|
assert _adapter(supports_draft_streaming=False).supports_draft_streaming() is False
|
|
assert _adapter(supports_draft_streaming=True).supports_draft_streaming() is True
|
|
|
|
|
|
def test_len_fn_utf16_counts_code_units():
|
|
a = _adapter(len_unit="utf16")
|
|
# An astral-plane emoji is two UTF-16 code units.
|
|
assert a.message_len_fn("\U0001f600") == 2
|
|
|
|
|
|
def test_len_fn_chars_uses_builtin_len():
|
|
a = _adapter(len_unit="chars")
|
|
assert a.message_len_fn("\U0001f600") == 1
|
|
|
|
|
|
def test_is_a_base_platform_adapter():
|
|
# stream_consumer's isinstance(adapter, BasePlatformAdapter) guard must pass.
|
|
from gateway.platforms.base import BasePlatformAdapter
|
|
|
|
assert isinstance(_adapter(), BasePlatformAdapter)
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_connect_without_transport_raises():
|
|
a = _adapter()
|
|
with pytest.raises(RuntimeError, match="no transport"):
|
|
await a.connect()
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_send_without_transport_returns_failure():
|
|
a = _adapter()
|
|
result = await a.send("chat1", "hello")
|
|
assert result.success is False
|
|
assert result.error == "no transport"
|
|
|
|
|
|
class _CaptureTransport:
|
|
"""Minimal RelayTransport stand-in that records the outbound action."""
|
|
|
|
def __init__(self):
|
|
self.sent = None
|
|
|
|
def set_inbound_handler(self, h): # noqa: D401
|
|
self._h = h
|
|
|
|
async def send_outbound(self, action):
|
|
self.sent = action
|
|
return {"success": True, "message_id": "m1"}
|
|
|
|
|
|
def _make_event(chat_id="chan-1", guild_id="guild-9"):
|
|
from gateway.platforms.base import MessageEvent, MessageType
|
|
from gateway.session import SessionSource
|
|
|
|
src = SessionSource(
|
|
platform=Platform.RELAY,
|
|
chat_id=chat_id,
|
|
chat_type="channel",
|
|
guild_id=guild_id,
|
|
)
|
|
return MessageEvent(text="hi", source=src, message_type=MessageType.TEXT)
|
|
|
|
|
|
def _make_dm_event(chat_id="dm-1", user_id="user-42"):
|
|
"""An inbound DM: no guild_id, carries the authentic author user_id."""
|
|
from gateway.platforms.base import MessageEvent, MessageType
|
|
from gateway.session import SessionSource
|
|
|
|
src = SessionSource(
|
|
platform=Platform.RELAY,
|
|
chat_id=chat_id,
|
|
chat_type="dm",
|
|
guild_id=None,
|
|
user_id=user_id,
|
|
)
|
|
return MessageEvent(text="hi", source=src, message_type=MessageType.TEXT)
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_send_reattaches_guild_id_from_inbound_scope():
|
|
"""The connector's egress guard resolves the owning tenant from
|
|
metadata.guild_id; the gateway's generic delivery path drops it, so the
|
|
relay adapter must re-attach the guild scope learned from the inbound event.
|
|
Regression for live 'discord egress declined: target not routed to an
|
|
onboarded tenant'."""
|
|
t = _CaptureTransport()
|
|
a = RelayAdapter(PlatformConfig(), make_desc(platform="discord"), transport=t)
|
|
# Simulate the connector delivering an inbound message in guild-9 / chan-1,
|
|
# but don't run the full handle_message pipeline — just the scope capture.
|
|
a._capture_scope(_make_event(chat_id="chan-1", guild_id="guild-9"))
|
|
|
|
await a.send("chan-1", "the reply")
|
|
|
|
assert t.sent["metadata"].get("guild_id") == "guild-9"
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_send_without_known_scope_omits_guild_id():
|
|
"""A chat we never saw inbound (e.g. a DM) gets no guild_id — no-op, never
|
|
invents a scope."""
|
|
t = _CaptureTransport()
|
|
a = RelayAdapter(PlatformConfig(), make_desc(platform="discord"), transport=t)
|
|
await a.send("unknown-chat", "hi")
|
|
assert "guild_id" not in t.sent["metadata"]
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_send_preserves_explicit_guild_id():
|
|
"""An explicitly-provided metadata.guild_id is never overwritten."""
|
|
t = _CaptureTransport()
|
|
a = RelayAdapter(PlatformConfig(), make_desc(platform="discord"), transport=t)
|
|
a._capture_scope(_make_event(chat_id="chan-1", guild_id="guild-9"))
|
|
await a.send("chan-1", "hi", metadata={"guild_id": "explicit-1"})
|
|
assert t.sent["metadata"]["guild_id"] == "explicit-1"
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_send_reattaches_dm_user_id_from_inbound_scope():
|
|
"""A DM reply has no guild_id, so the connector resolves the tenant from the
|
|
recipient's author binding — it needs metadata.user_id. The adapter must
|
|
re-attach the authentic author id learned from the inbound DM. Regression for
|
|
live 'discord egress declined: target not routed to an onboarded tenant' on
|
|
DM replies (the connector-side fix is gateway-gateway #67)."""
|
|
t = _CaptureTransport()
|
|
a = RelayAdapter(PlatformConfig(), make_desc(platform="discord"), transport=t)
|
|
a._capture_scope(_make_dm_event(chat_id="dm-1", user_id="user-42"))
|
|
|
|
await a.send("dm-1", "the reply")
|
|
|
|
assert t.sent["metadata"].get("user_id") == "user-42"
|
|
# A DM carries no guild_id — only the author discriminator.
|
|
assert "guild_id" not in t.sent["metadata"]
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_send_dm_does_not_invent_user_id_for_unknown_chat():
|
|
"""A chat we never saw inbound gets neither discriminator — no-op."""
|
|
t = _CaptureTransport()
|
|
a = RelayAdapter(PlatformConfig(), make_desc(platform="discord"), transport=t)
|
|
await a.send("unknown-dm", "hi")
|
|
assert "user_id" not in t.sent["metadata"]
|
|
assert "guild_id" not in t.sent["metadata"]
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_send_preserves_explicit_user_id():
|
|
"""An explicitly-provided metadata.user_id is never overwritten."""
|
|
t = _CaptureTransport()
|
|
a = RelayAdapter(PlatformConfig(), make_desc(platform="discord"), transport=t)
|
|
a._capture_scope(_make_dm_event(chat_id="dm-1", user_id="user-42"))
|
|
await a.send("dm-1", "hi", metadata={"user_id": "explicit-user"})
|
|
assert t.sent["metadata"]["user_id"] == "explicit-user"
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_guild_reply_does_not_carry_user_id():
|
|
"""A guild reply resolves by guild_id and must NOT carry a DM user_id even if
|
|
the same chat_id was somehow seen — guild capture wins and user_id stays out
|
|
(guild_id is the discriminator; user_id is the DM-only fallback)."""
|
|
t = _CaptureTransport()
|
|
a = RelayAdapter(PlatformConfig(), make_desc(platform="discord"), transport=t)
|
|
a._capture_scope(_make_event(chat_id="chan-1", guild_id="guild-9"))
|
|
await a.send("chan-1", "hi")
|
|
assert t.sent["metadata"].get("guild_id") == "guild-9"
|
|
assert "user_id" not in t.sent["metadata"]
|
|
|
|
|
|
# ── Phase 7 Unit 7d-B: terminal auth revocation → clean "relay disabled" ─────
|
|
|
|
|
|
class _RevokedTransport:
|
|
"""Transport stand-in that reports a terminal auth revocation (the
|
|
production WebSocketRelayTransport latches this after a 4401 close that
|
|
follows a successful handshake)."""
|
|
|
|
def __init__(self):
|
|
self.auth_revoked = True
|
|
|
|
def set_inbound_handler(self, h): # noqa: D401
|
|
self._h = h
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_revocation_marks_relay_disabled_non_retryable():
|
|
"""When the transport reports auth_revoked, the adapter surfaces a clean,
|
|
NON-retryable `relay_disabled` fatal and fires the fatal-error handler."""
|
|
a = RelayAdapter(PlatformConfig(), make_desc(platform="discord"), transport=_RevokedTransport())
|
|
notified = []
|
|
a.set_fatal_error_handler(lambda adapter: notified.append(adapter))
|
|
|
|
# Drive the monitor body directly (poll loop breaks immediately on the
|
|
# already-revoked transport).
|
|
await a._watch_for_revocation(poll_interval_s=0.01)
|
|
|
|
assert a.has_fatal_error is True
|
|
assert a.fatal_error_code == "relay_disabled"
|
|
assert a.fatal_error_retryable is False
|
|
assert "disabled" in (a.fatal_error_message or "").lower()
|
|
assert notified == [a]
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_no_revocation_no_fatal():
|
|
"""A transport that has NOT been revoked never trips the disabled fatal."""
|
|
|
|
class _LiveTransport:
|
|
auth_revoked = False
|
|
|
|
def set_inbound_handler(self, h): # noqa: D401
|
|
self._h = h
|
|
|
|
a = RelayAdapter(PlatformConfig(), make_desc(platform="discord"), transport=_LiveTransport())
|
|
# Run the monitor with a tiny window then cancel — it should never fire.
|
|
import asyncio
|
|
|
|
task = asyncio.create_task(a._watch_for_revocation(poll_interval_s=0.01))
|
|
await asyncio.sleep(0.05)
|
|
task.cancel()
|
|
try:
|
|
await task
|
|
except asyncio.CancelledError:
|
|
pass
|
|
assert a.has_fatal_error is False
|