2012-07-31 20:50:36 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
class AltoRouter {
|
2012-08-01 11:05:42 +02:00
|
|
|
|
2013-07-14 23:22:59 +02:00
|
|
|
protected $routes = array();
|
|
|
|
protected $namedRoutes = array();
|
|
|
|
protected $basePath = '';
|
2014-01-08 12:27:35 +01:00
|
|
|
protected $matchTypes = array(
|
|
|
|
'i' => '[0-9]++',
|
|
|
|
'a' => '[0-9A-Za-z]++',
|
|
|
|
'h' => '[0-9A-Fa-f]++',
|
|
|
|
'*' => '.+?',
|
|
|
|
'**' => '.++',
|
|
|
|
'' => '[^/\.]++'
|
|
|
|
);
|
2012-07-31 20:50:36 +02:00
|
|
|
|
2014-01-05 16:08:34 +03:00
|
|
|
/**
|
|
|
|
* Create router in one call from config.
|
|
|
|
*
|
2014-01-07 13:15:53 +03:00
|
|
|
* @param array $routes
|
2014-01-05 16:08:34 +03:00
|
|
|
* @param string $basePath
|
2014-01-08 12:27:35 +01:00
|
|
|
* @param array $matchTypes
|
2014-01-05 16:08:34 +03:00
|
|
|
*/
|
2014-01-08 12:27:35 +01:00
|
|
|
public function __construct( $routes = array(), $basePath = '', $matchTypes = array() ) {
|
2014-04-16 11:17:42 +02:00
|
|
|
$this->addRoutes($routes);
|
2014-01-08 14:39:18 +01:00
|
|
|
$this->setBasePath($basePath);
|
|
|
|
$this->addMatchTypes($matchTypes);
|
2014-04-16 11:17:42 +02:00
|
|
|
}
|
2014-01-08 12:27:35 +01:00
|
|
|
|
2014-04-16 11:17:42 +02:00
|
|
|
/**
|
|
|
|
* Add multiple routes at once from array in the following format:
|
|
|
|
*
|
|
|
|
* $routes = array(
|
|
|
|
* array($method, $route, $target, $name)
|
|
|
|
* );
|
|
|
|
*
|
|
|
|
* @param array $routes
|
|
|
|
* @return void
|
|
|
|
* @author Koen Punt
|
|
|
|
*/
|
|
|
|
public function addRoutes(array $routes){
|
|
|
|
foreach($routes as $route) {
|
|
|
|
call_user_func_array(array($this, 'map'), $route);
|
2014-01-05 16:08:34 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-08-01 11:05:42 +02:00
|
|
|
/**
|
2012-09-07 21:40:41 +02:00
|
|
|
* Set the base path.
|
|
|
|
* Useful if you are running your application from a subdirectory.
|
|
|
|
*/
|
2012-07-31 20:50:36 +02:00
|
|
|
public function setBasePath($basePath) {
|
|
|
|
$this->basePath = $basePath;
|
|
|
|
}
|
2013-03-26 19:38:53 +01:00
|
|
|
|
2014-01-08 12:27:35 +01:00
|
|
|
/**
|
2014-01-08 14:39:18 +01:00
|
|
|
* Add named match types. It uses array_merge so keys can be overwritten.
|
2014-01-08 12:27:35 +01:00
|
|
|
*
|
2014-01-08 14:39:18 +01:00
|
|
|
* @param array $matchTypes The key is the name and the value is the regex.
|
2014-01-08 12:27:35 +01:00
|
|
|
*/
|
2014-01-08 14:39:18 +01:00
|
|
|
public function addMatchTypes($matchTypes) {
|
2014-01-08 12:27:35 +01:00
|
|
|
$this->matchTypes = array_merge($this->matchTypes, $matchTypes);
|
|
|
|
}
|
|
|
|
|
2012-07-31 20:50:36 +02:00
|
|
|
/**
|
2012-09-07 21:40:41 +02:00
|
|
|
* Map a route to a target
|
|
|
|
*
|
|
|
|
* @param string $method One of 4 HTTP Methods, or a pipe-separated list of multiple HTTP Methods (GET|POST|PUT|DELETE)
|
|
|
|
* @param string $route The route regex, custom regex must start with an @. You can use multiple pre-set regex filters, like [i:id]
|
|
|
|
* @param mixed $target The target where this route should point to. Can be anything.
|
|
|
|
* @param string $name Optional name of this route. Supply if you want to reverse route this url in your application.
|
|
|
|
*
|
|
|
|
*/
|
2012-08-01 10:57:44 +02:00
|
|
|
public function map($method, $route, $target, $name = null) {
|
2013-03-06 11:09:46 +01:00
|
|
|
|
2012-08-01 10:57:44 +02:00
|
|
|
$this->routes[] = array($method, $route, $target, $name);
|
2013-03-06 11:09:46 +01:00
|
|
|
|
2012-08-01 10:57:44 +02:00
|
|
|
if($name) {
|
2013-03-06 11:09:46 +01:00
|
|
|
if(isset($this->namedRoutes[$name])) {
|
|
|
|
throw new \Exception("Can not redeclare route '{$name}'");
|
2012-08-01 10:57:44 +02:00
|
|
|
} else {
|
|
|
|
$this->namedRoutes[$name] = $route;
|
|
|
|
}
|
2013-03-06 11:09:46 +01:00
|
|
|
|
2012-07-31 20:50:36 +02:00
|
|
|
}
|
2012-08-01 11:05:42 +02:00
|
|
|
|
|
|
|
return;
|
2012-07-31 20:50:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2012-09-07 21:40:41 +02:00
|
|
|
* Reversed routing
|
|
|
|
*
|
|
|
|
* Generate the URL for a named route. Replace regexes with supplied parameters
|
|
|
|
*
|
|
|
|
* @param string $routeName The name of the route.
|
|
|
|
* @param array @params Associative array of parameters to replace placeholders with.
|
|
|
|
* @return string The URL of the route with named parameters in place.
|
|
|
|
*/
|
2012-07-31 20:50:36 +02:00
|
|
|
public function generate($routeName, array $params = array()) {
|
|
|
|
|
|
|
|
// Check if named route exists
|
|
|
|
if(!isset($this->namedRoutes[$routeName])) {
|
|
|
|
throw new \Exception("Route '{$routeName}' does not exist.");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Replace named parameters
|
|
|
|
$route = $this->namedRoutes[$routeName];
|
2013-11-08 15:58:39 +01:00
|
|
|
|
|
|
|
// prepend base path to route url again
|
|
|
|
$url = $this->basePath . $route;
|
2012-07-31 20:50:36 +02:00
|
|
|
|
|
|
|
if (preg_match_all('`(/|\.|)\[([^:\]]*+)(?::([^:\]]*+))?\](\?|)`', $route, $matches, PREG_SET_ORDER)) {
|
2013-03-06 11:09:46 +01:00
|
|
|
|
2012-07-31 20:50:36 +02:00
|
|
|
foreach($matches as $match) {
|
|
|
|
list($block, $pre, $type, $param, $optional) = $match;
|
|
|
|
|
2012-08-16 11:56:13 +01:00
|
|
|
if ($pre) {
|
|
|
|
$block = substr($block, 1);
|
|
|
|
}
|
|
|
|
|
2012-07-31 20:50:36 +02:00
|
|
|
if(isset($params[$param])) {
|
2012-08-16 11:56:13 +01:00
|
|
|
$url = str_replace($block, $params[$param], $url);
|
|
|
|
} elseif ($optional) {
|
2013-07-14 23:22:59 +02:00
|
|
|
$url = str_replace($pre . $block, '', $url);
|
2012-07-31 20:50:36 +02:00
|
|
|
}
|
|
|
|
}
|
2013-03-06 11:09:46 +01:00
|
|
|
|
2012-07-31 20:50:36 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return $url;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2012-09-07 21:40:41 +02:00
|
|
|
* Match a given Request Url against stored routes
|
|
|
|
* @param string $requestUrl
|
|
|
|
* @param string $requestMethod
|
|
|
|
* @return array|boolean Array with route information on success, false on failure (no match).
|
|
|
|
*/
|
2012-07-31 20:50:36 +02:00
|
|
|
public function match($requestUrl = null, $requestMethod = null) {
|
|
|
|
|
2012-08-01 10:57:44 +02:00
|
|
|
$params = array();
|
|
|
|
$match = false;
|
|
|
|
|
2012-07-31 20:50:36 +02:00
|
|
|
// set Request Url if it isn't passed as parameter
|
|
|
|
if($requestUrl === null) {
|
|
|
|
$requestUrl = isset($_SERVER['REQUEST_URI']) ? $_SERVER['REQUEST_URI'] : '/';
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:58:39 +01:00
|
|
|
// strip base path from request url
|
|
|
|
$requestUrl = substr($requestUrl, strlen($this->basePath));
|
|
|
|
|
2012-07-31 20:50:36 +02:00
|
|
|
// Strip query string (?a=b) from Request Url
|
2013-03-19 11:29:19 +07:00
|
|
|
if (($strpos = strpos($requestUrl, '?')) !== false) {
|
2013-03-26 19:38:53 +01:00
|
|
|
$requestUrl = substr($requestUrl, 0, $strpos);
|
2012-09-07 20:38:52 +02:00
|
|
|
}
|
2012-07-31 20:50:36 +02:00
|
|
|
|
|
|
|
// set Request Method if it isn't passed as a parameter
|
|
|
|
if($requestMethod === null) {
|
|
|
|
$requestMethod = isset($_SERVER['REQUEST_METHOD']) ? $_SERVER['REQUEST_METHOD'] : 'GET';
|
|
|
|
}
|
|
|
|
|
|
|
|
// Force request_order to be GP
|
2012-09-07 20:38:52 +02:00
|
|
|
// http://www.mail-archive.com/internals@lists.php.net/msg33119.html
|
|
|
|
$_REQUEST = array_merge($_GET, $_POST);
|
|
|
|
|
|
|
|
foreach($this->routes as $handler) {
|
|
|
|
list($method, $_route, $target, $name) = $handler;
|
|
|
|
|
|
|
|
$methods = explode('|', $method);
|
|
|
|
$method_match = false;
|
|
|
|
|
|
|
|
// Check if request method matches. If not, abandon early. (CHEAP)
|
|
|
|
foreach($methods as $method) {
|
|
|
|
if (strcasecmp($requestMethod, $method) === 0) {
|
|
|
|
$method_match = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Method did not match, continue to next route.
|
|
|
|
if(!$method_match) continue;
|
2012-07-31 20:50:36 +02:00
|
|
|
|
2012-09-07 20:38:52 +02:00
|
|
|
// Check for a wildcard (matches all)
|
|
|
|
if ($_route === '*') {
|
|
|
|
$match = true;
|
|
|
|
} elseif (isset($_route[0]) && $_route[0] === '@') {
|
|
|
|
$match = preg_match('`' . substr($_route, 1) . '`', $requestUrl, $params);
|
|
|
|
} else {
|
|
|
|
$route = null;
|
|
|
|
$regex = false;
|
|
|
|
$j = 0;
|
|
|
|
$n = isset($_route[0]) ? $_route[0] : null;
|
|
|
|
$i = 0;
|
|
|
|
|
|
|
|
// Find the longest non-regex substring and match it against the URI
|
|
|
|
while (true) {
|
|
|
|
if (!isset($_route[$i])) {
|
|
|
|
break;
|
|
|
|
} elseif (false === $regex) {
|
|
|
|
$c = $n;
|
|
|
|
$regex = $c === '[' || $c === '(' || $c === '.';
|
|
|
|
if (false === $regex && false !== isset($_route[$i+1])) {
|
|
|
|
$n = $_route[$i + 1];
|
|
|
|
$regex = $n === '?' || $n === '+' || $n === '*' || $n === '{';
|
|
|
|
}
|
|
|
|
if (false === $regex && $c !== '/' && (!isset($requestUrl[$j]) || $c !== $requestUrl[$j])) {
|
|
|
|
continue 2;
|
|
|
|
}
|
|
|
|
$j++;
|
|
|
|
}
|
|
|
|
$route .= $_route[$i++];
|
|
|
|
}
|
|
|
|
|
|
|
|
$regex = $this->compileRoute($route);
|
|
|
|
$match = preg_match($regex, $requestUrl, $params);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(($match == true || $match > 0)) {
|
|
|
|
|
|
|
|
if($params) {
|
|
|
|
foreach($params as $key => $value) {
|
|
|
|
if(is_numeric($key)) unset($params[$key]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return array(
|
|
|
|
'target' => $target,
|
|
|
|
'params' => $params,
|
|
|
|
'name' => $name
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
2012-07-31 20:50:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2012-09-07 21:40:41 +02:00
|
|
|
* Compile the regex for a given route (EXPENSIVE)
|
|
|
|
*/
|
2012-07-31 20:50:36 +02:00
|
|
|
private function compileRoute($route) {
|
|
|
|
if (preg_match_all('`(/|\.|)\[([^:\]]*+)(?::([^:\]]*+))?\](\?|)`', $route, $matches, PREG_SET_ORDER)) {
|
|
|
|
|
2014-01-08 12:27:35 +01:00
|
|
|
$matchTypes = $this->matchTypes;
|
2014-04-16 11:17:42 +02:00
|
|
|
foreach($matches as $match) {
|
2012-09-07 20:38:52 +02:00
|
|
|
list($block, $pre, $type, $param, $optional) = $match;
|
|
|
|
|
2014-01-08 12:27:35 +01:00
|
|
|
if (isset($matchTypes[$type])) {
|
|
|
|
$type = $matchTypes[$type];
|
2012-09-07 20:38:52 +02:00
|
|
|
}
|
|
|
|
if ($pre === '.') {
|
|
|
|
$pre = '\.';
|
|
|
|
}
|
|
|
|
|
|
|
|
//Older versions of PCRE require the 'P' in (?P<named>)
|
|
|
|
$pattern = '(?:'
|
|
|
|
. ($pre !== '' ? $pre : null)
|
|
|
|
. '('
|
|
|
|
. ($param !== '' ? "?P<$param>" : null)
|
|
|
|
. $type
|
|
|
|
. '))'
|
|
|
|
. ($optional !== '' ? '?' : null);
|
|
|
|
|
|
|
|
$route = str_replace($block, $pattern, $route);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
return "`^$route$`";
|
2012-07-31 20:50:36 +02:00
|
|
|
}
|
2013-03-26 19:38:53 +01:00
|
|
|
}
|