mirror of
https://github.com/HKUDS/nanobot.git
synced 2026-04-11 13:43:37 +00:00
When a user is idle for longer than a configured TTL, nanobot **proactively** compresses the session context into a summary. This reduces token cost and first-token latency when the user returns — instead of re-processing a long stale context with an expired KV cache, the model receives a compact summary and fresh input.
932 lines
36 KiB
Python
932 lines
36 KiB
Python
"""Tests for auto compact (idle TTL) feature."""
|
|
|
|
import asyncio
|
|
from datetime import datetime, timedelta
|
|
from unittest.mock import AsyncMock, MagicMock
|
|
from pathlib import Path
|
|
|
|
import pytest
|
|
|
|
from nanobot.agent.loop import AgentLoop
|
|
from nanobot.bus.events import InboundMessage
|
|
from nanobot.bus.queue import MessageBus
|
|
from nanobot.config.schema import AgentDefaults
|
|
from nanobot.command import CommandContext
|
|
from nanobot.providers.base import LLMResponse
|
|
|
|
|
|
def _make_loop(tmp_path: Path, session_ttl_minutes: int = 15) -> AgentLoop:
|
|
"""Create a minimal AgentLoop for testing."""
|
|
bus = MessageBus()
|
|
provider = MagicMock()
|
|
provider.get_default_model.return_value = "test-model"
|
|
provider.estimate_prompt_tokens.return_value = (10_000, "test")
|
|
provider.chat_with_retry = AsyncMock(return_value=LLMResponse(content="ok", tool_calls=[]))
|
|
provider.generation.max_tokens = 4096
|
|
loop = AgentLoop(
|
|
bus=bus,
|
|
provider=provider,
|
|
workspace=tmp_path,
|
|
model="test-model",
|
|
context_window_tokens=128_000,
|
|
session_ttl_minutes=session_ttl_minutes,
|
|
)
|
|
loop.tools.get_definitions = MagicMock(return_value=[])
|
|
return loop
|
|
|
|
|
|
class TestSessionTTLConfig:
|
|
"""Test session TTL configuration."""
|
|
|
|
def test_default_ttl_is_zero(self):
|
|
"""Default TTL should be 0 (disabled)."""
|
|
defaults = AgentDefaults()
|
|
assert defaults.session_ttl_minutes == 0
|
|
|
|
def test_custom_ttl(self):
|
|
"""Custom TTL should be stored correctly."""
|
|
defaults = AgentDefaults(session_ttl_minutes=30)
|
|
assert defaults.session_ttl_minutes == 30
|
|
|
|
|
|
class TestAgentLoopTTLParam:
|
|
"""Test that AutoCompact receives and stores session_ttl_minutes."""
|
|
|
|
def test_loop_stores_ttl(self, tmp_path):
|
|
"""AutoCompact should store the TTL value."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=25)
|
|
assert loop.auto_compact._ttl == 25
|
|
|
|
def test_loop_default_ttl_zero(self, tmp_path):
|
|
"""AutoCompact default TTL should be 0 (disabled)."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=0)
|
|
assert loop.auto_compact._ttl == 0
|
|
|
|
|
|
class TestAutoCompact:
|
|
"""Test the _archive method."""
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_is_expired_boundary(self, tmp_path):
|
|
"""Exactly at TTL boundary should be expired (>= not >)."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
ts = datetime.now() - timedelta(minutes=15)
|
|
assert loop.auto_compact._is_expired(ts) is True
|
|
ts2 = datetime.now() - timedelta(minutes=14, seconds=59)
|
|
assert loop.auto_compact._is_expired(ts2) is False
|
|
await loop.close_mcp()
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_is_expired_string_timestamp(self, tmp_path):
|
|
"""_is_expired should parse ISO string timestamps."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
ts = (datetime.now() - timedelta(minutes=20)).isoformat()
|
|
assert loop.auto_compact._is_expired(ts) is True
|
|
assert loop.auto_compact._is_expired(None) is False
|
|
assert loop.auto_compact._is_expired("") is False
|
|
await loop.close_mcp()
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_check_expired_only_archives_expired_sessions(self, tmp_path):
|
|
"""With multiple sessions, only the expired one should be archived."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
# Expired session
|
|
s1 = loop.sessions.get_or_create("cli:expired")
|
|
s1.add_message("user", "old")
|
|
s1.updated_at = datetime.now() - timedelta(minutes=20)
|
|
loop.sessions.save(s1)
|
|
# Active session
|
|
s2 = loop.sessions.get_or_create("cli:active")
|
|
s2.add_message("user", "recent")
|
|
loop.sessions.save(s2)
|
|
|
|
async def _fake_archive(messages):
|
|
return True
|
|
|
|
loop.consolidator.archive = _fake_archive
|
|
loop.auto_compact.check_expired(loop._schedule_background)
|
|
await asyncio.sleep(0.1)
|
|
|
|
active_after = loop.sessions.get_or_create("cli:active")
|
|
assert len(active_after.messages) == 1
|
|
assert active_after.messages[0]["content"] == "recent"
|
|
await loop.close_mcp()
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_auto_compact_archives_and_clears(self, tmp_path):
|
|
"""_archive should archive un-consolidated messages and clear session."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
for i in range(4):
|
|
session.add_message("user", f"msg{i}")
|
|
session.add_message("assistant", f"resp{i}")
|
|
loop.sessions.save(session)
|
|
|
|
archived_messages = []
|
|
|
|
async def _fake_archive(messages):
|
|
archived_messages.extend(messages)
|
|
return True
|
|
|
|
loop.consolidator.archive = _fake_archive
|
|
|
|
await loop.auto_compact._archive("cli:test")
|
|
|
|
assert len(archived_messages) == 8
|
|
session_after = loop.sessions.get_or_create("cli:test")
|
|
assert len(session_after.messages) == 0
|
|
await loop.close_mcp()
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_auto_compact_stores_summary(self, tmp_path):
|
|
"""_archive should store the summary in _summaries."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
session.add_message("user", "hello")
|
|
session.add_message("assistant", "hi there")
|
|
loop.sessions.save(session)
|
|
|
|
async def _fake_archive(messages):
|
|
return True
|
|
|
|
loop.consolidator.archive = _fake_archive
|
|
loop.consolidator.get_last_history_entry = lambda: {
|
|
"cursor": 1, "timestamp": "2026-01-01 00:00", "content": "User said hello.",
|
|
}
|
|
|
|
await loop.auto_compact._archive("cli:test")
|
|
|
|
entry = loop.auto_compact._summaries.get("cli:test")
|
|
assert entry is not None
|
|
assert entry[0] == "User said hello."
|
|
session_after = loop.sessions.get_or_create("cli:test")
|
|
assert len(session_after.messages) == 0
|
|
await loop.close_mcp()
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_auto_compact_empty_session(self, tmp_path):
|
|
"""_archive on empty session should not archive."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
|
|
archive_called = False
|
|
|
|
async def _fake_archive(messages):
|
|
nonlocal archive_called
|
|
archive_called = True
|
|
return True
|
|
|
|
loop.consolidator.archive = _fake_archive
|
|
|
|
await loop.auto_compact._archive("cli:test")
|
|
|
|
assert not archive_called
|
|
session_after = loop.sessions.get_or_create("cli:test")
|
|
assert len(session_after.messages) == 0
|
|
await loop.close_mcp()
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_auto_compact_respects_last_consolidated(self, tmp_path):
|
|
"""_archive should only archive un-consolidated messages."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
for i in range(10):
|
|
session.add_message("user", f"msg{i}")
|
|
session.add_message("assistant", f"resp{i}")
|
|
session.last_consolidated = 18
|
|
loop.sessions.save(session)
|
|
|
|
archived_count = 0
|
|
|
|
async def _fake_archive(messages):
|
|
nonlocal archived_count
|
|
archived_count = len(messages)
|
|
return True
|
|
|
|
loop.consolidator.archive = _fake_archive
|
|
|
|
await loop.auto_compact._archive("cli:test")
|
|
|
|
assert archived_count == 2
|
|
await loop.close_mcp()
|
|
|
|
|
|
class TestAutoCompactIdleDetection:
|
|
"""Test idle detection triggers auto-new in _process_message."""
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_no_auto_compact_when_ttl_disabled(self, tmp_path):
|
|
"""No auto-new should happen when TTL is 0 (disabled)."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=0)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
session.add_message("user", "old message")
|
|
session.updated_at = datetime.now() - timedelta(minutes=30)
|
|
loop.sessions.save(session)
|
|
|
|
msg = InboundMessage(channel="cli", sender_id="user", chat_id="test", content="new msg")
|
|
await loop._process_message(msg)
|
|
|
|
session_after = loop.sessions.get_or_create("cli:test")
|
|
assert any(m["content"] == "old message" for m in session_after.messages)
|
|
await loop.close_mcp()
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_auto_compact_triggers_on_idle(self, tmp_path):
|
|
"""Proactive auto-new archives expired session; _process_message reloads it."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
session.add_message("user", "old message")
|
|
session.updated_at = datetime.now() - timedelta(minutes=20)
|
|
loop.sessions.save(session)
|
|
|
|
archived_messages = []
|
|
|
|
async def _fake_archive(messages):
|
|
archived_messages.extend(messages)
|
|
return True
|
|
|
|
loop.consolidator.archive = _fake_archive
|
|
loop.consolidator.get_last_history_entry = lambda: {
|
|
"cursor": 1, "timestamp": "2026-01-01 00:00", "content": "Summary.",
|
|
}
|
|
|
|
# Simulate proactive archive completing before message arrives
|
|
await loop.auto_compact._archive("cli:test")
|
|
|
|
msg = InboundMessage(channel="cli", sender_id="user", chat_id="test", content="new msg")
|
|
await loop._process_message(msg)
|
|
|
|
session_after = loop.sessions.get_or_create("cli:test")
|
|
assert not any(m["content"] == "old message" for m in session_after.messages)
|
|
assert any(m["content"] == "new msg" for m in session_after.messages)
|
|
await loop.close_mcp()
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_no_auto_compact_when_active(self, tmp_path):
|
|
"""No auto-new should happen when session is recently active."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
session.add_message("user", "recent message")
|
|
loop.sessions.save(session)
|
|
|
|
msg = InboundMessage(channel="cli", sender_id="user", chat_id="test", content="new msg")
|
|
await loop._process_message(msg)
|
|
|
|
session_after = loop.sessions.get_or_create("cli:test")
|
|
assert any(m["content"] == "recent message" for m in session_after.messages)
|
|
await loop.close_mcp()
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_auto_compact_does_not_affect_priority_commands(self, tmp_path):
|
|
"""Priority commands (/stop, /restart) bypass _process_message entirely via run()."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
session.add_message("user", "old message")
|
|
session.updated_at = datetime.now() - timedelta(minutes=20)
|
|
loop.sessions.save(session)
|
|
|
|
# Priority commands are dispatched in run() before _process_message is called.
|
|
# Simulate that path directly via dispatch_priority.
|
|
raw = "/stop"
|
|
msg = InboundMessage(channel="cli", sender_id="user", chat_id="test", content=raw)
|
|
ctx = CommandContext(msg=msg, session=session, key="cli:test", raw=raw, loop=loop)
|
|
result = await loop.commands.dispatch_priority(ctx)
|
|
assert result is not None
|
|
assert "stopped" in result.content.lower() or "no active task" in result.content.lower()
|
|
|
|
# Session should be untouched since priority commands skip _process_message
|
|
session_after = loop.sessions.get_or_create("cli:test")
|
|
assert any(m["content"] == "old message" for m in session_after.messages)
|
|
await loop.close_mcp()
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_auto_compact_with_slash_new(self, tmp_path):
|
|
"""Auto-new fires before /new dispatches; session is cleared twice but idempotent."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
for i in range(4):
|
|
session.add_message("user", f"msg{i}")
|
|
session.add_message("assistant", f"resp{i}")
|
|
session.updated_at = datetime.now() - timedelta(minutes=20)
|
|
loop.sessions.save(session)
|
|
|
|
async def _fake_archive(messages):
|
|
return True
|
|
|
|
loop.consolidator.archive = _fake_archive
|
|
|
|
msg = InboundMessage(channel="cli", sender_id="user", chat_id="test", content="/new")
|
|
response = await loop._process_message(msg)
|
|
|
|
assert response is not None
|
|
assert "new session started" in response.content.lower()
|
|
|
|
session_after = loop.sessions.get_or_create("cli:test")
|
|
# Session is empty (auto-new archived and cleared, /new cleared again)
|
|
assert len(session_after.messages) == 0
|
|
await loop.close_mcp()
|
|
|
|
|
|
class TestAutoCompactSystemMessages:
|
|
"""Test that auto-new also works for system messages."""
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_auto_compact_triggers_for_system_messages(self, tmp_path):
|
|
"""Proactive auto-new archives expired session; system messages reload it."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
session.add_message("user", "old message from subagent context")
|
|
session.updated_at = datetime.now() - timedelta(minutes=20)
|
|
loop.sessions.save(session)
|
|
|
|
async def _fake_archive(messages):
|
|
return True
|
|
|
|
loop.consolidator.archive = _fake_archive
|
|
loop.consolidator.get_last_history_entry = lambda: {
|
|
"cursor": 1, "timestamp": "2026-01-01 00:00", "content": "Summary.",
|
|
}
|
|
|
|
# Simulate proactive archive completing before system message arrives
|
|
await loop.auto_compact._archive("cli:test")
|
|
|
|
msg = InboundMessage(
|
|
channel="system", sender_id="subagent", chat_id="cli:test",
|
|
content="subagent result",
|
|
)
|
|
await loop._process_message(msg)
|
|
|
|
session_after = loop.sessions.get_or_create("cli:test")
|
|
assert not any(
|
|
m["content"] == "old message from subagent context"
|
|
for m in session_after.messages
|
|
)
|
|
await loop.close_mcp()
|
|
|
|
|
|
class TestAutoCompactEdgeCases:
|
|
"""Edge cases for auto session new."""
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_auto_compact_with_nothing_summary(self, tmp_path):
|
|
"""Auto-new should not inject when archive produces '(nothing)'."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
session.add_message("user", "thanks")
|
|
session.add_message("assistant", "you're welcome")
|
|
session.updated_at = datetime.now() - timedelta(minutes=20)
|
|
loop.sessions.save(session)
|
|
|
|
loop.provider.chat_with_retry = AsyncMock(
|
|
return_value=LLMResponse(content="(nothing)", tool_calls=[])
|
|
)
|
|
|
|
await loop.auto_compact._archive("cli:test")
|
|
|
|
session_after = loop.sessions.get_or_create("cli:test")
|
|
assert len(session_after.messages) == 0
|
|
# "(nothing)" summary should not be stored
|
|
assert "cli:test" not in loop.auto_compact._summaries
|
|
|
|
await loop.close_mcp()
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_auto_compact_archive_failure_still_clears(self, tmp_path):
|
|
"""Auto-new should clear session even if LLM archive fails (raw_archive fallback)."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
session.add_message("user", "important data")
|
|
session.updated_at = datetime.now() - timedelta(minutes=20)
|
|
loop.sessions.save(session)
|
|
|
|
loop.provider.chat_with_retry = AsyncMock(side_effect=Exception("API down"))
|
|
|
|
# Should not raise
|
|
await loop.auto_compact._archive("cli:test")
|
|
|
|
session_after = loop.sessions.get_or_create("cli:test")
|
|
# Session should be cleared (archive falls back to raw dump)
|
|
assert len(session_after.messages) == 0
|
|
|
|
await loop.close_mcp()
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_auto_compact_preserves_runtime_checkpoint_before_check(self, tmp_path):
|
|
"""Runtime checkpoint is restored; proactive archive handles the expired session."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
session.metadata[AgentLoop._RUNTIME_CHECKPOINT_KEY] = {
|
|
"assistant_message": {"role": "assistant", "content": "interrupted response"},
|
|
"completed_tool_results": [],
|
|
"pending_tool_calls": [],
|
|
}
|
|
session.add_message("user", "previous message")
|
|
session.updated_at = datetime.now() - timedelta(minutes=20)
|
|
loop.sessions.save(session)
|
|
|
|
archived_messages = []
|
|
|
|
async def _fake_archive(messages):
|
|
archived_messages.extend(messages)
|
|
return True
|
|
|
|
loop.consolidator.archive = _fake_archive
|
|
loop.consolidator.get_last_history_entry = lambda: {
|
|
"cursor": 1, "timestamp": "2026-01-01 00:00", "content": "Summary.",
|
|
}
|
|
|
|
# Simulate proactive archive completing before message arrives
|
|
await loop.auto_compact._archive("cli:test")
|
|
|
|
msg = InboundMessage(channel="cli", sender_id="user", chat_id="test", content="continue")
|
|
await loop._process_message(msg)
|
|
|
|
# The checkpoint-restored message should have been archived by proactive path
|
|
assert len(archived_messages) >= 1
|
|
|
|
await loop.close_mcp()
|
|
|
|
|
|
class TestAutoCompactIntegration:
|
|
"""End-to-end test of auto session new feature."""
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_full_lifecycle(self, tmp_path):
|
|
"""
|
|
Full lifecycle: messages -> idle -> auto-new -> archive -> clear -> summary injected as runtime context.
|
|
"""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
|
|
# Phase 1: User has a conversation
|
|
session.add_message("user", "I'm learning English, teach me past tense")
|
|
session.add_message("assistant", "Past tense is used for actions completed in the past...")
|
|
session.add_message("user", "Give me an example")
|
|
session.add_message("assistant", '"I walked to the store yesterday."')
|
|
loop.sessions.save(session)
|
|
|
|
# Phase 2: Time passes (simulate idle)
|
|
session.updated_at = datetime.now() - timedelta(minutes=20)
|
|
loop.sessions.save(session)
|
|
|
|
# Phase 3: User returns with a new message
|
|
loop.provider.chat_with_retry = AsyncMock(
|
|
return_value=LLMResponse(
|
|
content="User is learning English past tense. Example: 'I walked to the store yesterday.'",
|
|
tool_calls=[],
|
|
)
|
|
)
|
|
|
|
msg = InboundMessage(
|
|
channel="cli", sender_id="user", chat_id="test",
|
|
content="Let's continue, teach me present perfect",
|
|
)
|
|
response = await loop._process_message(msg)
|
|
|
|
# Phase 4: Verify
|
|
session_after = loop.sessions.get_or_create("cli:test")
|
|
|
|
# Old messages should be gone
|
|
assert not any(
|
|
"past tense is used" in str(m.get("content", "")) for m in session_after.messages
|
|
)
|
|
|
|
# Summary should NOT be persisted in session (ephemeral, one-shot)
|
|
assert not any(
|
|
"[Resumed Session]" in str(m.get("content", "")) for m in session_after.messages
|
|
)
|
|
# Runtime context end marker should NOT be persisted
|
|
assert not any(
|
|
"[/Runtime Context]" in str(m.get("content", "")) for m in session_after.messages
|
|
)
|
|
|
|
# Pending summary should be consumed (one-shot)
|
|
assert "cli:test" not in loop.auto_compact._summaries
|
|
|
|
# The new message should be processed (response exists)
|
|
assert response is not None
|
|
|
|
await loop.close_mcp()
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_multi_paragraph_user_message_preserved(self, tmp_path):
|
|
"""Multi-paragraph user messages must be fully preserved after auto-new."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
session.add_message("user", "old message")
|
|
session.updated_at = datetime.now() - timedelta(minutes=20)
|
|
loop.sessions.save(session)
|
|
|
|
async def _fake_archive(messages):
|
|
return True
|
|
|
|
loop.consolidator.archive = _fake_archive
|
|
loop.consolidator.get_last_history_entry = lambda: {
|
|
"cursor": 1, "timestamp": "2026-01-01 00:00", "content": "Summary.",
|
|
}
|
|
|
|
# Simulate proactive archive completing before message arrives
|
|
await loop.auto_compact._archive("cli:test")
|
|
|
|
msg = InboundMessage(
|
|
channel="cli", sender_id="user", chat_id="test",
|
|
content="Paragraph one\n\nParagraph two\n\nParagraph three",
|
|
)
|
|
await loop._process_message(msg)
|
|
|
|
session_after = loop.sessions.get_or_create("cli:test")
|
|
user_msgs = [m for m in session_after.messages if m.get("role") == "user"]
|
|
assert len(user_msgs) >= 1
|
|
# All three paragraphs must be preserved
|
|
persisted = user_msgs[-1]["content"]
|
|
assert "Paragraph one" in persisted
|
|
assert "Paragraph two" in persisted
|
|
assert "Paragraph three" in persisted
|
|
# No runtime context markers in persisted message
|
|
assert "[Runtime Context" not in persisted
|
|
assert "[/Runtime Context]" not in persisted
|
|
await loop.close_mcp()
|
|
|
|
|
|
class TestProactiveAutoCompact:
|
|
"""Test proactive auto-new on idle ticks (TimeoutError path in run loop)."""
|
|
|
|
@staticmethod
|
|
async def _run_check_expired(loop):
|
|
"""Helper: run check_expired via callback and wait for background tasks."""
|
|
loop.auto_compact.check_expired(loop._schedule_background)
|
|
await asyncio.sleep(0.1)
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_no_check_when_ttl_disabled(self, tmp_path):
|
|
"""check_expired should be a no-op when TTL is 0."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=0)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
session.add_message("user", "old message")
|
|
session.updated_at = datetime.now() - timedelta(minutes=30)
|
|
loop.sessions.save(session)
|
|
|
|
await self._run_check_expired(loop)
|
|
|
|
session_after = loop.sessions.get_or_create("cli:test")
|
|
assert len(session_after.messages) == 1
|
|
await loop.close_mcp()
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_proactive_archive_on_idle_tick(self, tmp_path):
|
|
"""Expired session should be archived during idle tick."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
session.add_message("user", "old message")
|
|
session.add_message("assistant", "old response")
|
|
session.updated_at = datetime.now() - timedelta(minutes=20)
|
|
loop.sessions.save(session)
|
|
|
|
archived_messages = []
|
|
|
|
async def _fake_archive(messages):
|
|
archived_messages.extend(messages)
|
|
return True
|
|
|
|
loop.consolidator.archive = _fake_archive
|
|
loop.consolidator.get_last_history_entry = lambda: {
|
|
"cursor": 1, "timestamp": "2026-01-01 00:00", "content": "User chatted about old things.",
|
|
}
|
|
|
|
await self._run_check_expired(loop)
|
|
|
|
session_after = loop.sessions.get_or_create("cli:test")
|
|
assert len(session_after.messages) == 0
|
|
assert len(archived_messages) == 2
|
|
entry = loop.auto_compact._summaries.get("cli:test")
|
|
assert entry is not None
|
|
assert entry[0] == "User chatted about old things."
|
|
await loop.close_mcp()
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_no_proactive_archive_when_active(self, tmp_path):
|
|
"""Recently active session should NOT be archived on idle tick."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
session.add_message("user", "recent message")
|
|
loop.sessions.save(session)
|
|
|
|
await self._run_check_expired(loop)
|
|
|
|
session_after = loop.sessions.get_or_create("cli:test")
|
|
assert len(session_after.messages) == 1
|
|
await loop.close_mcp()
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_no_duplicate_archive(self, tmp_path):
|
|
"""Should not archive the same session twice if already in progress."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
session.add_message("user", "old message")
|
|
session.updated_at = datetime.now() - timedelta(minutes=20)
|
|
loop.sessions.save(session)
|
|
|
|
archive_count = 0
|
|
started = asyncio.Event()
|
|
block_forever = asyncio.Event()
|
|
|
|
async def _slow_archive(messages):
|
|
nonlocal archive_count
|
|
archive_count += 1
|
|
started.set()
|
|
await block_forever.wait()
|
|
return True
|
|
|
|
loop.consolidator.archive = _slow_archive
|
|
|
|
# First call starts archiving via callback
|
|
loop.auto_compact.check_expired(loop._schedule_background)
|
|
await started.wait()
|
|
assert archive_count == 1
|
|
|
|
# Second call should skip (key is in _archiving)
|
|
loop.auto_compact.check_expired(loop._schedule_background)
|
|
await asyncio.sleep(0.05)
|
|
assert archive_count == 1
|
|
|
|
# Clean up
|
|
block_forever.set()
|
|
await asyncio.sleep(0.1)
|
|
await loop.close_mcp()
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_proactive_archive_error_does_not_block(self, tmp_path):
|
|
"""Proactive archive failure should be caught and not block future ticks."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
session.add_message("user", "old message")
|
|
session.updated_at = datetime.now() - timedelta(minutes=20)
|
|
loop.sessions.save(session)
|
|
|
|
async def _failing_archive(messages):
|
|
raise RuntimeError("LLM down")
|
|
|
|
loop.consolidator.archive = _failing_archive
|
|
|
|
# Should not raise
|
|
await self._run_check_expired(loop)
|
|
|
|
# Key should be removed from _archiving (finally block)
|
|
assert "cli:test" not in loop.auto_compact._archiving
|
|
await loop.close_mcp()
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_proactive_archive_skips_empty_sessions(self, tmp_path):
|
|
"""Proactive archive should not call LLM for sessions with no un-consolidated messages."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
session.updated_at = datetime.now() - timedelta(minutes=20)
|
|
loop.sessions.save(session)
|
|
|
|
archive_called = False
|
|
|
|
async def _fake_archive(messages):
|
|
nonlocal archive_called
|
|
archive_called = True
|
|
return True
|
|
|
|
loop.consolidator.archive = _fake_archive
|
|
|
|
await self._run_check_expired(loop)
|
|
|
|
assert not archive_called
|
|
await loop.close_mcp()
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_no_reschedule_after_successful_archive(self, tmp_path):
|
|
"""Already-archived session should NOT be re-scheduled on subsequent ticks."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
session.add_message("user", "old message")
|
|
session.add_message("assistant", "old response")
|
|
session.updated_at = datetime.now() - timedelta(minutes=20)
|
|
loop.sessions.save(session)
|
|
|
|
archive_count = 0
|
|
|
|
async def _fake_archive(messages):
|
|
nonlocal archive_count
|
|
archive_count += 1
|
|
return True
|
|
|
|
loop.consolidator.archive = _fake_archive
|
|
loop.consolidator.get_last_history_entry = lambda: {
|
|
"cursor": 1, "timestamp": "2026-01-01 00:00", "content": "Summary.",
|
|
}
|
|
|
|
# First tick: archives the session
|
|
await self._run_check_expired(loop)
|
|
assert archive_count == 1
|
|
|
|
# Second tick: should NOT re-schedule (updated_at is fresh after clear)
|
|
await self._run_check_expired(loop)
|
|
assert archive_count == 1 # Still 1, not re-scheduled
|
|
await loop.close_mcp()
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_empty_skip_refreshes_updated_at_prevents_reschedule(self, tmp_path):
|
|
"""Empty session skip refreshes updated_at, preventing immediate re-scheduling."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
session.updated_at = datetime.now() - timedelta(minutes=20)
|
|
loop.sessions.save(session)
|
|
|
|
archive_count = 0
|
|
|
|
async def _fake_archive(messages):
|
|
nonlocal archive_count
|
|
archive_count += 1
|
|
return True
|
|
|
|
loop.consolidator.archive = _fake_archive
|
|
|
|
# First tick: skips (no messages), refreshes updated_at
|
|
await self._run_check_expired(loop)
|
|
assert archive_count == 0
|
|
|
|
# Second tick: should NOT re-schedule because updated_at is fresh
|
|
await self._run_check_expired(loop)
|
|
assert archive_count == 0
|
|
await loop.close_mcp()
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_session_can_be_compacted_again_after_new_messages(self, tmp_path):
|
|
"""After successful compact + user sends new messages + idle again, should compact again."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
session.add_message("user", "first conversation")
|
|
session.add_message("assistant", "first response")
|
|
session.updated_at = datetime.now() - timedelta(minutes=20)
|
|
loop.sessions.save(session)
|
|
|
|
archive_count = 0
|
|
|
|
async def _fake_archive(messages):
|
|
nonlocal archive_count
|
|
archive_count += 1
|
|
return True
|
|
|
|
loop.consolidator.archive = _fake_archive
|
|
loop.consolidator.get_last_history_entry = lambda: {
|
|
"cursor": 1, "timestamp": "2026-01-01 00:00", "content": "Summary.",
|
|
}
|
|
|
|
# First compact cycle
|
|
await loop.auto_compact._archive("cli:test")
|
|
assert archive_count == 1
|
|
|
|
# User returns, sends new messages
|
|
msg = InboundMessage(channel="cli", sender_id="user", chat_id="test", content="second topic")
|
|
await loop._process_message(msg)
|
|
|
|
# Simulate idle again
|
|
loop.sessions.invalidate("cli:test")
|
|
session2 = loop.sessions.get_or_create("cli:test")
|
|
session2.updated_at = datetime.now() - timedelta(minutes=20)
|
|
loop.sessions.save(session2)
|
|
|
|
# Second compact cycle should succeed
|
|
await loop.auto_compact._archive("cli:test")
|
|
assert archive_count == 2
|
|
await loop.close_mcp()
|
|
|
|
|
|
class TestSummaryPersistence:
|
|
"""Test that summary survives restart via session metadata."""
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_summary_persisted_in_session_metadata(self, tmp_path):
|
|
"""After archive, _last_summary should be in session metadata."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
session.add_message("user", "hello")
|
|
session.add_message("assistant", "hi there")
|
|
session.updated_at = datetime.now() - timedelta(minutes=20)
|
|
loop.sessions.save(session)
|
|
|
|
async def _fake_archive(messages):
|
|
return True
|
|
|
|
loop.consolidator.archive = _fake_archive
|
|
loop.consolidator.get_last_history_entry = lambda: {
|
|
"cursor": 1, "timestamp": "2026-01-01 00:00", "content": "User said hello.",
|
|
}
|
|
|
|
await loop.auto_compact._archive("cli:test")
|
|
|
|
# Summary should be persisted in session metadata
|
|
session_after = loop.sessions.get_or_create("cli:test")
|
|
meta = session_after.metadata.get("_last_summary")
|
|
assert meta is not None
|
|
assert meta["text"] == "User said hello."
|
|
assert "last_active" in meta
|
|
await loop.close_mcp()
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_summary_recovered_after_restart(self, tmp_path):
|
|
"""Summary should be recovered from metadata when _summaries is empty (simulates restart)."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
session.add_message("user", "hello")
|
|
session.add_message("assistant", "hi there")
|
|
last_active = datetime.now() - timedelta(minutes=20)
|
|
session.updated_at = last_active
|
|
loop.sessions.save(session)
|
|
|
|
async def _fake_archive(messages):
|
|
return True
|
|
|
|
loop.consolidator.archive = _fake_archive
|
|
loop.consolidator.get_last_history_entry = lambda: {
|
|
"cursor": 1, "timestamp": "2026-01-01 00:00", "content": "User said hello.",
|
|
}
|
|
|
|
# Archive
|
|
await loop.auto_compact._archive("cli:test")
|
|
|
|
# Simulate restart: clear in-memory state
|
|
loop.auto_compact._summaries.clear()
|
|
loop.sessions.invalidate("cli:test")
|
|
|
|
# prepare_session should recover summary from metadata
|
|
reloaded = loop.sessions.get_or_create("cli:test")
|
|
_, summary = loop.auto_compact.prepare_session(reloaded, "cli:test")
|
|
|
|
assert summary is not None
|
|
assert "User said hello." in summary
|
|
assert "Inactive for" in summary
|
|
# Metadata should be cleaned up after consumption
|
|
assert "_last_summary" not in reloaded.metadata
|
|
await loop.close_mcp()
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_metadata_cleanup_no_leak(self, tmp_path):
|
|
"""_last_summary should be removed from metadata after being consumed."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
session.add_message("user", "hello")
|
|
session.updated_at = datetime.now() - timedelta(minutes=20)
|
|
loop.sessions.save(session)
|
|
|
|
async def _fake_archive(messages):
|
|
return True
|
|
|
|
loop.consolidator.archive = _fake_archive
|
|
loop.consolidator.get_last_history_entry = lambda: {
|
|
"cursor": 1, "timestamp": "2026-01-01 00:00", "content": "Summary.",
|
|
}
|
|
|
|
await loop.auto_compact._archive("cli:test")
|
|
|
|
# Clear in-memory to force metadata path
|
|
loop.auto_compact._summaries.clear()
|
|
loop.sessions.invalidate("cli:test")
|
|
reloaded = loop.sessions.get_or_create("cli:test")
|
|
|
|
# First call: consumes from metadata
|
|
_, summary = loop.auto_compact.prepare_session(reloaded, "cli:test")
|
|
assert summary is not None
|
|
|
|
# Second call: no summary (already consumed)
|
|
_, summary2 = loop.auto_compact.prepare_session(reloaded, "cli:test")
|
|
assert summary2 is None
|
|
assert "_last_summary" not in reloaded.metadata
|
|
await loop.close_mcp()
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_metadata_cleanup_on_inmemory_path(self, tmp_path):
|
|
"""In-memory _summaries path should also clean up _last_summary from metadata."""
|
|
loop = _make_loop(tmp_path, session_ttl_minutes=15)
|
|
session = loop.sessions.get_or_create("cli:test")
|
|
session.add_message("user", "hello")
|
|
session.updated_at = datetime.now() - timedelta(minutes=20)
|
|
loop.sessions.save(session)
|
|
|
|
async def _fake_archive(messages):
|
|
return True
|
|
|
|
loop.consolidator.archive = _fake_archive
|
|
loop.consolidator.get_last_history_entry = lambda: {
|
|
"cursor": 1, "timestamp": "2026-01-01 00:00", "content": "Summary.",
|
|
}
|
|
|
|
await loop.auto_compact._archive("cli:test")
|
|
|
|
# Both _summaries and metadata have the summary
|
|
assert "cli:test" in loop.auto_compact._summaries
|
|
loop.sessions.invalidate("cli:test")
|
|
reloaded = loop.sessions.get_or_create("cli:test")
|
|
assert "_last_summary" in reloaded.metadata
|
|
|
|
# In-memory path is taken (no restart)
|
|
_, summary = loop.auto_compact.prepare_session(reloaded, "cli:test")
|
|
assert summary is not None
|
|
# Metadata should also be cleaned up
|
|
assert "_last_summary" not in reloaded.metadata
|
|
await loop.close_mcp()
|