Как проверить массив в Laravel?

106

Я пытаюсь проверить массив POST в Laravel:

$validator = Validator::make($request->all(), [
            "name.*" => 'required|distinct|min:3',
            "amount.*" => 'required|integer|min:1',
            "description.*" => "required|string"

        ]);

Я отправляю пустой POST и получаю это if ($validator->fails()) {}как False. Это означает, что проверка верна, но это не так.

Как проверить массив в Laravel? Когда я отправляю форму сinput name="name[]"

Дарама
источник

Ответы:

246

Символ звездочки (*) используется для проверки значений в массиве, а не для самого массива.

$validator = Validator::make($request->all(), [
    "names"    => "required|array|min:3",
    "names.*"  => "required|string|distinct|min:3",
]);

В приведенном выше примере:

  • "имена" должны быть массивом минимум из 3 элементов,
  • значения в массиве «names» должны быть отдельными (уникальными) строками, длиной не менее 3 символов.

РЕДАКТИРОВАТЬ: Начиная с Laravel 5.5, вы можете вызывать метод validate () непосредственно для объекта Request следующим образом:

$data = $request->validate([
    "name"    => "required|array|min:3",
    "name.*"  => "required|string|distinct|min:3",
]);
Филип Соболь
источник
не забудьте поместить его в пробный улов, если вы его используете $request->validate([...]). Если данные не прошли проверку, возникнет исключение.
daisura99
как получить сообщение об ошибке определенного поля? например, у меня есть 2 поля имени, а второе поле содержит только ошибку, как я могу этого добиться?
Eem
обязательное поле 'name. *' не требуется, поскольку оно проверяет его только в том случае, если оно существует
Бен Гудинг,
41

У меня есть этот массив в качестве данных моего запроса из сетки / таблицы данных HTML + Vue.js:

[0] => Array
    (
        [item_id] => 1
        [item_no] => 3123
        [size] => 3e
    )
[1] => Array
    (
        [item_id] => 2
        [item_no] => 7688
        [size] => 5b
    )

И используйте это, чтобы проверить, что работает правильно:

$this->validate($request, [
    '*.item_id' => 'required|integer',
    '*.item_no' => 'required|integer',
    '*.size'    => 'required|max:191',
]);
Нисал Гунавардана
источник
2
Это именно то, что мне нужно!
Крис
18

Рекомендуемый способ написания логики проверки и авторизации - поместить эту логику в отдельные классы запросов. Таким образом, код вашего контроллера останется чистым.

Вы можете создать класс запроса, выполнив php artisan make:request SomeRequest.

В каждом rules()методе класса запроса определите свои правила проверки:

//SomeRequest.php
public function rules()
{
   return [
    "name"    => [
          'required',
          'array', // input must be an array
          'min:3'  // there must be three members in the array
    ],
    "name.*"  => [
          'required',
          'string',   // input must be of type string
          'distinct', // members of the array must be unique
          'min:3'     // each string must have min 3 chars
    ]
  ];
}

В вашем контроллере напишите функцию маршрута следующим образом:

// SomeController.php
public function store(SomeRequest $request) 
{
  // Request is already validated before reaching this point.
  // Your controller logic goes here.
}

public function update(SomeRequest $request)
{
  // It isn't uncommon for the same validation to be required
  // in multiple places in the same controller. A request class
  // can be beneficial in this way.
}

Каждый класс запроса поставляется с перехватчиками / методами до и после проверки, которые можно настроить на основе бизнес-логики и особых случаев, чтобы изменить нормальное поведение класса запроса.

Вы можете создать родительские классы запросов для аналогичных типов запросов (например, webи api) запросов, а затем инкапсулировать некоторую общую логику запросов в эти родительские классы.

sumit
источник
6

Немного более сложные данные, смесь ответов @Laran и @Nisal Gunawardana

[ 
   {  
       "foodItemsList":[
    {
       "id":7,
       "price":240,
       "quantity":1
                },
               { 
                "id":8,
                "quantity":1
               }],
        "price":340,
        "customer_id":1
   },
   {   
      "foodItemsList":[
    {
       "id":7,
       "quantity":1
    },
    { 
        "id":8,
        "quantity":1
    }],
    "customer_id":2
   }
]

Правило проверки будет

 return [
            '*.customer_id' => 'required|numeric|exists:customers,id',
            '*.foodItemsList.*.id' => 'required|exists:food_items,id',
            '*.foodItemsList.*.quantity' => 'required|numeric',
        ];
Прафулла Кумар Саху
источник
4

Вам нужно перебрать входной массив и добавить правила для каждого входа, как описано здесь: Правила цикла

Вот вам код:

$input = Request::all();
$rules = [];

foreach($input['name'] as $key => $val)
{
    $rules['name.'.$key] = 'required|distinct|min:3';
}

$rules['amount'] = 'required|integer|min:1';
$rules['description'] = 'required|string';

$validator = Validator::make($input, $rules);

//Now check validation:
if ($validator->fails()) 
{ 
  /* do something */ 
}
Чад Фишер
источник
9
В этом нет необходимости - laravel.com/docs/5.4/validation#validating-arrays
Филип Соболь