2017-06-12 14:36:20 +03:00
|
|
|
<?php
|
2018-02-28 01:27:35 +03:00
|
|
|
namespace api\modules\oauth\models;
|
2017-06-12 14:36:20 +03:00
|
|
|
|
|
|
|
use api\components\OAuth2\Exception\AcceptRequiredException;
|
|
|
|
use api\components\OAuth2\Exception\AccessDeniedException;
|
2017-06-17 21:05:36 +03:00
|
|
|
use api\components\OAuth2\Grants\AuthCodeGrant;
|
|
|
|
use api\components\OAuth2\Grants\AuthorizeParams;
|
2017-06-12 14:36:20 +03:00
|
|
|
use common\models\Account;
|
|
|
|
use common\models\OauthClient;
|
2017-10-18 02:37:01 +03:00
|
|
|
use common\rbac\Permissions as P;
|
2017-06-12 14:36:20 +03:00
|
|
|
use League\OAuth2\Server\AuthorizationServer;
|
|
|
|
use League\OAuth2\Server\Exception\InvalidGrantException;
|
|
|
|
use League\OAuth2\Server\Exception\OAuthException;
|
|
|
|
use League\OAuth2\Server\Grant\GrantTypeInterface;
|
|
|
|
use Yii;
|
|
|
|
use yii\helpers\ArrayHelper;
|
|
|
|
|
|
|
|
class OauthProcess {
|
|
|
|
|
2017-10-18 02:37:01 +03:00
|
|
|
private const INTERNAL_PERMISSIONS_TO_PUBLIC_SCOPES = [
|
|
|
|
P::OBTAIN_OWN_ACCOUNT_INFO => 'account_info',
|
|
|
|
P::OBTAIN_ACCOUNT_EMAIL => 'account_email',
|
|
|
|
];
|
|
|
|
|
2017-06-12 14:36:20 +03:00
|
|
|
/**
|
|
|
|
* @var AuthorizationServer
|
|
|
|
*/
|
|
|
|
private $server;
|
|
|
|
|
|
|
|
public function __construct(AuthorizationServer $server) {
|
|
|
|
$this->server = $server;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-07-15 01:59:56 +03:00
|
|
|
* A request that should check the passed OAuth2 authorization params and build a response
|
|
|
|
* for our frontend application.
|
2017-06-12 14:36:20 +03:00
|
|
|
*
|
2019-07-15 01:59:56 +03:00
|
|
|
* The input data is the standard GET parameters list according to the OAuth2 standard:
|
2017-06-12 14:36:20 +03:00
|
|
|
* $_GET = [
|
|
|
|
* client_id,
|
|
|
|
* redirect_uri,
|
|
|
|
* response_type,
|
|
|
|
* scope,
|
|
|
|
* state,
|
|
|
|
* ];
|
|
|
|
*
|
2019-07-15 01:59:56 +03:00
|
|
|
* In addition, you can pass the description value to override the application's description.
|
2017-06-12 14:36:20 +03:00
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function validate(): array {
|
|
|
|
try {
|
|
|
|
$authParams = $this->getAuthorizationCodeGrant()->checkAuthorizeParams();
|
2017-06-17 21:05:36 +03:00
|
|
|
$client = $authParams->getClient();
|
2018-02-28 01:27:35 +03:00
|
|
|
/** @var OauthClient $clientModel */
|
|
|
|
$clientModel = $this->findClient($client->getId());
|
2017-06-12 14:36:20 +03:00
|
|
|
$response = $this->buildSuccessResponse(
|
|
|
|
Yii::$app->request->getQueryParams(),
|
|
|
|
$clientModel,
|
2017-06-17 21:05:36 +03:00
|
|
|
$authParams->getScopes()
|
2017-06-12 14:36:20 +03:00
|
|
|
);
|
|
|
|
} catch (OAuthException $e) {
|
|
|
|
$response = $this->buildErrorResponse($e);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $response;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-07-15 01:59:56 +03:00
|
|
|
* This method generates authorization_code and a link
|
|
|
|
* for the user's further redirect to the client's site.
|
2017-06-12 14:36:20 +03:00
|
|
|
*
|
2019-07-15 01:59:56 +03:00
|
|
|
* The input data are the same parameters that were necessary for validation request:
|
2017-06-12 14:36:20 +03:00
|
|
|
* $_GET = [
|
|
|
|
* client_id,
|
|
|
|
* redirect_uri,
|
|
|
|
* response_type,
|
|
|
|
* scope,
|
|
|
|
* state,
|
|
|
|
* ];
|
|
|
|
*
|
2019-07-15 01:59:56 +03:00
|
|
|
* Also, the accept field, which shows that the user has clicked on the "Accept" button.
|
|
|
|
* If the field is present, it will be interpreted as any value resulting in false positives.
|
|
|
|
* Otherwise, the value will be interpreted as "true".
|
2017-06-12 14:36:20 +03:00
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function complete(): array {
|
|
|
|
try {
|
2017-11-21 19:58:55 +03:00
|
|
|
Yii::$app->statsd->inc('oauth.complete.attempt');
|
2017-06-12 14:36:20 +03:00
|
|
|
$grant = $this->getAuthorizationCodeGrant();
|
|
|
|
$authParams = $grant->checkAuthorizeParams();
|
2018-02-28 01:27:35 +03:00
|
|
|
/** @var Account $account */
|
2017-09-19 20:06:16 +03:00
|
|
|
$account = Yii::$app->user->identity->getAccount();
|
2017-06-12 14:36:20 +03:00
|
|
|
/** @var \common\models\OauthClient $clientModel */
|
2018-02-28 01:27:35 +03:00
|
|
|
$clientModel = $this->findClient($authParams->getClient()->getId());
|
2017-06-12 14:36:20 +03:00
|
|
|
|
|
|
|
if (!$this->canAutoApprove($account, $clientModel, $authParams)) {
|
2017-11-21 19:58:55 +03:00
|
|
|
Yii::$app->statsd->inc('oauth.complete.approve_required');
|
2017-06-12 14:36:20 +03:00
|
|
|
$isAccept = Yii::$app->request->post('accept');
|
|
|
|
if ($isAccept === null) {
|
|
|
|
throw new AcceptRequiredException();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$isAccept) {
|
2017-06-17 21:05:36 +03:00
|
|
|
throw new AccessDeniedException($authParams->getRedirectUri());
|
2017-06-12 14:36:20 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$redirectUri = $grant->newAuthorizeRequest('user', $account->id, $authParams);
|
|
|
|
$response = [
|
|
|
|
'success' => true,
|
|
|
|
'redirectUri' => $redirectUri,
|
|
|
|
];
|
2017-11-21 19:58:55 +03:00
|
|
|
Yii::$app->statsd->inc('oauth.complete.success');
|
2017-06-12 14:36:20 +03:00
|
|
|
} catch (OAuthException $e) {
|
2017-11-21 19:58:55 +03:00
|
|
|
if (!$e instanceof AcceptRequiredException) {
|
|
|
|
Yii::$app->statsd->inc('oauth.complete.fail');
|
|
|
|
}
|
|
|
|
|
2017-06-12 14:36:20 +03:00
|
|
|
$response = $this->buildErrorResponse($e);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $response;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-07-15 01:59:56 +03:00
|
|
|
* The method is executed by the application server to which auth_token or refresh_token was given.
|
2017-06-12 14:36:20 +03:00
|
|
|
*
|
2019-07-15 01:59:56 +03:00
|
|
|
* Input data is a standard list of POST parameters according to the OAuth2 standard:
|
2018-01-02 20:45:04 +03:00
|
|
|
* $_POST = [
|
2017-06-12 14:36:20 +03:00
|
|
|
* client_id,
|
|
|
|
* client_secret,
|
|
|
|
* redirect_uri,
|
|
|
|
* code,
|
|
|
|
* grant_type,
|
|
|
|
* ]
|
2019-07-15 01:59:56 +03:00
|
|
|
* for request with grant_type = authentication_code:
|
2018-01-02 20:45:04 +03:00
|
|
|
* $_POST = [
|
2017-06-12 14:36:20 +03:00
|
|
|
* client_id,
|
|
|
|
* client_secret,
|
|
|
|
* refresh_token,
|
|
|
|
* grant_type,
|
|
|
|
* ]
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function getToken(): array {
|
2018-01-02 20:45:04 +03:00
|
|
|
$grantType = Yii::$app->request->post('grant_type', 'null');
|
2017-06-12 14:36:20 +03:00
|
|
|
try {
|
2018-01-02 20:45:04 +03:00
|
|
|
Yii::$app->statsd->inc("oauth.issueToken_{$grantType}.attempt");
|
2017-06-12 14:36:20 +03:00
|
|
|
$response = $this->server->issueAccessToken();
|
2018-01-02 20:45:04 +03:00
|
|
|
$clientId = Yii::$app->request->post('client_id');
|
|
|
|
Yii::$app->statsd->inc("oauth.issueToken_client.{$clientId}");
|
|
|
|
Yii::$app->statsd->inc("oauth.issueToken_{$grantType}.success");
|
2017-06-12 14:36:20 +03:00
|
|
|
} catch (OAuthException $e) {
|
2018-01-02 20:45:04 +03:00
|
|
|
Yii::$app->statsd->inc("oauth.issueToken_{$grantType}.fail");
|
2017-06-12 14:36:20 +03:00
|
|
|
Yii::$app->response->statusCode = $e->httpStatusCode;
|
|
|
|
$response = [
|
|
|
|
'error' => $e->errorType,
|
|
|
|
'message' => $e->getMessage(),
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
return $response;
|
|
|
|
}
|
|
|
|
|
2018-02-28 01:27:35 +03:00
|
|
|
private function findClient(string $clientId): ?OauthClient {
|
|
|
|
return OauthClient::findOne($clientId);
|
|
|
|
}
|
|
|
|
|
2017-06-12 14:36:20 +03:00
|
|
|
/**
|
2019-07-15 01:59:56 +03:00
|
|
|
* The method checks whether the current user can be automatically authorized for the specified client
|
|
|
|
* without requesting access to the necessary list of scopes
|
2017-06-12 14:36:20 +03:00
|
|
|
*
|
2017-06-17 21:05:36 +03:00
|
|
|
* @param Account $account
|
2017-06-12 14:36:20 +03:00
|
|
|
* @param OauthClient $client
|
2017-06-17 21:05:36 +03:00
|
|
|
* @param AuthorizeParams $oauthParams
|
2017-06-12 14:36:20 +03:00
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
2017-06-17 21:05:36 +03:00
|
|
|
private function canAutoApprove(Account $account, OauthClient $client, AuthorizeParams $oauthParams): bool {
|
2017-06-12 14:36:20 +03:00
|
|
|
if ($client->is_trusted) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @var \common\models\OauthSession|null $session */
|
|
|
|
$session = $account->getOauthSessions()->andWhere(['client_id' => $client->id])->one();
|
|
|
|
if ($session !== null) {
|
|
|
|
$existScopes = $session->getScopes()->members();
|
2017-06-17 21:05:36 +03:00
|
|
|
if (empty(array_diff(array_keys($oauthParams->getScopes()), $existScopes))) {
|
2017-06-12 14:36:20 +03:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-06-17 21:05:36 +03:00
|
|
|
/**
|
|
|
|
* @param array $queryParams
|
|
|
|
* @param OauthClient $client
|
|
|
|
* @param \api\components\OAuth2\Entities\ScopeEntity[] $scopes
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
private function buildSuccessResponse(array $queryParams, OauthClient $client, array $scopes): array {
|
2017-06-12 14:36:20 +03:00
|
|
|
return [
|
|
|
|
'success' => true,
|
2019-07-15 01:59:56 +03:00
|
|
|
// We return only those keys which are related to the OAuth2 standard parameters
|
2017-06-17 21:05:36 +03:00
|
|
|
'oAuth' => array_intersect_key($queryParams, array_flip([
|
2017-06-12 14:36:20 +03:00
|
|
|
'client_id',
|
|
|
|
'redirect_uri',
|
|
|
|
'response_type',
|
|
|
|
'scope',
|
|
|
|
'state',
|
|
|
|
])),
|
|
|
|
'client' => [
|
|
|
|
'id' => $client->id,
|
|
|
|
'name' => $client->name,
|
2017-06-17 21:05:36 +03:00
|
|
|
'description' => ArrayHelper::getValue($queryParams, 'description', $client->description),
|
2017-06-12 14:36:20 +03:00
|
|
|
],
|
|
|
|
'session' => [
|
2017-10-18 02:37:01 +03:00
|
|
|
'scopes' => $this->fixScopesNames(array_keys($scopes)),
|
2017-06-12 14:36:20 +03:00
|
|
|
],
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
2017-10-18 02:37:01 +03:00
|
|
|
private function fixScopesNames(array $scopes): array {
|
|
|
|
foreach ($scopes as &$scope) {
|
|
|
|
if (isset(self::INTERNAL_PERMISSIONS_TO_PUBLIC_SCOPES[$scope])) {
|
|
|
|
$scope = self::INTERNAL_PERMISSIONS_TO_PUBLIC_SCOPES[$scope];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $scopes;
|
|
|
|
}
|
|
|
|
|
2017-06-12 14:36:20 +03:00
|
|
|
private function buildErrorResponse(OAuthException $e): array {
|
|
|
|
$response = [
|
|
|
|
'success' => false,
|
|
|
|
'error' => $e->errorType,
|
|
|
|
'parameter' => $e->parameter,
|
|
|
|
'statusCode' => $e->httpStatusCode,
|
|
|
|
];
|
|
|
|
|
|
|
|
if ($e->shouldRedirect()) {
|
|
|
|
$response['redirectUri'] = $e->getRedirectUri();
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($e->httpStatusCode !== 200) {
|
|
|
|
Yii::$app->response->setStatusCode($e->httpStatusCode);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $response;
|
|
|
|
}
|
|
|
|
|
|
|
|
private function getGrant(string $grantType = null): GrantTypeInterface {
|
|
|
|
return $this->server->getGrantType($grantType ?? Yii::$app->request->get('grant_type'));
|
|
|
|
}
|
|
|
|
|
|
|
|
private function getAuthorizationCodeGrant(): AuthCodeGrant {
|
|
|
|
/** @var GrantTypeInterface $grantType */
|
|
|
|
$grantType = $this->getGrant('authorization_code');
|
|
|
|
if (!$grantType instanceof AuthCodeGrant) {
|
|
|
|
throw new InvalidGrantException('authorization_code grant have invalid realisation');
|
|
|
|
}
|
|
|
|
|
|
|
|
return $grantType;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|