sergiosgc / rest-router
Filesystem based request router for HTTP verb-aware applications (RESTful)
Requires
- php: >=7.0.0
- willdurand/negotiation: 3.*
This package is auto-updated.
Last update: 2025-03-11 19:56:52 UTC
README
Filesystem based request router for HTTP verb-aware applications (RESTful)
Objective
The usual PHP request routing pattern is that the script filename is obtained from the request URI, possibly after appending index.php
. This router follows the same logic, with a step up the RESTful path. It uses the request HTTP verb (GET
, POST
, PUT
, ...) to find out the correct script in the filesystem path that will handle the request.
Let's assume an application stored under the document root /srv/www/myapp
receiving three requests:
- A GET to
/customer/
- A POST to
/customer/
- A PUT to
/customer/10
Under typical PHP routing, the first two requests will be handled by /srv/www/myapp/customer/index.php
and the third one would require request rewriting at the webserver layer
Using this router, the requests will be handled by:
/srv/www/myapp/customer/get.php
/srv/www/myapp/customer/post.php
/srv/www/myapp/customer/put.php
(extraction of the customer id is possible and covered in this readme)
This eases development of RESTful apps, while leveraging the application directory structure to map out the URL space.
Installation
The easiest way to install and use is via Composer. On your composer.json:
# composer.json
{
"repositories": [
{
"type": "vcs",
"url": "https://github.com/sergiosgc/composer-fs-rest-router.git"
}
],
"require": {
"sergiosgc/rest-router": "dev-master"
}
}
Then issue either composer update
or composer install
.
Now, on your document root, create a catch-all script named index.php:
<?php
require_once('vendor/autoload.php');
(new \sergiosgc\router\Rest())->route();
And configure your webserver to route all requests for the vhost onto this index.php. For example, on nginx, the vhost file would be similar to:
server {
listen 80;
root /srv/www/myapp;
index /index.php;
location / {
try_files $uri /index.php?$args;
}
location ~ \.php$ {
fastcgi_split_path_info ^(.+\.php)(/.+)$;
fastcgi_pass unix:/run/php/php7.0-fpm.sock;
include fastcgi_params;
}
}
Basic Usage
Just create directories for your RESTful objects. For example, to handle requests for /customer/
, create a directory /srv/www/myapp/customer/
. Then, create one script file for each method you wish to handle:
post.php
for the POST method (create)get.php
for GET/HEAD methods (read)head.php
for the HEAD methodput.php
for the PUT method (update)delete.php
for the DELETE method (delete)
If head.php
is not present, the router will try to fallback to get.php
. Any method is accepted; the filename should be the lowercased method name with .php
appended.
Argument extraction from the URL
The router will set these two variables:
$_SERVER['ROUTER_PATHBOUND_REQUEST_URI']
: The part of the URL that matches filesystem directories$_SERVER['ROUTER_PATHBOUND_SCRIPT_FILENAME']
: The script filename obtained from$_SERVER['ROUTER_PATHBOUND_REQUEST_URI']
$_SERVER['ROUTER_PATHBOUND_REQUEST_URI'] is the result of consuming, from $_SERVER['REQUEST_URI'], the successive matches from regex files found in the matching path. It matches existing directories in the document root.
When using beautified URLs, you may extract extra data from the unbound part of the request URL. On the example above, a PUT to /customer/10
handled by /srv/www/myapp/customer/put.php
will have 10
as the value of $_SERVER['ROUTER_UNBOUND_REQUEST_URI']
.
For more complex URLs, you may use regular expressions to extract the parameters. On the same dir as the verb handling php script, drop a regular expression with named groups. Let's imagine the URL /customer/10/20
is interpreted as customers from minid 10 to maxid 20. Alongside /srv/www/myapp/customer/get.php
create /srv/www/myapp/customer/get.regex
with this content:
_^(?<minid>[0-9]+)/(?<maxid>[0-9]+)_
The request router will use this regex to:
- Match the unbound request uri and extract arguments minid and maxid. These will populate
$_REQUEST
- Validate that the unbound URI is valid (throw a 404 otherwise)
URL subtrees
If you wish a script to automatically handle all requests from that URL tree node down, you may do so with a regular expression that consumes the remaining URI.
For example, a script /srv/www/myapp/customer/order/get.php
may handle both the /customer/order/
and the /customer/order/return/
URLs. To do so, place a regex in /srv/www/myapp/customer/order/get.regex
with:
_^(?<remaining>.*)$_
The most specific script in the path will be used. For example, if these two scripts are present: /srv/www/myapp/customer/order/get.php
and /srv/www/myapp/customer/get.php
, the URL /customer/order/return/
will be handled by /srv/www/myapp/customer/order/get.php
.
Workaround for HTML verb limitation
While HTTP defines a rich set of methods, HTML actually allows only GET and POST. To workaround this limitation, in HTML you may use the x-verb
parameter in any request to override the HTTP verb. For example, the link /customer/12?x-verb=DELETE
, when clicked in a browser will issue a GET HTTP request, but will be processed as a DELETE request by the router.
Debugging
You may get a debug of URL matching by the router, instead of a regular response, by enabling the $debug static variable. This may easily be done in the catch-all script referred above, like so:
<?php
require_once('vendor/autoload.php');
\sergiosgc\router\Rest::$debug = true;
(new \sergiosgc\router\Rest())->route();