Laravel 表单验证之场景验证

本文介绍了如何在 Laravel 中创建自定义表单验证类和场景验证 Trait,以实现更灵活的验证规则。通过 `FormRequest` 基类和 `SceneValidator` Trait,可以方便地定义不同场景的验证规则,并在控制器中便捷地应用。例如,在 `UserRequest` 类中,针对注册和登录场景定义了不同的验证规则。当验证失败时,会抛出 401 错误并返回第一个错误信息。
摘要由CSDN通过智能技术生成

文章内容主体内容借鉴来源:《Laravel 开发 API 心得》
表单验证手册:《Laravel 8 中文文档 - 表单验证》


一、创建表单验证自定义类

1) FormRequest「表单验证基类」

Create File: app/Http/Requests/FormRequest.php

# [可选] artisan 方式创建
php artisan make:request FormRequest
<?php
namespace App\Http\Requests;
use App\Http\Requests\SceneValidator;
use Illuminate\Foundation\Http\FormRequest as BaseFormRequest;
use Illuminate\Contracts\Validation\Validator;

class FormRequest extends BaseFormRequest
{
    use SceneValidator; // 此处见第二步

    /**
     * 判断用户是否有请求权限
     *
     * @return bool
     */
    public function authorize()
    {
        return true;
    }

    /**
     * 预设验证场景
     *
     * @return array
     */
    public function scene()
    {
        return [];
    }

	/**
     * [覆盖方法] 处理失败的验证
     *
     * @param  \Illuminate\Validation\Validator  $validator
     * @return void
     *
     * @throws \Illuminate\Validation\ValidationException
     */
    protected function failedValidation(Validator $validator)
    {
//        throw (new \Illuminate\Validation\ValidationException($validator))
//            ->errorBag($this->errorBag)
//            ->redirectTo($this->getRedirectUrl());
        abort(401, $validator->errors()->first());
    }
}

2) SceneValidator「表单场景验证Trait类」

Create File: app/Http/Requests/SceneValidator.php

# [可选] artisan 方式创建
php artisan make:request SceneValidator
<?php
namespace App\Http\Requests;
use Illuminate\Contracts\Validation\{Factory, Validator};

trait SceneValidator
{
    protected $scene = null;

    protected $onlyRule = [];

    protected $autoValidate = true;

    /**
     * Validate.
     *
     * @param string|array $scene
     */
    public function validate($scene = '')
    {
        if (!$this->autoValidate) {
            if (is_array($scene)) {
                $this->onlyRule = $scene;
            } else {
                $this->scene = $scene;
            }

            $this->handleValidate();
        }
    }

    /**
     * 覆盖 ValidatesWhenResolvedTrait->validateResolved
     */
    public function validateResolved()
    {
        if (method_exists($this, 'autoValidate')) {
            $this->autoValidate = $this->container->call([$this, 'autoValidate']);
        }

        if ($this->autoValidate) {
            $this->handleValidate();
        }
    }

    /**
     * Handle validate.
     */
    protected function handleValidate()
    {
        parent::validateResolved();
    }

    /**
     * 定义 FormRequest->getValidatorInstance 下 validator 验证器
     *
     * @param Factory $factory
     *
     * @return Validator
     */
    public function validator(Factory $factory)
    {
        $validationData = $this->isMethod('GET') ? $this->query() : $this->post();
        return $factory->make($validationData, $this->getRules(), $this->messages(), $this->attributes());
    }

    /**
     * Get rules.
     *
     * @return array
     */
    protected function getRules()
    {
        return $this->handleScene($this->container->call([$this, 'rules']));
    }

    /**
     * Handle scene.
     *
     * @param array $rules
     *
     * @return array
     */
    protected function handleScene(array $rules)
    {
        if ($this->onlyRule) {
            return $this->handleRules($this->onlyRule, $rules);
        }

        if (!empty($this->scene) && method_exists($this, 'scene')) {
            $scene = $this->container->call([$this, 'scene']);
            if (array_key_exists($this->scene, $scene)) {
                return $this->handleRules($scene[$this->scene], $rules);
            }
        }

        return $rules;
    }

    /**
     * Handle rules.
     *
     * @param array $sceneRules
     * @param array $rules
     *
     * @return array
     */
    protected function handleRules(array $sceneRules, array $rules)
    {
        $result = [];
        foreach ($sceneRules as $key => $value) {
            if (is_numeric($key) && array_key_exists($value, $rules)) {
                $result[$value] = $rules[$value];
            } else {
                $result[$key] = $value;
            }
        }

        return $result;
    }
}

3) UserRequest「示例,表单业务验证类」

Create File: app/Http/Requests/UserRequest.php

# [可选] artisan 方式创建
php artisan make:request UserRequest
<?php
namespace App\Http\Requests\User;
use App\Http\Requests\FormRequest;

class UserRequest extends FormRequest
{
    protected $autoValidate = false;

	/**
     * 应用于请求的验证规则
     *
     * @return array
     */
    public function rules()
    {
    	$_rules = [
            'account' => ['required','string'],
            'password' => 'required|string|min:6',
        ];
        if ($this->scene === 'register') {
        	// 注册时,增加账号重复验证
            $_rules['account'][] = 'unique:\App\Models\Users,account';
        }
        return $_rules;
    }

	/**
     * 获取验证错误的自定义属性
     *
     * @return array
     */
    public function attributes()
    {
        return [
            'account' => '用户账号',
            'password' => '用户密码',
        ];
    }

    /**
     * 获取已定义验证规则的错误消息
     *
     * @return array
     */
    public function messages()
    {
        return [
            'account.required' => ':attribute必填',
            'password.required'  => ':attribute必填',
        ];
    }

	/**
     * 预设验证场景
     *
     * @return array
     */
    public function scene()
    {
    	$_default = [
            'account',
            'password',
        ];
        return [
            'register' => $_default,
            'login' => $_default,
        ];
    }
}

二、创建控制器使用示例

Create File: app/Http/Controllers/UserController.php

# [可选] artisan 方式创建
php artisan make:controller UserController
//...
use App\Http\Requests\User\UserRequest;
//...
public function register(UserRequest $request)
{
    $params = $request->post();
    // 验证失败时,抛出异常\Illuminate\Validation\ValidationException
    $request->validate('register');
	// 获取验证通过数据:
	dump($request->validated());
	// 获取所有请求参数数据
    dd($params);
}
//...
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

陀螺蚁

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值