mirror of
https://github.com/werkraum-media/events.git
synced 2024-11-22 04:36:10 +01:00
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:
parent
9bc0466e5d
commit
0fc2668d17
7 changed files with 174 additions and 48 deletions
|
@ -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;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->earliestTimeout = min($this->earliestTimeout, $newTimeout);
|
private function getTimeout(): ?int
|
||||||
|
{
|
||||||
|
if (!$this->endOfEvent instanceof DateTimeImmutable) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$executionTime = $this->context->getPropertyFromAspect('date', 'timestamp');
|
||||||
|
return ((int) $this->endOfEvent->format('U')) - $executionTime;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function register(): void
|
public static function register(): void
|
||||||
|
|
|
@ -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;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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',
|
||||||
|
|
|
@ -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',
|
||||||
|
|
|
@ -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'];
|
||||||
|
}
|
||||||
|
}
|
|
@ -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"
|
||||||
|
|
|
@ -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 {
|
||||||
|
|
Loading…
Reference in a new issue