hermes-agent/tests/hermes_cli/test_managed_scope.py
Ben 9cbcc0c9c8 feat(managed-scope): add managed_scope module (resolver, loaders, key helpers)
New hermes_cli/managed_scope.py resolves a system-level managed directory
(HERMES_MANAGED_DIR override > /etc/hermes), parses managed config.yaml/.env
with fail-open semantics, and exposes is_key_managed/is_env_managed helpers.
The system default is ignored under pytest and HERMES_MANAGED_DIR is added to
the conftest env scrub so a real managed scope can't leak into the suite.

Not wired into the load paths yet (Phases 2-3).
2026-06-19 07:46:33 -07:00

145 lines
4.9 KiB
Python

"""Unit tests for hermes_cli.managed_scope (resolver + loaders + key helpers)."""
import textwrap
import pytest
# ── Directory resolver ───────────────────────────────────────────────────────
def test_get_managed_dir_env_override(tmp_path, monkeypatch):
from hermes_cli import managed_scope
managed = tmp_path / "managed"
managed.mkdir()
monkeypatch.setenv("HERMES_MANAGED_DIR", str(managed))
assert managed_scope.get_managed_dir() == managed
def test_get_managed_dir_absent_override_returns_none(tmp_path, monkeypatch):
from hermes_cli import managed_scope
monkeypatch.setenv("HERMES_MANAGED_DIR", str(tmp_path / "nope"))
# Override points at a non-existent dir → no managed scope.
assert managed_scope.get_managed_dir() is None
def test_get_managed_dir_empty_override_falls_through(tmp_path, monkeypatch):
from hermes_cli import managed_scope
monkeypatch.setenv("HERMES_MANAGED_DIR", " ") # whitespace = unset
# Under pytest the /etc/hermes default is ignored, so this is None; the
# assertion that matters is that it does NOT raise.
result = managed_scope.get_managed_dir()
assert result is None or result.exists()
def test_get_managed_dir_default_ignored_under_pytest(monkeypatch):
"""The system default must be inert in the test suite (isolation guard)."""
from hermes_cli import managed_scope
monkeypatch.delenv("HERMES_MANAGED_DIR", raising=False)
assert managed_scope.get_managed_dir() is None
# ── Loaders + key helpers ────────────────────────────────────────────────────
def _write_managed(tmp_path, monkeypatch, *, config=None, env=None):
from hermes_cli import managed_scope
managed = tmp_path / "managed"
managed.mkdir(exist_ok=True)
if config is not None:
(managed / "config.yaml").write_text(textwrap.dedent(config), encoding="utf-8")
if env is not None:
(managed / ".env").write_text(textwrap.dedent(env), encoding="utf-8")
monkeypatch.setenv("HERMES_MANAGED_DIR", str(managed))
managed_scope.invalidate_managed_cache()
return managed
def test_load_managed_config(tmp_path, monkeypatch):
from hermes_cli import managed_scope
_write_managed(
tmp_path,
monkeypatch,
config="""
model:
default: managed/model
""",
)
assert managed_scope.load_managed_config() == {"model": {"default": "managed/model"}}
def test_load_managed_config_absent_is_empty(tmp_path, monkeypatch):
from hermes_cli import managed_scope
monkeypatch.setenv("HERMES_MANAGED_DIR", str(tmp_path / "nope"))
managed_scope.invalidate_managed_cache()
assert managed_scope.load_managed_config() == {}
def test_load_managed_config_malformed_fails_open(tmp_path, monkeypatch):
from hermes_cli import managed_scope
_write_managed(tmp_path, monkeypatch, config="model: : : not yaml :")
assert managed_scope.load_managed_config() == {} # fail-open, no raise
def test_managed_config_keys_are_dotted_leaves(tmp_path, monkeypatch):
from hermes_cli import managed_scope
_write_managed(
tmp_path,
monkeypatch,
config="""
model:
default: m
security:
redact_secrets: true
""",
)
assert managed_scope.managed_config_keys() == {
"model.default",
"security.redact_secrets",
}
def test_is_key_managed(tmp_path, monkeypatch):
from hermes_cli import managed_scope
_write_managed(tmp_path, monkeypatch, config="model:\n default: m\n")
assert managed_scope.is_key_managed("model.default") is True
assert managed_scope.is_key_managed("model.fallback") is False
def test_load_managed_env_and_is_env_managed(tmp_path, monkeypatch):
from hermes_cli import managed_scope
_write_managed(
tmp_path, monkeypatch, env="OPENAI_API_BASE=https://org.example/v1\n"
)
assert managed_scope.load_managed_env() == {
"OPENAI_API_BASE": "https://org.example/v1"
}
assert managed_scope.is_env_managed("OPENAI_API_BASE") is True
assert managed_scope.is_env_managed("OTHER") is False
def test_editing_managed_config_invalidates_cache(tmp_path, monkeypatch):
from hermes_cli import managed_scope
managed = _write_managed(tmp_path, monkeypatch, config="model:\n default: v1\n")
assert managed_scope.load_managed_config()["model"]["default"] == "v1"
(managed / "config.yaml").write_text("model:\n default: v2\n", encoding="utf-8")
managed_scope.invalidate_managed_cache()
assert managed_scope.load_managed_config()["model"]["default"] == "v2"
def test_managed_dir_env_scrubbed_by_default():
"""conftest must scrub HERMES_MANAGED_DIR so a dev-shell value can't leak in."""
import os
assert "HERMES_MANAGED_DIR" not in os.environ