FastRoute - Fast request router for PHP
This library provides a fast implementation of a regular expression based router. Blog post explaining how the
implementation works and why it is fast., (*1)
This is a fork of nikic/FastRoute that adds
PHP 5.3 compatibility patches., (*2)
Usage
Here's a basic usage example:, (*3)
<?php
require '/path/to/FastRoute/src/bootstrap.php';
$dispatcher = FastRoute\simpleDispatcher(function(FastRoute\RouteCollector $r) {
$r->addRoute('GET', '/user/{id:\d+}', 'handler1');
$r->addRoute('GET', '/user/{id:\d+}/{name}', 'handler2');
// Or alternatively
$r->addRoute('GET', '/user/{id:\d+}[/{name}]', 'common_handler');
});
$routeInfo = $dispatcher->dispatch($httpMethod, $uri);
switch ($routeInfo[0]) {
case FastRoute\Dispatcher::NOT_FOUND:
// ... 404 Not Found
break;
case FastRoute\Dispatcher::METHOD_NOT_ALLOWED:
$allowedMethods = $routeInfo[1];
// ... 405 Method Not Allowed
break;
case FastRoute\Dispatcher::FOUND:
$handler = $routeInfo[1];
$vars = $routeInfo[2];
// ... call $handler with $vars
break;
}
Defining routes
The routes are defined by calling the FastRoute\simpleDispatcher
function, which accepts
a callable taking a FastRoute\RouteCollector
instance. The routes are added by calling
addRoute()
on the collector instance., (*4)
This method accepts the HTTP method the route must match, the route pattern and an associated
handler. The handler does not necessarily have to be a callback (it could also be a controller
class name or any other kind of data you wish to associate with the route)., (*5)
By default a route pattern syntax is used where {foo}
specified a placeholder with name foo
and matching the string [^/]+
. To adjust the pattern the placeholder matches, you can specify
a custom pattern by writing {bar:[0-9]+}
., (*6)
Furthermore parts of the route enclosed in [...]
are considered optional, so that /foo[bar]
will match both /foo
and /foobar
. Optional parts are only supported in a trailing position,
not in the middle of a route., (*7)
A custom pattern for a route placeholder must not use capturing groups. For example {lang:(en|de)}
is not a valid placeholder, because ()
is a capturing group. Instead you can use either
{lang:en|de}
or {lang:(?:en|de)}
., (*8)
The reason simpleDispatcher
accepts a callback for defining the routes is to allow seamless
caching. By using cachedDispatcher
instead of simpleDispatcher
you can cache the generated
routing data and construct the dispatcher from the cached information:, (*9)
<?php
$dispatcher = FastRoute\cachedDispatcher(function(FastRoute\RouteCollector $r) {
$r->addRoute('GET', '/user/{name}/{id:[0-9]+}', 'handler0');
$r->addRoute('GET', '/user/{id:[0-9]+}', 'handler1');
$r->addRoute('GET', '/user/{name}', 'handler2');
}, [
'cacheFile' => __DIR__ . '/route.cache', /* required */
'cacheDisabled' => IS_DEBUG_ENABLED, /* optional, enabled by default */
]);
The second parameter to the function is an options array, which can be used to specify the cache
file location, among other things., (*10)
Dispatching a URI
A URI is dispatched by calling the dispatch()
method of the created dispatcher. This method
accepts the HTTP method and a URI. Getting those two bits of information (and normalizing them
appropriately) is your job - this library is not bound to the PHP web SAPIs., (*11)
The dispatch()
method returns an array whose first element contains a status code. It is one
of Dispatcher::NOT_FOUND
, Dispatcher::METHOD_NOT_ALLOWED
and Dispatcher::FOUND
. For the
method not allowed status the second array element contains a list of HTTP methods allowed for
the supplied URI. For example:, (*12)
[FastRoute\Dispatcher::METHOD_NOT_ALLOWED, ['GET', 'POST']]
NOTE: The HTTP specification requires that a 405 Method Not Allowed
response include the
Allow:
header to detail available methods for the requested resource. Applications using FastRoute
should use the second array element to add this header when relaying a 405 response., (*13)
For the found status the second array element is the handler that was associated with the route
and the third array element is a dictionary of placeholder names to their values. For example:, (*14)
/* Routing against GET /user/nikic/42 */
[FastRoute\Dispatcher::FOUND, 'handler0', ['name' => 'nikic', 'id' => '42']]
Overriding the route parser and dispatcher
The routing process makes use of three components: A route parser, a data generator and a
dispatcher. The three components adhere to the following interfaces:, (*15)
<?php
namespace FastRoute;
interface RouteParser {
public function parse($route);
}
interface DataGenerator {
public function addRoute($httpMethod, $routeData, $handler);
public function getData();
}
interface Dispatcher {
const NOT_FOUND = 0, FOUND = 1, METHOD_NOT_ALLOWED = 2;
public function dispatch($httpMethod, $uri);
}
The route parser takes a route pattern string and converts it into an array of route infos, where
each route info is again an array of it's parts. The structure is best understood using an example:, (*16)
/* The route /user/{id:\d+}[/{name}] converts to the following array: */
[
[
'/user/',
['name', '[^/]+'],
],
[
'/user/',
['name', '[^/]+'],
'/',
['id', '[0-9]+'],
],
]
This array can then be passed to the addRoute()
method of a data generator. After all routes have
been added the getData()
of the generator is invoked, which returns all the routing data required
by the dispatcher. The format of this data is not further specified - it is tightly coupled to
the corresponding dispatcher., (*17)
The dispatcher accepts the routing data via a constructor and provides a dispatch()
method, which
you're already familiar with., (*18)
The route parser can be overwritten individually (to make use of some different pattern syntax),
however the data generator and dispatcher should always be changed as a pair, as the output from
the former is tightly coupled to the input of the latter. The reason the generator and the
dispatcher are separate is that only the latter is needed when using caching (as the output of
the former is what is being cached.), (*19)
When using the simpleDispatcher
/ cachedDispatcher
functions from above the override happens
through the options array:, (*20)
<?php
$dispatcher = FastRoute\simpleDispatcher(function(FastRoute\RouteCollector $r) {
/* ... */
}, [
'routeParser' => 'FastRoute\\RouteParser\\Std',
'dataGenerator' => 'FastRoute\\DataGenerator\\GroupCountBased',
'dispatcher' => 'FastRoute\\Dispatcher\\GroupCountBased',
]);
The above options array corresponds to the defaults. By replacing GroupCountBased
by
GroupPosBased
you could switch to a different dispatching strategy., (*21)
A Note on HEAD Requests
The HTTP spec requires servers to support both GET and HEAD methods:, (*22)
The methods GET and HEAD MUST be supported by all general-purpose servers, (*23)
To avoid forcing users to manually register HEAD routes for each resource we fallback to matching an
available GET route for a given resource. The PHP web SAPI transparently removes the entity body
from HEAD responses so this behavior has no effect on the vast majority of users., (*24)
However, implementors using FastRoute outside the web SAPI environment (e.g. a custom server) MUST
NOT send entity bodies generated in response to HEAD requests. If you are a non-SAPI user this is
your responsibility; FastRoute has no purview to prevent you from breaking HTTP in such cases., (*25)
Finally, note that applications MAY always specify their own HEAD method route for a given
resource to bypass this behavior entirely., (*26)
Credits
This library is based on a router that Levi Morrison implemented for the Aerys server., (*27)
A large number of tests, as well as HTTP compliance considerations, were provided by Daniel Lowrey., (*28)