Major feature additions inspired by OpenClaw/ClawdBot integration analysis: Voice Message Transcription (STT): - Auto-transcribe voice/audio messages via OpenAI Whisper API - Download voice to ~/.hermes/audio_cache/ on Telegram/Discord/WhatsApp - Inject transcript as text so all models can understand voice input - Configurable model (whisper-1, gpt-4o-mini-transcribe, gpt-4o-transcribe) Telegram Sticker Understanding: - Describe static stickers via vision tool with JSON-backed cache - Cache keyed by file_unique_id avoids redundant API calls - Animated/video stickers get emoji-based fallback description Discord Rich UX: - Native slash commands (/ask, /reset, /status, /stop) via app_commands - Button-based exec approvals (Allow Once / Always Allow / Deny) - ExecApprovalView with user authorization and timeout handling Slack Integration: - Full SlackAdapter using slack-bolt with Socket Mode - DMs, channel messages (mention-gated), /hermes slash command - File attachment handling with bot-token-authenticated downloads DM Pairing System: - Code-based user authorization as alternative to static allowlists - 8-char codes from unambiguous alphabet, 1-hour expiry - Rate limiting, lockout after failed attempts, chmod 0600 on data - CLI: hermes pairing list/approve/revoke/clear-pending Event Hook System: - File-based hook discovery from ~/.hermes/hooks/ - HOOK.yaml + handler.py per hook, sync/async handler support - Events: gateway:startup, session:start/reset, agent:start/step/end - Wildcard matching (command:* catches all command events) Cross-Channel Messaging: - send_message agent tool for delivering to any connected platform - Enables cron job delivery and cross-platform notifications Human-Like Response Pacing: - Configurable delays between message chunks (off/natural/custom) - HERMES_HUMAN_DELAY_MODE env var with min/max ms settings Warm Injection Message Style: - Retrofitted image vision messages with friendly kawaii-consistent tone - All new injection messages (STT, stickers, errors) use warm style Also: updated config migration to prompt for optional keys interactively, bumped config version, updated README, AGENTS.md, .env.example, cli-config.yaml.example, install scripts, pyproject.toml, and toolsets.
151 lines
5.4 KiB
Python
151 lines
5.4 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
|
|
- session:reset -- User ran /new or /reset
|
|
- 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
|
|
import os
|
|
from pathlib import Path
|
|
from typing import Any, Callable, Dict, List, Optional
|
|
|
|
import yaml
|
|
|
|
|
|
HOOKS_DIR = Path(os.path.expanduser("~/.hermes/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 discover_and_load(self) -> None:
|
|
"""
|
|
Scan the hooks directory for hook directories and load their handlers.
|
|
|
|
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)
|
|
"""
|
|
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)
|
|
|
|
async def emit(self, event_type: str, context: Optional[Dict[str, Any]] = None) -> None:
|
|
"""
|
|
Fire all handlers registered for an event.
|
|
|
|
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 = {}
|
|
|
|
# Collect handlers: exact match + wildcard match
|
|
handlers = list(self._handlers.get(event_type, []))
|
|
|
|
# Check for wildcard patterns (e.g., "command:*" matches "command:reset")
|
|
if ":" in event_type:
|
|
base = event_type.split(":")[0]
|
|
wildcard_key = f"{base}:*"
|
|
handlers.extend(self._handlers.get(wildcard_key, []))
|
|
|
|
for fn in handlers:
|
|
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)
|