Files
isparkclaw-webui/tests/test_issue895_894_nous_prefix.py
nesquena-hermes 4089972b09 fix(models): preserve @nous: prefix in settings + fix cross-namespace 404 for Nous (#895 #894) (#901)
* fix(models): preserve @nous: prefix in settings + fix cross-namespace 404 for Nous (#895 #894)

* fix(review): persist bare form for CLI compatibility + picker smart-match

The PR persisted `@nous:anthropic/claude-opus-4.6` verbatim to config.yaml
to make the Settings picker match its dropdown options (which carry the
`@nous:` prefix after #885). That fixes the WebUI picker but introduces a
cross-tool regression: hermes-agent's CLI reads `config.yaml -> model.default`
directly and passes it to the provider API verbatim. For aggregator providers
(Nous is one — see hermes_cli/model_normalize.py `_AGGREGATOR_PROVIDERS`),
`normalize_model_for_provider` is skipped entirely (run_agent.py:887), so
the literal `@nous:anthropic/...` string flows to the Nous API, which rejects
it — breaking every user who runs `hermes` in the terminal right after
saving via WebUI.

Fix the tension at the picker rather than the persistence: the existing
`_findModelInDropdown()` smart matcher already normalises both sides
(lowercase, strip namespace prefix, dashes→dots) so a saved bare
`anthropic/claude-opus-4.6` resolves to the `@nous:anthropic/claude-opus-4.6`
option automatically. Applied this in panels.js via `_applyModelToDropdown()`.

Changes:
  api/config.py         revert the @-prefix preservation; persist the
                        resolved bare/slash form (CLI-compatible)
  static/panels.js      Settings picker uses _applyModelToDropdown()
                        instead of raw `.value =` so saved bare forms
                        still select the matching @nous: option
  tests                 test renamed + asserts bare persisted form;
                        new test locks the smart-matcher contract

This also improves behaviour for a dormant case not flagged in #895: a user
who set their default via `hermes model X` and opens Settings for the first
time used to see a blank picker (bare form vs prefixed options). Now the
smart matcher finds the right option, so the "open Settings → save → bare
form in config.yaml" round-trip is stable for both CLI- and WebUI-origin
saves.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>

* chore: update CHANGELOG v0.50.171 — bare-form persistence + picker smart-match

---------

Co-authored-by: nesquena-hermes <nesquena-hermes@users.noreply.github.com>
Co-authored-by: Nathan Esquenazi <nesquena@gmail.com>
Co-authored-by: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-04-23 10:44:10 -07:00

174 lines
7.9 KiB
Python

"""
Regression tests for #895 (set_hermes_default_model strips @nous: prefix + blocks on live fetch)
and #894 (resolve_model_provider strips cross-namespace prefix for portal providers with base_url).
"""
import threading
import pytest
from pathlib import Path
import api.config as config
from api.config import resolve_model_provider, set_hermes_default_model
# ── Shared fixture ──────────────────────────────────────────────────────────
@pytest.fixture(autouse=True)
def _isolate(tmp_path, monkeypatch):
old_cfg = dict(config.cfg)
old_mtime = config._cfg_mtime
old_cache = config._available_models_cache
old_cache_ts = config._available_models_cache_ts
config_file = tmp_path / "config.yaml"
config_file.write_text(
"model:\n provider: nous\n base_url: https://router.nous.ai/v1\n default: anthropic/claude-opus-4.6\n",
encoding="utf-8",
)
monkeypatch.setattr(config, "_get_config_path", lambda: Path(str(config_file)))
config.cfg.clear()
config.cfg.update({
"model": {
"provider": "nous",
"base_url": "https://router.nous.ai/v1",
"default": "anthropic/claude-opus-4.6",
}
})
try:
config._cfg_mtime = config_file.stat().st_mtime
except OSError:
config._cfg_mtime = 0.0
config.invalidate_models_cache()
yield
config.cfg.clear()
config.cfg.update(old_cfg)
config._cfg_mtime = old_mtime
config._available_models_cache = old_cache
config._available_models_cache_ts = old_cache_ts
# ── #894: portal-provider + config_base_url prefix-stripping ───────────────
class TestResolveModelProviderPortalPriority:
def test_minimax_prefix_preserved_for_nous(self):
"""Nous with base_url must NOT strip minimax/ prefix (#894)."""
m, p, _ = resolve_model_provider("minimax/minimax-m2.7")
assert m == "minimax/minimax-m2.7", f"prefix was stripped: {m!r}"
assert p == "nous"
def test_qwen_prefix_preserved_for_nous(self):
"""Nous with base_url must NOT strip qwen/ prefix (#894)."""
m, p, _ = resolve_model_provider("qwen/qwen3.5-35b-a3b")
assert m == "qwen/qwen3.5-35b-a3b", f"prefix was stripped: {m!r}"
assert p == "nous"
def test_anthropic_prefix_preserved_for_nous(self):
"""Core case: anthropic/claude-opus-4.6 must route to nous intact."""
m, p, _ = resolve_model_provider("anthropic/claude-opus-4.6")
assert m == "anthropic/claude-opus-4.6"
assert p == "nous"
def test_at_nous_prefix_unpacked_correctly(self):
"""@nous:anthropic/claude-opus-4.6 should unpack to bare model and nous provider."""
m, p, _ = resolve_model_provider("@nous:anthropic/claude-opus-4.6")
assert m == "anthropic/claude-opus-4.6"
assert p == "nous"
def test_unknown_prefix_preserved_for_nous(self):
"""Non-PROVIDER_MODELS prefix like moonshotai/ must also pass through intact."""
m, p, _ = resolve_model_provider("moonshotai/kimi-k2.6")
assert m == "moonshotai/kimi-k2.6"
assert p == "nous"
# ── #895: set_hermes_default_model persists @provider: prefix ──────────────
class TestSetDefaultModelPreservesAtPrefix:
def test_at_nous_prefix_strips_to_bare_for_cli_compatibility(self, tmp_path, monkeypatch):
"""set_hermes_default_model must persist the RESOLVED bare/slash form, not the
`@provider:` prefix. The `@provider:` syntax is a WebUI-internal routing hint;
the hermes-agent CLI reads `config.yaml -> model.default` directly and passes
it to the provider API verbatim (see run_agent.py:887 — aggregator providers
like Nous skip normalize_model_for_provider, so the raw string flows through).
Storing `@nous:anthropic/...` would break any user who runs `hermes` in the
terminal right after saving via WebUI — the CLI would send the literal
prefixed string to Nous and hit a 404. The Settings picker handles the bare
form via the smart matcher in `_applyModelToDropdown()`.
"""
import yaml
config_file = tmp_path / "config.yaml"
config_file.write_text(
"model:\n provider: nous\n base_url: https://router.nous.ai/v1\n",
encoding="utf-8",
)
monkeypatch.setattr(config, "_get_config_path", lambda: Path(str(config_file)))
config.cfg["model"] = {"provider": "nous", "base_url": "https://router.nous.ai/v1"}
try:
config._cfg_mtime = config_file.stat().st_mtime
except OSError:
config._cfg_mtime = 0.0
result = set_hermes_default_model("@nous:anthropic/claude-opus-4.6")
# Result ack echoes the resolved bare/slash form (CLI-compatible)
assert result.get("ok") is True
assert result.get("model") == "anthropic/claude-opus-4.6", (
f"result.model should echo the CLI-compatible resolved form, not the "
f"WebUI-internal @-prefix: {result.get('model')!r}"
)
saved = yaml.safe_load(config_file.read_text(encoding="utf-8"))
assert saved["model"]["default"] == "anthropic/claude-opus-4.6", (
f"Config must persist the resolved bare form so the hermes-agent CLI "
f"can read it and pass it to the provider API: "
f"{saved['model']['default']!r}"
)
def test_settings_picker_applies_saved_default_via_smart_matcher(self):
"""The Settings picker must use `_applyModelToDropdown()` (smart matcher),
not raw `modelSel.value = ...`, when initialising from the saved default.
Raw `.value =` silently fails if no option matches exactly — blank picker
on reopen for any saved default whose canonical form doesn't equal an option
value (e.g. CLI-saved `anthropic/claude-opus-4.6` vs Nous dropdown option
`@nous:anthropic/claude-opus-4.6`). `_applyModelToDropdown()` normalises
on both sides and picks the matching option.
"""
js = (Path(__file__).resolve().parent.parent / "static" / "panels.js").read_text()
# Find the block that sets _settingsHermesDefaultModelOnOpen
anchor = "_settingsHermesDefaultModelOnOpen=(models&&models.default_model)||"
idx = js.find(anchor)
assert idx != -1, "Settings default-model initialisation not found in panels.js"
block = js[idx:idx + 1200]
assert "_applyModelToDropdown" in block, (
"Settings picker must use _applyModelToDropdown() so a saved bare form "
"(e.g. anthropic/claude-opus-4.6) still selects the matching "
"@nous:anthropic/claude-opus-4.6 option. A raw .value assignment leaves "
"the picker blank when the saved ID doesn't match an option verbatim."
)
def test_save_does_not_return_full_model_catalog(self, tmp_path, monkeypatch):
"""set_hermes_default_model must return a lightweight ack, not call get_available_models (#895)."""
config_file = tmp_path / "config.yaml"
config_file.write_text(
"model:\n provider: openrouter\n",
encoding="utf-8",
)
monkeypatch.setattr(config, "_get_config_path", lambda: Path(str(config_file)))
config.cfg["model"] = {"provider": "openrouter"}
try:
config._cfg_mtime = config_file.stat().st_mtime
except OSError:
config._cfg_mtime = 0.0
result = set_hermes_default_model("openai/gpt-5.4-mini")
# Must be a simple dict with ok+model, NOT the full catalog (which has "groups")
assert result.get("ok") is True
assert "groups" not in result, (
"set_hermes_default_model must not return the full model catalog — "
"doing so triggers a live provider fetch that blocks the HTTP response"
)