hermes-agent/website/docs/user-guide/messaging/matrix.md
Chris 4717989c10
fix(matrix): isolate room context and restore reliable inbound dispatch (#18505)
* fix(matrix): isolate room context and inbound dispatch

* test(matrix): cover room isolation and dispatch regressions

* docs(matrix): document room isolation and session scope

* fix(matrix): stabilize CI requirement checks

* test(matrix): isolate mautrix stubs in requirements tests

* fix(matrix): port room-scoped status and resume to slash commands mixin

Move Matrix /status scope output and /resume same-room guards from the
pre-refactor gateway/run.py into gateway/slash_commands.py so PR #18505
foundation behavior survives the upstream god-file decomposition.

Uses i18n keys for Matrix resume/status messages. Preserves upstream
session.py fixes (role_authorized, DM user_id isolation).

* docs(matrix): explain inbound dispatch via handle_sync loop

Document why Hermes uses an explicit sync loop with handle_sync() rather than
client.start(), aligning with upstream #7914 diagnostics while preserving
Hermes background maintenance tasks.

* fix(i18n): add Matrix resume/status keys to all locale catalogs

The Matrix /resume and /status slash-command keys added in the foundation
PR must exist in every supported locale file. tests/agent/test_i18n.py
asserts key and placeholder parity across catalogs.

Non-English locales use English strings as interim placeholders until
community translators can localize them.

* fix(matrix): restore gateway authz for allowed_users; honor config require_mention

Revert the early MATRIX_ALLOWED_USERS gate in _on_room_message so inbound
sender authorization stays in gateway authz like main. Parse require_mention
from config.extra (platforms.matrix / top-level matrix yaml) with env fallback,
matching thread_require_mention and fixing Forge when require_mention is set
only in profile config.yaml.

* fix(matrix): harden status scope and allowlisted DMs

* fix(matrix): use session store lookup for resume scope
2026-06-11 07:41:43 -04:00

38 KiB

sidebar_position title description
9 Matrix Set up Hermes Agent as a Matrix bot

Matrix Setup

Hermes Agent integrates with Matrix, the open, federated messaging protocol. Matrix lets you run your own homeserver or use a public one like matrix.org — either way, you keep control of your communications. The bot connects via the mautrix Python SDK, processes messages through the Hermes Agent pipeline (including tool use, memory, and reasoning), and responds in real time. It supports text, file attachments, images, audio, video, and optional end-to-end encryption (E2EE).

Hermes works with any Matrix homeserver — Synapse, Conduit, Dendrite, or matrix.org.

Before setup, here's the part most people want to know: how Hermes behaves once it's connected.

How Hermes Behaves

Context Behavior
DMs Hermes responds to every message. No @mention needed. Each DM has its own session. Set MATRIX_DM_MENTION_THREADS=true to start a thread when the bot is @mentioned in a DM.
Rooms By default, Hermes requires an @mention to respond. Set MATRIX_REQUIRE_MENTION=false or add room IDs to MATRIX_FREE_RESPONSE_ROOMS for free-response rooms. Room invites are auto-accepted.
Threads Hermes supports Matrix threads (MSC3440). If you reply in a thread, Hermes keeps the thread context isolated from the main room timeline. Threads where the bot has already participated do not require a mention.
Auto-threading By default, Hermes auto-creates a thread for each message it responds to in a room. This keeps conversations isolated. Set MATRIX_AUTO_THREAD=false to disable. Set MATRIX_DM_AUTO_THREAD=true (default false) to also auto-create threads for DM messages — this is distinct from MATRIX_DM_MENTION_THREADS, which only starts a thread when the bot is @mentioned in a DM.
Commands Hermes accepts normal /commands when your Matrix client sends them. If your client reserves / for local commands, use !commands instead; Hermes normalizes known !command aliases to /command.
Interactive controls Dangerous-command approval and /model selection can use Matrix reactions. Approval reactions can be limited to the user who requested the action.
Thinking and tool activity Matrix uses threaded, editable thinking/tool-activity panes when gateway progress is enabled, so updates do not flood the main room timeline.
Shared rooms with multiple users By default, Hermes isolates session history per user inside the room. Two people talking in the same room do not share one transcript unless you explicitly disable that.

:::tip The bot automatically joins rooms when invited. Just invite the bot's Matrix user to any room and it will join and start responding. :::

Capability Matrix

This table is backed by the Matrix adapter capability declaration and Matrix test coverage. E2EE is mode-based because deployments choose whether encrypted rooms are disabled, opportunistic, or required.

Capability Matrix
text yes
threads yes
reactions yes
approvals yes
model picker yes
thinking panes yes
images yes
multiple images yes
files yes
voice/audio yes
video yes
E2EE off / optional / required
diagnostics yes

Session Model in Matrix

By default:

  • each DM gets its own session
  • each thread gets its own session namespace
  • each user in a shared room gets their own session inside that room

This is controlled by config.yaml:

group_sessions_per_user: true

Set it to false only if you explicitly want one shared conversation for the entire room:

group_sessions_per_user: false

Shared sessions can be useful for a collaborative room, but they also mean:

  • users share context growth and token costs
  • one person's long tool-heavy task can bloat everyone else's context
  • one person's in-flight run can interrupt another person's follow-up in the same room

Mention and Threading Configuration

You can configure mention and auto-threading behavior via environment variables or config.yaml:

matrix:
  require_mention: true           # Require @mention in rooms (default: true)
  allowed_users:                  # Matrix users allowed to trigger agent turns
    - "@alice:matrix.org"
  allowed_rooms:                  # Matrix rooms allowed to trigger agent turns
    - "!abc123:matrix.org"
  free_response_rooms:            # Rooms exempt from mention requirement
    - "!abc123:matrix.org"
  ignore_user_patterns:           # Bridge/appservice ghost users to ignore
    - "^@telegram_"
    - "^@whatsapp_"
  process_notices: false          # Ignore m.notice by default
  session_scope: room             # auto|room|thread; room is recommended for project rooms
  auto_thread: true               # Auto-create threads for responses (default: true)
  dm_mention_threads: false       # Create thread when @mentioned in DM (default: false)

Or via environment variables:

MATRIX_REQUIRE_MENTION=true
MATRIX_ALLOWED_USERS=@alice:matrix.org
MATRIX_ALLOWED_ROOMS=!abc123:matrix.org
MATRIX_FREE_RESPONSE_ROOMS=!abc123:matrix.org,!def456:matrix.org
MATRIX_IGNORE_USER_PATTERNS='^@telegram_,^@whatsapp_'
MATRIX_PROCESS_NOTICES=false
MATRIX_SESSION_SCOPE=room       # recommended for stable project-room context
MATRIX_AUTO_THREAD=true
MATRIX_DM_MENTION_THREADS=false
MATRIX_REACTIONS=true          # default: true — emoji reactions during processing
MATRIX_ALLOW_ROOM_MENTIONS=false

:::tip Disabling reactions MATRIX_REACTIONS=false turns off the processing-lifecycle emoji reactions (👀//) the bot posts on inbound messages. Useful for rooms where reaction events are noisy or aren't supported by all participating clients. :::

:::tip Room-wide mentions Hermes sends structured Matrix user mentions for explicit Matrix IDs such as @alice:example.org. Room-wide @room notifications are disabled by default; set MATRIX_ALLOW_ROOM_MENTIONS=true only in rooms where the bot is allowed to notify everyone. :::

:::note If you are upgrading from a version that did not have MATRIX_REQUIRE_MENTION, the bot previously responded to all messages in rooms. To preserve that behavior, set MATRIX_REQUIRE_MENTION=false. :::

Project Room Isolation

If you use the same Matrix bot in multiple project rooms, configure stable room-scoped sessions:

MATRIX_SESSION_SCOPE=room
MATRIX_AUTO_THREAD=false

MATRIX_SESSION_SCOPE accepts:

Scope Behavior
auto Backward-compatible default. Existing MATRIX_AUTO_THREAD behavior controls synthetic threads.
room Unthreaded room messages stay in one stable room session. Real Matrix threads still use their thread root.
thread Unthreaded room messages synthesize a thread/session from the triggering event ID.

Hermes now includes the current Matrix room name, room ID, topic, message ID, and a Matrix room-boundary note in the agent prompt. /status also shows the current Matrix room/session scope, and /resume will not silently resume a named session from another Matrix room unless you explicitly use /resume --cross-room <session name>.

MATRIX_SESSION_SCOPE=room controls the room/thread lane. The existing group_sessions_per_user setting still controls whether users inside that room share the lane. With group_sessions_per_user: true (default), Alice and Bob get separate Project B sessions. With group_sessions_per_user: false, the room has one shared Project B transcript.

This guide walks you through the full setup process — from creating your bot account to sending your first message.

Step 1: Create a Bot Account

You need a Matrix user account for the bot. There are several ways to do this:

If you run your own homeserver (Synapse, Conduit, Dendrite):

  1. Use the admin API or registration tool to create a new user:
# Synapse example
register_new_matrix_user -c /etc/synapse/homeserver.yaml http://localhost:8008
  1. Choose a username like hermes — the full user ID will be @hermes:your-server.org.

Option B: Use matrix.org or Another Public Homeserver

  1. Go to Element Web and create a new account.
  2. Pick a username for your bot (e.g., hermes-bot).

Option C: Use Your Own Account

You can also run Hermes as your own user. This means the bot posts as you — useful for personal assistants.

Step 2: Get an Access Token

Hermes needs an access token to authenticate with the homeserver. You have two options:

The most reliable way to get a token:

Via Element:

  1. Log in to Element with the bot account.
  2. Go to SettingsHelp & About.
  3. Scroll down and expand Advanced — the access token is displayed there.
  4. Copy it immediately.

Via the API:

curl -X POST https://your-server/_matrix/client/v3/login \
  -H "Content-Type: application/json" \
  -d '{
    "type": "m.login.password",
    "user": "@hermes:your-server.org",
    "password": "your-password"
  }'

The response includes an access_token field — copy it.

:::warning[Keep your access token safe] The access token gives full access to the bot's Matrix account. Never share it publicly or commit it to Git. If compromised, revoke it by logging out all sessions for that user. :::

Option B: Password Login

Instead of providing an access token, you can give Hermes the bot's user ID and password. Hermes will log in automatically on startup. This is simpler but means the password is stored in your .env file.

MATRIX_USER_ID=@hermes:your-server.org
MATRIX_PASSWORD=your-password

Step 3: Find Your Matrix User ID

Hermes Agent uses your Matrix User ID to control who can interact with the bot. Matrix User IDs follow the format @username:server.

To find yours:

  1. Open Element (or your preferred Matrix client).
  2. Click your avatar → Settings.
  3. Your User ID is displayed at the top of the profile (e.g., @alice:matrix.org).

:::tip Matrix User IDs always start with @ and contain a : followed by the server name. For example: @alice:matrix.org, @bob:your-server.com. :::

Step 4: Configure Hermes Agent

Run the guided setup command:

hermes gateway setup

Select Matrix when prompted, then provide your homeserver URL, access token (or user ID + password), and allowed user IDs when asked.

Option B: Manual Configuration

Add the following to your ~/.hermes/.env file:

Using an access token:

# Required
MATRIX_HOMESERVER=https://matrix.example.org
MATRIX_ACCESS_TOKEN=***

# Optional: user ID (auto-detected from token if omitted)
# MATRIX_USER_ID=@hermes:matrix.example.org

# Security: restrict who can interact with the bot
MATRIX_ALLOWED_USERS=@alice:matrix.example.org

# Optional: restrict which rooms can trigger the bot
MATRIX_ALLOWED_ROOMS=!abc123:matrix.example.org

# Multiple allowed users (comma-separated)
# MATRIX_ALLOWED_USERS=@alice:matrix.example.org,@bob:matrix.example.org

Using password login:

# Required
MATRIX_HOMESERVER=https://matrix.example.org
MATRIX_USER_ID=@hermes:matrix.example.org
MATRIX_PASSWORD=***

# Security
MATRIX_ALLOWED_USERS=@alice:matrix.example.org

Private Deployment Hardening

For private Matrix deployments, set both user and room allowlists. If MATRIX_ALLOWED_USERS is unset, any sender who can reach the bot in a joined room can trigger an agent turn. If MATRIX_ALLOWED_ROOMS is unset, any room the bot joins can trigger an agent turn. A locked-down deployment should set both:

MATRIX_ALLOWED_USERS=@alice:matrix.example.org,@bob:matrix.example.org
MATRIX_ALLOWED_ROOMS=!ops:matrix.example.org,!dmroom:matrix.example.org

Bridge and appservice deployments need extra loop protection. Hermes always ignores its own events, Matrix appservice-style users whose localpart starts with _, duplicate event IDs, old startup events, edit replacement events, and m.notice events by default. Add deployment-specific bridge ghost patterns when your bridge uses a different naming convention:

MATRIX_IGNORE_USER_PATTERNS='^@telegram_,^@slack_,^@whatsapp_'

Only enable notices when a trusted human workflow really sends m.notice:

MATRIX_PROCESS_NOTICES=true

Outbound whole-room notifications are disabled by default. Keep MATRIX_ALLOW_ROOM_MENTIONS=false unless the bot is explicitly allowed to wake the whole room with @room.

Diagnostics and debug payloads redact Matrix access tokens, recovery keys, device identifiers, and message bodies. Media downloads are limited to Matrix mxc:// content URIs and rejected when they exceed MATRIX_MAX_MEDIA_BYTES. Treat federated rooms and untrusted homeservers as untrusted input: keep room allowlists tight, prefer DMs or private rooms for tool-heavy work, and avoid authorizing bridge ghosts or appservice puppets as allowed users.

Optional behavior settings in ~/.hermes/config.yaml:

group_sessions_per_user: true
  • group_sessions_per_user: true keeps each participant's context isolated inside shared rooms

Start the Gateway

Once configured, start the Matrix gateway:

hermes gateway

The bot should connect to your homeserver and start syncing within a few seconds. Send it a message — either a DM or in a room it has joined — to test.

:::tip You can run hermes gateway in the background or as a systemd service for persistent operation. See the deployment docs for details. :::

End-to-End Encryption (E2EE)

Hermes supports Matrix end-to-end encryption, so you can chat with your bot in encrypted rooms.

Requirements

E2EE requires the mautrix library with encryption extras and the libolm C library:

# Install mautrix with E2EE support
pip install 'mautrix[encryption]'

# Or install with hermes extras
pip install 'hermes-agent[matrix]'

You also need libolm installed on your system:

# Debian/Ubuntu
sudo apt install libolm-dev

# macOS
brew install libolm

# Fedora
sudo dnf install libolm-devel

Enable E2EE

Add to your ~/.hermes/.env:

MATRIX_E2EE_MODE=required

MATRIX_E2EE_MODE accepts:

Mode Behavior
off Do not initialize Matrix E2EE.
optional Try E2EE when dependencies are available, but keep unencrypted rooms working if crypto cannot initialize.
required Fail closed if E2EE dependencies or crypto setup are not available.

Optional mode may fall back to non-E2EE operation when crypto setup is unavailable. Required mode fails closed instead of silently downgrading.

For backwards compatibility, MATRIX_ENCRYPTION=true still enables required E2EE behavior.

When E2EE is enabled, Hermes:

  • Stores encryption keys in ~/.hermes/platforms/matrix/store/ (legacy installs: ~/.hermes/matrix/store/)
  • Uploads device keys on first connection
  • Decrypts incoming messages and encrypts outgoing messages automatically
  • Auto-joins encrypted rooms when invited

Matrix Tools and Controls

In Matrix conversations, Hermes exposes Matrix-specific tools to the agent:

  • matrix_send_reaction
  • matrix_redact_message
  • matrix_create_room
  • matrix_invite_user
  • matrix_fetch_history
  • matrix_set_presence

These tools are scoped to Matrix contexts and are not available in non-Matrix toolsets. Admin-style tools are disabled by default: redaction requires MATRIX_TOOLS_ALLOW_REDACTION=true, invites require MATRIX_TOOLS_ALLOW_INVITES=true, and room creation requires MATRIX_TOOLS_ALLOW_ROOM_CREATE=true. Public room creation also requires MATRIX_ALLOW_PUBLIC_ROOMS=true. Matrix tools are limited to the current Matrix room by default. Explicit cross-room targets require MATRIX_TOOLS_ALLOW_CROSS_ROOM=true; redaction and invite-like cross-room actions additionally require MATRIX_TOOLS_ALLOW_CROSS_ROOM_DESTRUCTIVE=true. If MATRIX_ALLOWED_ROOMS is set, Matrix tools may only target those rooms.

Reaction controls use:

  • approve once
  • ♾️ approve always
  • deny
  • number reactions for /model choices

Set MATRIX_APPROVAL_REQUIRE_SENDER=false if you intentionally want any authorized Matrix user in the room to operate an approval/model picker prompt. The default is requester-bound when Hermes knows who requested the action.

Media Limits

Hermes uploads and downloads Matrix images, files, audio, and video through Matrix media APIs. Multiple generated images are sent as one ordered logical batch, preserving captions and thread context across the batch.

By default, Matrix media over 100 MB is rejected before upload/download. Override with:

MATRIX_MAX_MEDIA_BYTES=104857600

Inbound media must use Matrix mxc:// content URIs. Hermes rejects arbitrary HTTP(S) media URLs in Matrix events to avoid turning a federated room into an unrestricted downloader.

Synapse Integration Tests

Hermes includes an opt-in Synapse harness for local validation:

docker compose -f tests/e2e/matrix_synapse_gateway/docker-compose.yml up -d
HERMES_MATRIX_SYNAPSE_INTEGRATION=1 \
  scripts/run_tests.sh -m "integration and matrix_synapse" \
  tests/e2e/matrix_synapse_gateway/test_gateway.py
docker compose -f tests/e2e/matrix_synapse_gateway/docker-compose.yml down -v

The harness creates temporary users through Synapse shared-secret registration and covers private-room send/receive, named-room invite/join, media upload/download, bot response delivery, and startup old-event filtering. E2EE smoke coverage is separately marked with matrix_e2ee so it can stay opt-in on developer machines.

If your Matrix account has cross-signing enabled (the default in Element), set the recovery key so the bot can self-sign its device on startup. Without this, other Matrix clients may refuse to share encryption sessions with the bot after a device key rotation.

MATRIX_RECOVERY_KEY=EsT... your recovery key here

Where to find it: In Element, go to SettingsSecurity & PrivacyEncryption → your recovery key (also called the "Security Key"). This is the key you were asked to save when you first set up cross-signing.

On each startup, if MATRIX_RECOVERY_KEY is set, Hermes imports cross-signing keys from the homeserver's secure secret storage and signs the current device. This is idempotent and safe to leave enabled permanently.

If Hermes bootstraps a new Matrix recovery key, it never logs the raw key. Set MATRIX_RECOVERY_KEY_OUTPUT_FILE=/secure/path/matrix-recovery-key.txt before startup to write a generated key once with file mode 0600; the file is not overwritten if it already exists.

:::warning[Deleting the crypto store] If you delete ~/.hermes/platforms/matrix/store/crypto.db, the bot loses its encryption identity. Simply restarting with the same device ID will not fully recover — the homeserver still holds one-time keys signed with the old identity key, and peers cannot establish new Olm sessions.

Hermes detects this condition on startup and refuses to enable E2EE, logging: device XXXX has stale one-time keys on the server signed with a previous identity key.

Easiest recovery: generate a new access token (which gets a fresh device ID with no stale key history). See the "Upgrading from a previous version with E2EE" section below. This is the most reliable path and avoids touching the homeserver database.

Manual recovery (advanced — keeps the same device ID):

  1. Stop Synapse and delete the old device from its database:

    sudo systemctl stop matrix-synapse
    sudo sqlite3 /var/lib/matrix-synapse/homeserver.db "
      DELETE FROM e2e_device_keys_json WHERE device_id = 'DEVICE_ID' AND user_id = '@hermes:your-server';
      DELETE FROM e2e_one_time_keys_json WHERE device_id = 'DEVICE_ID' AND user_id = '@hermes:your-server';
      DELETE FROM e2e_fallback_keys_json WHERE device_id = 'DEVICE_ID' AND user_id = '@hermes:your-server';
      DELETE FROM devices WHERE device_id = 'DEVICE_ID' AND user_id = '@hermes:your-server';
    "
    sudo systemctl start matrix-synapse
    

    Or via the Synapse admin API (note the URL-encoded user ID):

    curl -X DELETE -H "Authorization: Bearer ADMIN_TOKEN" \
      'https://your-server/_synapse/admin/v2/users/%40hermes%3Ayour-server/devices/DEVICE_ID'
    

    Note: deleting a device via the admin API may also invalidate the associated access token. You may need to generate a new token afterward.

  2. Delete the local crypto store and restart Hermes:

    rm -f ~/.hermes/platforms/matrix/store/crypto.db*
    # restart hermes
    

Other Matrix clients (Element, matrix-commander) may cache the old device keys. After recovery, type /discardsession in Element to force a new encryption session with the bot. :::

:::info If mautrix[encryption] is not installed or libolm is missing, the bot falls back to a plain (unencrypted) client automatically. You'll see a warning in the logs. :::

Home Room

You can designate a "home room" where the bot sends proactive messages (such as cron job output, reminders, and notifications). There are two ways to set it:

Using the Slash Command

Type /sethome in any Matrix room where the bot is present. That room becomes the home room. If your Matrix client intercepts slash commands, type !sethome instead.

Manual Configuration

Add this to your ~/.hermes/.env:

MATRIX_HOME_ROOM=!abc123def456:matrix.example.org

Room allowlist (allowed_rooms)

Restrict the bot to a fixed set of Matrix rooms. When set, the bot only responds in rooms whose ID appears in the list — messages from any other room are silently ignored, even if the bot is mentioned.

DMs (direct chat rooms) are exempt from this filter, so authorized users can always reach the bot one-on-one.

matrix:
  allowed_rooms:
    - "!abc123def456:matrix.example.org"
    - "!opsroom789:matrix.example.org"

Or via env var (comma-separated):

MATRIX_ALLOWED_ROOMS="!abc123def456:matrix.example.org,!opsroom789:matrix.example.org"

Behavior:

  • Empty / unset → no restriction (default).
  • Non-empty → room ID must be on the list. The check runs before any other gating (mention requirement, sender allowlist, etc.).
  • Use the room's internal ID (!abc...:server), not its alias (#room:server). You can find a room's internal ID in Element via Room → Settings → Advanced.

See also: admin/user slash command split.

:::tip To find a Room ID: in Element, go to the room → SettingsAdvanced → the Internal room ID is shown there (starts with !). :::

Commands in Matrix

Hermes supports the same gateway commands in Matrix that it supports on other messaging platforms, including /commands, /model, /stop, /queue, /steer, /goal, /subgoal, /background, /bg, /btw, /tasks, and /yolo.

Some Matrix clients reserve leading / for local client commands and may not send unknown slash commands to the room. In that case, use ! as a Matrix-safe alias:

!commands
!model
!model gpt-5.5 --provider openrouter
!queue continue with the next task
!stop

Hermes only normalizes !command when the command is known to the gateway, a registered plugin command, or an installed skill command. Ordinary exclamations such as !important remain normal chat messages.

Troubleshooting

Bot is not responding to messages

Cause: The bot hasn't joined the room, MATRIX_ALLOWED_USERS doesn't include your User ID, MATRIX_ALLOWED_ROOMS doesn't include the room, or a room message did not mention the bot.

Fix: Invite the bot to the room — it auto-joins on invite. Verify your User ID is in MATRIX_ALLOWED_USERS (use the full @user:server format) and the room ID is in MATRIX_ALLOWED_ROOMS if that allowlist is configured. In rooms, mention the bot or add the room to MATRIX_FREE_RESPONSE_ROOMS. Restart the gateway.

Bot joins rooms but silently drops every message (clock skew)

Cause: The host's system clock is set ahead of real time. The Matrix adapter applies a 5-second startup-grace filter (event_ts < startup_ts - 5) to ignore events replayed from initial sync. When the wall clock is ahead, every incoming event looks "older than startup" and is dropped before reaching the message handler — the bot appears connected but never replies. See #12614.

Symptom: Gateway log shows Matrix: dropped N live events as 'too old' more than 30s after startup.

Fix: Sync the host clock with NTP and restart the bot:

# Debian/Ubuntu
sudo timedatectl set-ntp true
timedatectl status   # confirm "System clock synchronized: yes"

# macOS
sudo sntp -sS time.apple.com

"Failed to authenticate" / "whoami failed" on startup

Cause: The access token or homeserver URL is incorrect.

Fix: Verify MATRIX_HOMESERVER points to your homeserver (include https://, no trailing slash). Check that MATRIX_ACCESS_TOKEN is valid — try it with curl:

curl -H "Authorization: Bearer YOUR_TOKEN" \
  https://your-server/_matrix/client/v3/account/whoami

If this returns your user info, the token is valid. If it returns an error, generate a new token.

"mautrix not installed" error

Cause: The mautrix Python package is not installed.

Fix: Install it:

pip install 'mautrix[encryption]'

Or with Hermes extras:

pip install 'hermes-agent[matrix]'

Encryption errors / "could not decrypt event"

Cause: Missing encryption keys, libolm not installed, or the bot's device isn't trusted.

Fix:

  1. Verify libolm is installed on your system (see the E2EE section above).
  2. Make sure MATRIX_ENCRYPTION=true is set in your .env.
  3. In your Matrix client (Element), go to the bot's profile -> Sessions -> verify/trust the bot's device.
  4. If the bot just joined an encrypted room, it can only decrypt messages sent after it joined. Older messages are inaccessible.

Upgrading from a previous version with E2EE

:::tip If you also manually deleted crypto.db, see the "Deleting the crypto store" warning in the E2EE section above — there are additional steps to clear stale one-time keys from the homeserver. :::

If you previously used Hermes with MATRIX_ENCRYPTION=true and are upgrading to a version that uses the new SQLite-based crypto store, the bot's encryption identity has changed. Your Matrix client (Element) may cache the old device keys and refuse to share encryption sessions with the bot.

Symptoms: The bot connects and shows "E2EE enabled" in the logs, but all messages show "could not decrypt event" and the bot never responds.

What's happening: The old encryption state (from the previous matrix-nio or serialization-based mautrix backend) is incompatible with the new SQLite crypto store. The bot creates a fresh encryption identity, but your Matrix client still has the old keys cached and won't share the room's encryption session with a device whose keys changed. This is a Matrix security feature -- clients treat changed identity keys for the same device as suspicious.

Fix (one-time migration):

  1. Generate a new access token to get a fresh device ID. The simplest way:

    curl -X POST https://your-server/_matrix/client/v3/login \
      -H "Content-Type: application/json" \
      -d '{
        "type": "m.login.password",
        "identifier": {"type": "m.id.user", "user": "@hermes:your-server.org"},
        "password": "***",
        "initial_device_display_name": "Hermes Agent"
      }'
    

    Copy the new access_token and update MATRIX_ACCESS_TOKEN in ~/.hermes/.env.

  2. Delete old encryption state:

    rm -f ~/.hermes/platforms/matrix/store/crypto.db
    rm -f ~/.hermes/platforms/matrix/store/crypto_store.*
    
  3. Set your recovery key (if you use cross-signing — most Element users do). Add to ~/.hermes/.env:

    MATRIX_RECOVERY_KEY=EsT... your recovery key here
    

    This lets the bot self-sign with cross-signing keys on startup, so Element trusts the new device immediately. Without this, Element may see the new device as unverified and refuse to share encryption sessions. Find your recovery key in Element under SettingsSecurity & PrivacyEncryption.

  4. Force your Matrix client to rotate the encryption session. In Element, open the DM room with the bot and type /discardsession. This forces Element to create a new encryption session and share it with the bot's new device.

  5. Restart the gateway:

    hermes gateway run
    

    If MATRIX_RECOVERY_KEY is set, you should see Matrix: cross-signing verified via recovery key in the logs.

  6. Send a new message. The bot should decrypt and respond normally.

:::note After migration, messages sent before the upgrade cannot be decrypted -- the old encryption keys are gone. This only affects the transition; new messages work normally. :::

:::tip New installations are not affected. This migration is only needed if you had a working E2EE setup with a previous version of Hermes and are upgrading.

Why a new access token? Each Matrix access token is bound to a specific device ID. Reusing the same device ID with new encryption keys causes other Matrix clients to distrust the device (they see changed identity keys as a potential security breach). A new access token gets a new device ID with no stale key history, so other clients trust it immediately. :::

Proxy Mode (E2EE on macOS)

Matrix E2EE requires libolm, which doesn't compile on macOS ARM64 (Apple Silicon). The hermes-agent[matrix] extra is gated to Linux only. If you're on macOS, proxy mode lets you run E2EE in a Docker container on a Linux VM while the actual agent runs natively on macOS with full access to your local files, memory, and skills.

How It Works

macOS (Host):
  └─ hermes gateway
       ├─ api_server adapter ← listens on 0.0.0.0:8642
       ├─ AIAgent ← single source of truth
       ├─ Sessions, memory, skills
       └─ Local file access (Obsidian, projects, etc.)

Linux VM (Docker):
  └─ hermes gateway (proxy mode)
       ├─ Matrix adapter ← E2EE decryption/encryption
       └─ HTTP forward → macOS:8642/v1/chat/completions
           (no LLM API keys, no agent, no inference)

The Docker container only handles Matrix protocol + E2EE. When a message arrives, it decrypts it and forwards the text to the host via a standard HTTP request. The host runs the agent, calls tools, generates a response, and streams it back. The container encrypts and sends the response to Matrix. All sessions are unified — CLI, Matrix, Telegram, and any other platform share the same memory and conversation history.

Step 1: Configure the Host (macOS)

Enable the API server so the host accepts incoming requests from the Docker container.

Add to ~/.hermes/.env:

API_SERVER_ENABLED=true
API_SERVER_KEY=your-secret-key-here
API_SERVER_HOST=0.0.0.0
  • API_SERVER_HOST=0.0.0.0 binds to all interfaces so the Docker container can reach it.
  • API_SERVER_KEY is required for non-loopback binding. Pick a strong random string.
  • The API server runs on port 8642 by default (change with API_SERVER_PORT if needed).

Start the gateway:

hermes gateway

You should see the API server start alongside any other platforms you have configured. Verify it's reachable from the VM:

# From the Linux VM
curl http://<mac-ip>:8642/health

Step 2: Configure the Docker Container (Linux VM)

The container needs Matrix credentials and the proxy URL. It does NOT need LLM API keys.

docker-compose.yml:

services:
  hermes-matrix:
    build: .
    environment:
      # Matrix credentials
      MATRIX_HOMESERVER: "https://matrix.example.org"
      MATRIX_ACCESS_TOKEN: "syt_..."
      MATRIX_ALLOWED_USERS: "@you:matrix.example.org"
      MATRIX_ENCRYPTION: "true"
      MATRIX_DEVICE_ID: "HERMES_BOT"

      # Proxy mode — forward to host agent
      GATEWAY_PROXY_URL: "http://192.168.1.100:8642"
      GATEWAY_PROXY_KEY: "your-secret-key-here"
    volumes:
      - ./matrix-store:/root/.hermes/platforms/matrix/store

Dockerfile:

FROM python:3.11-slim

RUN apt-get update && apt-get install -y libolm-dev && rm -rf /var/lib/apt/lists/*
RUN pip install 'hermes-agent[matrix]'

CMD ["hermes", "gateway"]

That's the entire container. No API keys for OpenRouter, Anthropic, or any inference provider.

Step 3: Start Both

  1. Start the host gateway first:

    hermes gateway
    
  2. Start the Docker container:

    docker compose up -d
    
  3. Send a message in an encrypted Matrix room. The container decrypts it, forwards it to the host, and streams the response back.

Configuration Reference

Proxy mode is configured on the container side (the thin gateway):

Setting Description
GATEWAY_PROXY_URL URL of the remote Hermes API server (e.g., http://192.168.1.100:8642)
GATEWAY_PROXY_KEY Bearer token for authentication (must match API_SERVER_KEY on the host)
gateway.proxy_url Same as GATEWAY_PROXY_URL but in config.yaml

The host side needs:

Setting Description
API_SERVER_ENABLED Set to true
API_SERVER_KEY Bearer token (shared with the container)
API_SERVER_HOST Set to 0.0.0.0 for network access
API_SERVER_PORT Port number (default: 8642)

Works for Any Platform

Proxy mode is not limited to Matrix. Any platform adapter can use it — set GATEWAY_PROXY_URL on any gateway instance and it will forward to the remote agent instead of running one locally. This is useful for any deployment where the platform adapter needs to run in a different environment from the agent (network isolation, E2EE requirements, resource constraints).

:::tip Session continuity is maintained via the X-Hermes-Session-Id header. The host's API server tracks sessions by this ID, so conversations persist across messages just like they would with a local agent. :::

:::note Limitations (v1): Tool progress messages from the remote agent are not relayed back — the user sees the streamed final response only, not individual tool calls. Dangerous command approval prompts are handled on the host side, not relayed to the Matrix user. These can be addressed in future updates. :::

Bot connects and sends, but ignores inbound messages

Cause: Matrix event handlers only fire when sync payloads are dispatched through mautrix's handle_sync() machinery. A raw client.sync() poll that never calls handle_sync() can leave the adapter connected (send works) while inbound messages never reach _on_room_message.

Fix: Hermes uses an explicit sync loop that calls client.handle_sync() on both the initial sync and every incremental sync response. This matches the diagnosis in upstream issue #7914 and closed PR #37807, but keeps Hermes's own background maintenance tasks (joined-room tracking, invite handling, E2EE key share) instead of delegating the full lifecycle to client.start(). If inbound messages still fail after a gateway restart, verify handlers are registered before the first sync and check logs for sync event dispatch error.

Sync issues / bot falls behind

Cause: Long-running tool executions can delay the sync loop, or the homeserver is slow.

Fix: The sync loop automatically retries every 5 seconds on error. Check the Hermes logs for sync-related warnings. If the bot consistently falls behind, ensure your homeserver has adequate resources.

Bot is offline

Cause: The Hermes gateway isn't running, or it failed to connect.

Fix: Check that hermes gateway is running. Look at the terminal output for error messages. Common issues: wrong homeserver URL, expired access token, homeserver unreachable.

"User not allowed" / Bot ignores you

Cause: Your User ID isn't in MATRIX_ALLOWED_USERS.

Fix: Add your User ID to MATRIX_ALLOWED_USERS in ~/.hermes/.env and restart the gateway. Use the full @user:server format.

Bot ignores an entire room

Cause: MATRIX_ALLOWED_ROOMS is set and the current room ID is not listed, or the room requires a mention and the message did not mention the bot.

Fix: Add the room ID to MATRIX_ALLOWED_ROOMS, or remove the room allowlist if this is a personal deployment. To find a Room ID in Element, open room settings and check Advanced.

Bridge messages loop or echo

Cause: A bridge/appservice puppet is relaying bot output back as a new user message, or a bridge uses non-standard ghost user IDs.

Fix: Keep bridge ghosts out of MATRIX_ALLOWED_USERS, add a matching MATRIX_IGNORE_USER_PATTERNS entry, and leave MATRIX_PROCESS_NOTICES=false unless notices are part of a trusted workflow.

Security

:::warning Always set MATRIX_ALLOWED_USERS and, for shared/private deployments, MATRIX_ALLOWED_ROOMS. Without them, anyone who can message the bot in a joined room may trigger the agent. Only authorize people and rooms you trust — authorized users have full access to the agent's capabilities, including tool use and system access. :::

For more information on securing your Hermes Agent deployment, see the Security Guide.

Notes

  • Any homeserver: Works with Synapse, Conduit, Dendrite, matrix.org, or any spec-compliant Matrix homeserver. No specific homeserver software required.
  • Federation: If you're on a federated homeserver, the bot can communicate with users from other servers — just add their full @user:server IDs to MATRIX_ALLOWED_USERS.
  • Auto-join: The bot automatically accepts room invites and joins. It starts responding immediately after joining.
  • Media support: Hermes can send and receive images, audio, video, and file attachments. Media is uploaded to your homeserver using the Matrix content repository API.
  • Native voice messages (MSC3245): The Matrix adapter automatically tags outgoing voice messages with the org.matrix.msc3245.voice flag. This means TTS responses and voice audio are rendered as native voice bubbles in Element and other clients that support MSC3245, rather than as generic audio file attachments. Incoming voice messages with the MSC3245 flag are also correctly identified and routed to speech-to-text transcription. No configuration is needed — this works automatically.