在`Laravel`中,每一个请求都会被封装为一个`Request`对象,`Form Request`对象就是包含了额外验证逻辑(以及访问权限控制)的自定义`Request`类。 本文分析了FormRequest异常处理流程并提出了自定义处理FormRequest验证失败的思路。
所有示例基于Laravel 5.1.39 (LTS)
今天天气不错,我们来说说表单验证。
Controller中做表单验证
有的同学把表单验证逻辑写在Controller中,例如这个对用户提交评论内容的验证:
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 | <?php// ... useValidator;classCommentController{  {    $validator= Validator::make($request->all(), [      'comment'=> 'required', // 只是实例,就写个简单的规则,你的网站要是这么写欢迎在评论里贴网址    ]);    if($validator->fails()) {      returnredirect()        ->back()        ->withErrors($validator)        ->withInput();    }  } | 
这样写的话,表单验证和业务逻辑挤在一起,我们的Controller中就会有太多的代码,而且重复的验证规则基本也是复制粘贴。
我们可以利用Form Request来封装表单验证代码,从而精简Controller中的代码逻辑,使其专注于业务。而独立出去的表单验证逻辑甚至可以复用到其它请求中,例如修改评论。
什么是Form Request
在Laravel中,每一个请求都会被封装为一个Request对象,Form Request对象就是包含了额外验证逻辑(以及访问权限控制)的自定义Request类。
如何使用Form Request做表单验证
Laravel提供了生成Form Request的Artisan命令:
<code>$ php artisan make:request StoreCommentRequest</code>
于是就生成了app/Http/Requests/StoreCommentRequest.php,让我们来分析一下内容:
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 | <?phpnamespaceAppHttpRequests;useAppHttpRequestsRequest; // 可以看到,这个基类是在我们的项目中的,这意味着我们可以修改它classStoreCommentRequest extendsRequest{  /**   * Determine if the user is authorized to make this request.   *   * @return bool   */  publicfunctionauthorize() // 这个方法可以用来控制访问权限,例如禁止未付费用户评论…  {    returnfalse; // 注意!这里默认是false,记得改成true  }  /**   * Get the validation rules that apply to the request.   *   * @return array   */  publicfunctionrules() // 这个方法返回验证规则数组,也就是Validator的验证规则  {    return[      //    ];  }} | 
那么很容易,我们除了让authorize方法返回true之外,还得让rules方法返回我们的验证规则:
| 1 2 3 4 5 6 7 8 | <?php// ...  publicfunctionrules()  {    return[    ];  }// ... | 
接着修改我们的Controller:
| 1 2 3 4 5 6 7 8 | <?php// ...  // 之前:public function postStoreComment(Request $request)  publicfunctionpostStoreComment(AppHttpRequestsStoreCommentRequest $request)  {    // ...  }// ... | 
这样Laravel便会自动调用StoreCommentRequest进行表单验证了。
异常处理
如果表单验证失败,Laravel会重定向到之前的页面,并且将错误写到Session中,如果是AJAX请求,则会返回一段HTTP状态为422的JSON数据,类似这样:
<code>{comment: [“The comment field is required.”]}</code>
这里就不细说提示信息怎么修改了,如果有人想看相关教程,可以留言。
我们主要来说说怎么定制错误处理。
通常来说,Laravel中的错误都是异常(Exception),我们都可以在appExceptionshandler.php中进行统一处理。Form Request确实也抛出了一个IlluminateHttpExceptionHttpResponseException异常,但这个异常是在路由逻辑中就被特殊处理了。
首先我们来看看Form Request是如何被执行的:
IlluminateValidationValidationServiceProvider:
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 | <?phpnamespaceIlluminateValidation;useIlluminateSupportServiceProvider;useIlluminateContractsValidationValidatesWhenResolved;classValidationServiceProvider extendsServiceProvider{  /**   * Register the service provider.   *   * @return void   */  publicfunctionregister()  {    $this->registerValidationResolverHook(); // 看我看我看我    $this->registerPresenceVerifier();    $this->registerValidationFactory();  }  /**   * Register the "ValidatesWhenResolved" container hook.   *   * @return void   */  {    // 这里可以看到对`ValidatesWhenResolved`的实现做了一个监听    $this->app->afterResolving(function(ValidatesWhenResolved $resolved) {      $resolved->validate(); // 然后调用了它的`validate`方法进行验证    });  }// ... | 
你猜对了,Form Request就实现了这个IlluminateContractsValidationValidatesWhenResolved接口:
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 | <?php namespaceIlluminateFoundationHttp;useIlluminateHttpRequest;useIlluminateHttpResponse;useIlluminateHttpJsonResponse;useIlluminateRoutingRedirector;useIlluminateContainerContainer;useIlluminateContractsValidationValidator;useIlluminateHttpExceptionHttpResponseException;useIlluminateValidationValidatesWhenResolvedTrait;useIlluminateContractsValidationValidatesWhenResolved; // 是你useIlluminateContractsValidationFactory asValidationFactory;// 我们`appHttpRequestsRequest`便是继承于这个`FormRequest`类classFormRequest extendsRequest implementsValidatesWhenResolved // 就是你{  useValidatesWhenResolvedTrait; // 这个我们待会儿也要看看  // ... | 
FormRequest基类中的validate方法是由这个IlluminateValidationValidatesWhenResolvedTrait实现的:
IlluminateValidationValidatesWhenResolvedTrait:
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 | <?phpnamespaceIlluminateValidation;useIlluminateContractsValidationValidationException;useIlluminateContractsValidationUnauthorizedException;/** * Provides default implementation of ValidatesWhenResolved contract. */trait ValidatesWhenResolvedTrait{  /**   * Validate the class instance.   *   * @return void   */  publicfunctionvalidate() // 这里实现了`validate`方法  {    $instance= $this->getValidatorInstance(); // 这里获取了`Validator`实例    if(! $this->passesAuthorization()) {      $this->failedAuthorization(); // 这是调用了访问授权的失败处理    } elseif(! $instance->passes()) {      $this->failedValidation($instance); // 这里调用了验证失败的处理,我们主要看这里    }  }  // ... | 
在validate里,如果验证失败了就会调用$this->failedValidation(),继续:
IlluminateFoundationHttpFormRequest:
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 | <?php// ...  /**   * Handle a failed validation attempt.   *   * @param IlluminateContractsValidationValidator $validator   * @return mixed   */  protectedfunctionfailedValidation(Validator $validator)  {    thrownewHttpResponseException($this->response( // 这里抛出了传说中的异常      $this->formatErrors($validator)    ));  } | 
终于看到异常了!可是这个异常在另一个地方被处理了:
IlluminateRoutingRoute:
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 | <?php  // ...  /**   * Run the route action and return the response.   *   * @param IlluminateHttpRequest $request   * @return mixed   */  publicfunctionrun(Request $request)  {    $this->container = $this->container ?: newContainer;    try{      if(! is_string($this->action['uses'])) {        return$this->runCallable($request);      }      if($this->customDispatcherIsBound()) {        return$this->runWithCustomDispatcher($request);      }      return$this->runController($request);    } catch(HttpResponseException $e) { // 就是这里    }  }  // ... | 
至此,整个思路已然清晰,不过我们还是看看这里生成的HttpResponseException异常中的Response是怎么生成的:
IlluminateFoundationHttpFormRequest:
| 1 2 3 4 5 6 7 8 9 10 | <?php// ...  // 132行:    returnnewJsonResponse($errors, 422);  }  return$this->redirector->to($this->getRedirectUrl()) // 对普通表单提交的处理                  ->withInput($this->except($this->dontFlash))                  ->withErrors($errors, $this->errorBag);// ... | 
如何实现自定义错误处理,这里提供两个思路,都需要重写appHttpRequestsRequest的failedValidation:
抛出一个新异常,继承HttpResponseException异常,重新实现getResponse方法,这个异常类我们可以放到app/Exceptions/下便于管理,错误返回依然交给Laravel;
抛出一个我们自定义的异常,在appExceptionshandler中处理。
具体实现这里就不写啦(参阅Laravel文档中关于错误处理部分,中文文档传送门),如果你有别的方法或者想法可以在评论中和我交流。
补充
如果你的Controller使用IlluminateFoundationValidationValidatesRequests这个Trait的validate方法进行验证,同样的,这里验证失败也会抛出IlluminateHttpExceptionHttpResponseException异常,可以参考上面的解决方案进行处理。
联系信息:邮箱aoxolcom@163.com或见网站底部。


















请登录后发表评论
注册
社交帐号登录