loading…
Search for a command to run...
loading…
WhatsApp MCP server over Streamable HTTP with web admin UI (QR/status/settings), bidirectional media upload/download, and SQLite persistence.
WhatsApp MCP server over Streamable HTTP with web admin UI (QR/status/settings), bidirectional media upload/download, and SQLite persistence.
A WhatsApp MCP server built around Streamable HTTP transport, using Baileys for WhatsApp connectivity, with a web admin UI and bidirectional media flow (upload + download).
Key points:
/mcp/media hosting + MCP download tool# build and run
docker compose build
docker compose up -d
The server will be available at:
http://localhost:3003/adminhttp://localhost:3003/mcphttp://localhost:3003/media/<filename>Settings can be edited in the admin UI and are persisted to SETTINGS_PATH (defaults to MEDIA_DIR/settings.json).
Admin console with runtime settings, QR linking, export, and status.
Supported settings:
media_public_base_urlupload_max_mbupload_enabledmax_files_per_uploadrequire_upload_tokenupload_tokenauto_download_mediaauto_download_max_mbBuilt-in authentication is not implemented yet. In production, use a gateway that enforces auth. This project works well behind authmcp-gateway:
https://github.com/loglux/authmcp-gateway
Base64 JSON:
curl -X POST http://localhost:3003/api/upload \
-H "Content-Type: application/json" \
-d {filename:photo.jpg,mime_type:image/jpeg,data:<base64>}
Multipart (recommended for large files):
curl -X POST http://localhost:3003/api/upload-multipart \
-F "file=@/path/to/file.jpg"
Both return url and (if configured) publicUrl.
If require_upload_token=true, provide a token with either:
x-upload-token: <token>Authorization: Bearer <token>The server exposes Streamable HTTP at /mcp.
Typical flow:
POST /mcp with JSON-RPC initializemcp-session-id header for subsequent requestsPOST /mcp for tool callsNote: clients must send Accept: application/json, text/event-stream on initialize.
Quick regression smoke for MCP tools:
npm run smoke:mcp
Optional custom target:
MCP_BASE_URL=http://localhost:3003 npm run smoke:mcp
| Tool | Description |
|---|---|
get_qr_code |
Get the latest WhatsApp QR code as an image for authentication. |
check_auth_status |
Check if the WhatsApp client is authenticated and ready. |
logout |
Logout from WhatsApp and clear the current session. |
| Tool | Description |
|---|---|
search_contacts |
Search contacts by name or phone number. |
resolve_contact |
Resolve a contact by name or phone number (best matches). |
get_contact_by_id |
Get contact details by JID. |
get_profile_pic |
Get profile picture URL for a JID. |
get_group_info |
Get group metadata and participants by group JID. |
| Tool | Description |
|---|---|
list_chats |
List chats with metadata and optional last message. |
get_chat_by_id |
Get chat metadata by JID. |
list_groups |
List group chats only. |
get_direct_chat_by_contact_number |
Resolve a direct chat JID by phone number. |
get_chat_by_contact |
Resolve a contact by name or phone number and return chat metadata. |
analyze_group_overlaps |
Find members that appear across multiple groups. |
find_members_without_direct_chat |
Find group members with no direct chat. |
find_members_not_in_contacts |
Find group members missing from contacts. |
run_group_audit |
Run combined group audit as one routine operation. |
| Tool | Description |
|---|---|
list_messages |
Get messages from a specific chat. |
search_messages |
Search messages by text (optionally scoped to a chat). |
get_message_by_id |
Get a specific message by ID (jid:id). |
get_message_context |
Get recent messages around a specific message. |
get_last_interaction |
Get the most recent message for a JID. |
send_message |
Send a text message to a person or group. Supports optional idempotency_key. |
| Tool | Description |
|---|---|
send_media |
Send media (image/video/document/audio). Supports optional idempotency_key. |
download_media |
Download media from a message. |
| Tool | Description |
|---|---|
ping |
Health check tool. |
This service contains an intentional recovery workaround for Baileys/WhatsApp session-state corruption.
Why it exists:
failed to find key ... to decode mutation and failed to sync state from version.Current behavior:
forceResync().Connection Terminated, the service schedules a disconnect watchdog and escalates to an internal restart if the socket does not return to open in time.428 Connection Terminated, 503 Stream Errored) being auto-recovered back to open./healthz endpoint reports 503 only when the service is genuinely stuck outside the allowed recovery window./healthz, so the container is restarted only after in-process recovery has had a chance to work.These recovery mechanisms reduce operator intervention and improve resilience against common WhatsApp/Baileys session failures.
MIT
Chats and messages are persisted to a local SQLite database stored in the session volume.
Environment variables:
| Variable | Default | Description |
|---|---|---|
DB_PATH |
<SESSION_DIR>/store.sqlite |
SQLite database path for chats/messages persistence. |
WA_EVENT_LOG |
0 |
Enable detailed WhatsApp event logs. |
WA_EVENT_STREAM |
0 |
Write raw Baileys event stream to a file for deep debugging. |
WA_EVENT_STREAM_PATH |
/app/logs/wa-events.log |
File path for the event stream log. |
WA_RESYNC_RECONNECT |
1 |
Enable reconnect safety net after force resync. |
WA_RESYNC_RECONNECT_DELAY_MS |
15000 |
Delay before reconnect after force resync (ms). |
WA_SYNC_RECOVERY_COOLDOWN_MS |
300000 |
Minimum delay between automatic app-state recoveries. |
WA_SYNC_RECOVERY_WINDOW_MS |
900000 |
Time window used to count repeated app-state corruption failures. |
WA_SYNC_SOFT_RECOVERY_LIMIT |
2 |
Number of soft recoveries before escalating to an internal restart. |
WA_READINESS_GRACE_MS |
180000 |
Grace period during recovery/disconnect before /healthz turns unhealthy. |
WA_DISCONNECT_RECOVERY_DELAY_MS |
30000 |
How long to wait after a socket close before the disconnect watchdog forces reconnect/restart. |
WA_DISCONNECT_RECOVERY_RESTART_CODES |
428 |
Comma-separated disconnect status codes that should escalate straight to an internal restart watchdog. |
WA_SEND_DEDUP_WINDOW_MS |
45000 |
Suppress exact duplicate send_message requests to the same JID within this window. |
WA_IDEMPOTENCY_TTL_MS |
86400000 |
How long completed send_message idempotency records are retained in SQLite for safe retries. |
WA_MESSAGE_INDEX_MAX |
20000 |
Max in-memory entries for message index (jid:id -> raw message). |
WA_MESSAGE_KEY_INDEX_MAX |
20000 |
Max in-memory entries for message key index (id -> raw message). |
MCP_HTTP_ENABLE_JSON_RESPONSE |
1 |
Use direct JSON responses for Streamable HTTP POST requests by default. Set to 0 to force the older SSE-style POST response handling. |
Additional transport diagnostics:
/mcp POST requests now log request lifecycle events in logs/mcp-whatsapp.logtransport.handleRequest completion, and HTTP finish / closewhatsapp-mcp-stream or after that on the gateway/client sideExport a chat (JSON + optional downloaded media) via:
GET /api/export/chat/:jid?include_media=true
If include_media=true, the ZIP includes files already downloaded via download_media. It does not fetch missing media from WhatsApp.
Добавь это в claude_desktop_config.json и перезапусти Claude Desktop.
{
"mcpServers": {
"loglux-whatsapp-mcp-stream": {
"command": "npx",
"args": []
}
}
}