mirror of
https://github.com/Respect/Validation.git
synced 2026-03-15 14:55:44 +01:00
Creating a specific exception for each rule adds a painful overhead. If you want to make a custom message for your rule, you will need to create an exception and then register that exception namespace to be able to use it—all that is just for customizing the message of your rule. Having different namespaces also implies that you need to fetch the exception of the rule from another directory to change it. As Uncle Bob said, "Classes that change together belong together. Classes that are not reused together should not be grouped." This commit will drastically change this library, moving all the templates from the exceptions to the rules. Consequently, the Factory becomes much simpler, and the library gets a bit smaller, too. Signed-off-by: Henrique Moody <henriquemoody@gmail.com>
168 lines
3.9 KiB
PHP
168 lines
3.9 KiB
PHP
<?php
|
|
|
|
/*
|
|
* Copyright (c) Alexandre Gomes Gaigalas <alganet@gmail.com>
|
|
* SPDX-License-Identifier: MIT
|
|
*/
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace Respect\Validation\Rules;
|
|
|
|
use Respect\Validation\Attributes\ExceptionClass;
|
|
use Respect\Validation\Attributes\Template;
|
|
use Respect\Validation\Exceptions\ComponentException;
|
|
use Respect\Validation\Exceptions\NonOmissibleValidationException;
|
|
use Respect\Validation\NonNegatable;
|
|
use Respect\Validation\Validatable;
|
|
|
|
use function array_key_exists;
|
|
use function array_map;
|
|
use function count;
|
|
use function current;
|
|
use function is_array;
|
|
|
|
#[ExceptionClass(NonOmissibleValidationException::class)]
|
|
#[Template(
|
|
'All of the required rules must pass for {{name}}',
|
|
'',
|
|
self::TEMPLATE_NONE,
|
|
)]
|
|
#[Template(
|
|
'These rules must pass for {{name}}',
|
|
'',
|
|
self::TEMPLATE_SOME,
|
|
)]
|
|
#[Template(
|
|
'Must have keys {{keys}}',
|
|
'',
|
|
self::TEMPLATE_STRUCTURE,
|
|
)]
|
|
#[Template(
|
|
'Must not have keys {{extraKeys}}',
|
|
'',
|
|
self::TEMPLATE_STRUCTURE_EXTRA,
|
|
)]
|
|
final class KeySet extends AbstractWrapper implements NonNegatable
|
|
{
|
|
public const TEMPLATE_NONE = 'none';
|
|
public const TEMPLATE_SOME = 'some';
|
|
public const TEMPLATE_STRUCTURE = 'structure';
|
|
public const TEMPLATE_STRUCTURE_EXTRA = 'structure_extra';
|
|
|
|
/**
|
|
* @var mixed[]
|
|
*/
|
|
private readonly array $keys;
|
|
|
|
/**
|
|
* @var mixed[]
|
|
*/
|
|
private array $extraKeys = [];
|
|
|
|
/**
|
|
* @var Key[]
|
|
*/
|
|
private readonly array $keyRules;
|
|
|
|
public function __construct(Validatable ...$validatables)
|
|
{
|
|
$this->keyRules = array_map([$this, 'getKeyRule'], $validatables);
|
|
$this->keys = array_map([$this, 'getKeyReference'], $this->keyRules);
|
|
|
|
parent::__construct(new AllOf(...$this->keyRules));
|
|
}
|
|
|
|
public function assert(mixed $input): void
|
|
{
|
|
if (!$this->hasValidStructure($input)) {
|
|
throw $this->reportError($input);
|
|
}
|
|
|
|
parent::assert($input);
|
|
}
|
|
|
|
public function check(mixed $input): void
|
|
{
|
|
if (!$this->hasValidStructure($input)) {
|
|
throw $this->reportError($input);
|
|
}
|
|
|
|
parent::check($input);
|
|
}
|
|
|
|
public function validate(mixed $input): bool
|
|
{
|
|
if (!$this->hasValidStructure($input)) {
|
|
return false;
|
|
}
|
|
|
|
return parent::validate($input);
|
|
}
|
|
|
|
public function getTemplate(mixed $input): string
|
|
{
|
|
if ($this->template !== null) {
|
|
return $this->template;
|
|
}
|
|
|
|
if (count($this->extraKeys)) {
|
|
return self::TEMPLATE_STRUCTURE_EXTRA;
|
|
}
|
|
|
|
return KeySet::TEMPLATE_STRUCTURE;
|
|
}
|
|
|
|
/**
|
|
* @return array<string, mixed>
|
|
*/
|
|
public function getParams(): array
|
|
{
|
|
return [
|
|
'keys' => $this->keys,
|
|
'extraKeys' => $this->extraKeys,
|
|
];
|
|
}
|
|
|
|
/**
|
|
* @throws ComponentException
|
|
*/
|
|
private function getKeyRule(Validatable $validatable): Key
|
|
{
|
|
if ($validatable instanceof Key) {
|
|
return $validatable;
|
|
}
|
|
|
|
if (!$validatable instanceof AllOf || count($validatable->getRules()) !== 1) {
|
|
throw new ComponentException('KeySet rule accepts only Key rules');
|
|
}
|
|
|
|
return $this->getKeyRule(current($validatable->getRules()));
|
|
}
|
|
|
|
private function getKeyReference(Key $rule): mixed
|
|
{
|
|
return $rule->getReference();
|
|
}
|
|
|
|
private function hasValidStructure(mixed $input): bool
|
|
{
|
|
if (!is_array($input)) {
|
|
return false;
|
|
}
|
|
|
|
foreach ($this->keyRules as $keyRule) {
|
|
if (!array_key_exists($keyRule->getReference(), $input) && $keyRule->isMandatory()) {
|
|
return false;
|
|
}
|
|
|
|
unset($input[$keyRule->getReference()]);
|
|
}
|
|
|
|
foreach ($input as $extraKey => &$ignoreValue) {
|
|
$this->extraKeys[] = $extraKey;
|
|
}
|
|
|
|
return count($input) == 0;
|
|
}
|
|
}
|