mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-04-30 01:41:43 +00:00
BOOT.md was merged in PR #3733 before the feature was ready — the built-in hook spawned a bare AIAgent() with no model/runtime kwargs, which immediately 401s on any provider with a custom endpoint. Three separate community PRs (#5240, #12514, #14992) tried to paper over it. Remove the BOOT.md hook entirely and its user-facing docs/tips. Keep the gateway/builtin_hooks/ package and the HookRegistry._register_builtin_hooks() hook-point intact as the extension surface for future always-on gateway hooks. Closes #5239. Co-authored-by: teknium1 <teknium@users.noreply.github.com>
197 lines
7.2 KiB
Python
197 lines
7.2 KiB
Python
"""
|
|
Event Hook System
|
|
|
|
A lightweight event-driven system that fires handlers at key lifecycle points.
|
|
Hooks are discovered from ~/.hermes/hooks/ directories, each containing:
|
|
- HOOK.yaml (metadata: name, description, events list)
|
|
- handler.py (Python handler with async def handle(event_type, context))
|
|
|
|
Events:
|
|
- gateway:startup -- Gateway process starts
|
|
- session:start -- New session created (first message of a new session)
|
|
- session:end -- Session ends (user ran /new or /reset)
|
|
- session:reset -- Session reset completed (new session entry created)
|
|
- agent:start -- Agent begins processing a message
|
|
- agent:step -- Each turn in the tool-calling loop
|
|
- agent:end -- Agent finishes processing
|
|
- command:* -- Any slash command executed (wildcard match)
|
|
|
|
Errors in hooks are caught and logged but never block the main pipeline.
|
|
"""
|
|
|
|
import asyncio
|
|
import importlib.util
|
|
from typing import Any, Callable, Dict, List, Optional
|
|
|
|
import yaml
|
|
|
|
from hermes_cli.config import get_hermes_home
|
|
|
|
|
|
HOOKS_DIR = get_hermes_home() / "hooks"
|
|
|
|
|
|
class HookRegistry:
|
|
"""
|
|
Discovers, loads, and fires event hooks.
|
|
|
|
Usage:
|
|
registry = HookRegistry()
|
|
registry.discover_and_load()
|
|
await registry.emit("agent:start", {"platform": "telegram", ...})
|
|
"""
|
|
|
|
def __init__(self):
|
|
# event_type -> [handler_fn, ...]
|
|
self._handlers: Dict[str, List[Callable]] = {}
|
|
self._loaded_hooks: List[dict] = [] # metadata for listing
|
|
|
|
@property
|
|
def loaded_hooks(self) -> List[dict]:
|
|
"""Return metadata about all loaded hooks."""
|
|
return list(self._loaded_hooks)
|
|
|
|
def _register_builtin_hooks(self) -> None:
|
|
"""Register built-in hooks that are always active.
|
|
|
|
Currently empty — no shipped built-in hooks. Kept as the extension
|
|
point for future always-on gateway hooks so they drop in without
|
|
re-plumbing discover_and_load().
|
|
"""
|
|
return
|
|
|
|
def discover_and_load(self) -> None:
|
|
"""
|
|
Scan the hooks directory for hook directories and load their handlers.
|
|
|
|
Also registers built-in hooks that are always active.
|
|
|
|
Each hook directory must contain:
|
|
- HOOK.yaml with at least 'name' and 'events' keys
|
|
- handler.py with a top-level 'handle' function (sync or async)
|
|
"""
|
|
self._register_builtin_hooks()
|
|
|
|
if not HOOKS_DIR.exists():
|
|
return
|
|
|
|
for hook_dir in sorted(HOOKS_DIR.iterdir()):
|
|
if not hook_dir.is_dir():
|
|
continue
|
|
|
|
manifest_path = hook_dir / "HOOK.yaml"
|
|
handler_path = hook_dir / "handler.py"
|
|
|
|
if not manifest_path.exists() or not handler_path.exists():
|
|
continue
|
|
|
|
try:
|
|
manifest = yaml.safe_load(manifest_path.read_text(encoding="utf-8"))
|
|
if not manifest or not isinstance(manifest, dict):
|
|
print(f"[hooks] Skipping {hook_dir.name}: invalid HOOK.yaml", flush=True)
|
|
continue
|
|
|
|
hook_name = manifest.get("name", hook_dir.name)
|
|
events = manifest.get("events", [])
|
|
if not events:
|
|
print(f"[hooks] Skipping {hook_name}: no events declared", flush=True)
|
|
continue
|
|
|
|
# Dynamically load the handler module
|
|
spec = importlib.util.spec_from_file_location(
|
|
f"hermes_hook_{hook_name}", handler_path
|
|
)
|
|
if spec is None or spec.loader is None:
|
|
print(f"[hooks] Skipping {hook_name}: could not load handler.py", flush=True)
|
|
continue
|
|
|
|
module = importlib.util.module_from_spec(spec)
|
|
spec.loader.exec_module(module)
|
|
|
|
handle_fn = getattr(module, "handle", None)
|
|
if handle_fn is None:
|
|
print(f"[hooks] Skipping {hook_name}: no 'handle' function found", flush=True)
|
|
continue
|
|
|
|
# Register the handler for each declared event
|
|
for event in events:
|
|
self._handlers.setdefault(event, []).append(handle_fn)
|
|
|
|
self._loaded_hooks.append({
|
|
"name": hook_name,
|
|
"description": manifest.get("description", ""),
|
|
"events": events,
|
|
"path": str(hook_dir),
|
|
})
|
|
|
|
print(f"[hooks] Loaded hook '{hook_name}' for events: {events}", flush=True)
|
|
|
|
except Exception as e:
|
|
print(f"[hooks] Error loading hook {hook_dir.name}: {e}", flush=True)
|
|
|
|
def _resolve_handlers(self, event_type: str) -> List[Callable]:
|
|
"""Return all handlers that should fire for ``event_type``.
|
|
|
|
Exact matches fire first, followed by wildcard matches (e.g.
|
|
``command:*`` matches ``command:reset``).
|
|
"""
|
|
handlers = list(self._handlers.get(event_type, []))
|
|
if ":" in event_type:
|
|
base = event_type.split(":")[0]
|
|
wildcard_key = f"{base}:*"
|
|
handlers.extend(self._handlers.get(wildcard_key, []))
|
|
return handlers
|
|
|
|
async def emit(self, event_type: str, context: Optional[Dict[str, Any]] = None) -> None:
|
|
"""
|
|
Fire all handlers registered for an event, discarding return values.
|
|
|
|
Supports wildcard matching: handlers registered for "command:*" will
|
|
fire for any "command:..." event. Handlers registered for a base type
|
|
like "agent" won't fire for "agent:start" -- only exact matches and
|
|
explicit wildcards.
|
|
|
|
Args:
|
|
event_type: The event identifier (e.g. "agent:start").
|
|
context: Optional dict with event-specific data.
|
|
"""
|
|
if context is None:
|
|
context = {}
|
|
|
|
for fn in self._resolve_handlers(event_type):
|
|
try:
|
|
result = fn(event_type, context)
|
|
# Support both sync and async handlers
|
|
if asyncio.iscoroutine(result):
|
|
await result
|
|
except Exception as e:
|
|
print(f"[hooks] Error in handler for '{event_type}': {e}", flush=True)
|
|
|
|
async def emit_collect(
|
|
self,
|
|
event_type: str,
|
|
context: Optional[Dict[str, Any]] = None,
|
|
) -> List[Any]:
|
|
"""Fire handlers and return their non-None return values in order.
|
|
|
|
Like :meth:`emit` but captures each handler's return value. Used for
|
|
decision-style hooks (e.g. ``command:<name>`` policies that want to
|
|
allow/deny/rewrite the command before normal dispatch).
|
|
|
|
Exceptions from individual handlers are logged but do not abort the
|
|
remaining handlers.
|
|
"""
|
|
if context is None:
|
|
context = {}
|
|
|
|
results: List[Any] = []
|
|
for fn in self._resolve_handlers(event_type):
|
|
try:
|
|
result = fn(event_type, context)
|
|
if asyncio.iscoroutine(result):
|
|
result = await result
|
|
if result is not None:
|
|
results.append(result)
|
|
except Exception as e:
|
|
print(f"[hooks] Error in handler for '{event_type}': {e}", flush=True)
|
|
return results
|