zengapay / guzzle_retry_middleware
Guzzle v6+ retry middleware that handles status codes other than 200, 201, 202 and connection timeouts
Requires
- php: ^7.1|^8.0
- guzzlehttp/guzzle: ^6.3|^7.0
Requires (Dev)
- jaschilz/php-coverage-badger: ^2.0
- nesbot/carbon: ^2.0
- phpstan/extension-installer: ^1.0
- phpstan/phpstan: ^0.12
- phpunit/phpunit: ^7.5|^8.0|^9.0
- squizlabs/php_codesniffer: ^3.5
This package is auto-updated.
Last update: 2025-03-26 19:04:02 UTC
README
This is a Guzzle v6+ middleware library that implements automatic
retry of requests when HTTP servers respond with status codes other than 200
, 201
, 202
. It can also
be configured to retry requests that timeout.
If a server supplies a Retry-After header, this middleware will delay subsequent requests per the server's instructed wait period.
Unlike the built-in RetryAfter
middleware, this middleware provides some default behavior for negotiating retries
based on rules in the HTTP Spec. You can drop it right into your request stack without any additional configuration.
Features, at-a-glance:
- Automatically retries HTTP requests when a server responds with a status other than
200
,201
,202
(or any HTTP status code; this is configurable) - Sets a retry delay based on the
Retry-After
HTTP header, if it is sent, or automatically backs off exponentially if noRetry-After
header is sent (also configurable) - Optionally retries requests that time out (via the
connect_timeout
ortimeout
options) - Set an optional callback when a retry occurs (useful for logging/reporting)
- Specify a maximum number of retry attempts before giving up (default: 10)
- Near-100% test coverage, good inline documentation, and PSR-12 compliant
Install
Via Composer
$ composer require zengapay/guzzle_retry_middleware
Usage
Basically:
use GuzzleHttp\Client; use GuzzleHttp\HandlerStack; use GuzzleRetry\GuzzleRetryMiddleware; $stack = HandlerStack::create(); $stack->push(GuzzleRetryMiddleware::factory()); $client = new Client(['handler' => $stack]); // Make requests galore...
This is the default configuration. If a HTTP server responds with a status other than 200
, 201
, 202
, this
middleware with intercept the response and retry it up to 10 times before giving up and doing whatever
behavior Guzzle would do by default (by default, throwing a BadResponseException
).
If the server provides a RetryAfter
header, this middleware will wait the specified time before
attempting a request again. If not, then it will back off, waiting longer each time between requests
until giving up after 10 attempts.
Options
The following options are available:
Option | Type | Default | Summary |
---|---|---|---|
retry_enabled |
boolean | true | Is retry enabled (useful for disabling for individual requests) |
max_retry_attempts |
integer | 10 | Maximum number of retries per request |
max_allowable_timeout_secs |
integer | null | If set, specifies a hard ceiling in seconds that the client can wait between requests |
retry_only_if_retry_after_header |
boolean | false | Retry only if RetryAfter header sent |
retry_on_status_other_than |
array | 200, 201, 202 | Retry if server responds with a status other than these |
default_retry_multiplier |
float or callable | 1.5 | Value to multiply the number of requests by if RetryAfter not supplied (see below for details) |
on_retry_callback |
callable | null | Optional callback to call when a retry occurs |
retry_on_timeout |
boolean | false | Set to TRUE if you wish to retry requests that throw a ConnectException such as a timeout or 'connection refused' |
expose_retry_header |
boolean | false | Set to TRUE if you wish to expose the number of retries as a header on the response object |
retry_header |
string | X-Retry-Counter | The header key to use for the retry counter (if you need it) |
retry_after_header |
string | Retry-After | The header key to use for the retry after header. |
retry_after_date_format |
string | D, d M Y H:i:s T |
Optional customization for servers that return date/times that violate the HTTP spec |
Each option is discussed in detail below.
Configuring Options
Options can be set in one of three places:
// Per request, in the same array as other Guzzle options $response = $client->get('/some-url', [ 'max_retry_attempts' => 5, 'on_retry_callback' => $notifier ]); // When you instantiate Guzzle, in the same array as other Guzzle options $client = new \GuzzleHttp\Client([ // Standard Guzzle options 'base_url' => 'http://example.org', 'connect_timeout' => 10.0, // Retry options 'max_retry_attempts' => 5, 'on_retry_callback' => $notifier ]); // When you instantiate the Retry middleware $stack = \GuzzleHttp\Stack::create(); $stack->push(GuzzleRetryMiddleware::factory([ 'max_retry_attempts' => 5, 'on_retry_callback' => $notifier ]));
If you specify options in two or more places, the configuration is merged as follows:
- Request options take precedence over Guzzle constructor options
- Guzzle constructor options take precedence over middleware constructor options.
Setting maximum retry attempts
This value should be an integer equal to or greater than 0. Setting 0 or a negative effectively disables this middleware.
Setting this value to 0 is useful when you want to retry attempts by default, but disable retries for a particular request:
// Set the default retry attempts to 5 $client = new \GuzzleHttp\Client(['max_retry_attempts' => 5]); // Do not retry this request $client->get('/some/url', ['max_retry_attempts' => 0]);
Setting status codes NOT to retry
By default, this middleware will retry requests when the server does not respond with a 200
, 201
or 202
HTTP status code. But, you can configure this:
$response = $client->get('/some-path', [ 'retry_on_status_other_than' => [200, 201, 202] ]);
If the response includes a RetryAfter
header, but its status code is not in the list, it will not be processed.
Setting default retry delay
If the response includes a valid RetryAfter
header, this middleware will delay the next retry attempt the amount of
time that the server specifies in that header.
If the response includes a non-valid RetryAfter
or does not provide a RetryAfter
header, then this middleware
will use a default back-off algorithm: multipler * number-of-attempts
:
Response with RetryAfter
header:
Client Server
------ ------
GET /resource ->
<- 429 Response with `Retry-After: 120`
wait 120s
GET /resource ->
<- 200 OK
Without RetryAfter
, the number of requests is multiplied by the multiplier (default: 1.5
):
Client Server
------ ------
GET /resource ->
<- 429 Response (no Retry-After header)
wait 1.5 x 1s
GET /resource ->
<- 429 Response (no Retry-After header)
wait 1.5 x 2s
GET /resource ->
<- 429 Response (no Retry-After header)
wait 1.5 x 3s
GET /resource ->
<- 429 Response (no Retry-After header)
wait 1.5 x 4s
GET /resource ->
<- 200 OK
You can set a custom default multiplier:
$response = $client->get('/some-path', [ 'default_retry_multiplier' => 2.5 ]);
You can also pass in a custom algorithm for setting the default delay timeout if you specify a callable
for default_retry_multiplier
:
// Custom callback to determine default timeout. Note: $response may be NULL if a connect timeout occurred. $response = $client->get('/some-path', [ 'default_retry_multiplier' => function($numRequests, ?ResponseInterface $response): float { return (float) rand(1, 5); } ]);
Retrying requests that timeout
You can configure this middleware to retry requests that timeout. Simply set the retry_on_timeout
option to true
:
# Retry this request if it times out: $response = $client->get('/some-path', [ 'retry_on_timeout' => true, // Set the retry middleware to retry when the connection or response times out 'connect_timeout' => 20, // This is a built-in Guzzle option 'timeout' => 50 // This is also a built-in Guzzle option ]); # You can also set these as defaults for every request: $guzzle = new \GuzzleHttp\Client(['retry_on_timeout' => true, 'connect_timeout' => 20]); $response = $guzzle->get('https://example.org');
On-Retry callback
You can supply a callback method that will be called each time a request is retried. This is useful for logging, reporting, or anything else you can think of.
If you specify a callback, it will be called before the middleware calls the usleep()
delay function.
The request
and options
arguments are sent by reference in case you want to modify them in the callback.
use Psr\Http\Message\RequestInterface; use Psr\Http\Message\ResponseInterface; /** * Listen for retry events * * @param int $attemptNumber How many attempts have been tried for this particular request * @param float $delay How long the client will wait before retrying the request * @param RequestInterface $request Request * @param array $options Guzzle request options * @param ResponseInterface|null $response Response (or NULL if response not sent; e.g. connect timeout) */ $listener = function($attemptNumber, $delay, &$request, &$options, $response) { echo sprintf( "Retrying request to %s. Server responded with %s. Will wait %s seconds. This is attempt #%s", $request->getUri()->getPath(), $response->getStatusCode(), number_format($delay, 2), $attemptNumber ); } $client = new \GuzzleHttp\Client([ 'on_retry_callback' => $listener ]); $response = $client->get('/some/path');
Enabling or disabling per-request
Suppose that you have setup default retry options as follows:
$stack = \GuzzleHttp\Stack::create(); $stack->push(GuzzleRetryMiddleware::factory(['max_retry_attempts' => 5])); $client = new \GuzzleHttp\Client(['handler' => $stack]);
You can disable retry for individual requests as by setting the retry_enabled
parameter in the request options:
// Retry will NOT be attempted for this request.. $client->get('http://example.org', ['retry_enabled' => false]); // Retry WILL be attempted for this request... $client->get('http://example.org');
Adding a custom retry header to HTTP responses
Sometimes for debugging purposes, it is useful to know how many times a request was retried when getting a response.
For this purpose, this library can add a custom header to responses; simply set the expose_retry_header
option
to TRUE
.
Note: This modifies the HTTP response on the client. If you don't want to alter the response retrieved from the server, you can also use callbacks to get the request count.
Example:
# Retry this request if it times out: $response = $client->get('/some-path', [ 'expose_retry_header' => true // This adds the 'X-Retry-Counter' if a request was retried ]); # If a request was retried, the response will include the 'X-Retry-Counter' $numRetries = (int) $response->getHeaderLine('X-Retry-Counter');
You can also specify a custom header key:
# Retry this request if it times out: $response = $client->get('/some-path', [ 'expose_retry_header' => true, 'retry_header' => 'X-Retry-Count' ]); # If a request was retried, the response will include the 'X-Retry-Counter' $numRetries = (int) $response->getHeaderLine('X-Retry-Count');
Modifying the expected header name from Retry-After
You can change the header that the client expects the server to respond with. By default,
the client looks for the Retry-After
header, but in some edge-cases, servers may choose
to respond with a different header.
# Change the name of the expected retry after header to something else: $response = $client->get('/some-path', [ 'retry_after_header' => 'X-Custom-Retry-After-Seconds' ]); # Otherwise, the default `Retry-After` header will be used. $response = $client->get('/some-path');
Setting a custom date format for the Retry-After
header
You can change the expected date format expected from the server that the client library expects. By default, this library expects an RFC 2822 header as defined in the HTTP spec. In certain edge-cases, the server may implement some other date format. This library allows for the possibility of that.
# Change the expected date format of the `Retry-After` header $response = $client->get('/some-path', [ 'retry_after_date_format' => 'Y-m-d H:i:s' ]); # Otherwise, the default date format for the `Retry-After` header will be used. # (ex. 'Wed, 24 Nov 2020 07:28:00 GMT') $response = $client->get('/some-path');
Note: Be careful not to use this option with the Unix epoch (u
) format. The
client will interpret this value as an integer and subsequently timeout
for a very, very long time.
Setting a maximum allowable timeout value
If you want the client to not accept timeout values greater than a certain
value, set the max_allowable_timeout_secs
option. This will return a static
number once the timeout reaches a specified length regardless if it is calculated
using the default backoff algorithm or returned from the server via the Retry-After
header.
By default, this value is null
, which means there is no limit.
# Set the maximum allowable timeout # If the calculated value exceeds 120 seconds, then just return 120 seconds $response = $client->get('/some-path', [ 'max_allowable_timeout_secs' => 120 ]);
Change log
Please see CHANGELOG for more information on what has changed recently.
Testing
$ composer test
Note: Since this library tests timeouts, a few of the tests take a 2-3 seconds to run.
Contributing
Please see CONTRIBUTING for details.
Credits
License
The MIT License (MIT). Please see License File for more information.