Improve TYPO3 cache handling (#14)

There might be access to TSFE get_cache_timeout() prior rendering
events.
TYPO3 has a cache for timeout and won't re calculate again.
We therefore need to clear the cache if timeout would change.

That will lead to inconsistent cache information throughout a single
request. But the final cache timeout of the page will be correct. Other
parts might be longer, which probably is fine until they relate to the
events.

Relates: #10500
This commit is contained in:
Daniel Siepmann 2023-05-16 10:15:05 +02:00 committed by GitHub
parent 9bc0466e5d
commit 0fc2668d17
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
7 changed files with 174 additions and 48 deletions

View file

@ -25,6 +25,9 @@ namespace Wrm\Events\Caching;
use DateTime; use DateTime;
use DateTimeImmutable; use DateTimeImmutable;
use TYPO3\CMS\Core\Cache\CacheManager;
use TYPO3\CMS\Core\Cache\Frontend\FrontendInterface;
use TYPO3\CMS\Core\Context\Context;
use TYPO3\CMS\Core\SingletonInterface; use TYPO3\CMS\Core\SingletonInterface;
use Wrm\Events\Events\Controller\DateListVariables; use Wrm\Events\Events\Controller\DateListVariables;
@ -37,26 +40,45 @@ use Wrm\Events\Events\Controller\DateListVariables;
class PageCacheTimeout implements SingletonInterface class PageCacheTimeout implements SingletonInterface
{ {
/** /**
* @var int * @var null|DateTimeImmutable
*/ */
private $earliestTimeout = 0; private $endOfEvent = null;
/**
* @var FrontendInterface
*/
private $runtimeCache;
/**
* @var Context
*/
private $context;
public function __construct(
CacheManager $cacheManager,
Context $context
) {
$this->runtimeCache = $cacheManager->getCache('runtime');
$this->context = $context;
}
public function calculateCacheTimout( public function calculateCacheTimout(
array $parameters array $parameters
): int { ): int {
$timeout = $parameters['cacheTimeout']; $typo3Timeout = $parameters['cacheTimeout'];
$ourTimeout = $this->getTimeout();
if ($this->earliestTimeout <= 0) { if ($ourTimeout === null) {
return $timeout; return $typo3Timeout;
} }
return min($timeout, $this->earliestTimeout); return min($typo3Timeout, $ourTimeout);
} }
public function trackDates(DateListVariables $event): void public function trackDates(DateListVariables $event): void
{ {
if ($event->getDemand()->shouldShowFromMidnight()) { if ($event->getDemand()->shouldShowFromMidnight()) {
$this->updateTimeout((int) (new DateTimeImmutable('tomorrow midnight'))->format('U')); $this->updateTimeout((new DateTimeImmutable('tomorrow midnight')));
return; return;
} }
@ -66,23 +88,36 @@ class PageCacheTimeout implements SingletonInterface
continue; continue;
} }
$this->updateTimeout((int)DateTimeImmutable::createFromMutable($endDate)->format('U')); $this->updateTimeout(DateTimeImmutable::createFromMutable($endDate));
} }
} }
private function updateTimeout(int $timestamp): void private function updateTimeout(DateTimeImmutable $end): void
{ {
$newTimeout = $timestamp - time(); $now = new DateTimeImmutable();
if ($newTimeout <= 0) {
if (
$end <= $now
|| (
$this->endOfEvent instanceof DateTimeImmutable
&& $this->endOfEvent >= $end
)
) {
return; return;
} }
if ($this->earliestTimeout === 0) { $this->runtimeCache->remove('core-tslib_fe-get_cache_timeout');
$this->earliestTimeout = $newTimeout; $this->endOfEvent = $end;
return; }
private function getTimeout(): ?int
{
if (!$this->endOfEvent instanceof DateTimeImmutable) {
return null;
} }
$this->earliestTimeout = min($this->earliestTimeout, $newTimeout); $executionTime = $this->context->getPropertyFromAspect('date', 'timestamp');
return ((int) $this->endOfEvent->format('U')) - $executionTime;
} }
public static function register(): void public static function register(): void

View file

