Decoupled Drupal Authentication with OAuth 2.0

  • 10 minute read

Perhaps the most critical component of any decoupled Drupal architecture is a robust authentication mechanism that protects data transmitted between a Drupal site and API consumers like JavaScript applications and native mobile applications. While Drupal core makes available HTTP Basic Authentication and cookie-based authentication, both easy to use, neither of these approaches is sufficiently secure when it comes to best practices.

Fortunately, the Drupal contributed ecosystem contains several highly useful modules that leverage more recent authentication standards like OAuth 2.0 Bearer Token and JSON Web Tokens (JWT), both of which are seeing wide use in the Drupal community among decoupled Drupal practitioners. In the next two installments of Experience Express, we take a breather from voyaging to conferences and inspect authentication best practices in decoupled Drupal, starting with OAuth 2.0.

OAuth 2.0 Bearer Token authentication

At present, OAuth represents one of the most commonly found authentication methods as an open standard for access delegation. Somewhat confusingly, there are two separate modules in Drupal's contributed ecosystem that handle OAuth-based authentication, namely the OAuth module (providing support for the first version of OAuth, no longer recommended) and the Simple OAuth module (providing support for the second and current version, OAuth 2.0).

OAuth works by the server granting access to server resources on behalf of someone who exerts ownership over those resources. OAuth provides access tokens over HTTP to API consumers using an authorization server found in Drupal, which are then used by consumers in subsequent requests to retrieve data or inflict some change on the server.

Authored by Mateu Aguiló Bosch (e0ipso), the Simple OAuth module, by default, provides a password grant, which is an access grant that allows trusted first-party applications to access operations in Drupal. While there are other grant types provided in the OAuth 2.0 Authorization Framework (RFC 6749), and though password grants may allocate excessive control, it is a convenient foundation for authentication in decoupled Drupal.1

Installing Simple OAuth and assigning scopes

Install Simple OAuth using Composer in your project root. This command downloads and installs both the Simple OAuth module and its dependency, the OAuth2 Server package by the PHP League.

$ composer require drupal/simple_oauth:^3

With Simple OAuth installed, our next step is to generate a pair of keys, one public and the other private, to encrypt the tokens generated by Simple OAuth. These keys should be stored outside of the Drupal project root, but save the paths at which they are located for future reference.

$ openssl genrsa -out private.key 2048
$ openssl rsa -in private.key -pubout > public.key

In Drupal, scopes in OAuth 2.0 (which define operations that a consumer should be granted access to) are analogous to user roles. Generally, we can couple each consumer, such as a native mobile application, with an individual user role. Now, we can help Drupal know about the consumers that need to be granted OAuth 2.0 tokens.

If you navigate to Configuration » Simple OAuth » Consumers (/admin/config/services/consumer) and click on Add consumer (/admin/config/services/consumer/add), you will see a form that allows us to make your consumer known to Drupal. You'll need to create a name for your consumer and a secret (a password) that the consumer will include in every request.

image1_9.png

After creating our consumer in Drupal, we can see its scope and UUID.

Finally, head to the Simple OAuth module configuration page (/admin/config/people/simple_oauth) to add the paths to the public and private key that we generated earlier in the tutorial.2

Creating and verifying access tokens

Now that we've configured the Simple OAuth module, we need to request an access token from the resource located at /oauth/token. The OAuth 2.0 specification stipulates that the OAuth token resource on an authorization server must accept only POST requests whose bodies are formatted in form-data or x-www-form-urlencoded, and any JSON-formatted bodies will be rejected.

You'll need the following parameters to be present in the request body.

  • grant_type: In most cases, password.
  • client_id: The UUID of the consumer from the previous section.
  • client_secret: The client secret provided during the addition of the consumer.
  • username: The username of the account associated with the consumer.
  • password: The password for the account associated with the consumer.

When you issue this POST request, you'll receive the following response containing a JSON object with our access token represented, like the one below (tokens have been truncated).

{
  "token_type": "Bearer",
  "expires_in": 870000,
  "access_token": "eyJ0eXAi0iJKV1Qi[...]",
  "refresh_token": "uAXzh+B/7kCxsXkl[...]",
}

In order to make sure that our access token is operational, we can craft a GET request against the /oauth/debug resource with ?_format=json added to the end. Make sure to add an Authorization header containing "Bearer " (note the space with particular scrutiny), with the access token above appended.

