router.php 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153
  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. // If a route directory is being used, load the route file
  30. // corresponding to the first segment of the URI.
  31. // --------------------------------------------------------------
  32. if (is_dir(APP_PATH.'routes'))
  33. {
  34. if ($uri == '/')
  35. {
  36. if ( ! file_exists(APP_PATH.'routes/home'.EXT))
  37. {
  38. throw new \Exception("A [home] route file is required when using a route directory.");
  39. }
  40. static::$routes = require APP_PATH.'routes/home'.EXT;
  41. }
  42. else
  43. {
  44. $segments = explode('/', trim($uri, '/'));
  45. if ( ! file_exists(APP_PATH.'routes/'.$segments[0].EXT))
  46. {
  47. throw new \Exception("No route file defined for routes beginning with [".$segments[0]."]");
  48. }
  49. static::$routes = require APP_PATH.'routes/'.$segments[0].EXT;
  50. }
  51. }
  52. // --------------------------------------------------------------
  53. // If no route directory is being used, we can simply load the
  54. // routes file from the application directory.
  55. // --------------------------------------------------------------
  56. else
  57. {
  58. static::$routes = require APP_PATH.'routes'.EXT;
  59. }
  60. // --------------------------------------------------------------
  61. // Is there an exact match for the request?
  62. // --------------------------------------------------------------
  63. if (isset(static::$routes[$method.' '.$uri]))
  64. {
  65. return new Route(static::$routes[$method.' '.$uri]);
  66. }
  67. // --------------------------------------------------------------
  68. // No exact match... check each route individually.
  69. // --------------------------------------------------------------
  70. foreach (static::$routes as $keys => $callback)
  71. {
  72. // --------------------------------------------------------------
  73. // Only check routes that have multiple URIs or wildcards.
  74. // All other routes would have been caught by a literal match.
  75. // --------------------------------------------------------------
  76. if (strpos($keys, '(') !== false or strpos($keys, ',') !== false )
  77. {
  78. foreach (explode(', ', $keys) as $route)
  79. {
  80. $route = str_replace(':num', '[0-9]+', str_replace(':any', '.+', $route));
  81. if (preg_match('#^'.$route.'$#', $method.' '.$uri))
  82. {
  83. return new Route($callback, static::parameters(explode('/', $uri), explode('/', $route)));
  84. }
  85. }
  86. }
  87. }
  88. }
  89. /**
  90. * Find a route by name.
  91. *
  92. * @param string $name
  93. * @return array
  94. */
  95. public static function find($name)
  96. {
  97. if (array_key_exists($name, static::$names))
  98. {
  99. return static::$names[$name];
  100. }
  101. $arrayIterator = new \RecursiveArrayIterator(static::$routes);
  102. $recursiveIterator = new \RecursiveIteratorIterator($arrayIterator);
  103. foreach ($recursiveIterator as $iterator)
  104. {
  105. $route = $recursiveIterator->getSubIterator();
  106. if ($route['name'] == $name)
  107. {
  108. return static::$names[$name] = array($arrayIterator->key() => iterator_to_array($route));
  109. }
  110. }
  111. }
  112. /**
  113. * Get the parameters that should be passed to the route callback.
  114. *
  115. * @param array $uri_segments
  116. * @param array $route_segments
  117. * @return array
  118. */
  119. private static function parameters($uri_segments, $route_segments)
  120. {
  121. $parameters = array();
  122. for ($i = 0; $i < count($route_segments); $i++)
  123. {
  124. // --------------------------------------------------------------
  125. // Any segment wrapped in parentheses is a parameter.
  126. // --------------------------------------------------------------
  127. if (strpos($route_segments[$i], '(') === 0)
  128. {
  129. $parameters[] = $uri_segments[$i];
  130. }
  131. }
  132. return $parameters;
  133. }
  134. }