laravel 5.4在请求中验证之前修改数据[关闭]

Posted

技术标签:

【中文标题】laravel 5.4在请求中验证之前修改数据[关闭]【英文标题】:laravel 5.4 modify data before validation in request [closed] 【发布时间】:2017-07-19 06:42:28 【问题描述】:

我有我的自定义请求,它扩展了 Backpack CrudController。

现在我想覆盖 ValidatesWhenResolvedTrait 的 prepareForValidation,因为它看起来是修改传入数据的正确位置,但我不知道如何...

所以我的第一个问题是,我可以覆盖这个方法吗?它受保护...

protected function prepareForValidation()

我的第二个问题是,如何修改我对 Request 或 FormRreuqest 对象的输入?

这是我的请求类

<?php

namespace App\Http\Requests;

use App\Http\Requests\Request;
use Config;

class DonationsRequest extends \Backpack\CRUD\app\Http\Requests\CrudRequest



    /**
     * Determine if the user is authorized to make this request.
     *
     * @return bool
     */
    public function authorize()
    
        // only allow updates if the user is logged in
        return \Auth::check();
    

    /**
     * Get the validation rules that apply to the request.
     *
     * @return array
     */
    public function rules()
    
        return [
            'name' => 'required|max:255',
            'email' => 'required|email',
            'dob' => 'required|date',
            'newsletter' => 'required|boolean',
            'country' => 'sometimes|required|in:'.implode(',', Config::get('validation.countries')),
            'street' => 'sometimes|required|string|max:255',
            'zip' => 'sometimes|required|string|between:4,5',
            'city' => 'sometimes|required|string|between:4,255',
            'amount' => 'required|numeric|between:1,'.Config::get('donations.max'),
            'type' => 'required|in:oo,monthly',
            'provider' => 'sometimes|string|nullable',
            'product_id' => 'sometimes|exists:products,id|nullable',
            'campaign_id' => 'required|exists:campaigns,id',
            'status' => 'sometimes|required|in:pending,canceled,success,error',
            'profile' => 'sometimes|string|regex:/^profile[0-9]+$/|nullable',
        ];
    

    /**
     * Get the validation attributes that apply to the request.
     *
     * @return array
     */
    public function attributes()
    
        return [
            //
        ];
    

    /**
     * Get the validation messages that apply to the request.
     *
     * @return array
     */
    public function messages()
    
        return [
            //
        ];
    

    private function prepareForValidation()
    

        dd('getValidatorInstance custom');

        $this->sanitizeInput();

        return parent::getValidatorInstance();
    

    private function sanitizeInput()
    

        dd('sanitizeInput custom');

        $data = $this->all();

        dd($data);

        // overwrite the newsletter field value to match boolean validation
        $data['newsletter'] = ($data['newsletter'] == 'true' || $data['newsletter'] == '1' || $data['newsletter'] == true) ? true : false;

        return $data;
    

    private function validate() 
        dd('validate');
    

如您所见,我首先尝试覆盖 getValidatorInstance 方法,因为这看起来像是常见的方法,但它没有被执行(所以没有被覆盖 - 受保护?)。

【问题讨论】:

我目前无法测试,但这可能只是一个错字吗? private function prepareForValidation(),尝试将其更改为公开。 不,没有区别...不能覆盖该方法:/ thx! prepareForValidation() 是否受到保护并不重要,这是您应该重写以执行这些类型的任务的方法。在此方法中,调用 $this-&gt;replace() 将您的输入替换为经过清理的变体。 【参考方案1】:

虽然我没有尝试过,但它似乎应该可以工作,您可以从 Illuminate\Foundation\Http\FormRequest 类中覆盖 validationData

/**
 * Get data to be validated from the request.
 *
 * @return array
 */
protected function validationData()

    $all = parent::validationData();
    //e.g you have a field which may be json string or array
    if (is_string($playerIDs = array_get($all, 'player_id')))
        $playerIDs = json_decode($playerIDs, true);

    $all['player_id'] = $playerIDs
    return $all;

或者你可以覆盖Illuminate\Http\Concerns\InteractsWithInput trait中的all方法

/**
 * Get all of the input and files for the request.
 *
 * @return array
 */
public function all()

    $all = parent::all();
    //then do your operation
    if (is_string($playerIDs = array_get($all, 'player_id')))
        $playerIDs = json_decode($playerIDs, true);

    $all['player_id'] = $playerIDs
    return $all;

【讨论】:

我都试过了,但它不接受覆盖。我必须做一些特别的事情来完成这项工作吗?以某种方式提及特质或类别? @ThomasVenturini 我已经更新了我的答案,请检查历史记录,如果您没有成功,请更新您的问题,让我们知道您做了什么 我使用的是 Laravel 5.7。这相当于什么? @ggderas 适用于 5.7【参考方案2】:

你能修改请求吗?

$request->merge(['field' => 'new value']);

【讨论】:

我会在哪里这样做?如上所述,问题是我找不到要覆盖的方法。但是谢谢 :)【参考方案3】:

我敢肯定,这可以帮助修改输入,它对我有用。[laravel 5.4]

放置这个

$input['url'] = $url;
$this->replace($input);
dd($input);

在列表表单请求中。 (使用$all 而不是$input,如果您遵循上面使用的答案)。

这只会改变输入,即使在控制器中也可用。您仍然需要找到一种将其插入 DB 的方法,或者执行其他操作以使用修改后的输入在刀片中使用它。

【讨论】:

这可能是 Thomas Venturini 向@Tim Jai 提出的问题的答案。【参考方案4】:

好的,我发现了错误所在。我确实拆分了前端请求和后端请求调用。由于我正在处理后端请求,因此前端请求没有覆盖任何东西......所以这是我的错,那里没有错误,抱歉浪费时间,但非常感谢社区!

【讨论】:

以上是关于laravel 5.4在请求中验证之前修改数据[关闭]的主要内容,如果未能解决你的问题,请参考以下文章

Laravel 5.4 OAuth 与 Dingo 内部请求

Laravel 请求在验证失败时不返回修改后的请求

Laravel 5.4 表单请求唯一验证:如何访问 id?

Laravel 5.4 验证请求,如何处理更新时的唯一验证?

Laravel 5.4 有时验证规则不起作用

laravel :在验证之前清理请求数据