* SPDX-License-Identifier: MIT */ declare(strict_types=1); namespace Respect\Validation; use Respect\Validation\Exceptions\ValidationException; use Respect\Validation\Message\Formatter; use Respect\Validation\Message\Translator; use Respect\Validation\Mixins\Builder; use Respect\Validation\Rules\Core\Nameable; use Respect\Validation\Rules\Core\Reducer; use Throwable; use function is_array; use function is_callable; use function is_string; /** @mixin Builder */ final class Validator implements Rule, Nameable { /** @var array */ private array $rules = []; /** @var array */ private array $templates = []; private string|null $name = null; private string|null $template = null; /** @param array $ignoredBacktracePaths */ public function __construct( private readonly Factory $factory, private readonly Formatter $formatter, private readonly Translator $translator, private readonly array $ignoredBacktracePaths, ) { } public static function create(Rule ...$rules): self { $validator = new self( ValidatorDefaults::getFactory(), ValidatorDefaults::getFormatter(), ValidatorDefaults::getTranslator(), ValidatorDefaults::getIgnoredBacktracePaths(), ); $validator->rules = $rules; return $validator; } public function evaluate(mixed $input): Result { return $this->getRule()->evaluate($input); } public function isValid(mixed $input): bool { return $this->evaluate($input)->hasPassed; } /** @param array|callable(ValidationException): Throwable|string|Throwable|null $template */ public function assert(mixed $input, array|string|Throwable|callable|null $template = null): void { $result = $this->evaluate($input); if ($result->hasPassed) { return; } if ($template instanceof Throwable) { throw $template; } $templates = $this->templates; if (is_array($template)) { $templates = $template; } elseif (is_string($template)) { $templates = ['__root__' => $template]; } elseif ($this->getTemplate() != null) { $templates = ['__root__' => $this->getTemplate()]; } $exception = new ValidationException( $this->formatter->main($result, $templates, $this->translator), $this->formatter->full($result, $templates, $this->translator), $this->formatter->array($result, $templates, $this->translator), $this->ignoredBacktracePaths, ); if (!is_callable($template)) { throw $exception; } throw $template($exception); } /** @param array $templates */ public function setTemplates(array $templates): self { $this->templates = $templates; return $this; } public function addRule(Rule $rule): self { $this->rules[] = $rule; return $this; } public function getRule(): Reducer { return (new Reducer(...$this->rules))->withTemplate($this->template)->withName($this->name); } /** @return array */ public function getRules(): array { return $this->rules; } public function getName(): string|null { return $this->getRule()->getName() ?? $this->name; } public function setName(string $name): static { $this->name = $name; return $this; } public function getTemplate(): string|null { return $this->template; } public function setTemplate(string $template): static { $this->template = $template; return $this; } /** @param mixed[] $arguments */ public static function __callStatic(string $ruleName, array $arguments): self { return self::create()->__call($ruleName, $arguments); } /** @param mixed[] $arguments */ public function __call(string $ruleName, array $arguments): self { return $this->addRule($this->factory->rule($ruleName, $arguments)); } }