* * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA * 02110-1301, USA. */ namespace WerkraumMedia\Events\Caching; use DateTime; use DateTimeImmutable; use InvalidArgumentException; use TYPO3\CMS\Core\Cache\Frontend\FrontendInterface; use TYPO3\CMS\Core\Context\Context; use TYPO3\CMS\Core\SingletonInterface; use TYPO3\CMS\Frontend\Event\ModifyCacheLifetimeForPageEvent; use WerkraumMedia\Events\Domain\Model\Date; use WerkraumMedia\Events\Events\Controller\DateListVariables; /** * Teaches TYPO3 to set proper timeout for page cache. * * Takes timings of rendered dates into account. * Reduced timeout of page cache in case events (dates) might change before already calculated timeout. */ class PageCacheTimeout implements SingletonInterface { private ?DateTimeImmutable $timeout = null; public function __construct( private readonly FrontendInterface $runtimeCache, private readonly Context $context, ) { } public function modifyCacheLifetimeForPage(ModifyCacheLifetimeForPageEvent $event): void { $ourTimeout = $this->getTimeout(); if ($ourTimeout === null) { return; } $event->setCacheLifetime( min( $event->getCacheLifetime(), $ourTimeout ) ); } public function trackDates(DateListVariables $event): void { if ($event->getDemand()->shouldShowFromMidnight()) { $this->updateTimeout(new DateTimeImmutable('tomorrow midnight')); return; } if ($event->getDemand()->shouldShowUpcoming()) { $this->trackTimeoutByDate($event, static fn (Date $date) => $date->getStart()); } $this->trackTimeoutByDate($event, static fn (Date $date) => $date->getEnd()); } /** * @param callable $callback Receives Date as argument and should return DateTime to use as potential timeout. */ private function trackTimeoutByDate( DateListVariables $event, callable $callback ): void { foreach ($event->getDates() as $date) { $date = $callback($date); if (!$date instanceof DateTime) { continue; } $this->updateTimeout(DateTimeImmutable::createFromMutable($date)); } } private function updateTimeout(DateTimeImmutable $newTimeout): void { $now = $this->getExecution(); if ( $newTimeout <= $now || ( $this->timeout instanceof DateTimeImmutable && $this->timeout <= $newTimeout ) ) { return; } $id = $GLOBALS['TYPO3_REQUEST']->getAttribute('routing')->getPageId(); $this->runtimeCache->remove('cacheLifeTimeForPage_' . $id); $this->timeout = $newTimeout; } private function getTimeout(): ?int { if (!$this->timeout instanceof DateTimeImmutable) { return null; } return ((int)$this->timeout->format('U')) - ((int)$this->getExecution()->format('U')); } private function getExecution(): DateTimeImmutable { $execution = $this->context->getPropertyFromAspect('date', 'full'); if (!$execution instanceof DateTimeImmutable) { throw new InvalidArgumentException('Could not fetch DateTimeImmutable from context.', 1684740576); } return $execution; } }