Click here to Skip to main content
65,938 articles
CodeProject is changing. Read more.
Articles / desktop / Win32

Sending WhatsApp Messages from a Win32 C++ Program - Part 2

4.98/5 (17 votes)
13 Dec 2018CPOL3 min read 21.9K   1.6K  
A simple way for sending WhatsApp documents and images to an individual or to a group in C++

Won the Best Article of November 2018 3rd prize

 

Introduction

This article is the second part, following the first part. In this part, I will explain how to send images and documents to a group. As mentioned in part 1, there are several service providers and we have chosen one of them (WhatsAppMate) and started a free trial. However, their code samples for using their services are in almost any programming language except for C++. So we wrote our own C++ class for that purpose. Sending documents and files is a bit more complicated and will be explained in this article.

Background

WhatsApp is a multi-platform free service for chatting via video or voice and for sending messages to individuals or groups, including files, media, etc. WhatsApp is better than the old SMS because it is free and has more features. During our day to day work, we need to set up all sort of alerts to be sent to a group who share a job or work on a feature and when that's done automatically, it makes life easier to be notified.

Using the Code

For the GitHub repo, see this one (the class) and this one (the MFC based tool).

There are several types of messages that can be sent, including individual or group messages and with or without a photo or PDF file attached to them. You can read about it in this page.

Here are the building blocks you need to have. Alternatively, I will later attach a compiled (.exe) test application.

Image 1Image 2

C++
//
#define    GroupAdmin            <YOUR GROUP ADMIN MOBILE PHONE>
#define GroupName                <YOUR GROUP NAME>
#define CLIENT_ID                <YOUR CLIENT ID>
#define CLIENT_SECRET            <YOUR CLIENT SECRET>

#define GROUP_API_SERVER        L"api.whatsmate.net"
#define GROUP_API_PATH          L"/v3/whatsapp/group/text/message/12"
#define IMAGE_SINGLE_API_URL    L"http://api.whatsmate.net/v3/whatsapp/group/image/message/12"
//

Our User Interface

For the purpose of this article, we added some User Interface.

Image 3

As you can see, you can enter a message and select a document or an image to be sent with it. When you select a document or an image, the message appears as the caption of the image / document sent.

Sending Images and Documents

When images or documents are sent, we can add a "caption" to them which can be our message.

There is a different API for sending documents, as they appear as "attachment" on the recipient's phone. Images, however appear inline as part of the message. So in terms of the API, there is a difference between:

Our tool, checks the file's type and uses the API accordingly.

Converting an Image or a Document to base64

Here is how we convert an image or a document into base64 string. We will be using CryptBinaryToString().

C++
std::wstring document = _T("");
DWORD desiredLength = 0;
CryptBinaryToString(attachment, length, CRYPT_STRING_BASE64, NULL, &desiredLength);
desiredLength++;
TCHAR* base64content = (TCHAR*)malloc(desiredLength * sizeof(TCHAR));
CryptBinaryToString(attachment, length, CRYPT_STRING_BASE64, base64content, &desiredLength);

Connecting to the Internet

First, we open an Internet connection and connect to the API service:

C++
hOpenHandle = InternetOpen(_T("HTTP"), INTERNET_OPEN_TYPE_DIRECT, NULL, NULL, 0);
if (hOpenHandle == NULL)
{
    return false;
}

hConnectHandle = InternetConnect(hOpenHandle,
    GROUP_API_SERVER,
    INTERNET_DEFAULT_HTTP_PORT,
    NULL, NULL, INTERNET_SERVICE_HTTP,
    0, 1);

if (hConnectHandle == NULL)
{
    InternetCloseHandle(hOpenHandle);
    return false;
}

Opening a Request

C++
const wchar_t *AcceptTypes[] = { _T("application/json"),NULL };
HINTERNET hRequest = HttpOpenRequest(hConnectHandle, _T("POST"),
                     DOCUMENT_API_PATH, NULL, NULL, AcceptTypes, 0, 0);

if (hRequest == NULL)
{
    InternetCloseHandle(hConnectHandle);
    InternetCloseHandle(hOpenHandle);
    return false;
}

or when we wish to send an image:

C++
const wchar_t *AcceptTypes[] = { _T("application/json"),NULL };
HINTERNET hRequest = HttpOpenRequest(hConnectHandle, _T("POST"),
                     IMAGE_API_PATH, NULL, NULL, AcceptTypes, 0, 0);

if (hRequest == NULL)
{
    InternetCloseHandle(hConnectHandle);
    InternetCloseHandle(hOpenHandle);
    return false;
}

while we define the API paths as follows:

C++
#define DOCUMENT_API_PATH       L"/v3/whatsapp/group/document/message/12"
#define IMAGE_API_PATH          L"/v3/whatsapp/group/image/message/12"

The Header

Next, we post the Header which is composed using the following code:

C++
std::wstring HeaderData;

