2026-04-21 18:52:26 -05:00
|
|
|
"""Regression tests for the TUI gateway's ``session.list`` handler.
|
|
|
|
|
|
2026-04-26 18:49:48 -07:00
|
|
|
History:
|
|
|
|
|
- The original implementation hardcoded an allow-list of known gateway
|
|
|
|
|
sources (``tui, cli, telegram, discord, slack, ...``). New or unlisted
|
|
|
|
|
sources (``acp``, ``webhook``, user-defined ``HERMES_SESSION_SOURCE``
|
|
|
|
|
values, newly-added platforms) were silently dropped from the resume
|
|
|
|
|
picker — users reported "lots of sessions are missing from browse
|
|
|
|
|
but exist in .hermes/sessions."
|
|
|
|
|
- The handler now deny-lists only the internal/noisy source ``tool``
|
|
|
|
|
(sub-agent runs) and surfaces every other source to the picker.
|
|
|
|
|
- The default ``limit`` raised from 20 to 200 so longer-running users
|
|
|
|
|
can scroll through their history without hitting an artificial cap.
|
2026-04-21 18:52:26 -05:00
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
|
|
from tui_gateway import server
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class _StubDB:
|
|
|
|
|
def __init__(self, rows):
|
|
|
|
|
self.rows = rows
|
|
|
|
|
self.calls: list[dict] = []
|
|
|
|
|
|
|
|
|
|
def list_sessions_rich(self, **kwargs):
|
|
|
|
|
self.calls.append(kwargs)
|
|
|
|
|
return list(self.rows)
|
|
|
|
|
|
|
|
|
|
|
2026-04-26 18:49:48 -07:00
|
|
|
def _call(limit: int | None = None):
|
|
|
|
|
params: dict = {}
|
|
|
|
|
if limit is not None:
|
|
|
|
|
params["limit"] = limit
|
2026-04-21 18:52:26 -05:00
|
|
|
return server.handle_request({
|
|
|
|
|
"id": "1",
|
|
|
|
|
"method": "session.list",
|
2026-04-26 18:49:48 -07:00
|
|
|
"params": params,
|
2026-04-21 18:52:26 -05:00
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
2026-04-26 18:49:48 -07:00
|
|
|
def test_session_list_surfaces_all_user_facing_sources(monkeypatch):
|
|
|
|
|
"""acp / webhook / custom sources should all appear; only ``tool`` is hidden."""
|
2026-04-21 18:52:26 -05:00
|
|
|
rows = [
|
|
|
|
|
{"id": "tui-1", "source": "tui", "started_at": 9},
|
|
|
|
|
{"id": "tool-1", "source": "tool", "started_at": 8},
|
|
|
|
|
{"id": "tg-1", "source": "telegram", "started_at": 7},
|
|
|
|
|
{"id": "acp-1", "source": "acp", "started_at": 6},
|
|
|
|
|
{"id": "cli-1", "source": "cli", "started_at": 5},
|
2026-04-26 18:49:48 -07:00
|
|
|
{"id": "webhook-1", "source": "webhook", "started_at": 4},
|
|
|
|
|
{"id": "custom-1", "source": "my-custom-source", "started_at": 3},
|
2026-04-21 18:52:26 -05:00
|
|
|
]
|
|
|
|
|
db = _StubDB(rows)
|
|
|
|
|
monkeypatch.setattr(server, "_get_db", lambda: db)
|
|
|
|
|
|
|
|
|
|
resp = _call(limit=10)
|
2026-04-26 18:49:48 -07:00
|
|
|
ids = [s["id"] for s in resp["result"]["sessions"]]
|
|
|
|
|
|
|
|
|
|
# Every human-facing source — including previously-hidden acp, webhook,
|
|
|
|
|
# and custom sources — must surface in the picker now.
|
|
|
|
|
assert "tg-1" in ids
|
|
|
|
|
assert "tui-1" in ids
|
|
|
|
|
assert "cli-1" in ids
|
|
|
|
|
assert "acp-1" in ids, "acp sessions were being hidden by the old allow-list"
|
|
|
|
|
assert "webhook-1" in ids, "webhook sessions were being hidden by the old allow-list"
|
|
|
|
|
assert "custom-1" in ids, "custom HERMES_SESSION_SOURCE values were being hidden"
|
2026-04-21 18:52:26 -05:00
|
|
|
|
2026-04-26 18:49:48 -07:00
|
|
|
# Only internal sub-agent runs stay hidden.
|
|
|
|
|
assert "tool-1" not in ids
|
2026-04-21 18:52:26 -05:00
|
|
|
|
|
|
|
|
|
2026-04-26 18:49:48 -07:00
|
|
|
def test_session_list_default_limit_is_200(monkeypatch):
|
|
|
|
|
"""Default limit should be wide enough for long-running users."""
|
2026-04-21 18:52:26 -05:00
|
|
|
db = _StubDB([{"id": "x", "source": "cli", "started_at": 1}])
|
|
|
|
|
monkeypatch.setattr(server, "_get_db", lambda: db)
|
|
|
|
|
|
2026-04-26 18:49:48 -07:00
|
|
|
_call() # no explicit limit
|
|
|
|
|
# fetch_limit = max(limit * 2, 200); limit defaults to 200, so 400.
|
|
|
|
|
assert db.calls[0].get("limit") == 400, db.calls[0]
|
|
|
|
|
|
2026-04-21 18:52:26 -05:00
|
|
|
|
2026-04-26 18:49:48 -07:00
|
|
|
def test_session_list_respects_explicit_limit(monkeypatch):
|
|
|
|
|
db = _StubDB([{"id": "x", "source": "cli", "started_at": 1}])
|
|
|
|
|
monkeypatch.setattr(server, "_get_db", lambda: db)
|
|
|
|
|
|
|
|
|
|
_call(limit=10)
|
|
|
|
|
# fetch_limit = max(limit * 2, 200) = 200 when limit is small.
|
|
|
|
|
assert db.calls[0].get("limit") == 200, db.calls[0]
|
2026-04-21 18:52:26 -05:00
|
|
|
|
|
|
|
|
|
|
|
|
|
def test_session_list_preserves_ordering_after_filter(monkeypatch):
|
|
|
|
|
rows = [
|
|
|
|
|
{"id": "newest", "source": "telegram", "started_at": 5},
|
|
|
|
|
{"id": "internal", "source": "tool", "started_at": 4},
|
|
|
|
|
{"id": "middle", "source": "tui", "started_at": 3},
|
2026-04-26 18:49:48 -07:00
|
|
|
{"id": "also-visible", "source": "webhook", "started_at": 2},
|
2026-04-21 18:52:26 -05:00
|
|
|
{"id": "oldest", "source": "discord", "started_at": 1},
|
|
|
|
|
]
|
|
|
|
|
monkeypatch.setattr(server, "_get_db", lambda: _StubDB(rows))
|
|
|
|
|
|
|
|
|
|
resp = _call()
|
|
|
|
|
ids = [s["id"] for s in resp["result"]["sessions"]]
|
|
|
|
|
|
2026-04-26 18:49:48 -07:00
|
|
|
assert ids == ["newest", "middle", "also-visible", "oldest"]
|