route.php 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340
  1. <?php namespace Laravel\Routing;
  2. use Closure;
  3. use Laravel\Bundle;
  4. use Laravel\Request;
  5. use Laravel\Response;
  6. class Route {
  7. /**
  8. * The URI the route response to.
  9. *
  10. * @var string
  11. */
  12. public $uri;
  13. /**
  14. * The request method the route responds to.
  15. *
  16. * @var string
  17. */
  18. public $method;
  19. /**
  20. * The bundle in which the route was registered.
  21. *
  22. * @var string
  23. */
  24. public $bundle;
  25. /**
  26. * The action that is assigned to the route.
  27. *
  28. * @var mixed
  29. */
  30. public $action;
  31. /**
  32. * The parameters that will passed to the route callback.
  33. *
  34. * @var array
  35. */
  36. public $parameters;
  37. /**
  38. * Create a new Route instance.
  39. *
  40. * @param string $method
  41. * @param string $uri
  42. * @param array $action
  43. * @param array $parameters
  44. * @return void
  45. */
  46. public function __construct($method, $uri, $action, $parameters = array())
  47. {
  48. $this->uri = $uri;
  49. $this->method = $method;
  50. $this->action = $action;
  51. // Determine the bundle in which the route was registered. We will know
  52. // the bundle by using the bundle::handles method, which will return
  53. // the bundle assigned to that URI.
  54. $this->bundle = Bundle::handles($uri);
  55. // We'll set the parameters based on the number of parameters passed
  56. // compared to the parameters that were needed. If more parameters
  57. // are needed, we'll merge in defaults.
  58. $this->parameters($uri, $action, $parameters);
  59. }
  60. /**
  61. * Set the parameters array to the correct value.
  62. *
  63. * @param string $uri
  64. * @param array $action
  65. * @param array $parameters
  66. * @return void
  67. */
  68. protected function parameters($uri, $action, $parameters)
  69. {
  70. $defaults = (array) array_get($action, 'defaults');
  71. // If there are less parameters than wildcards, we will figure out how
  72. // many parameters we need to inject from the array of defaults and
  73. // merge them in into the main array for the route.
  74. if (count($defaults) > count($parameters))
  75. {
  76. $defaults = array_slice($defaults, count($parameters));
  77. $parameters = array_merge($parameters, $defaults);
  78. }
  79. $this->parameters = $parameters;
  80. }
  81. /**
  82. * Call a given route and return the route's response.
  83. *
  84. * @return Response
  85. */
  86. public function call()
  87. {
  88. // The route is responsible for running the global filters, and any
  89. // filters defined on the route itself, since all incoming requests
  90. // come through a route (either defined or ad-hoc).
  91. $response = Filter::run($this->filters('before'), array(), true);
  92. if (is_null($response))
  93. {
  94. $response = $this->response();
  95. }
  96. // We always return a Response instance from the route calls, so
  97. // we'll use the prepare method on the Response class to make
  98. // sure we have a valid Response isntance.
  99. $response = Response::prepare($response);
  100. Filter::run($this->filters('after'), array($response));
  101. return $response;
  102. }
  103. /**
  104. * Execute the route action and return the response.
  105. *
  106. * Unlike the "call" method, none of the attached filters will be run.
  107. *
  108. * @return mixed
  109. */
  110. public function response()
  111. {
  112. // If the action is a string, it is pointing the route to a controller
  113. // action, and we can just call the action and return its response.
  114. // We'll just pass the action off to the Controller class.
  115. $delegate = $this->delegate();
  116. if ( ! is_null($delegate))
  117. {
  118. return Controller::call($delegate, $this->parameters);
  119. }
  120. // If the route does not have a delegate, then it must be a Closure
  121. // instance or have a Closure in its action array, so we will try
  122. // to locate the Closure and call it directly.
  123. $handler = $this->handler();
  124. if ( ! is_null($handler))
  125. {
  126. return call_user_func_array($handler, $this->parameters);
  127. }
  128. }
  129. /**
  130. * Get the filters that are attached to the route for a given event.
  131. *
  132. * @param string $event
  133. * @return array
  134. */
  135. protected function filters($event)
  136. {
  137. $global = Bundle::prefix($this->bundle).$event;
  138. $filters = array_unique(array($event, $global));
  139. // Next we will check to see if there are any filters attached to
  140. // the route for the given event. If there are, we'll merge them
  141. // in with the global filters for the event.
  142. if (isset($this->action[$event]))
  143. {
  144. $assigned = Filter::parse($this->action[$event]);
  145. $filters = array_merge($filters, $assigned);
  146. }
  147. return array(new Filter_Collection($filters));
  148. }
  149. /**
  150. * Get the controller action delegate assigned to the route.
  151. *
  152. * If no delegate is assigned, null will be returned by the method.
  153. *
  154. * @return string
  155. */
  156. protected function delegate()
  157. {
  158. return array_get($this->action, 'uses');
  159. }
  160. /**
  161. * Get the anonymous function assigned to handle the route.
  162. *
  163. * @return Closure
  164. */
  165. protected function handler()
  166. {
  167. return array_first($this->action, function($key, $value)
  168. {
  169. return $value instanceof Closure;
  170. });
  171. }
  172. /**
  173. * Determine if the route has a given name.
  174. *
  175. * <code>
  176. * // Determine if the route is the "login" route
  177. * $login = Request::route()->is('login');
  178. * </code>
  179. *
  180. * @param string $name
  181. * @return bool
  182. */
  183. public function is($name)
  184. {
  185. return array_get($this->action, 'name') === $name;
  186. }
  187. /**
  188. * Register a controller with the router.
  189. *
  190. * @param string|array $controller
  191. * @param string|array $defaults
  192. * @return void
  193. */
  194. public static function controller($controllers, $defaults = 'index')
  195. {
  196. Router::controller($controllers, $defaults);
  197. }
  198. /**
  199. * Register a secure controller with the router.
  200. *
  201. * @param string|array $controllers
  202. * @param string|array $defaults
  203. * @return void
  204. */
  205. public static function secure_controller($controllers, $defaults = 'index')
  206. {
  207. Router::controller($controllers, $defaults, true);
  208. }
  209. /**
  210. * Register a GET route with the router.
  211. *
  212. * @param string|array $route
  213. * @param mixed $action
  214. * @return void
  215. */
  216. public static function get($route, $action)
  217. {
  218. Router::register('GET', $route, $action);
  219. }
  220. /**
  221. * Register a POST route with the router.
  222. *
  223. * @param string|array $route
  224. * @param mixed $action
  225. * @return void
  226. */
  227. public static function post($route, $action)
  228. {
  229. Router::register('POST', $route, $action);
  230. }
  231. /**
  232. * Register a PUT route with the router.
  233. *
  234. * @param string|array $route
  235. * @param mixed $action
  236. * @return void
  237. */
  238. public static function put($route, $action)
  239. {
  240. Router::register('PUT', $route, $action);
  241. }
  242. /**
  243. * Register a DELETE route with the router.
  244. *
  245. * @param string|array $route
  246. * @param mixed $action
  247. * @return void
  248. */
  249. public static function delete($route, $action)
  250. {
  251. Router::register('DELETE', $route, $action);
  252. }
  253. /**
  254. * Register a route that handles any request method.
  255. *
  256. * @param string|array $route
  257. * @param mixed $action
  258. * @return void
  259. */
  260. public static function any($route, $action)
  261. {
  262. Router::register('*', $route, $action);
  263. }
  264. /**
  265. * Register a group of routes that share attributes.
  266. *
  267. * @param array $attributes
  268. * @param Closure $callback
  269. * @return void
  270. */
  271. public static function group($attributes, Closure $callback)
  272. {
  273. Router::group($attributes, $callback);
  274. }
  275. /**
  276. * Register many request URIs to a single action.
  277. *
  278. * @param array $routes
  279. * @param mixed $action
  280. * @return void
  281. */
  282. public static function share($routes, $action)
  283. {
  284. Router::share($routes, $action);
  285. }
  286. /**
  287. * Register a HTTPS route with the router.
  288. *
  289. * @param string $method
  290. * @param string|array $route
  291. * @param mixed $action
  292. * @return void
  293. */
  294. public static function secure($method, $route, $action)
  295. {
  296. Router::secure($method, $route, $action);
  297. }
  298. }