| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416 | <?php/** * Slim - a micro PHP 5 framework * * @author      Josh Lockhart <[email protected]> * @copyright   2011 Josh Lockhart * @link        http://www.slimframework.com * @license     http://www.slimframework.com/license * @version     2.0.0 * @package     Slim * * MIT LICENSE * * Permission is hereby granted, free of charge, to any person obtaining * a copy of this software and associated documentation files (the * "Software"), to deal in the Software without restriction, including * without limitation the rights to use, copy, modify, merge, publish, * distribute, sublicense, and/or sell copies of the Software, and to * permit persons to whom the Software is furnished to do so, subject to * the following conditions: * * The above copyright notice and this permission notice shall be * included in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */namespace Slim;/** * Route * @package Slim * @author  Josh Lockhart, Thomas Bley * @since   1.0.0 */class Route{    /**     * @var string The route pattern (e.g. "/books/:id")     */    protected $pattern;    /**     * @var mixed The route callable     */    protected $callable;    /**     * @var array Conditions for this route's URL parameters     */    protected $conditions = array();    /**     * @var array Default conditions applied to all route instances     */    protected static $defaultConditions = array();    /**     * @var string The name of this route (optional)     */    protected $name;    /**     * @var array Key-value array of URL parameters     */    protected $params = array();    /**     * @var array value array of URL parameter names     */    protected $paramNames = array();    /**     * @var array key array of URL parameter names with + at the end     */    protected $paramNamesPath = array();    /**     * @var array HTTP methods supported by this Route     */    protected $methods = array();    /**     * @var array[Callable] Middleware to be run before only this route instance     */    protected $middleware = array();    /**     * Constructor     * @param string $pattern  The URL pattern (e.g. "/books/:id")     * @param mixed  $callable Anything that returns TRUE for is_callable()     */    public function __construct($pattern, $callable)    {        $this->setPattern($pattern);        $this->setCallable($callable);        $this->setConditions(self::getDefaultConditions());    }    /**     * Set default route conditions for all instances     * @param  array $defaultConditions     */    public static function setDefaultConditions(array $defaultConditions)    {        self::$defaultConditions = $defaultConditions;    }    /**     * Get default route conditions for all instances     * @return array     */    public static function getDefaultConditions()    {        return self::$defaultConditions;    }    /**     * Get route pattern     * @return string     */    public function getPattern()    {        return $this->pattern;    }    /**     * Set route pattern     * @param  string $pattern     */    public function setPattern($pattern)    {        $this->pattern = $pattern;    }    /**     * Get route callable     * @return mixed     */    public function getCallable()    {        return $this->callable;    }    /**     * Set route callable     * @param  mixed $callable     * @throws \InvalidArgumentException If argument is not callable     */    public function setCallable($callable)    {        if (!is_callable($callable)) {            throw new \InvalidArgumentException('Route callable must be callable');        }        $this->callable = $callable;    }    /**     * Get route conditions     * @return array     */    public function getConditions()    {        return $this->conditions;    }    /**     * Set route conditions     * @param  array $conditions     */    public function setConditions(array $conditions)    {        $this->conditions = $conditions;    }    /**     * Get route name     * @return string|null     */    public function getName()    {        return $this->name;    }    /**     * Set route name     * @param  string $name     */    public function setName($name)    {        $this->name = (string) $name;    }    /**     * Get route parameters     * @return array     */    public function getParams()    {        return $this->params;    }    /**     * Set route parameters     * @param  array $params     */    public function setParams($params)    {        $this->params = $params;    }    /**     * Get route parameter value     * @param  string                    $index     Name of URL parameter     * @return string     * @throws \InvalidArgumentException If route parameter does not exist at index     */    public function getParam($index)    {        if (!isset($this->params[$index])) {            throw new \InvalidArgumentException('Route parameter does not exist at specified index');        }        return $this->params[$index];    }    /**     * Set route parameter value     * @param  string                    $index     Name of URL parameter     * @param  mixed                     $value     The new parameter value     * @throws \InvalidArgumentException If route parameter does not exist at index     */    public function setParam($index, $value)    {        if (!isset($this->params[$index])) {            throw new \InvalidArgumentException('Route parameter does not exist at specified index');        }        $this->params[$index] = $value;    }    /**     * Add supported HTTP method(s)     */    public function setHttpMethods()    {        $args = func_get_args();        $this->methods = $args;    }    /**     * Get supported HTTP methods     * @return array     */    public function getHttpMethods()    {        return $this->methods;    }    /**     * Append supported HTTP methods     */    public function appendHttpMethods()    {        $args = func_get_args();        $this->methods = array_merge($this->methods, $args);    }    /**     * Append supported HTTP methods (alias for Route::appendHttpMethods)     * @return \Slim\Route     */    public function via()    {        $args = func_get_args();        $this->methods = array_merge($this->methods, $args);        return $this;    }    /**     * Detect support for an HTTP method     * @return bool     */    public function supportsHttpMethod($method)    {        return in_array($method, $this->methods);    }    /**     * Get middleware     * @return array[Callable]     */    public function getMiddleware()    {        return $this->middleware;    }    /**     * Set middleware     *     * This method allows middleware to be assigned to a specific Route.     * If the method argument `is_callable` (including callable arrays!),     * we directly append the argument to `$this->middleware`. Else, we     * assume the argument is an array of callables and merge the array     * with `$this->middleware`.  Each middleware is checked for is_callable()     * and an InvalidArgumentException is thrown immediately if it isn't.     *     * @param  Callable|array[Callable]     * @return \Slim\Route     * @throws \InvalidArgumentException If argument is not callable or not an array of callables.     */    public function setMiddleware($middleware)    {        if (is_callable($middleware)) {            $this->middleware[] = $middleware;        } elseif (is_array($middleware)) {            foreach($middleware as $callable) {                if (!is_callable($callable)) {                    throw new \InvalidArgumentException('All Route middleware must be callable');                }            }            $this->middleware = array_merge($this->middleware, $middleware);        } else {            throw new \InvalidArgumentException('Route middleware must be callable or an array of callables');        }        return $this;    }    /**     * Matches URI?     *     * Parse this route's pattern, and then compare it to an HTTP resource URI     * This method was modeled after the techniques demonstrated by Dan Sosedoff at:     *     * http://blog.sosedoff.com/2009/09/20/rails-like-php-url-router/     *     * @param  string $resourceUri A Request URI     * @return bool     */    public function matches($resourceUri)    {        //Convert URL params into regex patterns, construct a regex for this route, init params        $patternAsRegex = preg_replace_callback('#:([\w]+)\+?#', array($this, 'matchesCallback'),            str_replace(')', ')?', (string) $this->pattern));        if (substr($this->pattern, -1) === '/') {            $patternAsRegex .= '?';        }        //Cache URL params' names and values if this route matches the current HTTP request        if (!preg_match('#^' . $patternAsRegex . '$#', $resourceUri, $paramValues)) {            return false;        }        foreach ($this->paramNames as $name) {            if (isset($paramValues[$name])) {                if (isset($this->paramNamesPath[ $name ])) {                    $this->params[$name] = explode('/', urldecode($paramValues[$name]));                } else {                    $this->params[$name] = urldecode($paramValues[$name]);                }            }        }        return true;    }    /**     * Convert a URL parameter (e.g. ":id", ":id+") into a regular expression     * @param  array    URL parameters     * @return string   Regular expression for URL parameter     */    protected function matchesCallback($m)    {        $this->paramNames[] = $m[1];        if (isset($this->conditions[ $m[1] ])) {            return '(?P<' . $m[1] . '>' . $this->conditions[ $m[1] ] . ')';        }        if (substr($m[0], -1) === '+') {            $this->paramNamesPath[ $m[1] ] = 1;            return '(?P<' . $m[1] . '>.+)';        }        return '(?P<' . $m[1] . '>[^/]+)';    }    /**     * Set route name     * @param  string     $name The name of the route     * @return \Slim\Route     */    public function name($name)    {        $this->setName($name);        return $this;    }    /**     * Merge route conditions     * @param  array      $conditions Key-value array of URL parameter conditions     * @return \Slim\Route     */    public function conditions(array $conditions)    {        $this->conditions = array_merge($this->conditions, $conditions);        return $this;    }}
 |