| from __future__ import annotations |
|
|
| import importlib |
| import os |
| import sys |
| from unittest.mock import MagicMock, patch |
|
|
| import pytest |
|
|
|
|
| def _make_real_cli(**kwargs): |
| clean_config = { |
| "model": { |
| "default": "anthropic/claude-opus-4.6", |
| "base_url": "https://openrouter.ai/api/v1", |
| "provider": "auto", |
| }, |
| "display": {"compact": False, "tool_progress": "all"}, |
| "agent": {}, |
| "terminal": {"env_type": "local"}, |
| } |
| clean_env = {"LLM_MODEL": "", "HERMES_MAX_ITERATIONS": ""} |
| prompt_toolkit_stubs = { |
| "prompt_toolkit": MagicMock(), |
| "prompt_toolkit.history": MagicMock(), |
| "prompt_toolkit.styles": MagicMock(), |
| "prompt_toolkit.patch_stdout": MagicMock(), |
| "prompt_toolkit.application": MagicMock(), |
| "prompt_toolkit.layout": MagicMock(), |
| "prompt_toolkit.layout.processors": MagicMock(), |
| "prompt_toolkit.filters": MagicMock(), |
| "prompt_toolkit.layout.dimension": MagicMock(), |
| "prompt_toolkit.layout.menus": MagicMock(), |
| "prompt_toolkit.widgets": MagicMock(), |
| "prompt_toolkit.key_binding": MagicMock(), |
| "prompt_toolkit.completion": MagicMock(), |
| "prompt_toolkit.formatted_text": MagicMock(), |
| } |
| with patch.dict(sys.modules, prompt_toolkit_stubs), patch.dict( |
| "os.environ", clean_env, clear=False |
| ): |
| import cli as cli_mod |
|
|
| cli_mod = importlib.reload(cli_mod) |
| with patch.object(cli_mod, "get_tool_definitions", return_value=[]), patch.dict( |
| cli_mod.__dict__, {"CLI_CONFIG": clean_config} |
| ): |
| return cli_mod.HermesCLI(**kwargs) |
|
|
|
|
| class _DummyCLI: |
| def __init__(self, **kwargs): |
| self.kwargs = kwargs |
| self.session_id = "session-123" |
| self.system_prompt = "base prompt" |
| self.preloaded_skills = [] |
|
|
| def show_banner(self): |
| return None |
|
|
| def show_tools(self): |
| return None |
|
|
| def show_toolsets(self): |
| return None |
|
|
| def run(self): |
| return None |
|
|
|
|
| def test_main_applies_preloaded_skills_to_system_prompt(monkeypatch): |
| import cli as cli_mod |
|
|
| created = {} |
|
|
| def fake_cli(**kwargs): |
| created["cli"] = _DummyCLI(**kwargs) |
| return created["cli"] |
|
|
| monkeypatch.setattr(cli_mod, "HermesCLI", fake_cli) |
| monkeypatch.setattr( |
| cli_mod, |
| "build_preloaded_skills_prompt", |
| lambda skills, task_id=None: ("skill prompt", ["hermes-agent-dev", "github-auth"], []), |
| ) |
|
|
| with pytest.raises(SystemExit): |
| cli_mod.main(skills="hermes-agent-dev,github-auth", list_tools=True) |
|
|
| cli_obj = created["cli"] |
| assert cli_obj.system_prompt == "base prompt\n\nskill prompt" |
| assert cli_obj.preloaded_skills == ["hermes-agent-dev", "github-auth"] |
|
|
|
|
| def test_main_raises_for_unknown_preloaded_skill(monkeypatch): |
| import cli as cli_mod |
|
|
| monkeypatch.setattr(cli_mod, "HermesCLI", lambda **kwargs: _DummyCLI(**kwargs)) |
| monkeypatch.setattr( |
| cli_mod, |
| "build_preloaded_skills_prompt", |
| lambda skills, task_id=None: ("", [], ["missing-skill"]), |
| ) |
|
|
| with pytest.raises(ValueError, match=r"Unknown skill\(s\): missing-skill"): |
| cli_mod.main(skills="missing-skill", list_tools=True) |
|
|
|
|
| def test_show_banner_does_not_print_skills(): |
| """show_banner() no longer prints the activated skills line — it moved to run().""" |
| cli_obj = _make_real_cli(compact=False) |
| cli_obj.preloaded_skills = ["hermes-agent-dev", "github-auth"] |
| cli_obj.console = MagicMock() |
|
|
| with patch("cli.build_welcome_banner") as mock_banner, patch( |
| "shutil.get_terminal_size", return_value=os.terminal_size((120, 40)) |
| ): |
| cli_obj.show_banner() |
|
|
| print_calls = [ |
| call.args[0] |
| for call in cli_obj.console.print.call_args_list |
| if call.args and isinstance(call.args[0], str) |
| ] |
| startup_lines = [line for line in print_calls if "Activated skills:" in line] |
| assert len(startup_lines) == 0 |
| assert mock_banner.call_count == 1 |
|
|