mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-04-28 15:01:34 +08:00
85 lines
3.1 KiB
Python
85 lines
3.1 KiB
Python
|
|
"""Tests for hermes_cli.auth._update_config_for_provider clearing stale fields.
|
||
|
|
|
||
|
|
When the user switches from a custom provider (e.g. MiniMax with
|
||
|
|
``api_mode: anthropic_messages``, ``api_key: mxp-...``) to a built-in
|
||
|
|
provider (e.g. OpenRouter), the stale ``api_key`` and ``api_mode`` would
|
||
|
|
otherwise override the new provider's credentials and transport choice.
|
||
|
|
|
||
|
|
Built-in providers that legitimately need a specific ``api_mode`` (copilot,
|
||
|
|
xai) compute it at request-resolution time in
|
||
|
|
``_copilot_runtime_api_mode`` / ``_detect_api_mode_for_url``, so removing
|
||
|
|
the persisted value here is safe.
|
||
|
|
"""
|
||
|
|
|
||
|
|
from __future__ import annotations
|
||
|
|
|
||
|
|
import yaml
|
||
|
|
|
||
|
|
from hermes_cli.auth import _update_config_for_provider
|
||
|
|
from hermes_cli.config import get_config_path
|
||
|
|
|
||
|
|
|
||
|
|
def _read_model_cfg() -> dict:
|
||
|
|
path = get_config_path()
|
||
|
|
if not path.exists():
|
||
|
|
return {}
|
||
|
|
data = yaml.safe_load(path.read_text()) or {}
|
||
|
|
model = data.get("model", {})
|
||
|
|
return model if isinstance(model, dict) else {}
|
||
|
|
|
||
|
|
|
||
|
|
def _seed_custom_provider_config(api_mode: str = "anthropic_messages") -> None:
|
||
|
|
"""Write a config.yaml mimicking a user on a MiniMax-style custom provider."""
|
||
|
|
path = get_config_path()
|
||
|
|
path.parent.mkdir(parents=True, exist_ok=True)
|
||
|
|
path.write_text(
|
||
|
|
yaml.safe_dump(
|
||
|
|
{
|
||
|
|
"model": {
|
||
|
|
"provider": "custom",
|
||
|
|
"base_url": "https://api.minimax.io/anthropic",
|
||
|
|
"api_key": "mxp-stale-key",
|
||
|
|
"api_mode": api_mode,
|
||
|
|
"default": "claude-sonnet-4-6",
|
||
|
|
}
|
||
|
|
},
|
||
|
|
sort_keys=False,
|
||
|
|
)
|
||
|
|
)
|
||
|
|
|
||
|
|
|
||
|
|
class TestUpdateConfigForProviderClearsStaleCustomFields:
|
||
|
|
def test_switching_to_openrouter_clears_api_key_and_api_mode(self):
|
||
|
|
_seed_custom_provider_config()
|
||
|
|
|
||
|
|
_update_config_for_provider(
|
||
|
|
"openrouter",
|
||
|
|
"https://openrouter.ai/api/v1",
|
||
|
|
default_model="anthropic/claude-sonnet-4.6",
|
||
|
|
)
|
||
|
|
|
||
|
|
model_cfg = _read_model_cfg()
|
||
|
|
assert model_cfg.get("provider") == "openrouter"
|
||
|
|
assert model_cfg.get("base_url") == "https://openrouter.ai/api/v1"
|
||
|
|
assert "api_key" not in model_cfg, (
|
||
|
|
"Stale custom api_key would leak into OpenRouter requests — must be cleared"
|
||
|
|
)
|
||
|
|
assert "api_mode" not in model_cfg, (
|
||
|
|
"Stale api_mode=anthropic_messages from MiniMax would mis-route "
|
||
|
|
"OpenRouter requests to the Anthropic SDK — must be cleared"
|
||
|
|
)
|
||
|
|
|
||
|
|
def test_switching_to_nous_clears_stale_api_mode(self):
|
||
|
|
_seed_custom_provider_config()
|
||
|
|
_update_config_for_provider("nous", "https://inference-api.nousresearch.com/v1")
|
||
|
|
model_cfg = _read_model_cfg()
|
||
|
|
assert model_cfg.get("provider") == "nous"
|
||
|
|
assert "api_mode" not in model_cfg
|
||
|
|
assert "api_key" not in model_cfg
|
||
|
|
|
||
|
|
def test_switching_clears_codex_responses_api_mode(self):
|
||
|
|
"""Also covers codex_responses, not just anthropic_messages."""
|
||
|
|
_seed_custom_provider_config(api_mode="codex_responses")
|
||
|
|
_update_config_for_provider("openrouter", "https://openrouter.ai/api/v1")
|
||
|
|
assert "api_mode" not in _read_model_cfg()
|