hermes-agent/tests/gateway/test_pre_gateway_dispatch.py
Keira Voss 1ef1e4c669 feat(plugins): add pre_gateway_dispatch hook
Introduces a new plugin hook `pre_gateway_dispatch` fired once per
incoming MessageEvent in `_handle_message`, after the internal-event
guard but before the auth / pairing chain. Plugins may return a dict
to influence flow:

    {"action": "skip",    "reason": "..."}  -> drop (no reply)
    {"action": "rewrite", "text":   "..."}  -> replace event.text
    {"action": "allow"}  /  None             -> normal dispatch

Motivation: gateway-level message-flow patterns that don't fit cleanly
into any single adapter — e.g. listen-only group-chat windows (buffer
ambient messages, collapse on @mention), or human-handover silent
ingest (record messages while an owner handles the chat manually).
Today these require forking core; with this hook they can live in a
single profile-agnostic plugin.

Hook runs BEFORE auth so plugins can handle unauthorized senders
(e.g. customer-service handover ingest) without triggering the
pairing-code flow. Exceptions in plugin callbacks are caught and
logged; the first non-None action dict wins, remaining results are
ignored.

Includes:
- `VALID_HOOKS` entry + inline doc in `hermes_cli/plugins.py`
- Invocation block in `gateway/run.py::_handle_message`
- 5 new tests in `tests/gateway/test_pre_gateway_dispatch.py`
  (skip, rewrite, allow, exception safety, internal-event bypass)
- 2 additional tests in `tests/hermes_cli/test_plugins.py`
- Table entry in `website/docs/user-guide/features/plugins.md`

Made-with: Cursor
2026-04-24 03:02:03 -07:00

179 lines
5.6 KiB
Python

"""Tests for the pre_gateway_dispatch plugin hook.
The hook allows plugins to intercept incoming messages before auth and
agent dispatch. It runs in _handle_message and acts on returned action
dicts: {"action": "skip"|"rewrite"|"allow"}.
"""
from types import SimpleNamespace
from unittest.mock import AsyncMock, MagicMock
import pytest
from gateway.config import GatewayConfig, Platform, PlatformConfig
from gateway.platforms.base import MessageEvent
from gateway.session import SessionSource
def _clear_auth_env(monkeypatch) -> None:
for key in (
"TELEGRAM_ALLOWED_USERS",
"WHATSAPP_ALLOWED_USERS",
"GATEWAY_ALLOWED_USERS",
"TELEGRAM_ALLOW_ALL_USERS",
"WHATSAPP_ALLOW_ALL_USERS",
"GATEWAY_ALLOW_ALL_USERS",
):
monkeypatch.delenv(key, raising=False)
def _make_event(text: str = "hello", platform: Platform = Platform.WHATSAPP) -> MessageEvent:
return MessageEvent(
text=text,
message_id="m1",
source=SessionSource(
platform=platform,
user_id="15551234567@s.whatsapp.net",
chat_id="15551234567@s.whatsapp.net",
user_name="tester",
chat_type="dm",
),
)
def _make_runner(platform: Platform):
from gateway.run import GatewayRunner
config = GatewayConfig(
platforms={platform: PlatformConfig(enabled=True)},
)
runner = object.__new__(GatewayRunner)
runner.config = config
adapter = SimpleNamespace(send=AsyncMock())
runner.adapters = {platform: adapter}
runner.pairing_store = MagicMock()
runner.pairing_store.is_approved.return_value = False
runner.pairing_store._is_rate_limited.return_value = False
runner.session_store = MagicMock()
runner._running_agents = {}
runner._update_prompt_pending = {}
return runner, adapter
@pytest.mark.asyncio
async def test_hook_skip_short_circuits_dispatch(monkeypatch):
"""A plugin returning {'action': 'skip'} drops the message before auth."""
_clear_auth_env(monkeypatch)
def _fake_hook(name, **kwargs):
if name == "pre_gateway_dispatch":
return [{"action": "skip", "reason": "plugin-handled"}]
return []
monkeypatch.setattr("hermes_cli.plugins.invoke_hook", _fake_hook)
runner, adapter = _make_runner(Platform.WHATSAPP)
result = await runner._handle_message(_make_event("hi"))
assert result is None
adapter.send.assert_not_awaited()
runner.pairing_store.generate_code.assert_not_called()
@pytest.mark.asyncio
async def test_hook_rewrite_replaces_event_text(monkeypatch):
"""A plugin returning {'action': 'rewrite', 'text': ...} mutates event.text."""
_clear_auth_env(monkeypatch)
monkeypatch.setenv("WHATSAPP_ALLOWED_USERS", "*")
seen_text = {}
def _fake_hook(name, **kwargs):
if name == "pre_gateway_dispatch":
return [{"action": "rewrite", "text": "REWRITTEN"}]
return []
async def _capture(event, source, _quick_key, _run_generation):
seen_text["value"] = event.text
return "ok"
monkeypatch.setattr("hermes_cli.plugins.invoke_hook", _fake_hook)
runner, _adapter = _make_runner(Platform.WHATSAPP)
runner._handle_message_with_agent = _capture # noqa: SLF001
await runner._handle_message(_make_event("original"))
assert seen_text.get("value") == "REWRITTEN"
@pytest.mark.asyncio
async def test_hook_allow_falls_through_to_auth(monkeypatch):
"""A plugin returning {'action': 'allow'} continues to normal dispatch."""
_clear_auth_env(monkeypatch)
# No allowed users set → auth fails → pairing flow triggers.
monkeypatch.delenv("WHATSAPP_ALLOWED_USERS", raising=False)
def _fake_hook(name, **kwargs):
if name == "pre_gateway_dispatch":
return [{"action": "allow"}]
return []
monkeypatch.setattr("hermes_cli.plugins.invoke_hook", _fake_hook)
runner, adapter = _make_runner(Platform.WHATSAPP)
runner.pairing_store.generate_code.return_value = "12345"
result = await runner._handle_message(_make_event("hi"))
# auth chain ran → pairing code was generated
assert result is None
runner.pairing_store.generate_code.assert_called_once()
@pytest.mark.asyncio
async def test_hook_exception_does_not_break_dispatch(monkeypatch):
"""A raising plugin hook does not break the gateway."""
_clear_auth_env(monkeypatch)
monkeypatch.delenv("WHATSAPP_ALLOWED_USERS", raising=False)
def _fake_hook(name, **kwargs):
raise RuntimeError("plugin blew up")
monkeypatch.setattr("hermes_cli.plugins.invoke_hook", _fake_hook)
runner, _adapter = _make_runner(Platform.WHATSAPP)
runner.pairing_store.generate_code.return_value = None
# Should not raise; falls through to auth chain.
result = await runner._handle_message(_make_event("hi"))
assert result is None
@pytest.mark.asyncio
async def test_internal_events_bypass_hook(monkeypatch):
"""Internal events (event.internal=True) skip the plugin hook entirely."""
_clear_auth_env(monkeypatch)
monkeypatch.setenv("WHATSAPP_ALLOWED_USERS", "*")
called = {"count": 0}
def _fake_hook(name, **kwargs):
called["count"] += 1
return [{"action": "skip"}]
async def _capture(event, source, _quick_key, _run_generation):
return "ok"
monkeypatch.setattr("hermes_cli.plugins.invoke_hook", _fake_hook)
runner, _adapter = _make_runner(Platform.WHATSAPP)
runner._handle_message_with_agent = _capture # noqa: SLF001
event = _make_event("hi")
event.internal = True
# Even though the hook would say skip, internal events bypass it.
await runner._handle_message(event)
assert called["count"] == 0