HeaderData += _T("X-WM-CLIENT-ID: ");
HeaderData += _T(CLIENT_ID);
HeaderData += _T("\r\nX-WM-CLIENT-SECRET: ");
HeaderData += _T(CLIENT_SECRET);
HeaderData += _T("\r\n");
HttpAddRequestHeaders(hRequest, HeaderData.c_str(), HeaderData.size(), NULL);

The SendGroupDocument() Function

Here is the entire function for sending a document to a WhatsApp group:

C++
bool SGWhatsApp::SendGroupDocument(LPCTSTR ClientID, LPCTSTR ClientSecret, 
    LPCTSTR groupAdmin, LPCTSTR groupName, /*LPCTSTR message,*/ LPCTSTR filename, 
    LPBYTE attachment, int length)
{
    BOOL bResults = FALSE;
    HINTERNET hOpenHandle, hConnectHandle;
    const TCHAR* szHeaders = _T("Content-Type:application/json; charset=utf-8\r\n");


    hOpenHandle = InternetOpen(_T("HTTP"), INTERNET_OPEN_TYPE_DIRECT, NULL, NULL, 0);
    if (hOpenHandle == NULL)
    {
        return bResults;
    }

    hConnectHandle = InternetConnect(hOpenHandle,
        DOCUMENT_API_SERVER,
        INTERNET_DEFAULT_HTTP_PORT,
        NULL, NULL, INTERNET_SERVICE_HTTP,
        0, 1);

    if (hConnectHandle == NULL)
    {
        InternetCloseHandle(hOpenHandle);
        return bResults;
    }

    const wchar_t *AcceptTypes[] = { _T("application/json"),NULL };
    HINTERNET hRequest = HttpOpenRequest(hConnectHandle, _T("POST"), DOCUMENT_API_PATH, 
                         NULL, NULL, AcceptTypes, 0, 0);

    if (hRequest == NULL)
    {
        InternetCloseHandle(hConnectHandle);
        InternetCloseHandle(hOpenHandle);
        return bResults;
    }

    std::wstring HeaderData;

    HeaderData += _T("X-WM-CLIENT-ID: ");
    HeaderData += ClientID;
    HeaderData += _T("\r\nX-WM-CLIENT-SECRET: ");
    HeaderData += ClientSecret;
    HeaderData += _T("\r\n");
    HttpAddRequestHeaders(hRequest, HeaderData.c_str(), HeaderData.size(), NULL);

    std::wstring document = _T("");
    DWORD desiredLength = 0;
    CryptBinaryToString(attachment, length, CRYPT_STRING_BASE64, NULL, &desiredLength);
    desiredLength++;
    TCHAR* base64content = (TCHAR*)malloc(desiredLength * sizeof(TCHAR));
    CryptBinaryToString(attachment, length, CRYPT_STRING_BASE64, base64content, &desiredLength);

    std::wstring WJsonData;
    WJsonData += _T("{");
    WJsonData += _T("\"group_admin\":\"");
    WJsonData += groupAdmin;
    WJsonData += _T("\",");
    WJsonData += _T("\"group_name\":\"");
    WJsonData += groupName;
    WJsonData += _T("\",");

    WJsonData += _T("\"filename\":\"");
    WJsonData += filename;
    WJsonData += _T("\",");

    WJsonData += _T("\"document\":\"");    
    // Needed to remove CRLF and all spaces symbols
    for(size_t i=0; i<lstrlen(base64content); i++)
        if (!isspace(base64content[i]))
            WJsonData += base64content[i];
    WJsonData += _T("\"");

    WJsonData += _T("}");

    free(base64content);
    const std::string JsonData(WJsonData.begin(), WJsonData.end());

    bResults = HttpSendRequest(hRequest, NULL, 0, (LPVOID)(JsonData.c_str()), JsonData.size());

    TCHAR StatusText[BUFFER_LENGTH] = { 0 };
    DWORD StatusTextLen = BUFFER_LENGTH;
    HttpQueryInfo(hRequest, HTTP_QUERY_STATUS_TEXT, &StatusText, &StatusTextLen, NULL);
    bResults = (StatusTextLen && wcscmp(StatusText, L"OK") == FALSE);

    DWORD availableBytes = 0;
    DWORD downloadedBytes = 0;
    LPBYTE nextBytes = (LPBYTE)malloc((availableBytes +1) * sizeof(TCHAR));
    memset(nextBytes, 0, (availableBytes + 1) * sizeof(TCHAR));
    InternetQueryDataAvailable(hRequest, &availableBytes, 0, 0);
    InternetReadFile(hRequest, nextBytes, availableBytes, &downloadedBytes);
    free(nextBytes);

    InternetCloseHandle(hConnectHandle);
    InternetCloseHandle(hOpenHandle);

    return bResults;
}

The recipient will see a WhatsApp message like this:

Image 4

But to be sure, at this point, we need to know if everything went well. That is done by querying the result of our Http Request.

The StatusText we expect if it worked is "OK".

