* * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Symfony\Component\Serializer\Normalizer; use Symfony\Component\Serializer\Exception\InvalidArgumentException; use Symfony\Component\Serializer\Exception\UnexpectedValueException; /** * Normalizes an instance of {@see \DateInterval} to an interval string. * Denormalizes an interval string to an instance of {@see \DateInterval}. * * @author Jérôme Parmentier */ class DateIntervalNormalizer implements NormalizerInterface, DenormalizerInterface, CacheableSupportsMethodInterface { public const FORMAT_KEY = 'dateinterval_format'; private $defaultContext = [ self::FORMAT_KEY => '%rP%yY%mM%dDT%hH%iM%sS', ]; public function __construct(array $defaultContext = []) { $this->defaultContext = array_merge($this->defaultContext, $defaultContext); } /** * {@inheritdoc} * * @throws InvalidArgumentException */ public function normalize(mixed $object, string $format = null, array $context = []): string { if (!$object instanceof \DateInterval) { throw new InvalidArgumentException('The object must be an instance of "\DateInterval".'); } return $object->format($context[self::FORMAT_KEY] ?? $this->defaultContext[self::FORMAT_KEY]); } /** * {@inheritdoc} */ public function supportsNormalization(mixed $data, string $format = null): bool { return $data instanceof \DateInterval; } /** * {@inheritdoc} */ public function hasCacheableSupportsMethod(): bool { return __CLASS__ === static::class; } /** * {@inheritdoc} * * @throws InvalidArgumentException * @throws UnexpectedValueException */ public function denormalize(mixed $data, string $type, string $format = null, array $context = []): \DateInterval { if (!\is_string($data)) { throw new InvalidArgumentException(sprintf('Data expected to be a string, "%s" given.', get_debug_type($data))); } if (!$this->isISO8601($data)) { throw new UnexpectedValueException('Expected a valid ISO 8601 interval string.'); } $dateIntervalFormat = $context[self::FORMAT_KEY] ?? $this->defaultContext[self::FORMAT_KEY]; $signPattern = ''; switch (substr($dateIntervalFormat, 0, 2)) { case '%R': $signPattern = '[-+]'; $dateIntervalFormat = substr($dateIntervalFormat, 2); break; case '%r': $signPattern = '-?'; $dateIntervalFormat = substr($dateIntervalFormat, 2); break; } $valuePattern = '/^'.$signPattern.preg_replace('/%([yYmMdDhHiIsSwW])(\w)/', '(?:(?P<$1>\d+)$2)?', preg_replace('/(T.*)$/', '($1)?', $dateIntervalFormat)).'$/'; if (!preg_match($valuePattern, $data)) { throw new UnexpectedValueException(sprintf('Value "%s" contains intervals not accepted by format "%s".', $data, $dateIntervalFormat)); } try { if ('-' === $data[0]) { $interval = new \DateInterval(substr($data, 1)); $interval->invert = 1; return $interval; } if ('+' === $data[0]) { return new \DateInterval(substr($data, 1)); } return new \DateInterval($data); } catch (\Exception $e) { throw new UnexpectedValueException($e->getMessage(), $e->getCode(), $e); } } /** * {@inheritdoc} */ public function supportsDenormalization(mixed $data, string $type, string $format = null): bool { return \DateInterval::class === $type; } private function isISO8601(string $string): bool { return preg_match('/^[\-+]?P(?=\w*(?:\d|%\w))(?:\d+Y|%[yY]Y)?(?:\d+M|%[mM]M)?(?:(?:\d+D|%[dD]D)|(?:\d+W|%[wW]W))?(?:T(?:\d+H|[hH]H)?(?:\d+M|[iI]M)?(?:\d+S|[sS]S)?)?$/', $string); } }