2026-02-02 19:01:51 -08:00
|
|
|
"""
|
|
|
|
|
Gateway runner - entry point for messaging platform integrations.
|
|
|
|
|
|
|
|
|
|
This module provides:
|
|
|
|
|
- start_gateway(): Start all configured platform adapters
|
|
|
|
|
- GatewayRunner: Main class managing the gateway lifecycle
|
|
|
|
|
|
|
|
|
|
Usage:
|
|
|
|
|
# Start the gateway
|
|
|
|
|
python -m gateway.run
|
|
|
|
|
|
|
|
|
|
# Or from CLI
|
|
|
|
|
python cli.py --gateway
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
import asyncio
|
|
|
|
|
import os
|
2026-02-14 16:08:14 -08:00
|
|
|
import re
|
2026-02-02 19:01:51 -08:00
|
|
|
import sys
|
|
|
|
|
import signal
|
|
|
|
|
from pathlib import Path
|
|
|
|
|
from datetime import datetime
|
|
|
|
|
from typing import Dict, Optional, Any, List
|
|
|
|
|
|
|
|
|
|
# Add parent directory to path
|
|
|
|
|
sys.path.insert(0, str(Path(__file__).parent.parent))
|
|
|
|
|
|
2026-02-03 10:46:23 -08:00
|
|
|
# Load environment variables from ~/.hermes/.env first
|
2026-02-03 07:02:59 -08:00
|
|
|
from dotenv import load_dotenv
|
|
|
|
|
_env_path = Path.home() / '.hermes' / '.env'
|
|
|
|
|
if _env_path.exists():
|
|
|
|
|
load_dotenv(_env_path)
|
|
|
|
|
# Also try project .env as fallback
|
|
|
|
|
load_dotenv()
|
|
|
|
|
|
2026-02-03 10:46:23 -08:00
|
|
|
# Gateway runs in quiet mode - suppress debug output and use cwd directly (no temp dirs)
|
|
|
|
|
os.environ["HERMES_QUIET"] = "1"
|
|
|
|
|
|
2026-02-12 10:05:08 -08:00
|
|
|
# Enable interactive exec approval for dangerous commands on messaging platforms
|
|
|
|
|
os.environ["HERMES_EXEC_ASK"] = "1"
|
|
|
|
|
|
2026-02-03 10:46:23 -08:00
|
|
|
# Set terminal working directory for messaging platforms
|
|
|
|
|
# Uses MESSAGING_CWD if set, otherwise defaults to home directory
|
|
|
|
|
# This is separate from CLI which uses the directory where `hermes` is run
|
|
|
|
|
messaging_cwd = os.getenv("MESSAGING_CWD") or str(Path.home())
|
|
|
|
|
os.environ["TERMINAL_CWD"] = messaging_cwd
|
|
|
|
|
|
2026-02-02 19:01:51 -08:00
|
|
|
from gateway.config import (
|
|
|
|
|
Platform,
|
|
|
|
|
GatewayConfig,
|
|
|
|
|
load_gateway_config,
|
|
|
|
|
)
|
|
|
|
|
from gateway.session import (
|
|
|
|
|
SessionStore,
|
|
|
|
|
SessionSource,
|
|
|
|
|
SessionContext,
|
|
|
|
|
build_session_context,
|
|
|
|
|
build_session_context_prompt,
|
|
|
|
|
)
|
|
|
|
|
from gateway.delivery import DeliveryRouter, DeliveryTarget
|
2026-02-15 16:10:50 -08:00
|
|
|
from gateway.platforms.base import BasePlatformAdapter, MessageEvent, MessageType
|
2026-02-02 19:01:51 -08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
class GatewayRunner:
|
|
|
|
|
"""
|
|
|
|
|
Main gateway controller.
|
|
|
|
|
|
|
|
|
|
Manages the lifecycle of all platform adapters and routes
|
|
|
|
|
messages to/from the agent.
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
def __init__(self, config: Optional[GatewayConfig] = None):
|
|
|
|
|
self.config = config or load_gateway_config()
|
|
|
|
|
self.adapters: Dict[Platform, BasePlatformAdapter] = {}
|
|
|
|
|
self.session_store = SessionStore(self.config.sessions_dir, self.config)
|
|
|
|
|
self.delivery_router = DeliveryRouter(self.config)
|
|
|
|
|
self._running = False
|
|
|
|
|
self._shutdown_event = asyncio.Event()
|
2026-02-03 16:15:49 -08:00
|
|
|
|
|
|
|
|
# Track running agents per session for interrupt support
|
|
|
|
|
# Key: session_key, Value: AIAgent instance
|
|
|
|
|
self._running_agents: Dict[str, Any] = {}
|
|
|
|
|
self._pending_messages: Dict[str, str] = {} # Queued messages during interrupt
|
2026-02-12 10:05:08 -08:00
|
|
|
|
|
|
|
|
# Track pending exec approvals per session
|
|
|
|
|
# Key: session_key, Value: {"command": str, "pattern_key": str}
|
|
|
|
|
self._pending_approvals: Dict[str, Dict[str, str]] = {}
|
2026-02-02 19:01:51 -08:00
|
|
|
|
|
|
|
|
async def start(self) -> bool:
|
|
|
|
|
"""
|
|
|
|
|
Start the gateway and all configured platform adapters.
|
|
|
|
|
|
|
|
|
|
Returns True if at least one adapter connected successfully.
|
|
|
|
|
"""
|
|
|
|
|
print("[gateway] Starting Hermes Gateway...")
|
|
|
|
|
print(f"[gateway] Session storage: {self.config.sessions_dir}")
|
|
|
|
|
|
|
|
|
|
connected_count = 0
|
|
|
|
|
|
|
|
|
|
# Initialize and connect each configured platform
|
|
|
|
|
for platform, platform_config in self.config.platforms.items():
|
|
|
|
|
if not platform_config.enabled:
|
|
|
|
|
continue
|
|
|
|
|
|
|
|
|
|
adapter = self._create_adapter(platform, platform_config)
|
|
|
|
|
if not adapter:
|
|
|
|
|
print(f"[gateway] No adapter available for {platform.value}")
|
|
|
|
|
continue
|
|
|
|
|
|
|
|
|
|
# Set up message handler
|
|
|
|
|
adapter.set_message_handler(self._handle_message)
|
|
|
|
|
|
|
|
|
|
# Try to connect
|
|
|
|
|
print(f"[gateway] Connecting to {platform.value}...")
|
|
|
|
|
try:
|
|
|
|
|
success = await adapter.connect()
|
|
|
|
|
if success:
|
|
|
|
|
self.adapters[platform] = adapter
|
|
|
|
|
connected_count += 1
|
|
|
|
|
print(f"[gateway] ✓ {platform.value} connected")
|
|
|
|
|
else:
|
|
|
|
|
print(f"[gateway] ✗ {platform.value} failed to connect")
|
|
|
|
|
except Exception as e:
|
|
|
|
|
print(f"[gateway] ✗ {platform.value} error: {e}")
|
|
|
|
|
|
|
|
|
|
if connected_count == 0:
|
|
|
|
|
print("[gateway] No platforms connected. Check your configuration.")
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
# Update delivery router with adapters
|
|
|
|
|
self.delivery_router.adapters = self.adapters
|
|
|
|
|
|
|
|
|
|
self._running = True
|
|
|
|
|
print(f"[gateway] Gateway running with {connected_count} platform(s)")
|
|
|
|
|
print("[gateway] Press Ctrl+C to stop")
|
|
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
async def stop(self) -> None:
|
|
|
|
|
"""Stop the gateway and disconnect all adapters."""
|
|
|
|
|
print("[gateway] Stopping gateway...")
|
|
|
|
|
self._running = False
|
|
|
|
|
|
|
|
|
|
for platform, adapter in self.adapters.items():
|
|
|
|
|
try:
|
|
|
|
|
await adapter.disconnect()
|
|
|
|
|
print(f"[gateway] ✓ {platform.value} disconnected")
|
|
|
|
|
except Exception as e:
|
|
|
|
|
print(f"[gateway] ✗ {platform.value} disconnect error: {e}")
|
|
|
|
|
|
|
|
|
|
self.adapters.clear()
|
|
|
|
|
self._shutdown_event.set()
|
|
|
|
|
print("[gateway] Gateway stopped")
|
|
|
|
|
|
|
|
|
|
async def wait_for_shutdown(self) -> None:
|
|
|
|
|
"""Wait for shutdown signal."""
|
|
|
|
|
await self._shutdown_event.wait()
|
|
|
|
|
|
|
|
|
|
def _create_adapter(
|
|
|
|
|
self,
|
|
|
|
|
platform: Platform,
|
|
|
|
|
config: Any
|
|
|
|
|
) -> Optional[BasePlatformAdapter]:
|
|
|
|
|
"""Create the appropriate adapter for a platform."""
|
|
|
|
|
if platform == Platform.TELEGRAM:
|
|
|
|
|
from gateway.platforms.telegram import TelegramAdapter, check_telegram_requirements
|
|
|
|
|
if not check_telegram_requirements():
|
|
|
|
|
print(f"[gateway] Telegram: python-telegram-bot not installed")
|
|
|
|
|
return None
|
|
|
|
|
return TelegramAdapter(config)
|
|
|
|
|
|
|
|
|
|
elif platform == Platform.DISCORD:
|
|
|
|
|
from gateway.platforms.discord import DiscordAdapter, check_discord_requirements
|
|
|
|
|
if not check_discord_requirements():
|
|
|
|
|
print(f"[gateway] Discord: discord.py not installed")
|
|
|
|
|
return None
|
|
|
|
|
return DiscordAdapter(config)
|
|
|
|
|
|
|
|
|
|
elif platform == Platform.WHATSAPP:
|
|
|
|
|
from gateway.platforms.whatsapp import WhatsAppAdapter, check_whatsapp_requirements
|
|
|
|
|
if not check_whatsapp_requirements():
|
|
|
|
|
print(f"[gateway] WhatsApp: Node.js not installed or bridge not configured")
|
|
|
|
|
return None
|
|
|
|
|
return WhatsAppAdapter(config)
|
|
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
|
2026-02-03 10:46:23 -08:00
|
|
|
def _is_user_authorized(self, source: SessionSource) -> bool:
|
|
|
|
|
"""
|
|
|
|
|
Check if a user is authorized to use the bot.
|
|
|
|
|
|
|
|
|
|
Authorization is checked via environment variables:
|
|
|
|
|
- GATEWAY_ALLOWED_USERS: Comma-separated list of user IDs (all platforms)
|
|
|
|
|
- TELEGRAM_ALLOWED_USERS: Telegram-specific user IDs
|
|
|
|
|
- DISCORD_ALLOWED_USERS: Discord-specific user IDs
|
|
|
|
|
|
|
|
|
|
If no allowlist is configured, all users are allowed (open access).
|
|
|
|
|
"""
|
|
|
|
|
user_id = source.user_id
|
|
|
|
|
if not user_id:
|
|
|
|
|
return False # Can't verify unknown users
|
|
|
|
|
|
|
|
|
|
# Check platform-specific allowlist first
|
|
|
|
|
platform_env_map = {
|
|
|
|
|
Platform.TELEGRAM: "TELEGRAM_ALLOWED_USERS",
|
|
|
|
|
Platform.DISCORD: "DISCORD_ALLOWED_USERS",
|
|
|
|
|
Platform.WHATSAPP: "WHATSAPP_ALLOWED_USERS",
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
platform_allowlist = os.getenv(platform_env_map.get(source.platform, ""))
|
|
|
|
|
global_allowlist = os.getenv("GATEWAY_ALLOWED_USERS", "")
|
|
|
|
|
|
|
|
|
|
# If no allowlists configured, allow all (backward compatible)
|
|
|
|
|
if not platform_allowlist and not global_allowlist:
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
# Check if user is in any allowlist
|
|
|
|
|
allowed_ids = set()
|
|
|
|
|
if platform_allowlist:
|
|
|
|
|
allowed_ids.update(uid.strip() for uid in platform_allowlist.split(","))
|
|
|
|
|
if global_allowlist:
|
|
|
|
|
allowed_ids.update(uid.strip() for uid in global_allowlist.split(","))
|
|
|
|
|
|
|
|
|
|
return user_id in allowed_ids
|
|
|
|
|
|
2026-02-02 19:01:51 -08:00
|
|
|
async def _handle_message(self, event: MessageEvent) -> Optional[str]:
|
|
|
|
|
"""
|
|
|
|
|
Handle an incoming message from any platform.
|
|
|
|
|
|
|
|
|
|
This is the core message processing pipeline:
|
2026-02-03 10:46:23 -08:00
|
|
|
1. Check user authorization
|
|
|
|
|
2. Check for commands (/new, /reset, etc.)
|
2026-02-03 16:15:49 -08:00
|
|
|
3. Check for running agent and interrupt if needed
|
|
|
|
|
4. Get or create session
|
|
|
|
|
5. Build context for agent
|
|
|
|
|
6. Run agent conversation
|
|
|
|
|
7. Return response
|
2026-02-02 19:01:51 -08:00
|
|
|
"""
|
|
|
|
|
source = event.source
|
|
|
|
|
|
2026-02-03 10:46:23 -08:00
|
|
|
# Check if user is authorized
|
|
|
|
|
if not self._is_user_authorized(source):
|
|
|
|
|
print(f"[gateway] Unauthorized user: {source.user_id} ({source.user_name}) on {source.platform.value}")
|
|
|
|
|
return None # Silently ignore unauthorized users
|
|
|
|
|
|
2026-02-03 16:15:49 -08:00
|
|
|
# Check for commands
|
2026-02-02 19:01:51 -08:00
|
|
|
command = event.get_command()
|
|
|
|
|
if command in ["new", "reset"]:
|
|
|
|
|
return await self._handle_reset_command(event)
|
|
|
|
|
|
|
|
|
|
if command == "status":
|
|
|
|
|
return await self._handle_status_command(event)
|
|
|
|
|
|
2026-02-03 16:15:49 -08:00
|
|
|
if command == "stop":
|
|
|
|
|
return await self._handle_stop_command(event)
|
|
|
|
|
|
2026-02-12 10:05:08 -08:00
|
|
|
# Check for pending exec approval responses
|
|
|
|
|
session_key_preview = f"agent:main:{source.platform.value}:{source.chat_type}:{source.chat_id}" if source.chat_type != "dm" else f"agent:main:{source.platform.value}:dm"
|
|
|
|
|
if session_key_preview in self._pending_approvals:
|
|
|
|
|
user_text = event.text.strip().lower()
|
|
|
|
|
if user_text in ("yes", "y", "approve", "ok", "go", "do it"):
|
|
|
|
|
approval = self._pending_approvals.pop(session_key_preview)
|
|
|
|
|
cmd = approval["command"]
|
|
|
|
|
pattern_key = approval.get("pattern_key", "")
|
|
|
|
|
print(f"[gateway] ✅ User approved dangerous command: {cmd[:60]}...")
|
|
|
|
|
# Approve for session and re-run via terminal_tool with force=True
|
|
|
|
|
from tools.terminal_tool import terminal_tool, _session_approved_patterns
|
|
|
|
|
_session_approved_patterns.add(pattern_key)
|
|
|
|
|
result = terminal_tool(command=cmd, force=True)
|
|
|
|
|
return f"✅ Command approved and executed.\n\n```\n{result[:3500]}\n```"
|
|
|
|
|
elif user_text in ("no", "n", "deny", "cancel", "nope"):
|
|
|
|
|
self._pending_approvals.pop(session_key_preview)
|
|
|
|
|
return "❌ Command denied."
|
|
|
|
|
# If it's not clearly an approval/denial, fall through to normal processing
|
|
|
|
|
|
2026-02-02 19:01:51 -08:00
|
|
|
# Get or create session
|
|
|
|
|
session_entry = self.session_store.get_or_create_session(source)
|
2026-02-03 16:15:49 -08:00
|
|
|
session_key = session_entry.session_key
|
|
|
|
|
|
|
|
|
|
# Check if there's already a running agent for this session
|
|
|
|
|
if session_key in self._running_agents:
|
|
|
|
|
running_agent = self._running_agents[session_key]
|
|
|
|
|
print(f"[gateway] ⚡ Interrupting running agent for session {session_key[:20]}...")
|
|
|
|
|
running_agent.interrupt(event.text)
|
|
|
|
|
# Store the new message to be processed after current agent finishes
|
|
|
|
|
self._pending_messages[session_key] = event.text
|
|
|
|
|
return None # Don't respond yet - let the interrupt handle it
|
2026-02-02 19:01:51 -08:00
|
|
|
|
|
|
|
|
# Build session context
|
|
|
|
|
context = build_session_context(source, self.config, session_entry)
|
|
|
|
|
|
|
|
|
|
# Set environment variables for tools
|
|
|
|
|
self._set_session_env(context)
|
|
|
|
|
|
|
|
|
|
# Build the context prompt to inject
|
|
|
|
|
context_prompt = build_session_context_prompt(context)
|
|
|
|
|
|
|
|
|
|
# Load conversation history from transcript
|
|
|
|
|
history = self.session_store.load_transcript(session_entry.session_id)
|
|
|
|
|
|
2026-02-15 16:10:50 -08:00
|
|
|
# -----------------------------------------------------------------
|
|
|
|
|
# Auto-analyze images sent by the user
|
|
|
|
|
#
|
|
|
|
|
# If the user attached image(s), we run the vision tool eagerly so
|
|
|
|
|
# the conversation model always receives a text description. The
|
|
|
|
|
# local file path is also included so the model can re-examine the
|
|
|
|
|
# image later with a more targeted question via vision_analyze.
|
|
|
|
|
#
|
|
|
|
|
# We filter to image paths only (by media_type) so that non-image
|
|
|
|
|
# attachments (documents, audio, etc.) are not sent to the vision
|
|
|
|
|
# tool even when they appear in the same message.
|
|
|
|
|
# -----------------------------------------------------------------
|
|
|
|
|
message_text = event.text or ""
|
|
|
|
|
if event.media_urls:
|
|
|
|
|
image_paths = []
|
|
|
|
|
for i, path in enumerate(event.media_urls):
|
|
|
|
|
# Check media_types if available; otherwise infer from message type
|
|
|
|
|
mtype = event.media_types[i] if i < len(event.media_types) else ""
|
|
|
|
|
is_image = (
|
|
|
|
|
mtype.startswith("image/")
|
|
|
|
|
or event.message_type == MessageType.PHOTO
|
|
|
|
|
)
|
|
|
|
|
if is_image:
|
|
|
|
|
image_paths.append(path)
|
|
|
|
|
if image_paths:
|
|
|
|
|
message_text = await self._enrich_message_with_vision(
|
|
|
|
|
message_text, image_paths
|
|
|
|
|
)
|
|
|
|
|
|
2026-02-02 19:01:51 -08:00
|
|
|
try:
|
|
|
|
|
# Run the agent
|
|
|
|
|
response = await self._run_agent(
|
2026-02-15 16:10:50 -08:00
|
|
|
message=message_text,
|
2026-02-02 19:01:51 -08:00
|
|
|
context_prompt=context_prompt,
|
|
|
|
|
history=history,
|
|
|
|
|
source=source,
|
2026-02-03 16:15:49 -08:00
|
|
|
session_id=session_entry.session_id,
|
|
|
|
|
session_key=session_key
|
2026-02-02 19:01:51 -08:00
|
|
|
)
|
|
|
|
|
|
2026-02-12 10:05:08 -08:00
|
|
|
# Check if the agent encountered a dangerous command needing approval
|
|
|
|
|
# The terminal tool stores the last pending approval globally
|
|
|
|
|
try:
|
|
|
|
|
from tools.terminal_tool import _last_pending_approval
|
|
|
|
|
if _last_pending_approval:
|
|
|
|
|
self._pending_approvals[session_key] = _last_pending_approval.copy()
|
|
|
|
|
# Clear the global so it doesn't leak to other sessions
|
|
|
|
|
_last_pending_approval.clear()
|
|
|
|
|
except Exception:
|
|
|
|
|
pass
|
|
|
|
|
|
2026-02-15 16:10:50 -08:00
|
|
|
# Append to transcript (use the enriched message so vision context is preserved)
|
2026-02-02 19:01:51 -08:00
|
|
|
self.session_store.append_to_transcript(
|
|
|
|
|
session_entry.session_id,
|
2026-02-15 16:10:50 -08:00
|
|
|
{"role": "user", "content": message_text, "timestamp": datetime.now().isoformat()}
|
2026-02-02 19:01:51 -08:00
|
|
|
)
|
|
|
|
|
self.session_store.append_to_transcript(
|
|
|
|
|
session_entry.session_id,
|
|
|
|
|
{"role": "assistant", "content": response, "timestamp": datetime.now().isoformat()}
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
# Update session
|
|
|
|
|
self.session_store.update_session(session_entry.session_key)
|
|
|
|
|
|
|
|
|
|
return response
|
|
|
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
print(f"[gateway] Agent error: {e}")
|
|
|
|
|
return f"Sorry, I encountered an error: {str(e)}"
|
|
|
|
|
finally:
|
|
|
|
|
# Clear session env
|
|
|
|
|
self._clear_session_env()
|
|
|
|
|
|
|
|
|
|
async def _handle_reset_command(self, event: MessageEvent) -> str:
|
|
|
|
|
"""Handle /new or /reset command."""
|
|
|
|
|
source = event.source
|
|
|
|
|
|
|
|
|
|
# Get existing session key
|
|
|
|
|
session_key = f"agent:main:{source.platform.value}:" + \
|
|
|
|
|
(f"dm" if source.chat_type == "dm" else f"{source.chat_type}:{source.chat_id}")
|
|
|
|
|
|
|
|
|
|
# Reset the session
|
|
|
|
|
new_entry = self.session_store.reset_session(session_key)
|
|
|
|
|
|
|
|
|
|
if new_entry:
|
|
|
|
|
return "✨ Session reset! I've started fresh with no memory of our previous conversation."
|
|
|
|
|
else:
|
|
|
|
|
# No existing session, just create one
|
|
|
|
|
self.session_store.get_or_create_session(source, force_new=True)
|
|
|
|
|
return "✨ New session started!"
|
|
|
|
|
|
|
|
|
|
async def _handle_status_command(self, event: MessageEvent) -> str:
|
|
|
|
|
"""Handle /status command."""
|
|
|
|
|
source = event.source
|
|
|
|
|
session_entry = self.session_store.get_or_create_session(source)
|
|
|
|
|
|
|
|
|
|
connected_platforms = [p.value for p in self.adapters.keys()]
|
|
|
|
|
|
2026-02-03 16:15:49 -08:00
|
|
|
# Check if there's an active agent
|
|
|
|
|
session_key = session_entry.session_key
|
|
|
|
|
is_running = session_key in self._running_agents
|
|
|
|
|
|
2026-02-02 19:01:51 -08:00
|
|
|
lines = [
|
|
|
|
|
"📊 **Hermes Gateway Status**",
|
|
|
|
|
"",
|
|
|
|
|
f"**Session ID:** `{session_entry.session_id[:12]}...`",
|
|
|
|
|
f"**Created:** {session_entry.created_at.strftime('%Y-%m-%d %H:%M')}",
|
|
|
|
|
f"**Last Activity:** {session_entry.updated_at.strftime('%Y-%m-%d %H:%M')}",
|
|
|
|
|
f"**Tokens:** {session_entry.total_tokens:,}",
|
2026-02-03 16:15:49 -08:00
|
|
|
f"**Agent Running:** {'Yes ⚡' if is_running else 'No'}",
|
2026-02-02 19:01:51 -08:00
|
|
|
"",
|
|
|
|
|
f"**Connected Platforms:** {', '.join(connected_platforms)}",
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
return "\n".join(lines)
|
|
|
|
|
|
2026-02-03 16:15:49 -08:00
|
|
|
async def _handle_stop_command(self, event: MessageEvent) -> str:
|
|
|
|
|
"""Handle /stop command - interrupt a running agent."""
|
|
|
|
|
source = event.source
|
|
|
|
|
session_entry = self.session_store.get_or_create_session(source)
|
|
|
|
|
session_key = session_entry.session_key
|
|
|
|
|
|
|
|
|
|
if session_key in self._running_agents:
|
|
|
|
|
agent = self._running_agents[session_key]
|
|
|
|
|
agent.interrupt()
|
|
|
|
|
return "⚡ Stopping the current task... The agent will finish its current step and respond."
|
|
|
|
|
else:
|
|
|
|
|
return "No active task to stop."
|
|
|
|
|
|
2026-02-02 19:01:51 -08:00
|
|
|
def _set_session_env(self, context: SessionContext) -> None:
|
|
|
|
|
"""Set environment variables for the current session."""
|
|
|
|
|
os.environ["HERMES_SESSION_PLATFORM"] = context.source.platform.value
|
|
|
|
|
os.environ["HERMES_SESSION_CHAT_ID"] = context.source.chat_id
|
|
|
|
|
if context.source.chat_name:
|
|
|
|
|
os.environ["HERMES_SESSION_CHAT_NAME"] = context.source.chat_name
|
|
|
|
|
|
|
|
|
|
def _clear_session_env(self) -> None:
|
|
|
|
|
"""Clear session environment variables."""
|
|
|
|
|
for var in ["HERMES_SESSION_PLATFORM", "HERMES_SESSION_CHAT_ID", "HERMES_SESSION_CHAT_NAME"]:
|
|
|
|
|
if var in os.environ:
|
|
|
|
|
del os.environ[var]
|
|
|
|
|
|
2026-02-15 16:10:50 -08:00
|
|
|
async def _enrich_message_with_vision(
|
|
|
|
|
self,
|
|
|
|
|
user_text: str,
|
|
|
|
|
image_paths: List[str],
|
|
|
|
|
) -> str:
|
|
|
|
|
"""
|
|
|
|
|
Auto-analyze user-attached images with the vision tool and prepend
|
|
|
|
|
the descriptions to the message text.
|
|
|
|
|
|
|
|
|
|
Each image is analyzed with a general-purpose prompt. The resulting
|
|
|
|
|
description *and* the local cache path are injected so the model can:
|
|
|
|
|
1. Immediately understand what the user sent (no extra tool call).
|
|
|
|
|
2. Re-examine the image with vision_analyze if it needs more detail.
|
|
|
|
|
|
|
|
|
|
Args:
|
|
|
|
|
user_text: The user's original caption / message text.
|
|
|
|
|
image_paths: List of local file paths to cached images.
|
|
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
|
The enriched message string with vision descriptions prepended.
|
|
|
|
|
"""
|
|
|
|
|
from tools.vision_tools import vision_analyze_tool
|
|
|
|
|
import json as _json
|
|
|
|
|
|
|
|
|
|
analysis_prompt = (
|
|
|
|
|
"Describe everything visible in this image in thorough detail. "
|
|
|
|
|
"Include any text, code, data, objects, people, layout, colors, "
|
|
|
|
|
"and any other notable visual information."
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
enriched_parts = []
|
|
|
|
|
for path in image_paths:
|
|
|
|
|
try:
|
|
|
|
|
print(f"[gateway] Auto-analyzing user image: {path}", flush=True)
|
|
|
|
|
result_json = await vision_analyze_tool(
|
|
|
|
|
image_url=path,
|
|
|
|
|
user_prompt=analysis_prompt,
|
|
|
|
|
)
|
|
|
|
|
result = _json.loads(result_json)
|
|
|
|
|
if result.get("success"):
|
|
|
|
|
description = result.get("analysis", "")
|
|
|
|
|
enriched_parts.append(
|
|
|
|
|
f"[User sent an image. Vision analysis:\n{description}]\n"
|
|
|
|
|
f"[To examine this image further, use vision_analyze with "
|
|
|
|
|
f"image_url: {path}]"
|
|
|
|
|
)
|
|
|
|
|
else:
|
|
|
|
|
# Analysis failed -- still tell the model the image exists
|
|
|
|
|
enriched_parts.append(
|
|
|
|
|
f"[User sent an image but automatic analysis failed. "
|
|
|
|
|
f"You can try analyzing it with vision_analyze using "
|
|
|
|
|
f"image_url: {path}]"
|
|
|
|
|
)
|
|
|
|
|
except Exception as e:
|
|
|
|
|
print(f"[gateway] Vision auto-analysis error: {e}", flush=True)
|
|
|
|
|
enriched_parts.append(
|
|
|
|
|
f"[User sent an image but automatic analysis encountered an error. "
|
|
|
|
|
f"You can try analyzing it with vision_analyze using "
|
|
|
|
|
f"image_url: {path}]"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
# Combine: vision descriptions first, then the user's original text
|
|
|
|
|
if enriched_parts:
|
|
|
|
|
prefix = "\n\n".join(enriched_parts)
|
|
|
|
|
if user_text:
|
|
|
|
|
return f"{prefix}\n\n{user_text}"
|
|
|
|
|
return prefix
|
|
|
|
|
return user_text
|
|
|
|
|
|
2026-02-02 19:01:51 -08:00
|
|
|
async def _run_agent(
|
|
|
|
|
self,
|
|
|
|
|
message: str,
|
|
|
|
|
context_prompt: str,
|
|
|
|
|
history: List[Dict[str, Any]],
|
|
|
|
|
source: SessionSource,
|
2026-02-03 16:15:49 -08:00
|
|
|
session_id: str,
|
|
|
|
|
session_key: str = None
|
2026-02-02 19:01:51 -08:00
|
|
|
) -> str:
|
|
|
|
|
"""
|
|
|
|
|
Run the agent with the given message and context.
|
|
|
|
|
|
|
|
|
|
This is run in a thread pool to not block the event loop.
|
2026-02-03 16:15:49 -08:00
|
|
|
Supports interruption via new messages.
|
2026-02-02 19:01:51 -08:00
|
|
|
"""
|
|
|
|
|
from run_agent import AIAgent
|
2026-02-03 14:54:43 -08:00
|
|
|
import queue
|
2026-02-02 19:01:51 -08:00
|
|
|
|
|
|
|
|
# Determine toolset based on platform
|
|
|
|
|
toolset_map = {
|
|
|
|
|
Platform.LOCAL: "hermes-cli",
|
|
|
|
|
Platform.TELEGRAM: "hermes-telegram",
|
|
|
|
|
Platform.DISCORD: "hermes-discord",
|
|
|
|
|
Platform.WHATSAPP: "hermes-whatsapp",
|
|
|
|
|
}
|
|
|
|
|
toolset = toolset_map.get(source.platform, "hermes-telegram")
|
|
|
|
|
|
2026-02-03 14:54:43 -08:00
|
|
|
# Check if tool progress notifications are enabled
|
|
|
|
|
tool_progress_enabled = os.getenv("HERMES_TOOL_PROGRESS", "").lower() in ("1", "true", "yes")
|
|
|
|
|
progress_mode = os.getenv("HERMES_TOOL_PROGRESS_MODE", "new") # "all" or "new" (only new tools)
|
|
|
|
|
|
|
|
|
|
# Queue for progress messages (thread-safe)
|
|
|
|
|
progress_queue = queue.Queue() if tool_progress_enabled else None
|
|
|
|
|
last_tool = [None] # Mutable container for tracking in closure
|
|
|
|
|
|
|
|
|
|
def progress_callback(tool_name: str, preview: str = None):
|
|
|
|
|
"""Callback invoked by agent when a tool is called."""
|
|
|
|
|
if not progress_queue:
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
# "new" mode: only report when tool changes
|
|
|
|
|
if progress_mode == "new" and tool_name == last_tool[0]:
|
|
|
|
|
return
|
|
|
|
|
last_tool[0] = tool_name
|
|
|
|
|
|
2026-02-12 10:05:08 -08:00
|
|
|
# Build progress message with primary argument preview
|
2026-02-03 14:54:43 -08:00
|
|
|
tool_emojis = {
|
|
|
|
|
"terminal": "💻",
|
|
|
|
|
"web_search": "🔍",
|
|
|
|
|
"web_extract": "📄",
|
|
|
|
|
"read_file": "📖",
|
|
|
|
|
"write_file": "✍️",
|
2026-02-12 10:05:08 -08:00
|
|
|
"patch": "🔧",
|
|
|
|
|
"search": "🔎",
|
2026-02-03 14:54:43 -08:00
|
|
|
"list_directory": "📂",
|
|
|
|
|
"image_generate": "🎨",
|
2026-02-12 10:05:08 -08:00
|
|
|
"text_to_speech": "🔊",
|
2026-02-03 14:54:43 -08:00
|
|
|
"browser_navigate": "🌐",
|
|
|
|
|
"browser_click": "👆",
|
2026-02-12 10:05:08 -08:00
|
|
|
"browser_type": "⌨️",
|
|
|
|
|
"browser_snapshot": "📸",
|
2026-02-03 14:54:43 -08:00
|
|
|
"moa_query": "🧠",
|
2026-02-12 10:05:08 -08:00
|
|
|
"mixture_of_agents": "🧠",
|
|
|
|
|
"vision_analyze": "👁️",
|
|
|
|
|
"skill_view": "📚",
|
|
|
|
|
"skills_list": "📋",
|
2026-02-03 14:54:43 -08:00
|
|
|
}
|
|
|
|
|
emoji = tool_emojis.get(tool_name, "⚙️")
|
|
|
|
|
|
2026-02-12 10:05:08 -08:00
|
|
|
if preview:
|
|
|
|
|
# Truncate preview to keep messages clean
|
|
|
|
|
if len(preview) > 40:
|
|
|
|
|
preview = preview[:37] + "..."
|
|
|
|
|
msg = f"{emoji} {tool_name}... \"{preview}\""
|
2026-02-03 14:54:43 -08:00
|
|
|
else:
|
|
|
|
|
msg = f"{emoji} {tool_name}..."
|
|
|
|
|
|
|
|
|
|
progress_queue.put(msg)
|
|
|
|
|
|
|
|
|
|
# Background task to send progress messages
|
|
|
|
|
async def send_progress_messages():
|
|
|
|
|
if not progress_queue:
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
adapter = self.adapters.get(source.platform)
|
|
|
|
|
if not adapter:
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
while True:
|
|
|
|
|
try:
|
|
|
|
|
# Non-blocking check with small timeout
|
|
|
|
|
msg = progress_queue.get_nowait()
|
|
|
|
|
await adapter.send(chat_id=source.chat_id, content=msg)
|
2026-02-03 15:06:18 -08:00
|
|
|
# Restore typing indicator after sending progress message
|
|
|
|
|
await asyncio.sleep(0.3)
|
|
|
|
|
await adapter.send_typing(source.chat_id)
|
2026-02-03 14:54:43 -08:00
|
|
|
except queue.Empty:
|
|
|
|
|
await asyncio.sleep(0.3) # Check again soon
|
|
|
|
|
except asyncio.CancelledError:
|
|
|
|
|
# Drain remaining messages
|
|
|
|
|
while not progress_queue.empty():
|
|
|
|
|
try:
|
|
|
|
|
msg = progress_queue.get_nowait()
|
|
|
|
|
await adapter.send(chat_id=source.chat_id, content=msg)
|
|
|
|
|
except:
|
|
|
|
|
break
|
|
|
|
|
return
|
|
|
|
|
except Exception as e:
|
|
|
|
|
print(f"[Gateway] Progress message error: {e}")
|
|
|
|
|
await asyncio.sleep(1)
|
|
|
|
|
|
2026-02-03 16:15:49 -08:00
|
|
|
# We need to share the agent instance for interrupt support
|
|
|
|
|
agent_holder = [None] # Mutable container for the agent instance
|
|
|
|
|
result_holder = [None] # Mutable container for the result
|
|
|
|
|
|
2026-02-02 19:01:51 -08:00
|
|
|
def run_sync():
|
2026-02-03 14:48:19 -08:00
|
|
|
# Read from env var or use default (same as CLI)
|
|
|
|
|
max_iterations = int(os.getenv("HERMES_MAX_ITERATIONS", "60"))
|
|
|
|
|
|
2026-02-12 16:11:16 -08:00
|
|
|
# Map platform enum to the platform hint key the agent understands.
|
|
|
|
|
# Platform.LOCAL ("local") maps to "cli"; others pass through as-is.
|
|
|
|
|
platform_key = "cli" if source.platform == Platform.LOCAL else source.platform.value
|
|
|
|
|
|
2026-02-02 19:01:51 -08:00
|
|
|
agent = AIAgent(
|
2026-02-08 10:49:24 +00:00
|
|
|
model=os.getenv("HERMES_MODEL", "anthropic/claude-opus-4.6"),
|
2026-02-03 14:48:19 -08:00
|
|
|
max_iterations=max_iterations,
|
2026-02-02 19:01:51 -08:00
|
|
|
quiet_mode=True,
|
|
|
|
|
enabled_toolsets=[toolset],
|
|
|
|
|
ephemeral_system_prompt=context_prompt,
|
|
|
|
|
session_id=session_id,
|
2026-02-03 14:54:43 -08:00
|
|
|
tool_progress_callback=progress_callback if tool_progress_enabled else None,
|
2026-02-12 16:11:16 -08:00
|
|
|
platform=platform_key, # Tells the agent which interface to format for
|
2026-02-02 19:01:51 -08:00
|
|
|
)
|
|
|
|
|
|
2026-02-03 16:15:49 -08:00
|
|
|
# Store agent reference for interrupt support
|
|
|
|
|
agent_holder[0] = agent
|
|
|
|
|
|
2026-02-10 16:16:30 -08:00
|
|
|
# Convert history to agent format.
|
|
|
|
|
# Two cases:
|
|
|
|
|
# 1. Normal path (from transcript): simple {role, content, timestamp} dicts
|
|
|
|
|
# - Strip timestamps, keep role+content
|
|
|
|
|
# 2. Interrupt path (from agent result["messages"]): full agent messages
|
|
|
|
|
# that may include tool_calls, tool_call_id, reasoning, etc.
|
|
|
|
|
# - These must be passed through intact so the API sees valid
|
|
|
|
|
# assistant→tool sequences (dropping tool_calls causes 500 errors)
|
2026-02-03 15:42:54 -08:00
|
|
|
agent_history = []
|
|
|
|
|
for msg in history:
|
|
|
|
|
role = msg.get("role")
|
2026-02-10 16:16:30 -08:00
|
|
|
if not role:
|
|
|
|
|
continue
|
|
|
|
|
|
|
|
|
|
# Check if this is a rich agent message (has tool_calls or tool_call_id)
|
|
|
|
|
# If so, pass it through with full structure intact
|
|
|
|
|
has_tool_calls = "tool_calls" in msg
|
|
|
|
|
has_tool_call_id = "tool_call_id" in msg
|
|
|
|
|
is_tool_message = role == "tool"
|
|
|
|
|
|
|
|
|
|
if has_tool_calls or has_tool_call_id or is_tool_message:
|
|
|
|
|
# Preserve full message structure (tool_calls, tool_call_id, etc.)
|
|
|
|
|
# Only strip fields that are purely internal (e.g. timestamp)
|
|
|
|
|
clean_msg = {k: v for k, v in msg.items() if k != "timestamp"}
|
|
|
|
|
agent_history.append(clean_msg)
|
|
|
|
|
else:
|
|
|
|
|
# Simple text message - just need role and content
|
|
|
|
|
content = msg.get("content")
|
|
|
|
|
if content:
|
|
|
|
|
agent_history.append({"role": role, "content": content})
|
2026-02-02 19:01:51 -08:00
|
|
|
|
2026-02-03 15:42:54 -08:00
|
|
|
result = agent.run_conversation(message, conversation_history=agent_history)
|
2026-02-03 16:15:49 -08:00
|
|
|
result_holder[0] = result
|
2026-02-03 15:26:59 -08:00
|
|
|
|
|
|
|
|
# Return final response, or a message if something went wrong
|
|
|
|
|
final_response = result.get("final_response")
|
2026-02-14 16:08:14 -08:00
|
|
|
if not final_response:
|
|
|
|
|
if result.get("error"):
|
|
|
|
|
return f"⚠️ {result['error']}"
|
2026-02-03 15:26:59 -08:00
|
|
|
return "(No response generated)"
|
2026-02-14 16:08:14 -08:00
|
|
|
|
2026-02-14 16:16:54 -08:00
|
|
|
# Scan tool results for MEDIA:<path> tags that need to be delivered
|
|
|
|
|
# as native audio/file attachments. The TTS tool embeds MEDIA: tags
|
|
|
|
|
# in its JSON response, but the model's final text reply usually
|
|
|
|
|
# doesn't include them. We collect unique tags from tool results and
|
|
|
|
|
# append any that aren't already present in the final response, so the
|
|
|
|
|
# adapter's extract_media() can find and deliver the files exactly once.
|
|
|
|
|
if "MEDIA:" not in final_response:
|
|
|
|
|
media_tags = []
|
|
|
|
|
has_voice_directive = False
|
|
|
|
|
for msg in result.get("messages", []):
|
|
|
|
|
if msg.get("role") == "tool" or msg.get("role") == "function":
|
|
|
|
|
content = msg.get("content", "")
|
|
|
|
|
if "MEDIA:" in content:
|
|
|
|
|
for match in re.finditer(r'MEDIA:(\S+)', content):
|
|
|
|
|
path = match.group(1).strip().rstrip('",}')
|
|
|
|
|
if path:
|
|
|
|
|
media_tags.append(f"MEDIA:{path}")
|
|
|
|
|
if "[[audio_as_voice]]" in content:
|
|
|
|
|
has_voice_directive = True
|
|
|
|
|
|
|
|
|
|
if media_tags:
|
|
|
|
|
# Deduplicate while preserving order
|
|
|
|
|
seen = set()
|
|
|
|
|
unique_tags = []
|
|
|
|
|
for tag in media_tags:
|
|
|
|
|
if tag not in seen:
|
|
|
|
|
seen.add(tag)
|
|
|
|
|
unique_tags.append(tag)
|
|
|
|
|
if has_voice_directive:
|
|
|
|
|
unique_tags.insert(0, "[[audio_as_voice]]")
|
|
|
|
|
final_response = final_response + "\n" + "\n".join(unique_tags)
|
2026-02-14 16:08:14 -08:00
|
|
|
|
|
|
|
|
return final_response
|
2026-02-02 19:01:51 -08:00
|
|
|
|
2026-02-03 14:54:43 -08:00
|
|
|
# Start progress message sender if enabled
|
|
|
|
|
progress_task = None
|
|
|
|
|
if tool_progress_enabled:
|
|
|
|
|
progress_task = asyncio.create_task(send_progress_messages())
|
|
|
|
|
|
2026-02-03 16:15:49 -08:00
|
|
|
# Track this agent as running for this session (for interrupt support)
|
|
|
|
|
# We do this in a callback after the agent is created
|
|
|
|
|
async def track_agent():
|
|
|
|
|
# Wait for agent to be created
|
|
|
|
|
while agent_holder[0] is None:
|
|
|
|
|
await asyncio.sleep(0.05)
|
|
|
|
|
if session_key:
|
|
|
|
|
self._running_agents[session_key] = agent_holder[0]
|
|
|
|
|
|
|
|
|
|
tracking_task = asyncio.create_task(track_agent())
|
|
|
|
|
|
2026-02-03 20:10:15 -08:00
|
|
|
# Monitor for interrupts from the adapter (new messages arriving)
|
|
|
|
|
async def monitor_for_interrupt():
|
|
|
|
|
adapter = self.adapters.get(source.platform)
|
|
|
|
|
if not adapter:
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
chat_id = source.chat_id
|
|
|
|
|
while True:
|
|
|
|
|
await asyncio.sleep(0.2) # Check every 200ms
|
|
|
|
|
# Check if adapter has a pending interrupt for this session
|
|
|
|
|
if hasattr(adapter, 'has_pending_interrupt') and adapter.has_pending_interrupt(chat_id):
|
|
|
|
|
agent = agent_holder[0]
|
|
|
|
|
if agent:
|
|
|
|
|
pending_event = adapter.get_pending_message(chat_id)
|
|
|
|
|
pending_text = pending_event.text if pending_event else None
|
|
|
|
|
print(f"[gateway] ⚡ Interrupt detected from adapter, signaling agent...")
|
|
|
|
|
agent.interrupt(pending_text)
|
|
|
|
|
break
|
|
|
|
|
|
|
|
|
|
interrupt_monitor = asyncio.create_task(monitor_for_interrupt())
|
|
|
|
|
|
2026-02-03 14:54:43 -08:00
|
|
|
try:
|
|
|
|
|
# Run in thread pool to not block
|
|
|
|
|
loop = asyncio.get_event_loop()
|
|
|
|
|
response = await loop.run_in_executor(None, run_sync)
|
2026-02-03 16:15:49 -08:00
|
|
|
|
|
|
|
|
# Check if we were interrupted and have a pending message
|
|
|
|
|
result = result_holder[0]
|
2026-02-03 20:10:15 -08:00
|
|
|
adapter = self.adapters.get(source.platform)
|
|
|
|
|
|
|
|
|
|
# Get pending message from adapter if interrupted
|
|
|
|
|
pending = None
|
|
|
|
|
if result and result.get("interrupted") and adapter:
|
|
|
|
|
pending_event = adapter.get_pending_message(source.chat_id)
|
|
|
|
|
if pending_event:
|
|
|
|
|
pending = pending_event.text
|
|
|
|
|
elif result.get("interrupt_message"):
|
|
|
|
|
pending = result.get("interrupt_message")
|
|
|
|
|
|
|
|
|
|
if pending:
|
|
|
|
|
print(f"[gateway] 📨 Processing interrupted message: '{pending[:40]}...'")
|
2026-02-11 00:05:30 +00:00
|
|
|
|
|
|
|
|
# Clear the adapter's interrupt event so the next _run_agent call
|
|
|
|
|
# doesn't immediately re-trigger the interrupt before the new agent
|
|
|
|
|
# even makes its first API call (this was causing an infinite loop).
|
|
|
|
|
if adapter and hasattr(adapter, '_active_sessions') and source.chat_id in adapter._active_sessions:
|
|
|
|
|
adapter._active_sessions[source.chat_id].clear()
|
|
|
|
|
|
2026-02-10 16:34:27 -08:00
|
|
|
# Don't send the interrupted response to the user — it's just noise
|
|
|
|
|
# like "Operation interrupted." They already know they sent a new
|
|
|
|
|
# message, so go straight to processing it.
|
2026-02-03 20:10:15 -08:00
|
|
|
|
|
|
|
|
# Now process the pending message with updated history
|
|
|
|
|
updated_history = result.get("messages", history)
|
|
|
|
|
return await self._run_agent(
|
|
|
|
|
message=pending,
|
|
|
|
|
context_prompt=context_prompt,
|
|
|
|
|
history=updated_history,
|
|
|
|
|
source=source,
|
|
|
|
|
session_id=session_id,
|
|
|
|
|
session_key=session_key
|
|
|
|
|
)
|
2026-02-03 14:54:43 -08:00
|
|
|
finally:
|
2026-02-03 20:10:15 -08:00
|
|
|
# Stop progress sender and interrupt monitor
|
2026-02-03 14:54:43 -08:00
|
|
|
if progress_task:
|
|
|
|
|
progress_task.cancel()
|
2026-02-03 20:10:15 -08:00
|
|
|
interrupt_monitor.cancel()
|
2026-02-03 16:15:49 -08:00
|
|
|
|
|
|
|
|
# Clean up tracking
|
|
|
|
|
tracking_task.cancel()
|
|
|
|
|
if session_key and session_key in self._running_agents:
|
|
|
|
|
del self._running_agents[session_key]
|
2026-02-03 20:10:15 -08:00
|
|
|
|
|
|
|
|
# Wait for cancelled tasks
|
|
|
|
|
for task in [progress_task, interrupt_monitor, tracking_task]:
|
|
|
|
|
if task:
|
|
|
|
|
try:
|
|
|
|
|
await task
|
|
|
|
|
except asyncio.CancelledError:
|
|
|
|
|
pass
|
2026-02-02 19:01:51 -08:00
|
|
|
|
|
|
|
|
return response
|
|
|
|
|
|
|
|
|
|
|
2026-02-10 16:01:00 -08:00
|
|
|
async def start_gateway(config: Optional[GatewayConfig] = None) -> bool:
|
2026-02-02 19:01:51 -08:00
|
|
|
"""
|
|
|
|
|
Start the gateway and run until interrupted.
|
|
|
|
|
|
|
|
|
|
This is the main entry point for running the gateway.
|
2026-02-10 16:01:00 -08:00
|
|
|
Returns True if the gateway ran successfully, False if it failed to start.
|
|
|
|
|
A False return causes a non-zero exit code so systemd can auto-restart.
|
2026-02-02 19:01:51 -08:00
|
|
|
"""
|
|
|
|
|
runner = GatewayRunner(config)
|
|
|
|
|
|
|
|
|
|
# Set up signal handlers
|
|
|
|
|
def signal_handler():
|
|
|
|
|
asyncio.create_task(runner.stop())
|
|
|
|
|
|
|
|
|
|
loop = asyncio.get_event_loop()
|
|
|
|
|
for sig in (signal.SIGINT, signal.SIGTERM):
|
|
|
|
|
try:
|
|
|
|
|
loop.add_signal_handler(sig, signal_handler)
|
|
|
|
|
except NotImplementedError:
|
|
|
|
|
# Windows doesn't support add_signal_handler
|
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
# Start the gateway
|
|
|
|
|
success = await runner.start()
|
|
|
|
|
if not success:
|
2026-02-10 16:01:00 -08:00
|
|
|
return False
|
2026-02-02 19:01:51 -08:00
|
|
|
|
|
|
|
|
# Wait for shutdown
|
|
|
|
|
await runner.wait_for_shutdown()
|
2026-02-10 16:01:00 -08:00
|
|
|
return True
|
2026-02-02 19:01:51 -08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
|
"""CLI entry point for the gateway."""
|
|
|
|
|
import argparse
|
|
|
|
|
|
|
|
|
|
parser = argparse.ArgumentParser(description="Hermes Gateway - Multi-platform messaging")
|
|
|
|
|
parser.add_argument("--config", "-c", help="Path to gateway config file")
|
|
|
|
|
parser.add_argument("--verbose", "-v", action="store_true", help="Verbose output")
|
|
|
|
|
|
|
|
|
|
args = parser.parse_args()
|
|
|
|
|
|
|
|
|
|
config = None
|
|
|
|
|
if args.config:
|
|
|
|
|
import json
|
|
|
|
|
with open(args.config) as f:
|
|
|
|
|
data = json.load(f)
|
|
|
|
|
config = GatewayConfig.from_dict(data)
|
|
|
|
|
|
2026-02-10 16:01:00 -08:00
|
|
|
# Run the gateway - exit with code 1 if no platforms connected,
|
|
|
|
|
# so systemd Restart=on-failure will retry on transient errors (e.g. DNS)
|
|
|
|
|
success = asyncio.run(start_gateway(config))
|
|
|
|
|
if not success:
|
|
|
|
|
sys.exit(1)
|
2026-02-02 19:01:51 -08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
|
main()
|