router.php 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176
  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. * The named routes that have been found so far.
  11. *
  12. * @var array
  13. */
  14. public static $names = array();
  15. /**
  16. * Search a set of routes for the route matching a method and URI.
  17. *
  18. * @param string $method
  19. * @param string $uri
  20. * @return Route
  21. */
  22. public static function route($method, $uri)
  23. {
  24. // --------------------------------------------------------------
  25. // Force the URI to have a forward slash.
  26. // --------------------------------------------------------------
  27. $uri = ($uri != '/') ? '/'.$uri : $uri;
  28. // --------------------------------------------------------------
  29. // Load the application routes.
  30. // --------------------------------------------------------------
  31. if (is_null(static::$routes))
  32. {
  33. static::$routes = static::routes($uri);
  34. }
  35. // --------------------------------------------------------------
  36. // Is there an exact match for the request?
  37. // --------------------------------------------------------------
  38. if (isset(static::$routes[$method.' '.$uri]))
  39. {
  40. return new Route(static::$routes[$method.' '.$uri]);
  41. }
  42. // --------------------------------------------------------------
  43. // No exact match... check each route individually.
  44. // --------------------------------------------------------------
  45. foreach (static::$routes as $keys => $callback)
  46. {
  47. // --------------------------------------------------------------
  48. // Only check routes that have multiple URIs or wildcards.
  49. // All other routes would have been caught by a literal match.
  50. // --------------------------------------------------------------
  51. if (strpos($keys, '(') !== false or strpos($keys, ',') !== false )
  52. {
  53. foreach (explode(', ', $keys) as $route)
  54. {
  55. $route = str_replace(':num', '[0-9]+', str_replace(':any', '.+', $route));
  56. if (preg_match('#^'.$route.'$#', $method.' '.$uri))
  57. {
  58. return new Route($callback, static::parameters(explode('/', $uri), explode('/', $route)));
  59. }
  60. }
  61. }
  62. }
  63. }
  64. /**
  65. * Find a route by name.
  66. *
  67. * @param string $name
  68. * @return array
  69. */
  70. public static function find($name)
  71. {
  72. // --------------------------------------------------------------
  73. // Have we already located this route by name?
  74. // --------------------------------------------------------------
  75. if (array_key_exists($name, static::$names))
  76. {
  77. return static::$names[$name];
  78. }
  79. $arrayIterator = new \RecursiveArrayIterator(static::$routes);
  80. $recursiveIterator = new \RecursiveIteratorIterator($arrayIterator);
  81. // --------------------------------------------------------------
  82. // Find the named route.
  83. // --------------------------------------------------------------
  84. foreach ($recursiveIterator as $iterator)
  85. {
  86. $route = $recursiveIterator->getSubIterator();
  87. if ($route['name'] == $name)
  88. {
  89. return static::$names[$name] = array($arrayIterator->key() => iterator_to_array($route));
  90. }
  91. }
  92. }
  93. /**
  94. * Get the parameters that should be passed to the route callback.
  95. *
  96. * @param array $uri_segments
  97. * @param array $route_segments
  98. * @return array
  99. */
  100. private static function parameters($uri_segments, $route_segments)
  101. {
  102. $parameters = array();
  103. for ($i = 0; $i < count($route_segments); $i++)
  104. {
  105. // --------------------------------------------------------------
  106. // Any segment wrapped in parentheses is a parameter.
  107. // --------------------------------------------------------------
  108. if (strpos($route_segments[$i], '(') === 0)
  109. {
  110. $parameters[] = $uri_segments[$i];
  111. }
  112. }
  113. return $parameters;
  114. }
  115. /**
  116. * Load the routes based on the request URI.
  117. *
  118. * @param string $uri
  119. * @return void
  120. */
  121. private static function routes($uri)
  122. {
  123. // --------------------------------------------------------------
  124. // If a route directory is being used, load the route file
  125. // corresponding to the first segment of the URI.
  126. // --------------------------------------------------------------
  127. if (is_dir(APP_PATH.'routes'))
  128. {
  129. if ($uri == '/')
  130. {
  131. if ( ! file_exists(APP_PATH.'routes/home'.EXT))
  132. {
  133. throw new \Exception("A [home] route file is required when using a route directory.");
  134. }
  135. return require APP_PATH.'routes/home'.EXT;
  136. }
  137. else
  138. {
  139. $segments = explode('/', trim($uri, '/'));
  140. if ( ! file_exists(APP_PATH.'routes/'.$segments[0].EXT))
  141. {
  142. throw new \Exception("No route file defined for routes beginning with [".$segments[0]."]");
  143. }
  144. return require APP_PATH.'routes/'.$segments[0].EXT;
  145. }
  146. }
  147. // --------------------------------------------------------------
  148. // If no route directory is being used, we can simply load the
  149. // routes file from the application directory.
  150. // --------------------------------------------------------------
  151. else
  152. {
  153. return require APP_PATH.'routes'.EXT;
  154. }
  155. }
  156. }