Allows interaction with WhatsApp to manage contacts, search and list chats, send and receive text messages, and handle media uploads and downloads.
Click on "Install Server".
Wait a few minutes for the server to deploy. Once ready, it will show a "Started" state.
In the chat, type
@followed by the MCP server name and your instructions, e.g., "@WhatsApp MCP Streamsend a message to John Smith saying I'm on my way"
That's it! The server will respond to your query, and you can continue using it as needed.
Here is a step-by-step guide with screenshots.
WhatsApp MCP Stream
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:
Transport: Streamable HTTP at
/mcpEngine: Baileys
Admin UI: QR, status, logout, runtime settings
Media: upload endpoints +
/mediahosting + MCP download tool
Quick Start (Docker)
# build and run
docker compose build
docker compose up -dThe server will be available at:
Admin UI:
http://localhost:3003/adminMCP endpoint:
http://localhost:3003/mcpMedia files:
http://localhost:3003/media/<filename>
Runtime Settings
Settings can be edited in the admin UI and are persisted to SETTINGS_PATH (defaults to MEDIA_DIR/settings.json).
Admin UI
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_mb
Authentication
Built-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-gatewayMedia Upload API
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.
Upload Auth (Optional)
If require_upload_token=true, provide a token with either:
x-upload-token: <token>Authorization: Bearer <token>
MCP Transport
The server exposes Streamable HTTP at /mcp.
Typical flow:
POST /mcpwith JSON-RPCinitializeUse the returned
mcp-session-idheader for subsequent requestsPOST /mcpfor tool calls
Note: clients must send Accept: application/json, text/event-stream on initialize.
Smoke Test
Quick regression smoke for MCP tools:
npm run smoke:mcpOptional custom target:
MCP_BASE_URL=http://localhost:3003 npm run smoke:mcpMCP Tools
Auth
Tool | Description |
| Get the latest WhatsApp QR code as an image for authentication. |
| Check if the WhatsApp client is authenticated and ready. |
| Logout from WhatsApp and clear the current session. |
Contacts
Tool | Description |
| Search contacts by name or phone number. |
| Resolve a contact by name or phone number (best matches). |
| Get contact details by JID. |
| Get profile picture URL for a JID. |
| Get group metadata and participants by group JID. |
Chats
Tool | Description |
| List chats with metadata and optional last message. |
| Get chat metadata by JID. |
| List group chats only. |
| Resolve a direct chat JID by phone number. |
| Resolve a contact by name or phone number and return chat metadata. |
| Find members that appear across multiple groups. |
| Find group members with no direct chat. |
| Find group members missing from contacts. |
| Run combined group audit as one routine operation. |
Messages
Tool | Description |
| Get messages from a specific chat. |
| Search messages by text (optionally scoped to a chat). |
| Get a specific message by ID ( |
| Get recent messages around a specific message. |
| Get the most recent message for a JID. |
| Send a text message to a person or group. Supports optional |
Media
Tool | Description |
| Send media (image/video/document/audio). Supports optional |
| Download media from a message. |
Utility
Tool | Description |
| Health check tool. |
Recovery Notes
This service contains an intentional recovery workaround for Baileys/WhatsApp session-state corruption.
Why it exists:
In production we observed cases where the container stayed alive and MCP still answered, but the WhatsApp session was functionally broken.
The most common indicators were Baileys errors like
failed to find key ... to decode mutationandfailed to sync state from version.In that state, a manual container restart often restored service.
Current behavior:
On app-state corruption signals, the service first tries a soft recovery with
forceResync().If the same class of failure repeats within a time window, it escalates to an internal WhatsApp client restart.
On disconnects such as
Connection Terminated, the service schedules a disconnect watchdog and escalates to an internal restart if the socket does not return toopenin time.The reconnect lifecycle is guarded against nested lock deadlocks, so disconnect recovery can complete without requiring a manual container restart.
Recent production observations show repeated socket disconnects (
428 Connection Terminated,503 Stream Errored) being auto-recovered back toopen.A dedicated
/healthzendpoint reports503only when the service is genuinely stuck outside the allowed recovery window.Docker health checks use
/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.
License
MIT
Persistence
Chats and messages are persisted to a local SQLite database stored in the session volume.
Environment variables:
Variable | Default | Description |
|
| SQLite database path for chats/messages persistence. |
|
| Enable detailed WhatsApp event logs. |
|
| Write raw Baileys event stream to a file for deep debugging. |
|
| File path for the event stream log. |
|
| Enable reconnect safety net after force resync. |
|
| Delay before reconnect after force resync (ms). |
|
| Minimum delay between automatic app-state recoveries. |
|
| Time window used to count repeated app-state corruption failures. |
|
| Number of soft recoveries before escalating to an internal restart. |
|
| Grace period during recovery/disconnect before |
|
| How long to wait after a socket close before the disconnect watchdog forces reconnect/restart. |
|
| Comma-separated disconnect status codes that should escalate straight to an internal restart watchdog. |
|
| Suppress exact duplicate |
|
| How long completed |
|
| Max in-memory entries for message index ( |
|
| Max in-memory entries for message key index ( |
|
| Use direct JSON responses for Streamable HTTP POST requests by default. Set to |
Additional transport diagnostics:
/mcpPOST requests now log request lifecycle events inlogs/mcp-whatsapp.logthis includes request entry, transport dispatch,
transport.handleRequestcompletion, and HTTPfinish/closeuse these logs to determine whether latency happens before the response leaves
whatsapp-mcp-streamor after that on the gateway/client side
Export
Export 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.