GET /oauth/debug?_format=json HTTP/1.1
Authorization: Bearer eyJ0eXAi0iJKV1Qi[...]

The resulting response from Drupal's authorization server will include a JSON object with the access token repeated as well as the uid of the user whose credentials were supplied in the original request against /oauth/token. Finally, Drupal also adds a list of that user's roles and permissions, reflecting the scopes to which the user has access.3

Issuing requests with OAuth 2.0 authentication

In order to designate OAuth 2.0 as the preferred method to authenticate incoming requests from consumers, we can instruct Drupal to expose certain HTTP methods and REST resources through OAuth 2.0 authentication using either configuration imports or the REST UI module (both covered in a previous installment of Experience Express). We'll use the latter approach here.

Enable both the core REST and REST UI modules.

$ composer require drupal/restui
$ drush en -y rest restui

On the REST UI configuration page (/admin/config/services/rest), you will see a list of resources that core REST exposes in its API. On the Settings for resource Content page (/admin/config/services/rest/resource/entity%3Anode/edit), enable the oauth2 option that appears under the Authenticated providers list. Now, all requests against Drupal nodes will be authenticated through OAuth 2.0.

Once again, we need to retrieve our access token with a POST request with the request body formatted as form-data or x-www-form-urlencoded as before against /oauth/token.

grant_type: password
client_id: 24ac1dc6-9cd3-11e8-98d0-529269fb1459
client_secret: l0r3m1psum
username: admin
password: admin

Now, in the request you intend to submit, include an Authorization header with the Bearer prefix, as you can see in the example below. Once Drupal validates the token, it will proceed to serve the request.

Authorization: Bearer eyJ0eXAi0iJKV1Qi[...]

What to do if your access token expires

Once your Drupal site is in production, it isn't a good idea to have a lengthy time to expiration set for issued OAuth 2.0 tokens. If your token does expire, you can easily acquire a new token to keep submitting requests to Drupal. If you remember from several sections ago in this blog post, the response we receive from our request to /oauth/token also includes a refresh token, which lasts longer and are associated with another access token that we can use to substitute the expired token.

Enable the Simple OAuth Extras module contained within the Simple OAuth module in order to access refresh token functionality.

$ drush en -y simple_oauth_extras

Use the refresh token to acquire a new access token by issuing a POST request to /oauth/token containing the following parameters as form-data or x-www-form-urlencoded.

grant_type: refresh_token
 refresh_token: uAXzh+B/7kCxsXkl[...]
client_id: 24ac1dc6-9cd3-11e8-98d0-529269fb1459
 client_secret: l0r3m1psum

Drupal's authorization server will then respond with a JSON object that includes token_type (Bearer), expires_in (with the new expiry time of the access token), access_token (a new access token encrypted with the private key we previously provided), and refresh_token (a new refresh token for next time).4

If you've allowed your refresh token to expire, you'll have to repeat the entire process to acquire a new access token from the beginning, but you can avert this process by ensuring that your consumer automatically retrieves a new access token just before the refresh token itself expires.

Conclusion

OAuth 2.0 is the most widely used authentication mechanism for decoupled Drupal use cases, and it is far and away more secure than the Basic Authentication approach used in Drupal core by default. As you can see, protecting data from prying eyes is the responsibility of both the consumer and the authorization server, but Simple OAuth makes it much easier thanks to easy-to-use configuration interfaces and strict adherence to the OAuth 2.0 specification.

In the next installment of Experience Express, we examine another authentication method, JSON Web Tokens (JWT), which is newer but has a wide following and is also more secure than OAuth 2.0. While the module underpinning its functionality in the Drupal contributed ecosystem may be less stable, JWT provides another fascinating approach to authentication in decoupled Drupal that make it a compelling choice for practitioners.

Notes

  1. "OAuth 2.0." Drupal.org. 18 February 2018. Accessed 7 August 2018. https://www.drupal.org/project/simple_oauth
  2. Aguiló Bosch, Mateu. "2. Installation and set up." YouTube. 30 November 2016. Accessed 10 August 2018. https://www.youtube.com/watch?v=SI60hF4n8U8
  3. Aguiló Bosch, Mateu. "3. Password Grant." YouTube. 30 November 2016. Accessed 10 August 2018. https://www.youtube.com/watch?v=BEKKFExaBMM
  4. "Refresh token grant." The League of Extraordinary Packages. Accessed 22 August 2018. http://oauth2.thephpleague.com/authorization-server/refresh-token-grant