@ -24,20 +24,18 @@ declare(strict_types=1);
namespace Wrm\Events\Tests\Functional\Frontend; namespace Wrm\Events\Tests\Functional\Frontend;
use Codappix\Typo3PhpDatasets\PhpDataSet; use Codappix\Typo3PhpDatasets\PhpDataSet;
use Codappix\Typo3PhpDatasets\TestingFramework;
use DateTimeImmutable; use DateTimeImmutable;
use DateTimeZone; use DateTimeZone;
use Psr\Http\Message\ResponseInterface; use Psr\Http\Message\ResponseInterface;
use TYPO3\CMS\Core\TypoScript\TemplateService;
use TYPO3\TestingFramework\Core\Functional\Framework\Frontend\InternalRequest; use TYPO3\TestingFramework\Core\Functional\Framework\Frontend\InternalRequest;
use Wrm\Events\Tests\Functional\AbstractFunctionalTestCase; use TYPO3\TestingFramework\Core\Functional\Framework\Frontend\Internal\TypoScriptInstruction;
/** /**
* @covers \Wrm\Events\Caching\PageCacheTimeout * @covers \Wrm\Events\Caching\PageCacheTimeout
*/ */
class CacheTest extends AbstractFunctionalTestCase class CacheTest extends AbstractTestCase
{ {
use TestingFramework;
protected $testExtensionsToLoad = [ protected $testExtensionsToLoad = [
'typo3conf/ext/events', 'typo3conf/ext/events',
'typo3conf/ext/events/Tests/Functional/Frontend/Fixtures/Extensions/example', 'typo3conf/ext/events/Tests/Functional/Frontend/Fixtures/Extensions/example',
@ -71,9 +69,7 @@ class CacheTest extends AbstractFunctionalTestCase
*/ */
public function setupReturnsSystemDefaults(): void public function setupReturnsSystemDefaults(): void
{ {
$request = new InternalRequest(); $response = $this->executeFrontendRequest($this->getRequestWithSleep());
$request = $request->withPageId(1);
$response = $this->executeFrontendRequest($request);
self::assertSame(200, $response->getStatusCode()); self::assertSame(200, $response->getStatusCode());
self::assertSame('max-age=86400', $response->getHeaderLine('Cache-Control')); self::assertSame('max-age=86400', $response->getHeaderLine('Cache-Control'));
@ -101,9 +97,7 @@ class CacheTest extends AbstractFunctionalTestCase
], ],
]); ]);
$request = new InternalRequest(); $response = $this->executeFrontendRequest($this->getRequestWithSleep());
$request = $request->withPageId(1);
$response = $this->executeFrontendRequest($request);
self::assertSame(200, $response->getStatusCode()); self::assertSame(200, $response->getStatusCode());
self::assertSame('max-age=86400', $response->getHeaderLine('Cache-Control')); self::assertSame('max-age=86400', $response->getHeaderLine('Cache-Control'));
@ -115,6 +109,8 @@ class CacheTest extends AbstractFunctionalTestCase
*/ */
public function setupReturnsEarlierIfEventsChangeBeforeSystemDefault(): void public function setupReturnsEarlierIfEventsChangeBeforeSystemDefault(): void
{ {
$end = (new DateTimeImmutable('tomorrow midnight', new DateTimeZone('UTC')))->modify('+2 hours');
(new PhpDataSet())->import([ (new PhpDataSet())->import([
'tx_events_domain_model_event' => [ 'tx_events_domain_model_event' => [
[ [
@ -128,18 +124,15 @@ class CacheTest extends AbstractFunctionalTestCase
'pid' => '2', 'pid' => '2',
'event' => '1', 'event' => '1',
'start' => time(), 'start' => time(),
'end' => time() + 50, 'end' => $end->format('U'),
], ],
], ],
]); ]);
$request = new InternalRequest(); $response = $this->executeFrontendRequest($this->getRequestWithSleep());
$request = $request->withPageId(1);
$response = $this->executeFrontendRequest($request);
self::assertSame(200, $response->getStatusCode()); self::assertSame(200, $response->getStatusCode());
self::assertMaxAge(50, $response); self::assertCacheHeaders($end, $response);
self::assertSame('public', $response->getHeaderLine('Pragma'));
} }
/** /**
@ -147,6 +140,8 @@ class CacheTest extends AbstractFunctionalTestCase
*/ */
public function setupReturnsMidnightIfConfigured(): void public function setupReturnsMidnightIfConfigured(): void
{ {
$midnight = (new DateTimeImmutable('tomorrow midnight', new DateTimeZone('UTC')));
(new PhpDataSet())->import([ (new PhpDataSet())->import([
'tx_events_domain_model_event' => [ 'tx_events_domain_model_event' => [
[ [
@ -171,27 +166,31 @@ class CacheTest extends AbstractFunctionalTestCase
], ],
])); ]));
$request = new InternalRequest(); $response = $this->executeFrontendRequest($this->getRequestWithSleep());
$request = $request->withPageId(1);
$response = $this->executeFrontendRequest($request);
self::assertSame(200, $response->getStatusCode()); self::assertSame(200, $response->getStatusCode());
$midnight = (int) (new DateTimeImmutable('tomorrow midnight', new DateTimeZone('UTC')))->format('U'); self::assertCacheHeaders($midnight, $response);
$age = $midnight - time();
self::assertMaxAge($age, $response);
self::assertSame('public', $response->getHeaderLine('Pragma')); self::assertSame('public', $response->getHeaderLine('Pragma'));
} }
private static function assertMaxAge(int $age, ResponseInterface $response): void private static function assertCacheHeaders(DateTimeImmutable $end, ResponseInterface $response): void
{ {
[$prefix, $value] = explode('=', $response->getHeaderLine('Cache-Control')); self::assertSame('public', $response->getHeaderLine('Pragma'));
$expectedExpires = $end
->setTimezone(new DateTimeZone('GMT'))
->format(DateTimeImmutable::RFC7231)
;
self::assertSame($expectedExpires, $response->getHeaderLine('Expires'));
[$prefix, $value] = explode('=', $response->getHeaderLine('Cache-Control'));
self::assertSame('max-age', $prefix); self::assertSame('max-age', $prefix);
// We might be one sec off due to how fast code is executed, so add a small offset // We might be seconds off due to our created offset within the rendering.
$value = (int)$value; $value = (int)$value;
self::assertLessThanOrEqual($age + 1, $value, 'Max age of cached response is higher than expected.'); $age = ((int) $end->format('U')) - time();
self::assertGreaterThanOrEqual($age - 1, $value, 'Max age of cached response is less than expected.'); self::assertLessThanOrEqual($age + 3, $value, 'Max age of cached response is higher than expected.');
self::assertGreaterThanOrEqual($age - 3, $value, 'Max age of cached response is less than expected.');
} }
private function getTypoScriptFiles(): array private function getTypoScriptFiles(): array
@ -207,4 +206,24 @@ class CacheTest extends AbstractFunctionalTestCase
], ],
]; ];
} }
private function getRequestWithSleep(): InternalRequest
{
$request = new InternalRequest();
$request = $request->withPageId(1);
$request = $request->withInstructions([
$this->getTypoScriptInstruction()
->withTypoScript([
'page.' => [
'30.' => [
'userFunc.' => [
'sleep' => '2',
],
],
],
])
]);
return $request;
}
} }

