filter.php 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223
  1. <?php namespace Laravel\Routing;
  2. use Laravel\Request;
  3. class Filter {
  4. /**
  5. * The route filters for the application.
  6. *
  7. * @var array
  8. */
  9. protected static $filters = array();
  10. /**
  11. * Register an array of route filters.
  12. *
  13. * @param array $filters
  14. * @return void
  15. */
  16. public static function register($filters)
  17. {
  18. static::$filters = array_merge(static::$filters, $filters);
  19. }
  20. /**
  21. * Call a filter or set of filters.
  22. *
  23. * @param array|string $filters
  24. * @param array $parameters
  25. * @param bool $override
  26. * @return mixed
  27. */
  28. public static function run($filters, $parameters = array(), $override = false)
  29. {
  30. foreach (static::parse($filters) as $filter)
  31. {
  32. // Parameters may be passed into routes by specifying the list of
  33. // parameters after a colon. If parameters are present, we will
  34. // merge them into the parameter array that was passed to the
  35. // method and slice the parameters off of the filter string.
  36. if (($colon = strpos($filter, ':')) !== false)
  37. {
  38. $parameters = array_merge($parameters, explode(',', substr($filter, $colon + 1)));
  39. $filter = substr($filter, 0, $colon);
  40. }
  41. if ( ! isset(static::$filters[$filter])) continue;
  42. $response = call_user_func_array(static::$filters[$filter], $parameters);
  43. // "Before" filters may override the request cycle. For example,
  44. // an authentication filter may redirect a user to a login view
  45. // if they are not logged in. Because of this, we will return
  46. // the first filter response if overriding is enabled.
  47. if ( ! is_null($response) and $override) return $response;
  48. }
  49. }
  50. /**
  51. * Parse a string of filters into an array.
  52. *
  53. * @param string|array $filters
  54. * @return array
  55. */
  56. public static function parse($filters)
  57. {
  58. return (is_string($filters)) ? explode('|', $filters) : (array) $filters;
  59. }
  60. }
  61. class Filter_Collection {
  62. /**
  63. * The event being filtered.
  64. *
  65. * @var string
  66. */
  67. public $name;
  68. /**
  69. * The included controller methods.
  70. *
  71. * @var array
  72. */
  73. public $only = array();
  74. /**
  75. * The excluded controller methods.
  76. *
  77. * @var array
  78. */
  79. public $except = array();
  80. /**
  81. * The filters contained by the collection.
  82. *
  83. * @var string|array
  84. */
  85. public $filters = array();
  86. /**
  87. * The HTTP methods for which the filter applies.
  88. *
  89. * @var array
  90. */
  91. public $methods = array();
  92. /**
  93. * Create a new filter collection instance.
  94. *
  95. * @param string $name
  96. * @param string|array $filters
  97. */
  98. public function __construct($name, $filters)
  99. {
  100. $this->name = $name;
  101. $this->filters = Filter::parse($filters);
  102. }
  103. /**
  104. * Determine if this collection's filters apply to a given method.
  105. *
  106. * Methods may be included / excluded using the "only" and "except" methods on the
  107. * filter collection. Also, the "on" method may be used to set certain filters to
  108. * only run when the request uses a given HTTP verb.
  109. *
  110. * @param string $method
  111. * @return bool
  112. */
  113. public function applies($method)
  114. {
  115. if (count($this->only) > 0 and ! in_array($method, $this->only))
  116. {
  117. return false;
  118. }
  119. if (count($this->except) > 0 and in_array($method, $this->except))
  120. {
  121. return false;
  122. }
  123. if (count($this->methods) > 0 and ! in_array(strtolower(Request::method()), $this->methods))
  124. {
  125. return false;
  126. }
  127. return true;
  128. }
  129. /**
  130. * Set the excluded controller methods.
  131. *
  132. * When methods are excluded, the collection's filters will be run for each
  133. * controller method except those explicitly specified via this method.
  134. *
  135. * <code>
  136. * // Specify a filter for all methods except "index"
  137. * $this->filter('before', 'auth')->except('index');
  138. *
  139. * // Specify a filter for all methods except "index" and "home"
  140. * $this->filter('before', 'auth')->except(array('index', 'home'));
  141. * </code>
  142. *
  143. * @param array $methods
  144. * @return Filter_Collection
  145. */
  146. public function except($methods)
  147. {
  148. $this->except = (array) $methods;
  149. return $this;
  150. }
  151. /**
  152. * Set the included controller methods.
  153. *
  154. * This method is the inverse of the "except" methods. The methods specified
  155. * via this method are the only controller methods on which the collection's
  156. * filters will be run.
  157. *
  158. * <code>
  159. * // Specify a filter for only the "index" method
  160. * $this->filter('before', 'auth')->only('index');
  161. *
  162. * // Specify a filter for only the "index" and "home" methods
  163. * $this->filter('before', 'auth')->only(array('index', 'home'));
  164. * </code>
  165. *
  166. * @param array $methods
  167. * @return Filter_Collection
  168. */
  169. public function only($methods)
  170. {
  171. $this->only = (array) $methods;
  172. return $this;
  173. }
  174. /**
  175. * Set the HTTP methods for which the filter applies.
  176. *
  177. * Since some filters, such as the CSRF filter, only make sense in a POST
  178. * request context, this method allows you to limit which HTTP methods
  179. * the filter will apply to.
  180. *
  181. * <code>
  182. * // Specify that a filter only applies on POST requests
  183. * $this->filter('before', 'csrf')->on('post');
  184. *
  185. * // Specify that a filter applies for multiple HTTP request methods
  186. * $this->filter('before', 'csrf')->on(array('post', 'put'));
  187. * </code>
  188. *
  189. * @param array $methods
  190. * @return Filter_Collection
  191. */
  192. public function on($methods)
  193. {
  194. $this->methods = array_map('strtolower', (array) $methods);
  195. return $this;
  196. }
  197. }