true, 'DeleteBucket' => true, 'ListBuckets' => true]; const NO_PATTERN = 0; const DUALSTACK = 1; const ACCELERATE = 2; const ACCELERATE_DUALSTACK = 3; const PATH_STYLE = 4; const HOST_STYLE = 5; /** @var bool */ private $accelerateByDefault; /** @var bool */ private $dualStackByDefault; /** @var bool */ private $pathStyleByDefault; /** @var string */ private $region; /** @var callable */ private $endpointProvider; /** @var callable */ private $nextHandler; /** @var string */ private $endpoint; /** * Create a middleware wrapper function * * @param string $region * @param EndpointProvider $endpointProvider * @param array $options * * @return callable */ public static function wrap($region, $endpointProvider, array $options) { return function (callable $handler) use($region, $endpointProvider, $options) { return new self($handler, $region, $options, $endpointProvider); }; } public function __construct(callable $nextHandler, $region, array $options, $endpointProvider = null) { $this->pathStyleByDefault = isset($options['path_style']) ? (bool) $options['path_style'] : false; $this->dualStackByDefault = isset($options['dual_stack']) ? (bool) $options['dual_stack'] : false; $this->accelerateByDefault = isset($options['accelerate']) ? (bool) $options['accelerate'] : false; $this->region = (string) $region; $this->endpoint = isset($options['endpoint']) ? $options['endpoint'] : ""; $this->endpointProvider = is_null($endpointProvider) ? \UglyRobot\Infinite_Uploads\Aws\Endpoint\PartitionEndpointProvider::defaultProvider() : $endpointProvider; $this->nextHandler = $nextHandler; } public function __invoke(\UglyRobot\Infinite_Uploads\Aws\CommandInterface $command, \UglyRobot\Infinite_Uploads\Psr\Http\Message\RequestInterface $request) { if (!empty($this->endpoint)) { $request = $this->applyEndpoint($command, $request); } else { switch ($this->endpointPatternDecider($command, $request)) { case self::HOST_STYLE: $request = $this->applyHostStyleEndpoint($command, $request); break; case self::NO_PATTERN: case self::PATH_STYLE: break; case self::DUALSTACK: $request = $this->applyDualStackEndpoint($command, $request); break; case self::ACCELERATE: $request = $this->applyAccelerateEndpoint($command, $request, 's3-accelerate'); break; case self::ACCELERATE_DUALSTACK: $request = $this->applyAccelerateEndpoint($command, $request, 's3-accelerate.dualstack'); break; } } $nextHandler = $this->nextHandler; return $nextHandler($command, $request); } private static function isRequestHostStyleCompatible(\UglyRobot\Infinite_Uploads\Aws\CommandInterface $command, \UglyRobot\Infinite_Uploads\Psr\Http\Message\RequestInterface $request) { return \UglyRobot\Infinite_Uploads\Aws\S3\S3Client::isBucketDnsCompatible($command['Bucket']) && ($request->getUri()->getScheme() === 'http' || strpos($command['Bucket'], '.') === false) && filter_var($request->getUri()->getHost(), FILTER_VALIDATE_IP) === false; } private function endpointPatternDecider(\UglyRobot\Infinite_Uploads\Aws\CommandInterface $command, \UglyRobot\Infinite_Uploads\Psr\Http\Message\RequestInterface $request) { $accelerate = isset($command['@use_accelerate_endpoint']) ? $command['@use_accelerate_endpoint'] : $this->accelerateByDefault; $dualStack = isset($command['@use_dual_stack_endpoint']) ? $command['@use_dual_stack_endpoint'] : $this->dualStackByDefault; $pathStyle = isset($command['@use_path_style_endpoint']) ? $command['@use_path_style_endpoint'] : $this->pathStyleByDefault; if ($accelerate && $dualStack) { // When try to enable both for operations excluded from s3-accelerate, // only dualstack endpoints will be enabled. return $this->canAccelerate($command) ? self::ACCELERATE_DUALSTACK : self::DUALSTACK; } if ($accelerate && $this->canAccelerate($command)) { return self::ACCELERATE; } if ($dualStack) { return self::DUALSTACK; } if (!$pathStyle && self::isRequestHostStyleCompatible($command, $request)) { return self::HOST_STYLE; } return self::PATH_STYLE; } private function canAccelerate(\UglyRobot\Infinite_Uploads\Aws\CommandInterface $command) { return empty(self::$exclusions[$command->getName()]) && \UglyRobot\Infinite_Uploads\Aws\S3\S3Client::isBucketDnsCompatible($command['Bucket']); } private function getBucketStyleHost(\UglyRobot\Infinite_Uploads\Aws\CommandInterface $command, $host) { // For operations on the base host (e.g. ListBuckets) if (!isset($command['Bucket'])) { return $host; } return "{$command['Bucket']}.{$host}"; } private function applyHostStyleEndpoint(\UglyRobot\Infinite_Uploads\Aws\CommandInterface $command, \UglyRobot\Infinite_Uploads\Psr\Http\Message\RequestInterface $request) { $uri = $request->getUri(); $request = $request->withUri($uri->withHost($this->getBucketStyleHost($command, $uri->getHost()))->withPath($this->getBucketlessPath($uri->getPath(), $command))); return $request; } private function applyDualStackEndpoint(\UglyRobot\Infinite_Uploads\Aws\CommandInterface $command, \UglyRobot\Infinite_Uploads\Psr\Http\Message\RequestInterface $request) { $request = $request->withUri($request->getUri()->withHost($this->getDualStackHost())); if (empty($command['@use_path_style_endpoint']) && !$this->pathStyleByDefault && self::isRequestHostStyleCompatible($command, $request)) { $request = $this->applyHostStyleEndpoint($command, $request); } return $request; } private function getDualStackHost() { $dnsSuffix = $this->endpointProvider->getPartition($this->region, 's3')->getDnsSuffix(); return "s3.dualstack.{$this->region}.{$dnsSuffix}"; } private function applyAccelerateEndpoint(\UglyRobot\Infinite_Uploads\Aws\CommandInterface $command, \UglyRobot\Infinite_Uploads\Psr\Http\Message\RequestInterface $request, $pattern) { $request = $request->withUri($request->getUri()->withHost($this->getAccelerateHost($command, $pattern))->withPath($this->getBucketlessPath($request->getUri()->getPath(), $command))); return $request; } private function getAccelerateHost(\UglyRobot\Infinite_Uploads\Aws\CommandInterface $command, $pattern) { $dnsSuffix = $this->endpointProvider->getPartition($this->region, 's3')->getDnsSuffix(); return "{$command['Bucket']}.{$pattern}.{$dnsSuffix}"; } private function getBucketlessPath($path, \UglyRobot\Infinite_Uploads\Aws\CommandInterface $command) { $pattern = '/^\\/' . preg_quote($command['Bucket'], '/') . '/'; return preg_replace($pattern, '', $path) ?: '/'; } private function applyEndpoint(\UglyRobot\Infinite_Uploads\Aws\CommandInterface $command, \UglyRobot\Infinite_Uploads\Psr\Http\Message\RequestInterface $request) { $dualStack = isset($command['@use_dual_stack_endpoint']) ? $command['@use_dual_stack_endpoint'] : $this->dualStackByDefault; if (\UglyRobot\Infinite_Uploads\Aws\Arn\ArnParser::isArn($command['Bucket'])) { $arn = \UglyRobot\Infinite_Uploads\Aws\Arn\ArnParser::parse($command['Bucket']); $outpost = $arn->getService() == 's3-outposts'; if ($outpost && $dualStack) { throw new \UglyRobot\Infinite_Uploads\GuzzleHttp\Exception\InvalidArgumentException("Outposts + dualstack is not supported"); } } if ($dualStack) { throw new \UglyRobot\Infinite_Uploads\GuzzleHttp\Exception\InvalidArgumentException("Custom Endpoint + Dualstack not supported"); } $host = $this->pathStyleByDefault ? $this->endpoint : $this->getBucketStyleHost($command, $this->endpoint); $uri = new \UglyRobot\Infinite_Uploads\GuzzleHttp\Psr7\Uri($host); $scheme = $uri->getScheme(); if (empty($scheme)) { //The host contains s3-outposts here, that's where it gets added $request = $request->withUri($uri->withHost($host)); } else { $request = $request->withUri($uri); } return $request; } }