Skip to content

Commit 52e3e81

Browse files
committed
Fix string rule array merge
1 parent bb53227 commit 52e3e81

File tree

3 files changed

+55
-0
lines changed

3 files changed

+55
-0
lines changed

src/Resolvers/DataValidationRulesResolver.php

+4
Original file line numberDiff line numberDiff line change
@@ -244,6 +244,10 @@ protected function resolveRules(
244244
$rulesFromRulesMethod = app()->call([$class->name, 'rules'], ['context' => $validationContext]);
245245

246246
if ($this->shouldMergeRules($class)) {
247+
$rulesFromRulesMethod = collect($rulesFromRulesMethod)->map(
248+
fn (string|array $rules) => is_array($rules) ? $rules : explode('|', $rules)
249+
)->all();
250+
247251
$dataRules->rules = array_merge_recursive($dataRules->rules, $rulesFromRulesMethod);
248252
return;
249253
}

tests/Attributes/Validation/MergeRulesetTest.php

+27
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,9 @@
11
<?php
22

33
use Illuminate\Validation\ValidationException;
4+
use Spatie\LaravelData\Data;
45
use Spatie\LaravelData\Tests\Fakes\DataWithMergedRuleset;
6+
use Spatie\LaravelData\Tests\Fakes\DataWithMergedStringRuleset;
57

68
it('it will merge validation rules', function () {
79
try {
@@ -21,3 +23,28 @@
2123
}
2224
});
2325

26+
it('it will merge validation rules using string rules', function () {
27+
try {
28+
Data::validate(['first_name' => str_repeat('a', 1)]);
29+
} catch (ValidationException $exception) {
30+
expect($exception->errors())->toMatchArray([
31+
'first_name' => ['The first name field must be at least 2 characters.']
32+
]);
33+
}
34+
35+
try {
36+
DataWithMergedStringRuleset::validate(['first_name' => str_repeat('a', 11)]);
37+
} catch (ValidationException $exception) {
38+
expect($exception->errors())->toMatchArray([
39+
'first_name' => ['The first name field must not be greater than 10 characters.']
40+
]);
41+
}
42+
43+
try {
44+
DataWithMergedStringRuleset::validate(['first_name' => 'a123']);
45+
} catch (ValidationException $exception) {
46+
expect($exception->errors())->toMatchArray([
47+
'first_name' => ['The first name field must only contain letters.']
48+
]);
49+
}
50+
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
<?php
2+
3+
namespace Spatie\LaravelData\Tests\Fakes;
4+
5+
use Spatie\LaravelData\Attributes\MergeRules;
6+
use Spatie\LaravelData\Attributes\Validation\Max;
7+
use Spatie\LaravelData\Data;
8+
9+
#[MergeRules]
10+
class DataWithMergedStringRuleset extends Data
11+
{
12+
public function __construct(
13+
#[Max(10)]
14+
public string $first_name,
15+
) {
16+
}
17+
18+
public static function rules(): array
19+
{
20+
return [
21+
'first_name' => 'min:2|alpha'
22+
];
23+
}
24+
}

0 commit comments

Comments
 (0)