docs/handlers-and-middleware.md
Guzzle clients use a handler and middleware system to send HTTP requests.
A handler function accepts a Psr\Http\Message\RequestInterface and array of request options and returns a GuzzleHttp\Promise\PromiseInterface that is fulfilled with a Psr\Http\Message\ResponseInterface or rejected with a reason.
You can provide a custom handler to a client using the handler option of a client constructor. It is important to understand that several request options used by Guzzle require that specific middlewares wrap the handler used by the client. You can ensure that the handler you provide to a client uses the default middlewares by wrapping the handler in the GuzzleHttp\HandlerStack::create(callable $handler = null) static method.
use GuzzleHttp\Client;
use GuzzleHttp\HandlerStack;
use GuzzleHttp\Handler\CurlHandler;
$handler = new CurlHandler();
$stack = HandlerStack::create($handler); // Wrap w/ middleware
$client = new Client(['handler' => $stack]);
The create method adds default handlers to the HandlerStack. When the HandlerStack is resolved, the handlers will execute in the following order:
http_errors- No op when sending a request. The response status code is checked in the response processing when returning a response promise up the stack.allow_redirects- No op when sending a request. Following redirects occurs when a response promise is being returned up the stack.cookies- Adds cookies to requests.prepare_body- The body of an HTTP request will be prepared (e.g., add default headers like Content-Length, Content-Type, etc.).- <send request with handler>
prepare_body- no op on response processing.cookies- extracts response cookies into the cookie jar.allow_redirects- Follows redirects.http_errors- throws exceptions when the response status code>=400.
When provided no $handler argument, GuzzleHttp\HandlerStack::create() will choose the most appropriate handler based on the extensions available on your system.
[!IMPORTANT] The handler provided to a client determines how request options are applied and utilized for each request sent by a client. For example, if you do not have a cookie middleware associated with a client, then setting the
cookiesrequest option will have no effect on the request.
Middleware augments the functionality of handlers by invoking them in the process of generating responses. Middleware is implemented as a higher order function that takes the following form.
use Psr\Http\Message\RequestInterface;
function my_middleware()
{
return function (callable $handler) {
return function (RequestInterface $request, array $options) use ($handler) {
return $handler($request, $options);
};
};
}
Middleware functions return a function that accepts the next handler to invoke. This returned function then returns another function that acts as a composed handler-- it accepts a request and options, and returns a promise that is fulfilled with a response. Your composed middleware can modify the request, add custom request options, and modify the promise returned by the downstream handler.
Here's an example of adding a header to each request.
use Psr\Http\Message\RequestInterface;
function add_header($header, $value)
{
return function (callable $handler) use ($header, $value) {
return function (
RequestInterface $request,
array $options
) use ($handler, $header, $value) {
$request = $request->withHeader($header, $value);
return $handler($request, $options);
};
};
}
Once a middleware has been created, you can add it to a client by either wrapping the handler used by the client or by decorating a handler stack.
use GuzzleHttp\HandlerStack;
use GuzzleHttp\Handler\CurlHandler;
use GuzzleHttp\Client;
$stack = new HandlerStack();
$stack->setHandler(new CurlHandler());
$stack->push(add_header('X-Foo', 'bar'));
$client = new Client(['handler' => $stack]);
Now when you send a request, the client will use a handler composed with your added middleware, adding a header to each request.
Here's an example of creating a middleware that modifies the response of the downstream handler. This example adds a header to the response.
use Psr\Http\Message\RequestInterface;
use Psr\Http\Message\ResponseInterface;
use GuzzleHttp\HandlerStack;
use GuzzleHttp\Handler\CurlHandler;
use GuzzleHttp\Client;
function add_response_header($header, $value)
{
return function (callable $handler) use ($header, $value) {
return function (
RequestInterface $request,
array $options
) use ($handler, $header, $value) {
$promise = $handler($request, $options);
return $promise->then(
function (ResponseInterface $response) use ($header, $value) {
return $response->withHeader($header, $value);
}
);
};
};
}
$stack = new HandlerStack();
$stack->setHandler(new CurlHandler());
$stack->push(add_response_header('X-Foo', 'bar'));
$client = new Client(['handler' => $stack]);
Creating a middleware that modifies a request is made much simpler using the GuzzleHttp\Middleware::mapRequest() middleware. This middleware accepts a function that takes the request argument and returns the request to send.
use Psr\Http\Message\RequestInterface;
use GuzzleHttp\HandlerStack;
use GuzzleHttp\Handler\CurlHandler;
use GuzzleHttp\Client;
use GuzzleHttp\Middleware;
$stack = new HandlerStack();
$stack->setHandler(new CurlHandler());
$stack->push(Middleware::mapRequest(function (RequestInterface $request) {
return $request->withHeader('X-Foo', 'bar');
}));
$client = new Client(['handler' => $stack]);
Modifying a response is also much simpler using the GuzzleHttp\Middleware::mapResponse() middleware.
use Psr\Http\Message\ResponseInterface;
use GuzzleHttp\HandlerStack;
use GuzzleHttp\Handler\CurlHandler;
use GuzzleHttp\Client;
use GuzzleHttp\Middleware;
$stack = new HandlerStack();
$stack->setHandler(new CurlHandler());
$stack->push(Middleware::mapResponse(function (ResponseInterface $response) {
return $response->withHeader('X-Foo', 'bar');
}));
$client = new Client(['handler' => $stack]);
GuzzleHttp\Middleware::log() logs requests, responses, and errors using a GuzzleHttp\MessageFormatterInterface implementation. GuzzleHttp\MessageFormatter formats exactly the values requested by its template. Templates that include full messages, headers, bodies, URIs, URLs, or dynamic header placeholders can include sensitive data such as credentials, cookies, tokens, or request bodies. Avoid using debug or full-message templates in production unless logs are protected, or provide a custom formatter or logger processor that redacts sensitive data before logs are written.
A handler stack represents a stack of middleware to apply to a base handler function. You can push middleware to the stack to add to the top of the stack, and unshift middleware onto the stack to add to the bottom of the stack. When the stack is resolved, the handler is pushed onto the stack. Each value is then popped off of the stack, wrapping the previous value popped off of the stack.
use GuzzleHttp\Client;
use GuzzleHttp\HandlerStack;
use GuzzleHttp\Middleware;
use GuzzleHttp\Utils;
use Psr\Http\Message\RequestInterface;
$stack = new HandlerStack();
$stack->setHandler(Utils::chooseHandler());
$stack->push(Middleware::mapRequest(function (RequestInterface $r) {
echo 'A';
return $r;
}));
$stack->push(Middleware::mapRequest(function (RequestInterface $r) {
echo 'B';
return $r;
}));
$stack->push(Middleware::mapRequest(function (RequestInterface $r) {
echo 'C';
return $r;
}));
$client->request('GET', 'http://httpbin.org/');
// echoes 'ABC';
$stack->unshift(Middleware::mapRequest(function (RequestInterface $r) {
echo '0';
return $r;
}));
$client = new Client(['handler' => $stack]);
$client->request('GET', 'http://httpbin.org/');
// echoes '0ABC';
You can give middleware a name, which allows you to add middleware before other named middleware, after other named middleware, or remove middleware by name.
use Psr\Http\Message\RequestInterface;
use GuzzleHttp\Middleware;
// Add a middleware with a name
$stack->push(Middleware::mapRequest(function (RequestInterface $r) {
return $r->withHeader('X-Foo', 'Bar');
}, 'add_foo'));
// Add a middleware before a named middleware (unshift before).
$stack->before('add_foo', Middleware::mapRequest(function (RequestInterface $r) {
return $r->withHeader('X-Baz', 'Qux');
}, 'add_baz'));
// Add a middleware after a named middleware (pushed after).
$stack->after('add_baz', Middleware::mapRequest(function (RequestInterface $r) {
return $r->withHeader('X-Lorem', 'Ipsum');
}));
// Remove a middleware by name
$stack->remove('add_foo');
By default, Guzzle does not ask handlers to share low-level transport state.
Use the transport_sharing client constructor option when Guzzle creates the
default handler:
use GuzzleHttp\Client;
use GuzzleHttp\TransportSharing;
$client = new Client([
'transport_sharing' => TransportSharing::HANDLER_PREFER,
]);
TransportSharing::NONE disables transport sharing. This is the default.
TransportSharing::HANDLER_PREFER asks the selected handler to share transport
state for the lifetime of that handler when it can. Guzzle's cURL handlers use
cURL share handles to share DNS and SSL session cache state. If sharing cannot
be configured, or if the selected handler does not support sharing, Guzzle
continues without sharing.
TransportSharing::HANDLER_REQUIRE requires handler-lifetime transport
sharing. Guzzle fails when it cannot select a cURL handler with cURL share
support, when sharing cannot be configured, or when a request is routed to a
handler that does not support sharing.
Transport sharing does not share cookies. Cookies are managed by Guzzle middleware.
When constructing cURL handlers manually, configure sharing with the handler
transport_sharing option:
use GuzzleHttp\Handler\CurlHandler;
use GuzzleHttp\TransportSharing;
$handler = new CurlHandler([
'transport_sharing' => TransportSharing::HANDLER_REQUIRE,
]);
As stated earlier, a handler is a function that accepts a Psr\Http\Message\RequestInterface and an array of request options. A handler used with Guzzle middleware returns a GuzzleHttp\Promise\PromiseInterface that is fulfilled with a Psr\Http\Message\ResponseInterface or rejected with a reason.
use GuzzleHttp\Promise\PromiseInterface;
use Psr\Http\Message\RequestInterface;
function handler(RequestInterface $request, array $options): PromiseInterface
{
// Send the request and settle the returned promise with a response.
}
Most custom handlers should be wrapped with GuzzleHttp\HandlerStack::create($handler). This keeps Guzzle's default middleware behavior for redirects, cookies, HTTP errors, and request body preparation while still allowing the custom handler to own the underlying transport.
Synchronous client methods set GuzzleHttp\RequestOptions::SYNCHRONOUS before invoking the handler and then call wait() on the returned promise. The synchronous option is a hint that the caller intends to wait, not permission for a handler to return a response directly. A handler promise's wait() method must complete the transfer or throw the rejection reason. Asynchronous client methods return the promise without blocking.
Request options are applied by different parts of Guzzle. A custom handler should know which options are already reflected in the request it receives and which options still need transport support.
| Owner | Examples | Notes |
|---|---|---|
| Client-applied options | base_uri, headers, body, form_params, multipart, json, query, version, idn_conversion, basic auth | These options affect request construction or request mutation before the final handler sends the request. |
| Middleware-dependent options | allow_redirects, cookies, http_errors, expect | These options require the relevant middleware, normally from HandlerStack::create(). |
| Handler-owned options | delay, timeout, connect_timeout, read_timeout, stream, sink, verify, cert, ssl_key, proxy, force_ip_resolve, decode_content, progress, on_headers, on_stats, debug | These options describe transport behavior and need explicit handler support or clear unsupported behavior. |
Some options have split responsibilities. Basic auth adds an Authorization header before the handler runs, while digest and NTLM authentication are implemented through cURL options by Guzzle's built-in cURL handlers. The expect option is used by the body preparation middleware to add Expect: 100-Continue, but the transport still determines whether the protocol workflow is supported. The decode_content option can affect the Accept-Encoding request header, but response decoding is handled by the transport. Redirect middleware validates redirect targets with allow_redirects.protocols, but the handler is still responsible for enforcing which schemes it can send.
A handler is responsible for applying the following request options. These request options are a subset of request options called "transfer options".
certcert_typeconnect_timeoutcrypto_methoddebugdelaydecode_contentexpectforce_ip_resolveon_headerson_statsprogressprotocolsproxyread_timeoutsinktimeoutssl_keyssl_key_typestreamverifyTransport-specific options such as curl and stream_context are intended for the handlers that understand them. A non-cURL handler should reject or document how it treats cURL-specific options, and a non-stream handler should do the same for PHP stream context options.
Custom handlers do not need to support every handler-owned option, but they should not silently ignore options users reasonably expect to affect transport behavior. Prefer implementing the option, rejecting the request with a clear exception when the option is present, or documenting a deliberate no-op where the option has no meaningful transport equivalent.
The on_headers option is invoked after the response headers have been received and before response body bytes are written to the configured sink. In Guzzle 7, the callback receives the response object. If it throws, the request promise is rejected with a GuzzleHttp\Exception\RequestException that wraps the thrown exception.
The on_stats option is invoked when the handler has finished sending a request, with a GuzzleHttp\TransferStats object that describes the response received or the error encountered. Built-in cURL handlers may invoke on_stats per low-level transfer attempt.
The progress option is invoked with the documented argument order: the total number of bytes expected to be downloaded, the number of bytes downloaded so far, the total number of bytes expected to be uploaded, and the number of bytes uploaded so far. A handler that cannot provide progress information should reject the option or clearly document that progress reporting is unsupported.
Guzzle promises settle callbacks through the Guzzle promise task queue. The queue is drained automatically during synchronous wait(), but it is not automatically driven by arbitrary event loops. A handler that resolves or rejects Guzzle promises from an external scheduler must ensure the Guzzle promise task queue is drained while that scheduler is running.
After resolving or rejecting a Guzzle promise from an external scheduler, schedule GuzzleHttp\Promise\Utils::queue()->run() on that scheduler soon, without blocking the scheduler.
use GuzzleHttp\Promise\Utils as PromiseUtils;
// Pseudocode for an external scheduler callback.
$promise->resolve($response);
ExternalLoop::queue(static function (): void {
PromiseUtils::queue()->run();
});
Do not rely only on wait() to drain the queue because asynchronous users may attach callbacks and expect them to run while their event loop is active. Avoid running the queue in a tight polling loop, and avoid replacing the global task queue unless your library fully owns the process runtime. External futures or promises should generally be adapted by intentionally settling a Guzzle promise, because then(), wait(), and cancel() semantics often differ between promise implementations.