When auto-threading kicked in, the broadened backfill gate ran on the freshly-created thread — but the thread has no prior context to fetch, and the parent-channel reference passed to _fetch_channel_context would have leaked unrelated context (see #31467). Skip backfill when auto_threaded_channel is set. Also teach the _FakeTextChannel / _FakeThreadChannel test doubles to expose a no-op history() async generator so the broadened gate doesn't trip AttributeError → discord.Forbidden (MagicMock) → TypeError in the existing auto-thread tests. Add a regression test that asserts auto-threaded messages do not trigger backfill.
930 lines
36 KiB
Python
930 lines
36 KiB
Python
"""Tests for Discord free-response defaults and mention gating."""
|
|
|
|
from datetime import datetime, timezone
|
|
from types import SimpleNamespace
|
|
from unittest.mock import AsyncMock, MagicMock
|
|
import sys
|
|
|
|
import pytest
|
|
|
|
from gateway.config import PlatformConfig
|
|
|
|
|
|
def _ensure_discord_mock():
|
|
"""Install a mock discord module when discord.py isn't available."""
|
|
if "discord" in sys.modules and hasattr(sys.modules["discord"], "__file__"):
|
|
return
|
|
|
|
discord_mod = MagicMock()
|
|
discord_mod.Intents.default.return_value = MagicMock()
|
|
discord_mod.Client = MagicMock
|
|
discord_mod.File = MagicMock
|
|
discord_mod.DMChannel = type("DMChannel", (), {})
|
|
discord_mod.Thread = type("Thread", (), {})
|
|
discord_mod.ForumChannel = type("ForumChannel", (), {})
|
|
discord_mod.ui = SimpleNamespace(View=object, button=lambda *a, **k: (lambda fn: fn), Button=object)
|
|
discord_mod.ButtonStyle = SimpleNamespace(success=1, primary=2, secondary=2, danger=3, green=1, grey=2, blurple=2, red=3)
|
|
discord_mod.Color = SimpleNamespace(orange=lambda: 1, green=lambda: 2, blue=lambda: 3, red=lambda: 4, purple=lambda: 5)
|
|
discord_mod.Interaction = object
|
|
discord_mod.Embed = MagicMock
|
|
discord_mod.app_commands = SimpleNamespace(
|
|
describe=lambda **kwargs: (lambda fn: fn),
|
|
choices=lambda **kwargs: (lambda fn: fn),
|
|
Choice=lambda **kwargs: SimpleNamespace(**kwargs),
|
|
)
|
|
|
|
ext_mod = MagicMock()
|
|
commands_mod = MagicMock()
|
|
commands_mod.Bot = MagicMock
|
|
ext_mod.commands = commands_mod
|
|
|
|
sys.modules.setdefault("discord", discord_mod)
|
|
sys.modules.setdefault("discord.ext", ext_mod)
|
|
sys.modules.setdefault("discord.ext.commands", commands_mod)
|
|
|
|
|
|
_ensure_discord_mock()
|
|
|
|
import plugins.platforms.discord.adapter as discord_platform # noqa: E402
|
|
from plugins.platforms.discord.adapter import DiscordAdapter # noqa: E402
|
|
|
|
|
|
class FakeDMChannel:
|
|
def __init__(self, channel_id: int = 1, name: str = "dm"):
|
|
self.id = channel_id
|
|
self.name = name
|
|
|
|
|
|
class FakeTextChannel:
|
|
def __init__(self, channel_id: int = 1, name: str = "general", guild_name: str = "Hermes Server"):
|
|
self.id = channel_id
|
|
self.name = name
|
|
self.guild = SimpleNamespace(name=guild_name)
|
|
self.topic = None
|
|
|
|
def history(self, *, limit, before, after=None, oldest_first=None):
|
|
async def _iter():
|
|
return
|
|
yield
|
|
return _iter()
|
|
|
|
|
|
class FakeForumChannel:
|
|
def __init__(self, channel_id: int = 1, name: str = "support-forum", guild_name: str = "Hermes Server"):
|
|
self.id = channel_id
|
|
self.name = name
|
|
self.guild = SimpleNamespace(name=guild_name)
|
|
self.type = 15
|
|
self.topic = None
|
|
|
|
|
|
class FakeThread:
|
|
def __init__(self, channel_id: int = 1, name: str = "thread", parent=None, guild_name: str = "Hermes Server"):
|
|
self.id = channel_id
|
|
self.name = name
|
|
self.parent = parent
|
|
self.parent_id = getattr(parent, "id", None)
|
|
self.guild = getattr(parent, "guild", None) or SimpleNamespace(name=guild_name)
|
|
self.topic = None
|
|
|
|
def history(self, *, limit, before, after=None, oldest_first=None):
|
|
async def _iter():
|
|
return
|
|
yield
|
|
return _iter()
|
|
|
|
|
|
@pytest.fixture
|
|
def adapter(monkeypatch):
|
|
monkeypatch.setattr(discord_platform.discord, "DMChannel", FakeDMChannel, raising=False)
|
|
monkeypatch.setattr(discord_platform.discord, "Thread", FakeThread, raising=False)
|
|
monkeypatch.setattr(discord_platform.discord, "ForumChannel", FakeForumChannel, raising=False)
|
|
|
|
# Clear DISCORD_* env vars the test file exercises so tests don't leak
|
|
# process-env state from the contributor's shell into per-test behaviour.
|
|
# Individual tests still monkeypatch.setenv() for their own scenarios.
|
|
for _var in (
|
|
"DISCORD_REQUIRE_MENTION",
|
|
"DISCORD_THREAD_REQUIRE_MENTION",
|
|
"DISCORD_FREE_RESPONSE_CHANNELS",
|
|
"DISCORD_AUTO_THREAD",
|
|
"DISCORD_NO_THREAD_CHANNELS",
|
|
"DISCORD_ALLOWED_CHANNELS",
|
|
"DISCORD_IGNORED_CHANNELS",
|
|
"DISCORD_HISTORY_BACKFILL",
|
|
"DISCORD_HISTORY_BACKFILL_LIMIT",
|
|
"DISCORD_ALLOW_BOTS",
|
|
):
|
|
monkeypatch.delenv(_var, raising=False)
|
|
|
|
config = PlatformConfig(enabled=True, token="fake-token")
|
|
adapter = DiscordAdapter(config)
|
|
adapter._client = SimpleNamespace(user=SimpleNamespace(id=999))
|
|
adapter._text_batch_delay_seconds = 0 # disable batching for tests
|
|
adapter.handle_message = AsyncMock()
|
|
return adapter
|
|
|
|
|
|
def make_message(*, channel, content: str, mentions=None, msg_type=None):
|
|
author = SimpleNamespace(id=42, display_name="Jezza", name="Jezza")
|
|
return SimpleNamespace(
|
|
id=123,
|
|
content=content,
|
|
mentions=list(mentions or []),
|
|
attachments=[],
|
|
reference=None,
|
|
created_at=datetime.now(timezone.utc),
|
|
channel=channel,
|
|
author=author,
|
|
type=msg_type if msg_type is not None else discord_platform.discord.MessageType.default,
|
|
)
|
|
|
|
|
|
def make_history_message(
|
|
*,
|
|
author,
|
|
content: str,
|
|
msg_id: int,
|
|
msg_type=None,
|
|
attachments=None,
|
|
):
|
|
return SimpleNamespace(
|
|
id=msg_id,
|
|
author=author,
|
|
content=content,
|
|
attachments=list(attachments or []),
|
|
type=msg_type if msg_type is not None else discord_platform.discord.MessageType.default,
|
|
)
|
|
|
|
|
|
class FakeHistoryChannel(FakeTextChannel):
|
|
def __init__(self, history_messages, **kwargs):
|
|
super().__init__(**kwargs)
|
|
self._history_messages = list(history_messages)
|
|
|
|
def history(self, *, limit, before, after=None, oldest_first=None):
|
|
before_id = int(getattr(before, "id", before))
|
|
after_id = int(getattr(after, "id", after)) if after is not None else None
|
|
if oldest_first is None:
|
|
oldest_first = after is not None
|
|
|
|
messages = [
|
|
message for message in self._history_messages
|
|
if int(message.id) < before_id
|
|
and (after_id is None or int(message.id) > after_id)
|
|
]
|
|
messages.sort(key=lambda message: int(message.id), reverse=not oldest_first)
|
|
|
|
async def _iter():
|
|
for message in messages[:limit]:
|
|
yield message
|
|
|
|
return _iter()
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_defaults_to_require_mention(adapter, monkeypatch):
|
|
"""Default behavior: require @mention in server channels."""
|
|
monkeypatch.delenv("DISCORD_REQUIRE_MENTION", raising=False)
|
|
monkeypatch.delenv("DISCORD_FREE_RESPONSE_CHANNELS", raising=False)
|
|
|
|
message = make_message(channel=FakeTextChannel(channel_id=123), content="hello from channel")
|
|
|
|
await adapter._handle_message(message)
|
|
|
|
# Should be ignored — no mention, require_mention defaults to true
|
|
adapter.handle_message.assert_not_awaited()
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_free_response_in_server_channels(adapter, monkeypatch):
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "false")
|
|
monkeypatch.delenv("DISCORD_FREE_RESPONSE_CHANNELS", raising=False)
|
|
|
|
message = make_message(channel=FakeTextChannel(channel_id=123), content="hello from channel")
|
|
|
|
await adapter._handle_message(message)
|
|
|
|
adapter.handle_message.assert_awaited_once()
|
|
event = adapter.handle_message.await_args.args[0]
|
|
assert event.text == "hello from channel"
|
|
assert event.source.chat_id == "123"
|
|
assert event.source.chat_type == "group"
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_free_response_in_threads(adapter, monkeypatch):
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "false")
|
|
monkeypatch.delenv("DISCORD_FREE_RESPONSE_CHANNELS", raising=False)
|
|
|
|
thread = FakeThread(channel_id=456, name="Ghost reader skill")
|
|
message = make_message(channel=thread, content="hello from thread")
|
|
|
|
await adapter._handle_message(message)
|
|
|
|
adapter.handle_message.assert_awaited_once()
|
|
event = adapter.handle_message.await_args.args[0]
|
|
assert event.text == "hello from thread"
|
|
assert event.source.chat_id == "456"
|
|
assert event.source.thread_id == "456"
|
|
assert event.source.chat_type == "thread"
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_forum_threads_are_handled_as_threads(adapter, monkeypatch):
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "false")
|
|
monkeypatch.delenv("DISCORD_FREE_RESPONSE_CHANNELS", raising=False)
|
|
|
|
forum = FakeForumChannel(channel_id=222, name="support-forum")
|
|
thread = FakeThread(channel_id=456, name="Can Hermes reply here?", parent=forum)
|
|
message = make_message(channel=thread, content="hello from forum post")
|
|
|
|
await adapter._handle_message(message)
|
|
|
|
adapter.handle_message.assert_awaited_once()
|
|
event = adapter.handle_message.await_args.args[0]
|
|
assert event.text == "hello from forum post"
|
|
assert event.source.chat_id == "456"
|
|
assert event.source.thread_id == "456"
|
|
assert event.source.chat_type == "thread"
|
|
assert event.source.chat_name == "Hermes Server / support-forum / Can Hermes reply here?"
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_can_still_require_mentions_when_enabled(adapter, monkeypatch):
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "true")
|
|
monkeypatch.delenv("DISCORD_FREE_RESPONSE_CHANNELS", raising=False)
|
|
|
|
message = make_message(channel=FakeTextChannel(channel_id=789), content="ignored without mention")
|
|
|
|
await adapter._handle_message(message)
|
|
|
|
adapter.handle_message.assert_not_awaited()
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_free_response_channel_overrides_mention_requirement(adapter, monkeypatch):
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "true")
|
|
monkeypatch.setenv("DISCORD_FREE_RESPONSE_CHANNELS", "789,999")
|
|
|
|
message = make_message(channel=FakeTextChannel(channel_id=789), content="allowed without mention")
|
|
|
|
await adapter._handle_message(message)
|
|
|
|
adapter.handle_message.assert_awaited_once()
|
|
event = adapter.handle_message.await_args.args[0]
|
|
assert event.text == "allowed without mention"
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_free_response_channel_can_come_from_config_extra(adapter, monkeypatch):
|
|
monkeypatch.delenv("DISCORD_REQUIRE_MENTION", raising=False)
|
|
monkeypatch.delenv("DISCORD_FREE_RESPONSE_CHANNELS", raising=False)
|
|
adapter.config.extra["free_response_channels"] = ["789", "999"]
|
|
|
|
message = make_message(channel=FakeTextChannel(channel_id=789), content="allowed from config")
|
|
|
|
await adapter._handle_message(message)
|
|
|
|
adapter.handle_message.assert_awaited_once()
|
|
event = adapter.handle_message.await_args.args[0]
|
|
assert event.text == "allowed from config"
|
|
|
|
|
|
def test_discord_free_response_channels_bare_int(adapter, monkeypatch):
|
|
# YAML `discord.free_response_channels: 1491973769726791812` (single bare
|
|
# integer) is loaded as an int and previously fell through the
|
|
# isinstance(str) branch in _discord_free_response_channels, silently
|
|
# returning an empty set. Scalar → str coercion makes single-channel
|
|
# config work without having to quote the ID in YAML.
|
|
monkeypatch.delenv("DISCORD_FREE_RESPONSE_CHANNELS", raising=False)
|
|
adapter.config.extra["free_response_channels"] = 1491973769726791812
|
|
|
|
assert adapter._discord_free_response_channels() == {"1491973769726791812"}
|
|
|
|
|
|
def test_discord_free_response_channels_int_list(adapter, monkeypatch):
|
|
# YAML list form with bare numeric entries — each element should be coerced.
|
|
monkeypatch.delenv("DISCORD_FREE_RESPONSE_CHANNELS", raising=False)
|
|
adapter.config.extra["free_response_channels"] = [1491973769726791812, 99999]
|
|
|
|
assert adapter._discord_free_response_channels() == {"1491973769726791812", "99999"}
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_forum_parent_in_free_response_list_allows_forum_thread(adapter, monkeypatch):
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "true")
|
|
monkeypatch.setenv("DISCORD_FREE_RESPONSE_CHANNELS", "222")
|
|
|
|
forum = FakeForumChannel(channel_id=222, name="support-forum")
|
|
thread = FakeThread(channel_id=333, name="Forum topic", parent=forum)
|
|
message = make_message(channel=thread, content="allowed from forum thread")
|
|
|
|
await adapter._handle_message(message)
|
|
|
|
adapter.handle_message.assert_awaited_once()
|
|
event = adapter.handle_message.await_args.args[0]
|
|
assert event.text == "allowed from forum thread"
|
|
assert event.source.chat_id == "333"
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_accepts_and_strips_bot_mentions_when_required(adapter, monkeypatch):
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "true")
|
|
monkeypatch.delenv("DISCORD_FREE_RESPONSE_CHANNELS", raising=False)
|
|
|
|
bot_user = adapter._client.user
|
|
message = make_message(
|
|
channel=FakeTextChannel(channel_id=321),
|
|
content=f"<@{bot_user.id}> hello with mention",
|
|
mentions=[bot_user],
|
|
)
|
|
|
|
await adapter._handle_message(message)
|
|
|
|
adapter.handle_message.assert_awaited_once()
|
|
event = adapter.handle_message.await_args.args[0]
|
|
assert event.text == "hello with mention"
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_dms_ignore_mention_requirement(adapter, monkeypatch):
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "true")
|
|
monkeypatch.delenv("DISCORD_FREE_RESPONSE_CHANNELS", raising=False)
|
|
|
|
message = make_message(channel=FakeDMChannel(channel_id=654), content="dm without mention")
|
|
|
|
await adapter._handle_message(message)
|
|
|
|
adapter.handle_message.assert_awaited_once()
|
|
event = adapter.handle_message.await_args.args[0]
|
|
assert event.text == "dm without mention"
|
|
assert event.source.chat_type == "dm"
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_auto_thread_enabled_by_default(adapter, monkeypatch):
|
|
"""Auto-threading should be enabled by default (DISCORD_AUTO_THREAD defaults to 'true')."""
|
|
monkeypatch.delenv("DISCORD_AUTO_THREAD", raising=False)
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "false")
|
|
|
|
# Patch _auto_create_thread to return a fake thread
|
|
fake_thread = FakeThread(channel_id=999, name="auto-thread")
|
|
adapter._auto_create_thread = AsyncMock(return_value=fake_thread)
|
|
|
|
message = make_message(channel=FakeTextChannel(channel_id=123), content="hello")
|
|
|
|
await adapter._handle_message(message)
|
|
|
|
adapter._auto_create_thread.assert_awaited_once()
|
|
adapter.handle_message.assert_awaited_once()
|
|
event = adapter.handle_message.await_args.args[0]
|
|
assert event.source.chat_type == "thread"
|
|
assert event.source.thread_id == "999"
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_reply_message_skips_auto_thread(adapter, monkeypatch):
|
|
"""Quote-replies should stay in-channel instead of trying to create a thread."""
|
|
monkeypatch.delenv("DISCORD_AUTO_THREAD", raising=False)
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "true")
|
|
monkeypatch.setenv("DISCORD_FREE_RESPONSE_CHANNELS", "123")
|
|
|
|
adapter._auto_create_thread = AsyncMock()
|
|
|
|
message = make_message(
|
|
channel=FakeTextChannel(channel_id=123),
|
|
content="reply without mention",
|
|
msg_type=discord_platform.discord.MessageType.reply,
|
|
)
|
|
|
|
await adapter._handle_message(message)
|
|
|
|
adapter._auto_create_thread.assert_not_awaited()
|
|
adapter.handle_message.assert_awaited_once()
|
|
event = adapter.handle_message.await_args.args[0]
|
|
assert event.text == "reply without mention"
|
|
assert event.source.chat_id == "123"
|
|
assert event.source.chat_type == "group"
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_auto_thread_can_be_disabled(adapter, monkeypatch):
|
|
"""Setting auto_thread to false skips thread creation."""
|
|
monkeypatch.setenv("DISCORD_AUTO_THREAD", "false")
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "false")
|
|
|
|
adapter._auto_create_thread = AsyncMock()
|
|
|
|
message = make_message(channel=FakeTextChannel(channel_id=123), content="hello")
|
|
|
|
await adapter._handle_message(message)
|
|
|
|
adapter._auto_create_thread.assert_not_awaited()
|
|
adapter.handle_message.assert_awaited_once()
|
|
event = adapter.handle_message.await_args.args[0]
|
|
assert event.source.chat_type == "group"
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_bot_thread_skips_mention_requirement(adapter, monkeypatch):
|
|
"""Messages in a thread the bot has participated in should not require @mention."""
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "true")
|
|
monkeypatch.delenv("DISCORD_FREE_RESPONSE_CHANNELS", raising=False)
|
|
monkeypatch.setenv("DISCORD_AUTO_THREAD", "false")
|
|
|
|
# Simulate bot having previously participated in thread 456
|
|
adapter._threads.mark("456")
|
|
|
|
thread = FakeThread(channel_id=456, name="existing thread")
|
|
message = make_message(channel=thread, content="follow-up without mention")
|
|
|
|
await adapter._handle_message(message)
|
|
|
|
adapter.handle_message.assert_awaited_once()
|
|
event = adapter.handle_message.await_args.args[0]
|
|
assert event.text == "follow-up without mention"
|
|
assert event.source.chat_type == "thread"
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_unknown_thread_still_requires_mention(adapter, monkeypatch):
|
|
"""Messages in a thread the bot hasn't participated in should still require @mention."""
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "true")
|
|
monkeypatch.delenv("DISCORD_FREE_RESPONSE_CHANNELS", raising=False)
|
|
monkeypatch.setenv("DISCORD_AUTO_THREAD", "false")
|
|
|
|
# Bot has NOT participated in thread 789
|
|
thread = FakeThread(channel_id=789, name="some thread")
|
|
message = make_message(channel=thread, content="hello from unknown thread")
|
|
|
|
await adapter._handle_message(message)
|
|
|
|
adapter.handle_message.assert_not_awaited()
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_auto_thread_tracks_participation(adapter, monkeypatch):
|
|
"""Auto-created threads should be tracked for future mention-free replies."""
|
|
monkeypatch.delenv("DISCORD_AUTO_THREAD", raising=False)
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "false")
|
|
|
|
fake_thread = FakeThread(channel_id=555, name="auto-thread")
|
|
adapter._auto_create_thread = AsyncMock(return_value=fake_thread)
|
|
|
|
message = make_message(channel=FakeTextChannel(channel_id=123), content="start a thread")
|
|
|
|
await adapter._handle_message(message)
|
|
|
|
assert "555" in adapter._threads
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_thread_participation_tracked_on_dispatch(adapter, monkeypatch):
|
|
"""When the bot processes a message in a thread, it tracks participation."""
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "false")
|
|
monkeypatch.setenv("DISCORD_AUTO_THREAD", "false")
|
|
|
|
thread = FakeThread(channel_id=777, name="manually created thread")
|
|
message = make_message(channel=thread, content="hello in thread")
|
|
|
|
await adapter._handle_message(message)
|
|
|
|
assert "777" in adapter._threads
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_voice_linked_channel_skips_mention_requirement_and_auto_thread(adapter, monkeypatch):
|
|
"""Active voice-linked text channels should behave like free-response channels."""
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "true")
|
|
monkeypatch.delenv("DISCORD_FREE_RESPONSE_CHANNELS", raising=False)
|
|
monkeypatch.delenv("DISCORD_AUTO_THREAD", raising=False)
|
|
|
|
adapter._voice_text_channels[111] = 789
|
|
adapter._auto_create_thread = AsyncMock()
|
|
|
|
message = make_message(
|
|
channel=FakeTextChannel(channel_id=789),
|
|
content="follow-up from voice text chat",
|
|
)
|
|
|
|
await adapter._handle_message(message)
|
|
|
|
adapter._auto_create_thread.assert_not_awaited()
|
|
adapter.handle_message.assert_awaited_once()
|
|
event = adapter.handle_message.await_args.args[0]
|
|
assert event.text == "follow-up from voice text chat"
|
|
assert event.source.chat_type == "group"
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_free_response_channel_skips_auto_thread(adapter, monkeypatch):
|
|
"""Free-response channels should reply inline, never spawn a new thread.
|
|
|
|
Without this, every message in a free-response channel would auto-create
|
|
a fresh thread (since the channel bypasses the @mention gate, every
|
|
message looks like a fresh trigger). That turns a "lightweight chat"
|
|
channel into a thread-spawning machine — see the docs at
|
|
website/docs/user-guide/messaging/discord.md which already describe
|
|
this as the intended behavior.
|
|
"""
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "true")
|
|
monkeypatch.setenv("DISCORD_FREE_RESPONSE_CHANNELS", "789")
|
|
monkeypatch.delenv("DISCORD_AUTO_THREAD", raising=False) # default true
|
|
|
|
adapter._auto_create_thread = AsyncMock()
|
|
|
|
message = make_message(
|
|
channel=FakeTextChannel(channel_id=789),
|
|
content="casual chat in free-response channel",
|
|
)
|
|
|
|
await adapter._handle_message(message)
|
|
|
|
adapter._auto_create_thread.assert_not_awaited()
|
|
adapter.handle_message.assert_awaited_once()
|
|
event = adapter.handle_message.await_args.args[0]
|
|
assert event.text == "casual chat in free-response channel"
|
|
assert event.source.chat_type == "group"
|
|
|
|
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_voice_linked_parent_thread_still_requires_mention(adapter, monkeypatch):
|
|
"""Threads under a voice-linked channel should still require @mention."""
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "true")
|
|
monkeypatch.delenv("DISCORD_FREE_RESPONSE_CHANNELS", raising=False)
|
|
|
|
adapter._voice_text_channels[111] = 789
|
|
message = make_message(
|
|
channel=FakeThread(channel_id=790, parent=FakeTextChannel(channel_id=789)),
|
|
content="thread reply without mention",
|
|
)
|
|
|
|
await adapter._handle_message(message)
|
|
|
|
adapter.handle_message.assert_not_awaited()
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_thread_default_keeps_responding_after_participation(adapter, monkeypatch):
|
|
"""Default behavior: once the bot is in a thread, it auto-responds without @mention."""
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "true")
|
|
monkeypatch.delenv("DISCORD_FREE_RESPONSE_CHANNELS", raising=False)
|
|
monkeypatch.delenv("DISCORD_THREAD_REQUIRE_MENTION", raising=False)
|
|
|
|
thread = FakeThread(channel_id=456, name="follow-up")
|
|
adapter._threads.mark("456") # bot has previously participated
|
|
|
|
message = make_message(channel=thread, content="follow-up without mention")
|
|
await adapter._handle_message(message)
|
|
|
|
adapter.handle_message.assert_awaited_once()
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_thread_require_mention_gates_followups(adapter, monkeypatch):
|
|
"""When thread_require_mention=true, even bot-participated threads need @mention."""
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "true")
|
|
monkeypatch.setenv("DISCORD_THREAD_REQUIRE_MENTION", "true")
|
|
monkeypatch.delenv("DISCORD_FREE_RESPONSE_CHANNELS", raising=False)
|
|
|
|
thread = FakeThread(channel_id=456, name="multi-bot thread")
|
|
adapter._threads.mark("456") # bot has previously participated
|
|
|
|
message = make_message(channel=thread, content="ambient chatter — not for me")
|
|
await adapter._handle_message(message)
|
|
|
|
adapter.handle_message.assert_not_awaited()
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_thread_require_mention_still_responds_when_mentioned(adapter, monkeypatch):
|
|
"""thread_require_mention=true still lets explicit @mentions through in threads."""
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "true")
|
|
monkeypatch.setenv("DISCORD_THREAD_REQUIRE_MENTION", "true")
|
|
monkeypatch.delenv("DISCORD_FREE_RESPONSE_CHANNELS", raising=False)
|
|
|
|
thread = FakeThread(channel_id=456, name="multi-bot thread")
|
|
adapter._threads.mark("456")
|
|
bot_user = adapter._client.user
|
|
|
|
message = make_message(
|
|
channel=thread,
|
|
content=f"<@{bot_user.id}> hey, this one's for you",
|
|
mentions=[bot_user],
|
|
)
|
|
await adapter._handle_message(message)
|
|
|
|
adapter.handle_message.assert_awaited_once()
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_thread_require_mention_via_config_extra(adapter, monkeypatch):
|
|
"""thread_require_mention can also be set via config.extra (yaml)."""
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "true")
|
|
monkeypatch.delenv("DISCORD_THREAD_REQUIRE_MENTION", raising=False)
|
|
monkeypatch.delenv("DISCORD_FREE_RESPONSE_CHANNELS", raising=False)
|
|
adapter.config.extra["thread_require_mention"] = True
|
|
|
|
thread = FakeThread(channel_id=456, name="multi-bot thread")
|
|
adapter._threads.mark("456")
|
|
|
|
message = make_message(channel=thread, content="ambient — should be ignored")
|
|
await adapter._handle_message(message)
|
|
|
|
adapter.handle_message.assert_not_awaited()
|
|
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_fetch_channel_context_stops_at_self_message_and_reverses_to_chronological_order(adapter, monkeypatch):
|
|
monkeypatch.setenv("DISCORD_ALLOW_BOTS", "all")
|
|
adapter.config.extra["history_backfill_limit"] = 10
|
|
|
|
other_bot = SimpleNamespace(id=55, display_name="Gemini", name="Gemini", bot=True)
|
|
human = SimpleNamespace(id=56, display_name="Alice", name="Alice", bot=False)
|
|
old_human = SimpleNamespace(id=57, display_name="Bob", name="Bob", bot=False)
|
|
|
|
channel = FakeHistoryChannel(
|
|
[
|
|
make_history_message(author=human, content="latest human note", msg_id=4),
|
|
make_history_message(author=other_bot, content="latest bot note", msg_id=3),
|
|
make_history_message(author=adapter._client.user, content="our prior response", msg_id=2),
|
|
make_history_message(author=old_human, content="older than boundary", msg_id=1),
|
|
],
|
|
channel_id=123,
|
|
)
|
|
|
|
result = await adapter._fetch_channel_context(channel, before=make_message(channel=channel, content="trigger"))
|
|
|
|
assert result == (
|
|
"[Recent channel messages]\n"
|
|
"[Gemini [bot]] latest bot note\n"
|
|
"[Alice] latest human note"
|
|
)
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_fetch_channel_context_skips_other_bots_when_allow_bots_none(adapter, monkeypatch):
|
|
monkeypatch.setenv("DISCORD_ALLOW_BOTS", "none")
|
|
adapter.config.extra["history_backfill_limit"] = 10
|
|
|
|
other_bot = SimpleNamespace(id=55, display_name="Gemini", name="Gemini", bot=True)
|
|
human = SimpleNamespace(id=56, display_name="Alice", name="Alice", bot=False)
|
|
|
|
channel = FakeHistoryChannel(
|
|
[
|
|
make_history_message(author=human, content="human note", msg_id=3),
|
|
make_history_message(author=other_bot, content="bot note", msg_id=2),
|
|
],
|
|
channel_id=123,
|
|
)
|
|
|
|
result = await adapter._fetch_channel_context(channel, before=make_message(channel=channel, content="trigger"))
|
|
|
|
assert result == "[Recent channel messages]\n[Alice] human note"
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_fetch_channel_context_uses_cache_to_narrow_window(adapter, monkeypatch):
|
|
"""When _last_self_message_id is cached, the fetch passes after= to skip old messages."""
|
|
monkeypatch.setenv("DISCORD_ALLOW_BOTS", "all")
|
|
adapter.config.extra["history_backfill_limit"] = 50
|
|
|
|
human = SimpleNamespace(id=56, display_name="Alice", name="Alice", bot=False)
|
|
|
|
# Record the after= arg passed to history()
|
|
recorded_after = {}
|
|
|
|
class CacheTrackingChannel(FakeHistoryChannel):
|
|
def history(self, *, limit, before, after=None, oldest_first=None):
|
|
recorded_after["value"] = after
|
|
return super().history(
|
|
limit=limit,
|
|
before=before,
|
|
after=after,
|
|
oldest_first=oldest_first,
|
|
)
|
|
|
|
channel = CacheTrackingChannel(
|
|
[make_history_message(author=human, content="hello", msg_id=200)],
|
|
channel_id=777,
|
|
)
|
|
|
|
# Seed the cache — bot's last message in this channel was ID 100
|
|
adapter._last_self_message_id["777"] = "100"
|
|
|
|
trigger = make_message(channel=channel, content="trigger")
|
|
trigger.id = 300 # trigger is newer than cache
|
|
|
|
result = await adapter._fetch_channel_context(channel, before=trigger)
|
|
|
|
assert result == "[Recent channel messages]\n[Alice] hello"
|
|
# Verify cache was used: after= should be set (not None)
|
|
assert recorded_after["value"] is not None
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_fetch_channel_context_cache_uses_latest_window_when_after_set(adapter, monkeypatch):
|
|
"""Regression: discord.py defaults oldest_first=True when after= is provided.
|
|
|
|
The hot cache path passes both after= and before=. We still want the latest
|
|
messages before the trigger, not the earliest messages after our prior
|
|
response, otherwise tool traces can crowd out the final answer.
|
|
"""
|
|
monkeypatch.setenv("DISCORD_ALLOW_BOTS", "all")
|
|
adapter.config.extra["history_backfill_limit"] = 3
|
|
|
|
codex = SimpleNamespace(id=56, display_name="Codex", name="Codex", bot=True)
|
|
human = SimpleNamespace(id=57, display_name="Alice", name="Alice", bot=False)
|
|
|
|
channel = FakeHistoryChannel(
|
|
[
|
|
make_history_message(author=codex, content="old tool trace 1", msg_id=101),
|
|
make_history_message(author=codex, content="old tool trace 2", msg_id=102),
|
|
make_history_message(author=codex, content="old tool trace 3", msg_id=103),
|
|
make_history_message(author=codex, content="final analysis", msg_id=104),
|
|
make_history_message(author=human, content="latest follow-up", msg_id=105),
|
|
],
|
|
channel_id=777,
|
|
)
|
|
adapter._last_self_message_id["777"] = "100"
|
|
|
|
trigger = make_message(channel=channel, content="trigger")
|
|
trigger.id = 200
|
|
|
|
result = await adapter._fetch_channel_context(channel, before=trigger)
|
|
|
|
assert "[Codex [bot]] final analysis" in result
|
|
assert "[Alice] latest follow-up" in result
|
|
assert "old tool trace 1" not in result
|
|
assert "old tool trace 2" not in result
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_fetch_channel_context_ignores_stale_cache(adapter, monkeypatch):
|
|
"""If cached ID is >= trigger ID (stale/future), fall back to cold-start scan."""
|
|
monkeypatch.setenv("DISCORD_ALLOW_BOTS", "all")
|
|
adapter.config.extra["history_backfill_limit"] = 50
|
|
|
|
human = SimpleNamespace(id=56, display_name="Alice", name="Alice", bot=False)
|
|
|
|
recorded_after = {}
|
|
|
|
class CacheTrackingChannel(FakeHistoryChannel):
|
|
def history(self, *, limit, before, after=None, oldest_first=None):
|
|
recorded_after["value"] = after
|
|
return super().history(
|
|
limit=limit,
|
|
before=before,
|
|
after=after,
|
|
oldest_first=oldest_first,
|
|
)
|
|
|
|
channel = CacheTrackingChannel(
|
|
[make_history_message(author=human, content="hello", msg_id=50)],
|
|
channel_id=777,
|
|
)
|
|
|
|
# Cache has a NEWER ID than the trigger — stale/invalid
|
|
adapter._last_self_message_id["777"] = "500"
|
|
|
|
trigger = make_message(channel=channel, content="trigger")
|
|
trigger.id = 300
|
|
|
|
result = await adapter._fetch_channel_context(channel, before=trigger)
|
|
|
|
assert result == "[Recent channel messages]\n[Alice] hello"
|
|
# Cache should have been ignored — after= should be None
|
|
assert recorded_after["value"] is None
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_shared_channel_backfill_prepends_context(adapter, monkeypatch):
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "true")
|
|
monkeypatch.delenv("DISCORD_FREE_RESPONSE_CHANNELS", raising=False)
|
|
monkeypatch.setenv("DISCORD_AUTO_THREAD", "false")
|
|
adapter.config.extra["group_sessions_per_user"] = False
|
|
adapter.config.extra["history_backfill"] = True
|
|
adapter._fetch_channel_context = AsyncMock(return_value="[Recent channel messages]\n[Alice] context")
|
|
|
|
bot_user = adapter._client.user
|
|
message = make_message(
|
|
channel=FakeTextChannel(channel_id=321),
|
|
content=f"<@{bot_user.id}> hello with mention",
|
|
mentions=[bot_user],
|
|
)
|
|
|
|
await adapter._handle_message(message)
|
|
|
|
adapter._fetch_channel_context.assert_awaited_once()
|
|
event = adapter.handle_message.await_args.args[0]
|
|
assert event.text == "hello with mention"
|
|
assert event.channel_context == "[Recent channel messages]\n[Alice] context"
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_per_user_channel_backfills_too(adapter, monkeypatch):
|
|
"""Per-user sessions also benefit from backfill: Alice's session is missing
|
|
other-channel-participants' context and her own pre-mention messages."""
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "true")
|
|
monkeypatch.delenv("DISCORD_FREE_RESPONSE_CHANNELS", raising=False)
|
|
monkeypatch.setenv("DISCORD_AUTO_THREAD", "false")
|
|
adapter.config.extra["group_sessions_per_user"] = True
|
|
adapter.config.extra["history_backfill"] = True
|
|
adapter._fetch_channel_context = AsyncMock(return_value="[Recent channel messages]\n[Alice] context")
|
|
|
|
bot_user = adapter._client.user
|
|
message = make_message(
|
|
channel=FakeTextChannel(channel_id=321),
|
|
content=f"<@{bot_user.id}> hello with mention",
|
|
mentions=[bot_user],
|
|
)
|
|
|
|
await adapter._handle_message(message)
|
|
|
|
adapter._fetch_channel_context.assert_awaited_once()
|
|
event = adapter.handle_message.await_args.args[0]
|
|
assert event.text == "hello with mention"
|
|
assert event.channel_context == "[Recent channel messages]\n[Alice] context"
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_participated_thread_backfills_without_mention(adapter, monkeypatch):
|
|
"""Known threads still need recent thread context when mention gating is bypassed."""
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "true")
|
|
monkeypatch.delenv("DISCORD_FREE_RESPONSE_CHANNELS", raising=False)
|
|
monkeypatch.delenv("DISCORD_THREAD_REQUIRE_MENTION", raising=False)
|
|
adapter.config.extra["history_backfill"] = True
|
|
adapter._fetch_channel_context = AsyncMock(return_value="[Recent channel messages]\n[Alice] thread context")
|
|
|
|
thread = FakeThread(channel_id=456, name="follow-up")
|
|
adapter._threads.mark("456")
|
|
|
|
message = make_message(channel=thread, content="follow-up without mention")
|
|
await adapter._handle_message(message)
|
|
|
|
adapter._fetch_channel_context.assert_awaited_once()
|
|
event = adapter.handle_message.await_args.args[0]
|
|
assert event.text == "follow-up without mention"
|
|
assert event.channel_context == "[Recent channel messages]\n[Alice] thread context"
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_dm_does_not_backfill(adapter, monkeypatch):
|
|
"""DMs skip backfill — every DM triggers the bot, so there's no mention gap."""
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "true")
|
|
adapter.config.extra["history_backfill"] = True
|
|
adapter._fetch_channel_context = AsyncMock(return_value="[Recent channel messages]\n[Alice] context")
|
|
|
|
bot_user = adapter._client.user
|
|
dm_channel = SimpleNamespace(
|
|
id=999,
|
|
name=None,
|
|
guild=None,
|
|
topic=None,
|
|
)
|
|
# Make isinstance(channel, discord.DMChannel) return True
|
|
monkeypatch.setattr(
|
|
discord_platform.discord, "DMChannel", type(dm_channel), raising=False,
|
|
)
|
|
|
|
message = make_message(
|
|
channel=dm_channel,
|
|
content="hello in DM",
|
|
mentions=[],
|
|
)
|
|
|
|
await adapter._handle_message(message)
|
|
|
|
adapter._fetch_channel_context.assert_not_awaited()
|
|
if adapter.handle_message.await_args is not None:
|
|
event = adapter.handle_message.await_args.args[0]
|
|
assert event.channel_context is None
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_discord_auto_thread_skips_backfill(adapter, monkeypatch):
|
|
"""Auto-created threads skip backfill — the thread is brand new with no prior context."""
|
|
monkeypatch.setenv("DISCORD_REQUIRE_MENTION", "true")
|
|
monkeypatch.setenv("DISCORD_AUTO_THREAD", "true")
|
|
monkeypatch.delenv("DISCORD_NO_THREAD_CHANNELS", raising=False)
|
|
monkeypatch.delenv("DISCORD_FREE_RESPONSE_CHANNELS", raising=False)
|
|
adapter.config.extra["history_backfill"] = True
|
|
|
|
fake_thread = FakeThread(channel_id=777, name="auto-thread")
|
|
adapter._auto_create_thread = AsyncMock(return_value=fake_thread)
|
|
adapter._fetch_channel_context = AsyncMock(return_value="[Recent channel messages]\n[Alice] noise")
|
|
|
|
bot_user = adapter._client.user
|
|
parent = FakeTextChannel(channel_id=200, name="general")
|
|
message = make_message(channel=parent, content="hello", mentions=[bot_user])
|
|
await adapter._handle_message(message)
|
|
|
|
adapter._auto_create_thread.assert_awaited_once()
|
|
adapter._fetch_channel_context.assert_not_awaited()
|
|
|
|
|