fix(ui): context indicator prefers latest usage over stale session data (fixes #437)
This commit is contained in:
@@ -143,7 +143,15 @@ async function loadSession(sid){
|
|||||||
const _s=S.session;
|
const _s=S.session;
|
||||||
if(_s&&typeof _syncCtxIndicator==='function'){
|
if(_s&&typeof _syncCtxIndicator==='function'){
|
||||||
const u=S.lastUsage||{};
|
const u=S.lastUsage||{};
|
||||||
_syncCtxIndicator({input_tokens:_s.input_tokens||u.input_tokens||0,output_tokens:_s.output_tokens||u.output_tokens||0,estimated_cost:_s.estimated_cost||u.estimated_cost,context_length:u.context_length||0,last_prompt_tokens:u.last_prompt_tokens||0,threshold_tokens:u.threshold_tokens||0});
|
const _pick=(latest,stored,dflt=0)=>latest!=null?latest:(stored!=null?stored:dflt);
|
||||||
|
_syncCtxIndicator({
|
||||||
|
input_tokens: _pick(u.input_tokens, _s.input_tokens),
|
||||||
|
output_tokens: _pick(u.output_tokens, _s.output_tokens),
|
||||||
|
estimated_cost: _pick(u.estimated_cost, _s.estimated_cost),
|
||||||
|
context_length: _pick(u.context_length, _s.context_length),
|
||||||
|
last_prompt_tokens:_pick(u.last_prompt_tokens,_s.last_prompt_tokens),
|
||||||
|
threshold_tokens: _pick(u.threshold_tokens, _s.threshold_tokens),
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,107 +1,38 @@
|
|||||||
"""
|
"""Sprint 42 tests: context indicator prefers latest usage over stale session data (issue #437)"""
|
||||||
Sprint 42 Tests: SessionDB injection into AIAgent for WebUI sessions (PR #356).
|
import os
|
||||||
|
|
||||||
Covers:
|
SESSIONS_JS = os.path.join(os.path.dirname(__file__), '..', 'static', 'sessions.js')
|
||||||
- streaming.py: SessionDB is initialized inside _run_agent_streaming (import present)
|
|
||||||
- streaming.py: try/except guards SessionDB init so failures are non-fatal
|
|
||||||
- streaming.py: session_db= kwarg is passed to AIAgent constructor
|
|
||||||
- streaming.py: SessionDB init failure prints a WARNING (not silently swallowed)
|
|
||||||
- streaming.py: SessionDB init is placed before AIAgent construction
|
|
||||||
"""
|
|
||||||
import ast
|
|
||||||
import pathlib
|
|
||||||
import re
|
|
||||||
import unittest
|
|
||||||
|
|
||||||
REPO_ROOT = pathlib.Path(__file__).parent.parent
|
|
||||||
STREAMING_PY = (REPO_ROOT / "api" / "streaming.py").read_text()
|
|
||||||
|
|
||||||
|
|
||||||
class TestSessionDBInjection(unittest.TestCase):
|
def _read_sessions_js():
|
||||||
"""Verify SessionDB is initialized and passed to AIAgent in streaming.py."""
|
with open(SESSIONS_JS, 'r') as f:
|
||||||
|
return f.read()
|
||||||
def test_hermes_state_import_present(self):
|
|
||||||
"""SessionDB must be imported from hermes_state inside _run_agent_streaming."""
|
|
||||||
self.assertIn(
|
|
||||||
"from hermes_state import SessionDB",
|
|
||||||
STREAMING_PY,
|
|
||||||
"SessionDB import missing from streaming.py (PR #356)",
|
|
||||||
)
|
|
||||||
|
|
||||||
def test_session_db_kwarg_passed_to_agent(self):
|
|
||||||
"""session_db= must be passed to the AIAgent constructor call."""
|
|
||||||
self.assertIn(
|
|
||||||
"session_db=_session_db",
|
|
||||||
STREAMING_PY,
|
|
||||||
"session_db kwarg not passed to AIAgent (PR #356)",
|
|
||||||
)
|
|
||||||
|
|
||||||
def test_sessiondb_init_in_try_except(self):
|
|
||||||
"""SessionDB() init must be wrapped in try/except for non-fatal failure handling."""
|
|
||||||
# Check that the try/except pattern surrounding SessionDB() is present
|
|
||||||
pattern = r"try:\s*\n\s*from hermes_state import SessionDB\s*\n\s*_session_db\s*=\s*SessionDB\(\)"
|
|
||||||
self.assertRegex(
|
|
||||||
STREAMING_PY,
|
|
||||||
pattern,
|
|
||||||
"SessionDB() init must be inside a try block for non-fatal error handling (PR #356)",
|
|
||||||
)
|
|
||||||
|
|
||||||
def test_sessiondb_failure_logs_warning(self):
|
|
||||||
"""A failure initializing SessionDB must print a WARNING (not silently drop the error)."""
|
|
||||||
self.assertIn(
|
|
||||||
"WARNING: SessionDB init failed",
|
|
||||||
STREAMING_PY,
|
|
||||||
"SessionDB init failure must log a WARNING message (PR #356)",
|
|
||||||
)
|
|
||||||
|
|
||||||
def test_session_db_initialized_before_agent_construction(self):
|
|
||||||
"""SessionDB initialization must appear before the AIAgent(...) constructor call."""
|
|
||||||
db_pos = STREAMING_PY.find("from hermes_state import SessionDB")
|
|
||||||
agent_pos = STREAMING_PY.find("session_db=_session_db")
|
|
||||||
self.assertGreater(
|
|
||||||
agent_pos,
|
|
||||||
db_pos,
|
|
||||||
"SessionDB init must appear before AIAgent construction (PR #356)",
|
|
||||||
)
|
|
||||||
|
|
||||||
def test_session_db_default_is_none(self):
|
|
||||||
"""_session_db must be initialized to None before the try block (safe default)."""
|
|
||||||
# Pattern: _session_db = None followed (eventually) by the try/SessionDB block
|
|
||||||
pattern = r"_session_db\s*=\s*None\s*\n\s*try:"
|
|
||||||
self.assertRegex(
|
|
||||||
STREAMING_PY,
|
|
||||||
pattern,
|
|
||||||
"_session_db must default to None before try/except block (PR #356)",
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
class TestSessionDBAST(unittest.TestCase):
|
def test_context_indicator_uses_pick_helper():
|
||||||
"""AST-level checks: verify the try/except is not inside _ENV_LOCK (deadlock guard)."""
|
"""The _pick helper must be present in sessions.js to prefer latest over stale values."""
|
||||||
|
content = _read_sessions_js()
|
||||||
|
assert '_pick' in content, "_pick helper not found in static/sessions.js"
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
self.tree = ast.parse(STREAMING_PY)
|
|
||||||
|
|
||||||
def test_sessiondb_try_not_inside_env_lock(self):
|
def test_context_indicator_old_pattern_removed():
|
||||||
"""The try block that wraps SessionDB init must NOT be inside a 'with _ENV_LOCK:' block.
|
"""The old || pattern that preferred stale session data must be gone."""
|
||||||
|
content = _read_sessions_js()
|
||||||
|
assert '_s.input_tokens||u.input_tokens' not in content, \
|
||||||
|
"Old stale-data-first pattern '_s.input_tokens||u.input_tokens' still present in static/sessions.js"
|
||||||
|
|
||||||
Putting a try/except inside _ENV_LOCK is the deadlock pattern caught by test_sprint34.
|
|
||||||
The SessionDB try/except is outside the lock scope, which is correct.
|
def test_context_indicator_all_six_fields():
|
||||||
"""
|
"""All six token/cost fields must appear in the _syncCtxIndicator call."""
|
||||||
# Find all 'with _ENV_LOCK:' nodes; check none of their bodies contain
|
content = _read_sessions_js()
|
||||||
# a Try node that also contains 'from hermes_state import SessionDB'
|
fields = [
|
||||||
for node in ast.walk(self.tree):
|
'input_tokens',
|
||||||
if not isinstance(node, ast.With):
|
'output_tokens',
|
||||||
continue
|
'estimated_cost',
|
||||||
names = [getattr(item.context_expr, "id", "") for item in node.items]
|
'context_length',
|
||||||
if "_ENV_LOCK" not in names:
|
'last_prompt_tokens',
|
||||||
continue
|
'threshold_tokens',
|
||||||
# Walk the with-body for Try nodes
|
]
|
||||||
for stmt in node.body:
|
for field in fields:
|
||||||
if isinstance(stmt, ast.Try):
|
assert field in content, \
|
||||||
# Check if this try imports hermes_state
|
f"Field '{field}' not found in static/sessions.js _syncCtxIndicator call"
|
||||||
src = ast.unparse(stmt)
|
|
||||||
self.assertNotIn(
|
|
||||||
"hermes_state",
|
|
||||||
src,
|
|
||||||
"SessionDB try/except must NOT be inside _ENV_LOCK body (deadlock risk)",
|
|
||||||
)
|
|
||||||
|
|||||||
Reference in New Issue
Block a user