This repository has been archived on 2026-03-24. You can view files and clone it. You cannot open issues or pull requests or push a commit.
Files
Timmy-time-dashboard/tests/dashboard/test_world_api.py
2026-03-19 01:28:57 -04:00

248 lines
7.6 KiB
Python

"""Tests for GET /api/world/state endpoint and /api/world/ws relay."""
import json
import time
from unittest.mock import AsyncMock, patch
import pytest
from dashboard.routes.world import (
_STALE_THRESHOLD,
_build_world_state,
_read_presence_file,
broadcast_world_state,
)
# ---------------------------------------------------------------------------
# _build_world_state
# ---------------------------------------------------------------------------
def test_build_world_state_maps_fields():
presence = {
"version": 1,
"liveness": "2026-03-19T02:00:00Z",
"mood": "exploring",
"current_focus": "reviewing PR",
"energy": 0.8,
"confidence": 0.9,
"active_threads": [{"type": "thinking", "ref": "test", "status": "active"}],
"recent_events": [],
"concerns": [],
}
result = _build_world_state(presence)
assert result["timmyState"]["mood"] == "exploring"
assert result["timmyState"]["activity"] == "reviewing PR"
assert result["timmyState"]["energy"] == 0.8
assert result["timmyState"]["confidence"] == 0.9
assert result["updatedAt"] == "2026-03-19T02:00:00Z"
assert result["version"] == 1
assert result["visitorPresent"] is False
assert len(result["activeThreads"]) == 1
def test_build_world_state_defaults():
"""Missing fields get safe defaults."""
result = _build_world_state({})
assert result["timmyState"]["mood"] == "focused"
assert result["timmyState"]["energy"] == 0.5
assert result["version"] == 1
# ---------------------------------------------------------------------------
# _read_presence_file
# ---------------------------------------------------------------------------
def test_read_presence_file_missing(tmp_path):
with patch("dashboard.routes.world.PRESENCE_FILE", tmp_path / "nope.json"):
assert _read_presence_file() is None
def test_read_presence_file_stale(tmp_path):
f = tmp_path / "presence.json"
f.write_text(json.dumps({"version": 1}))
# Backdate the file
stale_time = time.time() - _STALE_THRESHOLD - 10
import os
os.utime(f, (stale_time, stale_time))
with patch("dashboard.routes.world.PRESENCE_FILE", f):
assert _read_presence_file() is None
def test_read_presence_file_fresh(tmp_path):
f = tmp_path / "presence.json"
f.write_text(json.dumps({"version": 1, "mood": "focused"}))
with patch("dashboard.routes.world.PRESENCE_FILE", f):
result = _read_presence_file()
assert result is not None
assert result["version"] == 1
def test_read_presence_file_bad_json(tmp_path):
f = tmp_path / "presence.json"
f.write_text("not json {{{")
with patch("dashboard.routes.world.PRESENCE_FILE", f):
assert _read_presence_file() is None
# ---------------------------------------------------------------------------
# Full endpoint via TestClient
# ---------------------------------------------------------------------------
@pytest.fixture
def client():
from fastapi import FastAPI
from fastapi.testclient import TestClient
app = FastAPI()
from dashboard.routes.world import router
app.include_router(router)
return TestClient(app)
def test_world_state_endpoint_with_file(client, tmp_path):
"""Endpoint returns data from presence file when fresh."""
f = tmp_path / "presence.json"
f.write_text(
json.dumps(
{
"version": 1,
"liveness": "2026-03-19T02:00:00Z",
"mood": "exploring",
"current_focus": "testing",
"active_threads": [],
"recent_events": [],
"concerns": [],
}
)
)
with patch("dashboard.routes.world.PRESENCE_FILE", f):
resp = client.get("/api/world/state")
assert resp.status_code == 200
data = resp.json()
assert data["timmyState"]["mood"] == "exploring"
assert data["timmyState"]["activity"] == "testing"
assert resp.headers["cache-control"] == "no-cache, no-store"
def test_world_state_endpoint_fallback(client, tmp_path):
"""Endpoint falls back to live state when file missing."""
with (
patch("dashboard.routes.world.PRESENCE_FILE", tmp_path / "nope.json"),
patch("timmy.workshop_state.get_state_dict") as mock_get,
):
mock_get.return_value = {
"version": 1,
"liveness": "2026-03-19T02:00:00Z",
"mood": "idle",
"current_focus": "",
"active_threads": [],
"recent_events": [],
"concerns": [],
}
resp = client.get("/api/world/state")
assert resp.status_code == 200
assert resp.json()["timmyState"]["mood"] == "idle"
def test_world_state_endpoint_full_fallback(client, tmp_path):
"""Endpoint returns safe defaults when everything fails."""
with (
patch("dashboard.routes.world.PRESENCE_FILE", tmp_path / "nope.json"),
patch(
"timmy.workshop_state.get_state_dict",
side_effect=RuntimeError("boom"),
),
):
resp = client.get("/api/world/state")
assert resp.status_code == 200
data = resp.json()
assert data["timmyState"]["mood"] == "idle"
assert data["version"] == 1
# ---------------------------------------------------------------------------
# broadcast_world_state
# ---------------------------------------------------------------------------
@pytest.mark.asyncio
async def test_broadcast_world_state_sends_timmy_state():
"""broadcast_world_state sends timmy_state JSON to connected clients."""
from dashboard.routes.world import _ws_clients
ws = AsyncMock()
_ws_clients.append(ws)
try:
presence = {
"version": 1,
"mood": "exploring",
"current_focus": "testing",
"energy": 0.8,
"confidence": 0.9,
}
await broadcast_world_state(presence)
ws.send_text.assert_called_once()
msg = json.loads(ws.send_text.call_args[0][0])
assert msg["type"] == "timmy_state"
assert msg["mood"] == "exploring"
assert msg["activity"] == "testing"
finally:
_ws_clients.clear()
@pytest.mark.asyncio
async def test_broadcast_world_state_removes_dead_clients():
"""Dead WebSocket connections are cleaned up on broadcast."""
from dashboard.routes.world import _ws_clients
dead_ws = AsyncMock()
dead_ws.send_text.side_effect = ConnectionError("gone")
_ws_clients.append(dead_ws)
try:
await broadcast_world_state({"mood": "idle"})
assert dead_ws not in _ws_clients
finally:
_ws_clients.clear()
def test_world_ws_endpoint_accepts_connection(client):
"""WebSocket endpoint at /api/world/ws accepts connections."""
with client.websocket_connect("/api/world/ws"):
pass # Connection accepted — just close it
def test_world_ws_sends_snapshot_on_connect(client, tmp_path):
"""WebSocket sends a world_state snapshot immediately on connect."""
f = tmp_path / "presence.json"
f.write_text(
json.dumps(
{
"version": 1,
"liveness": "2026-03-19T02:00:00Z",
"mood": "exploring",
"current_focus": "testing",
"active_threads": [],
"recent_events": [],
"concerns": [],
}
)
)
with patch("dashboard.routes.world.PRESENCE_FILE", f):
with client.websocket_connect("/api/world/ws") as ws:
msg = json.loads(ws.receive_text())
assert msg["type"] == "world_state"
assert msg["timmyState"]["mood"] == "exploring"
assert msg["timmyState"]["activity"] == "testing"
assert "updatedAt" in msg