Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/makise-co/http-router
HTTP Router based on FastRoute
https://github.com/makise-co/http-router
Last synced: 2 months ago
JSON representation
HTTP Router based on FastRoute
- Host: GitHub
- URL: https://github.com/makise-co/http-router
- Owner: makise-co
- License: mit
- Created: 2020-10-11T17:01:18.000Z (over 4 years ago)
- Default Branch: main
- Last Pushed: 2020-10-25T00:00:21.000Z (about 4 years ago)
- Last Synced: 2024-08-03T23:23:58.244Z (5 months ago)
- Language: PHP
- Size: 55.7 KB
- Stars: 3
- Watchers: 2
- Forks: 0
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
- awesome-psr15-middlewares - makise-co/http-router - A HTTP Router based on FastRoute with middleware and DI support. Optimized for long-running applications. (Packages / Router)
README
# PSR HTTP Router
A PSR HTTP Router based on FastRoute with Middleware support## Requirements
* PHP >= 7.4## Installation
`composer require makise-co/http-router`## Features
* Middlewares
* Dependency Injection to Route Handlers (through PSR Container and [php-di/invoker](https://github.com/PHP-DI/Invoker))
* Supports all of most common request handler declarations (Closures, callables, array-callables, class@method, class::method)
* Strict PSR standards following (PSR-7 and PSR-15), each route handler **MUST** return instance of PSR ResponseInterface
* High portability and customization (any implementation part can be replaced with your own implementation)## FAQ
* How to get the Route instance that is processing the request?```php
$route = $request->getAttribute(RouteInterface::class);
```
* How to get route arguments?```php
// method 1 - Using DependencyInjection, many frameworks provides this way
function apiEndpoint(int $id): \Psr\Http\Message\ResponseInterface
{
}
// method 2 - Manually getting route arguments
function apiEndpoint(\Psr\Http\Message\ServerRequestInterface $request): \Psr\Http\Message\ResponseInterface
{
$args = $request->getAttribute(Router::ROUTE_ARGS);
}
```## Performance
For three million calls (using php-di/invoker):
```
Time took: 4.95980096 secs (0.00000165 secs per request)
```For three million calls (using native function call implementation without DI):
```
Time took: 3.31491995 secs (0.00000110 secs per request)
```Benchmark code can be found [here](examples/benchmark.php).
* Benchmarks were performed on PHP 7.4 with OPcache enabled
* CPU: Intel Core i7-9750H 6 cores (CPU frequency during benchmarks: 4.07 GHz)
* OS: Ubuntu 20.04 (WSL 2)## Usage
```php
create(
// container that is used to resolve route handlers
// and inject dependencies into route handlers
new Container(),
);$collector->get('/', function (): Response {
return new Response(200, [], 'Welcome');
});$collector->addGroup(
'api',
[
'namespace' => 'App\\Http\\Controllers\\',
'middleware' => [AddHeaderMiddleware1::class]
],
function (RouteCollectorInterface $collector) {
$collector
->get('/balance', function (): Response {
return new Response(200, [], '228');
})
->setName('api_balance')
->withMiddleware(AddHeaderMiddleware2::class)
// or
->withMiddleware(new AddHeaderMiddleware2())
->withAttribute('auth_permissions', ['customer']);$collector->post('/posts', function (ServerRequestInterface $request): Response {
$body = $request->getParsedBody();return new Response(200, [], json_encode($body));
});$collector->get('/posts/{id:\d+}', function (int $id): Response {
return new Response(200, [], "Post {$id}");
});$collector->get('/user', 'UserController@index');
// or
$collector->get('/user', [$userController, 'index']);
// or
$collector->get('/user', 'UserController::index');
// or
$collector->get('/user', [UserController::class, 'index']);
}
);$router = $collector->getRouter();
$request = new ServerRequest('GET', '/api/balance');
$response = $router->handle($request);
```## Lazy resolving
When you write your route handlers using Controllers approach
you may want to inject some dependencies into your Controller constructor.
By default, all dependencies will be automatically injected into your controller after you register the route.This behavior can cause various bugs or even errors in your application.
Let's imagine your controller class looks like:
```php
class PostController
{
private ORMInterface $orm;public function __construct(ORMInterface $orm)
{
$this->orm = $orm;
}
public function index(): Response
{
$rows = $this
->orm
->getRepository(\App\Entity\Post::class)
->findAll();return new Response(json_encode($rows));
}
}
```And you are registering a route for the index method of PostController:
```php
$collector->get('/posts', 'PostController@index');
```After that an instance of your PostController class will be immediately created,
and an instance of ORM will be injected into it.
ORM can initiate database connections and query for database schema information.In multi-process app architecture (when your app has a master process and many worker processes)
this may lead to unnecessary resource consumption
(draining of database connections, memory consumption, slower application boot time).
And it can even lead to an error in the application, especially with Swoole backends, because all non-blocking I/O operations
MUST be performed in the Coroutine context.This package offers a simple solution to avoid all kind of these problems: Lazy route handlers resolution.
Just use `RouteCollectorLazyFactory` instead of `RouteCollectorFactory`.Instead of immediately resolving the route handlers, this process is postponed until the routes compilation stage.
The compilation stage of the routes occur when you call the "getRouter" method on the "RouteCollector" instance.
This means that the route handlers are resolved at the time the "getRouter" method is called.## Error handling
```php
getMessage());
}if ($e instanceof MethodNotAllowedException) {
// following https://tools.ietf.org/html/rfc7231#section-6.5.5
return new Response(405, ['Allow' => $e->getAllowedMethods()], $e->getMessage());
}return new Response(500, [], "Internal Server Error
{$e->getMessage()}");
}
}$router = $collector->getRouter();
$app = (new \MakiseCo\Middleware\MiddlewarePipeFactory())->create([
new ErrorHandlingMiddleware(new ErrorHandler()), // placing error handling middleware first
$router
]);
// or
$app = new \MakiseCo\Middleware\Dispatcher([
new ErrorHandlingMiddleware(new ErrorHandler()), // placing error handling middleware first
$router
]);
// or use any other middleware dispatcher$response = $app->handle($request);
```More complete examples can be found [here](examples/collector.php).