Cookbook
In This Article
Using Double-Pass Middleware
Expressive uses PSR-15 middleware and request handlers exclusively as of version 3.
In previous releases, however, we supported "double-pass" middleware, and a number of third-party packages provided double-pass middleware. How can you use this middleware with Expressive 3?
What is Double-Pass Middleware?
Double pass middleware receives both the request and a response in addition to the handler, and passes both the request and response to the handler when invoking it:
function (ServerRequestInterface $request, ResponseInterface $response, callable $next) { $response = $next($request, $response); return $response->withHeader('X-Test', time()); }It is termed "double pass" because you pass both the request and response when delegating to the next layer.
doublePassMiddleware function
zend-stratigility v2.2 and v3.0 ship a utility function,
Zend\Stratigility\doublePassMiddleware(), that will decorate a callable
double-pass middleware using a Zend\Stratigility\Middleware\DoublePassMiddlewareDecorator
instance; this latter is a PSR-15 impelementation, and can thus be used in your
middleware pipelines.
The function (and class) also expects a PSR-7
ResponseInterface instance as a second argument; this is then passed as the
$response argument to the double-pass middleware. The following examples
demostrate both piping and routing to double pass middleware using this
technique, and using zend-diactoros to provide the response instance.
use Zend\Diactoros\Response;
use function Zend\Stratigility\doublePassMiddleware;
$app->pipe(doublePassMiddleware(function ($request, $response, $next) {
    $response = $next($request, $response);
    return $response->withHeader('X-Clacks-Overhead', 'GNU Terry Pratchett');
}, new Response())); // <-- note the response
$app->get('/api/ping', doublePassMiddleware(function ($request, $response, $next) {
    return new Response\JsonResponse([
        'ack' => time(),
    ]);
}, new Response())); // <-- note the response
Double-Pass Middleware Services
What if you're piping or routing to a service — for instance, a class provided by a third-party implementation?
In this case, you have one of two options:
- Decorate the middleware before returning it from the factory that creates it.
- Use a delegator factory to decorate the middleware.
Decorating via factory
If you have control of the factory that creates the double-pass middleware you will be using in your application, you can use the strategy outlined above to decorate your middleware before returning it, with one minor change: you can pull a response factory from the container as well.
To demonstrate:
use Psr\Container\ContainerInterface;
use Psr\Http\Message\ResponseInterface;
use function Zend\Stratigility\doublePassMiddleware;
class SomeDoublePassMiddlewareFactory
{
    public function __invoke(ContainerInterface $container)
    {
        // Create the middleware instance somehow. This example
        // assumes it is in `$middleware` when done.
        return doublePassMiddleware(
            $middleware,
            ($container->get(ResponseInterface::class))()
        );
    }
}
That last line may look a little strange.
The Psr\Http\Response\ResponseInterface service returns a callable factory
for producing response instances, and not a response instance itself. As such,
we pull it, and then invoke it to produce the response instance for our
double-pass middleware.
This approach will work, but it means code duplication everywhere you have double-pass middleware. Let's look at the delegator factory solution.
Decorating via delegator factory
Delegator factories can be re-used for multiple services. In our case, we'll re-use it to decorate double-pass middleware.
The delegator factory would look like this:
use Psr\Container\ContainerInterface;
use Psr\Http\Message\ResponseInterface;
use function Zend\Stratigility\doublePassMiddleware;
class DoublePassMiddlewareDelegator
{
    public function __invoke(Container $container, string $serviceName, callable $callback)
    {
        return doublePassMiddleware(
            $callback(),
            ($container->get(ResponseInterface::class))()
        );
    }
}
This looks similar to our previous solution, but is self-contained; we rely on
the $callback argument to produce the middleware we want to decorate.
Then, for each service we have that represents double-pass middleware, we can provide configuration like the following:
return [
    'dependencies' => [
        'delegators' => [
            SomeDoublePassMiddleware::class => [
                DoublePassMiddlewareDelegator::class,
            ],
        ],
    ],
];
This approach has a couple of benefits:
- We do not need to change existing factories.
- We do not need to extend factories from third-party services.
- We can see explicitly in our configuration all services we consume that are double-pass middleware. This will help us identify projects we want to contribute PSR-15 patches to, or potentially migrate away from, or middleware of our own we need to refactor.
Extending the MiddlewareContainer
Another possibility is to extend Zend\Expressive\MiddlewareContainer to add
awareness of double-pass middleware, and have it auto-decorate them for you.
A contributor has created such a library:
- https://github.com/Moln/expressive-callable-middleware-compat
You can install it using composer require moln/expressive-callable-middleware-compat.
Once installed, add its Moln\ExpressiveCallableCompat\ConfigProvider as an
entry in your config/config.php after the Zend\Expressive\ConfigProvider
entry. This last point is particularly important: providers are merged in the order
presented, with later entries having precedence; you need to ensure the new
package overrides the MiddlewareContainer service provided by zend-expressive!
When you use this approach, it will automatically detect double-pass middleware and decorate it for you.
The main drawback with such an approach is that it will not help you identify double-pass middleware in your system.
Found a mistake or want to contribute to the documentation? Edit this page on GitHub!