mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-06-19 10:02:16 +00:00
`hermes login` was removed in favor of `hermes auth` / `hermes model`, but the subparser still validated `--provider` against a hardcoded choices list (nous, openai-codex, xai-oauth). Running `hermes login --provider anthropic` therefore crashed in argparse with `invalid choice: 'anthropic'` *before* the deprecation handler could print the redirect to `hermes model` — so a user trying to authenticate a perfectly valid provider just saw a hard error and assumed the feature was broken rather than relocated. - Drop the restrictive `choices=` so every `--provider` value reaches the deprecation handler (which ignores the value and prints guidance). - Omit the subparser `help=` kwarg so the dead command no longer advertises itself in `hermes --help` (#24756). Avoids the `==SUPPRESS==` placeholder leak that `help=argparse.SUPPRESS` emits for a top-level subparser on 3.12+. - `hermes login [--flags]` still reaches the actionable deprecation message for old scripts/aliases; `hermes login --help` shows the redirect. Picks up the intent of the inactivity-closed #24902, rebased onto the post-refactor parser location (hermes_cli/subcommands/login.py) and extended to fix the whole bug class (any provider value), not just hiding from --help. Tests: parametrized provider acceptance + help-suppression (no SUPPRESS leak).
146 lines
6.8 KiB
Python
146 lines
6.8 KiB
Python
"""Smoke tests for the batch-extracted subcommand parser builders.
|
|
|
|
Each ``build_<group>_parser`` should attach its subcommand to a subparsers
|
|
group and wire ``func`` to the injected handler. These are intentionally
|
|
light — the byte-identical ``--help`` verification done at extraction time is
|
|
the real behavioral guarantee; this just guards against a module failing to
|
|
import or a builder raising.
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
import argparse
|
|
|
|
import pytest
|
|
|
|
from hermes_cli.subcommands.auth import build_auth_parser
|
|
from hermes_cli.subcommands.backup import build_backup_parser
|
|
from hermes_cli.subcommands.config import build_config_parser
|
|
from hermes_cli.subcommands.dashboard import build_dashboard_parser
|
|
from hermes_cli.subcommands.debug import build_debug_parser
|
|
from hermes_cli.subcommands.doctor import build_doctor_parser
|
|
from hermes_cli.subcommands.dump import build_dump_parser
|
|
from hermes_cli.subcommands.gui import build_gui_parser
|
|
from hermes_cli.subcommands.hooks import build_hooks_parser
|
|
from hermes_cli.subcommands.import_cmd import build_import_cmd_parser
|
|
from hermes_cli.subcommands.login import build_login_parser
|
|
from hermes_cli.subcommands.logout import build_logout_parser
|
|
from hermes_cli.subcommands.logs import build_logs_parser
|
|
from hermes_cli.subcommands.model import build_model_parser
|
|
from hermes_cli.subcommands.postinstall import build_postinstall_parser
|
|
from hermes_cli.subcommands.prompt_size import build_prompt_size_parser
|
|
from hermes_cli.subcommands.security import build_security_parser
|
|
from hermes_cli.subcommands.setup import build_setup_parser
|
|
from hermes_cli.subcommands.slack import build_slack_parser
|
|
from hermes_cli.subcommands.status import build_status_parser
|
|
from hermes_cli.subcommands.uninstall import build_uninstall_parser
|
|
from hermes_cli.subcommands.update import build_update_parser
|
|
from hermes_cli.subcommands.version import build_version_parser
|
|
from hermes_cli.subcommands.webhook import build_webhook_parser
|
|
from hermes_cli.subcommands.whatsapp import build_whatsapp_parser
|
|
|
|
|
|
def _h(name):
|
|
def handler(args): # pragma: no cover - identity only
|
|
return name
|
|
handler.__name__ = f"cmd_{name}"
|
|
return handler
|
|
|
|
|
|
# (subcommand_name, builder, handler_kwargs, sample_argv)
|
|
SINGLE_HANDLER_CASES = [
|
|
("model", build_model_parser, "cmd_model", ["model"]),
|
|
("setup", build_setup_parser, "cmd_setup", ["setup"]),
|
|
("postinstall", build_postinstall_parser, "cmd_postinstall", ["postinstall"]),
|
|
("whatsapp", build_whatsapp_parser, "cmd_whatsapp", ["whatsapp"]),
|
|
("slack", build_slack_parser, "cmd_slack", ["slack"]),
|
|
("login", build_login_parser, "cmd_login", ["login"]),
|
|
("logout", build_logout_parser, "cmd_logout", ["logout"]),
|
|
("auth", build_auth_parser, "cmd_auth", ["auth"]),
|
|
("status", build_status_parser, "cmd_status", ["status"]),
|
|
("webhook", build_webhook_parser, "cmd_webhook", ["webhook"]),
|
|
("hooks", build_hooks_parser, "cmd_hooks", ["hooks"]),
|
|
("doctor", build_doctor_parser, "cmd_doctor", ["doctor"]),
|
|
("security", build_security_parser, "cmd_security", ["security"]),
|
|
("dump", build_dump_parser, "cmd_dump", ["dump"]),
|
|
("debug", build_debug_parser, "cmd_debug", ["debug"]),
|
|
("backup", build_backup_parser, "cmd_backup", ["backup"]),
|
|
("import", build_import_cmd_parser, "cmd_import", ["import", "/tmp/x.zip"]),
|
|
("config", build_config_parser, "cmd_config", ["config"]),
|
|
("version", build_version_parser, "cmd_version", ["version"]),
|
|
("update", build_update_parser, "cmd_update", ["update"]),
|
|
("uninstall", build_uninstall_parser, "cmd_uninstall", ["uninstall"]),
|
|
("gui", build_gui_parser, "cmd_gui", ["gui"]),
|
|
("logs", build_logs_parser, "cmd_logs", ["logs"]),
|
|
("prompt-size", build_prompt_size_parser, "cmd_prompt_size", ["prompt-size"]),
|
|
]
|
|
|
|
|
|
@pytest.mark.parametrize("name,builder,kw,argv", SINGLE_HANDLER_CASES, ids=[c[0] for c in SINGLE_HANDLER_CASES])
|
|
def test_single_handler_builders(name, builder, kw, argv):
|
|
parser = argparse.ArgumentParser(prog="hermes")
|
|
sub = parser.add_subparsers(dest="command")
|
|
handler = _h(name)
|
|
builder(sub, **{kw: handler})
|
|
ns = parser.parse_args(argv)
|
|
assert ns.func is handler
|
|
|
|
|
|
def test_dashboard_builder_two_handlers():
|
|
parser = argparse.ArgumentParser(prog="hermes")
|
|
sub = parser.add_subparsers(dest="command")
|
|
dash, reg = _h("dashboard"), _h("dashboard_register")
|
|
build_dashboard_parser(sub, cmd_dashboard=dash, cmd_dashboard_register=reg)
|
|
# bare dashboard -> launch handler
|
|
assert parser.parse_args(["dashboard"]).func is dash
|
|
# dashboard register -> register handler
|
|
assert parser.parse_args(["dashboard", "register"]).func is reg
|
|
|
|
|
|
# ── deprecated `hermes login` fails gracefully, not with argparse error ────
|
|
#
|
|
# `hermes login` is a removed command; its handler (`login_command` in
|
|
# `hermes_cli/auth.py`) prints a deprecation notice pointing at `hermes auth` /
|
|
# `hermes model` and exits 0. Two behavior contracts guard the UX:
|
|
# 1. ANY `--provider <value>` (including ones the user actually wants, like
|
|
# `anthropic`) must parse and reach the handler — never crash in argparse
|
|
# with `invalid choice` before the friendly redirect is printed (#24756).
|
|
# 2. The subcommand must not advertise itself in the parser help row.
|
|
|
|
|
|
def _login_parser():
|
|
parser = argparse.ArgumentParser(prog="hermes")
|
|
sub = parser.add_subparsers(dest="command")
|
|
build_login_parser(sub, cmd_login=_h("login"))
|
|
return parser
|
|
|
|
|
|
@pytest.mark.parametrize("provider", ["anthropic", "nous", "openai-codex", "totally-made-up"])
|
|
def test_login_accepts_any_provider_value(provider):
|
|
"""Deprecated `login` must route every `--provider` to the handler.
|
|
|
|
A restrictive `choices=` list (the pre-fix behavior) rejected providers
|
|
like `anthropic` with an argparse error *before* the deprecation message
|
|
could run, so the user just saw `invalid choice: 'anthropic'` and assumed
|
|
the feature was broken rather than relocated.
|
|
"""
|
|
ns = _login_parser().parse_args(["login", "--provider", provider])
|
|
assert ns.func.__name__ == "cmd_login"
|
|
assert ns.provider == provider
|
|
|
|
|
|
def test_login_subparser_help_is_suppressed():
|
|
"""The deprecated `login` row must not appear in `hermes --help`.
|
|
|
|
Must hold without leaking argparse's literal `==SUPPRESS==` placeholder,
|
|
which `help=argparse.SUPPRESS` emits for a top-level subparser on 3.12+.
|
|
The fix omits the `help=` kwarg entirely instead.
|
|
"""
|
|
parser = argparse.ArgumentParser(prog="hermes")
|
|
sub = parser.add_subparsers(dest="command")
|
|
build_login_parser(sub, cmd_login=_h("login"))
|
|
help_text = parser.format_help()
|
|
# The misleading old help string must be gone from the top-level usage.
|
|
assert "Authenticate with an inference provider" not in help_text
|
|
# And no leaked SUPPRESS placeholder row.
|
|
assert "==SUPPRESS==" not in help_text
|