Files
hermes-webui/tests/test_issue1436_context_indicator_load_path.py
2026-05-17 13:27:40 +08:00

393 lines
18 KiB
Python

"""Regression test for #1436 — context-window indicator broken on load path.
#1356 (closed Apr 30 2026) fixed the indicator on the **live SSE path** by adding
a model-metadata fallback when `agent.context_compressor` didn't provide
`context_length`. But the **GET /api/session load path** (used when clicking
older sessions from the sidebar) was missed — it returned `context_length=0`
verbatim from the persisted Session object for any session that pre-dates
#1318 (when the field was added).
Combined with two cascading frontend fallbacks (`promptTok = last_prompt_tokens
|| input_tokens`, `ctxWindow = context_length || 128*1024`), older sessions
loaded their indicator showing nonsense like "100 / 890% used (context exceeded)"
because:
- `context_length=0` → fallback to 131,072 (128K JS default)
- `last_prompt_tokens=0` → fallback to **cumulative** `input_tokens`
(often 1M+ on long sessions)
- 1.2M / 131K = ~890% → ring caps at 100, tooltip shows "890% used"
Two-layer fix:
1. Backend (api/routes.py:1295-1305) — add model_metadata fallback so
loaded sessions get a sane `context_length` even when persisted as 0.
2. Frontend (static/ui.js:1269) — drop the `input_tokens` fallback for
`promptTok`. Cumulative input is fundamentally wrong for "context window
% used"; better to render "·" + "tokens used" (honest no-data) than a
misleading >100% percentage.
Reported by @AvidFuturist in Discord (May 1 2026, "the 100 comes up way too
often"). Confirmed live on the dev server: 23 of 75 sessions had
`context_length=0` + `input_tokens > 128K`, all rendering >100%.
"""
import json
from pathlib import Path
from unittest.mock import MagicMock, patch
from urllib.parse import urlparse
ROUTES = Path(__file__).resolve().parent.parent / "api" / "routes.py"
UI_JS = Path(__file__).resolve().parent.parent / "static" / "ui.js"
# ─────────────────────────────────────────────────────────────────────────────
# Backend: GET /api/session load path
# ─────────────────────────────────────────────────────────────────────────────
class TestIssue1436BackendFallback:
"""The /api/session GET handler must resolve context_length via
agent.model_metadata.get_model_context_length when the persisted value is 0."""
def _stub_session(self, *, context_length, model, last_prompt_tokens=0,
input_tokens=0):
"""Build a mock Session that mimics the persisted-session shape."""
s = MagicMock()
s.context_length = context_length
s.threshold_tokens = 0
s.last_prompt_tokens = last_prompt_tokens
s.input_tokens = input_tokens
s.output_tokens = 0
s.model = model
s.title = "test-session"
s.session_id = "test-1436"
s.messages = []
s.tool_calls = []
s.active_stream_id = None
s.pending_user_message = None
s.pending_attachments = []
s.pending_started_at = None
# compact() returns a dict that gets merged with the response.
s.compact.return_value = {
"session_id": "test-1436",
"title": "test-session",
"model": model,
"input_tokens": input_tokens,
"output_tokens": 0,
"context_length": context_length,
"threshold_tokens": 0,
"last_prompt_tokens": last_prompt_tokens,
}
return s
def _invoke_get_session(self, session_obj, *, fallback_returns=0):
"""Hit handle_get(/api/session?session_id=...) and capture the JSON response."""
import api.routes as routes
captured = {}
def fake_j(h, data, status=200):
captured["data"] = data
captured["status"] = status
# Patch get_model_context_length so the test doesn't depend on the
# live agent.model_metadata bundle.
fake_module = MagicMock()
fake_module.get_model_context_length = MagicMock(return_value=fallback_returns)
handler = MagicMock()
parsed = urlparse("/api/session?session_id=test-1436&messages=0")
# Patch import so `from agent.model_metadata import ...` resolves to our fake.
with patch("api.routes.get_session", return_value=session_obj), \
patch("api.routes.j", side_effect=fake_j), \
patch.dict("sys.modules", {"agent.model_metadata": fake_module}):
routes.handle_get(handler, parsed)
return captured
def test_persisted_context_length_passed_through_unchanged(self):
"""Fast metadata loads keep the persisted value to avoid catalog work."""
s = self._stub_session(context_length=200_000, model="claude-sonnet-4.6")
result = self._invoke_get_session(s, fallback_returns=999_999)
body = result["data"]["session"]
assert body["context_length"] == 200_000, (
f"persisted context_length must pass through unchanged, "
f"got {body['context_length']}"
)
def test_resolved_model_load_refreshes_stale_persisted_context_length(self):
"""The deferred resolve_model=1 load must refresh stale context metadata.
Session switching first asks for messages=0&resolve_model=0 for speed,
then follows with messages=0&resolve_model=1 to hydrate the final
model/provider display. That second path is also where a stale
context_length from a prior model must be corrected; otherwise a
resumed DeepSeek 1M session can stay stuck on an old 200k window until
the user manually toggles models.
"""
import api.routes as routes
captured = {}
def fake_j(h, data, status=200):
captured["data"] = data
fake_module = MagicMock()
fake_module.get_model_context_length = MagicMock(return_value=1_000_000)
s = self._stub_session(context_length=200_000, model="deepseek-v4-pro")
handler = MagicMock()
parsed = urlparse(
"/api/session?session_id=test-1436&messages=0&resolve_model=1"
)
with patch("api.routes.get_session", return_value=s), \
patch("api.routes.j", side_effect=fake_j), \
patch.dict("sys.modules", {"agent.model_metadata": fake_module}):
routes.handle_get(handler, parsed)
body = captured["data"]["session"]
assert body["context_length"] == 1_000_000, (
"resolve_model=1 must refresh stale persisted context_length from "
"current model metadata"
)
def test_session_model_update_refreshes_context_metadata(self):
"""Changing the session model must not keep the prior model's window."""
import api.routes as routes
captured = {}
def fake_j(h, data, status=200):
captured["data"] = data
fake_module = MagicMock()
fake_module.get_model_context_length = MagicMock(return_value=1_000_000)
s = self._stub_session(context_length=200_000, model="old-model")
s.model_provider = "old-provider"
s.workspace = "/tmp"
s.threshold_tokens = 100_000
s.last_prompt_tokens = 80_000
s.save = MagicMock()
s.compact.return_value = {
**s.compact.return_value,
"model": "deepseek-v4-pro",
"model_provider": "deepseek",
"context_length": 1_000_000,
"threshold_tokens": 0,
"last_prompt_tokens": 0,
}
handler = MagicMock()
parsed = urlparse("/api/session/update")
body = {
"session_id": "test-1436",
"workspace": "/tmp",
"model": "deepseek-v4-pro",
"model_provider": "deepseek",
}
with patch("api.routes._check_csrf", return_value=True), \
patch("api.routes.read_body", return_value=body), \
patch("api.routes.get_session", return_value=s), \
patch("api.routes.resolve_trusted_workspace", return_value="/tmp"), \
patch("api.routes.j", side_effect=fake_j), \
patch.dict("sys.modules", {"agent.model_metadata": fake_module}):
routes.handle_post(handler, parsed)
assert s.model == "deepseek-v4-pro"
assert s.model_provider == "deepseek"
assert s.context_length == 1_000_000
assert s.threshold_tokens == 0
assert s.last_prompt_tokens == 0
s.save.assert_called_once()
assert captured["data"]["session"]["context_length"] == 1_000_000
def test_zero_context_length_falls_back_to_model_metadata(self):
"""Pre-#1318 sessions with context_length=0 must resolve via model_metadata."""
s = self._stub_session(context_length=0, model="claude-opus-4-7",
input_tokens=1_200_000)
result = self._invoke_get_session(s, fallback_returns=1_000_000)
body = result["data"]["session"]
assert body["context_length"] == 1_000_000, (
f"context_length=0 must resolve to model's 1M window via fallback, "
f"got {body['context_length']}"
)
def test_fallback_called_with_persisted_model(self):
"""Fallback must be called with Session.model (not empty string)."""
import api.routes as routes
captured = {}
def fake_j(h, data, status=200):
captured["data"] = data
fake_module = MagicMock()
fake_module.get_model_context_length = MagicMock(return_value=400_000)
s = self._stub_session(context_length=0, model="gpt-5-mini")
handler = MagicMock()
parsed = urlparse("/api/session?session_id=test-1436&messages=0")
with patch("api.routes.get_session", return_value=s), \
patch("api.routes.j", side_effect=fake_j), \
patch.dict("sys.modules", {"agent.model_metadata": fake_module}):
routes.handle_get(handler, parsed)
# First positional arg should be the model name; second is base_url ("")
called_args = fake_module.get_model_context_length.call_args
assert called_args is not None, "get_model_context_length was never called"
assert called_args.args[0] == "gpt-5-mini", (
f"fallback called with wrong model: {called_args}"
)
def test_empty_model_skips_fallback(self):
"""If Session.model is empty AND no effective_model is available, skip
the fallback rather than calling get_model_context_length('')."""
s = self._stub_session(context_length=0, model="")
# resolve_model=0 to skip _resolve_effective_session_model_for_display
import api.routes as routes
captured = {}
def fake_j(h, data, status=200):
captured["data"] = data
fake_module = MagicMock()
fake_module.get_model_context_length = MagicMock(return_value=256_000)
handler = MagicMock()
parsed = urlparse(
"/api/session?session_id=test-1436&messages=0&resolve_model=0"
)
with patch("api.routes.get_session", return_value=s), \
patch("api.routes.j", side_effect=fake_j), \
patch.dict("sys.modules", {"agent.model_metadata": fake_module}):
routes.handle_get(handler, parsed)
# When model is empty, fallback either isn't called OR returns no value
# we use. Either way context_length stays 0.
body = captured["data"]["session"]
assert body["context_length"] == 0, (
f"empty model must NOT trigger fallback (avoids the 256K default-for-unknown trap), "
f"got context_length={body['context_length']}"
)
def test_fallback_exception_does_not_break_response(self):
"""If the fallback raises (older agent build, missing module), the route
must still return a response — context_length just stays 0."""
import api.routes as routes
captured = {}
def fake_j(h, data, status=200):
captured["data"] = data
# Fallback raises on import
fake_module = MagicMock()
fake_module.get_model_context_length = MagicMock(
side_effect=RuntimeError("simulated agent error")
)
s = self._stub_session(context_length=0, model="claude-opus-4-7")
handler = MagicMock()
parsed = urlparse("/api/session?session_id=test-1436&messages=0")
with patch("api.routes.get_session", return_value=s), \
patch("api.routes.j", side_effect=fake_j), \
patch.dict("sys.modules", {"agent.model_metadata": fake_module}):
routes.handle_get(handler, parsed) # must not raise
body = captured["data"]["session"]
assert body["context_length"] == 0, (
"fallback exception must be swallowed; field stays 0"
)
# ─────────────────────────────────────────────────────────────────────────────
# Frontend: static/ui.js _syncCtxIndicator
# ─────────────────────────────────────────────────────────────────────────────
class TestIssue1436FrontendDefense:
"""The frontend must NOT fall back to cumulative input_tokens when
last_prompt_tokens is missing — that produces nonsense percentages."""
def test_promptTok_does_not_fall_back_to_input_tokens(self):
"""Verify the line `promptTok = usage.last_prompt_tokens||usage.input_tokens||0`
has been removed. The old fallback divides cumulative input by the
context window, producing the >100% bug."""
src = UI_JS.read_text(encoding="utf-8")
# The bug shape: the `||usage.input_tokens` fragment must NOT appear
# on any line that defines `promptTok`.
for line_num, line in enumerate(src.splitlines(), 1):
stripped = line.strip()
if stripped.startswith("//") or stripped.startswith("*"):
continue
if "promptTok" in line and "=" in line and "usage.last_prompt_tokens" in line:
assert "usage.input_tokens" not in line, (
f"static/ui.js:{line_num} still falls back to cumulative "
f"input_tokens for promptTok — this produces the >100% indicator "
f"bug from #1436. Line: {line.rstrip()!r}"
)
def test_promptTok_assignment_uses_last_prompt_tokens_only(self):
"""Verify the new assignment: `promptTok = usage.last_prompt_tokens || 0`."""
src = UI_JS.read_text(encoding="utf-8")
# Allow whitespace variations.
normalized = "".join(src.split())
assert "constpromptTok=usage.last_prompt_tokens||0" in normalized, (
"static/ui.js _syncCtxIndicator must assign "
"`promptTok = usage.last_prompt_tokens || 0` (no input_tokens fallback)"
)
def test_no_data_branch_renders_dot(self):
"""When promptTok is 0 (no last-prompt data), the `!hasPromptTok` branch
must render '·' (U+00B7) on the ring instead of computing a percentage.
This is the existing behavior; the test pins it so a future refactor
doesn't accidentally re-introduce a numeric fallback."""
src = UI_JS.read_text(encoding="utf-8")
assert "hasPromptTok=!!promptTok" in src.replace(" ", ""), (
"hasPromptTok must be a boolean of promptTok"
)
# The ring center text uses '·' when !hasPromptTok
assert "hasPromptTok?String(pct):'\\u00b7'" in src.replace(" ", ""), (
"ring center must show '·' (\\u00b7) when no last-prompt data"
)
# ─────────────────────────────────────────────────────────────────────────────
# Static-source assertions (defense in depth — pin the comment markers in place)
# ─────────────────────────────────────────────────────────────────────────────
class TestIssue1436SourceMarkers:
"""Pin the fix comments + import shape so a future refactor can't silently
drop the fallback."""
def test_routes_load_path_imports_get_model_context_length(self):
src = ROUTES.read_text(encoding="utf-8")
# The session load path can call a helper, but the lazy import must
# remain in routes.py so WebUI still works with older/missing agent
# bundles by swallowing metadata-resolution failures.
start = src.find('if parsed.path == "/api/session":')
end = src.find('if parsed.path == "/api/projects":', start)
block = src[start:end]
assert "_resolve_context_length_for_session_model" in block, (
"GET /api/session load-path block must resolve model context "
"metadata for the context_length fallback (#1436)"
)
assert "from agent.model_metadata import get_model_context_length" in src, (
"routes.py must lazy-import get_model_context_length for the "
"context_length fallback (#1436)"
)
def test_routes_load_path_marks_fix_with_issue_number(self):
"""Comment must reference #1436 so future maintainers find this trail."""
src = ROUTES.read_text(encoding="utf-8")
# Find the load-path block (between "if parsed.path == \"/api/session\":"
# and the next `if parsed.path` after it).
start = src.find('if parsed.path == "/api/session":')
end = src.find('if parsed.path == "/api/projects":', start)
block = src[start:end]
assert "#1436" in block, (
"GET /api/session load-path block must reference #1436 in a comment"
)