Whats91
Developers

Bulk upload contacts into one contact book using a JSON payload.

Upload Contacts

Summary

Bulk upload contacts into one contact book using a JSON payload.

Prerequisites

  • Authorization: Bearer w91_live_xxx
  • Content-Type: application/json for JSON requests

Endpoint: POST /api/v2/contact-books/{bookUid}/contacts/bulk. Bulk upload is JSON-only. CSV/XLSX multipart upload remains dashboard-only for now.

POST/api/v2/contact-books/{bookUid}/contacts/bulk
ParameterTypeRequiredDescription
AuthorizationheaderRequiredBearer w91_public_token_here. Must be a global public API token.
Content-TypeheaderRequiredContent-Type: application/json.
bookUidstringRequiredContact book UID, for example grp_abc.
defaultCountryCodestringOptionalCountry code used to normalize local phone numbers, for example 91.
contactsarrayRequiredArray of contact objects. Maximum 1000 contacts per request.
contacts[].phonestringRequiredContact phone number. It is normalized using defaultCountryCode.
contacts[].displayNamestringOptionalReadable contact name.
contacts[].firstNamestringOptionalOptional first name.
contacts[].lastNamestringOptionalOptional last name.
contacts[].emailstringOptionalOptional email address.
contacts[].companyNamestringOptionalOptional company name.
contacts[].customFieldsobjectOptionalOptional developer-defined metadata.
Bulk upload contacts
curl -X POST "https://graph.whats91.com/api/v2/contact-books/grp_abc/contacts/bulk" \
  -H "Authorization: Bearer w91_public_token_here" \
  -H "Content-Type: application/json" \
  -d '{
    "defaultCountryCode": "91",
    "contacts": [
      {
        "phone": "7000782082",
        "displayName": "Asha Rao",
        "firstName": "Asha",
        "lastName": "Rao",
        "email": "asha@example.com",
        "companyName": "Acme",
        "customFields": {
          "city": "Mumbai"
        }
      }
    ]
  }'
Upload response
{
  "success": true,
  "message": "Contacts uploaded",
  "data": {
    "contactBook": {
      "contactBookUid": "grp_abc",
      "name": "Retail Leads"
    },
    "summary": {
      "total": 3,
      "created": 1,
      "updated": 1,
      "skipped": 1,
      "invalid": 0
    },
    "results": [
      {
        "row": 1,
        "status": "UPDATED",
        "contact": { "contactUid": "ct_existing", "phone": "917000782082" }
      },
      {
        "row": 2,
        "status": "CREATED",
        "contact": { "contactUid": "ct_new", "phone": "917999999999" }
      },
      {
        "row": 3,
        "status": "SKIPPED",
        "reason": "DUPLICATE_IN_REQUEST",
        "phone": "917999999999"
      }
    ]
  }
}

Bulk Upload Behavior

  • Maximum 1000 contacts per request.
  • Phone numbers are normalized using defaultCountryCode.
  • Existing contacts are updated and linked to the contact book.
  • New contacts are created and linked to the contact book.
  • Duplicate numbers inside the same request are skipped with DUPLICATE_IN_REQUEST.
  • Invalid rows do not fail the whole request; they are returned in row-level results.

Validation Errors

HTTPError codeScenario
401MISSING_AUTH_TOKENNo public API token supplied.
401INVALID_AUTH_TOKENToken is invalid, expired, revoked, or user inactive.
403TOKEN_SCOPE_NOT_ALLOWEDToken is number-scoped; contact books require a global token.
403FEATURE_NOT_AVAILABLECustomer subscription does not include contacts.
400VALIDATION_FAILEDInvalid pagination, missing or long name, invalid contacts payload, too many contacts, or invalid custom fields.
404CONTACT_BOOK_NOT_FOUNDContact book UID does not belong to the authenticated customer.
415UNSUPPORTED_CONTENT_TYPEPOST request is not JSON.

SDK Examples

Use these examples as starting points for server-side implementations.

cURL
curl -X POST "https://graph.whats91.com/api/v2/contact-books/{bookUid}/contacts/bulk" \
  -H "Authorization: Bearer w91_live_xxx" \
  -H "Content-Type: application/json" \
  -d 'curl -X POST "https://graph.whats91.com/api/v2/contact-books/grp_abc/contacts/bulk" \
  -H "Authorization: Bearer w91_public_token_here" \
  -H "Content-Type: application/json" \
  -d '{
    "defaultCountryCode": "91",
    "contacts": [
      {
        "phone": "7000782082",
        "displayName": "Asha Rao",
        "firstName": "Asha",
        "lastName": "Rao",
        "email": "asha@example.com",
        "companyName": "Acme",
        "customFields": {
          "city": "Mumbai"
        }
      }
    ]
  }''
