Completed
Push — master ( 9bfb69...8c33b5 )
by Andrew
15s queued 12s
created

RefreshTokenGrant::respondToAccessTokenRequest()   B

Complexity

Conditions 6
Paths 13

Size

Total Lines 46
Code Lines 22

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 20
CRAP Score 6

Importance

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