enjoys / middleware-dispatcher
Requires
- php: ^8.1
- enjoys/functions: ^2.0
- psr/http-server-middleware: ^1.0
Requires (Dev)
- infection/infection: ^0.31.2
- phpunit/phpunit: ^12
- symfony/var-dumper: ^7
- vimeo/psalm: ^6
This package is auto-updated.
Last update: 2025-09-05 11:50:59 UTC
README
A PSR-15 compliant middleware dispatcher that provides flexible middleware queue management with support for various middleware formats.
Requirements
- PHP 8.1 or higher
- PSR-15 implementation
- PSR-7/PSR-17 implementations for HTTP messages
Installation
composer require enjoys/middleware-dispatcher
Usage
Basic Usage
use Enjoys\MiddlewareDispatcher\HttpMiddlewareDispatcher; use Psr\Http\Message\ServerRequestInterface; use Psr\Http\Server\MiddlewareInterface; use Psr\Http\Server\RequestHandlerInterface; use Psr\Http\Message\ResponseInterface; // Create your final request handler $finalHandler = new class implements RequestHandlerInterface { public function handle(ServerRequestInterface $request): ResponseInterface { // Return your final response return new Response(); } }; // Create the dispatcher $dispatcher = new HttpMiddlewareDispatcher($finalHandler); // Set up middleware queue $middlewares = [ new MyFirstMiddleware(), new MySecondMiddleware(), function ($request, $handler) { // Callable middleware return $handler->handle($request); } ]; $dispatcher->setQueue($middlewares); // Handle the request $response = $dispatcher->handle($request);
With Middleware Resolver
use Enjoys\MiddlewareDispatcher\MiddlewareResolverInterface; use Psr\Http\Server\MiddlewareInterface; class MyMiddlewareResolver implements MiddlewareResolverInterface { public function resolve(mixed $entry): ?MiddlewareInterface { if (is_string($entry) && class_exists($entry)) { return new $entry(); } if (is_array($entry) && count($entry) === 2) { return new ClassMethodMiddleware($entry[0], $entry[1]); } return $entry; } } $resolver = new MyMiddlewareResolver(); $dispatcher = new HttpMiddlewareDispatcher($finalHandler, $resolver);
Example PSR-11 Container Middleware Resolver
use Enjoys\MiddlewareDispatcher\MiddlewareResolverInterface; use Psr\Container\ContainerExceptionInterface; use Psr\Container\ContainerInterface; use Psr\Http\Server\MiddlewareInterface; final class Psr11ContainerMiddlewareResolver implements MiddlewareResolverInterface { public function __construct(private readonly ContainerInterface $container) { } public function resolve(mixed $entry): null|MiddlewareInterface|callable { if ($entry instanceof MiddlewareInterface || is_callable($entry)) { return $entry; } if (is_string($entry)) { try { $entry = $this->container->get($entry); if ($entry instanceof MiddlewareInterface) { return $entry; } } catch (ContainerExceptionInterface) { return null; } } return null; } }
Adding Middleware Dynamically
// Add middleware to the current position in the queue $dispatcher->addQueue([ new AdditionalMiddleware(), function ($request, $handler) { // Another callable middleware return $handler->handle($request); } ]);
API Reference
Constructor
public function __construct( RequestHandlerInterface $requestHandler, ?MiddlewareResolverInterface $resolver = null )
Methods
setQueue(ArrayIterator|array $queue): void
Sets the middleware queue. Throws InvalidArgumentException if queue is empty.
handle(ServerRequestInterface $request): ResponseInterface
Processes the request through the middleware queue.
addQueue(array $middlewares): void
Adds middleware(s) at the current position in the queue.
Middleware Formats
The dispatcher supports multiple middleware formats:
MiddlewareInterface instances: Objects implementing Psr\Http\Server\MiddlewareInterface
Callables: Functions or closures with signature function(ServerRequestInterface, RequestHandlerInterface): ResponseInterface
Resolvable entries: Any format that can be resolved by your MiddlewareResolverInterface
Error Handling
- Throws
InvalidArgumentException
when setting an empty queue - Throws
RuntimeException
when encountering invalid middleware entries