loading…
Search for a command to run...
loading…
An MCP server that enables interaction with WhatsApp using the Baileys library and Streamable HTTP transport. It supports managing contacts, chats, and messages
An MCP server that enables interaction with WhatsApp using the Baileys library and Streamable HTTP transport. It supports managing contacts, chats, and messages, while providing a web admin UI for QR code authentication and media handling.
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": {
"whatsapp-mcp-stream": {
"command": "npx",
"args": []
}
}
}