欢迎来到福编程网,本站提供各种互联网专业知识!

Laravel中使用FormRequest进行表单验证方法及问题汇总

发布时间:2016-06-19 作者:该叶无法找到 来源:转载
Laravel5.0带来了FormRequests,这是一种特殊的类型,用于在提交表单时进行数据的检查和验证.每个FormRequest类至少包含一个rules()方法,这个方法返回一组验证规则.除此之外还必须包含一个authorize()方法,该方法返回一个布尔值,代表是否允许用户执行本次请求.

在`Laravel`中,每一个请求都会被封装为一个`Request`对象,`Form Request`对象就是包含了额外验证逻辑(以及访问权限控制)的自定义`Request`类。 本文分析了FormRequest异常的处理流程并提出了自定义处理FormRequest验证失败的思路。

所有示例基于Laravel 5.1.39 (LTS)

今天天气不错,我们来说说表单验证。

Controller中做表单验证

有的同学把表单验证逻辑写在Controller中,例如这个对用户提交评论内容的验证:

代码
  1. <?php
  2. // ...
  3. use Validator;
  4. class CommentController
  5. {
  6. public function postStoreComment(Request $request)
  7. {
  8. $validator = Validator::make($request->all(), [
  9. 'comment' => 'required', // 只是实例,就写个简单的规则,你的网站要是这么写欢迎在评论里贴网址
  10. ]);
  11. if ($validator->fails()) {
  12. return redirect()
  13. ->back()
  14. ->withErrors($validator)
  15. ->withInput();
  16. }
  17. }

这样写的话,表单验证和业务逻辑挤在一起,我们的Controller中就会有太多的代码,而且重复的验证规则基本也是复制粘贴。

我们可以利用Form Request来封装表单验证代码,从而精简Controller中的代码逻辑,使其专注于业务。而独立出去的表单验证逻辑甚至可以复用到其它请求中,例如修改评论。

什么是Form Request

在Laravel中,每一个请求都会被封装为一个Request对象,Form Request对象就是包含了额外验证逻辑(以及访问权限控制)的自定义Request类。

如何使用Form Request做表单验证

Laravel提供了生成Form Request的Artisan命令:

$ php artisan make:request StoreCommentRequest

于是就生成了app/Http/Requests/StoreCommentRequest.php,让我们来分析一下内容:

代码
  1. <?php
  2. namespace AppHttpRequests;
  3. use AppHttpRequestsRequest; // 可以看到,这个基类是在我们的项目中的,这意味着我们可以修改它
  4. class StoreCommentRequest extends Request
  5. {
  6. /**
  7. * Determine if the user is authorized to make this request.
  8. *
  9. * @return bool
  10. */
  11. public function authorize() // 这个方法可以用来控制访问权限,例如禁止未付费用户评论…
  12. {
  13. return false; // 注意!这里默认是false,记得改成true
  14. }
  15. /**
  16. * Get the validation rules that apply to the request.
  17. *
  18. * @return array
  19. */
  20. public function rules() // 这个方法返回验证规则数组,也就是Validator的验证规则
  21. {
  22. return [
  23. //
  24. ];
  25. }
  26. }

那么很容易,我们除了让authorize方法返回true之外,还得让rules方法返回我们的验证规则:

代码
  1. <?php
  2. // ...
  3. public function rules()
  4. {
  5. return [
  6. ];
  7. }
  8. // ...

接着修改我们的Controller:

代码
  1. <?php
  2. // ...
  3. // 之前:public function postStoreComment(Request $request)
  4. public function postStoreComment(AppHttpRequestsStoreCommentRequest $request)
  5. {
  6. // ...
  7. }
  8. // ...

这样Laravel便会自动调用StoreCommentRequest进行表单验证了。

异常处理

如果表单验证失败,Laravel会重定向到之前的页面,并且将错误写到Session中,如果是AJAX请求,则会返回一段HTTP状态为422的JSON数据,类似这样:

{comment: ["The comment field is required."]}

这里就不细说提示信息怎么修改了,如果有人想看相关教程,可以留言。

我们主要来说说怎么定制错误处理。

通常来说,Laravel中的错误都是异常(Exception),我们都可以在appExceptionshandler.php中进行统一处理。Form Request确实也抛出了一个IlluminateHttpExceptionHttpResponseException异常,但这个异常是在路由逻辑中就被特殊处理了。

首先我们来看看Form Request是如何被执行的:

IlluminateValidationValidationServiceProvider:

代码
  1. <?php
  2. namespace IlluminateValidation;
  3. use IlluminateSupportServiceProvider;
  4. use IlluminateContractsValidationValidatesWhenResolved;
  5. class ValidationServiceProvider extends ServiceProvider
  6. {
  7. /**
  8. * Register the service provider.
  9. *
  10. * @return void
  11. */
  12. public function register()
  13. {
  14. $this->registerValidationResolverHook(); // 看我看我看我
  15. $this->registerPresenceVerifier();
  16. $this->registerValidationFactory();
  17. }
  18. /**
  19. * Register the "ValidatesWhenResolved" container hook.
  20. *
  21. * @return void
  22. */
  23. protected function registerValidationResolverHook() // 对,就是我
  24. {
  25. // 这里可以看到对`ValidatesWhenResolved`的实现做了一个监听
  26. $this->app->afterResolving(function (ValidatesWhenResolved $resolved) {
  27. $resolved->validate(); // 然后调用了它的`validate`方法进行验证
  28. });
  29. }
  30. // ...

你猜对了,Form Request就实现了这个IlluminateContractsValidationValidatesWhenResolved接口:

代码
  1. <?php
  2. namespace IlluminateFoundationHttp;
  3. use IlluminateHttpRequest;
  4. use IlluminateHttpResponse;
  5. use IlluminateHttpJsonResponse;
  6. use IlluminateRoutingRedirector;
  7. use IlluminateContainerContainer;
  8. use IlluminateContractsValidationValidator;
  9. use IlluminateHttpExceptionHttpResponseException;
  10. use IlluminateValidationValidatesWhenResolvedTrait;
  11. use IlluminateContractsValidationValidatesWhenResolved; // 是你
  12. use IlluminateContractsValidationFactory as ValidationFactory;
  13. // 我们`appHttpRequestsRequest`便是继承于这个`FormRequest`类
  14. class FormRequest extends Request implements ValidatesWhenResolved // 就是你
  15. {
  16. use ValidatesWhenResolvedTrait; // 这个我们待会儿也要看看
  17. // ...

FormRequest基类中的validate方法是由这个IlluminateValidationValidatesWhenResolvedTrait实现的:

IlluminateValidationValidatesWhenResolvedTrait:

代码
  1. <?php
  2. namespace IlluminateValidation;
  3. use IlluminateContractsValidationValidationException;
  4. use IlluminateContractsValidationUnauthorizedException;
  5. /**
  6. * Provides default implementation of ValidatesWhenResolved contract.
  7. */
  8. trait ValidatesWhenResolvedTrait
  9. {
  10. /**
  11. * Validate the class instance.
  12. *
  13. * @return void
  14. */
  15. public function validate() // 这里实现了`validate`方法
  16. {
  17. $instance = $this->getValidatorInstance(); // 这里获取了`Validator`实例
  18. if (! $this->passesAuthorization()) {
  19. $this->failedAuthorization(); // 这是调用了访问授权的失败处理
  20. } elseif (! $instance->passes()) {
  21. $this->failedValidation($instance); // 这里调用了验证失败的处理,我们主要看这里
  22. }
  23. }
  24. // ...

在validate里,如果验证失败了就会调用$this->failedValidation(),继续:

IlluminateFoundationHttpFormRequest:

代码
  1. <?php
  2. // ...
  3. /**
  4. * Handle a failed validation attempt.
  5. *
  6. * @param IlluminateContractsValidationValidator $validator
  7. * @return mixed
  8. */
  9. protected function failedValidation(Validator $validator)
  10. {
  11. throw new HttpResponseException($this->response( // 这里抛出了传说中的异常
  12. $this->formatErrors($validator)
  13. ));
  14. }

终于看到异常了!可是这个异常在另一个地方被处理了:

IlluminateRoutingRoute:

代码
  1. <?php
  2. // ...
  3. /**
  4. * Run the route action and return the response.
  5. *
  6. * @param IlluminateHttpRequest $request
  7. * @return mixed
  8. */
  9. public function run(Request $request)
  10. {
  11. $this->container = $this->container ?: new Container;
  12. try {
  13. if (! is_string($this->action['uses'])) {
  14. return $this->runCallable($request);
  15. }
  16. if ($this->customDispatcherIsBound()) {
  17. return $this->runWithCustomDispatcher($request);
  18. }
  19. return $this->runController($request);
  20. } catch (HttpResponseException $e) { // 就是这里
  21. return $e->getResponse(); // 这里直接返回了Response给客户端
  22. }
  23. }
  24. // ...

至此,整个思路已然清晰,不过我们还是看看这里生成的HttpResponseException异常中的Response是怎么生成的:

IlluminateFoundationHttpFormRequest:

代码
  1. <?php
  2. // ...
  3. // 132行:
  4. if ($this->ajax() || $this->wantsJson()) { // 对AJAX请求的处理
  5. return new JsonResponse($errors, 422);
  6. }
  7. return $this->redirector->to($this->getRedirectUrl()) // 对普通表单提交的处理
  8. ->withInput($this->except($this->dontFlash))
  9. ->withErrors($errors, $this->errorBag);
  10. // ...

相信你都看明白了。

如何实现自定义错误处理,这里提供两个思路,都需要重写appHttpRequestsRequest的failedValidation:

抛出一个新异常,继承HttpResponseException异常,重新实现getResponse方法,这个异常类我们可以放到app/Exceptions/下便于管理,错误返回依然交给Laravel;

抛出一个我们自定义的异常,在appExceptionshandler中处理。

具体实现这里就不写啦(参阅Laravel文档中关于错误处理部分,中文文档传送门),如果你有别的方法或者想法可以在评论中和我交流。

补充

如果你的Controller使用IlluminateFoundationValidationValidatesRequests这个Trait的validate方法进行验证,同样的,这里验证失败也会抛出IlluminateHttpExceptionHttpResponseException异常,可以参考上面的解决方案进行处理。

相关推荐

返回顶部