View file

@ -25,12 +25,11 @@ namespace Wrm\Events\Tests\Functional\Frontend;
use TYPO3\TestingFramework\Core\Functional\Framework\Frontend\InternalRequest; use TYPO3\TestingFramework\Core\Functional\Framework\Frontend\InternalRequest;
use Wrm\Events\Frontend\Dates; use Wrm\Events\Frontend\Dates;
use Wrm\Events\Tests\Functional\AbstractFunctionalTestCase;
/** /**
* @covers \Wrm\Events\Frontend\Dates * @covers \Wrm\Events\Frontend\Dates
*/ */
class DatesTest extends AbstractFunctionalTestCase class DatesTest extends AbstractTestCase
{ {
protected $testExtensionsToLoad = [ protected $testExtensionsToLoad = [
'typo3conf/ext/events', 'typo3conf/ext/events',

View file

@ -5,13 +5,12 @@ declare(strict_types=1);
namespace Wrm\Events\Tests\Functional\Frontend; namespace Wrm\Events\Tests\Functional\Frontend;
use TYPO3\TestingFramework\Core\Functional\Framework\Frontend\InternalRequest; use TYPO3\TestingFramework\Core\Functional\Framework\Frontend\InternalRequest;
use Wrm\Events\Tests\Functional\AbstractFunctionalTestCase;
/** /**
* @covers \Wrm\Events\Controller\DateController * @covers \Wrm\Events\Controller\DateController
* @covers \Wrm\Events\Domain\Repository\DateRepository * @covers \Wrm\Events\Domain\Repository\DateRepository
*/ */
class FilterTest extends AbstractFunctionalTestCase class FilterTest extends AbstractTestCase
{ {
protected $testExtensionsToLoad = [ protected $testExtensionsToLoad = [
'typo3conf/ext/events', 'typo3conf/ext/events',

View file

@ -0,0 +1,52 @@
<?php
declare(strict_types=1);
/*
* Copyright (C) 2023 Daniel Siepmann <coding@daniel-siepmann.de>
*
* 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 Wrm\EventsExample;
use TYPO3\CMS\Frontend\ContentObject\ContentObjectRenderer;
use TYPO3\CMS\Frontend\Controller\TypoScriptFrontendController;
class UserFunc
{
/**
* @var ContentObjectRenderer
*/
public $cObj;
public function accessTsfeTimeout(): string
{
return 'get_cache_timeout: ' . $this->getTsfe()->get_cache_timeout();
}
public function sleep(string $content, array $configuration): string
{
$sleep = (int) $this->cObj->stdWrapValue('sleep', $configuration['userFunc.'], 0);
sleep($sleep);
return 'Sleep for ' . $sleep . ' seconds';
}
public function getTsfe(): TypoScriptFrontendController
{
return $GLOBALS['TSFE'];
}
}

View file

@ -5,6 +5,11 @@
"require": { "require": {
"wrm/events": "*" "wrm/events": "*"
}, },
"autoload": {
"psr-4": {
"Wrm\\EventsExample\\": "Classes/"
}
},
"extra": { "extra": {
"typo3/cms": { "typo3/cms": {
"extension-key": "example" "extension-key": "example"

View file

@ -6,7 +6,24 @@ config {
page = PAGE page = PAGE
page { page {
10 < styles.content.get 10 = USER
10 {
// Simulates foreign access prior our rendering.
// TYPO3 has an internal cache in order to not recalculate timeout.
userFunc = Wrm\EventsExample\UserFunc->accessTsfeTimeout
}
20 < styles.content.get
30 = USER
30 {
// Simulates further long running rendering.
// In order to test that our ttl is calculated as expected.
userFunc = Wrm\EventsExample\UserFunc->sleep
userFunc {
sleep = 0
}
}
} }
plugin.tx_events { plugin.tx_events {