mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-04-28 06:51:16 +08:00
149 lines
5.9 KiB
Python
149 lines
5.9 KiB
Python
|
|
"""Regression tests for #15165 — gateway session shutdown must pass the
|
||
|
|
agent's conversation transcript to ``shutdown_memory_provider`` so memory
|
||
|
|
providers' ``on_session_end`` hooks see the real messages instead of an
|
||
|
|
empty list.
|
||
|
|
|
||
|
|
Before the fix, ``_cleanup_agent_resources`` called
|
||
|
|
``agent.shutdown_memory_provider()`` with no arguments, which in turn
|
||
|
|
invoked ``on_session_end([])`` on every memory provider. Providers with
|
||
|
|
an empty-guard (Holographic, Hindsight, etc.) exited early and never
|
||
|
|
persisted the session's facts, so the next gateway start-up surfaced no
|
||
|
|
memories from the prior conversation.
|
||
|
|
|
||
|
|
The fix reads ``agent._session_messages`` (set on ``AIAgent.__init__``
|
||
|
|
and refreshed every turn via ``_persist_session``) and forwards it to
|
||
|
|
``shutdown_memory_provider``. Test stubs built via ``object.__new__``
|
||
|
|
or plain ``MagicMock()`` still exercise the legacy no-arg path, so the
|
||
|
|
change is backward-compatible with existing suites.
|
||
|
|
"""
|
||
|
|
|
||
|
|
from __future__ import annotations
|
||
|
|
|
||
|
|
import sys
|
||
|
|
import types
|
||
|
|
from unittest.mock import MagicMock
|
||
|
|
|
||
|
|
import pytest
|
||
|
|
|
||
|
|
|
||
|
|
@pytest.fixture(autouse=True)
|
||
|
|
def _mock_dotenv(monkeypatch):
|
||
|
|
"""gateway.run imports dotenv at module load; stub so tests run bare."""
|
||
|
|
fake = types.ModuleType("dotenv")
|
||
|
|
fake.load_dotenv = lambda *a, **kw: None
|
||
|
|
monkeypatch.setitem(sys.modules, "dotenv", fake)
|
||
|
|
|
||
|
|
|
||
|
|
def _make_runner():
|
||
|
|
from gateway.run import GatewayRunner
|
||
|
|
|
||
|
|
runner = object.__new__(GatewayRunner)
|
||
|
|
return runner
|
||
|
|
|
||
|
|
|
||
|
|
# A lightweight stand-in for AIAgent so ``isinstance(..., list)`` correctly
|
||
|
|
# discriminates between "attribute set to a list" and "attribute absent /
|
||
|
|
# MagicMock auto-synthesised". Using MagicMock directly for the agent
|
||
|
|
# would also work for the populated case, but attribute access on a
|
||
|
|
# MagicMock always yields a child MagicMock — we want a real Python
|
||
|
|
# object we can shape per-test.
|
||
|
|
class _FakeAgent:
|
||
|
|
def __init__(self, session_messages=None, has_shutdown=True):
|
||
|
|
if session_messages is not None:
|
||
|
|
self._session_messages = session_messages
|
||
|
|
if has_shutdown:
|
||
|
|
self.shutdown_memory_provider = MagicMock()
|
||
|
|
self.close = MagicMock()
|
||
|
|
|
||
|
|
|
||
|
|
class TestCleanupAgentResourcesPassesMessages:
|
||
|
|
"""_cleanup_agent_resources forwards the agent's session messages."""
|
||
|
|
|
||
|
|
def test_populated_messages_forwarded(self):
|
||
|
|
"""Real-world path: an agent that ran a turn has a populated
|
||
|
|
``_session_messages`` list and the cleanup call forwards it."""
|
||
|
|
runner = _make_runner()
|
||
|
|
transcript = [
|
||
|
|
{"role": "user", "content": "remember my dog is named Biscuit"},
|
||
|
|
{"role": "assistant", "content": "Got it — Biscuit."},
|
||
|
|
]
|
||
|
|
agent = _FakeAgent(session_messages=transcript)
|
||
|
|
|
||
|
|
runner._cleanup_agent_resources(agent)
|
||
|
|
|
||
|
|
# The fix must call shutdown_memory_provider with the exact list
|
||
|
|
# identity — providers iterate it to extract facts.
|
||
|
|
agent.shutdown_memory_provider.assert_called_once_with(transcript)
|
||
|
|
|
||
|
|
def test_empty_list_still_forwarded(self):
|
||
|
|
"""An agent that initialised but ran no turns has an empty list
|
||
|
|
on ``_session_messages``. Forwarding it (rather than falling
|
||
|
|
through to the no-arg path) makes the absence of content
|
||
|
|
explicit to providers and matches the pre-fix observable
|
||
|
|
behaviour (``on_session_end([])``)."""
|
||
|
|
runner = _make_runner()
|
||
|
|
agent = _FakeAgent(session_messages=[])
|
||
|
|
|
||
|
|
runner._cleanup_agent_resources(agent)
|
||
|
|
|
||
|
|
agent.shutdown_memory_provider.assert_called_once_with([])
|
||
|
|
|
||
|
|
def test_missing_attribute_falls_back_to_no_arg(self):
|
||
|
|
"""Test stubs built via ``object.__new__(AIAgent)`` skip
|
||
|
|
``__init__`` and therefore have no ``_session_messages``
|
||
|
|
attribute. The fix must not explode — it falls back to the
|
||
|
|
legacy no-arg call so existing suites keep passing."""
|
||
|
|
runner = _make_runner()
|
||
|
|
agent = _FakeAgent(session_messages=None) # attribute not set
|
||
|
|
|
||
|
|
runner._cleanup_agent_resources(agent)
|
||
|
|
|
||
|
|
agent.shutdown_memory_provider.assert_called_once_with()
|
||
|
|
|
||
|
|
def test_non_list_attribute_falls_back_to_no_arg(self):
|
||
|
|
"""A MagicMock-based agent auto-synthesises ``_session_messages``
|
||
|
|
as a nested MagicMock. ``isinstance(mock, list)`` is False, so
|
||
|
|
we fall back to the no-arg path rather than passing a garbage
|
||
|
|
value to providers that expect ``List[Dict]``."""
|
||
|
|
runner = _make_runner()
|
||
|
|
agent = MagicMock()
|
||
|
|
# No explicit _session_messages assignment — MagicMock will
|
||
|
|
# synthesise one on access.
|
||
|
|
|
||
|
|
runner._cleanup_agent_resources(agent)
|
||
|
|
|
||
|
|
agent.shutdown_memory_provider.assert_called_once_with()
|
||
|
|
|
||
|
|
def test_provider_exception_is_swallowed(self):
|
||
|
|
"""Provider teardown must be best-effort — a raising
|
||
|
|
``shutdown_memory_provider`` must not prevent ``close()`` from
|
||
|
|
running (tool resource leak is worse than a missed memory
|
||
|
|
flush)."""
|
||
|
|
runner = _make_runner()
|
||
|
|
agent = _FakeAgent(session_messages=[{"role": "user", "content": "x"}])
|
||
|
|
agent.shutdown_memory_provider.side_effect = RuntimeError("boom")
|
||
|
|
|
||
|
|
# Must not raise.
|
||
|
|
runner._cleanup_agent_resources(agent)
|
||
|
|
|
||
|
|
# close() still invoked after the swallowed exception.
|
||
|
|
agent.close.assert_called_once()
|
||
|
|
|
||
|
|
def test_none_agent_is_noop(self):
|
||
|
|
"""Defensive: None agent short-circuits (idle sweeps may
|
||
|
|
observe a None entry in the cache during eviction races)."""
|
||
|
|
runner = _make_runner()
|
||
|
|
# Must not raise.
|
||
|
|
runner._cleanup_agent_resources(None)
|
||
|
|
|
||
|
|
def test_agent_without_shutdown_method_is_tolerated(self):
|
||
|
|
"""An agent without ``shutdown_memory_provider`` (old test
|
||
|
|
stub, partial mock) must still have ``close()`` called."""
|
||
|
|
runner = _make_runner()
|
||
|
|
agent = _FakeAgent(has_shutdown=False)
|
||
|
|
# No _session_messages either, to exercise the hasattr guard.
|
||
|
|
|
||
|
|
runner._cleanup_agent_resources(agent)
|
||
|
|
|
||
|
|
agent.close.assert_called_once()
|