1: | <?php |
2: | namespace Worldline\Acquiring\Sdk\Authentication; |
3: | |
4: | use InvalidArgumentException; |
5: | use Worldline\Acquiring\Sdk\Communication\Connection; |
6: | use Worldline\Acquiring\Sdk\Communication\DefaultConnection; |
7: | use Worldline\Acquiring\Sdk\Communication\ResponseBuilder; |
8: | use Worldline\Acquiring\Sdk\CommunicatorConfiguration; |
9: | use Worldline\Acquiring\Sdk\JSON\JSONUtil; |
10: | |
11: | |
12: | |
13: | |
14: | |
15: | |
16: | class OAuth2Authenticator implements Authenticator |
17: | { |
18: | |
19: | |
20: | |
21: | private const TOKEN_TYPES = [ |
22: | '' => [ |
23: | 'processing_payment', 'processing_refund', 'processing_credittransfer', 'processing_accountverification', |
24: | 'processing_balanceinquiry', 'processing_operation_reverse', 'processing_dcc_rate', 'services_ping' |
25: | ], |
26: | ]; |
27: | |
28: | |
29: | private $oauth2TokenUri; |
30: | |
31: | |
32: | private $oauth2ClientId; |
33: | |
34: | |
35: | private $oauth2ClientSecret; |
36: | |
37: | |
38: | private $tokenCache; |
39: | |
40: | |
41: | private $communicatorConfiguration; |
42: | |
43: | |
44: | |
45: | |
46: | |
47: | |
48: | public function __construct( |
49: | CommunicatorConfiguration $communicatorConfiguration, |
50: | $oauth2TokenUri, |
51: | OAuth2TokenCache $tokenCache = null |
52: | ) { |
53: | $this->communicatorConfiguration = $communicatorConfiguration; |
54: | $this->oauth2TokenUri = $oauth2TokenUri; |
55: | $this->oauth2ClientId = $communicatorConfiguration->getOAuth2ClientId(); |
56: | $this->oauth2ClientSecret = $communicatorConfiguration->getOAuth2ClientSecret(); |
57: | $this->tokenCache = $tokenCache ?: new DefaultOAuth2TokenCache(); |
58: | } |
59: | |
60: | |
61: | |
62: | |
63: | |
64: | |
65: | |
66: | public function getAuthorization($httpMethod, $uriPath, $requestHeaders) |
67: | { |
68: | $tokenType = self::getTokenType($uriPath); |
69: | $oauth2AccessToken = $this->tokenCache->getOAuth2AccessToken($tokenType); |
70: | if ($oauth2AccessToken) { |
71: | return 'Bearer ' . $oauth2AccessToken; |
72: | } |
73: | |
74: | $startTime = time(); |
75: | |
76: | $oauth2RequestHeaders = array(); |
77: | $oauth2RequestHeaders['Content-Type'] = 'application/x-www-form-urlencoded'; |
78: | |
79: | $requestBody = sprintf('grant_type=client_credentials&client_id=%s&client_secret=%s&scope=%s', $this->oauth2ClientId, $this->oauth2ClientSecret, self::getScopes($tokenType)); |
80: | |
81: | $responseBuilder = new ResponseBuilder(); |
82: | $responseHandler = function ($httpStatusCode, $data, $headers) use ($responseBuilder) { |
83: | $responseBuilder->setHttpStatusCode($httpStatusCode); |
84: | $responseBuilder->setHeaders($headers); |
85: | $responseBuilder->appendBody($data); |
86: | }; |
87: | |
88: | $connection = $this->createConnection(); |
89: | $connection->post($this->oauth2TokenUri, $oauth2RequestHeaders, $requestBody, $responseHandler); |
90: | |
91: | $response = $responseBuilder->getResponse(); |
92: | |
93: | $responseObject = JSONUtil::decode($response->getBody()); |
94: | |
95: | if ($response->getHttpStatusCode() !== 200) { |
96: | if (property_exists($responseObject, 'error_description')) { |
97: | throw new OAuth2Exception(sprintf( |
98: | 'There was an error while retrieving the OAuth2 access token: %s - %s', |
99: | $responseObject->error, |
100: | $responseObject->error_description |
101: | )); |
102: | } |
103: | throw new OAuth2Exception(sprintf( |
104: | 'There was an error while retrieving the OAuth2 access token: %s', |
105: | $responseObject->error |
106: | )); |
107: | } |
108: | $oauth2AccessToken = $responseObject->access_token; |
109: | $expirationTimestamp = $startTime + $responseObject->expires_in; |
110: | $this->tokenCache->storeOAuth2AccessToken($tokenType, $oauth2AccessToken, $expirationTimestamp); |
111: | |
112: | return 'Bearer ' . $oauth2AccessToken; |
113: | } |
114: | |
115: | |
116: | |
117: | |
118: | protected function createConnection() |
119: | { |
120: | return new DefaultConnection($this->communicatorConfiguration); |
121: | } |
122: | |
123: | private static function getTokenType($fullPath) |
124: | { |
125: | foreach (self::TOKEN_TYPES as $tokenType => $scopes) { |
126: | if (self::endsWith($fullPath, $tokenType) || self::contains($fullPath, $tokenType . '/')) { |
127: | return $tokenType; |
128: | } |
129: | } |
130: | |
131: | throw new InvalidArgumentException("Scope could not be found for path $fullPath"); |
132: | } |
133: | |
134: | private static function getScopes($tokenType) |
135: | { |
136: | if (!array_key_exists($tokenType, self::TOKEN_TYPES)) { |
137: | throw new InvalidArgumentException("Token type $tokenType does not exist."); |
138: | } |
139: | |
140: | return join(" ", self::TOKEN_TYPES[$tokenType]); |
141: | } |
142: | |
143: | private static function endsWith($haystack, $needle) |
144: | { |
145: | return substr_compare($haystack, $needle, -strlen($needle)) === 0; |
146: | } |
147: | |
148: | private static function contains($haystack, $needle) |
149: | { |
150: | return strpos($haystack, $needle) !== false; |
151: | } |
152: | } |
153: | |