mirror of
https://github.com/nesquena/hermes-webui.git
synced 2026-05-23 10:50:14 +00:00
164 lines
5.3 KiB
Python
164 lines
5.3 KiB
Python
"""Regression tests for #1240 — WebUI model catalog should delegate to Hermes CLI.
|
|
|
|
The WebUI picker should not freeze ordinary providers to its static
|
|
``_PROVIDER_MODELS`` snapshot when Hermes CLI can return a fresher provider
|
|
catalog. Static lists remain a fallback only.
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
import sys
|
|
import types
|
|
|
|
import api.config as config
|
|
|
|
|
|
_PROVIDER_ENV_VARS = (
|
|
"ANTHROPIC_API_KEY",
|
|
"OPENAI_API_KEY",
|
|
"OPENROUTER_API_KEY",
|
|
"GOOGLE_API_KEY",
|
|
"GEMINI_API_KEY",
|
|
"GLM_API_KEY",
|
|
"KIMI_API_KEY",
|
|
"DEEPSEEK_API_KEY",
|
|
"XIAOMI_API_KEY",
|
|
"OPENCODE_ZEN_API_KEY",
|
|
"OPENCODE_GO_API_KEY",
|
|
"MINIMAX_API_KEY",
|
|
"MINIMAX_CN_API_KEY",
|
|
"XAI_API_KEY",
|
|
"MISTRAL_API_KEY",
|
|
"OLLAMA_CLOUD_API_KEY",
|
|
"OLLAMA_API_KEY",
|
|
"NOUS_API_KEY",
|
|
"NVIDIA_API_KEY",
|
|
)
|
|
|
|
|
|
def _scrub_provider_env(monkeypatch):
|
|
for name in _PROVIDER_ENV_VARS:
|
|
monkeypatch.delenv(name, raising=False)
|
|
|
|
|
|
def _install_fake_hermes_cli(monkeypatch, *, provider_id: str, live_ids, raise_on_lookup: bool = False):
|
|
"""Install a hermes_cli stub that reports one authenticated provider."""
|
|
fake_pkg = types.ModuleType("hermes_cli")
|
|
fake_pkg.__path__ = []
|
|
|
|
fake_models = types.ModuleType("hermes_cli.models")
|
|
fake_models.list_available_providers = lambda: [
|
|
{"id": provider_id, "authenticated": True}
|
|
]
|
|
|
|
calls: list[str] = []
|
|
|
|
def provider_model_ids(pid):
|
|
calls.append(pid)
|
|
if raise_on_lookup:
|
|
raise RuntimeError("simulated provider_model_ids failure")
|
|
return list(live_ids) if pid == provider_id else []
|
|
|
|
fake_models.provider_model_ids = provider_model_ids
|
|
|
|
fake_auth = types.ModuleType("hermes_cli.auth")
|
|
|
|
def get_auth_status(pid):
|
|
if pid == provider_id:
|
|
return {"logged_in": True, "key_source": ""}
|
|
return {"logged_in": False, "key_source": ""}
|
|
|
|
fake_auth.get_auth_status = get_auth_status
|
|
|
|
monkeypatch.setitem(sys.modules, "hermes_cli", fake_pkg)
|
|
monkeypatch.setitem(sys.modules, "hermes_cli.models", fake_models)
|
|
monkeypatch.setitem(sys.modules, "hermes_cli.auth", fake_auth)
|
|
monkeypatch.delitem(sys.modules, "agent.credential_pool", raising=False)
|
|
monkeypatch.delitem(sys.modules, "agent", raising=False)
|
|
config.invalidate_models_cache()
|
|
return calls
|
|
|
|
|
|
def _configure(monkeypatch, tmp_path, *, provider: str, default: str = ""):
|
|
monkeypatch.setattr(config, "_get_config_path", lambda: tmp_path / "missing-config.yaml")
|
|
monkeypatch.setattr(config, "_models_cache_path", tmp_path / "models_cache.json")
|
|
monkeypatch.setattr(
|
|
config,
|
|
"cfg",
|
|
{
|
|
"model": {"provider": provider, "default": default},
|
|
"providers": {},
|
|
"fallback_providers": [],
|
|
},
|
|
)
|
|
monkeypatch.setattr(config, "_cfg_mtime", 0.0)
|
|
config.invalidate_models_cache()
|
|
|
|
|
|
def _provider_group(result: dict, provider_id: str) -> dict:
|
|
return next(g for g in result["groups"] if g.get("provider_id") == provider_id)
|
|
|
|
|
|
def _ids(group: dict) -> list[str]:
|
|
return [m.get("id") for m in group.get("models", [])]
|
|
|
|
|
|
def test_generic_provider_uses_hermes_cli_catalog_before_static_snapshot(monkeypatch, tmp_path):
|
|
"""A normal provider should show fresh CLI-discovered models.
|
|
|
|
``claude-sonnet-5.0`` is intentionally absent from WebUI's static Anthropic
|
|
list. Before this fix the group came entirely from ``_PROVIDER_MODELS`` and
|
|
this model was invisible even though Hermes CLI knew about it.
|
|
"""
|
|
_scrub_provider_env(monkeypatch)
|
|
calls = _install_fake_hermes_cli(
|
|
monkeypatch,
|
|
provider_id="anthropic",
|
|
live_ids=["claude-opus-4.7", "claude-sonnet-5.0"],
|
|
)
|
|
_configure(monkeypatch, tmp_path, provider="anthropic", default="claude-opus-4.7")
|
|
|
|
result = config.get_available_models()
|
|
group = _provider_group(result, "anthropic")
|
|
|
|
assert calls == ["anthropic"]
|
|
assert _ids(group) == ["claude-opus-4.7", "claude-sonnet-5.0"]
|
|
assert group["models"][1]["label"] == "Claude Sonnet 5.0"
|
|
|
|
|
|
def test_generic_provider_keeps_static_catalog_as_cli_failure_fallback(monkeypatch, tmp_path):
|
|
_scrub_provider_env(monkeypatch)
|
|
calls = _install_fake_hermes_cli(
|
|
monkeypatch,
|
|
provider_id="anthropic",
|
|
live_ids=[],
|
|
raise_on_lookup=True,
|
|
)
|
|
_configure(monkeypatch, tmp_path, provider="anthropic", default="claude-opus-4.7")
|
|
|
|
result = config.get_available_models()
|
|
group = _provider_group(result, "anthropic")
|
|
|
|
assert calls == ["anthropic"]
|
|
assert "claude-opus-4.7" in _ids(group)
|
|
assert "claude-sonnet-4.6" in _ids(group)
|
|
|
|
|
|
def test_generic_provider_prefixes_live_ids_when_not_active_provider(monkeypatch, tmp_path):
|
|
"""Provider-qualified live IDs must route through the selected provider."""
|
|
_scrub_provider_env(monkeypatch)
|
|
calls = _install_fake_hermes_cli(
|
|
monkeypatch,
|
|
provider_id="anthropic",
|
|
live_ids=["claude-sonnet-5.0"],
|
|
)
|
|
# Anthropic is authenticated via Hermes CLI, but OpenAI is the active
|
|
# default. The Anthropic row still has to be pickable/routable.
|
|
_configure(monkeypatch, tmp_path, provider="openai", default="gpt-5.5")
|
|
|
|
result = config.get_available_models()
|
|
group = _provider_group(result, "anthropic")
|
|
|
|
assert "anthropic" in calls
|
|
assert _ids(group) == ["@anthropic:claude-sonnet-5.0"]
|