OIDCRoleArnCredentialsProvider.php 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257
  1. <?php
  2. namespace AlibabaCloud\Credentials\Providers;
  3. use AlibabaCloud\Credentials\Utils\Helper;
  4. use AlibabaCloud\Credentials\Utils\Filter;
  5. use AlibabaCloud\Credentials\Request\Request;
  6. use GuzzleHttp\Psr7\Uri;
  7. use GuzzleHttp\Exception\GuzzleException;
  8. use InvalidArgumentException;
  9. use RuntimeException;
  10. use Exception;
  11. /**
  12. * @internal This class is intended for internal use within the package.
  13. * Class OIDCRoleArnCredentialsProvider
  14. *
  15. * @package AlibabaCloud\Credentials\Providers
  16. */
  17. class OIDCRoleArnCredentialsProvider extends SessionCredentialsProvider
  18. {
  19. /**
  20. * @var string
  21. */
  22. private $roleArn;
  23. /**
  24. * @var string
  25. */
  26. private $oidcProviderArn;
  27. /**
  28. * @var string
  29. */
  30. private $oidcTokenFilePath;
  31. /**
  32. * @var string
  33. */
  34. private $roleSessionName;
  35. /**
  36. * @description role session expiration
  37. * @example 3600
  38. * @var int
  39. */
  40. private $durationSeconds = 3600;
  41. /**
  42. * @var string
  43. */
  44. private $policy;
  45. /**
  46. * @var string
  47. */
  48. private $stsEndpoint;
  49. /**
  50. * @var int
  51. */
  52. private $connectTimeout = 5;
  53. /**
  54. * @var int
  55. */
  56. private $readTimeout = 5;
  57. /**
  58. * OIDCRoleArnCredentialsProvider constructor.
  59. *
  60. * @param array $params
  61. * @param array $options
  62. */
  63. public function __construct(array $params = [], array $options = [])
  64. {
  65. $this->filterOptions($options);
  66. $this->filterRoleArn($params);
  67. $this->filterOIDCProviderArn($params);
  68. $this->filterOIDCTokenFilePath($params);
  69. $this->filterRoleSessionName($params);
  70. $this->filterDurationSeconds($params);
  71. $this->filterPolicy($params);
  72. $this->filterSTSEndpoint($params);
  73. }
  74. private function filterRoleArn(array $params)
  75. {
  76. if (Helper::envNotEmpty('ALIBABA_CLOUD_ROLE_ARN')) {
  77. $this->roleArn = Helper::env('ALIBABA_CLOUD_ROLE_ARN');
  78. }
  79. if (isset($params['roleArn'])) {
  80. $this->roleArn = $params['roleArn'];
  81. }
  82. Filter::roleArn($this->roleArn);
  83. }
  84. private function filterOIDCProviderArn(array $params)
  85. {
  86. if (Helper::envNotEmpty('ALIBABA_CLOUD_OIDC_PROVIDER_ARN')) {
  87. $this->oidcProviderArn = Helper::env('ALIBABA_CLOUD_OIDC_PROVIDER_ARN');
  88. }
  89. if (isset($params['oidcProviderArn'])) {
  90. $this->oidcProviderArn = $params['oidcProviderArn'];
  91. }
  92. Filter::oidcProviderArn($this->oidcProviderArn);
  93. }
  94. private function filterOIDCTokenFilePath(array $params)
  95. {
  96. if (Helper::envNotEmpty('ALIBABA_CLOUD_OIDC_TOKEN_FILE')) {
  97. $this->oidcTokenFilePath = Helper::env('ALIBABA_CLOUD_OIDC_TOKEN_FILE');
  98. }
  99. if (isset($params['oidcTokenFilePath'])) {
  100. $this->oidcTokenFilePath = $params['oidcTokenFilePath'];
  101. }
  102. Filter::oidcTokenFilePath($this->oidcTokenFilePath);
  103. }
  104. private function filterRoleSessionName(array $params)
  105. {
  106. if (Helper::envNotEmpty('ALIBABA_CLOUD_ROLE_SESSION_NAME')) {
  107. $this->roleSessionName = Helper::env('ALIBABA_CLOUD_ROLE_SESSION_NAME');
  108. }
  109. if (isset($params['roleSessionName'])) {
  110. $this->roleSessionName = $params['roleSessionName'];
  111. }
  112. if (is_null($this->roleSessionName) || $this->roleSessionName === '') {
  113. $this->roleSessionName = 'phpSdkRoleSessionName';
  114. }
  115. }
  116. private function filterDurationSeconds(array $params)
  117. {
  118. if (isset($params['durationSeconds'])) {
  119. if (is_int($params['durationSeconds'])) {
  120. $this->durationSeconds = $params['durationSeconds'];
  121. }
  122. }
  123. if ($this->durationSeconds < 900) {
  124. throw new InvalidArgumentException('Role session expiration should be in the range of 900s - max session duration');
  125. }
  126. }
  127. private function filterPolicy(array $params)
  128. {
  129. if (isset($params['policy'])) {
  130. if (is_string($params['policy'])) {
  131. $this->policy = $params['policy'];
  132. }
  133. if (is_array($params['policy'])) {
  134. $this->policy = json_encode($params['policy']);
  135. }
  136. }
  137. }
  138. private function filterSTSEndpoint(array $params)
  139. {
  140. if (Helper::envNotEmpty('ALIBABA_CLOUD_STS_REGION')) {
  141. $this->stsEndpoint = 'sts' . Helper::env('ALIBABA_CLOUD_STS_REGION') . '.aliyuncs.com';
  142. }
  143. if (isset($params['stsRegionId'])) {
  144. $this->stsEndpoint = 'sts' . $params['stsRegionId'] . '.aliyuncs.com';
  145. }
  146. if (isset($params['stsEndpoint'])) {
  147. $this->stsEndpoint = $params['stsEndpoint'];
  148. }
  149. if (is_null($this->stsEndpoint) || $this->stsEndpoint === '') {
  150. $this->stsEndpoint = 'sts.aliyuncs.com';
  151. }
  152. }
  153. private function filterOptions(array $options)
  154. {
  155. if (isset($options['connectTimeout'])) {
  156. $this->connectTimeout = $options['connectTimeout'];
  157. }
  158. if (isset($options['readTimeout'])) {
  159. $this->readTimeout = $options['readTimeout'];
  160. }
  161. Filter::timeout($this->connectTimeout, $this->readTimeout);
  162. }
  163. /**
  164. * Get credentials by request.
  165. *
  166. * @return array
  167. * @throws RuntimeException
  168. * @throws GuzzleException
  169. */
  170. public function refreshCredentials()
  171. {
  172. $options = Request::commonOptions();
  173. $options['read_timeout'] = $this->readTimeout;
  174. $options['connect_timeout'] = $this->connectTimeout;
  175. $options['query']['Action'] = 'AssumeRoleWithOIDC';
  176. $options['query']['Version'] = '2015-04-01';
  177. $options['query']['Format'] = 'JSON';
  178. $options['query']['Timestamp'] = gmdate('Y-m-d\TH:i:s\Z');
  179. $options['query']['RoleArn'] = $this->roleArn;
  180. $options['query']['OIDCProviderArn'] = $this->oidcProviderArn;
  181. try {
  182. $oidcToken = file_get_contents($this->oidcTokenFilePath);
  183. $options['query']['OIDCToken'] = $oidcToken;
  184. } catch (Exception $exception) {
  185. throw new InvalidArgumentException($exception->getMessage());
  186. }
  187. $options['query']['RoleSessionName'] = $this->roleSessionName;
  188. $options['query']['DurationSeconds'] = (string) $this->durationSeconds;
  189. if (!is_null($this->policy)) {
  190. $options['query']['Policy'] = $this->policy;
  191. }
  192. $url = (new Uri())->withScheme('https')->withHost($this->stsEndpoint);
  193. $result = Request::createClient()->request('POST', $url, $options);
  194. if ($result->getStatusCode() !== 200) {
  195. throw new RuntimeException('Error refreshing credentials from OIDC, statusCode: ' . $result->getStatusCode() . ', result: ' . (string) $result);
  196. }
  197. $json = $result->toArray();
  198. $credentials = $json['Credentials'];
  199. if (!isset($credentials['AccessKeyId']) || !isset($credentials['AccessKeySecret']) || !isset($credentials['SecurityToken'])) {
  200. throw new RuntimeException('Error retrieving credentials from OIDC result:' . $result->toJson());
  201. }
  202. return $credentials;
  203. }
  204. public function key()
  205. {
  206. return 'oidc_role_arn#roleArn#' . $this->roleArn . '#oidcProviderArn#' . $this->oidcProviderArn . '#roleSessionName#' . $this->roleSessionName;
  207. }
  208. public function getProviderName()
  209. {
  210. return 'oidc_role_arn';
  211. }
  212. }