router.php 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142
  1. <?php namespace System;
  2. class Router {
  3. /**
  4. * The request method and URI.
  5. *
  6. * @var string
  7. */
  8. public $request;
  9. /**
  10. * All of the loaded routes.
  11. *
  12. * @var array
  13. */
  14. public $routes;
  15. /**
  16. * Create a new router for a request method and URI.
  17. *
  18. * @param string $method
  19. * @param string $uri
  20. * @param array $routes
  21. * @return void
  22. */
  23. public function __construct($method, $uri, $routes = null)
  24. {
  25. // Put the request method and URI in route form. Routes begin with
  26. // the request method and a forward slash.
  27. $this->request = $method.' /'.trim($uri, '/');
  28. $this->routes = (is_array($routes)) ? $routes : $this->load($uri);
  29. }
  30. /**
  31. * Create a new router for a request method and URI.
  32. *
  33. * @param string $method
  34. * @param string $uri
  35. * @param array $routes
  36. * @return Router
  37. */
  38. public static function make($method, $uri, $routes = null)
  39. {
  40. return new static($method, $uri, $routes);
  41. }
  42. /**
  43. * Load the appropriate routes for the request URI.
  44. *
  45. * @param string $uri
  46. * @return array
  47. */
  48. public function load($uri)
  49. {
  50. $base = require APP_PATH.'routes'.EXT;
  51. if ( ! is_dir(APP_PATH.'routes') or $uri == '')
  52. {
  53. return $base;
  54. }
  55. list($routes, $segments) = array(array(), explode('/', $uri));
  56. foreach (array_reverse($segments, true) as $key => $value)
  57. {
  58. if (file_exists($path = ROUTE_PATH.implode('/', array_slice($segments, 0, $key + 1)).EXT))
  59. {
  60. $routes = require $path;
  61. break;
  62. }
  63. }
  64. return array_merge($routes, $base);
  65. }
  66. /**
  67. * Search a set of routes for the route matching a method and URI.
  68. *
  69. * @return Route
  70. */
  71. public function route()
  72. {
  73. if (isset($this->routes[$this->request]))
  74. {
  75. return Request::$route = new Route($this->request, $this->routes[$this->request]);
  76. }
  77. foreach ($this->routes as $keys => $callback)
  78. {
  79. // Only check routes that have multiple URIs or wildcards.
  80. // Other routes would have been caught by the check for literal matches.
  81. if (strpos($keys, '(') !== false or strpos($keys, ',') !== false )
  82. {
  83. foreach (explode(', ', $keys) as $key)
  84. {
  85. if (preg_match('#^'.$this->translate_wildcards($key).'$#', $this->request))
  86. {
  87. return Request::$route = new Route($keys, $callback, $this->parameters($this->request, $key));
  88. }
  89. }
  90. }
  91. }
  92. }
  93. /**
  94. * Translate route URI wildcards into actual regular expressions.
  95. *
  96. * @param string $key
  97. * @return string
  98. */
  99. private function translate_wildcards($key)
  100. {
  101. $replacements = 0;
  102. // For optional parameters, first translate the wildcards to their
  103. // regex equivalent, sans the ")?" ending. We will add the endings
  104. // back on after we know how many replacements we made.
  105. $key = str_replace(array('/(:num?)', '/(:any?)'), array('(?:/([0-9]+)', '(?:/([a-zA-Z0-9\-_]+)'), $key, $replacements);
  106. $key .= ($replacements > 0) ? str_repeat(')?', $replacements) : '';
  107. return str_replace(array(':num', ':any'), array('[0-9]+', '[a-zA-Z0-9\-_]+'), $key);
  108. }
  109. /**
  110. * Extract the parameters from a URI based on a route URI.
  111. *
  112. * Any route segment wrapped in parentheses is considered a parameter.
  113. *
  114. * @param string $uri
  115. * @param string $route
  116. * @return array
  117. */
  118. private function parameters($uri, $route)
  119. {
  120. return array_values(array_intersect_key(explode('/', $uri), preg_grep('/\(.+\)/', explode('/', $route))));
  121. }
  122. }