Node.js
const response = await fetch("https://graph.whats91.com/api/v2/contact-books/{bookUid}/contacts/bulk", {
  method: "POST",
  headers: {
    "Authorization": "Bearer w91_live_xxx",
    "Content-Type": "application/json"
  },
  body: JSON.stringify(curl -X POST "https://graph.whats91.com/api/v2/contact-books/grp_abc/contacts/bulk" \
  -H "Authorization: Bearer w91_public_token_here" \
  -H "Content-Type: application/json" \
  -d '{
    "defaultCountryCode": "91",
    "contacts": [
      {
        "phone": "7000782082",
        "displayName": "Asha Rao",
        "firstName": "Asha",
        "lastName": "Rao",
        "email": "asha@example.com",
        "companyName": "Acme",
        "customFields": {
          "city": "Mumbai"
        }
      }
    ]
  }')
});

const data = await response.json();
console.log(data);
PHP
$ch = curl_init("https://graph.whats91.com/api/v2/contact-books/{bookUid}/contacts/bulk");
curl_setopt_array($ch, [
  CURLOPT_CUSTOMREQUEST => "POST",
  CURLOPT_RETURNTRANSFER => true,
  CURLOPT_HTTPHEADER => [
    "Authorization: Bearer w91_live_xxx",
    "Content-Type: application/json"
  ],
  CURLOPT_POSTFIELDS => json_encode(curl -X POST "https://graph.whats91.com/api/v2/contact-books/grp_abc/contacts/bulk" \
  -H "Authorization: Bearer w91_public_token_here" \
  -H "Content-Type: application/json" \
  -d '{
    "defaultCountryCode": "91",
    "contacts": [
      {
        "phone": "7000782082",
        "displayName": "Asha Rao",
        "firstName": "Asha",
        "lastName": "Rao",
        "email": "asha@example.com",
        "companyName": "Acme",
        "customFields": {
          "city": "Mumbai"
        }
      }
    ]
  }')
]);

$response = curl_exec($ch);
curl_close($ch);
echo $response;
Python
import requests

response = requests.request(
    "POST",
    "https://graph.whats91.com/api/v2/contact-books/{bookUid}/contacts/bulk",
    headers={
        "Authorization": "Bearer w91_live_xxx",
        "Content-Type": "application/json",
    },
    json=curl -X POST "https://graph.whats91.com/api/v2/contact-books/grp_abc/contacts/bulk" \
  -H "Authorization: Bearer w91_public_token_here" \
  -H "Content-Type: application/json" \
  -d '{
    "defaultCountryCode": "91",
    "contacts": [
      {
        "phone": "7000782082",
        "displayName": "Asha Rao",
        "firstName": "Asha",
        "lastName": "Rao",
        "email": "asha@example.com",
        "companyName": "Acme",
        "customFields": {
          "city": "Mumbai"
        }
      }
    ]
  }'
)

print(response.json())
C#
using var client = new HttpClient();
client.DefaultRequestHeaders.Add("Authorization", "Bearer w91_live_xxx");

var request = new HttpRequestMessage(HttpMethod.Post, "https://graph.whats91.com/api/v2/contact-books/{bookUid}/contacts/bulk");
request.Content = new StringContent(
  """curl -X POST \"https://graph.whats91.com/api/v2/contact-books/grp_abc/contacts/bulk\" \
  -H \"Authorization: Bearer w91_public_token_here\" \
  -H \"Content-Type: application/json\" \
  -d '{
    \"defaultCountryCode\": \"91\",
    \"contacts\": [
      {
        \"phone\": \"7000782082\",
        \"displayName\": \"Asha Rao\",
        \"firstName\": \"Asha\",
        \"lastName\": \"Rao\",
        \"email\": \"asha@example.com\",
        \"companyName\": \"Acme\",
        \"customFields\": {
          \"city\": \"Mumbai\"
        }
      }
    ]
  }'""",
  Encoding.UTF8,
  "application/json"
);

var response = await client.SendAsync(request);
Console.WriteLine(await response.Content.ReadAsStringAsync());

Related APIs