router.php 2.8 KB

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