hermes-agent/tests/gateway/test_session_load_bool.py
kshitijk4poor 9c994377ed fix(gateway): skip non-dict entries in session loading (#46994)
Corrupted sessions.json entries (e.g. a bare bool where a dict is
expected) caused TypeError on 'origin' in data' which escaped the
(ValueError, KeyError) inner except and aborted loading ALL remaining
sessions, not just the corrupted one.

Two-layer fix:
- Loop level: isinstance(entry_data, dict) guard before from_dict
- from_dict: isinstance(data['origin'], dict) instead of bare truthiness
- Added TypeError to the inner except as defense-in-depth

Closes #46994
2026-06-25 01:26:13 +05:30

126 lines
4.7 KiB
Python

"""Regression tests for issue #46994.
Corrupted sessions.json entries (e.g. a bare bool where a dict is expected)
must not crash the entire session loading loop. The TypeError from
`"origin" in True` escaped the (ValueError, KeyError) except and aborted
loading ALL remaining sessions, not just the corrupted one.
"""
import json
import threading
from pathlib import Path
from gateway.session import SessionStore
class TestSessionLoadBoolCorruption:
"""Verify that non-dict entries in sessions.json are skipped, not fatal."""
def _make_store(self, tmp_path: Path, sessions_data: dict) -> SessionStore:
"""Create a SessionStore with a pre-populated sessions.json."""
sessions_dir = tmp_path / "sessions"
sessions_dir.mkdir(parents=True)
(sessions_dir / "sessions.json").write_text(
json.dumps(sessions_data), encoding="utf-8"
)
# SessionStore requires a config object with session reset policy
class FakeConfig:
session_idle_ttl = 0
session_daily_ttl = 0
group_sessions_per_user = True
thread_sessions_per_user = False
multiplex_profiles = False
def get_reset_policy(self, *a, **kw):
return None
store = SessionStore.__new__(SessionStore)
store.sessions_dir = sessions_dir
store._entries = {}
store._loaded = False
store._lock = threading.RLock()
store.config = FakeConfig()
store._has_active_processes_fn = None
return store
def _valid_entry(self, session_id: str = "20260101_120000_abc12345") -> dict:
return {
"session_key": "agent:main:telegram:dm:123456",
"session_id": session_id,
"created_at": "2026-01-01T12:00:00",
"updated_at": "2026-01-01T12:30:00",
"origin": {
"platform": "telegram",
"chat_id": "123456",
"chat_type": "dm",
},
}
def test_bool_entry_skipped_not_fatal(self, tmp_path):
"""A bool entry must not crash the loop or block other sessions."""
data = {
"_README": "test sentinel",
"corrupted_key": True,
"valid_key": self._valid_entry(),
}
store = self._make_store(tmp_path, data)
store._ensure_loaded()
# The valid entry must still be loaded
assert "valid_key" in store._entries
assert store._entries["valid_key"].session_id == "20260101_120000_abc12345"
# The corrupted entry must NOT be loaded
assert "corrupted_key" not in store._entries
def test_string_entry_skipped(self, tmp_path):
"""A string entry must also be skipped without crashing."""
data = {
"bad_string": "not a dict",
"valid_key": self._valid_entry("20260101_130000_def67890"),
}
store = self._make_store(tmp_path, data)
store._ensure_loaded()
assert "valid_key" in store._entries
assert "bad_string" not in store._entries
def test_all_corrupted_entries_does_not_crash(self, tmp_path):
"""Multiple corrupted entries must not produce an unhandled exception."""
data = {
"bad1": True,
"bad2": 42,
"bad3": "string",
"bad4": [1, 2, 3],
}
store = self._make_store(tmp_path, data)
store._ensure_loaded()
assert len(store._entries) == 0
def test_origin_not_dict_skipped(self, tmp_path):
"""If origin is present but not a dict, from_dict must not crash."""
entry = self._valid_entry()
entry["origin"] = True # bool instead of dict
data = {"key_with_bad_origin": entry}
store = self._make_store(tmp_path, data)
store._ensure_loaded()
# Entry should still load, just with origin=None
assert "key_with_bad_origin" in store._entries
assert store._entries["key_with_bad_origin"].origin is None
def test_typeerror_in_from_dict_caught(self, tmp_path):
"""TypeError from from_dict must be caught, not escape to outer except."""
# An entry with a non-dict, non-bool value that could trigger TypeError
# in from_dict's datetime.fromisoformat or Platform() calls
entry = self._valid_entry()
entry["created_at"] = 12345 # int instead of ISO string
data = {
"bad_date": entry,
"valid_key": self._valid_entry(),
}
store = self._make_store(tmp_path, data)
store._ensure_loaded()
# The valid entry must still load despite the bad one
assert "valid_key" in store._entries
assert "bad_date" not in store._entries