88 lines
2.8 KiB
PHP
88 lines
2.8 KiB
PHP
|
<?php
|
||
|
|
||
|
/*
|
||
|
* This file is part of the Symfony package.
|
||
|
*
|
||
|
* (c) Fabien Potencier <fabien@symfony.com>
|
||
|
*
|
||
|
* For the full copyright and license information, please view the LICENSE
|
||
|
* file that was distributed with this source code.
|
||
|
*/
|
||
|
|
||
|
namespace Symfony\Component\HttpKernel\Controller\ArgumentResolver;
|
||
|
|
||
|
use Psr\Clock\ClockInterface;
|
||
|
use Symfony\Component\HttpFoundation\Request;
|
||
|
use Symfony\Component\HttpKernel\Attribute\MapDateTime;
|
||
|
use Symfony\Component\HttpKernel\Controller\ValueResolverInterface;
|
||
|
use Symfony\Component\HttpKernel\ControllerMetadata\ArgumentMetadata;
|
||
|
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
|
||
|
|
||
|
/**
|
||
|
* Convert DateTime instances from request attribute variable.
|
||
|
*
|
||
|
* @author Benjamin Eberlei <kontakt@beberlei.de>
|
||
|
* @author Tim Goudriaan <tim@codedmonkey.com>
|
||
|
*/
|
||
|
final class DateTimeValueResolver implements ValueResolverInterface
|
||
|
{
|
||
|
public function __construct(
|
||
|
private readonly ?ClockInterface $clock = null,
|
||
|
) {
|
||
|
}
|
||
|
|
||
|
public function resolve(Request $request, ArgumentMetadata $argument): array
|
||
|
{
|
||
|
if (!is_a($argument->getType(), \DateTimeInterface::class, true) || !$request->attributes->has($argument->getName())) {
|
||
|
return [];
|
||
|
}
|
||
|
|
||
|
$value = $request->attributes->get($argument->getName());
|
||
|
$class = \DateTimeInterface::class === $argument->getType() ? \DateTimeImmutable::class : $argument->getType();
|
||
|
|
||
|
if (!$value) {
|
||
|
if ($argument->isNullable()) {
|
||
|
return [null];
|
||
|
}
|
||
|
if (!$this->clock) {
|
||
|
return [new $class()];
|
||
|
}
|
||
|
$value = $this->clock->now();
|
||
|
}
|
||
|
|
||
|
if ($value instanceof \DateTimeInterface) {
|
||
|
return [$value instanceof $class ? $value : $class::createFromInterface($value)];
|
||
|
}
|
||
|
|
||
|
$format = null;
|
||
|
|
||
|
if ($attributes = $argument->getAttributes(MapDateTime::class, ArgumentMetadata::IS_INSTANCEOF)) {
|
||
|
$attribute = $attributes[0];
|
||
|
$format = $attribute->format;
|
||
|
}
|
||
|
|
||
|
if (null !== $format) {
|
||
|
$date = $class::createFromFormat($format, $value, $this->clock?->now()->getTimeZone());
|
||
|
|
||
|
if (($class::getLastErrors() ?: ['warning_count' => 0])['warning_count']) {
|
||
|
$date = false;
|
||
|
}
|
||
|
} else {
|
||
|
if (false !== filter_var($value, \FILTER_VALIDATE_INT, ['options' => ['min_range' => 0]])) {
|
||
|
$value = '@'.$value;
|
||
|
}
|
||
|
try {
|
||
|
$date = new $class($value, $this->clock?->now()->getTimeZone());
|
||
|
} catch (\Exception) {
|
||
|
$date = false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (!$date) {
|
||
|
throw new NotFoundHttpException(sprintf('Invalid date given for parameter "%s".', $argument->getName()));
|
||
|
}
|
||
|
|
||
|
return [$date];
|
||
|
}
|
||
|
}
|