mirror of
https://github.com/Respect/Validation.git
synced 2026-03-15 06:45:44 +01:00
We want to release version 3.0 as fresh as possible, without having to maintain backward compatibility with the previous versions. Because that version will be on for some time, we decided it will be best to support only PHP version 8.5 or higher. Acked-by: Alexandre Gomes Gaigalas <alganet@gmail.com>
143 lines
4.6 KiB
PHP
143 lines
4.6 KiB
PHP
<?php
|
|
|
|
/*
|
|
* Copyright (c) Alexandre Gomes Gaigalas <alganet@gmail.com>
|
|
* SPDX-License-Identifier: MIT
|
|
*/
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace Respect\Validation\Rules;
|
|
|
|
use Attribute;
|
|
use DateTimeImmutable;
|
|
use DateTimeInterface;
|
|
use Respect\Validation\Exceptions\InvalidRuleConstructorException;
|
|
use Respect\Validation\Helpers\CanValidateDateTime;
|
|
use Respect\Validation\Message\Template;
|
|
use Respect\Validation\Result;
|
|
use Respect\Validation\Rule;
|
|
use Throwable;
|
|
|
|
use function in_array;
|
|
use function ucfirst;
|
|
|
|
#[Attribute(Attribute::TARGET_PROPERTY | Attribute::IS_REPEATABLE)]
|
|
#[Template(
|
|
'The number of {{type|trans}} between now and',
|
|
'The number of {{type|trans}} between now and',
|
|
self::TEMPLATE_STANDARD
|
|
)]
|
|
#[Template(
|
|
'The number of {{type|trans}} between {{now}} and',
|
|
'The number of {{type|trans}} between {{now}} and',
|
|
self::TEMPLATE_CUSTOMIZED
|
|
)]
|
|
#[Template(
|
|
'For comparison with {{now|raw}}, {{name}} must be a valid datetime',
|
|
'For comparison with {{now|raw}}, {{name}} must not be a valid datetime',
|
|
self::TEMPLATE_NOT_A_DATE
|
|
)]
|
|
#[Template(
|
|
'For comparison with {{now|raw}}, {{name}} must be a valid datetime in the format {{sample|raw}}',
|
|
'For comparison with {{now|raw}}, {{name}} must not be a valid datetime in the format {{sample|raw}}',
|
|
self::TEMPLATE_WRONG_FORMAT
|
|
)]
|
|
final readonly class DateTimeDiff implements Rule
|
|
{
|
|
use CanValidateDateTime;
|
|
|
|
public const TEMPLATE_CUSTOMIZED = '__customized__';
|
|
public const TEMPLATE_NOT_A_DATE = '__not_a_date__';
|
|
public const TEMPLATE_WRONG_FORMAT = '__wrong_format__';
|
|
|
|
/** @param "years"|"months"|"days"|"hours"|"minutes"|"seconds"|"microseconds" $type */
|
|
public function __construct(
|
|
private string $type,
|
|
private Rule $rule,
|
|
private ?string $format = null,
|
|
private ?DateTimeImmutable $now = null,
|
|
) {
|
|
$availableTypes = ['years', 'months', 'days', 'hours', 'minutes', 'seconds', 'microseconds'];
|
|
if (!in_array($this->type, $availableTypes, true)) {
|
|
throw new InvalidRuleConstructorException(
|
|
'"%s" is not a valid type of age (Available: %s)',
|
|
$this->type,
|
|
$availableTypes
|
|
);
|
|
}
|
|
}
|
|
|
|
public function evaluate(mixed $input): Result
|
|
{
|
|
$now = $this->now ?? new DateTimeImmutable();
|
|
$compareTo = $this->createDateTimeObject($input);
|
|
if ($compareTo === null) {
|
|
$template = $this->format === null ? self::TEMPLATE_NOT_A_DATE : self::TEMPLATE_WRONG_FORMAT;
|
|
$parameters = ['sample' => $now->format($this->format ?? 'c'), 'now' => $this->nowParameter($now)];
|
|
|
|
return Result::failed($input, $this, $parameters, $template)
|
|
->withId('dateTimeDiff' . ucfirst($this->rule->evaluate($input)->id));
|
|
}
|
|
|
|
$nowPlaceholder = $this->nowParameter($now);
|
|
|
|
return Result::fromAdjacent(
|
|
$input,
|
|
'dateTimeDiff',
|
|
$this,
|
|
$this->rule->evaluate($this->comparisonValue($now, $compareTo)),
|
|
['type' => $this->type, 'now' => $nowPlaceholder],
|
|
$nowPlaceholder === 'now' ? self::TEMPLATE_STANDARD : self::TEMPLATE_CUSTOMIZED
|
|
);
|
|
}
|
|
|
|
private function comparisonValue(DateTimeInterface $now, DateTimeInterface $compareTo): int|float
|
|
{
|
|
return match ($this->type) {
|
|
'years' => $compareTo->diff($now)->y,
|
|
'months' => $compareTo->diff($now)->m,
|
|
'days' => $compareTo->diff($now)->d,
|
|
'hours' => $compareTo->diff($now)->h,
|
|
'minutes' => $compareTo->diff($now)->i,
|
|
'seconds' => $compareTo->diff($now)->s,
|
|
'microseconds' => $compareTo->diff($now)->f,
|
|
};
|
|
}
|
|
|
|
private function nowParameter(DateTimeInterface $now): string
|
|
{
|
|
if ($this->format === null && $this->now === null) {
|
|
return 'now';
|
|
}
|
|
|
|
if ($this->format === null) {
|
|
return $now->format('Y-m-d H:i:s.u');
|
|
}
|
|
|
|
return $now->format($this->format);
|
|
}
|
|
|
|
private function createDateTimeObject(mixed $input): ?DateTimeInterface
|
|
{
|
|
if ($input instanceof DateTimeInterface) {
|
|
return $input;
|
|
}
|
|
|
|
if ($this->format === null) {
|
|
try {
|
|
return new DateTimeImmutable((string) $input);
|
|
} catch (Throwable) {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
$format = $this->getExceptionalFormats()[$this->format] ?? $this->format;
|
|
$dateTime = DateTimeImmutable::createFromFormat($format, (string) $input);
|
|
if ($dateTime === false) {
|
|
return null;
|
|
}
|
|
|
|
return $dateTime;
|
|
}
|
|
}
|