5dbcfc7ccf
Her remaining two wishes from the 6-19 sketch: Proactive reach-out (#6, literal): lyra/notify.py pushes to ntfy so she can reach Brian when he's not in the app. thoughts.maybe_ping gates on salience, a cooldown, and local quiet hours (all config-tunable; eager defaults), uses ntfy JSON publish (UTF-8 titles/messages), links to /thoughts, and marks the thread surfaced so chat won't also re-raise it. Disabled unless NTFY_URL is set. External input feed (#1): lyra/feeds.py pulls configurable RSS/Atom feeds (stdlib ElementTree, no new dep; tolerant of RSS 2.0 + Atom), dedupes seen items in a feed_items table, and hands think() one fresh item at a time. New 'react' mode: a would-be new thread instead reacts to a world item (FEED_REACT_PROB). Dream cycle refreshes feeds on its cadence; failures degrade to no item. Config: NTFY_URL/NTFY_TOPIC/LYRA_WEB_URL, PING_SALIENCE/COOLDOWN/QUIET_HOURS, LYRA_TIMEZONE, LYRA_FEEDS, FEED_REACT_PROB (+ .env.example). thought_meta table for ping cooldown. 10 new tests (feeds parse, react mode, ping gating); suite 65. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
80 lines
3.0 KiB
Python
80 lines
3.0 KiB
Python
"""Dream-cycle tests: backlog sensing + a full forced pass, with LLM/embeddings
|
|
stubbed so nothing hits a real backend."""
|
|
from __future__ import annotations
|
|
|
|
import importlib
|
|
|
|
import pytest
|
|
|
|
|
|
@pytest.fixture
|
|
def lyra(tmp_path, monkeypatch):
|
|
"""A fresh Lyra wired to a temp DB with stubbed embeddings + LLM."""
|
|
monkeypatch.setenv("LYRA_DB_PATH", str(tmp_path / "test.db"))
|
|
monkeypatch.setenv("SUMMARY_BACKEND", "local")
|
|
monkeypatch.setenv("LYRA_FEEDS", "") # dream cycle refreshes feeds; keep it offline
|
|
|
|
from lyra import llm
|
|
# Deterministic 3-d embeddings; content-insensitive is fine for storage tests.
|
|
monkeypatch.setattr(llm, "embed", lambda texts: [[0.1, 0.2, 0.3] for _ in texts])
|
|
# reflect() expects JSON back; everything else just stores the text.
|
|
monkeypatch.setattr(
|
|
llm, "complete",
|
|
lambda messages, backend=None, model=None:
|
|
'{"mood":"focused","valence":0.7,"new_reflections":["I got some thinking done."]}',
|
|
)
|
|
|
|
import lyra.memory as memory
|
|
importlib.reload(memory) # drop any cached connection from another test/db
|
|
return memory
|
|
|
|
|
|
def _seed(memory, session_id, n, summarized_up_to=None):
|
|
ids = [memory.remember(session_id, "user", f"msg {i}") for i in range(n)]
|
|
if summarized_up_to is not None:
|
|
memory.store_summary(session_id, "gist", ids[summarized_up_to])
|
|
return ids
|
|
|
|
|
|
def test_backlog_stats(lyra):
|
|
memory = lyra
|
|
_seed(memory, "s-fresh", 5) # never summarized -> ripe
|
|
_seed(memory, "s-ripe", 25, summarized_up_to=0) # 24 new turns -> ripe
|
|
_seed(memory, "s-clean", 3, summarized_up_to=2) # caught up -> not dirty
|
|
|
|
stats = memory.backlog_stats(ripe_threshold=20)
|
|
assert stats["sessions"] == 3
|
|
assert stats["dirty"] == 2
|
|
assert stats["ripe"] == 2
|
|
assert stats["max_exchange_id"] == 33
|
|
|
|
|
|
def test_dream_cycle_consolidates_and_persists(lyra):
|
|
memory = lyra
|
|
from lyra import dream
|
|
|
|
# A big backlog: enough never-summarized sessions that continuity saturates
|
|
# and the resulting fresh gists push coherence past threshold too.
|
|
for k in range(7):
|
|
_seed(memory, f"s{k}", 4)
|
|
|
|
state = dream.dream_cycle(force=False)
|
|
|
|
# continuity built up and fired -> sessions got summarized
|
|
assert len(memory.list_summaries()) == 7
|
|
acts = state["dream"]["last_actions"]
|
|
assert any("consolidated" in a for a in acts)
|
|
# 7 fresh gists -> coherence crossed threshold -> profile got integrated
|
|
assert any("integrated" in a for a in acts)
|
|
assert memory.get_profile() is not None
|
|
|
|
# drives + bookkeeping persisted and reload-able
|
|
assert set(state["drives"]) == {"continuity", "coherence", "curiosity", "stability"}
|
|
assert state["dream"]["cycle_count"] == 1
|
|
assert memory.get_self_state()["dream"]["last_exchange_id"] == 28
|
|
|
|
# a second pass with no new activity should rest (continuity relieved)
|
|
state2 = dream.dream_cycle(force=False)
|
|
assert state2["dream"]["cycle_count"] == 2
|
|
assert state2["drives"]["continuity"] == 0.0
|