ash-rain / oauth2-server
PHP OAuth 2.0 implementation.
Requires
- symfony/http-foundation: ~2.4
Requires (Dev)
- mockery/mockery: ~0.9
- phpunit/phpunit: ~4.0
- predis/predis: ~0.8.5
Suggests
- predis/predis: Predis is required to use the Redis storage adapter.
This package is auto-updated.
Last update: 2024-11-07 20:14:36 UTC
README
A PHP OAuth 2.0 server implementation.
Installation
The package can be installed with Composer, either by modifying your composer.json
directly or using the composer require
command.
composer require ash-rain/oauth2-server
Note that this package is still under development and has not been tagged as stable.
Grant Types
All four OAuth 2.0 grant types detailed in the specification are implemented within this package.
Storage Adapters
As of v0.1.0 the following storage adapters are available.
Dingo\OAuth2\Storage\MySqlAdapter
Dingo\OAuth2\Storage\RedisAdapter
Using the dingo/oauth2-server-laravel package you also have.
Dingo\OAuth2\Storage\FluentAdapter
MySQL Table Structure
The following is the table structure required for the MySQL storage adapter. When developing your own storage adapters you can use this structure as a starting point.
CREATE TABLE IF NOT EXISTS `oauth_authorization_codes` ( `code` varchar(40) COLLATE utf8_unicode_ci NOT NULL, `client_id` varchar(255) COLLATE utf8_unicode_ci NOT NULL, `user_id` varchar(255) COLLATE utf8_unicode_ci NOT NULL, `redirect_uri` varchar(255) COLLATE utf8_unicode_ci DEFAULT NULL, `expires` datetime NOT NULL, PRIMARY KEY (`code`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci; CREATE TABLE IF NOT EXISTS `oauth_authorization_code_scopes` ( `id` int(11) NOT NULL AUTO_INCREMENT, `code` varchar(40) COLLATE utf8_unicode_ci NOT NULL, `scope` varchar(255) COLLATE utf8_unicode_ci NOT NULL, PRIMARY KEY (`id`), KEY `code` (`code`,`scope`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci AUTO_INCREMENT=1; CREATE TABLE IF NOT EXISTS `oauth_clients` ( `id` varchar(40) COLLATE utf8_unicode_ci NOT NULL, `secret` varchar(40) COLLATE utf8_unicode_ci NOT NULL, `name` varchar(100) COLLATE utf8_unicode_ci NOT NULL, `trusted` tinyint(1) NOT NULL DEFAULT '0', PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci; CREATE TABLE IF NOT EXISTS `oauth_client_endpoints` ( `id` int(11) NOT NULL AUTO_INCREMENT, `client_id` varchar(40) COLLATE utf8_unicode_ci NOT NULL, `uri` varchar(255) COLLATE utf8_unicode_ci NOT NULL, `is_default` tinyint(1) NOT NULL DEFAULT '0', PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci AUTO_INCREMENT=1; CREATE TABLE IF NOT EXISTS `oauth_scopes` ( `scope` varchar(255) COLLATE utf8_unicode_ci NOT NULL, `name` varchar(255) COLLATE utf8_unicode_ci NOT NULL, `description` text COLLATE utf8_unicode_ci NOT NULL, PRIMARY KEY (`scope`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci; CREATE TABLE IF NOT EXISTS `oauth_tokens` ( `token` varchar(40) COLLATE utf8_unicode_ci NOT NULL, `type` enum('access','refresh') COLLATE utf8_unicode_ci NOT NULL DEFAULT 'access', `client_id` varchar(40) COLLATE utf8_unicode_ci NOT NULL, `user_id` varchar(255) COLLATE utf8_unicode_ci DEFAULT NULL, `expires` datetime NOT NULL, PRIMARY KEY (`token`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci; CREATE TABLE IF NOT EXISTS `oauth_token_scopes` ( `id` int(11) NOT NULL AUTO_INCREMENT, `token` varchar(40) COLLATE utf8_unicode_ci NOT NULL, `scope` varchar(255) COLLATE utf8_unicode_ci NOT NULL, PRIMARY KEY (`id`), KEY `token` (`token`,`scope`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci AUTO_INCREMENT=1;
Usage Guide
This guide is very brief and is framework agnostic. It's purpose is to simply demonstrate how the pieces come together and is not meant to be used as a real world implementation. As such there will be no mention of where each piece of code belongs.
Before we continue you should be aware of what the following terms mean.
Clients
In OAuth 2.0 a client is an application that acts on behalf of a user and talks to the Authorization and Resource servers.
The package is capable of creating clients, however, no user interface is provided with the package. To create a client you'll need a storage adapter instance.
$storage = new Dingo\OAuth2\Storage\MySqlAdapter(new PDO('mysql:host=localhost;dbname=oauth', 'root'));
You can now get the client storage and create a new client.
$storage->get('client')->create('id', 'secret', 'name', [['uri' => 'http://example.com/code', 'default' => true]]);
A client is expected to have at least ONE associated endpoint and it should be defined as the default endpoint. Clients can have multiple endpoints for testing or staging servers.
$storage->get('client')->create('id', 'secret', 'name', [ ['uri' => 'http://example.com/code', 'default' => true], ['uri' => 'http://staging.example.com/code', 'default' => false] ]);
A client can be set as "trusted", meaning you can perform a quick check before authorizing and if the client is marked as "trusted" then it will be automatically authorized. The fifth parameter must be set to true
to mark the client as "trusted".
$storage->get('client')->create('id', 'secret', 'name', [['uri' => 'http://example.com/code', 'default' => true]], true);
You can also delete a client. This will also delete an associated endpoints.
$storage->get('client')->delete('id');
For the rest of the guide it will be assumed that you have created a client similar to the following.
$storage->get('client')->create('id', 'secret', 'name', [['uri' => 'http://localhost/example-client/auth/code', 'default' => true]]);
Scopes
When a client requests a users authorization the client will often request specific permissions, these permissions are refered to as a scope. A scope defines what the client has permission to see or do. Essentially they provide developers with even finer control over what a client can access.
The package is capable of creating scopes, however, no user interface is provided with the package. To create a scope you'll need a storage adapter instance.
$storage = new Dingo\OAuth2\Storage\MySqlAdapter(new PDO('mysql:host=localhost;dbname=oauth', 'root'));
You can now get the scope storage and create a new scope.
$storage->get('scope')->create('scope', 'name', 'description');
You can also delete a scope.
$storage->get('scope')->delete('scope');
This guide will not utilize scopes, however, feel free to create and use them.
Authorization Server
The responsibilities of the Authorization Server are to authorize and issue access tokens to clients. Depending on the configuration the Authorization Server will also issue a refresh token which the client should store for when the access token expires.
To issue an access token the Authorization Server must be configured with the desired storage adapter and grant types.
$storage = new Dingo\OAuth2\Storage\MySqlAdapter(new PDO('mysql:host=localhost;dbname=oauth', 'root')); $server = new Dingo\OAuth2\Server\Authorization($storage);
We can now register the four different grant types depending on the projects requirements. For this guide we'll only be using the standard Authorization Code grant type.
$server->registerGrant(new Dingo\OAuth2\Grant\AuthorizationCode);
We'll now need a route that will handle the attempted authorization. This guide will assume this route is at http://localhost/example-server/authorize
.
<?php // If the user is not logged in we'll redirect them to the login form // with the query string that was sent with the initial request. // The login form is not within the scope of this guide. if ( ! isset($_SESSION['user_id'])) { header("Location: /login?{$_SERVER['QUERY_STRING']}"); } else { try { $payload = $server->validateAuthorizationRequest(); } catch (Dingo\OAuth2\Exception\ClientException $exception) { echo $exception->getMessage(); exit; } if (isset($_POST['submit']) or $payload['client']->isTrusted()) { $response = $server->handleAuthorizationRequest($payload['client_id'], $_SESSION['user_id'], $payload['redirect_uri'], $payload['scopes']); header("Location: {$server->makeRedirectUri($response)}"); } else { ?> <p><?php echo $payload['client']->getName(); ?> wants permission to:</p> <table> <?php foreach($payload['scopes'] as $scope): ?> <tr> <td> <strong><?php echo $scope->getName(); ?></strong> </td> <td> <?php echo $scope->getDescription(); ?> </td> </tr> <?php endforeach; ?> </table> <form method="POST"> <input type="submit" name="submit" value="Authorize"> <input type="submit" name="cancel" value="Cancel"> </form> <?php } } ?>
The client can now prompt a user to authorize via OAuth 2.0 by directing the user to a similar URI as follows (note that spacing has been added for readability).
http://localhost/example-server/authorize
?response_type=code
&client_id=example
&redirect_uri=http%3A%2F%2Flocalhost%2Fexample-client%2Fauth%2Fcode
If the Authorization Server detects that the user is not logged in they will be redirected to the login page and requested to login. Once logged in the user should be redirected back where they are prompted to authorize the client unless the client has been marked as "trusted". If the user authorizes the client the Authorization Server will issue an authorization code which is sent back as part of the query string on the redirect URI that was provided.
Remember that if a redirection URI is provided it must match a redirection URI that was registered for the client. When no redirection URI is provided the default redirection URI is used.
Now that the client has the authorization code it needs to request an access token from the Authorization Server using the code. We'll need a route that will handle the issuing of access tokens. This guide will assume the route is at http://localhost/example-server/token
.
header('Content-Type: application/json'); echo json_encode($server->issueAccessToken());
The client can now request the access token by sending another request to the Authorization Server to a similar URI as follows (note that spacing has been added for readability).
http://localhost/example-server/token
?grant_type=authorization_code
&code=<authorization_code_returned_by_server>
&client_id=example
&client_secret=topsecret
The Authorization Server should respond with a JSON payload similar to the following.
{ "access_token": "nkwCbxJ8EAEqEM11vCrKLd2TAqJLfCN21beMjVGK", "token_type": "Bearer", "expires": 1396795320, "expires_in": 3600, "refresh_token": "vnzKgulkldV1cnDeVh4y8KbAjDHCqvWBMnxTUqWa" }
The client should save the refresh token and all subsequent requests to protected resources should include an Authorization
header similar to the following.
Authorization: Bearer nkwCbxJ8EAEqEM11vCrKLd2TAqJLfCN21beMjVGK
Authorized Callback
You can set an optional authorized callback that is fired once a client has been authorized. This is useful for when you want to avoid prompting a user to authorize a client that they've already authorized in the past with the same scopes. The callback receives two parameters, the first being an instance of Dingo\OAuth2\Entity\Token
and the second being an instance of Dingo\OAuth2\Entity\Client
.
$server->setAuthorizedCallback(function($token, $client) { // Insert a record into your database showing that $token->getUserId() has authorized // $client->getId() with $token->getScopes() and that in the future the server // can skip the prompt. });
It's up to you to implement the code required to check if the user has already authorized the client in the past. As an example you might adjust the if
statement earlier to check the existence of a database record.
$alreadyAuthorized = $db->table('user_authorized_clients') ->where('client_id', '=', $payload['client']->getId()) ->where('user_id', '=', $_SESSION['user']['id']) ->exists(); if (isset($_POST['submit']) or $alreadyAuthorized == true) { $response = $server->handleAuthorizationRequest($payload['client_id'], $payload['user_id'], $payload['redirect_uri'], $payload['scopes']); header("Location: {$server->makeRedirectUri($response)}"); }
Be sure to only authorize requests that have the same scopes or less. Never authorize a request that includes different scopes as the user should be give the chance the review and either approve or deny the request.
Resource Server
The responsibility of the Resource Server is to authenticate a request by validating the supplied access token.
$storage = new Dingo\OAuth2\Storage\MySqlAdapter(new PDO('mysql:host=localhost;dbname=oauth', 'root')); $server = new Dingo\OAuth2\Server\Resource($storage);
We can now validate that a request contains an access token that exists and has not expired.
try { $server->validateRequest(); } catch (Dingo\OAuth2\Exception\InvalidTokenException $exception) { header('Content-Type: application/json', true, $exception->getStatusCode()); echo json_encode(['error' => $exception->getError(), 'message' => $exception->getMessage()]); exit; }
Exception Errors
When issuing access tokens or validating requests for a protected resource exceptions may be thrown when something goes wrong, e.g., missing a required parameter in the request. Generally speaking all exceptions thrown will extend from Dingo\OAuth2\Exception\OAuthException
. Each exception will have an error type, a message, and an HTTP status code.
This allows you to return more informative responses to clients. Returning both the error type and error message is convenient for clients to distinguish what actually happened and react accordingly.
Refer to the Resource Server example above to see how exceptions should be caught and returned to the client.
Error Types
The following tables describes the error types and the reason as to why this error was raised.