mirror of
https://github.com/nesquena/hermes-webui.git
synced 2026-05-24 18:50:15 +00:00
213 lines
6.5 KiB
Python
213 lines
6.5 KiB
Python
"""Regression tests for /api/models disk cache metadata."""
|
|
|
|
import json
|
|
import time
|
|
|
|
import api.config as config
|
|
|
|
|
|
def _reset_memory_cache() -> None:
|
|
with config._available_models_cache_lock:
|
|
config._available_models_cache = None
|
|
config._available_models_cache_ts = 0.0
|
|
config._cache_build_in_progress = False
|
|
config._cache_build_cv.notify_all()
|
|
|
|
|
|
def test_save_models_cache_to_disk_preserves_response_metadata(tmp_path, monkeypatch):
|
|
cache_path = tmp_path / "models_cache.json"
|
|
monkeypatch.setattr(config, "_models_cache_path", cache_path)
|
|
|
|
payload = {
|
|
"active_provider": "openai",
|
|
"default_model": "gpt-5.4-mini",
|
|
"groups": [
|
|
{
|
|
"provider": "OpenAI",
|
|
"provider_id": "openai",
|
|
"models": [{"id": "gpt-5.4-mini", "label": "GPT 5.4 Mini"}],
|
|
}
|
|
],
|
|
}
|
|
|
|
config._save_models_cache_to_disk(payload)
|
|
|
|
assert json.loads(cache_path.read_text(encoding="utf-8")) == payload
|
|
assert config._load_models_cache_from_disk() == payload
|
|
|
|
|
|
def test_load_models_cache_from_disk_rejects_legacy_groups_only_cache(tmp_path, monkeypatch):
|
|
cache_path = tmp_path / "models_cache.json"
|
|
monkeypatch.setattr(config, "_models_cache_path", cache_path)
|
|
cache_path.write_text(
|
|
json.dumps(
|
|
{
|
|
"groups": [
|
|
{
|
|
"provider": "Legacy",
|
|
"provider_id": "legacy",
|
|
"models": [{"id": "legacy-model", "label": "Legacy Model"}],
|
|
}
|
|
]
|
|
}
|
|
),
|
|
encoding="utf-8",
|
|
)
|
|
|
|
assert config._load_models_cache_from_disk() is None
|
|
|
|
|
|
def test_load_models_cache_from_disk_rejects_partial_metadata_cache(
|
|
tmp_path,
|
|
monkeypatch,
|
|
):
|
|
cache_path = tmp_path / "models_cache.json"
|
|
monkeypatch.setattr(config, "_models_cache_path", cache_path)
|
|
|
|
valid_payload = {
|
|
"active_provider": "openai",
|
|
"default_model": "gpt-5.4-mini",
|
|
"groups": [
|
|
{
|
|
"provider": "OpenAI",
|
|
"provider_id": "openai",
|
|
"models": [{"id": "gpt-5.4-mini", "label": "GPT 5.4 Mini"}],
|
|
}
|
|
],
|
|
}
|
|
|
|
invalid_payloads = [
|
|
{key: value for key, value in valid_payload.items() if key != "active_provider"},
|
|
{key: value for key, value in valid_payload.items() if key != "default_model"},
|
|
{key: value for key, value in valid_payload.items() if key != "groups"},
|
|
{**valid_payload, "active_provider": 123},
|
|
{**valid_payload, "default_model": None},
|
|
{**valid_payload, "groups": {}},
|
|
]
|
|
|
|
for payload in invalid_payloads:
|
|
cache_path.write_text(json.dumps(payload), encoding="utf-8")
|
|
assert config._load_models_cache_from_disk() is None
|
|
|
|
|
|
def test_get_available_models_ignores_invalid_ttl_memory_cache(monkeypatch):
|
|
_reset_memory_cache()
|
|
|
|
stale_cache = {
|
|
"groups": [
|
|
{
|
|
"provider": "Stale",
|
|
"provider_id": "stale",
|
|
"models": [{"id": "stale-model", "label": "Stale Model"}],
|
|
}
|
|
]
|
|
}
|
|
|
|
saved_mtime = config._cfg_mtime
|
|
try:
|
|
with config._available_models_cache_lock:
|
|
config._available_models_cache = stale_cache
|
|
config._available_models_cache_ts = time.monotonic()
|
|
|
|
try:
|
|
config._cfg_mtime = config.Path(config._get_config_path()).stat().st_mtime
|
|
except OSError:
|
|
config._cfg_mtime = 0.0
|
|
|
|
result = config.get_available_models()
|
|
finally:
|
|
config._cfg_mtime = saved_mtime
|
|
_reset_memory_cache()
|
|
|
|
assert "active_provider" in result
|
|
assert "default_model" in result
|
|
assert "groups" in result
|
|
assert not any(group.get("provider") == "Stale" for group in result["groups"])
|
|
|
|
|
|
def test_get_available_models_does_not_use_disk_cache_after_config_mtime_change(
|
|
tmp_path,
|
|
monkeypatch,
|
|
):
|
|
cache_path = tmp_path / "models_cache.json"
|
|
monkeypatch.setattr(config, "_models_cache_path", cache_path)
|
|
cache_path.write_text(
|
|
json.dumps(
|
|
{
|
|
"active_provider": "stale-provider",
|
|
"default_model": "stale-model",
|
|
"groups": [
|
|
{
|
|
"provider": "Stale",
|
|
"provider_id": "stale",
|
|
"models": [{"id": "stale-model", "label": "Stale Model"}],
|
|
}
|
|
],
|
|
}
|
|
),
|
|
encoding="utf-8",
|
|
)
|
|
_reset_memory_cache()
|
|
|
|
saved_mtime = config._cfg_mtime
|
|
try:
|
|
config._cfg_mtime = -1.0
|
|
result = config.get_available_models()
|
|
finally:
|
|
config._cfg_mtime = saved_mtime
|
|
_reset_memory_cache()
|
|
|
|
assert result["active_provider"] != "stale-provider"
|
|
assert result["default_model"] != "stale-model"
|
|
assert not any(group.get("provider") == "Stale" for group in result["groups"])
|
|
|
|
written = json.loads(cache_path.read_text(encoding="utf-8"))
|
|
assert written["active_provider"] != "stale-provider"
|
|
assert written["default_model"] != "stale-model"
|
|
assert not any(group.get("provider") == "Stale" for group in written["groups"])
|
|
|
|
|
|
def test_get_available_models_ignores_legacy_disk_cache_and_rebuilds(
|
|
tmp_path,
|
|
monkeypatch,
|
|
):
|
|
cache_path = tmp_path / "models_cache.json"
|
|
monkeypatch.setattr(config, "_models_cache_path", cache_path)
|
|
cache_path.write_text(
|
|
json.dumps(
|
|
{
|
|
"groups": [
|
|
{
|
|
"provider": "Legacy",
|
|
"provider_id": "legacy",
|
|
"models": [{"id": "legacy-model", "label": "Legacy Model"}],
|
|
}
|
|
]
|
|
}
|
|
),
|
|
encoding="utf-8",
|
|
)
|
|
_reset_memory_cache()
|
|
|
|
saved_mtime = config._cfg_mtime
|
|
try:
|
|
try:
|
|
config._cfg_mtime = config.Path(config._get_config_path()).stat().st_mtime
|
|
except OSError:
|
|
config._cfg_mtime = 0.0
|
|
|
|
result = config.get_available_models()
|
|
finally:
|
|
config._cfg_mtime = saved_mtime
|
|
_reset_memory_cache()
|
|
|
|
assert "active_provider" in result
|
|
assert "default_model" in result
|
|
assert "groups" in result
|
|
assert not any(group.get("provider") == "Legacy" for group in result["groups"])
|
|
|
|
written = json.loads(cache_path.read_text(encoding="utf-8"))
|
|
assert "active_provider" in written
|
|
assert "default_model" in written
|
|
assert "groups" in written
|