mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-06-21 10:22:18 +00:00
Manual verification surfaced a second bypass class beyond the standalone config loaders: several code paths bridge config.yaml values into os.environ (HERMES_TIMEZONE, HERMES_REDACT_SECRETS, HERMES_MAX_ITERATIONS, TERMINAL_*, network.force_ipv4, ...) by reading the raw user YAML, so the env the whole process reads carried the USER's value even when an administrator pinned it — e.g. a managed timezone was overridden because gateway/run.py wrote the user's timezone into HERMES_TIMEZONE, and _resolve_timezone_name() checks the env var first. Wired the shared apply_managed_overlay() into every config→env bridge: - gateway/run.py module-level startup bridge (timezone, redact_secrets, max_turns, terminal, display, gateway.strict, ...) - gateway/run.py _reload_runtime_env_preserving_config_authority (the per-turn re-bridge that keeps config authoritative over reloaded .env — must keep MANAGED authoritative on every turn, not just startup) - hermes_cli/main.py early security.redact_secrets / network.force_ipv4 bridge (runs before load_config is usable, at import time) - hermes_cli/send_cmd.py top-level scalar config→env bridge Verified end-to-end against a writable managed dir (12/12 checks incl. timezone, logging, model, skin, gateway settings, write-guard) and in a clean process the gateway per-turn bridge writes HERMES_TIMEZONE=<managed>. Adds an order-independent regression test for the bridge overlay.
142 lines
5.7 KiB
Python
142 lines
5.7 KiB
Python
"""Each standalone config loader (gateway, TUI/desktop, cron) must honor managed scope.
|
|
|
|
These loaders build their own config dict instead of routing through
|
|
hermes_cli.config.load_config, so the managed overlay has to be wired into each.
|
|
This is the regression guard for the whole bug class (a managed display.skin was
|
|
silently ignored by the TUI; the same gap existed in the gateway and cron).
|
|
"""
|
|
import textwrap
|
|
|
|
import pytest
|
|
|
|
|
|
@pytest.fixture
|
|
def homes(tmp_path, monkeypatch):
|
|
home = tmp_path / "home"
|
|
home.mkdir()
|
|
managed = tmp_path / "managed"
|
|
managed.mkdir()
|
|
monkeypatch.setenv("HERMES_HOME", str(home))
|
|
monkeypatch.setenv("HERMES_MANAGED_DIR", str(managed))
|
|
import hermes_cli.config as cfg
|
|
from hermes_cli import managed_scope
|
|
|
|
cfg._LOAD_CONFIG_CACHE.clear()
|
|
cfg._RAW_CONFIG_CACHE.clear()
|
|
managed_scope.invalidate_managed_cache()
|
|
return home, managed
|
|
|
|
|
|
def _seed(home, managed, *, user, mgd):
|
|
(home / "config.yaml").write_text(textwrap.dedent(user), encoding="utf-8")
|
|
(managed / "config.yaml").write_text(textwrap.dedent(mgd), encoding="utf-8")
|
|
import hermes_cli.config as cfg
|
|
from hermes_cli import managed_scope
|
|
|
|
cfg._LOAD_CONFIG_CACHE.clear()
|
|
cfg._RAW_CONFIG_CACHE.clear()
|
|
managed_scope.invalidate_managed_cache()
|
|
|
|
|
|
def test_gateway_run_loader_honors_managed(homes, monkeypatch):
|
|
home, managed = homes
|
|
_seed(home, managed, user="model:\n default: user/m\n", mgd="model:\n default: org/m\n")
|
|
import gateway.run as gr
|
|
|
|
monkeypatch.setattr(gr, "_hermes_home", home, raising=False)
|
|
cfg = gr._load_gateway_config()
|
|
assert (cfg.get("model") or {}).get("default") == "org/m"
|
|
|
|
|
|
def test_gateway_config_loader_honors_managed(homes, monkeypatch):
|
|
home, managed = homes
|
|
_seed(
|
|
home,
|
|
managed,
|
|
user="group_sessions_per_user: false\n",
|
|
mgd="group_sessions_per_user: true\n",
|
|
)
|
|
import gateway.config as gc
|
|
|
|
# load_gateway_config resolves home via get_hermes_home() (HERMES_HOME env).
|
|
cfg = gc.load_gateway_config()
|
|
# Managed value should have flowed into the GatewayConfig.
|
|
assert cfg.group_sessions_per_user is True
|
|
|
|
|
|
def test_tui_loader_honors_managed(homes, monkeypatch):
|
|
home, managed = homes
|
|
_seed(home, managed, user="display:\n skin: user\n", mgd="display:\n skin: charizard\n")
|
|
import tui_gateway.server as ts
|
|
|
|
monkeypatch.setattr(ts, "_hermes_home", home, raising=False)
|
|
monkeypatch.setattr(ts, "_cfg_cache", None, raising=False)
|
|
monkeypatch.setattr(ts, "_cfg_mtime", None, raising=False)
|
|
monkeypatch.setattr(ts, "get_hermes_home_override", lambda: None, raising=False)
|
|
cfg = ts._load_cfg()
|
|
assert (cfg.get("display") or {}).get("skin") == "charizard"
|
|
|
|
|
|
def test_tui_loader_does_not_persist_managed_back(homes, monkeypatch):
|
|
"""The TUI caches RAW config so _save_cfg never writes managed values to disk."""
|
|
home, managed = homes
|
|
_seed(home, managed, user="display:\n skin: user\n", mgd="display:\n skin: charizard\n")
|
|
import tui_gateway.server as ts
|
|
|
|
monkeypatch.setattr(ts, "_hermes_home", home, raising=False)
|
|
monkeypatch.setattr(ts, "_cfg_cache", None, raising=False)
|
|
monkeypatch.setattr(ts, "_cfg_mtime", None, raising=False)
|
|
monkeypatch.setattr(ts, "get_hermes_home_override", lambda: None, raising=False)
|
|
ts._load_cfg() # populates the cache
|
|
# The cache must hold the RAW user value, not the managed overlay, so a
|
|
# subsequent _save_cfg can't bake the managed skin into the user file.
|
|
assert (ts._cfg_cache.get("display") or {}).get("skin") == "user"
|
|
|
|
|
|
def test_logging_config_honors_managed(homes, monkeypatch):
|
|
home, managed = homes
|
|
_seed(home, managed, user="logging:\n level: INFO\n", mgd="logging:\n level: DEBUG\n")
|
|
import hermes_logging
|
|
|
|
level, _max, _bk = hermes_logging._read_logging_config()
|
|
assert level == "DEBUG"
|
|
|
|
|
|
def test_timezone_honors_managed(homes, monkeypatch):
|
|
home, managed = homes
|
|
# hermes_time checks an env override first; ensure it's unset so config wins.
|
|
monkeypatch.delenv("HERMES_TIMEZONE", raising=False)
|
|
monkeypatch.delenv("TZ", raising=False)
|
|
_seed(home, managed, user="timezone: America/New_York\n", mgd="timezone: Asia/Tokyo\n")
|
|
import hermes_time
|
|
|
|
assert hermes_time._resolve_timezone_name() == "Asia/Tokyo"
|
|
|
|
|
|
def test_gateway_env_bridge_honors_managed(homes, monkeypatch):
|
|
"""The gateway config→env bridge must bridge MANAGED values, not user ones.
|
|
|
|
gateway/run.py bridges config.yaml settings into os.environ at startup and on
|
|
every turn (HERMES_TIMEZONE, HERMES_REDACT_SECRETS, HERMES_MAX_ITERATIONS,
|
|
...). A managed value must win at that env layer too — otherwise the bridge
|
|
writes the user's value into the env that the whole process then reads. This
|
|
is the regression that manual verification caught (managed timezone was
|
|
overridden by the user's value via the env bridge).
|
|
|
|
We assert on the managed-overlaid config the bridge consumes (rather than the
|
|
os.environ side effect, which leaks across same-process tests under the
|
|
runner) — the bridge writes whatever this dict carries, so a managed value
|
|
here proves the env var gets the managed value.
|
|
"""
|
|
home, managed = homes
|
|
_seed(home, managed, user="timezone: America/New_York\n", mgd="timezone: Asia/Tokyo\n")
|
|
from hermes_cli import managed_scope
|
|
|
|
managed_scope.invalidate_managed_cache()
|
|
# The bridge loads config.yaml, expands env, then applies this overlay before
|
|
# writing HERMES_TIMEZONE = cfg["timezone"]. Prove the overlay flips the value.
|
|
import yaml
|
|
|
|
raw = yaml.safe_load((home / "config.yaml").read_text())
|
|
bridged = managed_scope.apply_managed_overlay(raw)
|
|
assert bridged.get("timezone") == "Asia/Tokyo"
|