mirror of
https://github.com/nesquena/hermes-webui.git
synced 2026-06-04 16:10:24 +00:00
541c064a72
Use _merged_session_messages_for_display for is_messaging_session even in the metadata-only (messages=0) path. This ensures message_count and last_message_at match the full load path for Telegram / external messaging sessions that have stitched or duplicate rows in state.db + sidecar. Prevents spurious refresh loops, scroll resets, and open panel closures when resuming cross-surface sessions in the WebUI. No impact on CLI, non-messaging, or full-message paths. All 580 session tests pass. Fixes the root cause identified in the SessionDB / render interaction changes.
127 lines
5.4 KiB
Python
127 lines
5.4 KiB
Python
from urllib.parse import urlparse
|
|
|
|
|
|
def test_webui_session_metadata_load_skips_cli_metadata_scan(monkeypatch):
|
|
"""Opening a normal WebUI session should not scan imported CLI sessions."""
|
|
import api.routes as routes
|
|
from api.models import Session
|
|
|
|
session = Session(
|
|
session_id="webui_normal",
|
|
title="Normal WebUI chat",
|
|
messages=[{"role": "user", "content": "hello"}],
|
|
)
|
|
|
|
monkeypatch.setattr(routes, "get_session", lambda sid, metadata_only=False: session)
|
|
monkeypatch.setattr(routes, "_clear_stale_stream_state", lambda _session: None)
|
|
monkeypatch.setattr(routes, "redact_session_data", lambda payload: payload)
|
|
monkeypatch.setattr(routes, "j", lambda _handler, payload, status=200, extra_headers=None: payload)
|
|
monkeypatch.setattr(
|
|
routes,
|
|
"_lookup_cli_session_metadata",
|
|
lambda _sid: (_ for _ in ()).throw(AssertionError("normal WebUI loads should not scan CLI sessions")),
|
|
)
|
|
|
|
response = routes.handle_get(
|
|
object(),
|
|
urlparse("/api/session?session_id=webui_normal&messages=0&resolve_model=0"),
|
|
)
|
|
|
|
assert response["session"]["session_id"] == "webui_normal"
|
|
assert response["session"]["messages"] == []
|
|
|
|
|
|
def test_read_only_session_metadata_load_preserves_cli_metadata_lookup(monkeypatch):
|
|
"""Read-only imported sidecars still need CLI metadata for source identity."""
|
|
import api.routes as routes
|
|
from api.models import Session
|
|
|
|
session = Session(
|
|
session_id="readonly_sidecar",
|
|
title="Imported chat",
|
|
messages=[{"role": "user", "content": "hello"}],
|
|
read_only=True,
|
|
)
|
|
looked_up = []
|
|
|
|
monkeypatch.setattr(routes, "get_session", lambda sid, metadata_only=False: session)
|
|
monkeypatch.setattr(routes, "_clear_stale_stream_state", lambda _session: None)
|
|
monkeypatch.setattr(routes, "get_cli_session_messages", lambda _sid: [])
|
|
monkeypatch.setattr(routes, "redact_session_data", lambda payload: payload)
|
|
monkeypatch.setattr(routes, "j", lambda _handler, payload, status=200, extra_headers=None: payload)
|
|
|
|
def fake_lookup(sid):
|
|
looked_up.append(sid)
|
|
return {
|
|
"session_id": sid,
|
|
"read_only": True,
|
|
"source_label": "External Agent",
|
|
}
|
|
|
|
monkeypatch.setattr(routes, "_lookup_cli_session_metadata", fake_lookup)
|
|
|
|
response = routes.handle_get(
|
|
object(),
|
|
urlparse("/api/session?session_id=readonly_sidecar&messages=0&resolve_model=0"),
|
|
)
|
|
|
|
assert looked_up == ["readonly_sidecar"]
|
|
assert response["session"]["read_only"] is True
|
|
|
|
|
|
def test_messaging_session_metadata_load_preserves_cli_metadata_lookup(monkeypatch):
|
|
"""Messaging/imported sidecars still need CLI metadata for source identity."""
|
|
import api.routes as routes
|
|
from api.models import Session
|
|
|
|
session = Session(
|
|
session_id="messaging_sidecar",
|
|
title="Telegram chat",
|
|
messages=[{"role": "user", "content": "hello"}],
|
|
session_source="messaging",
|
|
raw_source="telegram",
|
|
)
|
|
looked_up = []
|
|
|
|
monkeypatch.setattr(routes, "get_session", lambda sid, metadata_only=False: session)
|
|
monkeypatch.setattr(routes, "_clear_stale_stream_state", lambda _session: None)
|
|
monkeypatch.setattr(routes, "get_cli_session_messages", lambda _sid: [])
|
|
monkeypatch.setattr(routes, "redact_session_data", lambda payload: payload)
|
|
monkeypatch.setattr(routes, "j", lambda _handler, payload, status=200, extra_headers=None: payload)
|
|
|
|
def fake_lookup(sid):
|
|
looked_up.append(sid)
|
|
return {
|
|
"session_id": sid,
|
|
"session_source": "messaging",
|
|
"raw_source": "telegram",
|
|
"source_label": "Telegram",
|
|
}
|
|
|
|
monkeypatch.setattr(routes, "_lookup_cli_session_metadata", fake_lookup)
|
|
|
|
response = routes.handle_get(
|
|
object(),
|
|
urlparse("/api/session?session_id=messaging_sidecar&messages=0&resolve_model=0"),
|
|
)
|
|
|
|
assert looked_up == ["messaging_sidecar"]
|
|
assert response["session"]["source_label"] == "Telegram"
|
|
|
|
|
|
def test_messaging_session_metadata_matches_full_display_merge(monkeypatch):
|
|
import api.routes as routes
|
|
from api.models import Session
|
|
sidecar = [{"role": "user", "content": "hi", "timestamp": 1000}, {"role": "assistant", "content": "ok", "timestamp": 1001}]
|
|
cli = sidecar + [{"role": "assistant", "content": "ok", "timestamp": 1001.7}]
|
|
session = Session(session_id="telegram_resume", title="Telegram", messages=sidecar, session_source="messaging", raw_source="telegram")
|
|
monkeypatch.setattr(routes, "get_session", lambda sid, metadata_only=False: session)
|
|
monkeypatch.setattr(routes, "_clear_stale_stream_state", lambda _session: None)
|
|
monkeypatch.setattr(routes, "_lookup_cli_session_metadata", lambda sid: {"session_id": sid, "session_source": "messaging", "raw_source": "telegram"})
|
|
monkeypatch.setattr(routes, "get_cli_session_messages", lambda _sid: cli)
|
|
monkeypatch.setattr(routes, "redact_session_data", lambda payload: payload)
|
|
monkeypatch.setattr(routes, "j", lambda _handler, payload, status=200, extra_headers=None: payload)
|
|
full = routes.handle_get(object(), urlparse("/api/session?session_id=telegram_resume&messages=1&resolve_model=0"))["session"]
|
|
meta = routes.handle_get(object(), urlparse("/api/session?session_id=telegram_resume&messages=0&resolve_model=0"))["session"]
|
|
assert (meta["message_count"], meta["last_message_at"]) == (full["message_count"], full["last_message_at"])
|