vulpes / router
Simple router & dispatcher implementation.
v1.4.3
2025-01-26 09:11 UTC
Requires
- php: >=8.2
- psr/container: ^2.0
- psr/http-client: ^1.0
- psr/http-message: ^2.0
- psr/http-server-handler: ^1.0
- psr/http-server-middleware: ^1.0
- psr/log: ^3.0
- psr/simple-cache: ^3.0
- vulpes/http: v1.4.1
- vulpes/http-exceptions: v1.2
Requires (Dev)
- phpunit/phpunit: 10.2.6
README
Available methods
- ANY available to all methods
- CLI used on command line prompt
- GET method requests a representation of the specified resource.
- HEAD method asks for a response identical to a GET request, but without the response body.
- POST method submits an entity to the specified resource, often causing a change in state or side effects on the server.
- PUT method replaces all current representations of the target resource with the request payload.
- DELETE method deletes the specified resource.
- CONNECT method establishes a tunnel to the server identified by the target resource.
- OPTIONS method describes the communication options for the target resource.
- TRACE method performs a message loop-back test along the path to the target resource.
- PATCH method applies partial modifications to a resource.
Path variants, and middlewares
<?php declare(strict_types=1); use Psr\Http\Message\ResponseInterface; use Psr\Http\Message\ServerRequestInterface; use Psr\Http\Server\MiddlewareInterface; use Psr\Http\Server\RequestHandlerInterface; require __DIR__ . '/vendor/autoload.php'; // composer require vulpes/container $container = new Container\Container; /** @var Psr\Container\ContainerInterface $container */ $dispatcher = new Http\Dispatcher(container: $container); // add anonymous Middleware globally as object $dispatcher->addMiddleware(new class implements MiddlewareInterface { public function process(ServerRequestInterface $request, RequestHandlerInterface $handler): ResponseInterface { // TODO before $response = $handler->handle($request); // TODO after return $response; } }); // add Middleware globally as string (the ContainerInterface will create it's instance) $dispatcher->addMiddleware(Middleware\AnotherMiddleware::class); // You can also insert middleware at the very beginning of the line, for example for error handling, // to catch the Http\Exception\NotFoundException thrown by the router $dispatcher->appendMiddleware(Middleware\ErrorHandlerMiddleware::class); $dispatcher // You can set prefix for paths via groups ->addGroup(name: 'rest-v1', path: 'api/v1') ->addMiddleware(Http\Middleware\JsonMiddleware::class) ->appendMiddleware('id-for-a-middleware-1') // You can use more groups for one simple path ->addGroup(name: 'rest-v2', path: 'api/v2') ->addMiddleware(Http\Middleware\JsonMiddleware::class) ->appendMiddleware('id-for-a-middleware-2') // You can use groups for simple authentication layers ->addGroup(name: 'admin') ->appendMiddleware('just-a-container-id-for-an-authentication-middleware') // add command line endpoint MyController->index() ->addPath(method: 'GET', path: '/', requestHandler: MyController::class . '::index') // add Middleware locally to current Path as string (classname), you can use object as well ->addMiddleware(Http\Middleware\JsonMiddleware::class) // add command line endpoint, MyController->service(int $thread) ->addPath(method: 'CLI', path: '/service/{thread:int}', requestHandler: MyController::class . '::service') // add command line endpoint, MyController->create() ->addPath( method: 'POST', // because of groups the full path will be: "/api/v1/entity/create", and "/api/v2/entity/create" path: '/entity/create', requestHandler: MyController::class . '::create', groups: 'rest-v1', 'rest-v2' ) // add Middleware locally to current path as string (classname), you can use object as well ->addMiddleware(Middleware\MyMiddleware::class) ->appendMiddleware(Middleware\AnotherErrorMiddleware::class) // The above middlewares will behave according to the pattern below (with global middlewares), with rest-v1 group // AnotherErrorMiddleware :before // id-for-a-middleware-1 :before // ErrorHandlerMiddleware :before // anonymous :before // AnotherMiddleware :before // JsonMiddleware :before // MyMiddleware :before // MyController->create() - throws Throwable, everything from controller // MyMiddleware :after // JsonMiddleware :after // AnotherMiddleware :after // anonymous :after // ErrorHandlerMiddleware :after - You can catch RouterInterface's NotFoundException here // id-for-a-middleware-1 :after // AnotherErrorMiddleware :after - ... or here // You can use RequestHandler (RequestHandlerInterface) as Controller too ->addPath(method: 'POST', path: '/entity/create', requestHandler: MyRequestHandler::class); // composer require vulpes/http $serverRequest = (new Http\Factory\ServerRequestFactory)->createServerRequestFromGlobals(); try { print $dispatcher->handle($serverRequest)/* ResponseInterface */->getBody(); } catch (Throwable $e) { print $e; }