router.php 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  1. <?php namespace System;
  2. class Router {
  3. /**
  4. * All of the loaded routes.
  5. *
  6. * @var array
  7. */
  8. public static $routes;
  9. /**
  10. * Search a set of routes for the route matching a method and URI.
  11. *
  12. * @param string $method
  13. * @param string $uri
  14. * @return Route
  15. */
  16. public static function route($method, $uri)
  17. {
  18. // Prepend a forward slash since all routes begin with one.
  19. $uri = ($uri != '/') ? '/'.$uri : $uri;
  20. if (is_null(static::$routes))
  21. {
  22. static::$routes = ( ! is_dir(APP_PATH.'routes')) ? require APP_PATH.'routes'.EXT : static::load($uri);
  23. }
  24. // Is there an exact match for the request?
  25. if (isset(static::$routes[$method.' '.$uri]))
  26. {
  27. return Request::$route = new Route($method.' '.$uri, static::$routes[$method.' '.$uri]);
  28. }
  29. foreach (static::$routes as $keys => $callback)
  30. {
  31. // Only check routes that have multiple URIs or wildcards. All other routes would
  32. // have been caught by a literal match.
  33. if (strpos($keys, '(') !== false or strpos($keys, ',') !== false )
  34. {
  35. foreach (explode(', ', $keys) as $key)
  36. {
  37. $key = str_replace(':num', '[0-9]+', str_replace(':any', '[a-zA-Z0-9\-_]+', $key));
  38. if (preg_match('#^'.$key.'$#', $method.' '.$uri))
  39. {
  40. // Remove the leading slashes from the route and request URIs. Also trim
  41. // the request method off of the route URI. This should get the request
  42. // and route URIs in the same format so we can extract the parameters.
  43. $uri = trim($uri, '/');
  44. $key = trim(substr($key, strlen($method.' ')), '/');
  45. return Request::$route = new Route($keys, $callback, static::parameters(explode('/', $uri), explode('/', $key)));
  46. }
  47. }
  48. }
  49. }
  50. }
  51. /**
  52. * Load the appropriate route file for the request URI.
  53. *
  54. * @param string $uri
  55. * @return array
  56. */
  57. public static function load($uri)
  58. {
  59. if ( ! file_exists(APP_PATH.'routes/home'.EXT))
  60. {
  61. throw new \Exception("A [home] route file is required when using a route directory.");
  62. }
  63. if ($uri == '/')
  64. {
  65. return require APP_PATH.'routes/home'.EXT;
  66. }
  67. else
  68. {
  69. $segments = explode('/', trim($uri, '/'));
  70. if ( ! file_exists(APP_PATH.'routes/'.$segments[0].EXT))
  71. {
  72. return require APP_PATH.'routes/home'.EXT;
  73. }
  74. return array_merge(require APP_PATH.'routes/'.$segments[0].EXT, require APP_PATH.'routes/home'.EXT);
  75. }
  76. }
  77. /**
  78. * Extract the parameters from a URI based on a route URI.
  79. *
  80. * Any route segment wrapped in parentheses is considered a parameter.
  81. *
  82. * @param array $uri
  83. * @param array $route
  84. * @return array
  85. */
  86. private static function parameters($uri, $route)
  87. {
  88. $parameters = array();
  89. for ($i = 0; $i < count($route); $i++)
  90. {
  91. if (strpos($route[$i], '(') === 0)
  92. {
  93. $parameters[] = $uri[$i];
  94. }
  95. }
  96. return $parameters;
  97. }
  98. }