from __future__ import annotations import pytest from chat.db.connection import open_db from chat.db.migrate import apply_migrations from chat.eventlog.log import append_event from chat.eventlog.projector import project import chat.state.memory # registers memory_written handler from chat.state.memory import get_memory, get_pinned, search_memories def _base_memory(**overrides): payload = { "owner_id": "bot_a", "chat_id": "chat_bot_a", "scene_id": 1, "pov_summary": "She laughed at his joke about owls.", "witness_you": 1, "witness_host": 1, "witness_guest": 0, "chat_clock_at": "2026-04-26T10:00:00", "source": "direct", "reliability": 1.0, "significance": 1, "pinned": 0, "auto_pinned": 0, } payload.update(overrides) return payload def test_memory_written_is_projected_and_readable(tmp_path): db = tmp_path / "t.db" apply_migrations(db) with open_db(db) as conn: append_event(conn, kind="memory_written", payload=_base_memory()) project(conn) row = conn.execute("SELECT id FROM memories").fetchone() assert row is not None mem = get_memory(conn, row[0]) assert mem is not None assert mem["owner_id"] == "bot_a" assert mem["chat_id"] == "chat_bot_a" assert mem["scene_id"] == 1 assert mem["pov_summary"] == "She laughed at his joke about owls." assert mem["witness_you"] == 1 assert mem["witness_host"] == 1 assert mem["witness_guest"] == 0 assert mem["chat_clock_at"] == "2026-04-26T10:00:00" assert mem["source"] == "direct" assert mem["reliability"] == 1.0 assert mem["significance"] == 1 assert mem["pinned"] == 0 assert mem["auto_pinned"] == 0 def test_get_memory_returns_none_for_missing_id(tmp_path): db = tmp_path / "t.db" apply_migrations(db) with open_db(db) as conn: assert get_memory(conn, 9999) is None def test_search_memories_filters_out_non_witness(tmp_path): db = tmp_path / "t.db" apply_migrations(db) with open_db(db) as conn: append_event(conn, kind="memory_written", payload=_base_memory( pov_summary="The cat sat on the mat.", witness_you=1, witness_host=1, witness_guest=0, )) project(conn) # guest did not witness => excluded results = search_memories(conn, "bot_a", "guest", "cat") assert results == [] def test_search_memories_includes_witnesses(tmp_path): db = tmp_path / "t.db" apply_migrations(db) with open_db(db) as conn: append_event(conn, kind="memory_written", payload=_base_memory( pov_summary="The cat sat on the mat.", witness_you=1, witness_host=1, witness_guest=0, )) project(conn) results = search_memories(conn, "bot_a", "host", "cat") assert len(results) == 1 assert results[0]["pov_summary"] == "The cat sat on the mat." assert "fts_rank" in results[0] def test_search_memories_fts_matches_only_relevant_text(tmp_path): db = tmp_path / "t.db" apply_migrations(db) with open_db(db) as conn: append_event(conn, kind="memory_written", payload=_base_memory( pov_summary="She loves owls and stars.", )) append_event(conn, kind="memory_written", payload=_base_memory( pov_summary="He fixed the broken kettle.", )) project(conn) results = search_memories(conn, "bot_a", "you", "owls") assert len(results) == 1 assert results[0]["pov_summary"] == "She loves owls and stars." def test_search_memories_filters_by_owner(tmp_path): db = tmp_path / "t.db" apply_migrations(db) with open_db(db) as conn: append_event(conn, kind="memory_written", payload=_base_memory( owner_id="bot_a", pov_summary="Owls hooted at midnight.", )) append_event(conn, kind="memory_written", payload=_base_memory( owner_id="bot_b", pov_summary="Owls hooted at midnight.", )) project(conn) results_a = search_memories(conn, "bot_a", "you", "owls") results_b = search_memories(conn, "bot_b", "you", "owls") assert len(results_a) == 1 assert results_a[0]["owner_id"] == "bot_a" assert len(results_b) == 1 assert results_b[0]["owner_id"] == "bot_b" def test_search_memories_returns_empty_on_no_match(tmp_path): db = tmp_path / "t.db" apply_migrations(db) with open_db(db) as conn: append_event(conn, kind="memory_written", payload=_base_memory( pov_summary="The cat sat on the mat.", )) project(conn) results = search_memories(conn, "bot_a", "you", "spaceship") assert results == [] def test_search_memories_invalid_witness_role_raises(tmp_path): db = tmp_path / "t.db" apply_migrations(db) with open_db(db) as conn: with pytest.raises(ValueError): search_memories(conn, "bot_a", "everyone", "cat") def test_search_memories_respects_k_limit(tmp_path): db = tmp_path / "t.db" apply_migrations(db) with open_db(db) as conn: for i in range(6): append_event(conn, kind="memory_written", payload=_base_memory( pov_summary=f"Owls hooted at midnight number {i}.", )) project(conn) results = search_memories(conn, "bot_a", "you", "owls", k=4) assert len(results) == 4 def test_get_pinned_returns_only_pinned(tmp_path): db = tmp_path / "t.db" apply_migrations(db) with open_db(db) as conn: append_event(conn, kind="memory_written", payload=_base_memory( pov_summary="Pinned moment.", pinned=1, )) append_event(conn, kind="memory_written", payload=_base_memory( pov_summary="Unpinned moment.", pinned=0, )) project(conn) pinned = get_pinned(conn, "bot_a") assert len(pinned) == 1 assert pinned[0]["pov_summary"] == "Pinned moment." assert pinned[0]["pinned"] == 1 def test_get_pinned_filters_by_owner(tmp_path): db = tmp_path / "t.db" apply_migrations(db) with open_db(db) as conn: append_event(conn, kind="memory_written", payload=_base_memory( owner_id="bot_a", pov_summary="A's pin.", pinned=1, )) append_event(conn, kind="memory_written", payload=_base_memory( owner_id="bot_b", pov_summary="B's pin.", pinned=1, )) project(conn) pinned_a = get_pinned(conn, "bot_a") assert len(pinned_a) == 1 assert pinned_a[0]["owner_id"] == "bot_a" def test_memory_payload_defaults_when_optional_missing(tmp_path): db = tmp_path / "t.db" apply_migrations(db) with open_db(db) as conn: append_event(conn, kind="memory_written", payload={ "owner_id": "bot_a", "chat_id": "chat_bot_a", "pov_summary": "Bare minimum memory.", "witness_you": 1, "witness_host": 1, "witness_guest": 1, }) project(conn) row = conn.execute("SELECT id FROM memories").fetchone() mem = get_memory(conn, row[0]) assert mem["scene_id"] is None assert mem["chat_clock_at"] is None assert mem["source"] == "direct" assert mem["reliability"] == 1.0 assert mem["significance"] == 1 assert mem["pinned"] == 0 assert mem["auto_pinned"] == 0 def test_schema_version_after_migration_is_at_least_6(tmp_path): db = tmp_path / "t.db" apply_migrations(db) with open_db(db) as conn: row = conn.execute( "SELECT value FROM meta WHERE key = 'schema_version'" ).fetchone() assert int(row[0]) >= 6