AssumeRoleWithWebIdentityCredentialProvider.php 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166
  1. <?php
  2. namespace Aws\Credentials;
  3. use Aws\Exception\AwsException;
  4. use Aws\Exception\CredentialsException;
  5. use Aws\Result;
  6. use Aws\Sts\StsClient;
  7. use GuzzleHttp\Promise;
  8. /**
  9. * Credential provider that provides credentials via assuming a role with a web identity
  10. * More Information, see: https://docs.aws.amazon.com/aws-sdk-php/v3/api/api-sts-2011-06-15.html#assumerolewithwebidentity
  11. */
  12. class AssumeRoleWithWebIdentityCredentialProvider
  13. {
  14. const ERROR_MSG = "Missing required 'AssumeRoleWithWebIdentityCredentialProvider' configuration option: ";
  15. const ENV_RETRIES = 'AWS_METADATA_SERVICE_NUM_ATTEMPTS';
  16. /** @var string */
  17. private $tokenFile;
  18. /** @var string */
  19. private $arn;
  20. /** @var string */
  21. private $session;
  22. /** @var StsClient */
  23. private $client;
  24. /** @var integer */
  25. private $retries;
  26. /** @var integer */
  27. private $authenticationAttempts;
  28. /** @var integer */
  29. private $tokenFileReadAttempts;
  30. /**
  31. * The constructor attempts to load config from environment variables.
  32. * If not set, the following config options are used:
  33. * - WebIdentityTokenFile: full path of token filename
  34. * - RoleArn: arn of role to be assumed
  35. * - SessionName: (optional) set by SDK if not provided
  36. *
  37. * @param array $config Configuration options
  38. * @throws \InvalidArgumentException
  39. */
  40. public function __construct(array $config = [])
  41. {
  42. if (!isset($config['RoleArn'])) {
  43. throw new \InvalidArgumentException(self::ERROR_MSG . "'RoleArn'.");
  44. }
  45. $this->arn = $config['RoleArn'];
  46. if (!isset($config['WebIdentityTokenFile'])) {
  47. throw new \InvalidArgumentException(self::ERROR_MSG . "'WebIdentityTokenFile'.");
  48. }
  49. $this->tokenFile = $config['WebIdentityTokenFile'];
  50. if (!preg_match("/^\w\:|^\/|^\\\/", $this->tokenFile)) {
  51. throw new \InvalidArgumentException("'WebIdentityTokenFile' must be an absolute path.");
  52. }
  53. $this->retries = (int) getenv(self::ENV_RETRIES) ?: (isset($config['retries']) ? $config['retries'] : 3);
  54. $this->authenticationAttempts = 0;
  55. $this->tokenFileReadAttempts = 0;
  56. $this->session = isset($config['SessionName'])
  57. ? $config['SessionName']
  58. : 'aws-sdk-php-' . round(microtime(true) * 1000);
  59. $region = isset($config['region'])
  60. ? $config['region']
  61. : 'us-east-1';
  62. if (isset($config['client'])) {
  63. $this->client = $config['client'];
  64. } else {
  65. $this->client = new StsClient([
  66. 'credentials' => false,
  67. 'region' => $region,
  68. 'version' => 'latest'
  69. ]);
  70. }
  71. }
  72. /**
  73. * Loads assume role with web identity credentials.
  74. *
  75. * @return Promise\PromiseInterface
  76. */
  77. public function __invoke()
  78. {
  79. return Promise\Coroutine::of(function () {
  80. $client = $this->client;
  81. $result = null;
  82. while ($result == null) {
  83. try {
  84. $token = @file_get_contents($this->tokenFile);
  85. if (false === $token) {
  86. clearstatcache(true, dirname($this->tokenFile) . "/" . readlink($this->tokenFile));
  87. clearstatcache(true, dirname($this->tokenFile) . "/" . dirname(readlink($this->tokenFile)));
  88. clearstatcache(true, $this->tokenFile);
  89. if (!@is_readable($this->tokenFile)) {
  90. throw new CredentialsException(
  91. "Unreadable tokenfile at location {$this->tokenFile}"
  92. );
  93. }
  94. $token = @file_get_contents($this->tokenFile);
  95. }
  96. if (empty($token)) {
  97. if ($this->tokenFileReadAttempts < $this->retries) {
  98. sleep((int) pow(1.2, $this->tokenFileReadAttempts));
  99. $this->tokenFileReadAttempts++;
  100. continue;
  101. }
  102. throw new CredentialsException("InvalidIdentityToken from file: {$this->tokenFile}");
  103. }
  104. } catch (\Exception $exception) {
  105. throw new CredentialsException(
  106. "Error reading WebIdentityTokenFile from " . $this->tokenFile,
  107. 0,
  108. $exception
  109. );
  110. }
  111. $assumeParams = [
  112. 'RoleArn' => $this->arn,
  113. 'RoleSessionName' => $this->session,
  114. 'WebIdentityToken' => $token
  115. ];
  116. try {
  117. $result = $client->assumeRoleWithWebIdentity($assumeParams);
  118. } catch (AwsException $e) {
  119. if ($e->getAwsErrorCode() == 'InvalidIdentityToken') {
  120. if ($this->authenticationAttempts < $this->retries) {
  121. sleep((int) pow(1.2, $this->authenticationAttempts));
  122. } else {
  123. throw new CredentialsException(
  124. "InvalidIdentityToken, retries exhausted"
  125. );
  126. }
  127. } else {
  128. throw new CredentialsException(
  129. "Error assuming role from web identity credentials",
  130. 0,
  131. $e
  132. );
  133. }
  134. } catch (\Exception $e) {
  135. throw new CredentialsException(
  136. "Error retrieving web identity credentials: " . $e->getMessage()
  137. . " (" . $e->getCode() . ")"
  138. );
  139. }
  140. $this->authenticationAttempts++;
  141. }
  142. yield $this->client->createCredentials($result);
  143. });
  144. }
  145. }