mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-06-21 10:22:18 +00:00
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).
145 lines
4.9 KiB
Python
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
|