123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457 |
- <?php
- /**
- * Lithium: the most rad php framework
- *
- * @copyright Copyright 2013, Union of RAD (http://union-of-rad.org)
- * @license http://opensource.org/licenses/bsd-license.php The BSD License
- */
- namespace lithium\net\http;
- use lithium\util\Inflector;
- use lithium\net\http\RoutingException;
- /**
- * The two primary responsibilities of the `Router` class are to generate URLs from parameter lists,
- * and to determine the correct set of dispatch parameters for incoming requests.
- *
- * Using `Route` objects, these two operations can be handled in a reciprocally consistent way.
- * For example, if you wanted the `/login` URL to be routed to
- * `myapp\controllers\SessionsController::add()`, you could set up a route like the following in
- * `config/routes.php`:
- *
- * {{{
- * use lithium\net\http\Router;
- *
- * Router::connect('/login', array('controller' => 'Sessions', 'action' => 'add'));
- *
- * // -- or --
- *
- * Router::connect('/login', 'Sessions::add');
- * }}}
- *
- * Not only would that correctly route all requests for `/login` to `SessionsController::add()`, but
- * any time the framework generated a route with matching parameters, `Router` would return the
- * correct short URL.
- *
- * While most framework components that work with URLs (and utilize routing) handle calling the
- * `Router` directly (i.e. controllers doing redirects, or helpers generating links), if you have a
- * scenario where you need to call the `Router` directly, you can use the `match()` method.
- *
- * This allows you to keep your application's URL structure nicely decoupled from the underlying
- * software design. For more information on parsing and generating URLs, see the `parse()` and
- * `match()` methods.
- */
- class Router extends \lithium\core\StaticObject {
- /**
- * An array of loaded `Route` objects used to match Request objects against.
- *
- * @see lithium\net\http\Route
- * @var array
- */
- protected static $_configurations = array();
- /**
- * An array of named closures matching up to corresponding route parameter values. Used to
- * format those values.
- *
- * @see lithium\net\http\Router::formatters()
- * @var array
- */
- protected static $_formatters = array();
- /**
- * Classes used by `Router`.
- *
- * @var array
- */
- protected static $_classes = array(
- 'route' => 'lithium\net\http\Route'
- );
- /**
- * Flag for generating Unicode-capable routes. Turn this off if you don't need it, or if you're
- * using a broken OS distribution (i.e. CentOS).
- */
- protected static $_unicode = true;
- /**
- * Modify `Router` configuration settings and dependencies.
- *
- * @param array $config Optional array to override configuration. Acceptable keys are
- * `'classes'` and `'unicode'`.
- * @return array Returns the current configuration settings.
- */
- public static function config($config = array()) {
- if (!$config) {
- return array('classes' => static::$_classes, 'unicode' => static::$_unicode);
- }
- if (isset($config['classes'])) {
- static::$_classes = $config['classes'] + static::$_classes;
- }
- if (isset($config['unicode'])) {
- static::$_unicode = $config['unicode'];
- }
- }
- /**
- * Connects a new route and returns the current routes array. This method creates a new
- * `Route` object and registers it with the `Router`. The order in which routes are connected
- * matters, since the order of precedence is taken into account in parsing and matching
- * operations.
- *
- * @see lithium\net\http\Route
- * @see lithium\net\http\Router::parse()
- * @see lithium\net\http\Router::match()
- * @param string $template An empty string, or a route string "/"
- * @param array $params An array describing the default or required elements of the route
- * @param array $options
- * @return array Array of routes
- */
- public static function connect($template, $params = array(), $options = array()) {
- if (is_object($template)) {
- return (static::$_configurations[] = $template);
- }
- if (is_string($params)) {
- $params = static::_parseString($params, false);
- }
- if (isset($params[0]) && is_array($tmp = static::_parseString($params[0], false))) {
- unset($params[0]);
- $params = $tmp + $params;
- }
- $params = static::_parseController($params);
- if (is_callable($options)) {
- $options = array('handler' => $options);
- }
- $config = compact('template', 'params') + $options + array(
- 'formatters' => static::formatters(),
- 'unicode' => static::$_unicode
- );
- return (static::$_configurations[] = static::_instance('route', $config));
- }
- /**
- * Wrapper method which takes a `Request` object, parses it through all attached `Route`
- * objects, assigns the resulting parameters to the `Request` object, and returns it.
- *
- * @param object $request A request object, usually an instance of `lithium\action\Request`.
- * @return object Returns a copy of the `Request` object with parameters applied.
- */
- public static function process($request) {
- if (!$result = static::parse($request)) {
- return $request;
- }
- return $result;
- }
- /**
- * Used to get or set an array of named formatter closures, which are used to format route
- * parameters when generating URLs. For example, for controller/action parameters to be dashed
- * instead of underscored or camelBacked, you could do the following:
- *
- * {{{
- * use lithium\util\Inflector;
- *
- * Router::formatters(array(
- * 'controller' => function($value) { return Inflector::slug($value); },
- * 'action' => function($value) { return Inflector::slug($value); }
- * ));
- * }}}
- *
- * _Note_: Because formatters are copied to `Route` objects on an individual basis, make sure
- * you append custom formatters _before_ connecting new routes.
- *
- * @param array $formatters An array of named formatter closures to append to (or overwrite) the
- * existing list.
- * @return array Returns the formatters array.
- */
- public static function formatters(array $formatters = array()) {
- if (!static::$_formatters) {
- static::$_formatters = array(
- 'args' => function($value) {
- return is_array($value) ? join('/', $value) : $value;
- },
- 'controller' => function($value) {
- return Inflector::underscore($value);
- }
- );
- }
- if ($formatters) {
- static::$_formatters = array_filter($formatters + static::$_formatters);
- }
- return static::$_formatters;
- }
- /**
- * Accepts an instance of `lithium\action\Request` (or a subclass) and matches it against each
- * route, in the order that the routes are connected.
- *
- * @see lithium\action\Request
- * @see lithium\net\http\Router::connect()
- * @param object $request A request object containing URL and environment data.
- * @return array Returns an array of parameters specifying how the given request should be
- * routed. The keys returned depend on the `Route` object that was matched, but
- * typically include `'controller'` and `'action'` keys.
- */
- public static function parse($request) {
- $orig = $request->params;
- $url = $request->url;
- foreach (static::$_configurations as $route) {
- if (!$match = $route->parse($request, compact('url'))) {
- continue;
- }
- $request = $match;
- if ($route->canContinue() && isset($request->params['args'])) {
- $url = '/' . join('/', $request->params['args']);
- unset($request->params['args']);
- continue;
- }
- return $request;
- }
- $request->params = $orig;
- }
- /**
- * Attempts to match an array of route parameters (i.e. `'controller'`, `'action'`, etc.)
- * against a connected `Route` object. For example, given the following route:
- *
- * {{{
- * Router::connect('/login', array('controller' => 'users', 'action' => 'login'));
- * }}}
- *
- * This will match:
- * {{{
- * $url = Router::match(array('controller' => 'users', 'action' => 'login'));
- * // returns /login
- * }}}
- *
- * For URLs templates with no insert parameters (i.e. elements like `{:id}` that are replaced
- * with a value), all parameters must match exactly as they appear in the route parameters.
- *
- * Alternatively to using a full array, you can specify routes using a more compact syntax. The
- * above example can be written as:
- *
- * {{{ $url = Router::match('Users::login'); // still returns /login }}}
- *
- * You can combine this with more complicated routes; for example:
- * {{{
- * Router::connect('/posts/{:id:\d+}', array('controller' => 'posts', 'action' => 'view'));
- * }}}
- *
- * This will match:
- * {{{
- * $url = Router::match(array('controller' => 'posts', 'action' => 'view', 'id' => '1138'));
- * // returns /posts/1138
- * }}}
- *
- * Again, you can specify the same URL with a more compact syntax, as in the following:
- * {{{
- * $url = Router::match(array('Posts::view', 'id' => '1138'));
- * // again, returns /posts/1138
- * }}}
- *
- * You can use either syntax anywhere a URL is accepted, i.e.
- * `lithium\action\Controller::redirect()`, or `lithium\template\helper\Html::link()`.
- *
- * @param string|array $url Options to match to a URL. Optionally, this can be a string
- * containing a manually generated URL.
- * @param object $context An instance of `lithium\action\Request`. This supplies the context for
- * any persistent parameters, as well as the base URL for the application.
- * @param array $options Options for the generation of the matched URL. Currently accepted
- * values are:
- * - `'absolute'` _boolean_: Indicates whether or not the returned URL should be an
- * absolute path (i.e. including scheme and host name).
- * - `'host'` _string_: If `'absolute'` is `true`, sets the host name to be used,
- * or overrides the one provided in `$context`.
- * - `'scheme'` _string_: If `'absolute'` is `true`, sets the URL scheme to be
- * used, or overrides the one provided in `$context`.
- * @return string Returns a generated URL, based on the URL template of the matched route, and
- * prefixed with the base URL of the application.
- */
- public static function match($url = array(), $context = null, array $options = array()) {
- if (is_string($url = static::_prepareParams($url, $context, $options))) {
- return $url;
- }
- $defaults = array('action' => 'index');
- $url += $defaults;
- $stack = array();
- $base = isset($context) ? $context->env('base') : '';
- $suffix = isset($url['#']) ? "#{$url['#']}" : null;
- unset($url['#']);
- foreach (static::$_configurations as $route) {
- if (!$match = $route->match($url, $context)) {
- continue;
- }
- if ($route->canContinue()) {
- $stack[] = $match;
- $export = $route->export();
- $keys = $export['match'] + $export['keys'] + $export['defaults'];
- unset($keys['args']);
- $url = array_diff_key($url, $keys);
- continue;
- }
- if ($stack) {
- $stack[] = $match;
- $match = static::_compileStack($stack);
- }
- $path = rtrim("{$base}{$match}{$suffix}", '/') ?: '/';
- $path = ($options) ? static::_prefix($path, $context, $options) : $path;
- return $path ?: '/';
- }
- $url = static::_formatError($url);
- throw new RoutingException("No parameter match found for URL `{$url}`.");
- }
- protected static function _compileStack($stack) {
- $result = null;
- foreach (array_reverse($stack) as $fragment) {
- if ($result) {
- $result = str_replace('{:args}', ltrim($result, '/'), $fragment);
- continue;
- }
- $result = $fragment;
- }
- return $result;
- }
- protected static function _formatError($url) {
- $match = array("\n", 'array (', ',)', '=> NULL', '( \'', ', ');
- $replace = array('', '(', ')', '=> null', '(\'', ', ');
- return str_replace($match, $replace, var_export($url, true));
- }
- protected static function _parseController(array $params) {
- if (!isset($params['controller'])) {
- return $params;
- }
- if (strpos($params['controller'], '.')) {
- $separated = explode('.', $params['controller'], 2);
- list($params['library'], $params['controller']) = $separated;
- }
- if (strpos($params['controller'], '\\') === false) {
- $params['controller'] = Inflector::camelize($params['controller']);
- }
- return $params;
- }
- protected static function _prepareParams($url, $context, array $options) {
- if (is_string($url)) {
- if (strpos($url, '://')) {
- return $url;
- }
- foreach (array('#', '//', 'mailto') as $prefix) {
- if (strpos($url, $prefix) === 0) {
- return $url;
- }
- }
- if (is_string($url = static::_parseString($url, $context))) {
- return static::_prefix($url, $context, $options);
- }
- }
- if (isset($url[0]) && is_array($params = static::_parseString($url[0], $context))) {
- unset($url[0]);
- $url = $params + $url;
- }
- return static::_persist(static::_parseController($url), $context);
- }
- /**
- * Returns the prefix (scheme + hostname) for a URL based on the passed `$options` and the
- * `$context`.
- *
- * @param string $path The URL to be prefixed.
- * @param object $context The request context.
- * @param array $options Options for generating the proper prefix. Currently accepted values
- * are: `'absolute' => true|false`, `'host' => string` and `'scheme' => string`.
- * @return string The prefixed URL, depending on the passed options.
- */
- protected static function _prefix($path, $context = null, array $options = array()) {
- $defaults = array('scheme' => null, 'host' => null, 'absolute' => false);
- if ($context) {
- $defaults['host'] = $context->env('HTTP_HOST');
- $defaults['scheme'] = $context->env('HTTPS') ? 'https://' : 'http://';
- }
- $options += $defaults;
- return ($options['absolute']) ? "{$options['scheme']}{$options['host']}{$path}" : $path;
- }
- /**
- * Copies persistent parameters (parameters in the request which have been designated to
- * persist) to the current URL, unless the parameter has been explicitly disabled from
- * persisting by setting the value in the URL to `null`, or by assigning some other value.
- *
- * For example:
- *
- * {{{ embed:lithium\tests\cases\net\http\RouterTest::testParameterPersistence(1-10) }}}
- *
- * @see lithium\action\Request::$persist
- * @param array $url The parameters that define the URL to be matched.
- * @param object $context Typically an instance of `lithium\action\Request`, which contains a
- * `$persist` property, which is an array of keys to be persisted in URLs between
- * requests.
- * @return array Returns the modified URL array.
- */
- protected static function _persist($url, $context) {
- if (!$context || !isset($context->persist)) {
- return $url;
- }
- foreach ($context->persist as $key) {
- $url += array($key => $context->params[$key]);
- if ($url[$key] === null) {
- unset($url[$key]);
- }
- }
- return $url;
- }
- /**
- * Returns a route from the loaded configurations, by name.
- *
- * @param string $route Name of the route to request.
- * @return lithium\net\http\Route
- */
- public static function get($route = null) {
- if ($route === null) {
- return static::$_configurations;
- }
- return isset(static::$_configurations[$route]) ? static::$_configurations[$route] : null;
- }
- /**
- * Resets the `Router` to its default state, unloading all routes.
- *
- * @return void
- */
- public static function reset() {
- static::$_configurations = array();
- }
- /**
- * Helper function for taking a path string and parsing it into a controller and action array.
- *
- * @param string $path Path string to parse.
- * @param boolean $context
- * @return array
- */
- protected static function _parseString($path, $context) {
- if (!preg_match('/^[A-Za-z0-9_]+::[A-Za-z0-9_]+$/', $path)) {
- $base = $context ? $context->env('base') : '';
- $path = trim($path, '/');
- return $context !== false ? "{$base}/{$path}" : null;
- }
- list($controller, $action) = explode('::', $path, 2);
- return compact('controller', 'action');
- }
- }
- ?>
|