"""Regression tests: /stop can interrupt a sibling participant's run in a per-user thread. When ``thread_sessions_per_user=True``, each participant in a thread gets an isolated session key (``...:{thread_id}:{user_id}``). A run another user started lives under a different key, so the caller's own ``/stop`` used to find nothing and reply "no active task to stop". Authorized users should be able to stop any run in the same thread. """ import pytest from gateway.run import GatewayRunner, _AGENT_PENDING_SENTINEL, _INTERRUPT_REASON_STOP from gateway.session import SessionSource, build_session_key from gateway.platforms.base import Platform, MessageEvent, MessageType class _FakeAgent: pass def _thread_source(uid, thread_id="thr1", chat_id="chan1"): return SessionSource( platform=Platform.DISCORD, chat_type="forum", chat_id=chat_id, thread_id=thread_id, user_id=uid, ) def _per_user_key(uid, thread_id="thr1", chat_id="chan1"): return build_session_key( _thread_source(uid, thread_id, chat_id), thread_sessions_per_user=True, ) # --------------------------------------------------------------------------- # _sibling_thread_run_keys # --------------------------------------------------------------------------- def test_sibling_finds_other_users_run_in_same_thread(): runner = object.__new__(GatewayRunner) key_a = _per_user_key("userA") key_b = _per_user_key("userB") runner._running_agents = {key_b: _FakeAgent()} assert runner._sibling_thread_run_keys(_thread_source("userA"), key_a) == [key_b] def test_sibling_excludes_callers_own_key(): runner = object.__new__(GatewayRunner) key_a = _per_user_key("userA") key_b = _per_user_key("userB") runner._running_agents = {key_a: _FakeAgent(), key_b: _FakeAgent()} assert runner._sibling_thread_run_keys(_thread_source("userA"), key_a) == [key_b] def test_sibling_skips_pending_sentinel(): runner = object.__new__(GatewayRunner) key_a = _per_user_key("userA") key_b = _per_user_key("userB") runner._running_agents = {key_b: _AGENT_PENDING_SENTINEL} assert runner._sibling_thread_run_keys(_thread_source("userA"), key_a) == [] def test_sibling_does_not_match_different_thread_same_chat(): # thr1 caller must not match a run in thr11 (prefix-collision guard). runner = object.__new__(GatewayRunner) key_a = _per_user_key("userA", thread_id="thr1") key_b_other = _per_user_key("userB", thread_id="thr11") runner._running_agents = {key_b_other: _FakeAgent()} assert runner._sibling_thread_run_keys(_thread_source("userA"), key_a) == [] def test_sibling_returns_empty_for_non_thread_source(): # Non-thread group/channel must NOT trigger the cross-user fallback. runner = object.__new__(GatewayRunner) nonthread = SessionSource( platform=Platform.DISCORD, chat_type="group", chat_id="chan1", user_id="userA" ) grp_b = build_session_key( SessionSource( platform=Platform.DISCORD, chat_type="group", chat_id="chan1", user_id="userB" ) ) runner._running_agents = {grp_b: _FakeAgent()} assert runner._sibling_thread_run_keys(nonthread, "agent:main:discord:group:chan1:userA") == [] # --------------------------------------------------------------------------- # _handle_stop_command fallback path # --------------------------------------------------------------------------- class _StoreEntry: def __init__(self, session_key): self.session_key = session_key class _FakeStore: def __init__(self, session_key): self._key = session_key def get_or_create_session(self, source): return _StoreEntry(self._key) @pytest.mark.asyncio async def test_stop_interrupts_sibling_thread_run_when_authorized(monkeypatch): runner = object.__new__(GatewayRunner) key_a = _per_user_key("userA") key_b = _per_user_key("userB") runner._running_agents = {key_b: _FakeAgent()} runner.session_store = _FakeStore(key_a) interrupted = [] async def _fake_interrupt(session_key, source, *, interrupt_reason, invalidation_reason): interrupted.append((session_key, interrupt_reason, invalidation_reason)) runner._interrupt_and_clear_session = _fake_interrupt runner._is_user_authorized = lambda source: True event = MessageEvent( text="/stop", message_type=MessageType.TEXT, source=_thread_source("userA") ) result = await runner._handle_stop_command(event) assert interrupted == [(key_b, _INTERRUPT_REASON_STOP, "stop_command_thread_sibling")] # EphemeralReply or str — both carry the "stopped" message, not "no_active". assert "no active" not in str(getattr(result, "text", result)).lower() @pytest.mark.asyncio async def test_stop_does_not_interrupt_sibling_when_unauthorized(monkeypatch): runner = object.__new__(GatewayRunner) key_a = _per_user_key("userA") key_b = _per_user_key("userB") runner._running_agents = {key_b: _FakeAgent()} runner.session_store = _FakeStore(key_a) interrupted = [] async def _fake_interrupt(session_key, source, *, interrupt_reason, invalidation_reason): interrupted.append(session_key) runner._interrupt_and_clear_session = _fake_interrupt runner._is_user_authorized = lambda source: False event = MessageEvent( text="/stop", message_type=MessageType.TEXT, source=_thread_source("userA") ) result = await runner._handle_stop_command(event) assert interrupted == [] assert "no active" in str(getattr(result, "text", result)).lower()