Pre-processing
Before you can use a custom validation rule, you must define the namespace prefix for the custom validation rule
ValidateConfig::instance()->setRulesPath('Itwmw\\App\\Model\\Validate\\Rules\\');
It is recommended to define the validate-related settings in Provider
.
Priority
The validator's Built-in Rules is greater than that of the extended rules, so care needs to be taken that the rule names are not consistent.
Extending global rules
Global rules refer to rules that can be used in all validators
Using rule objects
There are many useful validation rules within the validator; custom rules are also supported。 One way to register a custom validation rule is to create a new rule and inherit from Itwmw\Validate\Support\Rule\BaseRule
. The new rule is stored in the directory you have set up.
Once the rule is created, we can define its behavior. The passes
method takes the attribute value and name and returns true
or false
depending on whether the attribute value matches the rule. The message
attribute is the validation error message used in case of validation failure, which can be overridden by the message
defined in the validator.
ImplicitRule
If you would like a rule object to run when an attribute is empty, you should implement the Itwmw\Validation\Support\Interfaces\ImplicitRule
interface. This interface serves as a "marker interface" for the validator; therefore, it does not contain any methods you need to implement.
namespace Itwmw\App\Model\Validate\Rules;
class Chs extends BaseRule
{
/**
* Default error message
* @var string
*/
protected $message = 'The value of :attribute can only have Chinese';
/**
* Determine if the validation rules pass。
*
* @param mixed $attribute
* @param mixed $value
* @return bool
*/
public function passes($attribute, $value): bool
{
return is_scalar($value) && 1 === preg_match('/^[\x{4e00}-\x{9fa5}]+$/u', (string)$value);
}
}
Once the rule object has been defined, you can pass an instance of the rule object to the validator by passing it along with other validation rules:
protected $rule = [
'title' => 'required|chs',
];
The first letter of the custom extension rule can be lowercase, it is also recommended to use lowercase.
Rule object error message formatting support:Formatting error messages
The rules also support standalone calls to use
Chs::make()->check("name");
Validation returns true
, failure returns false
extend: Extend Custom validation rules
In validators, you can use the extend
method to register a rule. For example:
class UserValidator extends Validate
{
}
$v = UserValidator::make();
$v->extend("chs",function ($attribute, $value, $parameters, $validator) {
return is_scalar($value) && 1 === preg_match('/^[\x{4e00}-\x{9fa5}]+$/u', (string)$value);
}, ':attribute must only contain Chinese characters');
The custom validation closure takes four parameters: the name of the attribute to be validated $attribute
, the value of the attribute $value
, the array of parameters passed to the validation rule $parameters
, and the Validator
instance. In addition to using a closure, you can also pass a class and method to the extend
method:
class Rules
{
public function chs($attribute, $value, $parameters, $validator) {
return is_scalar($value) && 1 === preg_match('/^[\x{4e00}-\x{9fa5}]+$/u', (string)$value);
}
}
$v->extend("chs","Rules@chs", ':attribute must only contain Chinese characters');
If your method is a static method, you can also pass it in array form [full class name, method name]:
class Rules
{
public static function chs($attribute, $value, $parameters, $validator)
{
return is_scalar($value) && 1 === preg_match('/^[\x{4e00}-\x{9fa5}]+$/u', (string)$value);
}
}
$v->extend('chs', [Rules::class, 'chs'], ':attribute must only contain Chinese characters');
Note
When passing a class and method, the method must be of public
visibility, and the full namespace of the class must be provided. If no method name is passed, the default method name is validate
.
Replacer: Error Message Customizer
If you need to define error messages, you can pass them in as the third argument of extend
, or define them using the extendReplacer
method of the validator. The extendReplacer
method takes two parameters: ruleName
and a closure. The closure accepts four parameters: the error message $message
, the name of the validated attribute $attribute
, the current rule name $rule
, and the parameter array passed to the validation rule $parameters
.
$v->extendReplacer('chs', function ($message, $attribute, $rule, $parameters) {
return 'Custom error message';
});
In addition to closures, you can also pass classes and methods to the extendReplacer
method:
$v->extendReplacer('chs', "Rules@chsMessage");
// For static methods
$v->extendReplacer('chs', [Rules::class, 'chsMessage']);
Note
When passing a class and method to extendReplacer
, the method must be public
and the class must be the fully qualified namespace. If no method name is passed, the default method name is replace
, which can override the custom message of the default rule.
Usage is the same as using the rule object:
protected $rule = [
'title' => 'required|chs',
];
extendImplicit
By default, when an attribute being validated is not present or contains an empty string, normal validation rules, including custom extensions, are not run. For example, the unique
rule will not be run against an empty string:
$rules = ['name' => 'unique:users,name'];
$input = ['name' => ''];
// Validate passed
For a rule to run even when an attribute is empty, the rule must imply that the attribute is required. To create such an "implicit" extension, use the extendImplicit()
method in Validate
:
$v->extendImplicit('foo', function ($attribute, $value, $parameters, $validator) {
return $value == 'foo';
});
An "implicit" extension only implies that the attribute is required. Whether it actually invalidates a missing or empty attribute is up to you.
extendDependent
If you want to define a custom extension to validate an array, you may find that extend
and extendImplicit
do not parse *
. In this case, you can use the extendDependent
method in Validate
:
$v = new Validate();
$v->extendDependent('contains', function ($attribute, $value, $parameters, $validator) {
// The $parameters passed from the validator below is ['*.provider'], when we imply that this
// custom rule is dependent the validator tends to replace the asterisks with the current
// indices as per the original attribute we're validating, so *.provider will be replaced
// with 0.provider, now we can use array_get() to get the value of the other field.
// So this custom rule validates that the attribute value contains the value of the other given attribute
return str_contains($value,Arr::get($validator->getData(),$parameters[0]));
});
$v->setRules([
'*.email' => 'contains:*.provider'
])->check([
[
'email' => '995645888@qq.com', 'provider' => 'qq.com'
]
]);
Custom rules within the validator
Using class methods
Custom rules also support the direct use of methods under the current validator class, the rule is rule
+ rule name, e.g. ruleCheckLogin
Note
The checkLogin
here and other methods registered under the current validator must not duplicate the rule name, otherwise it will overwrite
The Custom Rule method takes four parameters: The name of the attribute to be validated $attribute
, the value of the attribute $value
, an array of parameters to be passed into the validation rule $parameters
, and a Validator
instance.
class LoginValidate extends Validate
{
protected $rule = [
'user' => 'required|alphaNum|checkLogin'
];
protected $message = [
'user.checkLogin' => 'Login failure'
];
public function ruleCheckLogin($attribute, $value, $parameters, $validator): bool
{
return 'admin' === $value;
}
}
You can also set error messages for class method rules with the $ruleMessage
parameter
class Test extends Validate
{
protected $ruleMessage = [
'chs' => 'The value of :attribute can only have Chinese'
];
protected function ruleChs($attribute, $value, $parameters, $validator): bool
{
return is_scalar($value) && 1 === preg_match('/^[\x{4e00}-\x{9fa5}]+$/u', (string)$value);
}
}
Regular Expression Rule
Supports direct use of regular validation, e.g:
'id' => 'regex:/^\d+$/',
If your regular expression contains the |
symbol, it must be defined in an array.
'id' => ['regex:/^\d+$/']
You can also implement a predefined regular expression and call it directly, for example, by defining the regex
property in the validator class and then using regex:
+ regular name
in the rule
class Test extends \Itwmw\Validate\Validate
{
protected $regex = [
'number' => '/^\d+$/'
];
protected $rule = [
'id' => 'regex:number'
];
}
Custom rule incoming parameters
Custom rules, like other rules, also support passing in parameters, similar to max:100
, in:0,1,2
, which will be passed into the constructor of the custom rule class in order. as follows:
class Length extends BaseRule
{
protected $message = 'The length of :attribute does not meet the requirement';
protected $size;
public function __construct(int $size)
{
$this->size = $size;
}
public function passes($attribute, $value): bool
{
return strlen($value) === $this->size;
}
}
Or Rule
When you need to pass validation if any of multiple rules pass, you can use the or
rule. For example:
class User extends Validate {
protected $rule = [
'username' => 'required|or:email,mobile'
];
}
This means that if username
satisfies either the email or mobile rule, it will pass validation:
User::make()->check([
'username' => '995645888@qq.com'
]);
User::make()->check([
'username' => '18899996666'
]);
Both 995645888@qq.com
and 18899996666
satisfy the rule.
Using Rule Groups
The or
rule also supports using rule groups. For example:
class User extends Validate {
protected $group = [
'username' => 'email|mobile'
];
protected $rule = [
'username' => 'required|or:username'
];
}
You can also use multiple rule groups, for example:
class User extends Validate
{
protected $group = [
'username' => 'email|mobile',
'username_nl' => 'alpha|chs'
];
protected $rule = [
'username' => 'required|or:username,username_nl'
];
}
In this definition, username
will pass validation as long as it satisfies any rule in either the username
or username_nl
rule groups.
You can also use rule groups and individual rules together, for example:
class User extends Validate
{
protected $group = [
'username_nl' => 'alpha|chs',
'username_specific' => 'in:admin123,root520'
];
protected $rule = [
'username' => 'required|or:username_specific,username_nl,email,mobile'
];
}
In this example, the username
will pass the validation if it satisfies any rule in either the username_specific
or username_nl
rule group, or if it satisfies any rule in the email
and mobile
rule groups.
User::make()->check([
'username' => '995645888@qq.com'
]);
User::make()->check([
'username' => '18899996666'
]);
User::make()->check([
'username' => '虞灪'
]);
User::make()->check([
'username' => 'yapyuyu'
]);
User::make()->check([
'username' => 'admin123'
]);
User::make()->check([
'username' => 'root520'
]);
All of the above are valid/acceptable for validation.