1. Vonage Learn
  2. Blog
  3. 2020
  4. 10
  5. 26
  6. Send Whatsapp Messages With Laravel Dr
How to Send WhatsApp Messages With Laravel

< Tutorial />

How to Send WhatsApp Messages With Laravel

Take two things that we love: WhatsApp messages and Laravel framework. Mix them together. What do you get? Fun with phones!

This tutorial shows you how to create a Laravel application that sends and responds to WhatsApp messages.

You will need:

  • A PHP development platform (Laravel 8 requires PHP 7.3 or later)
  • WhatsApp on your phone

Get the Code Running

The project is on GitHub at nexmo-community/laravel-messages, so go ahead and clone that repository to your computer.

Run the command composer install to get the dependencies needed for this project.

This application uses the phpdotenv library to manage its configuration on a dev platform. Copy the file .env.example to .env and edit the file as needed. In particular, you should update the NEXMO_API_KEY and NEXMO_API_SECRET lines at the end of the file to connect your Vonage account (we used to be called Nexmo, and old habits die hard!)

The application is ready to go! Start it with:

php artisan serve

By default, this will run your application on port 8000. Check that you have a Laravel homepage on http://localhost:8000 before moving on.

When Laravel runs, and is working, it looks like this

Set Up Ngrok

Since the application needs to be able to receive incoming webhooks for two-way messaging communication, we need a way to allow public URLs to access the dev platform. I usually use Ngrok for this; it's an excellent tool.

Start an Ngrok tunnel to port 8000 (or whatever port your application is running on):

ngrok http 8000

This starts an in-terminal console, so it looks something like this:

Ngrok running nicely in the terminal

Copy the "Forwarding" https URL. We will need this shortly to configure the Vonage Messages API sandbox. You can also check everything is wired up by requesting this URL in your browser and seeing the same Laravel landing page as before.

Configure the Messages Sandbox

To get incoming WhatsApp messages directed to your application, we need to do a little bit of configuration in the dashboard. Under "Messages and Dispatch," click on "Sandbox" we will use the Messages API Sandbox for the demo today, but if you have a WhatsApp Business account, you could use the same approach to message any user without the Sandbox or whitelisting process.

The QR code used to add your WhatsApp account to the sandbox

Add your phone number to the Sandbox by scanning the QR code or messaging the magic words to the number shown. I am not sure why I enjoy the magic words "auth method" so much, but it really seems like magic!

The webhooks set up

Time to configure the webhooks, and you will need the URL copied from the ngrok console earlier.

  • Inbound should be [url you copied earlier]/webhooks/inbound
  • Status should be [url you copied earlier]/webhooks/status

Don't forget to press "Save" here! That seems to trip me up more often than it ought to.

Inside the Code

Let's look at the various routes set up in this application and how they interact with the Messages API.

Sending From Laravel

The best place to start with this application is at http://localhost:8000/messages where you can add your phone number and send yourself a message.

Note that the number needs to be in the international (E.164) format without a leading +. So for US numbers, start with 1 and then add the full number with area code. For the UK, start with 44 and then add the whole number without the leading 0.

Here's what happens when that form gets submitted:

// in routes/web.php

Route::post('/message', function(Request $request) {
    // TODO: validate incoming params first!

    $url = "https://messages-sandbox.nexmo.com/v0.1/messages";
    $params = ["to" => ["type" => "whatsapp", "number" => $request->input('number')],
        "from" => ["type" => "whatsapp", "number" => "14157386170"],
        "message" => [
            "content" => [
                "type" => "text",
                "text" => "Hello from Vonage and Laravel :) Please reply to this message with a number between 1 and 100"
            ]
        ]
    ];
    $headers = ["Authorization" => "Basic " . base64_encode(env('NEXMO_API_KEY') . ":" . env('NEXMO_API_SECRET'))];

    $client = new \GuzzleHttp\Client();
    $response = $client->request('POST', $url, ["headers" => $headers, "json" => $params]);
    $data = $response->getBody();
    Log::Info($data);

    return view('thanks');
});

This route uses Guzzle to send a POST request to the Messages Sandbox, and if the credentials are correct and the supplied phone number is whitelisted to the application, it will send the message you can see defined here. The message prompts the user to reply, so let's see that code next.

Receiving and Responding to WhatsApp Messages

With the webhooks configured in the dashboard and the Ngrok tunnel running, the local development application can receive incoming WhatsApp messages. If you already replied to the challenge to supply a number, then you know what happens next :)

Here's the code:

// in routes/web.php

Route::post('/webhooks/inbound', function(Request $request) {
    $data = $request->all();

    $text = $data['message']['content']['text'];
    $number = intval($text);
    Log::Info($number);
    if($number > 0) {
        $random = rand(1, 8);
        Log::Info($random);
        $respond_number = $number * $random;
        Log::Info($respond_number);
        $url = "https://messages-sandbox.nexmo.com/v0.1/messages";
        $params = ["to" => ["type" => "whatsapp", "number" => $data['from']['number']],
            "from" => ["type" => "whatsapp", "number" => "14157386170"],
            "message" => [
                "content" => [
                    "type" => "text",
                    "text" => "The answer is " . $respond_number . ", we multiplied by " . $random . "."
                ]
            ]
        ];
        $headers = ["Authorization" => "Basic " . base64_encode(env('NEXMO_API_KEY') . ":" . env('NEXMO_API_SECRET'))];

        $client = new \GuzzleHttp\Client();
        $response = $client->request('POST', $url, ["headers" => $headers, "json" => $params]);
        $data = $response->getBody();
    }
    Log::Info($data);
});

This example probably isn't exactly what you'd want your application to do, but it includes the moving parts you might need! The incoming webhook arrives, and we grab the data and try to read the message content as an integer. We do a little fun maths operation with a randomly generated number and send a reply to the user exactly as we did in the first code example.

You could also use the Laravel HTTP Client in place of Guzzle if you're more familiar with that. The Laravel HTTP Client is a wrapper for Guzzle.

Status Updates

Remember the status webhook we configured earlier? Events about the messages such as when they are submitted (sent from Vonage to WhatsApp), delivered (arrived on the user's device), and read (the user opened the message) are delivered to this endpoint. This application doesn't do a lot with them, but it is great to have access and be able to respond to them.

The status endpoints are also pretty handy for debugging. The route in the example application looks like this:

// in routes/web.php

Route::post('/webhooks/status', function(Request $request) {
    $data = $request->all();
    Log::Info($data);
});

So it just logs each event as it happens. Logging is incredibly valuable when debugging issues! By default, you can find these logs in storage/logs/laravel.log and it is well worth keeping an eye on what is happening in that file as you develop an application like this one!

Fun With Phones

If you didn't try it already, then go ahead and enjoy a WhatsApp chat with your application.

Messages coming in on an Android phone

Today's example is simple but hopefully serves to get you started with WhatsApp and Laravel. We would love to know what you build, so let us know, and of course, always reach out to us if you have questions!

More Resources