2013-01-04 14:44:02 -05:00
< ? php
namespace OAuth2 ;
2013-01-29 16:23:41 +00:00
use OAuth2\Util ;
2013-01-29 14:16:47 +00:00
use OAuth2\Storage\SessionInterface ;
use OAuth2\Storage\ClientInterface ;
use OAuth2\Storage\ScopeInterface ;
2013-01-22 11:33:09 -05:00
class AuthServer
2013-01-04 14:44:02 -05:00
{
2013-01-29 14:18:13 +00:00
/**
* The delimeter between scopes specified in the scope query string parameter
*
* The OAuth 2 specification states it should be a space but that is stupid
* and everyone excepted Google use a comma instead .
*
* @ var string
*/
2013-01-04 14:44:02 -05:00
protected $scopeDelimeter = ',' ;
protected $expiresIn = 3600 ;
2013-01-29 14:17:56 +00:00
protected $responseTypes = array ();
2013-01-04 14:44:02 -05:00
protected $storages = array ();
protected $grantTypes = array ();
2013-01-22 11:25:51 -05:00
protected $request = null ;
2013-01-29 14:19:23 +00:00
/**
* Exception error codes
* @ var array
*/
protected $exceptionCodes = array (
0 => 'invalid_request' ,
1 => 'unauthorized_client' ,
2 => 'access_denied' ,
3 => 'unsupported_response_type' ,
4 => 'invalid_scope' ,
5 => 'server_error' ,
6 => 'temporarily_unavailable' ,
7 => 'unsupported_grant_type' ,
8 => 'invalid_client' ,
9 => 'invalid_grant'
);
/**
* Exception error messages
* @ var array
*/
protected $exceptionMessages = array (
'invalid_request' => 'The request is missing a required parameter, includes an invalid parameter value, includes a parameter more than once, or is otherwise malformed. Check the "%s" parameter.' ,
'unauthorized_client' => 'The client is not authorized to request an access token using this method.' ,
'access_denied' => 'The resource owner or authorization server denied the request.' ,
'unsupported_response_type' => 'The authorization server does not support obtaining an access token using this method.' ,
'invalid_scope' => 'The requested scope is invalid, unknown, or malformed. Check the "%s" scope.' ,
'server_error' => 'The authorization server encountered an unexpected condition which prevented it from fulfilling the request.' ,
'temporarily_unavailable' => 'The authorization server is currently unable to handle the request due to a temporary overloading or maintenance of the server.' ,
'unsupported_grant_type' => 'The authorization grant type "%s" is not supported by the authorization server' ,
'invalid_client' => 'Client authentication failed' ,
'invalid_grant' => 'The provided authorization grant is invalid, expired, revoked, does not match the redirection URI used in the authorization request, or was issued to another client. Check the "%s" parameter.' ,
'invalid_credentials' => 'The user credentials were incorrect.' ,
'invalid_refresh' => 'The refresh token is invalid.' ,
);
2013-01-04 14:44:02 -05:00
2013-01-29 14:16:47 +00:00
public function __construct ( ClientInterface $client , SessionInterface $session , ScopeInterface $scope )
{
$this -> storages = array (
'client' => $client ,
'session' => $session ,
'scope' => $scope
);
2013-01-04 14:44:02 -05:00
}
2013-01-29 14:17:56 +00:00
public function addGrantType ( GrantTypeInterface $grantType , $identifier = null )
2013-01-04 14:44:02 -05:00
{
if ( is_null ( $identifier )) {
2013-01-29 14:17:56 +00:00
$identifier = $grantType -> getIdentifier ();
}
$this -> grantTypes [ $identifier ] = $grantType ;
2013-01-29 16:21:53 +00:00
if ( ! is_null ( $grantType -> getResponseType ())) {
2013-01-29 14:17:56 +00:00
$this -> responseTypes [] = $grantType -> getResponseType ();
2013-01-04 14:44:02 -05:00
}
}
2013-01-22 11:25:51 -05:00
public function getScopeDelimeter ()
{
return $this -> scopeDelimeter ;
}
2013-01-04 14:44:02 -05:00
public function setScopeDelimeter ( $scope_delimeter )
{
$this -> scopeDelimeter = $scope_delimeter ;
}
2013-01-22 11:25:51 -05:00
public function getExpiresIn ()
{
return $this -> expiresIn ;
}
2013-01-04 14:44:02 -05:00
public function setExpiresIn ( $expires_in )
{
$this -> expiresIn = $expires_in ;
}
2013-01-22 11:25:51 -05:00
/**
* Sets the Request Object
*
* @ param RequestInterface The Request Object
*/
public function setRequest ( RequestInterface $request )
{
$this -> request = $request ;
}
/**
* Gets the Request object . It will create one from the globals if one is not set .
*
* @ return RequestInterface
*/
public function getRequest ()
{
if ( $this -> request === null ) {
$this -> request = Request :: buildFromGlobals ();
}
return $this -> request ;
}
2013-01-29 14:16:47 +00:00
public function getStorage ( $obj )
{
return $this -> storages [ $obj ];
}
2013-01-22 11:25:51 -05:00
/**
* Check client authorise parameters
*
* @ access public
* @ param array $authParams Optional array of parsed $_GET keys
* @ return array Authorise request parameters
*/
public function checkClientAuthoriseParams ( $authParams = null )
{
2013-01-29 14:56:17 +00:00
$params = array ();
2013-01-22 11:25:51 -05:00
2013-01-29 14:56:17 +00:00
// Client ID
$params [ 'client_id' ] = ( isset ( $authParams [ 'client_id' ])) ?
$authParams [ 'client_id' ] :
$this -> getRequest () -> get ( 'client_id' );
if ( is_null ( $params [ 'client_id' ])) {
throw new Exception\ClientException ( sprintf ( $this -> exceptionMessages [ 'invalid_request' ], 'client_id' ), 0 );
}
// Redirect URI
$params [ 'redirect_uri' ] = ( isset ( $authParams [ 'redirect_uri' ])) ?
$authParams [ 'redirect_uri' ] :
$this -> getRequest () -> get ( 'redirect_uri' );
if ( is_null ( $params [ 'redirect_uri' ])) {
throw new Exception\ClientException ( sprintf ( $this -> exceptionMessages [ 'invalid_request' ], 'redirect_uri' ), 0 );
}
// Validate client ID and redirect URI
$clientDetails = $this -> getStorage ( 'client' ) -> get ( $params [ 'client_id' ], null , $params [ 'redirect_uri' ]);
if ( $clientDetails === false ) {
throw new Exception\ClientException ( $this -> exceptionMessages [ 'invalid_client' ], 8 );
}
$params [ 'client_details' ] = $clientDetails ;
// Response type
$params [ 'response_type' ] = ( isset ( $authParams [ 'response_type' ])) ?
$authParams [ 'response_type' ] :
$this -> getRequest () -> get ( 'response_type' );
if ( is_null ( $params [ 'response_type' ])) {
throw new Exception\ClientException ( sprintf ( $this -> exceptionMessages [ 'invalid_request' ], 'response_type' ), 0 );
}
// Ensure response type is one that is recognised
if ( ! in_array ( $params [ 'response_type' ], $this -> responseTypes )) {
throw new Exception\ClientException ( $this -> exceptionMessages [ 'unsupported_response_type' ], 3 );
}
// Get and validate scopes
$scopes = ( isset ( $authParams [ 'scope' ])) ?
$authParams [ 'scope' ] :
$this -> getRequest () -> get ( 'scope' , '' );
$scopes = explode ( $this -> _config [ 'scope_delimeter' ], $scopes );
for ( $i = 0 ; $i < count ( $scopes ); $i ++ ) {
$scopes [ $i ] = trim ( $scopes [ $i ]);
if ( $scopes [ $i ] === '' ) unset ( $scopes [ $i ]); // Remove any junk scopes
}
if ( count ( $scopes ) === 0 ) {
throw new Exception\ClientException ( sprintf ( $this -> exceptionMessages [ 'invalid_request' ], 'scope' ), 0 );
}
$params [ 'scopes' ] = array ();
foreach ( $scopes as $scope ) {
$scopeDetails = $this -> getStorage ( 'scope' ) -> get ( $scope );
if ( $scopeDetails === false ) {
throw new Exception\ClientException ( sprintf ( $this -> exceptionMessages [ 'invalid_scope' ], $scope ), 4 );
}
$params [ 'scopes' ][] = $scopeDetails ;
}
return $params ;
2013-01-22 11:25:51 -05:00
}
/**
* Parse a new authorise request
*
* @ param string $type The session owner ' s type
* @ param string $typeId The session owner ' s ID
* @ param array $authoriseParams The authorise request $_GET parameters
* @ return string An authorisation code
*/
public function newAuthoriseRequest ( $type , $typeId , $authoriseParams )
{
2013-01-29 16:23:41 +00:00
// Generate an auth code
$authCode = SecureKey :: make ();
2013-01-22 11:25:51 -05:00
2013-01-29 16:23:41 +00:00
// Remove any old sessions the user might have
$this -> getStorage ( 'session' ) -> delete ( $authoriseParams [ 'client_id' ], $type , $typeId );
// Create a new session
$sessionId = $this -> getStorage ( 'session' ) -> create ( $authoriseParams [ 'client_id' ], $authoriseParams [ 'redirect_uri' ], $type , $typeId , $authCode );
// Associate scopes with the new session
foreach ( $authoriseParams [ 'scopes' ] as $scope )
{
$this -> getStorage ( 'session' ) -> associateScope ( $sessionId , $scope [ 'id' ]);
}
return $authCode ;
2013-01-22 11:25:51 -05:00
}
/**
* Issue an access token
*
* @ access public
* @ param array $authParams Optional array of parsed $_POST keys
* @ return array Authorise request parameters
*/
public function issueAccessToken ( $authParams = null )
{
2013-01-29 16:24:28 +00:00
$params [ 'grant_type' ] = ( isset ( $authParams [ 'grant_type' ])) ?
$authParams [ 'grant_type' ] :
$this -> getRequest () -> post ( 'grant_type' );
2013-01-22 11:25:51 -05:00
2013-01-29 16:24:28 +00:00
if ( is_null ( $params [ 'grant_type' ])) {
throw new Exception\ClientException ( sprintf ( $this -> errors [ 'invalid_request' ], 'grant_type' ), 0 );
}
2013-01-22 11:25:51 -05:00
2013-01-29 16:24:28 +00:00
// Ensure grant type is one that is recognised and is enabled
if ( ! in_array ( $params [ 'grant_type' ], array_keys ( $this -> grantTypes ))) {
throw new Exception\ClientException ( sprintf ( $this -> errors [ 'unsupported_grant_type' ], $params [ 'grant_type' ]), 7 );
}
2013-01-22 11:25:51 -05:00
2013-01-29 16:24:48 +00:00
// Complete the flow
return $this -> getCurrentGrantType ( $params [ 'grant_type' ]) -> completeFlow ( $authParams , $params );
}
2013-01-22 11:25:51 -05:00
2013-01-29 16:24:48 +00:00
protected function getCurrentGrantType ( $grantType )
{
return $this -> grantTypes [ $grantType ];
2013-01-22 11:25:51 -05:00
}
2013-01-04 14:44:02 -05:00
}