caller.php 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209
  1. <?php namespace Laravel\Routing;
  2. use Closure;
  3. use Laravel\Response;
  4. use Laravel\Container;
  5. class Caller {
  6. /**
  7. * The IoC container instance.
  8. *
  9. * @var Container
  10. */
  11. protected $container;
  12. /**
  13. * The route filters defined for the application.
  14. *
  15. * @var array
  16. */
  17. protected $filters;
  18. /**
  19. * The path to the application's controllers.
  20. *
  21. * @var string
  22. */
  23. protected $path;
  24. /**
  25. * Create a new route caller instance.
  26. *
  27. * @param Container $container
  28. * @param Delegator $delegator
  29. * @param array $filters
  30. * @return void
  31. */
  32. public function __construct(Container $container, $filters, $path)
  33. {
  34. $this->path = $path;
  35. $this->filters = $filters;
  36. $this->container = $container;
  37. }
  38. /**
  39. * Call a given route and return the route's response.
  40. *
  41. * @param Route $route
  42. * @return Response
  43. */
  44. public function call(Route $route)
  45. {
  46. // Since "before" filters can halt the request cycle, we will return any response
  47. // from the before filters. Allowing the filters to halt the request cycle makes
  48. // common tasks like authorization convenient to implement.
  49. if ( ! is_null($response = $this->before($route)))
  50. {
  51. return $this->finish($route, $response);
  52. }
  53. if ( ! is_null($response = $route->call()))
  54. {
  55. // If a route returns a string, it also means the route is delegating the
  56. // handling of the request to a controller method. So, we will pass the
  57. // string to the route delegator, exploding on "::".
  58. if (is_string($response)) $response = $this->delegate($route, explode('::', $response));
  59. return $this->finish($route, $response);
  60. }
  61. // If we get to this point, no response was returned from the filters or the route.
  62. // The 404 response will be returned to the browser instead of a blank screen.
  63. return $this->finish($route, $this->container->resolve('laravel.response')->error('404'));
  64. }
  65. /**
  66. * Run the "before" filters for the route.
  67. *
  68. * If a before filter returns a value, that value will be considered the response to the
  69. * request and the route function / controller will not be used to handle the request.
  70. *
  71. * @param Route $route
  72. * @return mixed
  73. */
  74. protected function before(Route $route)
  75. {
  76. $before = array_merge(array('before'), $route->filters('before'));
  77. return $this->filter($before, array(), true);
  78. }
  79. /**
  80. * Handle the delegation of a route to a controller method.
  81. *
  82. * @param Route $route
  83. * @param array $delegate
  84. * @return mixed
  85. */
  86. protected function delegate(Route $route, $delegate)
  87. {
  88. list($controller, $method) = array($delegate[0], $delegate[1]);
  89. $controller = $this->resolve($controller);
  90. // If the controller doesn't exist or the request is to an invalid method, we will
  91. // return the 404 error response. The "before" method and any method beginning with
  92. // an underscore are not publicly available.
  93. if (is_null($controller) or ($method == 'before' or strncmp($method, '_', 1) === 0))
  94. {
  95. return $this->container->resolve('laravel.response')->error('404');
  96. }
  97. $controller->container = $this->container;
  98. // Again, as was the case with route closures, if the controller "before" method returns
  99. // a response, it will be considered the response to the request and the controller method
  100. // will not be used to handle the request to the application.
  101. $response = $controller->before();
  102. return (is_null($response)) ? call_user_func_array(array($controller, $method), $route->parameters) : $response;
  103. }
  104. /**
  105. * Resolve a controller name to a controller instance.
  106. *
  107. * @param string $controller
  108. * @return Controller
  109. */
  110. protected function resolve($controller)
  111. {
  112. if ( ! $this->load($controller)) return;
  113. // If the controller is registered in the IoC container, we will resolve it out
  114. // of the container. Using constructor injection on controllers via the container
  115. // allows more flexible and testable development of applications.
  116. if ($this->container->registered('controllers.'.$controller))
  117. {
  118. return $this->container->resolve('controllers.'.$controller);
  119. }
  120. // If the controller was not registered in the container, we will instantiate
  121. // an instance of the controller manually. All controllers are suffixed with
  122. // "_Controller" to avoid namespacing. Allowing controllers to exist in the
  123. // global namespace gives the developer a convenient API for using the framework.
  124. $controller = str_replace(' ', '_', ucwords(str_replace('.', ' ', $controller))).'_Controller';
  125. return new $controller;
  126. }
  127. /**
  128. * Load the file for a given controller.
  129. *
  130. * @param string $controller
  131. * @return bool
  132. */
  133. protected function load($controller)
  134. {
  135. if (file_exists($path = $this->path.strtolower(str_replace('.', '/', $controller)).EXT))
  136. {
  137. require $path;
  138. return true;
  139. }
  140. return false;
  141. }
  142. /**
  143. * Finish the route handling for the request.
  144. *
  145. * The route response will be converted to a Response instance and the "after" filters will be run.
  146. *
  147. * @param Route $route
  148. * @param mixed $response
  149. * @return Response
  150. */
  151. protected function finish(Route $route, $response)
  152. {
  153. if ( ! $response instanceof Response) $response = new Response($response);
  154. $this->filter(array_merge($route->filters('after'), array('after')), array($response));
  155. return $response;
  156. }
  157. /**
  158. * Call a filter or set of filters.
  159. *
  160. * @param array $filters
  161. * @param array $parameters
  162. * @param bool $override
  163. * @return mixed
  164. */
  165. protected function filter($filters, $parameters = array(), $override = false)
  166. {
  167. foreach ((array) $filters as $filter)
  168. {
  169. if ( ! isset($this->filters[$filter])) continue;
  170. $response = call_user_func_array($this->filters[$filter], $parameters);
  171. // "Before" filters may override the request cycle. For example, an authentication
  172. // filter may redirect a user to a login view if they are not logged in. Because of
  173. // this, we will return the first filter response if overriding is enabled.
  174. if ( ! is_null($response) and $override) return $response;
  175. }
  176. }
  177. }