C++
TCHAR StatusText[BUFFER_LENGTH] = { 0 };
DWORD StatusTextLen = BUFFER_LENGTH;
HttpQueryInfo(hRequest, HTTP_QUERY_STATUS_TEXT, &StatusText, &StatusTextLen, NULL);
bResults = (StatusTextLen && wcscmp(StatusText, L"OK")==FALSE);

The SendGroupImage() Function

Next, we have the SendGroupImage() function which is used to send an image to a WhatsApp group. An image can be a .png or .jpg, which will be displayed inline as a new message.

C++
bool SGWhatsApp::SendGroupDocument(LPCTSTR ClientID, LPCTSTR ClientSecret, 
    LPCTSTR groupAdmin, LPCTSTR groupName, /*LPCTSTR message,*/ LPCTSTR filename, 
    LPBYTE attachment, int length)
{
    BOOL bResults = FALSE;
    HINTERNET hOpenHandle, hConnectHandle;
    const TCHAR* szHeaders = _T("Content-Type:application/json; charset=utf-8\r\n");

    hOpenHandle = InternetOpen(_T("HTTP"), INTERNET_OPEN_TYPE_DIRECT, NULL, NULL, 0);
    if (hOpenHandle == NULL)
    {
        return bResults;
    }

    hConnectHandle = InternetConnect(hOpenHandle,
        DOCUMENT_API_SERVER,
        INTERNET_DEFAULT_HTTP_PORT,
        NULL, NULL, INTERNET_SERVICE_HTTP,
        0, 1);

    if (hConnectHandle == NULL)
    {
        InternetCloseHandle(hOpenHandle);
        return bResults;
    }

    const wchar_t *AcceptTypes[] = { _T("application/json"),NULL };
    HINTERNET hRequest = HttpOpenRequest(hConnectHandle, _T("POST"), 
                         DOCUMENT_API_PATH, NULL, NULL, AcceptTypes, 0, 0);

    if (hRequest == NULL)
    {
        InternetCloseHandle(hConnectHandle);
        InternetCloseHandle(hOpenHandle);
        return bResults;
    }

    std::wstring HeaderData;

    HeaderData += _T("X-WM-CLIENT-ID: ");
    HeaderData += ClientID;
    HeaderData += _T("\r\nX-WM-CLIENT-SECRET: ");
    HeaderData += ClientSecret;
    HeaderData += _T("\r\n");
    HttpAddRequestHeaders(hRequest, HeaderData.c_str(), HeaderData.size(), NULL);

    std::wstring document = _T("");
    DWORD desiredLength = 0;
    CryptBinaryToString(attachment, length, CRYPT_STRING_BASE64, NULL, &desiredLength);
    desiredLength++;
    TCHAR* base64content = (TCHAR*)malloc(desiredLength * sizeof(TCHAR));
    CryptBinaryToString(attachment, length, CRYPT_STRING_BASE64, base64content, &desiredLength);

    std::wstring WJsonData;
    WJsonData += _T("{");
    WJsonData += _T("\"group_admin\":\"");
    WJsonData += groupAdmin;
    WJsonData += _T("\",");
    WJsonData += _T("\"group_name\":\"");
    WJsonData += groupName;
    WJsonData += _T("\",");
    //WJsonData += _T("\"message\":\"");
    //WJsonData += message;
    //WJsonData += _T("\"");

    WJsonData += _T("\"filename\":\"");
    WJsonData += filename;
    WJsonData += _T("\",");

    WJsonData += _T("\"document\":\"");    
    // Needed to remove CRLF and all spaces symbols
    for(size_t i=0; i<lstrlen(base64content); i++)
        if (!isspace(base64content[i]))
            WJsonData += base64content[i];
    WJsonData += _T("\"");

    WJsonData += _T("}");

    free(base64content);
    const std::string JsonData(WJsonData.begin(), WJsonData.end());

    bResults = HttpSendRequest(hRequest, NULL, 0, (LPVOID)(JsonData.c_str()), JsonData.size());

    TCHAR StatusText[BUFFER_LENGTH] = { 0 };
    DWORD StatusTextLen = BUFFER_LENGTH;
    HttpQueryInfo(hRequest, HTTP_QUERY_STATUS_TEXT, &StatusText, &StatusTextLen, NULL);
    bResults = (StatusTextLen && wcscmp(StatusText, L"OK") == FALSE);

    DWORD availableBytes = 0;
    DWORD downloadedBytes = 0;
    LPBYTE nextBytes = (LPBYTE)malloc((availableBytes +1) * sizeof(TCHAR));
    memset(nextBytes, 0, (availableBytes + 1) * sizeof(TCHAR));
    InternetQueryDataAvailable(hRequest, &availableBytes, 0, 0);
    InternetReadFile(hRequest, nextBytes, availableBytes, &downloadedBytes);
    free(nextBytes);

    InternetCloseHandle(hConnectHandle);
    InternetCloseHandle(hOpenHandle);

    return bResults;
}

Thank you

Thanks for Ivan Voloschuk for his help. 

History

  • 6th November, 2018: Initial version

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)