2016-11-29 23:15:56 +03:00
|
|
|
<?php
|
2019-09-22 00:17:21 +03:00
|
|
|
declare(strict_types=1);
|
2016-11-29 23:15:56 +03:00
|
|
|
|
2024-12-06 01:34:09 +01:00
|
|
|
namespace common\components\OAuth2\Grants;
|
2017-06-17 21:05:36 +03:00
|
|
|
|
2019-12-09 19:31:54 +03:00
|
|
|
use api\components\Tokens\TokenReader;
|
2024-12-02 15:10:55 +05:00
|
|
|
use Carbon\FactoryImmutable;
|
2024-12-06 01:34:09 +01:00
|
|
|
use common\components\OAuth2\CryptTrait;
|
2019-09-22 02:42:08 +03:00
|
|
|
use common\models\OauthSession;
|
2019-12-09 19:31:54 +03:00
|
|
|
use InvalidArgumentException;
|
2024-12-02 15:10:55 +05:00
|
|
|
use Lcobucci\JWT\Validation\Constraint\LooseValidAt;
|
|
|
|
use Lcobucci\JWT\Validation\Validator;
|
2019-09-22 00:17:21 +03:00
|
|
|
use League\OAuth2\Server\Entities\AccessTokenEntityInterface;
|
|
|
|
use League\OAuth2\Server\Entities\RefreshTokenEntityInterface;
|
2019-09-22 02:42:08 +03:00
|
|
|
use League\OAuth2\Server\Exception\OAuthServerException;
|
2019-09-22 00:17:21 +03:00
|
|
|
use League\OAuth2\Server\Grant\RefreshTokenGrant as BaseRefreshTokenGrant;
|
2019-09-22 02:42:08 +03:00
|
|
|
use Psr\Http\Message\ServerRequestInterface;
|
2024-12-02 15:10:55 +05:00
|
|
|
use Throwable;
|
2019-09-22 02:42:08 +03:00
|
|
|
use Yii;
|
2017-06-17 21:05:36 +03:00
|
|
|
|
2024-12-06 01:34:09 +01:00
|
|
|
final class RefreshTokenGrant extends BaseRefreshTokenGrant {
|
2019-12-06 14:37:51 +03:00
|
|
|
use CryptTrait;
|
2016-11-29 23:15:56 +03:00
|
|
|
|
2019-09-22 02:42:08 +03:00
|
|
|
/**
|
2019-09-23 00:53:13 +03:00
|
|
|
* Previously, refresh tokens were stored in Redis.
|
2019-09-22 02:42:08 +03:00
|
|
|
* If received refresh token is matches the legacy token template,
|
|
|
|
* restore the information from the legacy storage.
|
|
|
|
*
|
2024-12-06 01:34:09 +01:00
|
|
|
* @inheritDoc
|
2019-09-22 02:42:08 +03:00
|
|
|
*/
|
2024-12-02 15:10:55 +05:00
|
|
|
protected function validateOldRefreshToken(ServerRequestInterface $request, string $clientId): array {
|
2019-09-22 02:42:08 +03:00
|
|
|
$refreshToken = $this->getRequestParameter('refresh_token', $request);
|
|
|
|
if ($refreshToken !== null && mb_strlen($refreshToken) === 40) {
|
|
|
|
return $this->validateLegacyRefreshToken($refreshToken);
|
|
|
|
}
|
|
|
|
|
2019-12-09 19:31:54 +03:00
|
|
|
return $this->validateAccessToken($refreshToken);
|
2019-09-22 02:42:08 +03:00
|
|
|
}
|
|
|
|
|
2017-07-13 13:44:06 +03:00
|
|
|
/**
|
2019-09-22 00:17:21 +03:00
|
|
|
* Currently we're not rotating refresh tokens.
|
2024-12-02 15:10:55 +05:00
|
|
|
* So we're overriding this method to always return null, which means,
|
2019-09-22 00:17:21 +03:00
|
|
|
* that refresh_token will not be issued.
|
2017-07-13 13:44:06 +03:00
|
|
|
*/
|
2019-09-22 00:17:21 +03:00
|
|
|
protected function issueRefreshToken(AccessTokenEntityInterface $accessToken): ?RefreshTokenEntityInterface {
|
|
|
|
return null;
|
2016-11-30 02:19:14 +03:00
|
|
|
}
|
|
|
|
|
2019-12-04 13:40:12 +03:00
|
|
|
/**
|
2024-12-06 01:34:09 +01:00
|
|
|
* @return array<string, mixed>
|
2019-12-04 13:40:12 +03:00
|
|
|
* @throws OAuthServerException
|
|
|
|
*/
|
2019-09-22 02:42:08 +03:00
|
|
|
private function validateLegacyRefreshToken(string $refreshToken): array {
|
|
|
|
$result = Yii::$app->redis->get("oauth:refresh:tokens:{$refreshToken}");
|
|
|
|
if ($result === null) {
|
|
|
|
throw OAuthServerException::invalidRefreshToken('Token has been revoked');
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
[
|
|
|
|
'access_token_id' => $accessTokenId,
|
|
|
|
'session_id' => $sessionId,
|
2024-12-02 15:10:55 +05:00
|
|
|
] = json_decode((string)$result, true, 512, JSON_THROW_ON_ERROR);
|
|
|
|
} catch (Throwable $e) {
|
2019-09-22 02:42:08 +03:00
|
|
|
throw OAuthServerException::invalidRefreshToken('Cannot decrypt the refresh token', $e);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @var OauthSession|null $relatedSession */
|
|
|
|
$relatedSession = OauthSession::findOne(['legacy_id' => $sessionId]);
|
|
|
|
if ($relatedSession === null) {
|
|
|
|
throw OAuthServerException::invalidRefreshToken('Token has been revoked');
|
|
|
|
}
|
|
|
|
|
|
|
|
return [
|
|
|
|
'client_id' => $relatedSession->client_id,
|
|
|
|
'refresh_token_id' => $refreshToken,
|
|
|
|
'access_token_id' => $accessTokenId,
|
|
|
|
'scopes' => $relatedSession->getScopes(),
|
|
|
|
'user_id' => $relatedSession->account_id,
|
|
|
|
'expire_time' => null,
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
2019-12-09 19:31:54 +03:00
|
|
|
/**
|
2024-12-06 01:34:09 +01:00
|
|
|
* @return array<string, mixed>
|
2019-12-09 19:31:54 +03:00
|
|
|
* @throws OAuthServerException
|
|
|
|
*/
|
|
|
|
private function validateAccessToken(string $jwt): array {
|
|
|
|
try {
|
|
|
|
$token = Yii::$app->tokens->parse($jwt);
|
|
|
|
} catch (InvalidArgumentException $e) {
|
|
|
|
throw OAuthServerException::invalidRefreshToken('Cannot decrypt the refresh token', $e);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!Yii::$app->tokens->verify($token)) {
|
|
|
|
throw OAuthServerException::invalidRefreshToken('Cannot decrypt the refresh token');
|
|
|
|
}
|
|
|
|
|
2024-12-02 15:10:55 +05:00
|
|
|
if (!(new Validator())->validate($token, new LooseValidAt(FactoryImmutable::getDefaultInstance()))) {
|
2019-12-09 19:31:54 +03:00
|
|
|
throw OAuthServerException::invalidRefreshToken('Token has expired');
|
|
|
|
}
|
|
|
|
|
|
|
|
$reader = new TokenReader($token);
|
|
|
|
|
|
|
|
return [
|
|
|
|
'client_id' => $reader->getClientId(),
|
|
|
|
'refresh_token_id' => '', // This value used only to invalidate old token
|
|
|
|
'access_token_id' => '', // This value used only to invalidate old token
|
|
|
|
'scopes' => $reader->getScopes(),
|
|
|
|
'user_id' => $reader->getAccountId(),
|
|
|
|
'expire_time' => null,
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
2016-11-29 23:15:56 +03:00
|
|
|
}
|