| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169 | <?phpnamespace Aws\Auth;use Aws\Auth\Exception\UnresolvedAuthSchemeException;use Aws\Identity\AwsCredentialIdentity;use Aws\Identity\BearerTokenIdentity;use GuzzleHttp\Promise\PromiseInterface;/** * Houses logic for selecting an auth scheme modeled in a service's `auth` trait. * The `auth` trait can be modeled either in a service's metadata, or at the operation level. */class AuthSchemeResolver implements AuthSchemeResolverInterface{    const UNSIGNED_BODY = '-unsigned-body';    /**     * @var string[] Default mapping of modeled auth trait auth schemes     *               to the SDK's supported signature versions.     */    private static $defaultAuthSchemeMap = [        'aws.auth#sigv4' => 'v4',        'aws.auth#sigv4a' => 'v4a',        'smithy.api#httpBearerAuth' => 'bearer',        'smithy.api#noAuth' => 'anonymous'    ];    /**     * @var array Mapping of auth schemes to signature versions used in     *            resolving a signature version.     */    private $authSchemeMap;    private $tokenProvider;    private $credentialProvider;    public function __construct(        callable $credentialProvider,        callable $tokenProvider = null,        array $authSchemeMap = []    ){        $this->credentialProvider = $credentialProvider;        $this->tokenProvider = $tokenProvider;        $this->authSchemeMap = empty($authSchemeMap)            ? self::$defaultAuthSchemeMap            : $authSchemeMap;    }    /**     * Accepts a priority-ordered list of auth schemes and an Identity     * and selects the first compatible auth schemes, returning a normalized     * signature version.  For example, based on the default auth scheme mapping,     * if `aws.auth#sigv4` is selected, `v4` will be returned.     *     * @param array $authSchemes     * @param $identity     *     * @return string     * @throws UnresolvedAuthSchemeException     */    public function selectAuthScheme(        array $authSchemes,        array $args = []    ): string    {        $failureReasons = [];        foreach($authSchemes as $authScheme) {            $normalizedAuthScheme = $this->authSchemeMap[$authScheme] ?? $authScheme;            if ($this->isCompatibleAuthScheme($normalizedAuthScheme)) {                if ($normalizedAuthScheme === 'v4' && !empty($args['unsigned_payload'])) {                    return $normalizedAuthScheme . self::UNSIGNED_BODY;                }                return $normalizedAuthScheme;            } else {                $failureReasons[] = $this->getIncompatibilityMessage($normalizedAuthScheme);            }        }        throw new UnresolvedAuthSchemeException(            'Could not resolve an authentication scheme: '            . implode('; ', $failureReasons)        );    }    /**     * Determines compatibility based on either Identity or the availability     * of the CRT extension.     *     * @param $authScheme     *     * @return bool     */    private function isCompatibleAuthScheme($authScheme): bool    {        switch ($authScheme) {            case 'v4':            case 'anonymous':                return $this->hasAwsCredentialIdentity();            case 'v4a':                return extension_loaded('awscrt') && $this->hasAwsCredentialIdentity();            case 'bearer':                return $this->hasBearerTokenIdentity();            default:                return false;        }    }    /**     * Provides incompatibility messages in the event an incompatible auth scheme     * is encountered.     *     * @param $authScheme     *     * @return string     */    private function getIncompatibilityMessage($authScheme): string    {        switch ($authScheme) {            case 'v4':                return 'Signature V4 requires AWS credentials for request signing';            case 'anonymous':                return 'Anonymous signatures require AWS credentials for request signing';            case 'v4a':                return 'The aws-crt-php extension and AWS credentials are required to use Signature V4A';            case 'bearer':                return 'Bearer token credentials must be provided to use Bearer authentication';            default:                return "The service does not support `{$authScheme}` authentication.";        }    }    /**     * @return bool     */    private function hasAwsCredentialIdentity(): bool    {        $fn = $this->credentialProvider;        $result = $fn();        if ($result instanceof PromiseInterface) {            return $result->wait() instanceof AwsCredentialIdentity;        }        return $result instanceof AwsCredentialIdentity;    }    /**     * @return bool     */    private function hasBearerTokenIdentity(): bool    {        if ($this->tokenProvider) {            $fn = $this->tokenProvider;            $result = $fn();            if ($result instanceof PromiseInterface) {                return $result->wait() instanceof BearerTokenIdentity;            }            return $result instanceof BearerTokenIdentity;        }        return false;    }}
 |