mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-05-21 03:39:54 +00:00
143 lines
4.2 KiB
Python
143 lines
4.2 KiB
Python
"""CLI handlers for the ``hermes proxy`` subcommand."""
|
|
|
|
from __future__ import annotations
|
|
|
|
import asyncio
|
|
import logging
|
|
import sys
|
|
from typing import Any
|
|
|
|
from hermes_cli.proxy.adapters import ADAPTERS, get_adapter
|
|
from hermes_cli.proxy.server import (
|
|
AIOHTTP_AVAILABLE,
|
|
DEFAULT_HOST,
|
|
DEFAULT_PORT,
|
|
run_server,
|
|
)
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
def _print_aiohttp_missing() -> None:
|
|
print(
|
|
"hermes proxy requires aiohttp. Install one of:\n"
|
|
" pip install 'hermes-agent[messaging]'\n"
|
|
" pip install aiohttp",
|
|
file=sys.stderr,
|
|
)
|
|
|
|
|
|
def cmd_proxy_start(args: Any) -> int:
|
|
"""Run the proxy server in the foreground.
|
|
|
|
Returns process exit code (0 on clean shutdown).
|
|
"""
|
|
if not AIOHTTP_AVAILABLE:
|
|
_print_aiohttp_missing()
|
|
return 1
|
|
|
|
provider = getattr(args, "provider", None) or "nous"
|
|
try:
|
|
adapter = get_adapter(provider)
|
|
except ValueError as exc:
|
|
print(f"Error: {exc}", file=sys.stderr)
|
|
return 2
|
|
|
|
if not adapter.is_authenticated():
|
|
auth_hint = getattr(adapter, "auth_hint", f"hermes login {adapter.name}")
|
|
print(
|
|
f"Not logged into {adapter.display_name}. "
|
|
f"Run `{auth_hint}` first.",
|
|
file=sys.stderr,
|
|
)
|
|
return 2
|
|
|
|
host = getattr(args, "host", None) or DEFAULT_HOST
|
|
port = getattr(args, "port", None) or DEFAULT_PORT
|
|
|
|
print(
|
|
f"Starting Hermes proxy for {adapter.display_name}\n"
|
|
f" Listening on: http://{host}:{port}/v1\n"
|
|
f" Forwarding to: (resolved per-request from your subscription)\n"
|
|
f" Use any bearer token in the client — the proxy attaches your real credential.\n"
|
|
f"\n"
|
|
f"Press Ctrl+C to stop.",
|
|
file=sys.stderr,
|
|
)
|
|
|
|
try:
|
|
asyncio.run(run_server(adapter, host=host, port=port))
|
|
except KeyboardInterrupt:
|
|
print("\nproxy: stopped", file=sys.stderr)
|
|
except OSError as exc:
|
|
print(f"proxy: failed to bind {host}:{port}: {exc}", file=sys.stderr)
|
|
return 1
|
|
return 0
|
|
|
|
|
|
def cmd_proxy_status(args: Any) -> int:
|
|
"""Print the status of each configured upstream adapter."""
|
|
print("Hermes proxy upstream adapters\n")
|
|
for name in sorted(ADAPTERS):
|
|
adapter = get_adapter(name)
|
|
if not adapter.is_authenticated():
|
|
print(f" [{name:8s}] {adapter.display_name} — not logged in")
|
|
continue
|
|
try:
|
|
cred = adapter.get_credential()
|
|
except Exception as exc:
|
|
print(
|
|
f" [{name:8s}] {adapter.display_name} — credentials need attention "
|
|
f"({exc})"
|
|
)
|
|
continue
|
|
expires = f" (bearer expires {cred.expires_at})" if cred.expires_at else ""
|
|
print(f" [{name:8s}] {adapter.display_name} — ready{expires}")
|
|
print(
|
|
"\nStart the proxy with: hermes proxy start [--provider <name>]"
|
|
)
|
|
return 0
|
|
|
|
|
|
def cmd_proxy_list_providers(args: Any) -> int:
|
|
"""List available proxy upstream providers."""
|
|
print("Available proxy upstream providers:")
|
|
for name in sorted(ADAPTERS):
|
|
adapter = get_adapter(name)
|
|
print(f" {name} — {adapter.display_name}")
|
|
return 0
|
|
|
|
|
|
def cmd_proxy(args: Any) -> int:
|
|
"""Dispatch ``hermes proxy <subcommand>``."""
|
|
sub = getattr(args, "proxy_command", None)
|
|
if sub == "start":
|
|
return cmd_proxy_start(args)
|
|
if sub == "status":
|
|
return cmd_proxy_status(args)
|
|
if sub in {"providers", "list"}:
|
|
return cmd_proxy_list_providers(args)
|
|
# No subcommand → print short help.
|
|
print(
|
|
"hermes proxy — local OpenAI-compatible proxy that attaches your\n"
|
|
"OAuth-authenticated provider credentials to outbound requests.\n"
|
|
"\n"
|
|
"Subcommands:\n"
|
|
" hermes proxy start [--provider nous|xai] [--host 127.0.0.1] [--port 8645]\n"
|
|
" Run the proxy in the foreground.\n"
|
|
" hermes proxy status\n"
|
|
" Show which upstream adapters are ready.\n"
|
|
" hermes proxy providers\n"
|
|
" List available upstream providers.\n",
|
|
file=sys.stderr,
|
|
)
|
|
return 0
|
|
|
|
|
|
__all__ = [
|
|
"cmd_proxy",
|
|
"cmd_proxy_start",
|
|
"cmd_proxy_status",
|
|
"cmd_proxy_list_providers",
|
|
]
|