mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-04-28 23:11:37 +08:00
Enter while the agent is busy can now inject the typed text via /steer — arriving at the agent after the next tool call — instead of interrupting (current default) or queueing for the next turn. Changes: - cli.py: keybinding honors busy_input_mode='steer' by calling agent.steer(text) on the UI thread (thread-safe), with automatic fallback to 'queue' when the agent is missing, steer() is unavailable, images are attached, or steer() rejects the payload. /busy accepts 'steer' as a fourth argument alongside queue/interrupt/status. - gateway/run.py: busy-message handler and the PRIORITY running-agent path both route through running_agent.steer() when the mode is 'steer', with the same fallback-to-queue safety net. Ack wording tells users their message was steered into the current run. Restart-drain queueing now also activates for 'steer' so messages aren't lost across restarts. - agent/onboarding.py: first-touch hint has a steer branch for both CLI and gateway. - hermes_cli/commands.py: /busy args_hint updated to include steer, and 'steer' is registered as a subcommand (completions). - hermes_cli/web_server.py: dashboard select widget offers steer. - hermes_cli/config.py, cli-config.yaml.example, hermes_cli/tips.py: inline docs updated. - website/docs/user-guide/cli.md + messaging/index.md: documented. - Tests: steer set/status path for /busy; onboarding hints; _load_busy_input_mode accepts steer; busy-session ack exercises steer success + two fallback-to-queue branches. Requested on X by @CodingAcct. Default is unchanged (interrupt).
124 lines
4.4 KiB
Python
124 lines
4.4 KiB
Python
"""Tests for the /busy CLI command and busy-input-mode config handling."""
|
|
|
|
import unittest
|
|
from types import SimpleNamespace
|
|
from unittest.mock import patch
|
|
|
|
|
|
def _import_cli():
|
|
import hermes_cli.config as config_mod
|
|
|
|
if not hasattr(config_mod, "save_env_value_secure"):
|
|
config_mod.save_env_value_secure = lambda key, value: {
|
|
"success": True,
|
|
"stored_as": key,
|
|
"validated": False,
|
|
}
|
|
|
|
import cli as cli_mod
|
|
|
|
return cli_mod
|
|
|
|
|
|
class TestHandleBusyCommand(unittest.TestCase):
|
|
def _make_cli(self, busy_input_mode="interrupt"):
|
|
return SimpleNamespace(
|
|
busy_input_mode=busy_input_mode,
|
|
agent=None,
|
|
)
|
|
|
|
def test_no_args_shows_status(self):
|
|
cli_mod = _import_cli()
|
|
stub = self._make_cli("queue")
|
|
with (
|
|
patch.object(cli_mod, "_cprint") as mock_cprint,
|
|
patch.object(cli_mod, "save_config_value") as mock_save,
|
|
):
|
|
cli_mod.HermesCLI._handle_busy_command(stub, "/busy")
|
|
|
|
mock_save.assert_not_called()
|
|
printed = " ".join(str(c) for c in mock_cprint.call_args_list)
|
|
self.assertIn("queue", printed)
|
|
self.assertIn("interrupt", printed)
|
|
|
|
def test_queue_argument_sets_queue_mode_and_saves(self):
|
|
cli_mod = _import_cli()
|
|
stub = self._make_cli("interrupt")
|
|
with (
|
|
patch.object(cli_mod, "_cprint"),
|
|
patch.object(cli_mod, "save_config_value", return_value=True) as mock_save,
|
|
):
|
|
cli_mod.HermesCLI._handle_busy_command(stub, "/busy queue")
|
|
|
|
self.assertEqual(stub.busy_input_mode, "queue")
|
|
mock_save.assert_called_once_with("display.busy_input_mode", "queue")
|
|
|
|
def test_interrupt_argument_sets_interrupt_mode_and_saves(self):
|
|
cli_mod = _import_cli()
|
|
stub = self._make_cli("queue")
|
|
with (
|
|
patch.object(cli_mod, "_cprint"),
|
|
patch.object(cli_mod, "save_config_value", return_value=True) as mock_save,
|
|
):
|
|
cli_mod.HermesCLI._handle_busy_command(stub, "/busy interrupt")
|
|
|
|
self.assertEqual(stub.busy_input_mode, "interrupt")
|
|
mock_save.assert_called_once_with("display.busy_input_mode", "interrupt")
|
|
|
|
def test_steer_argument_sets_steer_mode_and_saves(self):
|
|
cli_mod = _import_cli()
|
|
stub = self._make_cli("interrupt")
|
|
with (
|
|
patch.object(cli_mod, "_cprint") as mock_cprint,
|
|
patch.object(cli_mod, "save_config_value", return_value=True) as mock_save,
|
|
):
|
|
cli_mod.HermesCLI._handle_busy_command(stub, "/busy steer")
|
|
|
|
self.assertEqual(stub.busy_input_mode, "steer")
|
|
mock_save.assert_called_once_with("display.busy_input_mode", "steer")
|
|
printed = " ".join(str(c) for c in mock_cprint.call_args_list)
|
|
self.assertIn("steer", printed.lower())
|
|
|
|
def test_status_reports_steer_behavior(self):
|
|
cli_mod = _import_cli()
|
|
stub = self._make_cli("steer")
|
|
with (
|
|
patch.object(cli_mod, "_cprint") as mock_cprint,
|
|
patch.object(cli_mod, "save_config_value") as mock_save,
|
|
):
|
|
cli_mod.HermesCLI._handle_busy_command(stub, "/busy status")
|
|
|
|
mock_save.assert_not_called()
|
|
printed = " ".join(str(c) for c in mock_cprint.call_args_list)
|
|
self.assertIn("steer", printed.lower())
|
|
# The usage line should also advertise the steer option
|
|
self.assertIn("steer", printed)
|
|
|
|
def test_invalid_argument_prints_usage(self):
|
|
cli_mod = _import_cli()
|
|
stub = self._make_cli()
|
|
with (
|
|
patch.object(cli_mod, "_cprint") as mock_cprint,
|
|
patch.object(cli_mod, "save_config_value") as mock_save,
|
|
):
|
|
cli_mod.HermesCLI._handle_busy_command(stub, "/busy nonsense")
|
|
|
|
mock_save.assert_not_called()
|
|
printed = " ".join(str(c) for c in mock_cprint.call_args_list)
|
|
self.assertIn("Usage: /busy", printed)
|
|
|
|
|
|
class TestBusyCommandRegistry(unittest.TestCase):
|
|
def test_busy_in_registry(self):
|
|
from hermes_cli.commands import COMMAND_REGISTRY
|
|
|
|
names = [c.name for c in COMMAND_REGISTRY]
|
|
assert "busy" in names
|
|
|
|
def test_busy_subcommands_documented(self):
|
|
from hermes_cli.commands import COMMAND_REGISTRY
|
|
|
|
busy = next(c for c in COMMAND_REGISTRY if c.name == "busy")
|
|
assert busy.args_hint == "[queue|steer|interrupt|status]"
|
|
assert busy.category == "Configuration"
|