2026-02-02 19:01:51 -08:00
|
|
|
"""
|
|
|
|
|
WhatsApp platform adapter.
|
|
|
|
|
|
|
|
|
|
WhatsApp integration is more complex than Telegram/Discord because:
|
|
|
|
|
- No official bot API for personal accounts
|
|
|
|
|
- Business API requires Meta Business verification
|
|
|
|
|
- Most solutions use web-based automation
|
|
|
|
|
|
|
|
|
|
This adapter supports multiple backends:
|
|
|
|
|
1. WhatsApp Business API (requires Meta verification)
|
|
|
|
|
2. whatsapp-web.js (via Node.js subprocess) - for personal accounts
|
|
|
|
|
3. Baileys (via Node.js subprocess) - alternative for personal accounts
|
|
|
|
|
|
|
|
|
|
For simplicity, we'll implement a generic interface that can work
|
|
|
|
|
with different backends via a bridge pattern.
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
import asyncio
|
2026-02-21 03:32:11 -08:00
|
|
|
import logging
|
2026-02-25 21:04:36 -08:00
|
|
|
import os
|
2026-03-01 01:54:27 +03:00
|
|
|
import platform
|
2026-02-02 19:01:51 -08:00
|
|
|
import subprocess
|
2026-03-01 01:54:27 +03:00
|
|
|
|
|
|
|
|
_IS_WINDOWS = platform.system() == "Windows"
|
2026-02-02 19:01:51 -08:00
|
|
|
from pathlib import Path
|
chore: remove ~100 unused imports across 55 files (#3016)
Automated cleanup via pyflakes + autoflake with manual review.
Changes:
- Removed unused stdlib imports (os, sys, json, pathlib.Path, etc.)
- Removed unused typing imports (List, Dict, Any, Optional, Tuple, Set, etc.)
- Removed unused internal imports (hermes_cli.auth, hermes_cli.config, etc.)
- Fixed cli.py: removed 8 shadowed banner imports (imported from hermes_cli.banner
then immediately redefined locally — only build_welcome_banner is actually used)
- Added noqa comments to imports that appear unused but serve a purpose:
- Re-exports (gateway/session.py SessionResetPolicy, tools/terminal_tool.py
is_interrupted/_interrupt_event)
- SDK presence checks in try/except (daytona, fal_client, discord)
- Test mock targets (auxiliary_client.py Path, mcp_config.py get_hermes_home)
Zero behavioral changes. Full test suite passes (6162/6162, 2 pre-existing
streaming test failures unrelated to this change).
2026-03-25 15:02:03 -07:00
|
|
|
from typing import Dict, Optional, Any
|
2026-02-02 19:01:51 -08:00
|
|
|
|
fix(cli): respect HERMES_HOME in all remaining hardcoded ~/.hermes paths
Several files resolved paths via Path.home() / ".hermes" or
os.path.expanduser("~/.hermes/..."), bypassing the HERMES_HOME
environment variable. This broke isolation when running multiple
Hermes instances with distinct HERMES_HOME directories.
Replace all hardcoded paths with calls to get_hermes_home() from
hermes_cli.config, consistent with the rest of the codebase.
Files fixed:
- tools/process_registry.py (processes.json)
- gateway/pairing.py (pairing/)
- gateway/sticker_cache.py (sticker_cache.json)
- gateway/channel_directory.py (channel_directory.json, sessions.json)
- gateway/config.py (gateway.json, config.yaml, sessions_dir)
- gateway/mirror.py (sessions/)
- gateway/hooks.py (hooks/)
- gateway/platforms/base.py (image_cache/, audio_cache/, document_cache/)
- gateway/platforms/whatsapp.py (whatsapp/session)
- gateway/delivery.py (cron/output)
- agent/auxiliary_client.py (auth.json)
- agent/prompt_builder.py (SOUL.md)
- cli.py (config.yaml, images/, pastes/, history)
- run_agent.py (logs/)
- tools/environments/base.py (sandboxes/)
- tools/environments/modal.py (modal_snapshots.json)
- tools/environments/singularity.py (singularity_snapshots.json)
- tools/tts_tool.py (audio_cache)
- hermes_cli/status.py (cron/jobs.json, sessions.json)
- hermes_cli/gateway.py (logs/, whatsapp session)
- hermes_cli/main.py (whatsapp/session)
Tests updated to use HERMES_HOME env var instead of patching Path.home().
Closes #892
(cherry picked from commit 78ac1bba43b8b74a934c6172f2c29bb4d03164b9)
2026-03-11 07:31:41 +01:00
|
|
|
from hermes_cli.config import get_hermes_home
|
2026-03-28 15:22:19 -07:00
|
|
|
from hermes_constants import get_hermes_dir
|
fix(cli): respect HERMES_HOME in all remaining hardcoded ~/.hermes paths
Several files resolved paths via Path.home() / ".hermes" or
os.path.expanduser("~/.hermes/..."), bypassing the HERMES_HOME
environment variable. This broke isolation when running multiple
Hermes instances with distinct HERMES_HOME directories.
Replace all hardcoded paths with calls to get_hermes_home() from
hermes_cli.config, consistent with the rest of the codebase.
Files fixed:
- tools/process_registry.py (processes.json)
- gateway/pairing.py (pairing/)
- gateway/sticker_cache.py (sticker_cache.json)
- gateway/channel_directory.py (channel_directory.json, sessions.json)
- gateway/config.py (gateway.json, config.yaml, sessions_dir)
- gateway/mirror.py (sessions/)
- gateway/hooks.py (hooks/)
- gateway/platforms/base.py (image_cache/, audio_cache/, document_cache/)
- gateway/platforms/whatsapp.py (whatsapp/session)
- gateway/delivery.py (cron/output)
- agent/auxiliary_client.py (auth.json)
- agent/prompt_builder.py (SOUL.md)
- cli.py (config.yaml, images/, pastes/, history)
- run_agent.py (logs/)
- tools/environments/base.py (sandboxes/)
- tools/environments/modal.py (modal_snapshots.json)
- tools/environments/singularity.py (singularity_snapshots.json)
- tools/tts_tool.py (audio_cache)
- hermes_cli/status.py (cron/jobs.json, sessions.json)
- hermes_cli/gateway.py (logs/, whatsapp session)
- hermes_cli/main.py (whatsapp/session)
Tests updated to use HERMES_HOME env var instead of patching Path.home().
Closes #892
(cherry picked from commit 78ac1bba43b8b74a934c6172f2c29bb4d03164b9)
2026-03-11 07:31:41 +01:00
|
|
|
|
2026-02-21 03:32:11 -08:00
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
2026-03-05 17:13:14 +03:00
|
|
|
|
|
|
|
|
def _kill_port_process(port: int) -> None:
|
|
|
|
|
"""Kill any process listening on the given TCP port."""
|
|
|
|
|
try:
|
|
|
|
|
if _IS_WINDOWS:
|
|
|
|
|
# Use netstat to find the PID bound to this port, then taskkill
|
|
|
|
|
result = subprocess.run(
|
|
|
|
|
["netstat", "-ano", "-p", "TCP"],
|
|
|
|
|
capture_output=True, text=True, timeout=5,
|
|
|
|
|
)
|
|
|
|
|
for line in result.stdout.splitlines():
|
|
|
|
|
parts = line.split()
|
|
|
|
|
if len(parts) >= 5 and parts[3] == "LISTENING":
|
|
|
|
|
local_addr = parts[1]
|
|
|
|
|
if local_addr.endswith(f":{port}"):
|
|
|
|
|
try:
|
|
|
|
|
subprocess.run(
|
|
|
|
|
["taskkill", "/PID", parts[4], "/F"],
|
|
|
|
|
capture_output=True, timeout=5,
|
|
|
|
|
)
|
|
|
|
|
except subprocess.SubprocessError:
|
|
|
|
|
pass
|
|
|
|
|
else:
|
|
|
|
|
result = subprocess.run(
|
|
|
|
|
["fuser", f"{port}/tcp"],
|
|
|
|
|
capture_output=True, timeout=5,
|
|
|
|
|
)
|
|
|
|
|
if result.returncode == 0:
|
|
|
|
|
subprocess.run(
|
|
|
|
|
["fuser", "-k", f"{port}/tcp"],
|
|
|
|
|
capture_output=True, timeout=5,
|
|
|
|
|
)
|
|
|
|
|
except Exception:
|
|
|
|
|
pass
|
|
|
|
|
|
2026-02-02 19:01:51 -08:00
|
|
|
import sys
|
2026-02-21 04:17:27 -08:00
|
|
|
sys.path.insert(0, str(Path(__file__).resolve().parents[2]))
|
2026-02-02 19:01:51 -08:00
|
|
|
|
|
|
|
|
from gateway.config import Platform, PlatformConfig
|
|
|
|
|
from gateway.platforms.base import (
|
|
|
|
|
BasePlatformAdapter,
|
|
|
|
|
MessageEvent,
|
|
|
|
|
MessageType,
|
|
|
|
|
SendResult,
|
fix(whatsapp): download documents, audio, and video media from messages (#2978)
Add downloadMediaMessage() calls for documents, audio/voice notes, and
video in bridge.js — previously only images were downloaded, leaving all
other file types inaccessible to the agent.
Handle local file paths from the bridge for DOCUMENT, VOICE, and VIDEO
types in whatsapp.py with proper MIME detection. Inject text content
inline for readable files (.txt, .md, .csv, .json, etc.).
Follow-up fixes applied during salvage:
- Remove unused cache_document_from_bytes import
- Add 100KB size cap on text injection (matches Telegram/Discord/Slack)
- Align injection format with other platforms
Cherry-picked from PR #2818. Also fixes #2856 (bugs 1 & 2).
PR #2865 by ayberkesn fixed the same voice note issue.
Co-authored-by: noestelar <hola@noeali.com>
2026-03-25 08:37:28 -07:00
|
|
|
SUPPORTED_DOCUMENT_TYPES,
|
2026-02-15 16:10:50 -08:00
|
|
|
cache_image_from_url,
|
Add messaging platform enhancements: STT, stickers, Discord UX, Slack, pairing, hooks
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.
2026-02-15 21:38:59 -08:00
|
|
|
cache_audio_from_url,
|
2026-02-02 19:01:51 -08:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def check_whatsapp_requirements() -> bool:
|
|
|
|
|
"""
|
|
|
|
|
Check if WhatsApp dependencies are available.
|
|
|
|
|
|
|
|
|
|
WhatsApp requires a Node.js bridge for most implementations.
|
|
|
|
|
"""
|
|
|
|
|
# Check for Node.js
|
|
|
|
|
try:
|
|
|
|
|
result = subprocess.run(
|
|
|
|
|
["node", "--version"],
|
|
|
|
|
capture_output=True,
|
|
|
|
|
text=True,
|
|
|
|
|
timeout=5
|
|
|
|
|
)
|
|
|
|
|
return result.returncode == 0
|
|
|
|
|
except Exception:
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class WhatsAppAdapter(BasePlatformAdapter):
|
|
|
|
|
"""
|
|
|
|
|
WhatsApp adapter.
|
|
|
|
|
|
|
|
|
|
This implementation uses a simple HTTP bridge pattern where:
|
|
|
|
|
1. A Node.js process runs the WhatsApp Web client
|
|
|
|
|
2. Messages are forwarded via HTTP/IPC to this Python adapter
|
|
|
|
|
3. Responses are sent back through the bridge
|
|
|
|
|
|
|
|
|
|
The actual Node.js bridge implementation can vary:
|
|
|
|
|
- whatsapp-web.js based
|
|
|
|
|
- Baileys based
|
|
|
|
|
- Business API based
|
|
|
|
|
|
|
|
|
|
Configuration:
|
|
|
|
|
- bridge_script: Path to the Node.js bridge script
|
|
|
|
|
- bridge_port: Port for HTTP communication (default: 3000)
|
|
|
|
|
- session_path: Path to store WhatsApp session data
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
# WhatsApp message limits
|
|
|
|
|
MAX_MESSAGE_LENGTH = 65536 # WhatsApp allows longer messages
|
|
|
|
|
|
2026-02-25 21:04:36 -08:00
|
|
|
# Default bridge location relative to the hermes-agent install
|
|
|
|
|
_DEFAULT_BRIDGE_DIR = Path(__file__).resolve().parents[2] / "scripts" / "whatsapp-bridge"
|
|
|
|
|
|
2026-02-02 19:01:51 -08:00
|
|
|
def __init__(self, config: PlatformConfig):
|
|
|
|
|
super().__init__(config, Platform.WHATSAPP)
|
|
|
|
|
self._bridge_process: Optional[subprocess.Popen] = None
|
|
|
|
|
self._bridge_port: int = config.extra.get("bridge_port", 3000)
|
2026-02-25 21:04:36 -08:00
|
|
|
self._bridge_script: Optional[str] = config.extra.get(
|
|
|
|
|
"bridge_script",
|
|
|
|
|
str(self._DEFAULT_BRIDGE_DIR / "bridge.js"),
|
|
|
|
|
)
|
2026-02-02 19:01:51 -08:00
|
|
|
self._session_path: Path = Path(config.extra.get(
|
|
|
|
|
"session_path",
|
2026-03-28 15:22:19 -07:00
|
|
|
get_hermes_dir("platforms/whatsapp/session", "whatsapp/session")
|
2026-02-02 19:01:51 -08:00
|
|
|
))
|
feat: OpenAI-compatible API server + WhatsApp configurable reply prefix (#1756)
* feat: OpenAI-compatible API server platform adapter
Salvaged from PR #956, updated for current main.
Adds an HTTP API server as a gateway platform adapter that exposes
hermes-agent via the OpenAI Chat Completions and Responses APIs.
Any OpenAI-compatible frontend (Open WebUI, LobeChat, LibreChat,
AnythingLLM, NextChat, ChatBox, etc.) can connect by pointing at
http://localhost:8642/v1.
Endpoints:
- POST /v1/chat/completions — stateless Chat Completions API
- POST /v1/responses — stateful Responses API with chaining
- GET /v1/responses/{id} — retrieve stored response
- DELETE /v1/responses/{id} — delete stored response
- GET /v1/models — list hermes-agent as available model
- GET /health — health check
Features:
- Real SSE streaming via stream_delta_callback (uses main's streaming)
- In-memory LRU response store for Responses API conversation chaining
- Named conversations via 'conversation' parameter
- Bearer token auth (optional, via API_SERVER_KEY)
- CORS support for browser-based frontends
- System prompt layering (frontend system messages on top of core)
- Real token usage tracking in responses
Integration points:
- Platform.API_SERVER in gateway/config.py
- _create_adapter() branch in gateway/run.py
- API_SERVER_* env vars in hermes_cli/config.py
- Env var overrides in gateway/config.py _apply_env_overrides()
Changes vs original PR #956:
- Removed streaming infrastructure (already on main via stream_consumer.py)
- Removed Telegram reply_to_mode (separate feature, not included)
- Updated _resolve_model() -> _resolve_gateway_model()
- Updated stream_callback -> stream_delta_callback
- Updated connect()/disconnect() to use _mark_connected()/_mark_disconnected()
- Adapted to current Platform enum (includes MATTERMOST, MATRIX, DINGTALK)
Tests: 72 new tests, all passing
Docs: API server guide, Open WebUI integration guide, env var reference
* feat(whatsapp): make reply prefix configurable via config.yaml
Reworked from PR #1764 (ifrederico) to use config.yaml instead of .env.
The WhatsApp bridge prepends a header to every outgoing message.
This was hardcoded to '⚕ *Hermes Agent*'. Users can now customize
or disable it via config.yaml:
whatsapp:
reply_prefix: '' # disable header
reply_prefix: '🤖 *My Bot*\n───\n' # custom prefix
How it works:
- load_gateway_config() reads whatsapp.reply_prefix from config.yaml
and stores it in PlatformConfig.extra['reply_prefix']
- WhatsAppAdapter reads it from config.extra at init
- When spawning bridge.js, the adapter passes it as
WHATSAPP_REPLY_PREFIX in the subprocess environment
- bridge.js handles undefined (default), empty (no header),
or custom values with \\n escape support
- Self-chat echo suppression uses the configured prefix
Also fixes _config_version: was 9 but ENV_VARS_BY_VERSION had a
key 10 (TAVILY_API_KEY), so existing users at v9 would never be
prompted for Tavily. Bumped to 10 to close the gap. Added a
regression test to prevent this from happening again.
Credit: ifrederico (PR #1764) for the bridge.js implementation
and the config version gap discovery.
---------
Co-authored-by: Test <test@test.com>
2026-03-17 10:44:37 -07:00
|
|
|
self._reply_prefix: Optional[str] = config.extra.get("reply_prefix")
|
2026-02-02 19:01:51 -08:00
|
|
|
self._message_queue: asyncio.Queue = asyncio.Queue()
|
2026-03-04 04:58:21 -08:00
|
|
|
self._bridge_log_fh = None
|
|
|
|
|
self._bridge_log: Optional[Path] = None
|
2026-03-26 14:36:24 -07:00
|
|
|
self._poll_task: Optional[asyncio.Task] = None
|
fix(whatsapp): reuse persistent aiohttp session across requests (#3818)
Replace per-request aiohttp.ClientSession() in every WhatsApp adapter
method with a single persistent self._http_session, matching the pattern
used by Mattermost, HomeAssistant, and SMS adapters.
Changes:
- Create self._http_session in connect(), close in disconnect()
- All bridge HTTP calls (send, edit, send-media, typing, get_chat_info,
poll_messages) now use the shared session
- Explicitly cancel _poll_task on disconnect() instead of relying
solely on self._running = False
- Health-check sessions in connect() remain ephemeral (persistent
session not yet created at that point)
- Remove per-method ImportError guards for aiohttp (always available
when gateway runs via [messaging] extras)
Salvaged from PR #1851 by Himess. The _poll_task storage was already
on main from PR #3267; this adds the disconnect cancellation and the
persistent session.
Tests: 4 new tests for session close, already-closed skip, poll task
cancellation, and done-task skip.
2026-03-29 16:25:20 -07:00
|
|
|
self._http_session: Optional["aiohttp.ClientSession"] = None
|
feat: add profiles — run multiple isolated Hermes instances (#3681)
Each profile is a fully independent HERMES_HOME with its own config,
API keys, memory, sessions, skills, gateway, cron, and state.db.
Core module: hermes_cli/profiles.py (~900 lines)
- Profile CRUD: create, delete, list, show, rename
- Three clone levels: blank, --clone (config), --clone-all (everything)
- Export/import: tar.gz archive for backup and migration
- Wrapper alias scripts (~/.local/bin/<name>)
- Collision detection for alias names
- Sticky default via ~/.hermes/active_profile
- Skill seeding via subprocess (handles module-level caching)
- Auto-stop gateway on delete with disable-before-stop for services
- Tab completion generation for bash and zsh
CLI integration (hermes_cli/main.py):
- _apply_profile_override(): pre-import -p/--profile flag + sticky default
- Full 'hermes profile' subcommand: list, use, create, delete, show,
alias, rename, export, import
- 'hermes completion bash/zsh' command
- Multi-profile skill sync in hermes update
Display (cli.py, banner.py, gateway/run.py):
- CLI prompt: 'coder ❯' when using a non-default profile
- Banner shows profile name
- Gateway startup log includes profile name
Gateway safety:
- Token locks: Discord, Slack, WhatsApp, Signal (extends Telegram pattern)
- Port conflict detection: API server, webhook adapter
Diagnostics (hermes_cli/doctor.py):
- Profile health section: lists profiles, checks config, .env, aliases
- Orphan alias detection: warns when wrapper points to deleted profile
Tests (tests/hermes_cli/test_profiles.py):
- 71 automated tests covering: validation, CRUD, clone levels, rename,
export/import, active profile, isolation, alias collision, completion
- Full suite: 6760 passed, 0 new failures
Documentation:
- website/docs/user-guide/profiles.md: full user guide (12 sections)
- website/docs/reference/profile-commands.md: command reference (12 commands)
- website/docs/reference/faq.md: 6 profile FAQ entries
- website/sidebars.ts: navigation updated
2026-03-29 10:41:20 -07:00
|
|
|
self._session_lock_identity: Optional[str] = None
|
2026-02-02 19:01:51 -08:00
|
|
|
|
|
|
|
|
async def connect(self) -> bool:
|
|
|
|
|
"""
|
|
|
|
|
Start the WhatsApp bridge.
|
|
|
|
|
|
|
|
|
|
This launches the Node.js bridge process and waits for it to be ready.
|
|
|
|
|
"""
|
|
|
|
|
if not check_whatsapp_requirements():
|
2026-02-25 21:04:36 -08:00
|
|
|
logger.warning("[%s] Node.js not found. WhatsApp requires Node.js.", self.name)
|
2026-02-02 19:01:51 -08:00
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
bridge_path = Path(self._bridge_script)
|
|
|
|
|
if not bridge_path.exists():
|
2026-02-25 21:04:36 -08:00
|
|
|
logger.warning("[%s] Bridge script not found: %s", self.name, bridge_path)
|
2026-02-02 19:01:51 -08:00
|
|
|
return False
|
|
|
|
|
|
2026-02-25 21:04:36 -08:00
|
|
|
logger.info("[%s] Bridge found at %s", self.name, bridge_path)
|
|
|
|
|
|
feat: add profiles — run multiple isolated Hermes instances (#3681)
Each profile is a fully independent HERMES_HOME with its own config,
API keys, memory, sessions, skills, gateway, cron, and state.db.
Core module: hermes_cli/profiles.py (~900 lines)
- Profile CRUD: create, delete, list, show, rename
- Three clone levels: blank, --clone (config), --clone-all (everything)
- Export/import: tar.gz archive for backup and migration
- Wrapper alias scripts (~/.local/bin/<name>)
- Collision detection for alias names
- Sticky default via ~/.hermes/active_profile
- Skill seeding via subprocess (handles module-level caching)
- Auto-stop gateway on delete with disable-before-stop for services
- Tab completion generation for bash and zsh
CLI integration (hermes_cli/main.py):
- _apply_profile_override(): pre-import -p/--profile flag + sticky default
- Full 'hermes profile' subcommand: list, use, create, delete, show,
alias, rename, export, import
- 'hermes completion bash/zsh' command
- Multi-profile skill sync in hermes update
Display (cli.py, banner.py, gateway/run.py):
- CLI prompt: 'coder ❯' when using a non-default profile
- Banner shows profile name
- Gateway startup log includes profile name
Gateway safety:
- Token locks: Discord, Slack, WhatsApp, Signal (extends Telegram pattern)
- Port conflict detection: API server, webhook adapter
Diagnostics (hermes_cli/doctor.py):
- Profile health section: lists profiles, checks config, .env, aliases
- Orphan alias detection: warns when wrapper points to deleted profile
Tests (tests/hermes_cli/test_profiles.py):
- 71 automated tests covering: validation, CRUD, clone levels, rename,
export/import, active profile, isolation, alias collision, completion
- Full suite: 6760 passed, 0 new failures
Documentation:
- website/docs/user-guide/profiles.md: full user guide (12 sections)
- website/docs/reference/profile-commands.md: command reference (12 commands)
- website/docs/reference/faq.md: 6 profile FAQ entries
- website/sidebars.ts: navigation updated
2026-03-29 10:41:20 -07:00
|
|
|
# Acquire scoped lock to prevent duplicate sessions
|
|
|
|
|
try:
|
|
|
|
|
from gateway.status import acquire_scoped_lock
|
|
|
|
|
|
|
|
|
|
self._session_lock_identity = str(self._session_path)
|
|
|
|
|
acquired, existing = acquire_scoped_lock(
|
|
|
|
|
"whatsapp-session",
|
|
|
|
|
self._session_lock_identity,
|
|
|
|
|
metadata={"platform": self.platform.value},
|
|
|
|
|
)
|
|
|
|
|
if not acquired:
|
|
|
|
|
owner_pid = existing.get("pid") if isinstance(existing, dict) else None
|
|
|
|
|
message = (
|
|
|
|
|
"Another local Hermes gateway is already using this WhatsApp session"
|
|
|
|
|
+ (f" (PID {owner_pid})." if owner_pid else ".")
|
|
|
|
|
+ " Stop the other gateway before starting a second WhatsApp bridge."
|
|
|
|
|
)
|
|
|
|
|
logger.error("[%s] %s", self.name, message)
|
|
|
|
|
self._set_fatal_error("whatsapp_session_lock", message, retryable=False)
|
|
|
|
|
return False
|
|
|
|
|
except Exception as e:
|
|
|
|
|
logger.warning("[%s] Could not acquire session lock (non-fatal): %s", self.name, e)
|
|
|
|
|
|
2026-02-25 21:04:36 -08:00
|
|
|
# Auto-install npm dependencies if node_modules doesn't exist
|
|
|
|
|
bridge_dir = bridge_path.parent
|
|
|
|
|
if not (bridge_dir / "node_modules").exists():
|
|
|
|
|
print(f"[{self.name}] Installing WhatsApp bridge dependencies...")
|
|
|
|
|
try:
|
|
|
|
|
install_result = subprocess.run(
|
|
|
|
|
["npm", "install", "--silent"],
|
|
|
|
|
cwd=str(bridge_dir),
|
|
|
|
|
capture_output=True,
|
|
|
|
|
text=True,
|
|
|
|
|
timeout=60,
|
|
|
|
|
)
|
|
|
|
|
if install_result.returncode != 0:
|
|
|
|
|
print(f"[{self.name}] npm install failed: {install_result.stderr}")
|
|
|
|
|
return False
|
|
|
|
|
print(f"[{self.name}] Dependencies installed")
|
|
|
|
|
except Exception as e:
|
|
|
|
|
print(f"[{self.name}] Failed to install dependencies: {e}")
|
|
|
|
|
return False
|
|
|
|
|
|
2026-02-02 19:01:51 -08:00
|
|
|
try:
|
|
|
|
|
# Ensure session directory exists
|
|
|
|
|
self._session_path.mkdir(parents=True, exist_ok=True)
|
|
|
|
|
|
2026-03-20 09:37:48 -07:00
|
|
|
# Check if bridge is already running and connected
|
|
|
|
|
import aiohttp
|
|
|
|
|
import asyncio
|
|
|
|
|
try:
|
|
|
|
|
async with aiohttp.ClientSession() as session:
|
|
|
|
|
async with session.get(
|
|
|
|
|
f"http://127.0.0.1:{self._bridge_port}/health",
|
|
|
|
|
timeout=aiohttp.ClientTimeout(total=2)
|
|
|
|
|
) as resp:
|
|
|
|
|
if resp.status == 200:
|
|
|
|
|
data = await resp.json()
|
|
|
|
|
bridge_status = data.get("status", "unknown")
|
|
|
|
|
if bridge_status == "connected":
|
|
|
|
|
print(f"[{self.name}] Using existing bridge (status: {bridge_status})")
|
2026-03-21 09:38:52 -07:00
|
|
|
self._mark_connected()
|
2026-03-20 09:37:48 -07:00
|
|
|
self._bridge_process = None # Not managed by us
|
fix(whatsapp): reuse persistent aiohttp session across requests (#3818)
Replace per-request aiohttp.ClientSession() in every WhatsApp adapter
method with a single persistent self._http_session, matching the pattern
used by Mattermost, HomeAssistant, and SMS adapters.
Changes:
- Create self._http_session in connect(), close in disconnect()
- All bridge HTTP calls (send, edit, send-media, typing, get_chat_info,
poll_messages) now use the shared session
- Explicitly cancel _poll_task on disconnect() instead of relying
solely on self._running = False
- Health-check sessions in connect() remain ephemeral (persistent
session not yet created at that point)
- Remove per-method ImportError guards for aiohttp (always available
when gateway runs via [messaging] extras)
Salvaged from PR #1851 by Himess. The _poll_task storage was already
on main from PR #3267; this adds the disconnect cancellation and the
persistent session.
Tests: 4 new tests for session close, already-closed skip, poll task
cancellation, and done-task skip.
2026-03-29 16:25:20 -07:00
|
|
|
self._http_session = aiohttp.ClientSession()
|
2026-03-26 14:36:24 -07:00
|
|
|
self._poll_task = asyncio.create_task(self._poll_messages())
|
2026-03-20 09:37:48 -07:00
|
|
|
return True
|
|
|
|
|
else:
|
|
|
|
|
print(f"[{self.name}] Bridge found but not connected (status: {bridge_status}), restarting")
|
|
|
|
|
except Exception:
|
|
|
|
|
pass # Bridge not running, start a new one
|
|
|
|
|
|
2026-02-25 21:04:36 -08:00
|
|
|
# Kill any orphaned bridge from a previous gateway run
|
2026-03-05 17:13:14 +03:00
|
|
|
_kill_port_process(self._bridge_port)
|
2026-03-09 17:16:26 +03:00
|
|
|
await asyncio.sleep(1)
|
2026-02-25 21:04:36 -08:00
|
|
|
|
2026-03-04 04:58:21 -08:00
|
|
|
# Start the bridge process in its own process group.
|
|
|
|
|
# Route output to a log file so QR codes, errors, and reconnection
|
|
|
|
|
# messages are preserved for troubleshooting.
|
2026-03-02 17:51:33 -08:00
|
|
|
whatsapp_mode = os.getenv("WHATSAPP_MODE", "self-chat")
|
2026-03-04 04:58:21 -08:00
|
|
|
self._bridge_log = self._session_path.parent / "bridge.log"
|
|
|
|
|
bridge_log_fh = open(self._bridge_log, "a")
|
|
|
|
|
self._bridge_log_fh = bridge_log_fh
|
feat: OpenAI-compatible API server + WhatsApp configurable reply prefix (#1756)
* feat: OpenAI-compatible API server platform adapter
Salvaged from PR #956, updated for current main.
Adds an HTTP API server as a gateway platform adapter that exposes
hermes-agent via the OpenAI Chat Completions and Responses APIs.
Any OpenAI-compatible frontend (Open WebUI, LobeChat, LibreChat,
AnythingLLM, NextChat, ChatBox, etc.) can connect by pointing at
http://localhost:8642/v1.
Endpoints:
- POST /v1/chat/completions — stateless Chat Completions API
- POST /v1/responses — stateful Responses API with chaining
- GET /v1/responses/{id} — retrieve stored response
- DELETE /v1/responses/{id} — delete stored response
- GET /v1/models — list hermes-agent as available model
- GET /health — health check
Features:
- Real SSE streaming via stream_delta_callback (uses main's streaming)
- In-memory LRU response store for Responses API conversation chaining
- Named conversations via 'conversation' parameter
- Bearer token auth (optional, via API_SERVER_KEY)
- CORS support for browser-based frontends
- System prompt layering (frontend system messages on top of core)
- Real token usage tracking in responses
Integration points:
- Platform.API_SERVER in gateway/config.py
- _create_adapter() branch in gateway/run.py
- API_SERVER_* env vars in hermes_cli/config.py
- Env var overrides in gateway/config.py _apply_env_overrides()
Changes vs original PR #956:
- Removed streaming infrastructure (already on main via stream_consumer.py)
- Removed Telegram reply_to_mode (separate feature, not included)
- Updated _resolve_model() -> _resolve_gateway_model()
- Updated stream_callback -> stream_delta_callback
- Updated connect()/disconnect() to use _mark_connected()/_mark_disconnected()
- Adapted to current Platform enum (includes MATTERMOST, MATRIX, DINGTALK)
Tests: 72 new tests, all passing
Docs: API server guide, Open WebUI integration guide, env var reference
* feat(whatsapp): make reply prefix configurable via config.yaml
Reworked from PR #1764 (ifrederico) to use config.yaml instead of .env.
The WhatsApp bridge prepends a header to every outgoing message.
This was hardcoded to '⚕ *Hermes Agent*'. Users can now customize
or disable it via config.yaml:
whatsapp:
reply_prefix: '' # disable header
reply_prefix: '🤖 *My Bot*\n───\n' # custom prefix
How it works:
- load_gateway_config() reads whatsapp.reply_prefix from config.yaml
and stores it in PlatformConfig.extra['reply_prefix']
- WhatsAppAdapter reads it from config.extra at init
- When spawning bridge.js, the adapter passes it as
WHATSAPP_REPLY_PREFIX in the subprocess environment
- bridge.js handles undefined (default), empty (no header),
or custom values with \\n escape support
- Self-chat echo suppression uses the configured prefix
Also fixes _config_version: was 9 but ENV_VARS_BY_VERSION had a
key 10 (TAVILY_API_KEY), so existing users at v9 would never be
prompted for Tavily. Bumped to 10 to close the gap. Added a
regression test to prevent this from happening again.
Credit: ifrederico (PR #1764) for the bridge.js implementation
and the config version gap discovery.
---------
Co-authored-by: Test <test@test.com>
2026-03-17 10:44:37 -07:00
|
|
|
|
|
|
|
|
# Build bridge subprocess environment.
|
|
|
|
|
# Pass WHATSAPP_REPLY_PREFIX from config.yaml so the Node bridge
|
|
|
|
|
# can use it without the user needing to set a separate env var.
|
|
|
|
|
bridge_env = os.environ.copy()
|
|
|
|
|
if self._reply_prefix is not None:
|
|
|
|
|
bridge_env["WHATSAPP_REPLY_PREFIX"] = self._reply_prefix
|
|
|
|
|
|
2026-02-02 19:01:51 -08:00
|
|
|
self._bridge_process = subprocess.Popen(
|
|
|
|
|
[
|
|
|
|
|
"node",
|
|
|
|
|
str(bridge_path),
|
|
|
|
|
"--port", str(self._bridge_port),
|
|
|
|
|
"--session", str(self._session_path),
|
2026-03-02 17:51:33 -08:00
|
|
|
"--mode", whatsapp_mode,
|
2026-02-02 19:01:51 -08:00
|
|
|
],
|
2026-03-04 04:58:21 -08:00
|
|
|
stdout=bridge_log_fh,
|
|
|
|
|
stderr=bridge_log_fh,
|
2026-03-01 01:54:27 +03:00
|
|
|
preexec_fn=None if _IS_WINDOWS else os.setsid,
|
feat: OpenAI-compatible API server + WhatsApp configurable reply prefix (#1756)
* feat: OpenAI-compatible API server platform adapter
Salvaged from PR #956, updated for current main.
Adds an HTTP API server as a gateway platform adapter that exposes
hermes-agent via the OpenAI Chat Completions and Responses APIs.
Any OpenAI-compatible frontend (Open WebUI, LobeChat, LibreChat,
AnythingLLM, NextChat, ChatBox, etc.) can connect by pointing at
http://localhost:8642/v1.
Endpoints:
- POST /v1/chat/completions — stateless Chat Completions API
- POST /v1/responses — stateful Responses API with chaining
- GET /v1/responses/{id} — retrieve stored response
- DELETE /v1/responses/{id} — delete stored response
- GET /v1/models — list hermes-agent as available model
- GET /health — health check
Features:
- Real SSE streaming via stream_delta_callback (uses main's streaming)
- In-memory LRU response store for Responses API conversation chaining
- Named conversations via 'conversation' parameter
- Bearer token auth (optional, via API_SERVER_KEY)
- CORS support for browser-based frontends
- System prompt layering (frontend system messages on top of core)
- Real token usage tracking in responses
Integration points:
- Platform.API_SERVER in gateway/config.py
- _create_adapter() branch in gateway/run.py
- API_SERVER_* env vars in hermes_cli/config.py
- Env var overrides in gateway/config.py _apply_env_overrides()
Changes vs original PR #956:
- Removed streaming infrastructure (already on main via stream_consumer.py)
- Removed Telegram reply_to_mode (separate feature, not included)
- Updated _resolve_model() -> _resolve_gateway_model()
- Updated stream_callback -> stream_delta_callback
- Updated connect()/disconnect() to use _mark_connected()/_mark_disconnected()
- Adapted to current Platform enum (includes MATTERMOST, MATRIX, DINGTALK)
Tests: 72 new tests, all passing
Docs: API server guide, Open WebUI integration guide, env var reference
* feat(whatsapp): make reply prefix configurable via config.yaml
Reworked from PR #1764 (ifrederico) to use config.yaml instead of .env.
The WhatsApp bridge prepends a header to every outgoing message.
This was hardcoded to '⚕ *Hermes Agent*'. Users can now customize
or disable it via config.yaml:
whatsapp:
reply_prefix: '' # disable header
reply_prefix: '🤖 *My Bot*\n───\n' # custom prefix
How it works:
- load_gateway_config() reads whatsapp.reply_prefix from config.yaml
and stores it in PlatformConfig.extra['reply_prefix']
- WhatsAppAdapter reads it from config.extra at init
- When spawning bridge.js, the adapter passes it as
WHATSAPP_REPLY_PREFIX in the subprocess environment
- bridge.js handles undefined (default), empty (no header),
or custom values with \\n escape support
- Self-chat echo suppression uses the configured prefix
Also fixes _config_version: was 9 but ENV_VARS_BY_VERSION had a
key 10 (TAVILY_API_KEY), so existing users at v9 would never be
prompted for Tavily. Bumped to 10 to close the gap. Added a
regression test to prevent this from happening again.
Credit: ifrederico (PR #1764) for the bridge.js implementation
and the config version gap discovery.
---------
Co-authored-by: Test <test@test.com>
2026-03-17 10:44:37 -07:00
|
|
|
env=bridge_env,
|
2026-02-02 19:01:51 -08:00
|
|
|
)
|
|
|
|
|
|
2026-03-04 04:58:21 -08:00
|
|
|
# Wait for the bridge to connect to WhatsApp.
|
|
|
|
|
# Phase 1: wait for the HTTP server to come up (up to 15s).
|
|
|
|
|
# Phase 2: wait for WhatsApp status: connected (up to 15s more).
|
2026-02-25 21:04:36 -08:00
|
|
|
import aiohttp
|
2026-03-04 04:58:21 -08:00
|
|
|
http_ready = False
|
2026-03-04 19:11:48 +03:00
|
|
|
data = {}
|
2026-02-25 21:04:36 -08:00
|
|
|
for attempt in range(15):
|
|
|
|
|
await asyncio.sleep(1)
|
|
|
|
|
if self._bridge_process.poll() is not None:
|
|
|
|
|
print(f"[{self.name}] Bridge process died (exit code {self._bridge_process.returncode})")
|
2026-03-04 04:58:21 -08:00
|
|
|
print(f"[{self.name}] Check log: {self._bridge_log}")
|
2026-03-04 19:11:48 +03:00
|
|
|
self._close_bridge_log()
|
2026-02-25 21:04:36 -08:00
|
|
|
return False
|
|
|
|
|
try:
|
|
|
|
|
async with aiohttp.ClientSession() as session:
|
|
|
|
|
async with session.get(
|
2026-03-20 09:37:48 -07:00
|
|
|
f"http://127.0.0.1:{self._bridge_port}/health",
|
2026-02-25 21:04:36 -08:00
|
|
|
timeout=aiohttp.ClientTimeout(total=2)
|
|
|
|
|
) as resp:
|
|
|
|
|
if resp.status == 200:
|
2026-03-04 04:58:21 -08:00
|
|
|
http_ready = True
|
2026-02-25 21:04:36 -08:00
|
|
|
data = await resp.json()
|
2026-03-04 04:58:21 -08:00
|
|
|
if data.get("status") == "connected":
|
|
|
|
|
print(f"[{self.name}] Bridge ready (status: connected)")
|
|
|
|
|
break
|
2026-02-25 21:04:36 -08:00
|
|
|
except Exception:
|
|
|
|
|
continue
|
2026-03-04 19:11:48 +03:00
|
|
|
|
2026-03-04 04:58:21 -08:00
|
|
|
if not http_ready:
|
|
|
|
|
print(f"[{self.name}] Bridge HTTP server did not start in 15s")
|
|
|
|
|
print(f"[{self.name}] Check log: {self._bridge_log}")
|
2026-03-04 19:11:48 +03:00
|
|
|
self._close_bridge_log()
|
2026-02-02 19:01:51 -08:00
|
|
|
return False
|
|
|
|
|
|
2026-03-04 04:58:21 -08:00
|
|
|
# Phase 2: HTTP is up but WhatsApp may still be connecting.
|
|
|
|
|
# Give it more time to authenticate with saved credentials.
|
|
|
|
|
if data.get("status") != "connected":
|
|
|
|
|
print(f"[{self.name}] Bridge HTTP ready, waiting for WhatsApp connection...")
|
|
|
|
|
for attempt in range(15):
|
|
|
|
|
await asyncio.sleep(1)
|
|
|
|
|
if self._bridge_process.poll() is not None:
|
|
|
|
|
print(f"[{self.name}] Bridge process died during connection")
|
|
|
|
|
print(f"[{self.name}] Check log: {self._bridge_log}")
|
2026-03-04 19:11:48 +03:00
|
|
|
self._close_bridge_log()
|
2026-03-04 04:58:21 -08:00
|
|
|
return False
|
|
|
|
|
try:
|
|
|
|
|
async with aiohttp.ClientSession() as session:
|
|
|
|
|
async with session.get(
|
2026-03-20 09:37:48 -07:00
|
|
|
f"http://127.0.0.1:{self._bridge_port}/health",
|
2026-03-04 04:58:21 -08:00
|
|
|
timeout=aiohttp.ClientTimeout(total=2)
|
|
|
|
|
) as resp:
|
|
|
|
|
if resp.status == 200:
|
|
|
|
|
data = await resp.json()
|
|
|
|
|
if data.get("status") == "connected":
|
|
|
|
|
print(f"[{self.name}] Bridge ready (status: connected)")
|
|
|
|
|
break
|
|
|
|
|
except Exception:
|
|
|
|
|
continue
|
|
|
|
|
else:
|
|
|
|
|
# Still not connected — warn but proceed (bridge may
|
|
|
|
|
# auto-reconnect later, e.g. after a code 515 restart).
|
|
|
|
|
print(f"[{self.name}] ⚠ WhatsApp not connected after 30s")
|
|
|
|
|
print(f"[{self.name}] Bridge log: {self._bridge_log}")
|
|
|
|
|
print(f"[{self.name}] If session expired, re-pair: hermes whatsapp")
|
|
|
|
|
|
fix(whatsapp): reuse persistent aiohttp session across requests (#3818)
Replace per-request aiohttp.ClientSession() in every WhatsApp adapter
method with a single persistent self._http_session, matching the pattern
used by Mattermost, HomeAssistant, and SMS adapters.
Changes:
- Create self._http_session in connect(), close in disconnect()
- All bridge HTTP calls (send, edit, send-media, typing, get_chat_info,
poll_messages) now use the shared session
- Explicitly cancel _poll_task on disconnect() instead of relying
solely on self._running = False
- Health-check sessions in connect() remain ephemeral (persistent
session not yet created at that point)
- Remove per-method ImportError guards for aiohttp (always available
when gateway runs via [messaging] extras)
Salvaged from PR #1851 by Himess. The _poll_task storage was already
on main from PR #3267; this adds the disconnect cancellation and the
persistent session.
Tests: 4 new tests for session close, already-closed skip, poll task
cancellation, and done-task skip.
2026-03-29 16:25:20 -07:00
|
|
|
# Create a persistent HTTP session for all bridge communication
|
|
|
|
|
self._http_session = aiohttp.ClientSession()
|
|
|
|
|
|
2026-02-02 19:01:51 -08:00
|
|
|
# Start message polling task
|
2026-03-26 14:36:24 -07:00
|
|
|
self._poll_task = asyncio.create_task(self._poll_messages())
|
2026-02-02 19:01:51 -08:00
|
|
|
|
2026-03-21 09:38:52 -07:00
|
|
|
self._mark_connected()
|
2026-02-02 19:01:51 -08:00
|
|
|
print(f"[{self.name}] Bridge started on port {self._bridge_port}")
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
except Exception as e:
|
feat: add profiles — run multiple isolated Hermes instances (#3681)
Each profile is a fully independent HERMES_HOME with its own config,
API keys, memory, sessions, skills, gateway, cron, and state.db.
Core module: hermes_cli/profiles.py (~900 lines)
- Profile CRUD: create, delete, list, show, rename
- Three clone levels: blank, --clone (config), --clone-all (everything)
- Export/import: tar.gz archive for backup and migration
- Wrapper alias scripts (~/.local/bin/<name>)
- Collision detection for alias names
- Sticky default via ~/.hermes/active_profile
- Skill seeding via subprocess (handles module-level caching)
- Auto-stop gateway on delete with disable-before-stop for services
- Tab completion generation for bash and zsh
CLI integration (hermes_cli/main.py):
- _apply_profile_override(): pre-import -p/--profile flag + sticky default
- Full 'hermes profile' subcommand: list, use, create, delete, show,
alias, rename, export, import
- 'hermes completion bash/zsh' command
- Multi-profile skill sync in hermes update
Display (cli.py, banner.py, gateway/run.py):
- CLI prompt: 'coder ❯' when using a non-default profile
- Banner shows profile name
- Gateway startup log includes profile name
Gateway safety:
- Token locks: Discord, Slack, WhatsApp, Signal (extends Telegram pattern)
- Port conflict detection: API server, webhook adapter
Diagnostics (hermes_cli/doctor.py):
- Profile health section: lists profiles, checks config, .env, aliases
- Orphan alias detection: warns when wrapper points to deleted profile
Tests (tests/hermes_cli/test_profiles.py):
- 71 automated tests covering: validation, CRUD, clone levels, rename,
export/import, active profile, isolation, alias collision, completion
- Full suite: 6760 passed, 0 new failures
Documentation:
- website/docs/user-guide/profiles.md: full user guide (12 sections)
- website/docs/reference/profile-commands.md: command reference (12 commands)
- website/docs/reference/faq.md: 6 profile FAQ entries
- website/sidebars.ts: navigation updated
2026-03-29 10:41:20 -07:00
|
|
|
if self._session_lock_identity:
|
|
|
|
|
try:
|
|
|
|
|
from gateway.status import release_scoped_lock
|
|
|
|
|
release_scoped_lock("whatsapp-session", self._session_lock_identity)
|
|
|
|
|
except Exception:
|
|
|
|
|
pass
|
2026-02-25 21:04:36 -08:00
|
|
|
logger.error("[%s] Failed to start bridge: %s", self.name, e, exc_info=True)
|
2026-03-04 19:11:48 +03:00
|
|
|
self._close_bridge_log()
|
2026-02-02 19:01:51 -08:00
|
|
|
return False
|
|
|
|
|
|
2026-03-04 19:11:48 +03:00
|
|
|
def _close_bridge_log(self) -> None:
|
|
|
|
|
"""Close the bridge log file handle if open."""
|
|
|
|
|
if self._bridge_log_fh:
|
|
|
|
|
try:
|
|
|
|
|
self._bridge_log_fh.close()
|
|
|
|
|
except Exception:
|
|
|
|
|
pass
|
|
|
|
|
self._bridge_log_fh = None
|
|
|
|
|
|
2026-03-21 09:38:52 -07:00
|
|
|
async def _check_managed_bridge_exit(self) -> Optional[str]:
|
|
|
|
|
"""Return a fatal error message if the managed bridge child exited."""
|
|
|
|
|
if self._bridge_process is None:
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
returncode = self._bridge_process.poll()
|
|
|
|
|
if returncode is None:
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
message = f"WhatsApp bridge process exited unexpectedly (code {returncode})."
|
|
|
|
|
if not self.has_fatal_error:
|
|
|
|
|
logger.error("[%s] %s", self.name, message)
|
|
|
|
|
self._set_fatal_error("whatsapp_bridge_exited", message, retryable=True)
|
|
|
|
|
self._close_bridge_log()
|
|
|
|
|
await self._notify_fatal_error()
|
|
|
|
|
return self.fatal_error_message or message
|
|
|
|
|
|
2026-02-02 19:01:51 -08:00
|
|
|
async def disconnect(self) -> None:
|
2026-02-25 21:04:36 -08:00
|
|
|
"""Stop the WhatsApp bridge and clean up any orphaned processes."""
|
2026-02-02 19:01:51 -08:00
|
|
|
if self._bridge_process:
|
|
|
|
|
try:
|
2026-02-25 21:04:36 -08:00
|
|
|
# Kill the entire process group so child node processes die too
|
|
|
|
|
import signal
|
|
|
|
|
try:
|
2026-03-01 01:54:27 +03:00
|
|
|
if _IS_WINDOWS:
|
|
|
|
|
self._bridge_process.terminate()
|
|
|
|
|
else:
|
|
|
|
|
os.killpg(os.getpgid(self._bridge_process.pid), signal.SIGTERM)
|
2026-02-25 21:04:36 -08:00
|
|
|
except (ProcessLookupError, PermissionError):
|
|
|
|
|
self._bridge_process.terminate()
|
2026-02-02 19:01:51 -08:00
|
|
|
await asyncio.sleep(1)
|
|
|
|
|
if self._bridge_process.poll() is None:
|
2026-02-25 21:04:36 -08:00
|
|
|
try:
|
2026-03-01 01:54:27 +03:00
|
|
|
if _IS_WINDOWS:
|
|
|
|
|
self._bridge_process.kill()
|
|
|
|
|
else:
|
|
|
|
|
os.killpg(os.getpgid(self._bridge_process.pid), signal.SIGKILL)
|
2026-02-25 21:04:36 -08:00
|
|
|
except (ProcessLookupError, PermissionError):
|
|
|
|
|
self._bridge_process.kill()
|
2026-02-02 19:01:51 -08:00
|
|
|
except Exception as e:
|
|
|
|
|
print(f"[{self.name}] Error stopping bridge: {e}")
|
2026-03-20 09:37:48 -07:00
|
|
|
else:
|
|
|
|
|
# Bridge was not started by us, don't kill it
|
|
|
|
|
print(f"[{self.name}] Disconnecting (external bridge left running)")
|
fix(whatsapp): reuse persistent aiohttp session across requests (#3818)
Replace per-request aiohttp.ClientSession() in every WhatsApp adapter
method with a single persistent self._http_session, matching the pattern
used by Mattermost, HomeAssistant, and SMS adapters.
Changes:
- Create self._http_session in connect(), close in disconnect()
- All bridge HTTP calls (send, edit, send-media, typing, get_chat_info,
poll_messages) now use the shared session
- Explicitly cancel _poll_task on disconnect() instead of relying
solely on self._running = False
- Health-check sessions in connect() remain ephemeral (persistent
session not yet created at that point)
- Remove per-method ImportError guards for aiohttp (always available
when gateway runs via [messaging] extras)
Salvaged from PR #1851 by Himess. The _poll_task storage was already
on main from PR #3267; this adds the disconnect cancellation and the
persistent session.
Tests: 4 new tests for session close, already-closed skip, poll task
cancellation, and done-task skip.
2026-03-29 16:25:20 -07:00
|
|
|
|
|
|
|
|
# Cancel the poll task explicitly
|
|
|
|
|
if self._poll_task and not self._poll_task.done():
|
|
|
|
|
self._poll_task.cancel()
|
|
|
|
|
try:
|
|
|
|
|
await self._poll_task
|
|
|
|
|
except (asyncio.CancelledError, Exception):
|
|
|
|
|
pass
|
|
|
|
|
self._poll_task = None
|
|
|
|
|
|
|
|
|
|
# Close the persistent HTTP session
|
|
|
|
|
if self._http_session and not self._http_session.closed:
|
|
|
|
|
await self._http_session.close()
|
|
|
|
|
self._http_session = None
|
|
|
|
|
|
feat: add profiles — run multiple isolated Hermes instances (#3681)
Each profile is a fully independent HERMES_HOME with its own config,
API keys, memory, sessions, skills, gateway, cron, and state.db.
Core module: hermes_cli/profiles.py (~900 lines)
- Profile CRUD: create, delete, list, show, rename
- Three clone levels: blank, --clone (config), --clone-all (everything)
- Export/import: tar.gz archive for backup and migration
- Wrapper alias scripts (~/.local/bin/<name>)
- Collision detection for alias names
- Sticky default via ~/.hermes/active_profile
- Skill seeding via subprocess (handles module-level caching)
- Auto-stop gateway on delete with disable-before-stop for services
- Tab completion generation for bash and zsh
CLI integration (hermes_cli/main.py):
- _apply_profile_override(): pre-import -p/--profile flag + sticky default
- Full 'hermes profile' subcommand: list, use, create, delete, show,
alias, rename, export, import
- 'hermes completion bash/zsh' command
- Multi-profile skill sync in hermes update
Display (cli.py, banner.py, gateway/run.py):
- CLI prompt: 'coder ❯' when using a non-default profile
- Banner shows profile name
- Gateway startup log includes profile name
Gateway safety:
- Token locks: Discord, Slack, WhatsApp, Signal (extends Telegram pattern)
- Port conflict detection: API server, webhook adapter
Diagnostics (hermes_cli/doctor.py):
- Profile health section: lists profiles, checks config, .env, aliases
- Orphan alias detection: warns when wrapper points to deleted profile
Tests (tests/hermes_cli/test_profiles.py):
- 71 automated tests covering: validation, CRUD, clone levels, rename,
export/import, active profile, isolation, alias collision, completion
- Full suite: 6760 passed, 0 new failures
Documentation:
- website/docs/user-guide/profiles.md: full user guide (12 sections)
- website/docs/reference/profile-commands.md: command reference (12 commands)
- website/docs/reference/faq.md: 6 profile FAQ entries
- website/sidebars.ts: navigation updated
2026-03-29 10:41:20 -07:00
|
|
|
if self._session_lock_identity:
|
|
|
|
|
try:
|
|
|
|
|
from gateway.status import release_scoped_lock
|
|
|
|
|
release_scoped_lock("whatsapp-session", self._session_lock_identity)
|
|
|
|
|
except Exception as e:
|
|
|
|
|
logger.warning("[%s] Error releasing WhatsApp session lock: %s", self.name, e, exc_info=True)
|
|
|
|
|
|
2026-03-21 09:38:52 -07:00
|
|
|
self._mark_disconnected()
|
2026-02-02 19:01:51 -08:00
|
|
|
self._bridge_process = None
|
2026-03-04 19:11:48 +03:00
|
|
|
self._close_bridge_log()
|
feat: add profiles — run multiple isolated Hermes instances (#3681)
Each profile is a fully independent HERMES_HOME with its own config,
API keys, memory, sessions, skills, gateway, cron, and state.db.
Core module: hermes_cli/profiles.py (~900 lines)
- Profile CRUD: create, delete, list, show, rename
- Three clone levels: blank, --clone (config), --clone-all (everything)
- Export/import: tar.gz archive for backup and migration
- Wrapper alias scripts (~/.local/bin/<name>)
- Collision detection for alias names
- Sticky default via ~/.hermes/active_profile
- Skill seeding via subprocess (handles module-level caching)
- Auto-stop gateway on delete with disable-before-stop for services
- Tab completion generation for bash and zsh
CLI integration (hermes_cli/main.py):
- _apply_profile_override(): pre-import -p/--profile flag + sticky default
- Full 'hermes profile' subcommand: list, use, create, delete, show,
alias, rename, export, import
- 'hermes completion bash/zsh' command
- Multi-profile skill sync in hermes update
Display (cli.py, banner.py, gateway/run.py):
- CLI prompt: 'coder ❯' when using a non-default profile
- Banner shows profile name
- Gateway startup log includes profile name
Gateway safety:
- Token locks: Discord, Slack, WhatsApp, Signal (extends Telegram pattern)
- Port conflict detection: API server, webhook adapter
Diagnostics (hermes_cli/doctor.py):
- Profile health section: lists profiles, checks config, .env, aliases
- Orphan alias detection: warns when wrapper points to deleted profile
Tests (tests/hermes_cli/test_profiles.py):
- 71 automated tests covering: validation, CRUD, clone levels, rename,
export/import, active profile, isolation, alias collision, completion
- Full suite: 6760 passed, 0 new failures
Documentation:
- website/docs/user-guide/profiles.md: full user guide (12 sections)
- website/docs/reference/profile-commands.md: command reference (12 commands)
- website/docs/reference/faq.md: 6 profile FAQ entries
- website/sidebars.ts: navigation updated
2026-03-29 10:41:20 -07:00
|
|
|
self._session_lock_identity = None
|
2026-02-02 19:01:51 -08:00
|
|
|
print(f"[{self.name}] Disconnected")
|
|
|
|
|
|
|
|
|
|
async def send(
|
|
|
|
|
self,
|
|
|
|
|
chat_id: str,
|
|
|
|
|
content: str,
|
|
|
|
|
reply_to: Optional[str] = None,
|
|
|
|
|
metadata: Optional[Dict[str, Any]] = None
|
|
|
|
|
) -> SendResult:
|
|
|
|
|
"""Send a message via the WhatsApp bridge."""
|
fix(whatsapp): reuse persistent aiohttp session across requests (#3818)
Replace per-request aiohttp.ClientSession() in every WhatsApp adapter
method with a single persistent self._http_session, matching the pattern
used by Mattermost, HomeAssistant, and SMS adapters.
Changes:
- Create self._http_session in connect(), close in disconnect()
- All bridge HTTP calls (send, edit, send-media, typing, get_chat_info,
poll_messages) now use the shared session
- Explicitly cancel _poll_task on disconnect() instead of relying
solely on self._running = False
- Health-check sessions in connect() remain ephemeral (persistent
session not yet created at that point)
- Remove per-method ImportError guards for aiohttp (always available
when gateway runs via [messaging] extras)
Salvaged from PR #1851 by Himess. The _poll_task storage was already
on main from PR #3267; this adds the disconnect cancellation and the
persistent session.
Tests: 4 new tests for session close, already-closed skip, poll task
cancellation, and done-task skip.
2026-03-29 16:25:20 -07:00
|
|
|
if not self._running or not self._http_session:
|
2026-02-02 19:01:51 -08:00
|
|
|
return SendResult(success=False, error="Not connected")
|
2026-03-21 09:38:52 -07:00
|
|
|
bridge_exit = await self._check_managed_bridge_exit()
|
|
|
|
|
if bridge_exit:
|
|
|
|
|
return SendResult(success=False, error=bridge_exit)
|
2026-02-02 19:01:51 -08:00
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
import aiohttp
|
fix(whatsapp): reuse persistent aiohttp session across requests (#3818)
Replace per-request aiohttp.ClientSession() in every WhatsApp adapter
method with a single persistent self._http_session, matching the pattern
used by Mattermost, HomeAssistant, and SMS adapters.
Changes:
- Create self._http_session in connect(), close in disconnect()
- All bridge HTTP calls (send, edit, send-media, typing, get_chat_info,
poll_messages) now use the shared session
- Explicitly cancel _poll_task on disconnect() instead of relying
solely on self._running = False
- Health-check sessions in connect() remain ephemeral (persistent
session not yet created at that point)
- Remove per-method ImportError guards for aiohttp (always available
when gateway runs via [messaging] extras)
Salvaged from PR #1851 by Himess. The _poll_task storage was already
on main from PR #3267; this adds the disconnect cancellation and the
persistent session.
Tests: 4 new tests for session close, already-closed skip, poll task
cancellation, and done-task skip.
2026-03-29 16:25:20 -07:00
|
|
|
|
|
|
|
|
payload = {
|
|
|
|
|
"chatId": chat_id,
|
|
|
|
|
"message": content,
|
|
|
|
|
}
|
|
|
|
|
if reply_to:
|
|
|
|
|
payload["replyTo"] = reply_to
|
2026-02-02 19:01:51 -08:00
|
|
|
|
fix(whatsapp): reuse persistent aiohttp session across requests (#3818)
Replace per-request aiohttp.ClientSession() in every WhatsApp adapter
method with a single persistent self._http_session, matching the pattern
used by Mattermost, HomeAssistant, and SMS adapters.
Changes:
- Create self._http_session in connect(), close in disconnect()
- All bridge HTTP calls (send, edit, send-media, typing, get_chat_info,
poll_messages) now use the shared session
- Explicitly cancel _poll_task on disconnect() instead of relying
solely on self._running = False
- Health-check sessions in connect() remain ephemeral (persistent
session not yet created at that point)
- Remove per-method ImportError guards for aiohttp (always available
when gateway runs via [messaging] extras)
Salvaged from PR #1851 by Himess. The _poll_task storage was already
on main from PR #3267; this adds the disconnect cancellation and the
persistent session.
Tests: 4 new tests for session close, already-closed skip, poll task
cancellation, and done-task skip.
2026-03-29 16:25:20 -07:00
|
|
|
async with self._http_session.post(
|
|
|
|
|
f"http://127.0.0.1:{self._bridge_port}/send",
|
|
|
|
|
json=payload,
|
|
|
|
|
timeout=aiohttp.ClientTimeout(total=30)
|
|
|
|
|
) as resp:
|
|
|
|
|
if resp.status == 200:
|
|
|
|
|
data = await resp.json()
|
|
|
|
|
return SendResult(
|
|
|
|
|
success=True,
|
|
|
|
|
message_id=data.get("messageId"),
|
|
|
|
|
raw_response=data
|
|
|
|
|
)
|
|
|
|
|
else:
|
|
|
|
|
error = await resp.text()
|
|
|
|
|
return SendResult(success=False, error=error)
|
2026-02-02 19:01:51 -08:00
|
|
|
except Exception as e:
|
|
|
|
|
return SendResult(success=False, error=str(e))
|
2026-03-02 14:13:35 -03:00
|
|
|
|
|
|
|
|
async def edit_message(
|
|
|
|
|
self,
|
|
|
|
|
chat_id: str,
|
|
|
|
|
message_id: str,
|
|
|
|
|
content: str,
|
|
|
|
|
) -> SendResult:
|
|
|
|
|
"""Edit a previously sent message via the WhatsApp bridge."""
|
fix(whatsapp): reuse persistent aiohttp session across requests (#3818)
Replace per-request aiohttp.ClientSession() in every WhatsApp adapter
method with a single persistent self._http_session, matching the pattern
used by Mattermost, HomeAssistant, and SMS adapters.
Changes:
- Create self._http_session in connect(), close in disconnect()
- All bridge HTTP calls (send, edit, send-media, typing, get_chat_info,
poll_messages) now use the shared session
- Explicitly cancel _poll_task on disconnect() instead of relying
solely on self._running = False
- Health-check sessions in connect() remain ephemeral (persistent
session not yet created at that point)
- Remove per-method ImportError guards for aiohttp (always available
when gateway runs via [messaging] extras)
Salvaged from PR #1851 by Himess. The _poll_task storage was already
on main from PR #3267; this adds the disconnect cancellation and the
persistent session.
Tests: 4 new tests for session close, already-closed skip, poll task
cancellation, and done-task skip.
2026-03-29 16:25:20 -07:00
|
|
|
if not self._running or not self._http_session:
|
2026-03-02 14:13:35 -03:00
|
|
|
return SendResult(success=False, error="Not connected")
|
2026-03-21 09:38:52 -07:00
|
|
|
bridge_exit = await self._check_managed_bridge_exit()
|
|
|
|
|
if bridge_exit:
|
|
|
|
|
return SendResult(success=False, error=bridge_exit)
|
2026-03-02 14:13:35 -03:00
|
|
|
try:
|
|
|
|
|
import aiohttp
|
fix(whatsapp): reuse persistent aiohttp session across requests (#3818)
Replace per-request aiohttp.ClientSession() in every WhatsApp adapter
method with a single persistent self._http_session, matching the pattern
used by Mattermost, HomeAssistant, and SMS adapters.
Changes:
- Create self._http_session in connect(), close in disconnect()
- All bridge HTTP calls (send, edit, send-media, typing, get_chat_info,
poll_messages) now use the shared session
- Explicitly cancel _poll_task on disconnect() instead of relying
solely on self._running = False
- Health-check sessions in connect() remain ephemeral (persistent
session not yet created at that point)
- Remove per-method ImportError guards for aiohttp (always available
when gateway runs via [messaging] extras)
Salvaged from PR #1851 by Himess. The _poll_task storage was already
on main from PR #3267; this adds the disconnect cancellation and the
persistent session.
Tests: 4 new tests for session close, already-closed skip, poll task
cancellation, and done-task skip.
2026-03-29 16:25:20 -07:00
|
|
|
async with self._http_session.post(
|
|
|
|
|
f"http://127.0.0.1:{self._bridge_port}/edit",
|
|
|
|
|
json={
|
|
|
|
|
"chatId": chat_id,
|
|
|
|
|
"messageId": message_id,
|
|
|
|
|
"message": content,
|
|
|
|
|
},
|
|
|
|
|
timeout=aiohttp.ClientTimeout(total=15)
|
|
|
|
|
) as resp:
|
|
|
|
|
if resp.status == 200:
|
|
|
|
|
return SendResult(success=True, message_id=message_id)
|
|
|
|
|
else:
|
|
|
|
|
error = await resp.text()
|
|
|
|
|
return SendResult(success=False, error=error)
|
2026-03-02 14:13:35 -03:00
|
|
|
except Exception as e:
|
|
|
|
|
return SendResult(success=False, error=str(e))
|
|
|
|
|
|
2026-03-02 16:34:49 -03:00
|
|
|
async def _send_media_to_bridge(
|
|
|
|
|
self,
|
|
|
|
|
chat_id: str,
|
|
|
|
|
file_path: str,
|
|
|
|
|
media_type: str,
|
|
|
|
|
caption: Optional[str] = None,
|
|
|
|
|
file_name: Optional[str] = None,
|
|
|
|
|
) -> SendResult:
|
|
|
|
|
"""Send any media file via bridge /send-media endpoint."""
|
fix(whatsapp): reuse persistent aiohttp session across requests (#3818)
Replace per-request aiohttp.ClientSession() in every WhatsApp adapter
method with a single persistent self._http_session, matching the pattern
used by Mattermost, HomeAssistant, and SMS adapters.
Changes:
- Create self._http_session in connect(), close in disconnect()
- All bridge HTTP calls (send, edit, send-media, typing, get_chat_info,
poll_messages) now use the shared session
- Explicitly cancel _poll_task on disconnect() instead of relying
solely on self._running = False
- Health-check sessions in connect() remain ephemeral (persistent
session not yet created at that point)
- Remove per-method ImportError guards for aiohttp (always available
when gateway runs via [messaging] extras)
Salvaged from PR #1851 by Himess. The _poll_task storage was already
on main from PR #3267; this adds the disconnect cancellation and the
persistent session.
Tests: 4 new tests for session close, already-closed skip, poll task
cancellation, and done-task skip.
2026-03-29 16:25:20 -07:00
|
|
|
if not self._running or not self._http_session:
|
2026-03-02 16:34:49 -03:00
|
|
|
return SendResult(success=False, error="Not connected")
|
2026-03-21 09:38:52 -07:00
|
|
|
bridge_exit = await self._check_managed_bridge_exit()
|
|
|
|
|
if bridge_exit:
|
|
|
|
|
return SendResult(success=False, error=bridge_exit)
|
2026-03-02 16:34:49 -03:00
|
|
|
try:
|
|
|
|
|
import aiohttp
|
|
|
|
|
|
|
|
|
|
if not os.path.exists(file_path):
|
|
|
|
|
return SendResult(success=False, error=f"File not found: {file_path}")
|
|
|
|
|
|
|
|
|
|
payload: Dict[str, Any] = {
|
|
|
|
|
"chatId": chat_id,
|
|
|
|
|
"filePath": file_path,
|
|
|
|
|
"mediaType": media_type,
|
|
|
|
|
}
|
|
|
|
|
if caption:
|
|
|
|
|
payload["caption"] = caption
|
|
|
|
|
if file_name:
|
|
|
|
|
payload["fileName"] = file_name
|
|
|
|
|
|
fix(whatsapp): reuse persistent aiohttp session across requests (#3818)
Replace per-request aiohttp.ClientSession() in every WhatsApp adapter
method with a single persistent self._http_session, matching the pattern
used by Mattermost, HomeAssistant, and SMS adapters.
Changes:
- Create self._http_session in connect(), close in disconnect()
- All bridge HTTP calls (send, edit, send-media, typing, get_chat_info,
poll_messages) now use the shared session
- Explicitly cancel _poll_task on disconnect() instead of relying
solely on self._running = False
- Health-check sessions in connect() remain ephemeral (persistent
session not yet created at that point)
- Remove per-method ImportError guards for aiohttp (always available
when gateway runs via [messaging] extras)
Salvaged from PR #1851 by Himess. The _poll_task storage was already
on main from PR #3267; this adds the disconnect cancellation and the
persistent session.
Tests: 4 new tests for session close, already-closed skip, poll task
cancellation, and done-task skip.
2026-03-29 16:25:20 -07:00
|
|
|
async with self._http_session.post(
|
|
|
|
|
f"http://127.0.0.1:{self._bridge_port}/send-media",
|
|
|
|
|
json=payload,
|
|
|
|
|
timeout=aiohttp.ClientTimeout(total=120),
|
|
|
|
|
) as resp:
|
|
|
|
|
if resp.status == 200:
|
|
|
|
|
data = await resp.json()
|
|
|
|
|
return SendResult(
|
|
|
|
|
success=True,
|
|
|
|
|
message_id=data.get("messageId"),
|
|
|
|
|
raw_response=data,
|
|
|
|
|
)
|
|
|
|
|
else:
|
|
|
|
|
error = await resp.text()
|
|
|
|
|
return SendResult(success=False, error=error)
|
2026-03-02 16:34:49 -03:00
|
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
return SendResult(success=False, error=str(e))
|
|
|
|
|
|
|
|
|
|
async def send_image(
|
|
|
|
|
self,
|
|
|
|
|
chat_id: str,
|
|
|
|
|
image_url: str,
|
|
|
|
|
caption: Optional[str] = None,
|
|
|
|
|
reply_to: Optional[str] = None,
|
|
|
|
|
) -> SendResult:
|
|
|
|
|
"""Download image URL to cache, send natively via bridge."""
|
|
|
|
|
try:
|
|
|
|
|
local_path = await cache_image_from_url(image_url)
|
|
|
|
|
return await self._send_media_to_bridge(chat_id, local_path, "image", caption)
|
|
|
|
|
except Exception:
|
|
|
|
|
return await super().send_image(chat_id, image_url, caption, reply_to)
|
|
|
|
|
|
|
|
|
|
async def send_image_file(
|
|
|
|
|
self,
|
|
|
|
|
chat_id: str,
|
|
|
|
|
image_path: str,
|
|
|
|
|
caption: Optional[str] = None,
|
|
|
|
|
reply_to: Optional[str] = None,
|
2026-03-28 13:28:04 -07:00
|
|
|
**kwargs,
|
2026-03-02 16:34:49 -03:00
|
|
|
) -> SendResult:
|
|
|
|
|
"""Send a local image file natively via bridge."""
|
|
|
|
|
return await self._send_media_to_bridge(chat_id, image_path, "image", caption)
|
|
|
|
|
|
|
|
|
|
async def send_video(
|
|
|
|
|
self,
|
|
|
|
|
chat_id: str,
|
|
|
|
|
video_path: str,
|
|
|
|
|
caption: Optional[str] = None,
|
|
|
|
|
reply_to: Optional[str] = None,
|
2026-03-28 13:28:04 -07:00
|
|
|
**kwargs,
|
2026-03-02 16:34:49 -03:00
|
|
|
) -> SendResult:
|
|
|
|
|
"""Send a video natively via bridge — plays inline in WhatsApp."""
|
|
|
|
|
return await self._send_media_to_bridge(chat_id, video_path, "video", caption)
|
|
|
|
|
|
|
|
|
|
async def send_document(
|
|
|
|
|
self,
|
|
|
|
|
chat_id: str,
|
|
|
|
|
file_path: str,
|
|
|
|
|
caption: Optional[str] = None,
|
|
|
|
|
file_name: Optional[str] = None,
|
|
|
|
|
reply_to: Optional[str] = None,
|
2026-03-28 13:28:04 -07:00
|
|
|
**kwargs,
|
2026-03-02 16:34:49 -03:00
|
|
|
) -> SendResult:
|
|
|
|
|
"""Send a document/file as a downloadable attachment via bridge."""
|
|
|
|
|
return await self._send_media_to_bridge(
|
|
|
|
|
chat_id, file_path, "document", caption,
|
|
|
|
|
file_name or os.path.basename(file_path),
|
|
|
|
|
)
|
|
|
|
|
|
2026-03-10 06:26:16 -07:00
|
|
|
async def send_typing(self, chat_id: str, metadata=None) -> None:
|
2026-02-02 19:01:51 -08:00
|
|
|
"""Send typing indicator via bridge."""
|
fix(whatsapp): reuse persistent aiohttp session across requests (#3818)
Replace per-request aiohttp.ClientSession() in every WhatsApp adapter
method with a single persistent self._http_session, matching the pattern
used by Mattermost, HomeAssistant, and SMS adapters.
Changes:
- Create self._http_session in connect(), close in disconnect()
- All bridge HTTP calls (send, edit, send-media, typing, get_chat_info,
poll_messages) now use the shared session
- Explicitly cancel _poll_task on disconnect() instead of relying
solely on self._running = False
- Health-check sessions in connect() remain ephemeral (persistent
session not yet created at that point)
- Remove per-method ImportError guards for aiohttp (always available
when gateway runs via [messaging] extras)
Salvaged from PR #1851 by Himess. The _poll_task storage was already
on main from PR #3267; this adds the disconnect cancellation and the
persistent session.
Tests: 4 new tests for session close, already-closed skip, poll task
cancellation, and done-task skip.
2026-03-29 16:25:20 -07:00
|
|
|
if not self._running or not self._http_session:
|
2026-02-02 19:01:51 -08:00
|
|
|
return
|
2026-03-21 09:38:52 -07:00
|
|
|
if await self._check_managed_bridge_exit():
|
|
|
|
|
return
|
2026-02-02 19:01:51 -08:00
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
import aiohttp
|
fix(whatsapp): reuse persistent aiohttp session across requests (#3818)
Replace per-request aiohttp.ClientSession() in every WhatsApp adapter
method with a single persistent self._http_session, matching the pattern
used by Mattermost, HomeAssistant, and SMS adapters.
Changes:
- Create self._http_session in connect(), close in disconnect()
- All bridge HTTP calls (send, edit, send-media, typing, get_chat_info,
poll_messages) now use the shared session
- Explicitly cancel _poll_task on disconnect() instead of relying
solely on self._running = False
- Health-check sessions in connect() remain ephemeral (persistent
session not yet created at that point)
- Remove per-method ImportError guards for aiohttp (always available
when gateway runs via [messaging] extras)
Salvaged from PR #1851 by Himess. The _poll_task storage was already
on main from PR #3267; this adds the disconnect cancellation and the
persistent session.
Tests: 4 new tests for session close, already-closed skip, poll task
cancellation, and done-task skip.
2026-03-29 16:25:20 -07:00
|
|
|
|
|
|
|
|
await self._http_session.post(
|
|
|
|
|
f"http://127.0.0.1:{self._bridge_port}/typing",
|
|
|
|
|
json={"chatId": chat_id},
|
|
|
|
|
timeout=aiohttp.ClientTimeout(total=5)
|
|
|
|
|
)
|
2026-02-02 19:01:51 -08:00
|
|
|
except Exception:
|
|
|
|
|
pass # Ignore typing indicator failures
|
|
|
|
|
|
|
|
|
|
async def get_chat_info(self, chat_id: str) -> Dict[str, Any]:
|
|
|
|
|
"""Get information about a WhatsApp chat."""
|
fix(whatsapp): reuse persistent aiohttp session across requests (#3818)
Replace per-request aiohttp.ClientSession() in every WhatsApp adapter
method with a single persistent self._http_session, matching the pattern
used by Mattermost, HomeAssistant, and SMS adapters.
Changes:
- Create self._http_session in connect(), close in disconnect()
- All bridge HTTP calls (send, edit, send-media, typing, get_chat_info,
poll_messages) now use the shared session
- Explicitly cancel _poll_task on disconnect() instead of relying
solely on self._running = False
- Health-check sessions in connect() remain ephemeral (persistent
session not yet created at that point)
- Remove per-method ImportError guards for aiohttp (always available
when gateway runs via [messaging] extras)
Salvaged from PR #1851 by Himess. The _poll_task storage was already
on main from PR #3267; this adds the disconnect cancellation and the
persistent session.
Tests: 4 new tests for session close, already-closed skip, poll task
cancellation, and done-task skip.
2026-03-29 16:25:20 -07:00
|
|
|
if not self._running or not self._http_session:
|
2026-02-02 19:01:51 -08:00
|
|
|
return {"name": "Unknown", "type": "dm"}
|
2026-03-21 09:38:52 -07:00
|
|
|
if await self._check_managed_bridge_exit():
|
|
|
|
|
return {"name": chat_id, "type": "dm"}
|
2026-02-02 19:01:51 -08:00
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
import aiohttp
|
fix(whatsapp): reuse persistent aiohttp session across requests (#3818)
Replace per-request aiohttp.ClientSession() in every WhatsApp adapter
method with a single persistent self._http_session, matching the pattern
used by Mattermost, HomeAssistant, and SMS adapters.
Changes:
- Create self._http_session in connect(), close in disconnect()
- All bridge HTTP calls (send, edit, send-media, typing, get_chat_info,
poll_messages) now use the shared session
- Explicitly cancel _poll_task on disconnect() instead of relying
solely on self._running = False
- Health-check sessions in connect() remain ephemeral (persistent
session not yet created at that point)
- Remove per-method ImportError guards for aiohttp (always available
when gateway runs via [messaging] extras)
Salvaged from PR #1851 by Himess. The _poll_task storage was already
on main from PR #3267; this adds the disconnect cancellation and the
persistent session.
Tests: 4 new tests for session close, already-closed skip, poll task
cancellation, and done-task skip.
2026-03-29 16:25:20 -07:00
|
|
|
|
|
|
|
|
async with self._http_session.get(
|
|
|
|
|
f"http://127.0.0.1:{self._bridge_port}/chat/{chat_id}",
|
|
|
|
|
timeout=aiohttp.ClientTimeout(total=10)
|
|
|
|
|
) as resp:
|
|
|
|
|
if resp.status == 200:
|
|
|
|
|
data = await resp.json()
|
|
|
|
|
return {
|
|
|
|
|
"name": data.get("name", chat_id),
|
|
|
|
|
"type": "group" if data.get("isGroup") else "dm",
|
|
|
|
|
"participants": data.get("participants", []),
|
|
|
|
|
}
|
2026-02-21 03:32:11 -08:00
|
|
|
except Exception as e:
|
|
|
|
|
logger.debug("Could not get WhatsApp chat info for %s: %s", chat_id, e)
|
2026-02-02 19:01:51 -08:00
|
|
|
|
|
|
|
|
return {"name": chat_id, "type": "dm"}
|
|
|
|
|
|
|
|
|
|
async def _poll_messages(self) -> None:
|
|
|
|
|
"""Poll the bridge for incoming messages."""
|
fix(whatsapp): reuse persistent aiohttp session across requests (#3818)
Replace per-request aiohttp.ClientSession() in every WhatsApp adapter
method with a single persistent self._http_session, matching the pattern
used by Mattermost, HomeAssistant, and SMS adapters.
Changes:
- Create self._http_session in connect(), close in disconnect()
- All bridge HTTP calls (send, edit, send-media, typing, get_chat_info,
poll_messages) now use the shared session
- Explicitly cancel _poll_task on disconnect() instead of relying
solely on self._running = False
- Health-check sessions in connect() remain ephemeral (persistent
session not yet created at that point)
- Remove per-method ImportError guards for aiohttp (always available
when gateway runs via [messaging] extras)
Salvaged from PR #1851 by Himess. The _poll_task storage was already
on main from PR #3267; this adds the disconnect cancellation and the
persistent session.
Tests: 4 new tests for session close, already-closed skip, poll task
cancellation, and done-task skip.
2026-03-29 16:25:20 -07:00
|
|
|
import aiohttp
|
|
|
|
|
|
2026-02-02 19:01:51 -08:00
|
|
|
while self._running:
|
fix(whatsapp): reuse persistent aiohttp session across requests (#3818)
Replace per-request aiohttp.ClientSession() in every WhatsApp adapter
method with a single persistent self._http_session, matching the pattern
used by Mattermost, HomeAssistant, and SMS adapters.
Changes:
- Create self._http_session in connect(), close in disconnect()
- All bridge HTTP calls (send, edit, send-media, typing, get_chat_info,
poll_messages) now use the shared session
- Explicitly cancel _poll_task on disconnect() instead of relying
solely on self._running = False
- Health-check sessions in connect() remain ephemeral (persistent
session not yet created at that point)
- Remove per-method ImportError guards for aiohttp (always available
when gateway runs via [messaging] extras)
Salvaged from PR #1851 by Himess. The _poll_task storage was already
on main from PR #3267; this adds the disconnect cancellation and the
persistent session.
Tests: 4 new tests for session close, already-closed skip, poll task
cancellation, and done-task skip.
2026-03-29 16:25:20 -07:00
|
|
|
if not self._http_session:
|
|
|
|
|
break
|
2026-03-21 09:38:52 -07:00
|
|
|
bridge_exit = await self._check_managed_bridge_exit()
|
|
|
|
|
if bridge_exit:
|
|
|
|
|
print(f"[{self.name}] {bridge_exit}")
|
|
|
|
|
break
|
2026-02-02 19:01:51 -08:00
|
|
|
try:
|
fix(whatsapp): reuse persistent aiohttp session across requests (#3818)
Replace per-request aiohttp.ClientSession() in every WhatsApp adapter
method with a single persistent self._http_session, matching the pattern
used by Mattermost, HomeAssistant, and SMS adapters.
Changes:
- Create self._http_session in connect(), close in disconnect()
- All bridge HTTP calls (send, edit, send-media, typing, get_chat_info,
poll_messages) now use the shared session
- Explicitly cancel _poll_task on disconnect() instead of relying
solely on self._running = False
- Health-check sessions in connect() remain ephemeral (persistent
session not yet created at that point)
- Remove per-method ImportError guards for aiohttp (always available
when gateway runs via [messaging] extras)
Salvaged from PR #1851 by Himess. The _poll_task storage was already
on main from PR #3267; this adds the disconnect cancellation and the
persistent session.
Tests: 4 new tests for session close, already-closed skip, poll task
cancellation, and done-task skip.
2026-03-29 16:25:20 -07:00
|
|
|
async with self._http_session.get(
|
|
|
|
|
f"http://127.0.0.1:{self._bridge_port}/messages",
|
|
|
|
|
timeout=aiohttp.ClientTimeout(total=30)
|
|
|
|
|
) as resp:
|
|
|
|
|
if resp.status == 200:
|
|
|
|
|
messages = await resp.json()
|
|
|
|
|
for msg_data in messages:
|
|
|
|
|
event = await self._build_message_event(msg_data)
|
|
|
|
|
if event:
|
|
|
|
|
await self.handle_message(event)
|
2026-02-02 19:01:51 -08:00
|
|
|
except asyncio.CancelledError:
|
|
|
|
|
break
|
|
|
|
|
except Exception as e:
|
2026-03-21 09:38:52 -07:00
|
|
|
bridge_exit = await self._check_managed_bridge_exit()
|
|
|
|
|
if bridge_exit:
|
|
|
|
|
print(f"[{self.name}] {bridge_exit}")
|
|
|
|
|
break
|
2026-02-02 19:01:51 -08:00
|
|
|
print(f"[{self.name}] Poll error: {e}")
|
|
|
|
|
await asyncio.sleep(5)
|
|
|
|
|
|
|
|
|
|
await asyncio.sleep(1) # Poll interval
|
|
|
|
|
|
2026-02-15 16:10:50 -08:00
|
|
|
async def _build_message_event(self, data: Dict[str, Any]) -> Optional[MessageEvent]:
|
|
|
|
|
"""Build a MessageEvent from bridge message data, downloading images to cache."""
|
2026-02-02 19:01:51 -08:00
|
|
|
try:
|
|
|
|
|
# Determine message type
|
|
|
|
|
msg_type = MessageType.TEXT
|
|
|
|
|
if data.get("hasMedia"):
|
|
|
|
|
media_type = data.get("mediaType", "")
|
|
|
|
|
if "image" in media_type:
|
|
|
|
|
msg_type = MessageType.PHOTO
|
|
|
|
|
elif "video" in media_type:
|
|
|
|
|
msg_type = MessageType.VIDEO
|
|
|
|
|
elif "audio" in media_type or "ptt" in media_type: # ptt = voice note
|
|
|
|
|
msg_type = MessageType.VOICE
|
|
|
|
|
else:
|
|
|
|
|
msg_type = MessageType.DOCUMENT
|
|
|
|
|
|
|
|
|
|
# Determine chat type
|
|
|
|
|
is_group = data.get("isGroup", False)
|
|
|
|
|
chat_type = "group" if is_group else "dm"
|
|
|
|
|
|
|
|
|
|
# Build source
|
|
|
|
|
source = self.build_source(
|
|
|
|
|
chat_id=data.get("chatId", ""),
|
|
|
|
|
chat_name=data.get("chatName"),
|
|
|
|
|
chat_type=chat_type,
|
|
|
|
|
user_id=data.get("senderId"),
|
|
|
|
|
user_name=data.get("senderName"),
|
|
|
|
|
)
|
|
|
|
|
|
fix(whatsapp): download documents, audio, and video media from messages (#2978)
Add downloadMediaMessage() calls for documents, audio/voice notes, and
video in bridge.js — previously only images were downloaded, leaving all
other file types inaccessible to the agent.
Handle local file paths from the bridge for DOCUMENT, VOICE, and VIDEO
types in whatsapp.py with proper MIME detection. Inject text content
inline for readable files (.txt, .md, .csv, .json, etc.).
Follow-up fixes applied during salvage:
- Remove unused cache_document_from_bytes import
- Add 100KB size cap on text injection (matches Telegram/Discord/Slack)
- Align injection format with other platforms
Cherry-picked from PR #2818. Also fixes #2856 (bugs 1 & 2).
PR #2865 by ayberkesn fixed the same voice note issue.
Co-authored-by: noestelar <hola@noeali.com>
2026-03-25 08:37:28 -07:00
|
|
|
# Download media URLs to the local cache so agent tools
|
2026-02-15 16:10:50 -08:00
|
|
|
# can access them reliably regardless of URL expiration.
|
|
|
|
|
raw_urls = data.get("mediaUrls", [])
|
|
|
|
|
cached_urls = []
|
|
|
|
|
media_types = []
|
|
|
|
|
for url in raw_urls:
|
|
|
|
|
if msg_type == MessageType.PHOTO and url.startswith(("http://", "https://")):
|
|
|
|
|
try:
|
|
|
|
|
cached_path = await cache_image_from_url(url, ext=".jpg")
|
|
|
|
|
cached_urls.append(cached_path)
|
|
|
|
|
media_types.append("image/jpeg")
|
|
|
|
|
print(f"[{self.name}] Cached user image: {cached_path}", flush=True)
|
|
|
|
|
except Exception as e:
|
|
|
|
|
print(f"[{self.name}] Failed to cache image: {e}", flush=True)
|
Add messaging platform enhancements: STT, stickers, Discord UX, Slack, pairing, hooks
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.
2026-02-15 21:38:59 -08:00
|
|
|
cached_urls.append(url)
|
2026-02-15 16:10:50 -08:00
|
|
|
media_types.append("image/jpeg")
|
2026-03-20 09:37:48 -07:00
|
|
|
elif msg_type == MessageType.PHOTO and os.path.isabs(url):
|
|
|
|
|
# Local file path — bridge already downloaded the image
|
|
|
|
|
cached_urls.append(url)
|
|
|
|
|
media_types.append("image/jpeg")
|
|
|
|
|
print(f"[{self.name}] Using bridge-cached image: {url}", flush=True)
|
Add messaging platform enhancements: STT, stickers, Discord UX, Slack, pairing, hooks
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.
2026-02-15 21:38:59 -08:00
|
|
|
elif msg_type == MessageType.VOICE and url.startswith(("http://", "https://")):
|
|
|
|
|
try:
|
|
|
|
|
cached_path = await cache_audio_from_url(url, ext=".ogg")
|
|
|
|
|
cached_urls.append(cached_path)
|
|
|
|
|
media_types.append("audio/ogg")
|
|
|
|
|
print(f"[{self.name}] Cached user voice: {cached_path}", flush=True)
|
|
|
|
|
except Exception as e:
|
|
|
|
|
print(f"[{self.name}] Failed to cache voice: {e}", flush=True)
|
|
|
|
|
cached_urls.append(url)
|
|
|
|
|
media_types.append("audio/ogg")
|
fix(whatsapp): download documents, audio, and video media from messages (#2978)
Add downloadMediaMessage() calls for documents, audio/voice notes, and
video in bridge.js — previously only images were downloaded, leaving all
other file types inaccessible to the agent.
Handle local file paths from the bridge for DOCUMENT, VOICE, and VIDEO
types in whatsapp.py with proper MIME detection. Inject text content
inline for readable files (.txt, .md, .csv, .json, etc.).
Follow-up fixes applied during salvage:
- Remove unused cache_document_from_bytes import
- Add 100KB size cap on text injection (matches Telegram/Discord/Slack)
- Align injection format with other platforms
Cherry-picked from PR #2818. Also fixes #2856 (bugs 1 & 2).
PR #2865 by ayberkesn fixed the same voice note issue.
Co-authored-by: noestelar <hola@noeali.com>
2026-03-25 08:37:28 -07:00
|
|
|
elif msg_type == MessageType.VOICE and os.path.isabs(url):
|
|
|
|
|
# Local file path — bridge already downloaded the audio
|
|
|
|
|
cached_urls.append(url)
|
|
|
|
|
media_types.append("audio/ogg")
|
|
|
|
|
print(f"[{self.name}] Using bridge-cached audio: {url}", flush=True)
|
|
|
|
|
elif msg_type == MessageType.DOCUMENT and os.path.isabs(url):
|
|
|
|
|
# Local file path — bridge already downloaded the document
|
|
|
|
|
cached_urls.append(url)
|
|
|
|
|
ext = Path(url).suffix.lower()
|
|
|
|
|
mime = SUPPORTED_DOCUMENT_TYPES.get(ext, "application/octet-stream")
|
|
|
|
|
media_types.append(mime)
|
|
|
|
|
print(f"[{self.name}] Using bridge-cached document: {url}", flush=True)
|
|
|
|
|
elif msg_type == MessageType.VIDEO and os.path.isabs(url):
|
|
|
|
|
cached_urls.append(url)
|
|
|
|
|
media_types.append("video/mp4")
|
|
|
|
|
print(f"[{self.name}] Using bridge-cached video: {url}", flush=True)
|
2026-02-15 16:10:50 -08:00
|
|
|
else:
|
|
|
|
|
cached_urls.append(url)
|
|
|
|
|
media_types.append("unknown")
|
fix(whatsapp): download documents, audio, and video media from messages (#2978)
Add downloadMediaMessage() calls for documents, audio/voice notes, and
video in bridge.js — previously only images were downloaded, leaving all
other file types inaccessible to the agent.
Handle local file paths from the bridge for DOCUMENT, VOICE, and VIDEO
types in whatsapp.py with proper MIME detection. Inject text content
inline for readable files (.txt, .md, .csv, .json, etc.).
Follow-up fixes applied during salvage:
- Remove unused cache_document_from_bytes import
- Add 100KB size cap on text injection (matches Telegram/Discord/Slack)
- Align injection format with other platforms
Cherry-picked from PR #2818. Also fixes #2856 (bugs 1 & 2).
PR #2865 by ayberkesn fixed the same voice note issue.
Co-authored-by: noestelar <hola@noeali.com>
2026-03-25 08:37:28 -07:00
|
|
|
|
|
|
|
|
# For text-readable documents, inject file content directly into
|
|
|
|
|
# the message text so the agent can read it inline.
|
|
|
|
|
# Cap at 100KB to match Telegram/Discord/Slack behaviour.
|
|
|
|
|
body = data.get("body", "")
|
|
|
|
|
MAX_TEXT_INJECT_BYTES = 100 * 1024
|
|
|
|
|
if msg_type == MessageType.DOCUMENT and cached_urls:
|
|
|
|
|
for doc_path in cached_urls:
|
|
|
|
|
ext = Path(doc_path).suffix.lower()
|
|
|
|
|
if ext in (".txt", ".md", ".csv", ".json", ".xml", ".yaml", ".yml", ".log", ".py", ".js", ".ts", ".html", ".css"):
|
|
|
|
|
try:
|
|
|
|
|
file_size = Path(doc_path).stat().st_size
|
|
|
|
|
if file_size > MAX_TEXT_INJECT_BYTES:
|
|
|
|
|
print(f"[{self.name}] Skipping text injection for {doc_path} ({file_size} bytes > {MAX_TEXT_INJECT_BYTES})", flush=True)
|
|
|
|
|
continue
|
|
|
|
|
content = Path(doc_path).read_text(errors="replace")
|
|
|
|
|
fname = Path(doc_path).name
|
|
|
|
|
# Remove the doc_<hex>_ prefix for display
|
|
|
|
|
display_name = fname
|
|
|
|
|
if "_" in fname:
|
|
|
|
|
parts = fname.split("_", 2)
|
|
|
|
|
if len(parts) >= 3:
|
|
|
|
|
display_name = parts[2]
|
|
|
|
|
injection = f"[Content of {display_name}]:\n{content}"
|
|
|
|
|
if body:
|
|
|
|
|
body = f"{injection}\n\n{body}"
|
|
|
|
|
else:
|
|
|
|
|
body = injection
|
|
|
|
|
print(f"[{self.name}] Injected text content from: {doc_path}", flush=True)
|
|
|
|
|
except Exception as e:
|
|
|
|
|
print(f"[{self.name}] Failed to read document text: {e}", flush=True)
|
|
|
|
|
|
2026-02-02 19:01:51 -08:00
|
|
|
return MessageEvent(
|
fix(whatsapp): download documents, audio, and video media from messages (#2978)
Add downloadMediaMessage() calls for documents, audio/voice notes, and
video in bridge.js — previously only images were downloaded, leaving all
other file types inaccessible to the agent.
Handle local file paths from the bridge for DOCUMENT, VOICE, and VIDEO
types in whatsapp.py with proper MIME detection. Inject text content
inline for readable files (.txt, .md, .csv, .json, etc.).
Follow-up fixes applied during salvage:
- Remove unused cache_document_from_bytes import
- Add 100KB size cap on text injection (matches Telegram/Discord/Slack)
- Align injection format with other platforms
Cherry-picked from PR #2818. Also fixes #2856 (bugs 1 & 2).
PR #2865 by ayberkesn fixed the same voice note issue.
Co-authored-by: noestelar <hola@noeali.com>
2026-03-25 08:37:28 -07:00
|
|
|
text=body,
|
2026-02-02 19:01:51 -08:00
|
|
|
message_type=msg_type,
|
|
|
|
|
source=source,
|
|
|
|
|
raw_message=data,
|
|
|
|
|
message_id=data.get("messageId"),
|
2026-02-15 16:10:50 -08:00
|
|
|
media_urls=cached_urls,
|
|
|
|
|
media_types=media_types,
|
2026-02-02 19:01:51 -08:00
|
|
|
)
|
|
|
|
|
except Exception as e:
|
|
|
|
|
print(f"[{self.name}] Error building event: {e}")
|
|
|
|
|
return None
|