In a previous article, we found out about Guzzle and how it can aid us in the task of establishing communication with third party APIs over HTTP. We used it to get the output of a random number generator and for basic interaction with Github’s API. It also offers a series of ‘subscribers’, log-subscriber being one of them and we saw how easy it was to integrate monolog into it.
While interacting with Github’s API we discovered that it supports basic authentication (sending plain username/password). But what if the API we want to use just offers OAUTH authentication?
Guzzle’s Oauth
Implementing OAUTH from scratch can be a hard and time consuming task, and would be like reinventing the wheel… You could grab a third party library to deal with OAUTH requests, but why do that when Guzzle has its own OAUTH subscriber? Yay!
First of all, I have to make you aware that Guzzle has dropped support for PHP 5.3, so you’ll need PHP 5.4 to follow this exercise. The installation instructions from the previous article changed a little bit, yet composer makes our lives a lot easier.
Our goal is to interact with the Twitter API and get our own timeline for which ever purpose we need. We’ll begin with our composer.json file:
{
"name": "johndoe/guzzle-twitter",
"description": "PoC for Sitepoint article",
"authors": [
{
"name": "John Doe",
"email": "john.doe@gmail.tst"
}
],
"minimum-stability": "dev",
"require": {
"guzzlehttp/guzzle": "4.*",
"guzzlehttp/log-subscriber": "1.*",
"monolog/monolog": "*",
"guzzlehttp/oauth-subscriber": "*",
}
}
Please note that we are using Guzzle 4.x, the log subscriber, monolog as our logging back end, and finally the main goal of this article: oauth-subscriber.
Now we should be able to perform a composer update
command to install everything needed.
As for the code, it is very simple:
<?php
require 'vendor/autoload.php';
use GuzzleHttp\Client;
use GuzzleHttp\Subscriber\Oauth\Oauth1;
use GuzzleHttp\Subscriber\Log\LogSubscriber;
use GuzzleHttp\Subscriber\Log\Formatter;
use Monolog\Logger;
use Monolog\Handler\StreamHandler;
date_default_timezone_set('America/Phoenix'); //Set to a proper timezone
/*
* Setting up the logging backend,
*/
$log = new Logger('guzzle');
$log->pushHandler(new StreamHandler('guzzle.log')); // Log will be found at a file named guzzle.log
$subscriber = new LogSubscriber($log, Formatter::SHORT); //To see full details, you can use Formatter::DEBUG
/*
* Creating the Guzzle client, we are setting oauth as the default authentication method
*/
$client = new Client(['base_url' => 'https://api.twitter.com', 'defaults' => ['auth' => 'oauth']]);
/*
* Setting up the oauth subscriber parameters. Parameter values have to be generated at the Twitter site
*/
$oauth = new Oauth1([
'consumer_key' => '',
'consumer_secret' => '',
'token' => '',
'token_secret' => ''
]);
/*
* Attaching the oauth and the log subscriber to the client
*/
$client->getEmitter()->attach($oauth);
$client->getEmitter()->attach($subscriber);
/*
* Executing a GET request on the timeline service, pass the result to the json parser
*/
$res = $client->get('1.1/statuses/home_timeline.json')->json();
print_r($res); //we have the parsed response in an array!
Now let’s see the interesting lines here
<?php
// ......
$client = new Client(['base_url' => 'https://api.twitter.com', 'defaults' => ['auth' => 'oauth']]);
// ......
The Guzzle HTTP’s Client object constructor accepts an array defining several configuration options. The defaults
parameter is defined in the documentation as “Default request options to apply to each request” (see the code on Github). In this case, we are defining that for every request the client performs, it has to use the OAUTH authorization mechanism.
// ......
$oauth = new Oauth1([
'consumer_key' => '',
'consumer_secret' => '',
'token' => '',
'token_secret' => ''
]);
// ......
Next we create an Oauth1
object, containing the parameters needed for an OAUTH authorization. The parameter values needed depend on whether we set up OAUTH in 1-legged or 3-legged mode. You can see this article for an explanation on the differences between those modes.
The essential parameters are consumer_key
and consumer_secret
. Twitter will generate those parameters when you register an app as follows:
First of all, go to https://apps.twitter.com/ and click on the [Create New App] button. Fill in the Name, Description and Website field, but leave the Callback URL blank for now. Agree to the TOS and finally click on the [Create your Twitter application] button.
Once you register you application, go to the [Api Keys] tab. You should see the Api Key and API Secret codes; those are your consumer_key
and consumer_secret
parameters.
OAUTH 1-legged mode is the easiest to code. To tell OUTH we want to use this mode, we have to supply the token
and token_secret
parameters. We will have to generate those on Twitter as follows:
Just below where your API key and API secret codes are, you should see a [Create my access tokens] button. Clicking it, you will see your Access token and Access token secret codes: those are the token
and token_secret
parameters. Now you have everything to fill in the Oauth1 object.
To finally be able to make our code work, we just need to attach the OAUTH subscriber to the Guzzle client with the $client->getEmitter()->attach($oauth);
command. We are ready to execute our first request!
On Twitter you can use 3-legged Oauth by making some changes.
First we have to create an entry point for our application with Twitter to validate a user:
<?php
session_start();
require 'vendor/autoload.php';
use GuzzleHttp\Client;
use GuzzleHttp\Subscriber\Oauth\Oauth1;
use GuzzleHttp\Subscriber\Log\LogSubscriber;
use Monolog\Logger;
use Monolog\Handler\StreamHandler;
use GuzzleHttp\Subscriber\Log\Formatter;
date_default_timezone_set('America/Phoenix');
$log = new Logger('guzzle');
$log->pushHandler(new StreamHandler('guzzle.log'));
$client = new Client(['base_url' => 'https://api.twitter.com', 'defaults' => ['auth' => 'oauth']]);
$subscriber = new LogSubscriber($log, Formatter::SHORT);
$client->getEmitter()->attach($subscriber);
$oauth = new Oauth1([
'consumer_key' => '[your_api_key]',
'consumer_secret' => '[your_api_secret]'
]);
$client->getEmitter()->attach($oauth);
// Set the "auth" request option to "oauth" to sign using oauth
$res = $client->post('oauth/request_token', ['body' => ['oauth_callback' => 'http://[yourawesomehost]/callback.php']]);
$params = (string)$res->getBody();
parse_str($params);
$_SESSION['oauth_token'] = $oauth_token;
$_SESSION['oauth_token_secret'] = $oauth_token_secret;
header("Location: https://api.twitter.com/oauth/authenticate?oauth_token={$oauth_token}");
The most relevant parts of the code are
// ......
$oauth = new Oauth1([
'consumer_key' => '[your_api_key]',
'consumer_secret' => '[your_api_secret]'
]);
// ......
Please notice that we are not using the token
and token_secret
parameters (yet). You still need to generate your API key and API secret as explained above.
// ......
$res = $client->post('oauth/request_token', ['body' => ['oauth_callback' => 'http://[yourawesomehost]/callback.php']]);
// ......
The statement above will send a POST request to oauth/request_token (this is the first leg of OAUTH 3-legged process). We are passing as a parameter a required callback url where Twitter will redirect the user once he has logged in and granted authorization to our application to use his account.
A successful response will include 2 parameters: oauth_token
and oauth_token_secret
. Be sure to store these two values; in the example above they are stored in $_SESSION
.
// ......
header("Location: https://api.twitter.com/oauth/authenticate?oauth_token={$oauth_token}");
// ......
As a final step, we will redirect the user to a Twitter login screen. There, he will need to sign in and grant our application access to his account information (this is the second leg). We need to send as a parameter the oauth_token
parameter we received in the first request.
Now let’s look at the contents of the callback.php file:
<?php
session_start();
require 'vendor/autoload.php';
use GuzzleHttp\Client;
use GuzzleHttp\Subscriber\Oauth\Oauth1;
use GuzzleHttp\Subscriber\Log\LogSubscriber;
use Monolog\Logger;
use Monolog\Handler\StreamHandler;
use GuzzleHttp\Subscriber\Log\Formatter;
date_default_timezone_set('America/Phoenix');
$authToken = $_GET['oauth_token'];
$authVerifier = $_GET['oauth_verifier'];
$log = new Logger('guzzle');
$log->pushHandler(new StreamHandler('guzzle.log'));
$client = new Client(['base_url' => 'https://api.twitter.com', 'defaults' => ['auth' => 'oauth']]);
$subscriber = new LogSubscriber($log, Formatter::SHORT);
$client->getEmitter()->attach($subscriber);
$oauth = new Oauth1([
'consumer_key' => '[your_api_key]',
'consumer_secret' => '[your_api_secret]',
'token' => $_SESSION['oauth_token']
]);
$client->getEmitter()->attach($oauth);
if ($authToken == $_SESSION['oauth_token'])
{
$res = $client->post('oauth/access_token', ['body' => ['oauth_verifier' => $authVerifier]]);
$params = (string)$res->getBody();
parse_str($params);
$_SESSION['oauth_token'] = $oauth_token;
$_SESSION['oauth_token_secret'] = $oauth_token_secret;
$_SESSION['userId'] = $user_id;
$_SESSION['screenName'] = $screen_name;
header("Location: timeline.php");
}
If the user signed in and authorized our application, Twitter will redirect him to our callback file. It will provide two parameters: oauth_token
and oauth_verifier
. We need to catch these two.
<?php
// ......
$authToken = $_GET['oauth_token'];
$authVerifier = $_GET['oauth_verifier'];
// ......
Please beware that getting external parameters without filtering and validating them first, is a great security hazard for your application. For simplicity’s sake, we are not sanitizing any input/output, just make sure YOU DO! ;)
Let’s look at Guzzle’s Oauth subscriber now
<?php
// ......
$oauth = new Oauth1([
'consumer_key' => '[your_api_key]',
'consumer_secret' => '[your_api_secret]',
'token' => $_SESSION['oauth_token']
]);
// ......
Now the token
parameter is present again from the first request.
// ......
if ($authToken == $_SESSION['oauth_token'])
// ......
For security purposes, we have to validate if the oauth_token
parameter provided to our callback is the same as the one we received in the first request.
Here comes the third leg:
// ......
$res = $client->post('oauth/access_token', ['body' => ['oauth_verifier' => $authVerifier]]);
// ......
We have to send a POST request to oauth/access_token
and send the oauth_verifier
parameter we received in the callback to get the final oauth_token
and oauth_token_secret
parameters needed for a full OAUTH request. If the request is successful, along with those two parameters, we will receive others like the user_id
and screen_name
. We can store all those in session to finally make an authenticated OAUTH request as our 1-legged example:
<?php
session_start();
require 'vendor/autoload.php';
use GuzzleHttp\Client;
use GuzzleHttp\Subscriber\Oauth\Oauth1;
use GuzzleHttp\Subscriber\Log\LogSubscriber;
use GuzzleHttp\Subscriber\Log\Formatter;
use Monolog\Logger;
use Monolog\Handler\StreamHandler;
date_default_timezone_set('America/Phoenix'); //Set to a proper timezone
/*
* Setting up the logging backend,
*/
$log = new Logger('guzzle');
$log->pushHandler(new StreamHandler('guzzle.log')); // Log will be found at a file named guzzle.log
$subscriber = new LogSubscriber($log, Formatter::SHORT); //To see full details, you can use Formatter::DEBUG
/*
* Creating the Guzzle client, we are setting oauth as the default authentication method
*/
$client = new Client(['base_url' => 'https://api.twitter.com', 'defaults' => ['auth' => 'oauth']]);
/*
* Setting up the oauth subscriber parameters. Parameter values
* have to be generated at the Twitter site
*/
$oauth = new Oauth1([
'consumer_key' => '[your_api_key]',
'consumer_secret' => '[your_api_secret]',
'token' => $_SESSION['oauth_token'],
'token_secret' => $_SESSION['oauth_token_secret']
]);
/*
* Attaching the oauth and the log subscriber to the client
*/
$client->getEmitter()->attach($oauth);
$client->getEmitter()->attach($subscriber);
/*
* Executing a GET request on the timeline service, pass the result to the json parser
*/
$res = $client->get('1.1/statuses/home_timeline.json')->json();
echo '<pre>';
echo 'Timeline for user ' . $_SESSION['screenName'] . ':' . PHP_EOL . '<hr>';
foreach ($res as $tweet)
{
echo 'From: ' . $tweet['user']['name'] . ' (@' . $tweet['user']['screen_name'] . ')' . PHP_EOL;
echo ' ' . htmlentities($tweet['text']) . PHP_EOL . '<hr>';
}
At the end, we have a complete Guzzle OAUTH subscriber object. This object has the token
and token_secret
parameters received from the third leg request and stored in session:
// ......
$oauth = new Oauth1([
'consumer_key' => '[your_api_key]',
'consumer_secret' => '[your_api_secret]',
'token' => $_SESSION['oauth_token'],
'token_secret' => $_SESSION['oauth_token_secret']
]);
// ......
Conclusion
As you can see, Guzzle makes performing OAUTH authorization a snap, without having to worry about the gory details of request signing. If you have any questions or suggestions, please let me know in the comments below. All the source code for this tutorial is available on Github. Any constructive feedback will always be welcome!
Frequently Asked Questions (FAQs) on Using Guzzle with Twitter via OAuth
How Do I Install Guzzle for Twitter OAuth?
To install Guzzle for Twitter OAuth, you need to use Composer, a dependency management tool in PHP. Run the command composer require guzzlehttp/guzzle
in your terminal. This command will install the latest version of Guzzle. If you want to install a specific version, specify it in the command, like composer require guzzlehttp/guzzle:^6.0
.
How Do I Use Guzzle to Send a GET Request to Twitter API?
After setting up Guzzle and Twitter OAuth, you can send a GET request to Twitter API. First, create a client instance with your base URI. Then, use the request
method to send a GET request. Here’s a sample code:$client = new GuzzleHttp\Client(['base_uri' => 'https://api.twitter.com/']);
$response = $client->request('GET', '1.1/statuses/user_timeline.json');
How Do I Handle Errors in Guzzle?
Guzzle throws exceptions for HTTP error responses. You can use a try-catch block to handle these exceptions. For instance:try {
$response = $client->request('GET', '1.1/statuses/user_timeline.json');
} catch (GuzzleHttp\Exception\ClientException $e) {
echo $e->getMessage();
}
How Do I Send a POST Request to Twitter API Using Guzzle?
To send a POST request, use the request
method with ‘POST’ as the first argument. The second argument is the API endpoint, and the third argument is an array of form parameters. Here’s an example:$response = $client->request('POST', '1.1/statuses/update.json', [
'form_params' => [
'status' => 'Hello, Twitter!'
]
]);
How Do I Use Guzzle to Upload Media to Twitter?
To upload media to Twitter, you need to send a POST request to the media/upload endpoint. The media parameter should be base64-encoded. Here’s a sample code:$response = $client->request('POST', '1.1/media/upload.json', [
'form_params' => [
'media' => base64_encode(file_get_contents('path/to/your/image.jpg'))
]
]);
How Do I Use Guzzle to Stream Large Files?
Guzzle supports streaming large files out of the box. You can use the stream
option to request the body of a response to be streamed. If you’re downloading a file, you can use the sink
option to specify the path to the file. Here’s an example:$response = $client->request('GET', '/stream/large/file', ['stream' => true]);
$stream = $response->getBody();
while (!$stream->eof()) {
echo $stream->read(1024);
}
How Do I Use Guzzle to Send Asynchronous Requests?
Guzzle supports sending asynchronous requests with the requestAsync
method. This method returns a promise that you can use to handle the response when it’s ready. Here’s a sample code:$promise = $client->requestAsync('GET', '1.1/statuses/user_timeline.json');
$promise->then(function ($response) {
echo 'Got a response! ' . $response->getStatusCode();
});
How Do I Use Guzzle to Send Multiple Requests Concurrently?
Guzzle supports sending multiple requests concurrently with the each
function. This function takes an iterator of promises and a concurrency limit. Here’s an example:$promises = (function () {
for ($i = 0; $i < 1000; $i++) {
yield $client->requestAsync('GET', '1.1/statuses/user_timeline.json');
}
})();
GuzzleHttp\Promise\each($promises, 50);
How Do I Use Guzzle to Send Requests with Custom Headers?
To send a request with custom headers, use the headers
option in the request method. Here’s a sample code:$response = $client->request('GET', '1.1/statuses/user_timeline.json', [
'headers' => [
'Authorization' => 'Bearer ' . $token,
'Accept' => 'application/json'
]
]);
How Do I Use Guzzle to Send Requests with Query Parameters?
To send a request with query parameters, use the query
option in the request method. Here’s an example:$response = $client->request('GET', '1.1/search/tweets.json', [
'query' => [
'q' => 'Guzzle',
'count' => 10
]
]);
Web application developer, database administrator, project lead in a variety of enterprise apps and now article author. Interested in database design and optimization. Amazed and involved in distributed systems development. Cryptography and information security fan.