mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-06-09 08:21:50 +00:00
Salvage of PR #27978 cherry-picked onto current main, resolving conflicts with main's intervening SimpleX plugin fixes (resp-envelope normalization, health-monitor reconnect-churn fix, bare-form DM addressing). What's new: - Group support via SIMPLEX_GROUP_ALLOWED (comma-separated IDs or '*'); inbound items surface chat_id=group:<id> + chat_type=group. Disabled by default so a bot in a group doesn't process every member's traffic. - Inbound files/voice via rcvFileDescrReady (immediate /freceive) deferred through _pending_file_transfers, replayed on rcvFileComplete. Voice notes -> MessageType.VOICE. - Native outbound media: send_image (PNG/JPEG + inline thumbnail), send_voice (msgContent.type=voice), send_video, send_document. All addressed by numeric ID via /_send ... json [...]. - MEDIA:<path> tags in agent replies stripped and dispatched as voice/document. - Text-burst batching (HERMES_SIMPLEX_TEXT_BATCH_DELAY, default 0.8s). - Auto-accept contact requests (SIMPLEX_AUTO_ACCEPT, default true). - Group send path uses structured /_send #<id> json form (the bracket #[<id>] form is parsed as display-name lookup and silently drops). plugin.yaml bumped to 1.1.0; docs updated. All inside plugins/platforms/simplex/ - no core edits. Co-authored-by: Juraj Bednar <juraj@bednar.io>
398 lines
14 KiB
Python
398 lines
14 KiB
Python
"""Tests for the SimpleX Chat platform-plugin adapter.
|
|
|
|
Loaded via the ``_plugin_adapter_loader`` helper so this lives under
|
|
``plugin_adapter_simplex`` in ``sys.modules`` and cannot collide with
|
|
sibling platform-plugin tests on the same xdist worker.
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
import asyncio
|
|
import json
|
|
from unittest.mock import AsyncMock, MagicMock
|
|
|
|
import pytest
|
|
|
|
from tests.gateway._plugin_adapter_loader import load_plugin_adapter
|
|
|
|
_simplex = load_plugin_adapter("simplex")
|
|
|
|
SimplexAdapter = _simplex.SimplexAdapter
|
|
check_requirements = _simplex.check_requirements
|
|
validate_config = _simplex.validate_config
|
|
is_connected = _simplex.is_connected
|
|
register = _simplex.register
|
|
_env_enablement = _simplex._env_enablement
|
|
_standalone_send = _simplex._standalone_send
|
|
_guess_extension = _simplex._guess_extension
|
|
_is_image_ext = _simplex._is_image_ext
|
|
_is_audio_ext = _simplex._is_audio_ext
|
|
_CORR_PREFIX = _simplex._CORR_PREFIX
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# 1. Platform enum (plugin-discovered, not bundled)
|
|
# ---------------------------------------------------------------------------
|
|
|
|
def test_platform_enum_resolves_via_plugin_scan():
|
|
"""The plugin filesystem scan should expose Platform("simplex")."""
|
|
from gateway.config import Platform
|
|
p = Platform("simplex")
|
|
assert p.value == "simplex"
|
|
# Identity stability — repeated lookups return the same pseudo-member
|
|
assert Platform("simplex") is p
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# 2. check_requirements / validate_config / is_connected
|
|
# ---------------------------------------------------------------------------
|
|
|
|
def test_check_requirements_needs_url(monkeypatch):
|
|
monkeypatch.delenv("SIMPLEX_WS_URL", raising=False)
|
|
assert check_requirements() is False
|
|
|
|
|
|
def test_check_requirements_true_when_configured(monkeypatch):
|
|
monkeypatch.setenv("SIMPLEX_WS_URL", "ws://127.0.0.1:5225")
|
|
# websockets is a dev dep in this repo via the test plugins; the
|
|
# check_requirements() gate also asserts the package imports.
|
|
websockets_present = True
|
|
try:
|
|
import websockets # noqa: F401
|
|
except ImportError:
|
|
websockets_present = False
|
|
assert check_requirements() is websockets_present
|
|
|
|
|
|
def test_validate_config_uses_env_or_extra():
|
|
from gateway.config import PlatformConfig
|
|
# Empty extra + no env → invalid
|
|
cfg = PlatformConfig(enabled=True)
|
|
assert validate_config(cfg) is False
|
|
# extra-only path → valid
|
|
cfg2 = PlatformConfig(enabled=True, extra={"ws_url": "ws://localhost:5225"})
|
|
assert validate_config(cfg2) is True
|
|
|
|
|
|
def test_is_connected_mirrors_validate(monkeypatch):
|
|
from gateway.config import PlatformConfig
|
|
monkeypatch.delenv("SIMPLEX_WS_URL", raising=False)
|
|
cfg = PlatformConfig(enabled=True, extra={"ws_url": "ws://x"})
|
|
assert is_connected(cfg) is True
|
|
assert is_connected(PlatformConfig(enabled=True)) is False
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# 3. _env_enablement seeds PlatformConfig.extra
|
|
# ---------------------------------------------------------------------------
|
|
|
|
def test_env_enablement_none_when_unset(monkeypatch):
|
|
monkeypatch.delenv("SIMPLEX_WS_URL", raising=False)
|
|
assert _env_enablement() is None
|
|
|
|
|
|
def test_env_enablement_seeds_ws_url(monkeypatch):
|
|
monkeypatch.setenv("SIMPLEX_WS_URL", "ws://127.0.0.1:5225")
|
|
monkeypatch.delenv("SIMPLEX_HOME_CHANNEL", raising=False)
|
|
seed = _env_enablement()
|
|
assert seed == {"ws_url": "ws://127.0.0.1:5225"}
|
|
|
|
|
|
def test_env_enablement_seeds_home_channel(monkeypatch):
|
|
monkeypatch.setenv("SIMPLEX_WS_URL", "ws://127.0.0.1:5225")
|
|
monkeypatch.setenv("SIMPLEX_HOME_CHANNEL", "42")
|
|
monkeypatch.setenv("SIMPLEX_HOME_CHANNEL_NAME", "Personal")
|
|
seed = _env_enablement()
|
|
assert seed["home_channel"] == {"chat_id": "42", "name": "Personal"}
|
|
|
|
|
|
def test_env_enablement_home_channel_defaults_name_to_id(monkeypatch):
|
|
monkeypatch.setenv("SIMPLEX_WS_URL", "ws://127.0.0.1:5225")
|
|
monkeypatch.setenv("SIMPLEX_HOME_CHANNEL", "42")
|
|
monkeypatch.delenv("SIMPLEX_HOME_CHANNEL_NAME", raising=False)
|
|
seed = _env_enablement()
|
|
assert seed["home_channel"] == {"chat_id": "42", "name": "42"}
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# 4. Adapter init
|
|
# ---------------------------------------------------------------------------
|
|
|
|
def test_adapter_init_custom_url():
|
|
from gateway.config import PlatformConfig
|
|
cfg = PlatformConfig(enabled=True, extra={"ws_url": "ws://localhost:5225"})
|
|
adapter = SimplexAdapter(cfg)
|
|
assert adapter.ws_url == "ws://localhost:5225"
|
|
assert adapter._running is False
|
|
assert adapter._ws is None
|
|
|
|
|
|
def test_adapter_init_default_url():
|
|
from gateway.config import PlatformConfig
|
|
cfg = PlatformConfig(enabled=True)
|
|
adapter = SimplexAdapter(cfg)
|
|
assert adapter.ws_url == "ws://127.0.0.1:5225"
|
|
|
|
|
|
def test_adapter_platform_identity():
|
|
"""Adapter should expose Platform("simplex") identity."""
|
|
from gateway.config import Platform, PlatformConfig
|
|
cfg = PlatformConfig(enabled=True)
|
|
adapter = SimplexAdapter(cfg)
|
|
assert adapter.platform is Platform("simplex")
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# 5. Helper functions (magic-byte detection)
|
|
# ---------------------------------------------------------------------------
|
|
|
|
def test_guess_extension_png():
|
|
assert _guess_extension(b"\x89PNG\r\n\x1a\n") == ".png"
|
|
|
|
|
|
def test_guess_extension_jpg():
|
|
assert _guess_extension(b"\xff\xd8\xff\xe0") == ".jpg"
|
|
|
|
|
|
def test_guess_extension_ogg():
|
|
assert _guess_extension(b"OggS\x00\x02") == ".ogg"
|
|
|
|
|
|
def test_guess_extension_unknown():
|
|
assert _guess_extension(b"\x00\x01\x02\x03") == ".bin"
|
|
|
|
|
|
def test_is_image_ext():
|
|
assert _is_image_ext(".png") is True
|
|
assert _is_image_ext(".webp") is True
|
|
assert _is_image_ext(".ogg") is False
|
|
|
|
|
|
def test_is_audio_ext():
|
|
assert _is_audio_ext(".ogg") is True
|
|
assert _is_audio_ext(".mp3") is True
|
|
assert _is_audio_ext(".pdf") is False
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# 6. Correlation IDs
|
|
# ---------------------------------------------------------------------------
|
|
|
|
def test_corr_id_starts_with_prefix_and_tracks_pending():
|
|
from gateway.config import PlatformConfig
|
|
cfg = PlatformConfig(enabled=True, extra={"ws_url": "ws://localhost:5225"})
|
|
adapter = SimplexAdapter(cfg)
|
|
corr_id = adapter._make_corr_id()
|
|
assert corr_id.startswith(_CORR_PREFIX)
|
|
assert corr_id in adapter._pending_corr_ids
|
|
|
|
|
|
def test_corr_id_pending_set_self_trims():
|
|
from gateway.config import PlatformConfig
|
|
cfg = PlatformConfig(enabled=True, extra={"ws_url": "ws://localhost:5225"})
|
|
adapter = SimplexAdapter(cfg)
|
|
adapter._max_pending_corr = 4
|
|
for _ in range(10):
|
|
adapter._make_corr_id()
|
|
# After many additions, the pending set should be bounded by the trim
|
|
# logic — at most one trim window above the cap.
|
|
assert len(adapter._pending_corr_ids) <= adapter._max_pending_corr + 1
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# 7. Outbound send (mocked WS)
|
|
# ---------------------------------------------------------------------------
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_send_dm():
|
|
"""DMs use the bare ``@<id> text`` chat-command form.
|
|
|
|
The bracketed form ``@[<id>] text`` is what the daemon's man page
|
|
documents, but in practice both addressing styles route through
|
|
the same chat-command parser; bare ``@<id>`` matches what every
|
|
Hermes deployment has been using in production for months.
|
|
"""
|
|
from gateway.config import PlatformConfig
|
|
cfg = PlatformConfig(enabled=True, extra={"ws_url": "ws://localhost:5225"})
|
|
adapter = SimplexAdapter(cfg)
|
|
|
|
mock_ws = AsyncMock()
|
|
adapter._ws = mock_ws
|
|
|
|
result = await adapter.send("contact-42", "Hello, SimpleX!")
|
|
mock_ws.send.assert_called_once()
|
|
payload = json.loads(mock_ws.send.call_args[0][0])
|
|
assert payload["cmd"] == "@contact-42 Hello, SimpleX!"
|
|
assert payload["corrId"].startswith(_CORR_PREFIX)
|
|
assert result.success is True
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_send_group():
|
|
"""Groups use the structured ``/_send #<id> json [...]`` form.
|
|
|
|
The bracket chat-command form ``#[<id>] text`` *looks* like an exact
|
|
ID match in the daemon docs but is parsed as a display-name lookup
|
|
— so messages to groups whose display name isn't literally the ID
|
|
silently drop. The structured ``/_send`` form addresses by numeric
|
|
ID and survives newlines/quoting through ``json.dumps``.
|
|
"""
|
|
from gateway.config import PlatformConfig
|
|
cfg = PlatformConfig(enabled=True, extra={"ws_url": "ws://localhost:5225"})
|
|
adapter = SimplexAdapter(cfg)
|
|
|
|
mock_ws = AsyncMock()
|
|
adapter._ws = mock_ws
|
|
|
|
result = await adapter.send("group:grp-99", "Hello, group!")
|
|
payload = json.loads(mock_ws.send.call_args[0][0])
|
|
assert payload["cmd"].startswith("/_send #grp-99 json ")
|
|
msg_content = json.loads(payload["cmd"].split(" json ", 1)[1])[0][
|
|
"msgContent"
|
|
]
|
|
assert msg_content == {"type": "text", "text": "Hello, group!"}
|
|
assert result.success is True
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_send_when_ws_not_connected_does_not_crash():
|
|
from gateway.config import PlatformConfig
|
|
cfg = PlatformConfig(enabled=True, extra={"ws_url": "ws://localhost:5225"})
|
|
adapter = SimplexAdapter(cfg)
|
|
# No _ws assigned — _send_ws should drop quietly
|
|
result = await adapter.send("contact-42", "hi")
|
|
assert result.success is True # send() always returns success — fire-and-forget
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# 8. Inbound: filter own-echo by corrId prefix
|
|
# ---------------------------------------------------------------------------
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_handle_event_filters_own_corr_id():
|
|
from gateway.config import PlatformConfig
|
|
cfg = PlatformConfig(enabled=True, extra={"ws_url": "ws://localhost:5225"})
|
|
adapter = SimplexAdapter(cfg)
|
|
# Pretend we sent a command with this corrId
|
|
own = adapter._make_corr_id()
|
|
handler_mock = AsyncMock()
|
|
adapter._handle_new_chat_item = handler_mock # type: ignore
|
|
|
|
await adapter._handle_event({"corrId": own, "type": "newChatItem"})
|
|
handler_mock.assert_not_called()
|
|
assert own not in adapter._pending_corr_ids # discarded
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# 9. Standalone (out-of-process) send for cron
|
|
# ---------------------------------------------------------------------------
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_standalone_send_missing_websockets(monkeypatch):
|
|
"""When websockets is unimportable, return a clean error dict.
|
|
|
|
Implementation detail: the standalone path does ``import websockets``
|
|
inside the function body. We simulate the package being absent by
|
|
pulling it out of ``sys.modules`` and pointing the finder at None.
|
|
"""
|
|
import sys
|
|
saved_websockets = sys.modules.pop("websockets", None)
|
|
saved_meta = list(sys.meta_path)
|
|
|
|
class _Blocker:
|
|
@staticmethod
|
|
def find_spec(name, path=None, target=None):
|
|
if name == "websockets" or name.startswith("websockets."):
|
|
raise ImportError("websockets blocked for test")
|
|
return None
|
|
|
|
sys.meta_path.insert(0, _Blocker())
|
|
try:
|
|
pconfig = MagicMock()
|
|
pconfig.extra = {"ws_url": "ws://localhost:5225"}
|
|
result = await _standalone_send(pconfig, "contact-42", "hi")
|
|
assert isinstance(result, dict)
|
|
assert "error" in result
|
|
assert "websockets" in result["error"]
|
|
finally:
|
|
sys.meta_path[:] = saved_meta
|
|
if saved_websockets is not None:
|
|
sys.modules["websockets"] = saved_websockets
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_standalone_send_defaults_to_local_daemon(monkeypatch):
|
|
monkeypatch.delenv("SIMPLEX_WS_URL", raising=False)
|
|
pconfig = MagicMock()
|
|
pconfig.extra = {}
|
|
|
|
sent_payloads = []
|
|
|
|
class DummyWs:
|
|
async def __aenter__(self):
|
|
return self
|
|
|
|
async def __aexit__(self, exc_type, exc, tb):
|
|
return None
|
|
|
|
async def send(self, payload):
|
|
sent_payloads.append(json.loads(payload))
|
|
|
|
def fake_connect(url, **kwargs):
|
|
assert url == "ws://127.0.0.1:5225"
|
|
assert kwargs["open_timeout"] == 10
|
|
assert kwargs["close_timeout"] == 5
|
|
return DummyWs()
|
|
|
|
import websockets
|
|
monkeypatch.setattr(websockets, "connect", fake_connect)
|
|
|
|
result = await _standalone_send(pconfig, "contact-42", "hi")
|
|
assert result == {"success": True, "platform": "simplex", "chat_id": "contact-42"}
|
|
assert sent_payloads[0]["cmd"] == "@contact-42 hi"
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_health_monitor_does_not_reconnect_quiet_healthy_ws(monkeypatch):
|
|
from gateway.config import PlatformConfig
|
|
cfg = PlatformConfig(enabled=True, extra={"ws_url": "ws://localhost:5225"})
|
|
adapter = SimplexAdapter(cfg)
|
|
adapter._running = True
|
|
adapter._last_ws_activity = 0
|
|
adapter._ws = AsyncMock()
|
|
|
|
monkeypatch.setattr(_simplex, "HEALTH_CHECK_INTERVAL", 0.01)
|
|
monkeypatch.setattr(_simplex, "HEALTH_CHECK_STALE_THRESHOLD", 0.01)
|
|
|
|
task = asyncio.create_task(adapter._health_monitor())
|
|
await asyncio.sleep(0.03)
|
|
adapter._running = False
|
|
await asyncio.wait_for(task, timeout=1)
|
|
|
|
adapter._ws.close.assert_not_called()
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
# 10. register() — plugin-side metadata
|
|
# ---------------------------------------------------------------------------
|
|
|
|
def test_register_calls_register_platform():
|
|
ctx = MagicMock()
|
|
register(ctx)
|
|
ctx.register_platform.assert_called_once()
|
|
kwargs = ctx.register_platform.call_args.kwargs
|
|
assert kwargs["name"] == "simplex"
|
|
assert kwargs["label"] == "SimpleX Chat"
|
|
assert kwargs["required_env"] == ["SIMPLEX_WS_URL"]
|
|
assert kwargs["allowed_users_env"] == "SIMPLEX_ALLOWED_USERS"
|
|
assert kwargs["allow_all_env"] == "SIMPLEX_ALLOW_ALL_USERS"
|
|
assert kwargs["cron_deliver_env_var"] == "SIMPLEX_HOME_CHANNEL"
|
|
assert callable(kwargs["check_fn"])
|
|
assert callable(kwargs["validate_config"])
|
|
assert callable(kwargs["is_connected"])
|
|
assert callable(kwargs["env_enablement_fn"])
|
|
assert callable(kwargs["standalone_sender_fn"])
|
|
assert callable(kwargs["adapter_factory"])
|
|
assert callable(kwargs["setup_fn"])
|
|
# SimpleX uses opaque IDs only — no PII to redact.
|
|
assert kwargs["pii_safe"] is True
|