alleyinteractive / laminas-validator-extensions
Additional validation classes for the laminas-validator framework.
Installs: 239 632
Dependents: 2
Suggesters: 0
Security: 0
Stars: 0
Watchers: 21
Forks: 2
Open Issues: 2
pkg:composer/alleyinteractive/laminas-validator-extensions
Requires
- php: ^8.0
- laminas/laminas-validator: ^2.20
Requires (Dev)
- friendsofphp/php-cs-fixer: ^3.8
- phpunit/phpunit: ^9.5
- squizlabs/php_codesniffer: ^3.6
README
This package provides additional validation classes for the Laminas Validator framework, plus a custom base validation class.
Installation
Install the latest version with:
$ composer require alleyinteractive/laminas-validator-extensions
Basic usage
For more information about what validators do, how to use them, and how to write your own, visit the Laminas documentation.
Base validators
ExtendedAbstractValidator
The abstract Alley\Validator\ExtendedAbstractValidator class standardizes the implementation of custom validators with \Laminas\Validator\AbstractValidator.
When extending ExtendedAbstractValidator, validation logic goes into a new testValue() method, which is responsible only for applying the logic and adding any validation errors. It's no longer necessary to call setValue() prior to evaluating the input, and isValid() will return true if there are no error messages after evaluating the input and false if there are any messages.
Before:
<?php class Float extends \Laminas\Validator\AbstractValidator { const FLOAT = 'float'; protected $messageTemplates = [ self::FLOAT => "'%value%' is not a floating point value", ]; public function isValid($value) { $this->setValue($value); if (! is_float($value)) { $this->error(self::FLOAT); return false; } return true; } }
After:
<?php class Float extends \Alley\Validator\ExtendedAbstractValidator { const FLOAT = 'float'; protected $messageTemplates = [ self::FLOAT => "'%value%' is not a floating point value", ]; public function testValue($value): void { if (! is_float($value)) { $this->error(self::FLOAT); } } }
FreeformValidator
The standalone, abstract Alley\Validator\FreeformValidator class leaves most of the implementation details to your discretion, but it's often easier to use for validators that are project-specific or not ready for wider distribution.
Like the ExtendedAbstractValidator class, the FreeformValidator expects that validation logic goes into a testValue() method, and isValid() will return true or false based on whether there are error messages.
Validation errors can be added using the error() method, which accepts the message key and text.
<?php class Float extends \Alley\Validator\FreeformValidator { public function testValue($value): void { if (! is_float($value)) { $this->error('float', 'Please enter a floating point value'); } } }
"Any Validator" chains
\Alley\Validator\AnyValidator is like a Laminas validator chain except that it connects the validators with "OR," marking input as valid as soon it passes one of the given validators.
Unlike a Laminas validator chain, validators can only be attached, not prepended, and there is no $priority argument.
Basic usage
<?php $valid = new \Alley\Validator\AnyValidator([new \Laminas\Validator\LessThan(['max' => 10])]); $valid->attach(new \Laminas\Validator\GreaterThan(['min' => 90])); $valid->isValid(9); // true $valid->isValid(99); // true $valid->isValid(42); // false
"Fast fail" validator chains
\Alley\Validator\FastFailValidatorChain is like a Laminas validator chain except that if a validator fails, the chain will automatically be broken; there is no $breakChainOnFailure parameter.
Unlike a Laminas validator chain, validators can only be attached, not prepended, and there is no $priority argument.
Basic usage
$valid = new \Alley\Validator\FastFailValidatorChain([new \Laminas\Validator\LessThan(['max' => 10])]); $valid->attach(new \Laminas\Validator\GreaterThan(['min' => 90])); $valid->isValid(42); // false count($valid->getMessages()); // 1
Validators by operator name
\Alley\Validator\ValidatorByOperator allows you to access a validator using a readable operator name, such as REGEX or NOT IN.
Its primary use case is to allow you to write functions that accept the readable operator names as parameters while using validators internally. Here's a demonstrative function call from the wp-match-blocks library:
<?php $images = \Alley\WP\match_blocks( $post, [ 'name' => 'core/image', 'attrs' => [ [ 'key' => 'credit', 'value' => '/(The )?Associated Press/i', 'operator' => 'REGEX', ], ], ], );
The supported operator names are:
CONTAINSandNOT CONTAINS, which forward to\Alley\Validator\ContainsStringusing a case-sensitive search.INandNOT IN, which forward to\Alley\Validator\OneOf.LIKEandNOT LIKE, which forward to\Alley\Validator\ContainsStringusing a case-insensitive search.REGEXandNOT REGEX, which forward to\Laminas\Validator\Regex.===,!==, and the other operators supported by\Alley\Validator\Comparison.
Any operator name that isn't forwarded to a different validator must be a valid Comparison operator.
Basic usage
$valid = new \Alley\Validator\ValidatorByOperator('REGEX', '/^foo/'); $valid->isValid('foobar'); // true $valid = new \Alley\Validator\ValidatorByOperator('NOT IN', ['bar', 'baz']); $valid->isValid('bar'); // false $valid = new \Alley\Validator\ValidatorByOperator('!==', 42); $valid->isValid(43); // true
Validators
AlwaysValid
\Alley\Validator\AlwaysValid marks all input as valid. It can be used to satisfy type requirements when full validation needs to be disabled or is impractical.
Supported options
None.
Basic usage
<?php $valid = new \Alley\Validator\AlwaysValid(); $valid->isValid(42); // true $valid->isValid(false); // true $valid->isValid('abcdefghijklmnopqrstuvwxyz'); // true
Comparison
\Alley\Validator\Comparison compares input to another value using a PHP comparison operator. The input passes validation if the comparison is true. Input is placed on the left side of the operator.
Supported options
The following options are supported for \Alley\Validator\Comparison:
compared: The value the inputs are compared to. It is placed on the right side of the operator.operator: The PHP comparison operator used to compare the input andcompared.
Basic usage
<?php $valid = new \Alley\Validator\Comparison( [ 'operator' => '<=', 'compared' => 100, ] ); $valid->isValid(101); // false $valid = new \Alley\Validator\Comparison( [ 'operator' => '!==', 'compared' => false, ] ); $valid->isValid(true); // true
ContainsString
\Alley\Validator\ContainsString is a validator around the str_contains() function. Each instance of the validator represents the "needle" string and validates whether the string is found within the input "haystack." Inputs will automatically be cast to strings.
Supported options
The following options are supported for \Alley\Validator\ContainsString:
needle: The string or instance of\Stringablethe inputs are searched for. It will automatically be cast to a string at the time of validation.ignoreCase: Whether to perform a case-insensitive search. False by default.
Basic usage
<?php $valid = new \Alley\Validator\ContainsString( [ 'needle' => 'foo', ], ); $valid->isValid('foobar'); // true $valid->isValid('barbaz'); // false
DivisibleBy
\Alley\Validator\DivisibleBy allows you to validate whether the input is evenly divisible by a given numeric value. Inputs will automatically be cast to integers.
Supported options
The following options are supported for \Alley\Validator\DivisibleBy:
divisor: The value the inputs are divided by. It will automatically be cast to an integer.
Basic usage
<?php $valid = new \Alley\Validator\DivisibleBy( [ 'divisor' => 3, ], ); $valid->isValid(9); // true $valid->isValid(10); // false
Not
Alley\Validator\Not inverts the validity of a given validator. It allows for creating validators that test whether input is, for example, "not one of" in addition to "one of."
Supported options
None.
Basic usage
<?php $origin = new \Alley\Validator\OneOf(['haystack' => ['foo', 'bar']]); $valid = new \Alley\Validator\Not($origin, 'The input was invalid.'); $valid->isValid('foo'); // false $valid->isValid('baz'); // true
OneOf
Alley\Validator\OneOf validates whether an array of scalar values contains the input.
OneOf is a simpler version of \Laminas\Validator\InArray that accepts only scalar values in the haystack and does only strict comparisons. In return, it produces a friendlier error message that lists the allowed values.
Supported options
The following options are supported for \Alley\Validator\OneOf:
haystack: The array to be searched for the input.
Basic Usage
<?php $valid = new \Alley\Validator\OneOf(['haystack' => ['one', 'two', 'three']]); $valid->isValid('four'); // false $valid->getMessages(); // ['notOneOf' => 'Must be one of [one, two, three] but is four.']
Type
\Alley\Validator\Type allows you to validate whether the input is of the given PHP type. The input passes if it is the expected type.
This validator is inspired by PHPUnit's \PHPUnit\Framework\Constraint\IsType class.
Supported options
The following options are supported for \Alley\Validator\Type:
type: The expected PHP type. Supported types arearray,bool,boolean,callable,double,float,int,integer,iterable,null,numeric,object,real,resource,string, andscalar.
Basic usage
<?php $valid = new \Alley\Validator\Type(['type' => 'callable']); $valid->isValid('date_create_immutable'); // true $valid = new \Alley\Validator\Type(['type' => 'bool']); $valid->isValid([]); // false
WithMessage
Alley\Validator\WithMessage allows you to decorate a validator with a custom failure code and message, replacing the validator's usual failure messages.
Supported options
None.
Basic usage
<?php $origin = new \Laminas\Validator\GreaterThan(42); $valid = new \Alley\Validator\WithMessage('tooSmall', 'Please enter a number greater than 42.', $origin); $valid->isValid(41); // false $valid->getMessages(); // ['tooSmall' => 'Please enter a number greater than 42.']