Вход Регистрация
Файл: includes/library/aws/GuzzleHttp/RedirectMiddleware.php
Строк: 194
<?php
namespace GuzzleHttp;

use 
GuzzleHttpExceptionBadResponseException;
use 
GuzzleHttpExceptionTooManyRedirectsException;
use 
GuzzleHttpPromisePromiseInterface;
use 
GuzzleHttpPsr7;
use 
PsrHttpMessageRequestInterface;
use 
PsrHttpMessageResponseInterface;
use 
PsrHttpMessageUriInterface;

/**
 * Request redirect middleware.
 *
 * Apply this middleware like other middleware using
 * {@see GuzzleHttpMiddleware::redirect()}.
 */
class RedirectMiddleware
{
    public static 
$defaultSettings = [
        
'max'       => 5,
        
'protocols' => ['http''https'],
        
'strict'    => false,
        
'referer'   => false
    
];

    
/** @var callable  */
    
private $nextHandler;

    
/**
     * @param callable $nextHandler Next handler to invoke.
     */
    
public function __construct(callable $nextHandler)
    {
        
$this->nextHandler $nextHandler;
    }

    
/**
     * @param RequestInterface $request
     * @param array            $options
     *
     * @return PromiseInterface
     */
    
public function __invoke(RequestInterface $request, array $options)
    {
        
$fn $this->nextHandler;

        if (empty(
$options['allow_redirects'])) {
            return 
$fn($request$options);
        }

        if (
$options['allow_redirects'] === true) {
            
$options['allow_redirects'] = self::$defaultSettings;
        } elseif (!
is_array($options['allow_redirects'])) {
            throw new 
InvalidArgumentException('allow_redirects must be true, false, or array');
        } else {
            
// Merge the default settings with the provided settings
            
$options['allow_redirects'] += self::$defaultSettings;
        }

        if (empty(
$options['allow_redirects']['max'])) {
            return 
$fn($request$options);
        }

        return 
$fn($request$options)
            ->
then(function (ResponseInterface $response) use ($request$options) {
                return 
$this->checkRedirect($request$options$response);
            });
    }

    
/**
     * @param RequestInterface  $request
     * @param array             $options
     * @param ResponseInterface|PromiseInterface $response
     *
     * @return ResponseInterface|PromiseInterface
     */
    
public function checkRedirect(
        
RequestInterface $request,
        array 
$options,
        
ResponseInterface $response
    
) {
        if (
substr($response->getStatusCode(), 01) != '3'
            
|| !$response->hasHeader('Location')
        ) {
            return 
$response;
        }

        
$this->guardMax($request$options);
        
$nextRequest $this->modifyRequest($request$options$response);

        return 
$this($nextRequest$options);
    }

    private function 
guardMax(RequestInterface $request, array &$options)
    {
        
$current = isset($options['__redirect_count'])
            ? 
$options['__redirect_count']
            : 
0;
        
$options['__redirect_count'] = $current 1;
        
$max $options['allow_redirects']['max'];

        if (
$options['__redirect_count'] > $max) {
            throw new 
TooManyRedirectsException(
                
"Will not follow more than {$max} redirects",
                
$request
            
);
        }
    }

    
/**
     * @param RequestInterface  $request
     * @param array             $options
     * @param ResponseInterface $response
     *
     * @return RequestInterface
     */
    
public function modifyRequest(
        
RequestInterface $request,
        array 
$options,
        
ResponseInterface $response
    
) {
        
// Request modifications to apply.
        
$modify = [];
        
$protocols $options['allow_redirects']['protocols'];

        
// Use a GET request if this is an entity enclosing request and we are
        // not forcing RFC compliance, but rather emulating what all browsers
        // would do.
        
$statusCode $response->getStatusCode();
        if (
$statusCode == 303 ||
            (
$statusCode <= 302 && $request->getBody() && !$options['allow_redirects']['strict'])
        ) {
            
$modify['method'] = 'GET';
            
$modify['body'] = '';
        }

        
$modify['uri'] = $this->redirectUri($request$response$protocols);
        
Psr7rewind_body($request);

        
// Add the Referer header if it is told to do so and only
        // add the header if we are not redirecting from https to http.
        
if ($options['allow_redirects']['referer']
            && 
$modify['uri']->getScheme() === $request->getUri()->getScheme()
        ) {
            
$uri $request->getUri()->withUserInfo('''');
            
$modify['set_headers']['Referer'] = (string) $uri;
        } else {
            
$modify['remove_headers'][] = 'Referer';
        }

        return 
Psr7modify_request($request$modify);
    }

    
/**
     * Set the appropriate URL on the request based on the location header
     *
     * @param RequestInterface  $request
     * @param ResponseInterface $response
     * @param array             $protocols
     *
     * @return UriInterface
     */
    
private function redirectUri(
        
RequestInterface $request,
        
ResponseInterface $response,
        array 
$protocols
    
) {
        
$location Psr7Uri::resolve(
            
$request->getUri(),
            
$response->getHeaderLine('Location')
        );

        
// Ensure that the redirect URI is allowed based on the protocols.
        
if (!in_array($location->getScheme(), $protocols)) {
            throw new 
BadResponseException(
                
sprintf(
                    
'Redirect URI, %s, does not use one of the allowed redirect protocols: %s',
                    
$location,
                    
implode(', '$protocols)
                ),
                
$request,
                
$response
            
);
        }

        return 
$location;
    }
}
Онлайн: 1
Реклама