mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-06-22 10:32:00 +00:00
Salvage of PR #41284 onto current main. Relocates the last 9 inline messaging adapters (+ satellites: telegram_network, feishu_comment/_rules/meeting_invite, wecom_crypto, wecom_callback) from gateway/platforms/ into self-contained bundled plugins under plugins/platforms/<x>/, discovered via the platform registry. Strips the per-platform core touchpoints from gateway/run.py, gateway/config.py, hermes_cli/gateway.py, hermes_cli/setup.py, and tools/send_message_tool.py. Carries forward the migration fixes (explicit enabled:false honored, get_connected_platforms forces discovery, plugin is_connected via gateway.get_env_value, logs --component gateway matches plugins.platforms.*, matrix hidden on Windows). Additionally ports config keys main added since the PR base: the matrix plugin's _apply_yaml_config now also covers allowed_users, ignore_user_patterns, process_notices, and session_scope (the inline gateway/config.py matrix block gained these in the 1340 commits the PR sat open; they would otherwise have been silently dropped on deletion).
107 lines
3.2 KiB
Python
107 lines
3.2 KiB
Python
"""Text-debounce batching for the WhatsApp adapter (issue #35301).
|
|
|
|
WhatsApp delivers rapid multi-message bursts (forwarded batches, paste-splits)
|
|
individually. Without debounce each fragment triggers a separate agent
|
|
invocation, wasting tokens and flooding the user with reply fragments. This
|
|
mirrors the Telegram/WeCom/Feishu pattern.
|
|
|
|
Batch delays are read from ``config.extra`` (config.yaml), not env vars.
|
|
"""
|
|
|
|
import asyncio
|
|
|
|
from gateway.config import Platform, PlatformConfig
|
|
from gateway.platforms.base import MessageEvent, MessageType
|
|
from plugins.platforms.whatsapp.adapter import WhatsAppAdapter
|
|
from gateway.session import SessionSource
|
|
|
|
|
|
def _make_adapter(**extra):
|
|
base = {"session_name": "test"}
|
|
base.update(extra)
|
|
return WhatsAppAdapter(PlatformConfig(enabled=True, extra=base))
|
|
|
|
|
|
def _event(text):
|
|
src = SessionSource(
|
|
platform=Platform.WHATSAPP,
|
|
chat_id="chat123",
|
|
chat_type="dm",
|
|
user_id="user1",
|
|
user_name="tester",
|
|
)
|
|
return MessageEvent(text=text, message_type=MessageType.TEXT, source=src)
|
|
|
|
|
|
def test_batch_delays_default_from_config():
|
|
adapter = _make_adapter()
|
|
assert adapter._text_batch_delay_seconds == 5.0
|
|
assert adapter._text_batch_split_delay_seconds == 10.0
|
|
|
|
|
|
def test_batch_delays_overridden_via_config_extra():
|
|
adapter = _make_adapter(
|
|
text_batch_delay_seconds="2.5",
|
|
text_batch_split_delay_seconds=7,
|
|
)
|
|
assert adapter._text_batch_delay_seconds == 2.5
|
|
assert adapter._text_batch_split_delay_seconds == 7.0
|
|
|
|
|
|
def test_invalid_config_value_falls_back_to_default():
|
|
adapter = _make_adapter(
|
|
text_batch_delay_seconds="garbage",
|
|
text_batch_split_delay_seconds=-3,
|
|
)
|
|
assert adapter._text_batch_delay_seconds == 5.0
|
|
assert adapter._text_batch_split_delay_seconds == 10.0
|
|
|
|
|
|
def test_env_var_is_ignored(monkeypatch):
|
|
# Config-only path: the legacy HERMES_* env var must NOT influence delays.
|
|
monkeypatch.setenv("HERMES_WHATSAPP_TEXT_BATCH_DELAY_SECONDS", "99")
|
|
adapter = _make_adapter()
|
|
assert adapter._text_batch_delay_seconds == 5.0
|
|
|
|
|
|
def test_rapid_texts_collapse_into_single_dispatch():
|
|
adapter = _make_adapter(
|
|
text_batch_delay_seconds=0.05,
|
|
text_batch_split_delay_seconds=0.05,
|
|
)
|
|
dispatched = []
|
|
|
|
async def _capture(event):
|
|
dispatched.append(event.text)
|
|
|
|
adapter.handle_message = _capture
|
|
|
|
async def _drive():
|
|
adapter._enqueue_text_event(_event("one"))
|
|
adapter._enqueue_text_event(_event("two"))
|
|
adapter._enqueue_text_event(_event("three"))
|
|
assert dispatched == [] # nothing flushed during the burst
|
|
await asyncio.sleep(0.2)
|
|
|
|
asyncio.run(_drive())
|
|
assert dispatched == ["one\ntwo\nthree"]
|
|
|
|
|
|
def test_lone_message_dispatched_alone():
|
|
adapter = _make_adapter(
|
|
text_batch_delay_seconds=0.05,
|
|
text_batch_split_delay_seconds=0.05,
|
|
)
|
|
dispatched = []
|
|
|
|
async def _capture(event):
|
|
dispatched.append(event.text)
|
|
|
|
adapter.handle_message = _capture
|
|
|
|
async def _drive():
|
|
adapter._enqueue_text_event(_event("solo"))
|
|
await asyncio.sleep(0.2)
|
|
|
|
asyncio.run(_drive())
|
|
assert dispatched == ["solo"]
|