| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255 | 
							- <?php
 
- namespace Aws\Credentials;
 
- use Aws\Exception\CredentialsException;
 
- use GuzzleHttp\Exception\ConnectException;
 
- use GuzzleHttp\Exception\GuzzleException;
 
- use GuzzleHttp\Psr7\Request;
 
- use GuzzleHttp\Promise;
 
- use GuzzleHttp\Promise\PromiseInterface;
 
- use Psr\Http\Message\ResponseInterface;
 
- /**
 
-  * Credential provider that fetches container credentials with GET request.
 
-  * container environment variables are used in constructing request URI.
 
-  */
 
- class EcsCredentialProvider
 
- {
 
-     const SERVER_URI = 'http://169.254.170.2';
 
-     const ENV_URI = "AWS_CONTAINER_CREDENTIALS_RELATIVE_URI";
 
-     const ENV_FULL_URI = "AWS_CONTAINER_CREDENTIALS_FULL_URI";
 
-     const ENV_AUTH_TOKEN = "AWS_CONTAINER_AUTHORIZATION_TOKEN";
 
-     const ENV_AUTH_TOKEN_FILE = "AWS_CONTAINER_AUTHORIZATION_TOKEN_FILE";
 
-     const ENV_TIMEOUT = 'AWS_METADATA_SERVICE_TIMEOUT';
 
-     const EKS_SERVER_HOST_IPV4 = '169.254.170.23';
 
-     const EKS_SERVER_HOST_IPV6 = 'fd00:ec2::23';
 
-     const ENV_RETRIES = 'AWS_METADATA_SERVICE_NUM_ATTEMPTS';
 
-     const DEFAULT_ENV_TIMEOUT = 1.0;
 
-     const DEFAULT_ENV_RETRIES = 3;
 
-     /** @var callable */
 
-     private $client;
 
-     /** @var float|mixed */
 
-     private $timeout;
 
-     /** @var int */
 
-     private $retries;
 
-     /** @var int */
 
-     private $attempts;
 
-     /**
 
-      *  The constructor accepts following options:
 
-      *  - timeout: (optional) Connection timeout, in seconds, default 1.0
 
-      *  - retries: Optional number of retries to be attempted, default 3.
 
-      *  - client: An EcsClient to make request from
 
-      *
 
-      * @param array $config Configuration options
 
-      */
 
-     public function __construct(array $config = [])
 
-     {
 
-         $this->timeout = (float) isset($config['timeout'])
 
-             ? $config['timeout']
 
-             : (getenv(self::ENV_TIMEOUT) ?: self::DEFAULT_ENV_TIMEOUT);
 
-         $this->retries = (int) isset($config['retries'])
 
-             ? $config['retries']
 
-             : ((int) getenv(self::ENV_RETRIES) ?: self::DEFAULT_ENV_RETRIES);
 
-         $this->client = $config['client'] ?? \Aws\default_http_handler();
 
-     }
 
-     /**
 
-      * Load container credentials.
 
-      *
 
-      * @return PromiseInterface
 
-      * @throws GuzzleException
 
-      */
 
-     public function __invoke()
 
-     {
 
-         $this->attempts = 0;
 
-         $uri = $this->getEcsUri();
 
-         if ($this->isCompatibleUri($uri)) {
 
-             return Promise\Coroutine::of(function () {
 
-                 $client = $this->client;
 
-                 $request = new Request('GET', $this->getEcsUri());
 
-                 $headers = $this->getHeadersForAuthToken();
 
-                 $credentials = null;
 
-                 while ($credentials === null) {
 
-                     $credentials = (yield $client(
 
-                         $request,
 
-                         [
 
-                             'timeout' => $this->timeout,
 
-                             'proxy' => '',
 
-                             'headers' => $headers,
 
-                         ]
 
-                     )->then(function (ResponseInterface $response) {
 
-                         $result = $this->decodeResult((string)$response->getBody());
 
-                         return new Credentials(
 
-                             $result['AccessKeyId'],
 
-                             $result['SecretAccessKey'],
 
-                             $result['Token'],
 
-                             strtotime($result['Expiration'])
 
-                         );
 
-                     })->otherwise(function ($reason) {
 
-                         $reason = is_array($reason) ? $reason['exception'] : $reason;
 
-                         $isRetryable = $reason instanceof ConnectException;
 
-                         if ($isRetryable && ($this->attempts < $this->retries)) {
 
-                             sleep((int)pow(1.2, $this->attempts));
 
-                         } else {
 
-                             $msg = $reason->getMessage();
 
-                             throw new CredentialsException(
 
-                                 sprintf('Error retrieving credentials from container metadata after attempt %d/%d (%s)', $this->attempts, $this->retries, $msg)
 
-                             );
 
-                         }
 
-                     }));
 
-                     $this->attempts++;
 
-                 }
 
-                 yield $credentials;
 
-             });
 
-         }
 
-         throw new CredentialsException("Uri '{$uri}' contains an unsupported host.");
 
-     }
 
-     /**
 
-      * Returns the number of attempts that have been done.
 
-      *
 
-      * @return int
 
-      */
 
-     public function getAttempts(): int
 
-     {
 
-         return $this->attempts;
 
-     }
 
-     /**
 
-      * Retrieves authorization token.
 
-      *
 
-      * @return array|false|string
 
-      */
 
-     private function getEcsAuthToken()
 
-     {
 
-         if (!empty($path = getenv(self::ENV_AUTH_TOKEN_FILE))) {
 
-             $token =  @file_get_contents($path);
 
-             if (false === $token) {
 
-                 clearstatcache(true, dirname($path) . DIRECTORY_SEPARATOR . @readlink($path));
 
-                 clearstatcache(true, dirname($path) . DIRECTORY_SEPARATOR . dirname(@readlink($path)));
 
-                 clearstatcache(true, $path);
 
-             }
 
-             if (!is_readable($path)) {
 
-                 throw new CredentialsException("Failed to read authorization token from '{$path}': no such file or directory.");
 
-             }
 
-             $token = @file_get_contents($path);
 
-             if (empty($token)) {
 
-                 throw new CredentialsException("Invalid authorization token read from `$path`. Token file is empty!");
 
-             }
 
-             return $token;
 
-         }
 
-         return getenv(self::ENV_AUTH_TOKEN);
 
-     }
 
-     /**
 
-      * Provides headers for credential metadata request.
 
-      *
 
-      * @return array|array[]|string[]
 
-      */
 
-     private function getHeadersForAuthToken()
 
-     {
 
-         $authToken = self::getEcsAuthToken();
 
-         $headers = [];
 
-         if (!empty($authToken))
 
-             $headers = ['Authorization' => $authToken];
 
-         return $headers;
 
-     }
 
-     /** @deprecated */
 
-     public function setHeaderForAuthToken()
 
-     {
 
-         $authToken = self::getEcsAuthToken();
 
-         $headers = [];
 
-         if (!empty($authToken))
 
-             $headers = ['Authorization' => $authToken];
 
-         return $headers;
 
-     }
 
-     /**
 
-      * Fetch container metadata URI from container environment variable.
 
-      *
 
-      * @return string Returns container metadata URI
 
-      */
 
-     private function getEcsUri()
 
-     {
 
-         $credsUri = getenv(self::ENV_URI);
 
-         if ($credsUri === false) {
 
-             $credsUri = $_SERVER[self::ENV_URI] ?? '';
 
-         }
 
-         if (empty($credsUri)){
 
-             $credFullUri = getenv(self::ENV_FULL_URI);
 
-             if ($credFullUri === false){
 
-                 $credFullUri = $_SERVER[self::ENV_FULL_URI] ?? '';
 
-             }
 
-             if (!empty($credFullUri))
 
-                 return $credFullUri;
 
-         }
 
-         return self::SERVER_URI . $credsUri;
 
-     }
 
-     private function decodeResult($response)
 
-     {
 
-         $result = json_decode($response, true);
 
-         if (!isset($result['AccessKeyId'])) {
 
-             throw new CredentialsException('Unexpected container metadata credentials value');
 
-         }
 
-         return $result;
 
-     }
 
-     /**
 
-      * Determines whether or not a given request URI is a valid
 
-      * container credential request URI.
 
-      *
 
-      * @param $uri
 
-      *
 
-      * @return bool
 
-      */
 
-     private function isCompatibleUri($uri)
 
-     {
 
-         $parsed = parse_url($uri);
 
-         if ($parsed['scheme'] !== 'https') {
 
-             $host = trim($parsed['host'], '[]');
 
-             $ecsHost = parse_url(self::SERVER_URI)['host'];
 
-             $eksHost = self::EKS_SERVER_HOST_IPV4;
 
-             if ($host !== $ecsHost
 
-                 && $host !== $eksHost
 
-                 && $host !== self::EKS_SERVER_HOST_IPV6
 
-                 && !CredentialsUtils::isLoopBackAddress(gethostbyname($host))
 
-             ) {
 
-                 return false;
 
-             }
 
-         }
 
-         return true;
 
-     }
 
- }
 
 
  |