API Documentation
Complete guide for integrating with the Zapini API
Base URL
https://instance-{id}.zapini.app
Authentication
Bearer Token (Sanctum)
Rate Limit
100 requests / 15 min
Getting Started
The Zapini REST API allows you to integrate WhatsApp messaging capabilities into your applications. All requests must use HTTPS and include Bearer token authentication.
API Types
Zapini offers two types of APIs for different needs. Choose the one that best fits your use case.
Universal API
https://instance-{id}.zapini.app
Complete REST API to manage all your instances, conversations, contacts, and settings. Use this API for web/mobile applications that need full system access.
Instance API
https://instance-{id}.zapini.app/
Direct API for sending messages with low latency. Each WhatsApp instance has its own dedicated URL. Ideal for bots and high-volume automations.
Which API should I use?
| Use Case | Recommended API |
|---|---|
| Web/mobile application | Universal (Sanctum Token) |
| Message bot | Instance (sk_ Token) |
| CRM / Support system | Universal (Sanctum Token) |
| Bulk messaging / campaigns | Instance (sk_ Token) |
| Simple external integration | Instance (sk_ Token) |
Instance ID
The instance UUID (used in Universal API) can be found on the Instances page in the admin panel. Copy the ID shown on each instance card.
Required Headers
Authorization: Bearer {token}
Accept: application/json
Content-Type: application/json
Response Format
Success Response
{
"success": true,
"message": "Operation completed",
"data": { ... }
}
Error Response
{
"success": false,
"error": {
"code": "ERROR_CODE",
"message": "Error description",
"details": {}
}
}
HTTP Status Codes
| Code | Description |
|---|---|
| 200 | OK - Successful request |
| 201 | Created - Resource created successfully |
| 400 | Bad Request - Invalid parameters |
| 401 | Unauthorized - Invalid or missing token |
| 403 | Forbidden - No permission for this resource |
| 404 | Not Found - Resource not found |
| 422 | Unprocessable Entity - Validation error |
| 429 | Too Many Requests - Rate limit exceeded |
| 500 | Internal Server Error - Server error |
Authentication
Token Types
There are two types of authentication tokens, each suited for different scenarios.
Sanctum Token
User token obtained via login. Gives access to all instances and resources in your tenant. Ideal for web/mobile applications where the user logs in.
- • How to get: POST /auth/login with email and password
- • Format: 1|abc123xyz...
- • Access: All tenant instances
Instance Token (sk_)
Token specific to a WhatsApp instance. Generated in the admin panel. Ideal for external integrations and bots.
- • How to get: Generate in panel at Instances > API Tokens
- • Format: sk_xxxxxxxxxxxxxxxx
- • Access: Only the specific instance
Both tokens work the same way in the Authorization header:
Authorization: Bearer {token}
/auth/login
Authenticates a user and returns an access token.
Parameters
| Field | Type | Required | Description |
|---|---|---|---|
email |
string | Yes | User email |
password |
string | Yes | User password |
device_name |
string | No | Device name for token identification |
Example
curl -X POST https://instance-{id}.zapini.app/auth/login \
-H "Content-Type: application/json" \
-H "Accept: application/json" \
-d '{
"email": "user@example.com",
"password": "your_password",
"device_name": "My App"
}'
Response
{
"success": true,
"message": "Login successful",
"data": {
"user": {
"id": 1,
"name": "User Name",
"email": "user@example.com",
"role": "admin"
},
"token": "1|abc123xyz...",
"token_type": "Bearer"
}
}
/auth/me
Returns information about the authenticated user.
Example
curl -X GET https://instance-{id}.zapini.app/auth/me \
-H "Authorization: Bearer {token}" \
-H "Accept: application/json"
/auth/logout
Revokes the current access token.
Example
curl -X POST https://instance-{id}.zapini.app/auth/logout \
-H "Authorization: Bearer {token}" \
-H "Accept: application/json"
/auth/refresh
Refreshes the access token.
Example
curl -X POST https://instance-{id}.zapini.app/auth/refresh \
-H "Authorization: Bearer {token}" \
-H "Accept: application/json"
Instances
/instances
Lists all WhatsApp instances in your account.
Query Parameters
| Field | Type | Description |
|---|---|---|
status |
string | connected, disconnected, qr_ready, pending |
Example
curl -X GET "https://instance-{id}.zapini.app/instances?status=connected" \
-H "Authorization: Bearer {token}" \
-H "Accept: application/json"
Response
{
"success": true,
"data": {
"instances": [
{
"uuid": "550e8400-e29b-41d4-a716-446655440000",
"phone_number": "5511999999999",
"status": "connected",
"connected": true,
"messages_sent_today": 50,
"remaining_quota": 450,
"can_send_messages": true,
"last_activity_at": "2025-01-15T10:30:00Z"
}
]
}
}
/instances/{uuid}
Returns details of a specific instance.
Example
curl -X GET https://instance-{id}.zapini.app/instances/550e8400-e29b-41d4-a716-446655440000 \
-H "Authorization: Bearer {token}" \
-H "Accept: application/json"
/instances/{uuid}/qr
Returns the QR code to connect the instance.
Example
curl -X GET https://instance-{id}.zapini.app/instances/550e8400-e29b-41d4-a716-446655440000/qr \
-H "Authorization: Bearer {token}" \
-H "Accept: application/json"
Response
{
"success": true,
"data": {
"status": "qr_ready",
"qr_code": "data:image/png;base64,...",
"qr_code_text": "2@...",
"qr_generated_at": "2025-01-15T10:30:00Z",
"qr_expired": false,
"message": "QR code ready to scan"
}
}
/instances/{uuid}/status
Returns the current status of the instance.
Query Parameters
| Field | Type | Description |
|---|---|---|
realtime |
boolean | If true, checks status in real-time on the server |
Example
curl -X GET "https://instance-{id}.zapini.app/instances/550e8400-e29b-41d4-a716-446655440000/status?realtime=true" \
-H "Authorization: Bearer {token}" \
-H "Accept: application/json"
/instances/{uuid}/stats
Returns usage statistics for the instance.
Response
{
"success": true,
"data": {
"messages_sent_today": 50,
"remaining_quota": 450,
"daily_limit": 500,
"messages_this_week": 200,
"messages_this_month": 800,
"messages_received_today": 30,
"conversations_count": 150,
"contacts_count": 500
}
}
/instances/{uuid}/disconnect
Disconnects the WhatsApp instance.
Example
curl -X POST https://instance-{id}.zapini.app/instances/550e8400-e29b-41d4-a716-446655440000/disconnect \
-H "Authorization: Bearer {token}" \
-H "Accept: application/json"
/instances/{uuid}/reconnect
Initiates the instance reconnection process.
Example
curl -X POST https://instance-{id}.zapini.app/instances/550e8400-e29b-41d4-a716-446655440000/reconnect \
-H "Authorization: Bearer {token}" \
-H "Accept: application/json"
Messages
Instance API (Baileys)
Direct API for sending messages with low latency. Each WhatsApp instance has its own dedicated URL. Ideal for bots and high-volume automations.
Base URL:
https://instance-{id}.zapini.app
Universal API - /chat/send
Para enviar mensagens via API Universal, use o endpoint /chat/send (consulte a aba Chat API).
/send-message
Sends a text message to a WhatsApp number.
Parameters
| Field | Type | Required | Description |
|---|---|---|---|
recipient
(ou number, to)
|
string | Yes | Recipient WhatsApp number (with country code) (5511999999999) |
message |
string | Yes | Text message content |
reply_to
(ou quoted_message_id)
|
string | No | Message ID to reply to |
recipient, number e to são equivalentes.
Assim como reply_to e quoted_message_id.
Example
curl -X POST https://instance-{id}.zapini.app/send-message \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-d '{
"recipient": "5511999999999",
"message": "Olá! Esta é uma mensagem de teste."
}'
Response
{
"success": true,
"message": "Message sent successfully",
"message_id": "3EB0B430A8B7F23C1D12",
"timestamp": "2025-12-30T17:30:00.000Z"
}
/send-media
Sends a message with media (image, video, audio, document).
Parameters
| Field | Type | Required | Description |
|---|---|---|---|
recipient
(ou number, to)
|
string | Yes | Recipient WhatsApp number (with country code) (5511999999999) |
media_url
(ou url)
|
string | Yes | Public URL of the media file |
media_type
(ou type)
|
string | Yes | image, video, audio, document |
caption |
string | No | Caption for images and videos |
filename |
string | No | Filename for documents |
recipient/number/to,
media_url/url e
media_type/type são equivalentes.
Example
curl -X POST https://instance-{id}.zapini.app/send-media \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-d '{
"recipient": "5511999999999",
"media_url": "https://example.com/image.jpg",
"media_type": "image",
"caption": "Confira esta imagem!"
}'
Response
{
"success": true,
"message": "Media sent successfully",
"message_id": "3EB0B430A8B7F23C1D13",
"timestamp": "2025-12-30T17:30:00.000Z"
}
/send-reaction
Sends a reaction emoji to a specific message.
Parameters
| Field | Type | Required | Description |
|---|---|---|---|
to |
string | Yes | JID do chat |
messageId |
string | Yes | ID da mensagem para reagir |
emoji |
string | Yes | Emoji da reação (ex: 👍, ❤️, 😂) |
Example
curl -X POST https://instance-{id}.zapini.app/send-reaction \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-d '{
"to": "5511999999999@s.whatsapp.net",
"messageId": "3EB0B430A8B7F23C1D12",
"emoji": "👍"
}'
/status
Returns the current connection status of the WhatsApp instance.
Example
curl -X GET https://instance-{id}.zapini.app/status \
-H "Authorization: Bearer {token}"
Response
{
"success": true,
"status": "connected",
"phone": "5511999999999",
"name": "Meu WhatsApp",
"platform": "android"
}
/qr
Returns the QR code to connect the WhatsApp instance. Only available when status is qr_ready.
Example
curl -X GET https://instance-{id}.zapini.app/qr \
-H "Authorization: Bearer {token}"
Response
{
"success": true,
"qr": "...",
"status": "qr_ready"
}
/health
Returns the health status of the API server. No authentication required.
Example
curl -X GET https://instance-{id}.zapini.app/health
Response
{
"status": "ok",
"uptime": 123456,
"timestamp": "2025-12-15T14:30:00.000Z"
}
Conversations
/conversations
Lists all active conversations.
Query Parameters
| Field | Type | Description |
|---|---|---|
instance_id |
string | Filter by instance (UUID) |
unread |
boolean | Show only unread conversations |
search |
string | Search by name or number |
is_group |
boolean | Filter by groups (true/false) |
per_page |
integer | Items per page (max 100) |
include_tags |
boolean | Include conversation tags in response |
tag_id |
uuid | Filter by tag ID |
tag_ids |
uuid[] | Filter by multiple tag UUIDs (comma-separated or array) |
untagged |
boolean | Only show conversations without tags |
format |
string | Use chat-optimized response format (chat) |
?format=chat: The Chat API returns the same data as standard endpoints but with additional fields optimized for UI rendering. You can also use ?format=chat on standard endpoints.
Example
curl -X GET "https://instance-{id}.zapini.app/conversations?unread=true&per_page=20" \
-H "Authorization: Bearer {token}" \
-H "Accept: application/json"
Response
{
"success": true,
"data": [
{
"id": "550e8400-e29b-41d4-a716-446655440045",
"contact_number": "5511999999999",
"contact_name": "John Doe",
"display_name": "John Doe",
"is_group": false,
"profile_picture_url": "https://...",
"last_message": "Ok, deal!",
"last_message_at": "2025-01-15T10:30:00Z",
"unread_count": 2,
"is_archived": false
}
],
"meta": {
"pagination": {
"total": 100,
"per_page": 20,
"current_page": 1,
"last_page": 5
}
}
}
/conversations/{uuid}
Returns details of a specific conversation.
/conversations/{uuid}/messages
Returns message history for a conversation.
Query Parameters
| Field | Type | Description |
|---|---|---|
from |
datetime | Start date (ISO 8601) |
to |
datetime | End date (ISO 8601) |
before_id |
integer | Messages before this ID |
after_id |
integer | Messages after this ID |
limit |
integer | Message limit (max 100) |
format |
string | Use chat-optimized response format (chat) |
?format=chat: Returns messages with formatted_body (resolved @mentions) and grouped reactions.
/conversations/{uuid}/mark-read
Marks a conversation as read.
/conversations/{uuid}/archive
Archives a conversation.
Response
{
"success": true,
"message": "Conversation archived.",
"data": {
"archived_id": "550e8400-e29b-41d4-a716-446655440045",
"archived_at": "2025-12-29T12:00:00.000000Z"
}
}
/conversations/{uuid}/archive
Restores an archived conversation back to active status.
The UUID must be from an archived conversation. Use GET /conversations/archived to list archived conversations.
Response
{
"success": true,
"message": "Conversation restored.",
"data": {
"conversation_id": "550e8400-e29b-41d4-a716-446655440045",
"restored": true
}
}
/conversations/{uuid}
Deletes a conversation and its messages.
Automation Control
Manage AI automation for conversations. When automation is active, manual messages are blocked until you take over the chat.
Important: Automation Behavior
- When automation is active, manual messages are blocked to prevent conflicts with AI responses.
- Use "pause" to take over the chat and send messages manually.
- Use "resume" to let the AI handle the conversation again.
- IMPORTANT: To resume automation, you must first archive the conversation.
Security: Archive Required to Resume
Automation can only be resumed after archiving the conversation. This ensures the manual service session is properly closed before returning control to AI. When resuming, the conversation will be automatically unarchived.
/conversations/{uuid}/automation
Returns the current automation status for a conversation.
Response
{
"success": true,
"data": {
"conversation_id": "550e8400-e29b-41d4-a716-446655440045",
"automation_status": "active",
"is_automation_active": true,
"can_send_manual_message": false,
"paused_at": null,
"paused_by": null,
"pause_reason": null
}
}
/conversations/{uuid}/automation/pause
Pauses automation for a conversation, allowing manual messages.
Request Body
| Field | Type | Required | Description |
|---|---|---|---|
reason |
string | No | Optional reason for pausing automation |
Example
curl -X POST "https://instance-{id}.zapini.app/conversations/550e8400-e29b-41d4-a716-446655440045/automation/pause" \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-d '{"reason": "Manual takeover for VIP client"}'
Response
{
"success": true,
"message": "Automation paused. You can now send messages manually.",
"data": {
"automation_status": "paused",
"paused_at": "2025-12-20T14:30:00Z",
"paused_by": "John Doe"
}
}
/conversations/{uuid}/automation/resume
Resumes automation for a conversation, letting AI handle responses.
Response
{
"success": true,
"message": "Automation resumed. AI will handle this conversation.",
"data": {
"automation_status": "active"
}
}
/conversations/automation/bulk-pause
Pauses automation for multiple conversations at once.
Request Body
| Field | Type | Required | Description |
|---|---|---|---|
conversation_ids |
array | Yes | Array of conversation IDs to process |
reason |
string | No | Optional reason for pausing automation |
Example
curl -X POST "https://instance-{id}.zapini.app/conversations/automation/bulk-pause" \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-d '{"conversation_ids": ["550e8400-e29b-41d4-a716-446655440001", "550e8400-e29b-41d4-a716-446655440002", "550e8400-e29b-41d4-a716-446655440003"]}'
Response
{
"success": true,
"message": "3 conversations paused successfully."
}
/conversations/automation/bulk-resume
Resumes automation for multiple conversations at once.
Request Body
| Field | Type | Required | Description |
|---|---|---|---|
conversation_ids |
array | Yes | Array of conversation IDs to process |
Response
{
"success": true,
"message": "3 conversations resumed successfully."
}
Contacts
/contacts
Lists all saved contacts.
Query Parameters
| Field | Type | Description |
|---|---|---|
instance_id |
string | Filter by instance (UUID) |
search |
string | Search by name or number |
per_page |
integer | Items per page (max 100) |
Example
curl -X GET "https://instance-{id}.zapini.app/contacts?search=john&per_page=50" \
-H "Authorization: Bearer {token}" \
-H "Accept: application/json"
Response
{
"success": true,
"data": [
{
"id": 1,
"phone_number": "5511999999999",
"name": "John Doe",
"display_name": "John",
"email": "john@example.com",
"company": "Acme Inc",
"notes": "VIP customer",
"profile_picture_url": "https://...",
"is_business": false,
"created_at": "2025-01-15T10:30:00Z"
}
],
"meta": {
"pagination": {
"total": 500,
"per_page": 50,
"current_page": 1,
"last_page": 10
}
}
}
/contacts
Creates a new contact.
Parameters
| Field | Type | Required | Description |
|---|---|---|---|
instance_id |
string | Yes | ID da instância |
phone_number |
string | Yes | Contact phone number |
name |
string | No | Contact name |
email |
string | No | Contact email |
company |
string | No | Contact company |
notes |
string | No | Notes about the contact |
Example
curl -X POST https://instance-{id}.zapini.app/contacts \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-d '{
"instance_id": "550e8400-e29b-41d4-a716-446655440000",
"phone_number": "+5511999999999",
"name": "John Doe",
"email": "john@example.com",
"company": "Acme Inc",
"notes": "Met at conference"
}'
/contacts/{id}
Returns details of a specific contact.
/contacts/{id}
Updates an existing contact.
Parameters
| Field | Type | Description |
|---|---|---|
name |
string | Contact name |
email |
string | Contact email |
company |
string | Contact company |
notes |
string | Notes about the contact |
/contacts/{id}
Deletes a contact.
/contacts/import
Imports multiple contacts at once.
Parameters
| Field | Type | Required | Description |
|---|---|---|---|
instance_id |
string | Yes | ID da instância |
contacts |
array | Yes | Array of contact objects |
Example
curl -X POST https://instance-{id}.zapini.app/contacts/import \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-d '{
"instance_id": "550e8400-e29b-41d4-a716-446655440000",
"contacts": [
{"phone_number": "+5511999999999", "name": "John Doe"},
{"phone_number": "+5511888888888", "name": "Jane Smith"}
]
}'
Response
{
"success": true,
"message": "Contacts imported successfully",
"data": {
"imported": 2,
"skipped": 0,
"errors": []
}
}
/contacts/export
Exports contacts in JSON or CSV format.
Query Parameters
| Field | Type | Description |
|---|---|---|
instance_id |
string | Filter by instance (UUID) |
format |
string | json, csv (default: json) |
Example
curl -X GET "https://instance-{id}.zapini.app/contacts/export?format=csv" \
-H "Authorization: Bearer {token}" \
-H "Accept: application/json"
Webhooks
Webhook System
Webhooks allow you to receive real-time notifications about events in your WhatsApp instances.
Incoming Webhooks
URLs that Zapini uses to receive data from your WhatsApp instances.
Outgoing Webhooks
URLs you configure to receive notifications from Zapini.
Outgoing Webhooks
Receive real-time notifications on your server
Configure webhooks to receive real-time notifications when events occur in your WhatsApp instance. Zapini will send HTTP POST requests to your configured URL with event data.
How to Configure
- Go to your API Token settings and click "Configure Webhook"
- Enter your webhook URL (must be HTTPS in production)
- Generate or enter a secret key for signature verification
- Select which events you want to receive and save
Available Events
| Event | Description |
|---|---|
message.received |
New message received |
message.sent |
Message sent |
message.status |
Message status changed (sent, delivered, read) |
instance.connected |
Instance connected to WhatsApp |
instance.disconnected |
Instance disconnected |
instance.qr |
New QR code generated |
Understanding Message IDs
Each message has two identifiers. Use the correct one depending on your use case:
| Field | Format | Usage |
|---|---|---|
message_id |
UUID (e.g., 055f01b2-da7b-476b-...) |
Primary key for deduplication and tracking in your system |
whatsapp_message_id |
WhatsApp ID (e.g., 3EB0ABC123456789) |
Required for reactions, replies, and WhatsApp operations |
Tip: How to prevent duplicates
Store message_id when you receive a webhook. Before processing new webhooks, check if message_id already exists in your database. If it does, update the existing record instead of creating a new one.
Request Format
Each webhook request is sent as an HTTP POST with the following headers:
POST https://your-server.com/webhook
Content-Type: application/json
X-Webhook-Signature: a1b2c3d4e5f6...
X-Webhook-Event: message.received
X-Instance-UUID: 550e8400-e29b-41d4-a716-446655440000
User-Agent: Zapini-Webhook/1.0
Webhook Headers
| Header | Description |
|---|---|
X-Webhook-Signature |
HMAC-SHA256 signature of the JSON payload using your secret key |
X-Webhook-Event |
The event type that triggered this webhook |
X-Instance-UUID |
UUID of the WhatsApp instance |
User-Agent |
Zapini-Webhook/1.0 |
Payload Examples
message.received
{
"event": "message.received",
"instance_uuid": "550e8400-e29b-41d4-a716-446655440000",
"timestamp": "2025-01-15T10:30:00+00:00",
"data": {
"message_id": "055f01b2-da7b-476b-9a5d-dcecb3fe510c",
"whatsapp_message_id": "3EB0ABC123456789",
"sender_number": "5511999999999",
"sender_name": "John Doe",
"content": "Hello, I need help!",
"media_type": null,
"media_url": null,
"timestamp": "2025-01-15T10:30:00+00:00",
"conversation_id": 123
}
}
message.sent
{
"event": "message.sent",
"instance_uuid": "550e8400-e29b-41d4-a716-446655440000",
"timestamp": "2025-01-15T10:31:00+00:00",
"data": {
"message_id": "f4a2e8c9-7b3d-4e5f-8a1c-2d9e6f0a3b7c",
"whatsapp_message_id": "3EB0ABC123456789",
"recipient_number": "5511888888888",
"content": "Thanks for contacting us!",
"media_type": null,
"media_url": null,
"timestamp": "2025-01-15T10:31:00+00:00",
"conversation_id": 123
}
}
message.status
{
"event": "message.status",
"instance_uuid": "550e8400-e29b-41d4-a716-446655440000",
"timestamp": "2025-01-15T10:32:00+00:00",
"data": {
"message_id": "f4a2e8c9-7b3d-4e5f-8a1c-2d9e6f0a3b7c",
"whatsapp_message_id": "3EB0ABC123456789",
"previous_status": "sent",
"new_status": "delivered",
"recipient_number": "5511888888888",
"timestamp": "2025-01-15T10:32:00+00:00"
}
}
instance.connected
{
"event": "instance.connected",
"instance_uuid": "550e8400-e29b-41d4-a716-446655440000",
"timestamp": "2025-01-15T10:00:00+00:00",
"data": {
"previous_status": "qr_ready",
"new_status": "connected",
"phone_number": "5511999999999"
}
}
/webhooks/instance/{uuid}/incoming
Receives incoming messages from WhatsApp instances.
Payload Format
{
"remoteJid": "5511999999999@s.whatsapp.net",
"message": {
"conversation": "Hello, how are you?",
"extendedTextMessage": null,
"imageMessage": null,
"videoMessage": null,
"audioMessage": null,
"documentMessage": null
},
"messageTimestamp": 1705312200,
"pushName": "John Doe",
"key": {
"remoteJid": "5511999999999@s.whatsapp.net",
"fromMe": false,
"id": "3EB0ABC123456789"
},
"participant": null,
"isGroup": false
}
Message Types
| Type | Field | Description |
|---|---|---|
| Text | message.conversation |
Simple text message |
| Extended Text | message.extendedTextMessage |
Text with link preview |
| Image | message.imageMessage |
Image with caption |
| Video | message.videoMessage |
Video message |
| Audio | message.audioMessage |
Audio/voice message |
| Document | message.documentMessage |
Document/file |
Example - Text Message
curl -X POST https://zapini.app/webhooks/instance/{uuid}/incoming \
-H "Content-Type: application/json" \
-d '{
"remoteJid": "5511999999999@s.whatsapp.net",
"message": {
"conversation": "Hello!"
},
"messageTimestamp": 1705312200,
"pushName": "Customer Name",
"key": {
"remoteJid": "5511999999999@s.whatsapp.net",
"fromMe": false,
"id": "ABC123"
}
}'
/webhooks/instance/{uuid}/status
Receives status updates for sent messages.
Payload Format
{
"key": {
"remoteJid": "5511999999999@s.whatsapp.net",
"id": "3EB0ABC123456789",
"fromMe": true
},
"status": 3,
"messageTimestamp": 1705312200
}
Status Codes
| Code | Status | Description |
|---|---|---|
1 |
pending | Message pending to be sent |
2 |
sent | Message sent to server |
3 |
delivered | Message delivered to recipient |
4 |
read | Message read by recipient |
5 |
played | Audio/video played |
/webhooks/instance/{uuid}/qr
Receives QR code updates for connection.
Payload Format
{
"qr": "2@ABC123...",
"qr_image": "...",
"generated_at": "2025-01-15T10:30:00Z"
}
/webhooks/instance/{uuid}/connection
Receives instance connection status updates.
Payload Format
{
"status": "connected",
"phone_number": "5511999999999",
"timestamp": "2025-01-15T10:30:00Z"
}
Connection Statuses
| Status | Description |
|---|---|
| connected | Instance connected and working |
| qr_ready | QR code ready to scan |
| disconnected | Instance disconnected |
| offline | Instance offline |
/webhooks/instance/{uuid}/reaction
Receives emoji reactions on messages.
Payload Format
{
"key": {
"remoteJid": "5511999999999@s.whatsapp.net",
"id": "3EB0ABC123456789",
"fromMe": false
},
"reaction": {
"text": "👍",
"key": {
"id": "3EB0DEF987654321"
}
},
"senderJid": "5511888888888@s.whatsapp.net"
}
/instances/{uuid}/webhook
Configure an external URL to receive event notifications.
Parameters
| Field | Type | Required | Description |
|---|---|---|---|
webhook_url |
string | Yes | URL that will receive notifications via POST |
webhook_secret |
string | No | Secret key for request signing |
events |
array | No | Events that will trigger the webhook |
is_active |
boolean | No | Enable or disable the webhook |
Available Events
Example
curl -X PUT https://instance-{id}.zapini.app/instances/550e8400-e29b-41d4-a716-446655440000/webhook \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-d '{
"webhook_url": "https://your-server.com/webhook",
"webhook_secret": "your_secret_key_here",
"events": ["message.incoming", "message.status"],
"is_active": true
}'
Webhook Security
All webhook requests include an HMAC signature for authenticity verification.
Signature Verification
Each request includes an X-Webhook-Signature header with the HMAC-SHA256 signature of the payload.
Header:
X-Webhook-Signature: a1b2c3d4e5f6789...
Verify Signature (PHP)
<?php
$payload = file_get_contents('php://input');
$signature = $_SERVER['HTTP_X_WEBHOOK_SIGNATURE'] ?? '';
$secret = 'your_webhook_secret';
$expected = hash_hmac('sha256', $payload, $secret);
if (hash_equals($expected, $signature)) {
// Signature is valid
$data = json_decode($payload, true);
// Process webhook...
} else {
http_response_code(401);
exit('Invalid signature');
}
Verify Signature (Node.js)
const crypto = require('crypto');
app.post('/webhook', (req, res) => {
const signature = req.headers['x-webhook-signature'];
const secret = 'your_webhook_secret';
const payload = JSON.stringify(req.body);
const expected = crypto
.createHmac('sha256', secret)
.update(payload)
.digest('hex');
if (crypto.timingSafeEqual(
Buffer.from(signature),
Buffer.from(expected)
)) {
// Signature is valid
console.log('Webhook received:', req.body);
res.status(200).send('OK');
} else {
res.status(401).send('Invalid signature');
}
});
Best Practices
- Always verify the signature before processing the webhook.
- Respond quickly (within 5 seconds) to avoid timeouts.
- Use a queue to process webhooks asynchronously.
- Implement idempotency to handle duplicate webhooks.
- Log all received webhooks for debugging.
Media
Media Management
Send and manage media files such as images, videos, audio, and documents.
/media/upload
Uploads a media file for later use.
Parameters
| Field | Type | Required | Description |
|---|---|---|---|
file |
file | Yes | Media file for upload (multipart/form-data) |
type |
string | No | image, video, audio, document (auto-detected) |
Example
curl -X POST https://instance-{id}.zapini.app/media/upload \
-H "Authorization: Bearer {token}" \
-F "file=@/path/to/image.jpg" \
-F "type=image"
Response
{
"success": true,
"data": {
"id": "media_abc123xyz",
"url": "https://storage.zapini.app/media/abc123xyz.jpg",
"type": "image",
"mime_type": "image/jpeg",
"size": 102400,
"filename": "image.jpg",
"expires_at": "2025-01-22T10:30:00Z"
}
}
/media/{id}
Returns information about a media file.
Response
{
"success": true,
"data": {
"id": "media_abc123xyz",
"url": "https://storage.zapini.app/media/abc123xyz.jpg",
"type": "image",
"mime_type": "image/jpeg",
"size": 102400,
"filename": "image.jpg",
"created_at": "2025-01-15T10:30:00Z",
"expires_at": "2025-01-22T10:30:00Z"
}
}
/media/{id}
Deletes a media file.
Example
curl -X DELETE https://instance-{id}.zapini.app/media/media_abc123xyz \
-H "Authorization: Bearer {token}"
/messages/send-media
Sends a message with attached media.
Parameters
| Field | Type | Required | Description |
|---|---|---|---|
instance_id |
string | Yes | ID da instância |
recipient |
string | Yes | Recipient WhatsApp number (with country code) |
media_url |
string | Yes* | Public URL of the media file |
media_id |
string | Yes* | ID of a previously uploaded media |
media_type |
string | Yes | image, video, audio, document |
caption |
string | No | Caption for images and videos |
filename |
string | No | Filename for documents |
* Provide media_url OR media_id
Example - Image
curl -X POST https://instance-{id}.zapini.app/messages/send-media \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-d '{
"instance_id": "550e8400-e29b-41d4-a716-446655440000",
"recipient": "+5511999999999",
"media_url": "https://example.com/image.jpg",
"media_type": "image",
"caption": "Check this image!"
}'
Example - Document
curl -X POST https://instance-{id}.zapini.app/messages/send-media \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-d '{
"instance_id": "550e8400-e29b-41d4-a716-446655440000",
"recipient": "+5511999999999",
"media_url": "https://example.com/report.pdf",
"media_type": "document",
"filename": "Monthly_Report.pdf",
"caption": "Here is the monthly report"
}'
Media Limits
| Type | Max Size | Formats |
|---|---|---|
| Images | 16 MB | jpg, jpeg, png, gif, webp |
| Videos | 64 MB | mp4, 3gp, mov, avi, mkv |
| Audio | 16 MB | mp3, ogg, wav, opus, aac, m4a |
| Documents | 100 MB | pdf, doc, docx, xls, xlsx, ppt, pptx, txt, zip |
/messages/{uuid}/media
Downloads media received in a message.
Example
curl -X GET https://instance-{id}.zapini.app/messages/660e8400-e29b-41d4-a716-446655440001/media \
-H "Authorization: Bearer {token}" \
--output downloaded_media.jpg
Response
Returns the binary file with the appropriate Content-Type.
Groups
Manage WhatsApp groups through the API. Create groups, add/remove participants, update settings, and more.
/instances/{uuid}/groups
Returns all WhatsApp groups from the instance.
Example
curl -X GET "https://instance-{id}.zapini.app/instances/{uuid}/groups" \
-H "Authorization: Bearer {token}" \
-H "Accept: application/json"
Response
{
"success": true,
"data": {
"groups": [
{
"jid": "120363001234567890@g.us",
"subject": "Marketing Team",
"owner": "5511987654321@s.whatsapp.net",
"size": 25,
"description": "Team discussions"
}
],
"total": 5
}
}
/instances/{uuid}/groups
Creates a new WhatsApp group.
Request Body
| Field | Type | Required | Description |
|---|---|---|---|
subject |
string | Yes | api-docs.group_subject_desc |
participants |
array | Yes | api-docs.group_participants_desc |
Example
curl -X POST "https://instance-{id}.zapini.app/instances/{uuid}/groups" \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-d '{
"subject": "My New Group",
"participants": ["5511987654321", "5511976543210"]
}'
Response
{
"success": true,
"data": {
"group": {
"jid": "120363001234567890@g.us",
"subject": "My New Group"
},
"message": "Group created successfully"
}
}
/instances/{uuid}/groups/{jid}
Returns detailed metadata of a specific group.
Response
{
"success": true,
"data": {
"jid": "120363001234567890@g.us",
"subject": "Marketing Team",
"description": "Team discussions",
"owner": "5511987654321@s.whatsapp.net",
"participants": [
{"jid": "5511987654321@s.whatsapp.net", "admin": "superadmin"},
{"jid": "5511976543210@s.whatsapp.net", "admin": null}
],
"size": 25
}
}
/instances/{uuid}/groups/{jid}
Updates the group name and/or description.
Request Body
| Field | Type | Description |
|---|---|---|
subject |
string | New group name |
description |
string | New group description |
Example
curl -X PATCH "https://instance-{id}.zapini.app/instances/{uuid}/groups/{jid}" \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-d '{
"subject": "New Group Name",
"description": "Updated description"
}'
/instances/{uuid}/groups/{jid}/participants
Adds one or more participants to the group.
Example
curl -X POST "https://instance-{id}.zapini.app/instances/{uuid}/groups/{jid}/participants" \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-d '{
"participants": ["5511987654321", "5511976543210"]
}'
/instances/{uuid}/groups/{jid}/participants
Removes one or more participants from the group.
Example
curl -X DELETE "https://instance-{id}.zapini.app/instances/{uuid}/groups/{jid}/participants" \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-d '{
"participants": ["5511987654321"]
}'
/instances/{uuid}/groups/{jid}/admins
api-docs.groups_promote_desc
Example
curl -X POST "https://instance-{id}.zapini.app/instances/{uuid}/groups/{jid}/admins" \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-d '{
"participants": ["5511987654321"]
}'
/instances/{uuid}/groups/{jid}/admins
api-docs.groups_demote_desc
Example
curl -X DELETE "https://instance-{id}.zapini.app/instances/{uuid}/groups/{jid}/admins" \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-d '{
"participants": ["5511987654321"]
}'
/instances/{uuid}/groups/{jid}/leave
Makes the instance leave a group.
Example
curl -X POST "https://instance-{id}.zapini.app/instances/{uuid}/groups/{jid}/leave" \
-H "Authorization: Bearer {token}"
/instances/{uuid}/groups/{jid}/invite
api-docs.groups_invite_get_desc
Response
{
"success": true,
"data": {
"code": "AbCdEfGhIjKl",
"invite_link": "https://chat.whatsapp.com/AbCdEfGhIjKl"
}
}
/instances/{uuid}/groups/{jid}/invite
api-docs.groups_invite_revoke_desc
Response
{
"success": true,
"data": {
"new_code": "MnOpQrStUvWx",
"new_invite_link": "https://chat.whatsapp.com/MnOpQrStUvWx"
},
"message": "Invite link revoked"
}
Tags
Manage conversation tags for organizing and categorizing chats. Tags are tenant-scoped and can be assigned to multiple conversations.
/tags
Lists all tags for the current tenant.
Example
curl -X GET "https://instance-{id}.zapini.app/tags" \
-H "Authorization: Bearer {token}" \
-H "Accept: application/json"
Response
{
"success": true,
"data": {
"tags": [
{
"id": "550e8400-e29b-41d4-a716-446655440001",
"name": "VIP",
"color": "#f59e0b",
"conversations_count": 15,
"created_at": "2025-01-15T10:30:00Z",
"updated_at": "2025-01-15T10:30:00Z"
},
{
"id": "550e8400-e29b-41d4-a716-446655440002",
"name": "Support",
"color": "#3b82f6",
"conversations_count": 42,
"created_at": "2025-01-15T10:35:00Z",
"updated_at": "2025-01-15T10:35:00Z"
}
]
}
}
/tags
Creates a new tag.
Request Body
| Field | Type | Required | Description |
|---|---|---|---|
name |
string | Yes | Tag name (unique per tenant) (max 50) |
color |
string | Yes | Hex color code for the tag (#RRGGBB) |
Example
curl -X POST "https://instance-{id}.zapini.app/tags" \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-H "Accept: application/json" \
-d '{
"name": "Important",
"color": "#ef4444"
}'
Response
{
"success": true,
"message": "Tag created successfully",
"data": {
"tag": {
"id": "550e8400-e29b-41d4-a716-446655440003",
"name": "Important",
"color": "#ef4444",
"conversations_count": 0,
"created_at": "2025-01-15T12:00:00Z",
"updated_at": "2025-01-15T12:00:00Z"
}
}
}
/tags/{uuid}
Returns details of a specific tag.
Example
curl -X GET "https://instance-{id}.zapini.app/tags/550e8400-e29b-41d4-a716-446655440001" \
-H "Authorization: Bearer {token}" \
-H "Accept: application/json"
/tags/{uuid}
Updates an existing tag.
Request Body
| Field | Type | Required | Description |
|---|---|---|---|
name |
string | No | Tag name (unique per tenant) (max 50) |
color |
string | No | Hex color code for the tag (#RRGGBB) |
Example
curl -X PATCH "https://instance-{id}.zapini.app/tags/550e8400-e29b-41d4-a716-446655440001" \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-H "Accept: application/json" \
-d '{
"name": "Very Important",
"color": "#dc2626"
}'
/tags/{uuid}
Deletes a tag. The tag will be automatically removed from all conversations.
Example
curl -X DELETE "https://instance-{id}.zapini.app/tags/550e8400-e29b-41d4-a716-446655440001" \
-H "Authorization: Bearer {token}" \
-H "Accept: application/json"
Response
{
"success": true,
"message": "Tag deleted successfully"
}
/conversations/{uuid}/tags
Updates the tags assigned to a conversation. This replaces all existing tags with the provided list.
Request Body
| Field | Type | Required | Description |
|---|---|---|---|
tag_ids |
array | Yes | Array of tag UUIDs to assign to the conversation |
Example
curl -X PUT "https://instance-{id}.zapini.app/conversations/550e8400-e29b-41d4-a716-446655440000/tags" \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-H "Accept: application/json" \
-d '{
"tag_ids": ["550e8400-e29b-41d4-a716-446655440001", "550e8400-e29b-41d4-a716-446655440002"]
}'
Response
{
"success": true,
"message": "Conversation tags updated",
"data": {
"tags": [
{"id": "550e8400-e29b-41d4-a716-446655440001", "name": "VIP", "color": "#f59e0b"},
{"id": "550e8400-e29b-41d4-a716-446655440002", "name": "Support", "color": "#3b82f6"},
{"id": "550e8400-e29b-41d4-a716-446655440003", "name": "Important", "color": "#ef4444"}
]
}
}
To remove all tags from a conversation, send an empty array: {"tag_ids": []}
Chat API
UI-OPTIMIZEDEndpoints optimized for building WhatsApp-like chat interfaces. Returns pre-formatted data with resolved @mentions, grouped reactions, and alignment helpers.
formatted_body
Pre-resolved @mentions in HTML format
grouped_reactions
Grouped reactions with counts
is_from_me
Easy message alignment with is_from_me
tags
Conversation tags included
The Chat API returns the same data as standard endpoints but with additional fields optimized for UI rendering. You can also use ?format=chat on standard endpoints.
/api/v1/conversations?format=chat
/api/v1/chat/conversations
Lists conversations with tags, profile pictures, and unread counts optimized for chat list UI.
Query Parameters
| Field | Type | Description |
|---|---|---|
instance_id |
string | Filter by instance (UUID) |
search |
string | Search by name or number |
unread |
boolean | Show only unread conversations |
per_page |
integer | Items per page (max 100) |
Example
curl -X GET "https://zapini.app/api/v1/chat/conversations?unread=true" \
-H "Authorization: Bearer {token}"
Response
{
"success": true,
"data": [
{
"id": "550e8400-e29b-41d4-a716-446655440045",
"contact_number": "5521999999999",
"display_name": "John Doe",
"is_group": false,
"profile_picture_url": "https://...",
"last_message": "Hello!",
"last_message_at": "2025-12-16T10:00:00Z",
"unread_count": 3,
"tags": [
{"id": "bf287f4d-037c-4280-8413-c7b7d08df5a0", "name": "VIP", "color": "#f59e0b"}
],
"instance": {
"uuid": "21ed5c9b-51a0-4569-85db-05433ea414b7",
"phone_number": "5511988887777",
"status": "connected"
}
}
],
"meta": {
"pagination": {
"total": 50,
"per_page": 15,
"current_page": 1,
"last_page": 4
}
}
}
/api/v1/chat/conversations/{uuid}/messages
Returns messages with formatted_body (resolved @mentions) and grouped reactions.
Query Parameters
| Field | Type | Description |
|---|---|---|
before_id |
integer | Messages before this ID |
after_id |
integer | Messages after this ID |
limit |
integer | Message limit (max 100) |
Example
curl -X GET "https://zapini.app/api/v1/chat/conversations/550e8400-e29b-41d4-a716-446655440045/messages?limit=50" \
-H "Authorization: Bearer {token}"
Response
{
"success": true,
"data": {
"messages": [
{
"id": 456,
"unique_id": "msg-abc-123",
"direction": "incoming",
"is_from_me": false,
"message_body": "Ola @5521999999999!",
"formatted_body": "Ola <span class=\"mention\" data-phone=\"5521999999999\">@John Doe</span>!",
"mentioned_jids": ["5521999999999@s.whatsapp.net"],
"sender_name": "Maria",
"status": "read",
"grouped_reactions": [
{
"emoji": "👍",
"count": 2,
"has_my_reaction": true,
"reactors": ["5521988887777", "5521966665555"],
"reactor_names": ["Carlos", "Ana"]
}
],
"created_at": "2025-12-16T10:00:00Z"
}
],
"has_more": true,
"oldest_id": 400,
"newest_id": 456
}
}
Key Features
formatted_body
Message text with @mentions resolved to contact names. Returns HTML with clickable spans.
Hello @John Smith!
is_from_me
Boolean indicating if the message was sent by your instance. Use for message alignment.
grouped_reactions
Reactions grouped by emoji with counts and reactor information.
{"emoji": "👍", "count": 2, "has_my_reaction": true}
mentioned_jids
Array of WhatsApp JIDs mentioned in the message.
/api/v1/chat/send
Sends a text message and returns the message in chat-optimized format.
Parameters
| Field | Type | Required | Description |
|---|---|---|---|
instance_id |
string | Yes | Instance UUID (required) |
recipient |
string | Yes | Recipient phone number with country code |
message |
string | Yes | Text message content |
reply_to |
string | No | Message ID to reply to |
Example
curl -X POST "https://zapini.app/api/v1/chat/send" \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-d '{
"instance_id": "18",
"recipient": "5521999999999",
"message": "Hello from Chat API!"
}'
/api/v1/chat/send-media
Sends media (image, video, audio, document) and returns chat-optimized response.
Parameters
| Field | Type | Required | Description |
|---|---|---|---|
instance_id |
string | Yes | Instance UUID (required) |
recipient |
string | Yes | Recipient phone number with country code |
media_url |
string | Yes | URL of the media file to send |
media_type |
string | Yes | Type of media (image, video, audio, document) |
caption |
string | No | Caption for image/video |
Example
curl -X POST "https://zapini.app/api/v1/chat/send-media" \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-d '{
"instance_id": "18",
"recipient": "5521999999999",
"media_url": "https://example.com/image.jpg",
"media_type": "image",
"caption": "Check this out!"
}'
/api/v1/chat/messages/{uuid}
Edits a sent message content.
Parameters
| Field | Type | Required | Description |
|---|---|---|---|
message |
string | Yes | New message content |
Example
curl -X PATCH "https://zapini.app/api/v1/chat/messages/msg-abc-123" \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-d '{"message": "Updated message content"}'
/api/v1/chat/messages/{uuid}
Deletes a sent message.
Example
curl -X DELETE "https://zapini.app/api/v1/chat/messages/msg-abc-123" \
-H "Authorization: Bearer {token}"
/api/v1/chat/messages/{uuid}/reaction
Adds an emoji reaction to a message.
Parameters
| Field | Type | Required | Description |
|---|---|---|---|
emoji |
string | Yes | Emoji to react with |
Example
curl -X POST "https://zapini.app/api/v1/chat/messages/msg-abc-123/reaction" \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-d '{"emoji": "👍"}'
/api/v1/chat/messages/{uuid}/reaction
Removes a reaction from a message.
Example
curl -X DELETE "https://zapini.app/api/v1/chat/messages/msg-abc-123/reaction" \
-H "Authorization: Bearer {token}"
/api/v1/chat/conversations/{uuid}/mark-read
Marks a conversation as read.
Example
curl -X POST "https://zapini.app/api/v1/chat/conversations/550e8400-e29b-41d4-a716-446655440045/mark-read" \
-H "Authorization: Bearer {token}"
/api/v1/chat/conversations/{uuid}/archive
Archives a conversation.
Example
curl -X POST "https://zapini.app/api/v1/chat/conversations/550e8400-e29b-41d4-a716-446655440045/archive" \
-H "Authorization: Bearer {token}"
Response
{
"success": true,
"message": "Conversation archived.",
"data": {
"archived_id": "550e8400-e29b-41d4-a716-446655440045",
"archived_at": "2025-12-29T12:00:00.000000Z"
}
}
/api/v1/chat/conversations/{uuid}/archive
Restores an archived conversation back to active status.
The UUID must be from an archived conversation. Use GET /conversations/archived to list archived conversations.
Example
curl -X DELETE "https://zapini.app/api/v1/chat/conversations/550e8400-e29b-41d4-a716-446655440045/archive" \
-H "Authorization: Bearer {token}"
Response
{
"success": true,
"message": "Conversation restored.",
"data": {
"conversation_id": "550e8400-e29b-41d4-a716-446655440045",
"restored": true
}
}
/api/v1/chat/conversations/{uuid}/pause-automation
Pause automation for a conversation to allow manual messaging.
Request Body
| Field | Type | Required | Description |
|---|---|---|---|
reason |
string | No | Optional reason for pausing automation |
Example
curl -X POST "https://zapini.app/api/v1/chat/conversations/550e8400-e29b-41d4-a716-446655440045/pause-automation" \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-d '{"reason": "Manual takeover"}'
/api/v1/chat/conversations/{uuid}/resume-automation
Resume automation for a conversation, returning control to the automated system.
Example
curl -X POST "https://zapini.app/api/v1/chat/conversations/550e8400-e29b-41d4-a716-446655440045/resume-automation" \
-H "Authorization: Bearer {token}"
Automations
IMPORTANTImportant Notice about Automations
If your instance has active automations (flows or integrations), sending messages via API may be blocked. Read this section to understand how it works.
What are Automations?
Automations are automated flows that respond to messages automatically. The system has two types of automation:
Flows
Conversation flows created in the visual editor that respond automatically based on rules and conditions.
Integrations
Connections with external systems (webhooks, APIs, AI) that process and respond to messages automatically.
Message Blocking
When a conversation has active automation, sending messages via API is blocked to avoid conflicts between automatic and manual responses.
Error Response
{
"success": false,
"error": {
"code": "AUTOMATION_ACTIVE",
"message": "Cannot send messages. This conversation has active automation. Pause the automation first to send messages manually.",
"conversation_id": 123
}
}
When is Sending Blocked?
- • The instance has an active flow or integration
- • The conversation was not manually paused
- • The conversation is still being managed by automation
How to Send Messages with Active Automation
To send messages manually when automation is active, you need to pause the automation for the specific conversation:
Check the Status
Use the conversation details endpoint to check the automation_status field.
Pause the Automation
If automation_status is "active", use the pause endpoint to allow manual messages.
Send the Message
After pausing, you can send messages normally via API.
/api/v1/chat/conversations/{uuid}
Check the automation status of a conversation before attempting to send messages.
Response Fields
| Field | Type | Description |
|---|---|---|
automation_status |
string | Current automation status (active or paused) |
can_send_manual_message |
boolean | Whether you can send manual messages |
automation_paused_at |
datetime|null | Date/time when automation was paused |
Status Values
| Status | Description | Can Send via API? |
|---|---|---|
active |
Automation is responding to messages automatically | No |
paused |
Automation paused, user has taken control | Yes |
/api/v1/chat/conversations/{uuid}/pause-automation
Pauses automation to allow sending manual messages via API.
Request Body (optional)
| Field | Type | Description |
|---|---|---|
reason |
string | Optional reason for pausing (e.g., "Manual intervention required") |
Example
curl -X POST https://instance-{id}.zapini.app/api/v1/chat/conversations/a1b2c3d4-e5f6-7890-abcd-ef1234567890/pause-automation \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-d '{"reason": "Manual intervention required"}'
Response Example
{
"success": true,
"message": "Automation paused successfully",
"data": {
"conversation_id": "a1b2c3d4-e5f6-7890-abcd-ef1234567890",
"automation_paused": true,
"paused_at": "2025-01-15T10:30:00+00:00",
"paused_by": "John Doe",
"reason": "Manual intervention required"
}
}
/api/v1/chat/conversations/{uuid}/resume-automation
Resumes automation, returning control to the automated system.
Example
curl -X POST https://instance-{id}.zapini.app/api/v1/chat/conversations/a1b2c3d4-e5f6-7890-abcd-ef1234567890/resume-automation \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json"
Response Example
{
"success": true,
"message": "Automation resumed successfully",
"data": {
"conversation_id": "a1b2c3d4-e5f6-7890-abcd-ef1234567890",
"automation_paused": false,
"resumed_at": "2025-01-15T11:00:00+00:00"
}
}
Bulk Operations
Pause or resume automation for multiple conversations at once. Useful for maintenance, campaigns, or batch processing. Maximum 100 conversations per request.
/api/v1/conversations/automation/bulk-pause
Pauses automation for multiple conversations at once. Skips conversations that are already paused or have no active automation.
Request Body
| Field | Type | Required | Description |
|---|---|---|---|
conversation_ids |
array | Yes | Array of conversation UUIDs (max 100) |
reason |
string | No | Optional reason for pausing (e.g., "Manual intervention required") |
Example
curl -X POST https://instance-{id}.zapini.app/api/v1/conversations/automation/bulk-pause \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-d '{
"conversation_ids": [
"a1b2c3d4-e5f6-7890-abcd-ef1234567890",
"b2c3d4e5-f6a7-8901-bcde-f12345678901"
],
"reason": "Bulk pause for maintenance"
}'
Response
{
"success": true,
"message": "2 conversations paused",
"data": {
"paused": 2,
"skipped": 0,
"errors": 0
}
}
/api/v1/conversations/automation/bulk-resume
Resumes automation for multiple conversations at once. Skips conversations that are not paused or have no active automation.
Request Body
| Field | Type | Required | Description |
|---|---|---|---|
conversation_ids |
array | Yes | Array of conversation UUIDs (max 100) |
Example
curl -X POST https://instance-{id}.zapini.app/api/v1/conversations/automation/bulk-resume \
-H "Authorization: Bearer {token}" \
-H "Content-Type: application/json" \
-d '{
"conversation_ids": [
"a1b2c3d4-e5f6-7890-abcd-ef1234567890",
"b2c3d4e5-f6a7-8901-bcde-f12345678901"
]
}'
Response
{
"success": true,
"message": "2 automations resumed",
"data": {
"resumed": 2,
"skipped": 0,
"errors": 0
}
}
Best Practices
- Always check can_send_manual_message before sending messages to avoid errors.
- Pause automation only when necessary for human intervention.
- Resume automation after manual service to maintain consistent experience.
Ready to Get Started?
Create your account now and start integrating WhatsApp into your applications with our powerful and easy-to-use API.