2015-11-13 23:07:37 +05:30
< ? php
namespace League\OAuth2\Server\Exception ;
use League\OAuth2\Server\Utils\RedirectUri ;
use Psr\Http\Message\ResponseInterface ;
use Zend\Diactoros\Response ;
use Zend\Diactoros\ServerRequest ;
class OAuthServerException extends \Exception
{
/**
* @ var int
*/
private $httpStatusCode ;
/**
* @ var string
*/
private $errorType ;
/**
* @ var null | string
*/
private $hint ;
/**
* @ var null | string
*/
private $redirectUri ;
/**
* Throw a new exception
*
* @ param string $message Error message
* @ param string $errorType Error type
* @ param int $httpStatusCode HTTP status code to send ( default = 400 )
* @ param null | string $hint A helper hint
* @ param null | string $redirectUri A HTTP URI to redirect the user back to
*/
public function __construct ( $message , $errorType , $httpStatusCode = 400 , $hint = null , $redirectUri = null )
{
parent :: __construct ( $message );
$this -> httpStatusCode = $httpStatusCode ;
$this -> errorType = $errorType ;
$this -> hint = $hint ;
$this -> redirectUri = $redirectUri ;
}
/**
* Invalid grant type error
*
* @ param null | string $localizedError
* @ param null | string $localizedHint
*
* @ return static
*/
public static function invalidGrantType (
$localizedError = null ,
$localizedHint = null
) {
$errorMessage = ( is_null ( $localizedError ))
? '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.'
: $localizedError ;
$hint = ( is_null ( $localizedHint ))
? 'Check the `grant_type` parameter'
: $localizedHint ;
return new static ( $errorMessage , 'invalid_grant' , 400 , $hint );
}
/**
* Unsupported grant type error
*
* @ param null | string $localizedError
* @ param null | string $localizedHint
*
* @ return static
*/
public static function unsupportedGrantType (
$localizedError = null ,
$localizedHint = null
) {
$errorMessage = ( is_null ( $localizedError ))
? 'The authorization grant type is not supported by the authorization server.'
: $localizedError ;
$hint = ( is_null ( $localizedHint ))
? 'Check the `grant_type` parameter'
: $localizedHint ;
return new static ( $errorMessage , 'unsupported_grant_type' , 400 , $hint );
}
/**
* Invalid request error
*
* @ param string $parameter The invalid parameter
* @ param null | string $localizedError
* @ param null | string $localizedHint
*
* @ return static
*/
public static function invalidRequest (
$parameter ,
$localizedError = null ,
$localizedHint = null
) {
$errorMessage = ( is_null ( $localizedError ))
? 'The request is missing a required parameter, includes an invalid parameter value, includes a parameter more than once, or is otherwise malformed.'
: $localizedError ;
$hint = ( is_null ( $localizedHint ))
? sprintf ( 'Check the `%s` parameter' , $parameter )
: sprintf ( $localizedHint , $parameter );
return new static ( $errorMessage , 'invalid_request' , 400 , $hint );
}
/**
* Invalid client error
*
* @ param null | string $localizedError
*
* @ return static
*/
public static function invalidClient ( $localizedError = null )
{
$errorMessage = ( is_null ( $localizedError ))
? 'Client authentication failed'
: $localizedError ;
return new static ( $errorMessage , 'invalid_client' , 401 );
}
/**
* Invalid scope error
*
* @ param string $scope The bad scope
* @ param null | string $localizedError A localized error message
* @ param null | string $localizedHint A localized error hint
* @ param null | string $redirectUri A HTTP URI to redirect the user back to
*
* @ return static
*/
public static function invalidScope ( $scope , $localizedError = null , $localizedHint = null , $redirectUri = null )
{
$errorMessage = ( is_null ( $localizedError ))
? 'The requested scope is invalid, unknown, or malformed'
: $localizedError ;
$hint = ( is_null ( $localizedHint ))
? sprintf ( 'Check the `%s` scope' , $scope )
: sprintf ( $localizedHint , $scope );
return new static ( $errorMessage , 'invalid_scope' , 400 , $hint , $redirectUri );
}
2015-11-16 18:27:59 +05:30
/**
* Invalid credentials error
*
* @ return static
*/
public static function invalidCredentials ()
{
return new static ( 'The user credentials were incorrect.' , 'invalid_credentials' , 401 );
}
2016-01-13 04:29:14 +05:30
/**
* Server error
*
* @ param $hint
*
* @ return static
*/
public static function serverError ( $hint )
{
2016-01-15 05:17:41 +05:30
return new static (
'The authorization server encountered an unexpected condition which prevented it from fulfilling'
. 'the request.' ,
'server_error' ,
500 ,
$hint
);
}
2016-01-15 05:17:49 +05:30
/**
* Invalid refresh token
*
* @ return static
*/
public static function invalidRefreshToken ( $hint = null )
{
return new static ( 'The refresh token is invalid.' , 'invalid_request' , 400 , $hint );
2016-01-13 04:29:14 +05:30
}
2016-01-17 18:28:00 +05:30
/**
* Access denied
*
* @ param null | string $hint
*
* @ return static
*/
public static function accessDenied ( $hint = null )
{
return new static ( 'The resource owner or authorization server denied the request.' , 401 , $hint );
}
2015-11-16 18:27:59 +05:30
/**
* @ return string
*/
public function getErrorType ()
{
return $this -> errorType ;
}
/**
* Generate a HTTP response
*
2016-01-17 18:27:50 +05:30
* @ param \Psr\Http\Message\ResponseInterface $response
*
* @ return \Psr\Http\Message\ResponseInterface
2015-11-16 18:27:59 +05:30
*/
2016-01-15 18:32:47 +05:30
public function generateHttpResponse ( ResponseInterface $response = null )
2015-11-16 18:27:59 +05:30
{
2016-01-15 18:32:47 +05:30
if ( ! $response instanceof ResponseInterface ) {
$response = new Response ();
}
2015-11-16 18:27:59 +05:30
$headers = $this -> getHttpHeaders ();
$payload = [
'error' => $this -> errorType ,
'message' => $this -> getMessage ()
];
if ( $this -> hint !== null ) {
$payload [ 'hint' ] = $this -> hint ;
}
if ( $this -> redirectUri !== null ) {
$headers [ 'Location' ] = RedirectUri :: make ( $this -> redirectUri , $payload );
}
2016-01-15 18:32:47 +05:30
foreach ( $headers as $header => $content ) {
2016-01-16 00:02:53 +05:30
$response = $response -> withHeader ( $header , $content );
2016-01-15 18:32:47 +05:30
}
2016-01-16 00:02:53 +05:30
return $response
2016-01-15 18:32:47 +05:30
-> withStatus ( $this -> getHttpStatusCode ())
-> getBody () -> write ( json_encode ( $payload ));
2015-11-16 18:27:59 +05:30
}
/**
* Get all headers that have to be send with the error response
*
* @ return array Array with header values
*/
public function getHttpHeaders ()
{
$headers = [
'Content-type' => 'application/json'
];
// Add "WWW-Authenticate" header
//
// RFC 6749, section 5.2.:
// "If the client attempted to authenticate via the 'Authorization'
// request header field, the authorization server MUST
// respond with an HTTP 401 (Unauthorized) status code and
// include the "WWW-Authenticate" response header field
// matching the authentication scheme used by the client.
// @codeCoverageIgnoreStart
if ( $this -> errorType === 'invalid_client' ) {
$authScheme = null ;
$request = new ServerRequest ();
2016-01-15 18:32:47 +05:30
if ( isset ( $request -> getServerParams ()[ 'PHP_AUTH_USER' ]) &&
2015-11-16 18:27:59 +05:30
$request -> getServerParams ()[ 'PHP_AUTH_USER' ] !== null
) {
$authScheme = 'Basic' ;
} else {
$authHeader = $request -> getHeader ( 'authorization' );
if ( $authHeader !== []) {
if ( strpos ( $authHeader [ 0 ], 'Bearer' ) === 0 ) {
$authScheme = 'Bearer' ;
} elseif ( strpos ( $authHeader [ 0 ], 'Basic' ) === 0 ) {
$authScheme = 'Basic' ;
}
}
}
if ( $authScheme !== null ) {
$headers [] = 'WWW-Authenticate: ' . $authScheme . ' realm="OAuth"' ;
}
}
// @codeCoverageIgnoreEnd
return $headers ;
}
/**
* Returns the HTTP status code to send when the exceptions is output
*
* @ return int
*/
public function getHttpStatusCode ()
{
return $this -> httpStatusCode ;
}
2015-11-13 23:07:37 +05:30
}