Laravel 学习笔记之 request validation

在用laravel写api时,当前端传进来的request是POST/PUT/PATH等method时,那须要作request validation,尽管对于先后端分离程序,前端程序Angular/Vue已经作了validation,可是ajax传过来的json input,在后端也须要作validation。前端

那该如何优雅的编写request validation呢?laravel官方文档已经包含了这个feature: Form Request Validationlaravel

这里能够写一个JsonRequest:ajax

class JsonRequest extends Illuminate\Foundation\Http\FormRequest
{
    public function rules()
    {
        $method = $this->method();
        
        assert(in_array($method, [static::METHOD_POST, static::METHOD_PUT, static::METHOD_PATCH], true));
        
        $controller = $this->route()->getController();
        $rules      = $controller::RULES;

        return ($rules[$this->method()] ?? []) + ($rules['*'] ?? []);
    }

    public function authorize()
    {
        return true;
    }
}

这样就能够在众多Model Controller里使用JsonRequest就行,如:json

use Illuminate\Http\Request;

final class AccountController extends \App\Http\Controllers\Controller
{
    public const RULES = [
        Request::METHOD_POST => [
            'bank_account' => 'required_if:type,bank',
            'loan_account' => 'required_if:type,loan',
        ],
        Request::METHOD_PUT => [
            // ...
        ],
        '*' => [
            // ...
        ],
    ];
}

这样就能够校验前端传进来的json input是否合法。
(1)若是前端传进来的json input是:后端

{
    "name": "lx1036",
    "type": "loan",
    "bank_account": {
        "source": "bank",
    }
}

那就validation失败,不合法。
(2) 若是前端传进来的json input是:api

{
    "name": "lx1036",
    "type": "bank",
    "loan_account": {
        "source": "loan",
    }
}

那就validation失败,不合法。app

这样就能够校验json input,不合法就直接弹回throw 一个HttpException,再也不用在进入下一步逻辑。对于这样嵌套的json input,使用request validation来校验对象间关系很重要,能够看作是进入核心业务逻辑前的初步校验。。固然最后写表时还有model validation,避免坏数据进入db。前后端分离

最后一点,laravel文档只是说了用法,没有说明原理。代码在\Illuminate\Foundation\Providers\FormRequestServiceProvider::class:ide

public function boot()
    {
        // \Illuminate\Foundation\Http\FormRequest use 了 ValidatesWhenResolvedTrait,extends 了 \Illuminate\Contracts\Validation\ValidatesWhenResolved
        $this->app->afterResolving(ValidatesWhenResolved::class, function ($resolved) {
            $resolved->validate();
        });

        // ...
    }

因此当从容器中resolve完\Illuminate\Foundation\Http\FormRequest后就会当即执行\Illuminate\Foundation\Http\FormRequest::validate()方法,具体不详述,可看laravel源码。ui

OK,总之,在写程序时,validation很重要,须要去写,包括request validation和model validation。。。

相关文章
相关标签/搜索