解析如何进行Laravel表单验证分层设计和验证场景应用

 4190

思考

我发现大多数的Laravel用户会将验证规则写在Controller下,但这会有一个问题,使代码杂乱且无法复用相同的规则。在CRUD为主的项目中,我们会发现好多字段以及它们的验证规则是相同的,我们一般会化身为CV工程师,再复制一份,那么为什么不加一层验证层?

伴随这个问题,然后去看了别的开源项目的代码,发现TP用户的验证会放在一个单独的类中,并且有验证场景来支持他们复用规则,于是我寻找Laravel中有没有对应的解决方案,然后发现了微擎的一个验证器w7/engine-validate,接下里的文章就依赖此扩展来讲如何增加一个验证层

安装

  1. composer require w7/engine-validate

介绍

首先此验证器也是基于illuminate/validation的,关于它的详细说明可以直接查看它的文档,这里就不过多说明了。

使用

首先我们建立一个和控制器相对应的验证层

  1. app
  2. ├── Http
  3. │   ├── Controllers
  4. │   │   └── UserController.php
  5. │   └── Validate
  6. │       └── UserValidate.php

验证器

验证器代码:

  1. class UserValidate extends Validate{
  2.     protected $rule = [
  3.         'user' => 'required|email',
  4.         'pass' => 'required|alpha_num'
  5.     ];
  6.  
  7.     protected $message = [
  8.         'user.required' => '账号不可为空',
  9.         'pass.required' => '密码不可为空',
  10.     ];
  11.  
  12.     protected $customAttributes = [
  13.        'user' => '账号',
  14.        'pass' => '密码',
  15.     ];
  16. }

控制器代码:

  1. class UserController extends BaseController{
  2.     use AuthorizesRequests, DispatchesJobs, ValidatesRequests;
  3.  
  4.     public function login(Request $request)
  5.     {
  6.         $data = UserValidate::make()->check($request->all());
  7.     }
  8. }

这个时候,如果值不符合要求,会抛出一个ValidateException异常,我们可以选择手动捕获,也可以选择在异常捕获类里面做一个全局的处理:

  1. public function render($request, Throwable $e)
  2. {
  3.     if ($e instanceof ValidateException) {
  4.         return response()->json([
  5.             'code'    => -1,
  6.             'message' => $e->getMessage()
  7.         ]);
  8.     }
  9.  
  10.     return parent::render($request, $e);
  11. }

这样我们就不需要再控制器里面管理任何的验证异常了。

此时,控制器中的$data一定是符合我们需求的数据,然后可以紧接着做对应的业务处理

验证场景

这个时候,如果我们的UserController控制器中又添加了一个register方法,在以上两个字段的需求同时增加了name,应该如何处理呢?

我们可以使用类的$scene属性来指定某一场景下需要验证的字段

验证器代码修改如下:

  1. class UserValidate extends Validate{
  2.     protected $rule = [
  3.         'user' => 'required|email',
  4.         'pass' => 'required|alpha_num',
  5.         'name' => 'required|alpha'
  6.     ];
  7.  
  8.     protected $message = [
  9.         'user.required' => '账号不可为空',
  10.         'pass.required' => '密码不可为空',
  11.     ];
  12.  
  13.     protected $customAttributes = [
  14.        'user' => '账号',
  15.        'pass' => '密码',
  16.        'name' => '用户名称',
  17.     ];
  18.  
  19.     protected $scene = [
  20.        'login'    => ['user', 'pass'],
  21.        'register' => ['user', 'pass', 'name']
  22.     ];
  23. }

此时:login场景对应的验证user和pass字段,而register场景对应的验证user,pass,name字段

控制器代码修改如下:

  1. class UserController extends BaseController{
  2.     use AuthorizesRequests, DispatchesJobs, ValidatesRequests;
  3.  
  4.     public function login(Request $request)
  5.     {
  6.         $data = UserValidate::make()->scene('login')->check($request->all());
  7.     }
  8.  
  9.     public function register(Request $request)
  10.     {
  11.         $data = UserValidate::make()->scene('register')->check($request->all());
  12.     }
  13. }

使用验证器的scene方法来指定当前需要验证的场景名称


验证中间件

默认我们一个控制器对应一个验证器,一个方法对应一个场景名称,基于此特点,我们可以编写一些更为简易的方法来解决验证,我们可以编写一个中间件来解决此问题,点击查看文档说明,中间件完整命名空间为:Itwmw\Validate\Middleware\Laravel\ValidateMiddleware


中间件设置

首先我们需要为他指定控制器和验证器的对应关系,在app/Providers目录下新建一个ValidateServiceProvider.php文件,写入如下代码:

  1. <?php
  2. namespace App\Providers;
  3. use Illuminate\Support\ServiceProvider;
  4. use Itwmw\Validate\Middleware\ValidateMiddlewareConfig;
  5. class ValidateServiceProvider extends ServiceProvider{
  6.     public function register()
  7.     {
  8.         ValidateMiddlewareConfig::instance()
  9.             ->setAutoValidatePath('App\\Http\\Controllers\\', 'App\\Http\\Validate\\');
  10.     }
  11. }

其中setAutoValidatePath为指定控制器和验证器的对应关系,可以设定多个,不限制数量,然后我们将ValidateServiceProvider注册一下,在config/app.php文件中找到providers,在其中添加App\Providers\ValidateServiceProvider::class

注意:不可以放在Illuminate\Validation\ValidationServiceProvider::class之前


使用中间件

这个时候,我们可以将中间件注册为全局中间件,也可以不注册,注册方法:中间件《Laravel 7 中文文档》

定义路由:

  1. Route::middleware(ValidateMiddleware::class)->group(function () {
  2.     Route::any('/login', [\App\Http\Controllers\UserController::class, 'login']);
  3.     Route::any('/register', [\App\Http\Controllers\UserController::class, 'register']);
  4. });

控制器代码修改如下:

  1. class UserController extends BaseController{
  2.     use AuthorizesRequests, DispatchesJobs, ValidatesRequests;
  3.  
  4.     public function login(Request $request)
  5.     {
  6.         $data = get_validate_data($request);
  7.     }
  8.  
  9.     public function register(Request $request)
  10.     {
  11.         $data = get_validate_data($request);
  12.     }
  13. }

这个时候,我们就不需要手动指定验证器已经场景名了,中间件会自动处理对应的验证规则,我们只需要使用get_validate_data方法来接收值即可。

结尾

此文章只写了这个扩展的一些基本用法,还有更多的功能,需要大家自己看看文档,来完成自己合适的验证层。


本文网址:https://www.zztuku.com/detail-9046.html
站长图库 - 解析如何进行Laravel表单验证分层设计和验证场景应用
申明:如有侵犯,请 联系我们 删除。

评论(0)条

您还没有登录,请 登录 后发表评论!

提示:请勿发布广告垃圾评论,否则封号处理!!

    编辑推荐