Skip to main content

Lightweight Webex Mercury WebSocket + KMS decryption for receiving bot messages without the full Webex SDK

Project description

webex-message-handler

Lightweight Webex Mercury WebSocket + KMS decryption for receiving bot messages — no Webex SDK required.

Python port of the TypeScript webex-message-handler.

Why?

  • The Webex Python SDK has heavy dependencies and limited WebSocket support
  • Bots behind corporate firewalls need persistent connections, not webhooks
  • This package extracts only the essential Mercury + KMS logic (~2 dependencies)

Install

pip install webex-message-handler

Quick Start

import asyncio
from webex_message_handler import WebexMessageHandler, WebexMessageHandlerConfig, console_logger

handler = WebexMessageHandler(
    WebexMessageHandlerConfig(
        token="YOUR_BOT_TOKEN",
        logger=console_logger,
    )
)

@handler.on("message:created")
async def on_message(msg):
    print(f"[{msg.person_email}] {msg.text}")
    if msg.html:
        print(f"  HTML: {msg.html}")

@handler.on("message:deleted")
def on_deleted(data):
    print(f"Message {data.message_id} deleted by {data.person_id}")

@handler.on("connected")
def on_connected():
    print("Connected to Webex")

@handler.on("disconnected")
def on_disconnected(reason):
    print(f"Disconnected: {reason}")

@handler.on("reconnecting")
def on_reconnecting(attempt):
    print(f"Reconnecting (attempt {attempt})...")

@handler.on("error")
def on_error(err):
    print(f"Error: {err}")

async def main():
    await handler.connect()
    # Keep running until interrupted
    try:
        await asyncio.Event().wait()
    finally:
        await handler.disconnect()

asyncio.run(main())

See examples/basic_bot.py for a complete working example.

Important: Implementing Loop Detection

This library only handles the receive side of messaging — it decrypts incoming messages from the Mercury WebSocket. It has no visibility into messages your bot sends via the REST API. This means it cannot detect message loops on its own.

If your bot replies to incoming messages, you must implement loop detection in your wrapper code. Without it, a bug or misconfiguration could cause your bot to endlessly reply to its own messages. Webex enforces a server-side rate limit (approximately 11 consecutive messages before throttling), but that still results in spam before the cutoff.

Recommended approach: Track your bot's outgoing message rate. If it exceeds a threshold (e.g., 5 messages in 3 seconds to the same room), pause sending and log a warning.

The ignore_self_messages option (default: True) provides a first line of defense by filtering out messages sent by this bot's own identity. If the library cannot verify the bot's identity during connect() (e.g., /people/me API failure), connection will fail rather than silently running without protection. Set ignore_self_messages=False to opt out, but only if you have your own loop prevention in place.

Proxy Support (Enterprise)

For corporate environments behind a proxy, pass a configured connector:

import aiohttp
from aiohttp_socks import ProxyConnector

# Using HTTP/HTTPS proxy
connector = ProxyConnector.from_url(
    "http://proxy.example.com:8080"
)

handler = WebexMessageHandler(
    WebexMessageHandlerConfig(
        token="YOUR_BOT_TOKEN",
        connector=connector,  # Pass configured connector
        logger=console_logger,
    )
)

await handler.connect()

Or using environment variables:

import os
import aiohttp
from aiohttp_socks import ProxyConnector

proxy_url = os.getenv("HTTPS_PROXY") or os.getenv("HTTP_PROXY")
connector = ProxyConnector.from_url(proxy_url) if proxy_url else None

handler = WebexMessageHandler(
    WebexMessageHandlerConfig(
        token=os.getenv("WEBEX_BOT_TOKEN"),
        connector=connector,
        logger=console_logger,
    )
)

Requires: pip install aiohttp-socks[asyncio]

API Reference

WebexMessageHandler

Main class for receiving and decrypting Webex messages.

Constructor

WebexMessageHandler(config: WebexMessageHandlerConfig)

Configuration options:

Option Type Default Description
token str required Webex bot access token
logger Logger noop Custom logger (console_logger provided)
ignore_self_messages bool True Filter out messages sent by this bot
connector aiohttp.BaseConnector None HTTP/HTTPS connector for proxy support
ping_interval float 15.0 Mercury ping interval (seconds)
pong_timeout float 14.0 Pong response timeout (seconds)
reconnect_backoff_max float 32.0 Max reconnect backoff (seconds)
max_reconnect_attempts int 10 Max reconnect attempts

Methods

  • await connect() — Connects to Webex (registers device, initializes KMS, opens Mercury WebSocket)
  • await disconnect() — Gracefully disconnects (closes WebSocket, unregisters device)
  • await reconnect(new_token) — Update token and re-establish connection
  • status() — Returns HandlerStatus health check
  • connectedbool property: whether currently connected

Events

Event Payload Description
message:created DecryptedMessage New message received and decrypted
message:deleted DeletedMessage Message was deleted
connected Connected/reconnected to Mercury
disconnected reason: str Disconnected from Mercury
reconnecting attempt: int Attempting to reconnect
error Exception Error occurred

DecryptedMessage

@dataclass
class DecryptedMessage:
    id: str
    room_id: str
    person_id: str
    person_email: str
    text: str
    created: str
    html: str | None
    room_type: str | None   # "direct" | "group"
    raw: MercuryActivity | None

Architecture

WebexMessageHandler (orchestrator)
├── DeviceManager  — WDM registration
├── MercurySocket  — WebSocket + ping/pong + reconnect
├── KmsClient      — ECDH handshake + key retrieval
└── MessageDecryptor — JWE decryption

License

MIT

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

webex_message_handler-0.6.3.tar.gz (31.3 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

webex_message_handler-0.6.3-py3-none-any.whl (26.9 kB view details)

Uploaded Python 3

File details

Details for the file webex_message_handler-0.6.3.tar.gz.

File metadata

  • Download URL: webex_message_handler-0.6.3.tar.gz
  • Upload date:
  • Size: 31.3 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for webex_message_handler-0.6.3.tar.gz
Algorithm Hash digest
SHA256 b2555a26b4c81af33c82d42a4217fe1d8423b1b7f68ecf0dee7818c9443bcf44
MD5 304701f288368757d5bf3d6dd836ad10
BLAKE2b-256 c40429c5466c65bc317d54394c2a981dd18e553eb8c625203fb08752b2fda545

See more details on using hashes here.

Provenance

The following attestation bundles were made for webex_message_handler-0.6.3.tar.gz:

Publisher: publish.yml on 3rg0n/webex-message-handler

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file webex_message_handler-0.6.3-py3-none-any.whl.

File metadata

File hashes

Hashes for webex_message_handler-0.6.3-py3-none-any.whl
Algorithm Hash digest
SHA256 bd156628625ca0eb02d020feed4a0cb5049833f37e255ed67bf68662ca53149f
MD5 c3f094580b1dd39f6a8543cca71c18bf
BLAKE2b-256 3c0f612b9b8a06a8b64b366e439ae112876c2a83f5b19f9a0128aee6027cc8d1

See more details on using hashes here.

Provenance

The following attestation bundles were made for webex_message_handler-0.6.3-py3-none-any.whl:

Publisher: publish.yml on 3rg0n/webex-message-handler

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page