| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119 | 
							- <?php
 
- namespace GuzzleHttp;
 
- use GuzzleHttp\Promise as P;
 
- use GuzzleHttp\Promise\PromiseInterface;
 
- use Psr\Http\Message\RequestInterface;
 
- use Psr\Http\Message\ResponseInterface;
 
- /**
 
-  * Middleware that retries requests based on the boolean result of
 
-  * invoking the provided "decider" function.
 
-  *
 
-  * @final
 
-  */
 
- class RetryMiddleware
 
- {
 
-     /**
 
-      * @var callable(RequestInterface, array): PromiseInterface
 
-      */
 
-     private $nextHandler;
 
-     /**
 
-      * @var callable
 
-      */
 
-     private $decider;
 
-     /**
 
-      * @var callable(int)
 
-      */
 
-     private $delay;
 
-     /**
 
-      * @param callable                                            $decider     Function that accepts the number of retries,
 
-      *                                                                         a request, [response], and [exception] and
 
-      *                                                                         returns true if the request is to be
 
-      *                                                                         retried.
 
-      * @param callable(RequestInterface, array): PromiseInterface $nextHandler Next handler to invoke.
 
-      * @param (callable(int): int)|null                           $delay       Function that accepts the number of retries
 
-      *                                                                         and returns the number of
 
-      *                                                                         milliseconds to delay.
 
-      */
 
-     public function __construct(callable $decider, callable $nextHandler, callable $delay = null)
 
-     {
 
-         $this->decider = $decider;
 
-         $this->nextHandler = $nextHandler;
 
-         $this->delay = $delay ?: __CLASS__.'::exponentialDelay';
 
-     }
 
-     /**
 
-      * Default exponential backoff delay function.
 
-      *
 
-      * @return int milliseconds.
 
-      */
 
-     public static function exponentialDelay(int $retries): int
 
-     {
 
-         return (int) 2 ** ($retries - 1) * 1000;
 
-     }
 
-     public function __invoke(RequestInterface $request, array $options): PromiseInterface
 
-     {
 
-         if (!isset($options['retries'])) {
 
-             $options['retries'] = 0;
 
-         }
 
-         $fn = $this->nextHandler;
 
-         return $fn($request, $options)
 
-             ->then(
 
-                 $this->onFulfilled($request, $options),
 
-                 $this->onRejected($request, $options)
 
-             );
 
-     }
 
-     /**
 
-      * Execute fulfilled closure
 
-      */
 
-     private function onFulfilled(RequestInterface $request, array $options): callable
 
-     {
 
-         return function ($value) use ($request, $options) {
 
-             if (!($this->decider)(
 
-                 $options['retries'],
 
-                 $request,
 
-                 $value,
 
-                 null
 
-             )) {
 
-                 return $value;
 
-             }
 
-             return $this->doRetry($request, $options, $value);
 
-         };
 
-     }
 
-     /**
 
-      * Execute rejected closure
 
-      */
 
-     private function onRejected(RequestInterface $req, array $options): callable
 
-     {
 
-         return function ($reason) use ($req, $options) {
 
-             if (!($this->decider)(
 
-                 $options['retries'],
 
-                 $req,
 
-                 null,
 
-                 $reason
 
-             )) {
 
-                 return P\Create::rejectionFor($reason);
 
-             }
 
-             return $this->doRetry($req, $options);
 
-         };
 
-     }
 
-     private function doRetry(RequestInterface $request, array $options, ResponseInterface $response = null): PromiseInterface
 
-     {
 
-         $options['delay'] = ($this->delay)(++$options['retries'], $response, $request);
 
-         return $this($request, $options);
 
-     }
 
- }
 
 
  |