mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-04-28 06:51:16 +08:00
288 lines
10 KiB
Python
288 lines
10 KiB
Python
|
|
"""Tests for the setup wizard's returning-user behavior.
|
||
|
|
|
||
|
|
On an existing install:
|
||
|
|
- Bare `hermes setup` drops straight into the full reconfigure wizard
|
||
|
|
(every prompt shows the current value as its default).
|
||
|
|
- `hermes setup --quick` runs the narrower "fill in missing items" flow.
|
||
|
|
- `hermes setup --reconfigure` is a backwards-compat alias for the
|
||
|
|
bare-setup default.
|
||
|
|
|
||
|
|
On a fresh install, all three are no-ops — fall through to first-time setup.
|
||
|
|
"""
|
||
|
|
|
||
|
|
from argparse import Namespace
|
||
|
|
from contextlib import ExitStack
|
||
|
|
from unittest.mock import patch
|
||
|
|
|
||
|
|
import pytest
|
||
|
|
|
||
|
|
|
||
|
|
def _make_setup_args(**overrides):
|
||
|
|
return Namespace(
|
||
|
|
non_interactive=overrides.get("non_interactive", False),
|
||
|
|
section=overrides.get("section", None),
|
||
|
|
reset=overrides.get("reset", False),
|
||
|
|
reconfigure=overrides.get("reconfigure", False),
|
||
|
|
quick=overrides.get("quick", False),
|
||
|
|
)
|
||
|
|
|
||
|
|
|
||
|
|
@pytest.fixture
|
||
|
|
def existing_install(tmp_path, monkeypatch):
|
||
|
|
"""Simulate a returning user with an existing configured install."""
|
||
|
|
home = tmp_path / ".hermes"
|
||
|
|
home.mkdir()
|
||
|
|
monkeypatch.setattr("pathlib.Path.home", lambda: tmp_path)
|
||
|
|
monkeypatch.setenv("HERMES_HOME", str(home))
|
||
|
|
return home
|
||
|
|
|
||
|
|
|
||
|
|
@pytest.fixture
|
||
|
|
def fresh_install(tmp_path, monkeypatch):
|
||
|
|
"""Simulate a first-time user with no existing configuration."""
|
||
|
|
home = tmp_path / ".hermes"
|
||
|
|
home.mkdir()
|
||
|
|
monkeypatch.setattr("pathlib.Path.home", lambda: tmp_path)
|
||
|
|
monkeypatch.setenv("HERMES_HOME", str(home))
|
||
|
|
return home
|
||
|
|
|
||
|
|
|
||
|
|
def _enter_existing_install_patches(stack, **extra):
|
||
|
|
"""Apply standard existing-install mocks via an ExitStack.
|
||
|
|
|
||
|
|
Returns a dict of mocks from the `extra` kwargs (which map mock-name to
|
||
|
|
target path) so callers can assert on them.
|
||
|
|
"""
|
||
|
|
# Unconditional mocks (no return values to assert against).
|
||
|
|
for target, kwargs in [
|
||
|
|
("hermes_cli.setup.ensure_hermes_home", {}),
|
||
|
|
("hermes_cli.setup.is_interactive_stdin", {"return_value": True}),
|
||
|
|
("hermes_cli.config.is_managed", {"return_value": False}),
|
||
|
|
("hermes_cli.setup.load_config", {"return_value": {}}),
|
||
|
|
("hermes_cli.setup.save_config", {}),
|
||
|
|
("hermes_cli.setup.get_env_value", {"return_value": None}),
|
||
|
|
("hermes_cli.auth.get_active_provider", {"return_value": "openrouter"}),
|
||
|
|
("hermes_cli.setup._print_setup_summary", {}),
|
||
|
|
("hermes_cli.setup._offer_launch_chat", {}),
|
||
|
|
("hermes_cli.setup._offer_openclaw_migration", {"return_value": False}),
|
||
|
|
]:
|
||
|
|
stack.enter_context(patch(target, **kwargs))
|
||
|
|
|
||
|
|
# Named mocks caller wants to assert on.
|
||
|
|
named = {}
|
||
|
|
for name, target in extra.items():
|
||
|
|
named[name] = stack.enter_context(patch(target))
|
||
|
|
return named
|
||
|
|
|
||
|
|
|
||
|
|
def _enter_fresh_install_patches(stack, **extra):
|
||
|
|
for target, kwargs in [
|
||
|
|
("hermes_cli.setup.ensure_hermes_home", {}),
|
||
|
|
("hermes_cli.setup.is_interactive_stdin", {"return_value": True}),
|
||
|
|
("hermes_cli.config.is_managed", {"return_value": False}),
|
||
|
|
("hermes_cli.setup.load_config", {"return_value": {}}),
|
||
|
|
("hermes_cli.setup.save_config", {}),
|
||
|
|
("hermes_cli.auth.get_active_provider", {"return_value": None}),
|
||
|
|
("hermes_cli.setup.get_env_value", {"return_value": None}),
|
||
|
|
("hermes_cli.setup._offer_openclaw_migration", {"return_value": False}),
|
||
|
|
]:
|
||
|
|
stack.enter_context(patch(target, **kwargs))
|
||
|
|
|
||
|
|
named = {}
|
||
|
|
for name, target_spec in extra.items():
|
||
|
|
if isinstance(target_spec, tuple):
|
||
|
|
target, kwargs = target_spec
|
||
|
|
named[name] = stack.enter_context(patch(target, **kwargs))
|
||
|
|
else:
|
||
|
|
named[name] = stack.enter_context(patch(target_spec))
|
||
|
|
return named
|
||
|
|
|
||
|
|
|
||
|
|
class TestExistingInstallDefault:
|
||
|
|
"""Bare `hermes setup` on an existing install = full reconfigure wizard."""
|
||
|
|
|
||
|
|
def test_bare_setup_runs_full_reconfigure_without_menu(self, existing_install):
|
||
|
|
"""No menu, no prompt_choice — just run every section in sequence."""
|
||
|
|
args = _make_setup_args() # no flags
|
||
|
|
|
||
|
|
with ExitStack() as stack:
|
||
|
|
m = _enter_existing_install_patches(
|
||
|
|
stack,
|
||
|
|
prompt_choice="hermes_cli.setup.prompt_choice",
|
||
|
|
quick="hermes_cli.setup._run_quick_setup",
|
||
|
|
model="hermes_cli.setup.setup_model_provider",
|
||
|
|
terminal="hermes_cli.setup.setup_terminal_backend",
|
||
|
|
agent="hermes_cli.setup.setup_agent_settings",
|
||
|
|
gateway="hermes_cli.setup.setup_gateway",
|
||
|
|
tools="hermes_cli.setup.setup_tools",
|
||
|
|
)
|
||
|
|
from hermes_cli.setup import run_setup_wizard
|
||
|
|
run_setup_wizard(args)
|
||
|
|
|
||
|
|
# No menu shown.
|
||
|
|
m["prompt_choice"].assert_not_called()
|
||
|
|
# Quick-setup path NOT taken.
|
||
|
|
m["quick"].assert_not_called()
|
||
|
|
# All five sections ran.
|
||
|
|
m["model"].assert_called_once()
|
||
|
|
m["terminal"].assert_called_once()
|
||
|
|
m["agent"].assert_called_once()
|
||
|
|
m["gateway"].assert_called_once()
|
||
|
|
m["tools"].assert_called_once()
|
||
|
|
|
||
|
|
def test_reconfigure_flag_is_backwards_compat_noop(self, existing_install):
|
||
|
|
"""`hermes setup --reconfigure` behaves the same as bare `hermes setup`."""
|
||
|
|
args = _make_setup_args(reconfigure=True)
|
||
|
|
|
||
|
|
with ExitStack() as stack:
|
||
|
|
m = _enter_existing_install_patches(
|
||
|
|
stack,
|
||
|
|
prompt_choice="hermes_cli.setup.prompt_choice",
|
||
|
|
model="hermes_cli.setup.setup_model_provider",
|
||
|
|
terminal="hermes_cli.setup.setup_terminal_backend",
|
||
|
|
agent="hermes_cli.setup.setup_agent_settings",
|
||
|
|
gateway="hermes_cli.setup.setup_gateway",
|
||
|
|
tools="hermes_cli.setup.setup_tools",
|
||
|
|
)
|
||
|
|
from hermes_cli.setup import run_setup_wizard
|
||
|
|
run_setup_wizard(args)
|
||
|
|
|
||
|
|
m["prompt_choice"].assert_not_called()
|
||
|
|
m["model"].assert_called_once()
|
||
|
|
m["terminal"].assert_called_once()
|
||
|
|
m["agent"].assert_called_once()
|
||
|
|
m["gateway"].assert_called_once()
|
||
|
|
m["tools"].assert_called_once()
|
||
|
|
|
||
|
|
|
||
|
|
class TestQuickFlag:
|
||
|
|
"""`--quick` on an existing install runs the fill-missing flow."""
|
||
|
|
|
||
|
|
def test_quick_flag_runs_quick_setup_only(self, existing_install):
|
||
|
|
args = _make_setup_args(quick=True)
|
||
|
|
|
||
|
|
with ExitStack() as stack:
|
||
|
|
m = _enter_existing_install_patches(
|
||
|
|
stack,
|
||
|
|
quick="hermes_cli.setup._run_quick_setup",
|
||
|
|
model="hermes_cli.setup.setup_model_provider",
|
||
|
|
terminal="hermes_cli.setup.setup_terminal_backend",
|
||
|
|
agent="hermes_cli.setup.setup_agent_settings",
|
||
|
|
gateway="hermes_cli.setup.setup_gateway",
|
||
|
|
tools="hermes_cli.setup.setup_tools",
|
||
|
|
)
|
||
|
|
from hermes_cli.setup import run_setup_wizard
|
||
|
|
run_setup_wizard(args)
|
||
|
|
|
||
|
|
m["quick"].assert_called_once()
|
||
|
|
# Full reconfigure sections must NOT run.
|
||
|
|
m["model"].assert_not_called()
|
||
|
|
m["terminal"].assert_not_called()
|
||
|
|
m["agent"].assert_not_called()
|
||
|
|
m["gateway"].assert_not_called()
|
||
|
|
m["tools"].assert_not_called()
|
||
|
|
|
||
|
|
|
||
|
|
class TestFreshInstall:
|
||
|
|
"""On a fresh install (no active provider), flags are no-ops."""
|
||
|
|
|
||
|
|
def test_bare_setup_runs_first_time_flow(self, fresh_install):
|
||
|
|
args = _make_setup_args()
|
||
|
|
|
||
|
|
with ExitStack() as stack:
|
||
|
|
m = _enter_fresh_install_patches(
|
||
|
|
stack,
|
||
|
|
prompt=("hermes_cli.setup.prompt_choice", {"return_value": 0}),
|
||
|
|
first="hermes_cli.setup._run_first_time_quick_setup",
|
||
|
|
)
|
||
|
|
from hermes_cli.setup import run_setup_wizard
|
||
|
|
run_setup_wizard(args)
|
||
|
|
|
||
|
|
m["prompt"].assert_called_once() # quick-vs-full prompt
|
||
|
|
m["first"].assert_called_once()
|
||
|
|
|
||
|
|
def test_reconfigure_on_fresh_install_falls_through(self, fresh_install):
|
||
|
|
args = _make_setup_args(reconfigure=True)
|
||
|
|
|
||
|
|
with ExitStack() as stack:
|
||
|
|
m = _enter_fresh_install_patches(
|
||
|
|
stack,
|
||
|
|
prompt=("hermes_cli.setup.prompt_choice", {"return_value": 0}),
|
||
|
|
first="hermes_cli.setup._run_first_time_quick_setup",
|
||
|
|
)
|
||
|
|
from hermes_cli.setup import run_setup_wizard
|
||
|
|
run_setup_wizard(args)
|
||
|
|
|
||
|
|
m["prompt"].assert_called_once()
|
||
|
|
m["first"].assert_called_once()
|
||
|
|
|
||
|
|
def test_quick_on_fresh_install_falls_through(self, fresh_install):
|
||
|
|
args = _make_setup_args(quick=True)
|
||
|
|
|
||
|
|
with ExitStack() as stack:
|
||
|
|
m = _enter_fresh_install_patches(
|
||
|
|
stack,
|
||
|
|
prompt=("hermes_cli.setup.prompt_choice", {"return_value": 0}),
|
||
|
|
first="hermes_cli.setup._run_first_time_quick_setup",
|
||
|
|
)
|
||
|
|
from hermes_cli.setup import run_setup_wizard
|
||
|
|
run_setup_wizard(args)
|
||
|
|
|
||
|
|
m["prompt"].assert_called_once()
|
||
|
|
m["first"].assert_called_once()
|
||
|
|
|
||
|
|
|
||
|
|
class TestArgparse:
|
||
|
|
"""The flags are plumbed through argparse to cmd_setup."""
|
||
|
|
|
||
|
|
def test_reconfigure_flag_reaches_cmd_setup(self, monkeypatch):
|
||
|
|
import sys
|
||
|
|
from hermes_cli.main import main
|
||
|
|
|
||
|
|
captured = {}
|
||
|
|
monkeypatch.setattr(
|
||
|
|
"hermes_cli.setup.run_setup_wizard",
|
||
|
|
lambda args: captured.setdefault("args", args),
|
||
|
|
)
|
||
|
|
monkeypatch.setattr(sys, "argv", ["hermes", "setup", "--reconfigure"])
|
||
|
|
try:
|
||
|
|
main()
|
||
|
|
except SystemExit:
|
||
|
|
pass
|
||
|
|
assert captured["args"].reconfigure is True
|
||
|
|
assert captured["args"].quick is False
|
||
|
|
|
||
|
|
def test_quick_flag_reaches_cmd_setup(self, monkeypatch):
|
||
|
|
import sys
|
||
|
|
from hermes_cli.main import main
|
||
|
|
|
||
|
|
captured = {}
|
||
|
|
monkeypatch.setattr(
|
||
|
|
"hermes_cli.setup.run_setup_wizard",
|
||
|
|
lambda args: captured.setdefault("args", args),
|
||
|
|
)
|
||
|
|
monkeypatch.setattr(sys, "argv", ["hermes", "setup", "--quick"])
|
||
|
|
try:
|
||
|
|
main()
|
||
|
|
except SystemExit:
|
||
|
|
pass
|
||
|
|
assert captured["args"].quick is True
|
||
|
|
assert captured["args"].reconfigure is False
|
||
|
|
|
||
|
|
def test_bare_setup_has_both_flags_false(self, monkeypatch):
|
||
|
|
import sys
|
||
|
|
from hermes_cli.main import main
|
||
|
|
|
||
|
|
captured = {}
|
||
|
|
monkeypatch.setattr(
|
||
|
|
"hermes_cli.setup.run_setup_wizard",
|
||
|
|
lambda args: captured.setdefault("args", args),
|
||
|
|
)
|
||
|
|
monkeypatch.setattr(sys, "argv", ["hermes", "setup"])
|
||
|
|
try:
|
||
|
|
main()
|
||
|
|
except SystemExit:
|
||
|
|
pass
|
||
|
|
assert captured["args"].reconfigure is False
|
||
|
|
assert captured["args"].quick is False
|