from types import SimpleNamespace from unittest.mock import MagicMock import hermes_cli.memory_setup as memory_setup from hermes_cli.memory_setup import _CANCELLED, _curses_select def test_curses_select_cancel_defaults_to_selected(monkeypatch): captured = {} def fake_radiolist(title, items, selected=0, *, cancel_returns=None): captured.update({ "title": title, "items": items, "selected": selected, "cancel_returns": cancel_returns, }) return cancel_returns monkeypatch.setattr("hermes_cli.curses_ui.curses_radiolist", fake_radiolist) result = _curses_select("Pick one", [("first", "desc"), ("second", "")], default=1) assert result == 1 assert captured == { "title": "Pick one", "items": ["first - desc", "second"], "selected": 1, "cancel_returns": 1, } def test_curses_select_accepts_explicit_cancel_value(monkeypatch): captured = {} def fake_radiolist(title, items, selected=0, *, cancel_returns=None): captured["cancel_returns"] = cancel_returns return cancel_returns monkeypatch.setattr("hermes_cli.curses_ui.curses_radiolist", fake_radiolist) result = _curses_select("Pick one", [("first", "")], default=0, cancel_returns=_CANCELLED) assert result == _CANCELLED assert captured["cancel_returns"] == _CANCELLED def test_cmd_setup_top_level_cancel_writes_nothing(monkeypatch): save_config = MagicMock() load_config = MagicMock(side_effect=AssertionError("cancel should not load config")) monkeypatch.setattr(memory_setup, "_get_available_providers", lambda: [("fake", "local", object())]) monkeypatch.setattr(memory_setup, "_curses_select", lambda *args, **kwargs: kwargs["cancel_returns"]) monkeypatch.setattr("hermes_cli.config.load_config", load_config) monkeypatch.setattr("hermes_cli.config.save_config", save_config) memory_setup.cmd_setup(SimpleNamespace()) load_config.assert_not_called() save_config.assert_not_called() def test_cmd_setup_builtin_selection_still_saves_builtin(monkeypatch): save_config = MagicMock() config = {"memory": {"provider": "openviking"}} providers = [("fake", "local", object())] monkeypatch.setattr(memory_setup, "_get_available_providers", lambda: providers) monkeypatch.setattr(memory_setup, "_curses_select", lambda *args, **kwargs: len(providers)) monkeypatch.setattr("hermes_cli.config.load_config", lambda: config) monkeypatch.setattr("hermes_cli.config.save_config", save_config) memory_setup.cmd_setup(SimpleNamespace()) assert config["memory"]["provider"] == "" save_config.assert_called_once_with(config) def test_cmd_setup_clears_interactive_picker_before_provider_post_setup(monkeypatch): events = [] class PostSetupProvider: def post_setup(self, hermes_home, config): events.append("post_setup") monkeypatch.setattr(memory_setup, "_get_available_providers", lambda: [("openviking", "local", PostSetupProvider())]) monkeypatch.setattr(memory_setup, "_curses_select", lambda *args, **kwargs: events.append("select") or 0) monkeypatch.setattr(memory_setup, "_clear_interactive_transition", lambda: events.append("clear"), raising=False) monkeypatch.setattr(memory_setup, "_install_dependencies", lambda name: events.append("install")) monkeypatch.setattr(memory_setup, "get_hermes_home", lambda: "/tmp/hermes-test") monkeypatch.setattr("hermes_cli.config.load_config", lambda: {"memory": {}}) memory_setup.cmd_setup(SimpleNamespace()) assert events == ["select", "clear", "install", "post_setup"] def test_cmd_setup_generic_choice_cancel_writes_nothing(tmp_path, monkeypatch): class ChoiceProvider: def __init__(self): self.save_config = MagicMock() def get_config_schema(self): return [{ "key": "mode", "description": "Mode", "default": "one", "choices": ["one", "two"], }] provider = ChoiceProvider() selections = iter([0, _CANCELLED]) save_config = MagicMock() install_dependencies = MagicMock() monkeypatch.setattr(memory_setup, "_get_available_providers", lambda: [("fake", "local", provider)]) monkeypatch.setattr(memory_setup, "_curses_select", lambda *args, **kwargs: next(selections)) monkeypatch.setattr(memory_setup, "_install_dependencies", install_dependencies) monkeypatch.setattr(memory_setup, "get_hermes_home", lambda: tmp_path) monkeypatch.setattr("hermes_cli.config.load_config", lambda: {"memory": {}}) monkeypatch.setattr("hermes_cli.config.save_config", save_config) memory_setup.cmd_setup(SimpleNamespace()) install_dependencies.assert_called_once_with("fake") save_config.assert_not_called() provider.save_config.assert_not_called() assert not (tmp_path / ".env").exists()