Completed
Pull Request — master (#1102)
by
unknown
08:28
created

RefreshTokenGrant::__construct()   A

Complexity

Conditions 1
Paths 1

Size

Total Lines 5
Code Lines 2

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 0
CRAP Score 2

Importance

Changes 0
Metric Value
cc 1
eloc 2
nc 1
nop 1
dl 0
loc 5
ccs 0
cts 4
cp 0
crap 2
rs 10
c 0
b 0
f 0
1
<?php
2
/**
3
 * OAuth 2.0 Refresh token grant.
4
 *
5
 * @author      Alex Bilbie <[email protected]>
6
 * @copyright   Copyright (c) Alex Bilbie
7
 * @license     http://mit-license.org/
8
 *
9
 * @link        https://github.com/thephpleague/oauth2-server
10
 */
11
12
namespace League\OAuth2\Server\Grant;
13
14
use DateInterval;
15
use Exception;
16
use League\OAuth2\Server\Exception\OAuthServerException;
17
use League\OAuth2\Server\Repositories\RefreshTokenRepositoryInterface;
18
use League\OAuth2\Server\RequestEvent;
19
use League\OAuth2\Server\ResponseTypes\ResponseTypeInterface;
20
use Psr\Http\Message\ServerRequestInterface;
21
22
/**
23
 * Refresh token grant.
24
 */
25
class RefreshTokenGrant extends AbstractGrant
26
{
27
    /**
28
     * @param RefreshTokenRepositoryInterface $refreshTokenRepository
29
     */
30
    public function __construct(RefreshTokenRepositoryInterface $refreshTokenRepository)
31
    {
32
        $this->setRefreshTokenRepository($refreshTokenRepository);
33
34
        $this->refreshTokenTTL = new DateInterval('P1M');
35
    }
36
37
    /**
38
     * {@inheritdoc}
39
     */
40
    public function respondToAccessTokenRequest(
41
        ServerRequestInterface $request,
42
        ResponseTypeInterface $responseType,
43
        DateInterval $accessTokenTTL
44
    ) {
45
        // Validate request
46
        $client = $this->validateClient($request);
47
        $oldRefreshToken = $this->validateOldRefreshToken($request, $client->getIdentifier());
48
        $scopes = $this->validateScopes($this->getRequestParameter(
49
            'scope',
50
            $request,
51
            \implode(self::SCOPE_DELIMITER_STRING, $oldRefreshToken['scopes']))
52
        );
53
54
        // The OAuth spec says that a refreshed access token can have the original scopes or fewer so ensure
55
        // the request doesn't include any new scopes
56
        foreach ($scopes as $scope) {
57
            if (\in_array($scope->getIdentifier(), $oldRefreshToken['scopes'], true) === false) {
58
                throw OAuthServerException::invalidScope($scope->getIdentifier());
59
            }
60
        }
61
62
        // Expire old tokens
63
        $this->accessTokenRepository->revokeAccessToken($oldRefreshToken['access_token_id']);
64
        $this->refreshTokenRepository->revokeRefreshToken($oldRefreshToken['refresh_token_id']);
65
66
        // Issue and persist new access token
67
        $accessToken = $this->issueAccessToken($accessTokenTTL, $client, $oldRefreshToken['user_id'], $scopes);
68
        $this->getEmitter()->emit(new RequestEvent(RequestEvent::ACCESS_TOKEN_ISSUED, $request));
69
        $responseType->setAccessToken($accessToken);
70
71
        // Issue and persist new refresh token if given
72
        $refreshToken = $this->issueRefreshToken($accessToken);
73
74
        if ($refreshToken !== null) {
75
            $this->getEmitter()->emit(new RequestEvent(RequestEvent::REFRESH_TOKEN_ISSUED, $request));
76
            $responseType->setRefreshToken($refreshToken);
77
        }
78
79
        return $responseType;
80
    }
81
82
    /**
83
     * @param ServerRequestInterface $request
84
     * @param string                 $clientId
85
     *
86
     * @throws OAuthServerException
87
     *
88
     * @return array
89
     */
90
    protected function validateOldRefreshToken(ServerRequestInterface $request, $clientId)
91
    {
92
        $encryptedRefreshToken = $this->getRequestParameter('refresh_token', $request);
93
        if (\is_null($encryptedRefreshToken)) {
94
            throw OAuthServerException::invalidRequest('refresh_token');
95
        }
96
97
        // Validate refresh token
98
        try {
99
            $refreshToken = $this->decrypt($encryptedRefreshToken);
100
        } catch (Exception $e) {
101
            throw OAuthServerException::invalidRefreshToken('Cannot decrypt the refresh token', $e);
102
        }
103
104
        $refreshTokenData = \json_decode($refreshToken, true);
105
        if ($refreshTokenData['client_id'] !== $clientId) {
106
            $this->getEmitter()->emit(new RequestEvent(RequestEvent::REFRESH_TOKEN_CLIENT_FAILED, $request));
107
            throw OAuthServerException::invalidRefreshToken('Token is not linked to client');
108
        }
109
110
        if ($refreshTokenData['expire_time'] < \time()) {
111
            throw OAuthServerException::invalidRefreshToken('Token has expired');
112
        }
113
114
        if ($this->refreshTokenRepository->isRefreshTokenRevoked($refreshTokenData['refresh_token_id']) === true) {
115
            throw OAuthServerException::invalidRefreshToken('Token has been revoked');
116
        }
117
118
        return $refreshTokenData;
119
    }
120
121
    /**
122
     * {@inheritdoc}
123
     */
124
    public function getIdentifier()
125
    {
126
        return 'refresh_token';
127
    }
128
}
129