Skip to content

Commit

Permalink
Merge pull request #398 from CPS-IT/feature/1.x/cacheable
Browse files Browse the repository at this point in the history
[FEATURE] Populate and read cacheable state of template path providers
  • Loading branch information
eliashaeussler authored Jan 15, 2025
2 parents 518fa53 + 18b8c59 commit 6546a4c
Show file tree
Hide file tree
Showing 6 changed files with 148 additions and 30 deletions.
5 changes: 5 additions & 0 deletions Classes/Renderer/Template/Path/GlobalPathProvider.php
Original file line number Diff line number Diff line change
Expand Up @@ -70,6 +70,11 @@ public function getTemplateRootPaths(): array
return $this->templateRootPaths;
}

public function isCacheable(): bool
{
return true;
}

public static function getPriority(): int
{
return 0;
Expand Down
2 changes: 2 additions & 0 deletions Classes/Renderer/Template/Path/PathProvider.php
Original file line number Diff line number Diff line change
Expand Up @@ -47,5 +47,7 @@ public function getPartialRootPaths(): array;
*/
public function getTemplateRootPaths(): array;

public function isCacheable(): bool;

public static function getPriority(): int;
}
5 changes: 5 additions & 0 deletions Classes/Renderer/Template/Path/TypoScriptPathProvider.php
Original file line number Diff line number Diff line change
Expand Up @@ -53,6 +53,11 @@ public function getTemplateRootPaths(): array
return $this->getViewConfiguration()[self::TEMPLATES] ?? [];
}

public function isCacheable(): bool
{
return true;
}

/**
* @return array{partialRootPaths?: array<int, string>, templateRootPaths?: array<int, string>}
*/
Expand Down
42 changes: 27 additions & 15 deletions Classes/Renderer/Template/TemplatePaths.php
Original file line number Diff line number Diff line change
Expand Up @@ -56,44 +56,56 @@ public function __construct(
*/
public function getPartialRootPaths(): array
{
if ($this->partialRootPaths === null) {
$this->partialRootPaths = $this->resolvePaths(
static fn(Path\PathProvider $pathProvider) => $pathProvider->getPartialRootPaths(),
);
}

return $this->partialRootPaths;
return $this->resolvePaths(
static fn(Path\PathProvider $pathProvider) => $pathProvider->getPartialRootPaths(),
$this->partialRootPaths,
);
}

/**
* @return array<int, string>
*/
public function getTemplateRootPaths(): array
{
if ($this->templateRootPaths === null) {
$this->templateRootPaths = $this->resolvePaths(
static fn(Path\PathProvider $pathProvider) => $pathProvider->getTemplateRootPaths(),
);
}

return $this->templateRootPaths;
return $this->resolvePaths(
static fn(Path\PathProvider $pathProvider) => $pathProvider->getTemplateRootPaths(),
$this->templateRootPaths,
);
}

/**
* @param callable(Path\PathProvider): array<int, string> $mapFunction
* @param array<int, string>|null $rootPaths
* @return array<int, string>
*/
private function resolvePaths(callable $mapFunction): array
private function resolvePaths(callable $mapFunction, ?array &$rootPaths): array
{
// Early return if root paths are already resolved and cached
if ($rootPaths !== null) {
return $rootPaths;
}

$cacheable = true;
$paths = [];

// Resolve root paths from path providers
foreach ($this->pathProviders as $pathProvider) {
\array_unshift($paths, $mapFunction($pathProvider));

if (!$pathProvider->isCacheable()) {
$cacheable = false;
}
}

// Merge and sort all root paths
$mergedPaths = array_replace(...$paths);
ksort($mergedPaths);

// Cache root paths if possible
if ($cacheable) {
$rootPaths = $mergedPaths;
}

return $mergedPaths;
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,7 @@ final class DummyPathProvider implements Renderer\Template\Path\PathProvider
public function __construct(
public array $templateRootPaths = [],
public array $partialRootPaths = [],
public bool $cacheable = true,
) {}

public function getPartialRootPaths(): array
Expand All @@ -53,6 +54,11 @@ public function getTemplateRootPaths(): array
return $this->templateRootPaths;
}

public function isCacheable(): bool
{
return $this->cacheable;
}

public static function getPriority(): int
{
return 10;
Expand Down
118 changes: 103 additions & 15 deletions Tests/Unit/Renderer/Template/TemplatePathsTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -53,6 +53,65 @@ protected function setUp(): void
]);
}

/**
* @param array<int, string> $partialRootPaths
* @param array<int, string> $expected
*/
#[Framework\Attributes\Test]
#[Framework\Attributes\DataProvider('getPartialRootPathsMergesConfigurationFromPathProvidersDataProvider')]
public function getPartialRootPathsMergesConfigurationFromPathProviders(
array $partialRootPaths,
array $expected,
): void {
$this->pathProvider->partialRootPaths = $partialRootPaths;

self::assertSame($expected, $this->subject->getPartialRootPaths());
}

#[Framework\Attributes\Test]
public function getPartialRootPathsCachesRootPathsIfAllPathProvidersAreCacheable(): void
{
$this->pathProvider->partialRootPaths = [
20 => 'foo',
];

$expected = [
10 => dirname(__DIR__, 2) . '/Fixtures/Partials',
20 => 'foo',
];

self::assertSame($expected, $this->subject->getPartialRootPaths());

$this->pathProvider->partialRootPaths = [
20 => 'baz',
];

self::assertSame($expected, $this->subject->getPartialRootPaths());
}

#[Framework\Attributes\Test]
public function getPartialRootPathsDoesNotCacheRootPathsIfAnyPathProviderIsNonCacheable(): void
{
$this->pathProvider->cacheable = false;

$expected = [
10 => dirname(__DIR__, 2) . '/Fixtures/Partials',
];

self::assertSame($expected, $this->subject->getPartialRootPaths());

$this->pathProvider->partialRootPaths = [
20 => 'foo',
];

$expected = [
10 => dirname(__DIR__, 2) . '/Fixtures/Partials',
20 => 'foo',
];

self::assertSame($expected, $this->subject->getPartialRootPaths());
}

/**
* @param array<int, string> $templateRootPaths
* @param array<int, string> $expected
Expand All @@ -68,30 +127,59 @@ public function getTemplateRootPathsMergesConfigurationFromPathProviders(
self::assertSame($expected, $this->subject->getTemplateRootPaths());
}

/**
* @param array<int, string> $partialRootPaths
* @param array<int, string> $expected
*/
#[Framework\Attributes\Test]
#[Framework\Attributes\DataProvider('getPartialRootPathsMergesConfigurationFromPathProvidersDataProvider')]
public function getPartialRootPathsMergesConfigurationFromPathProviders(
array $partialRootPaths,
array $expected,
): void {
$this->pathProvider->partialRootPaths = $partialRootPaths;
public function getTemplateRootPathsCachesRootPathsIfAllPathProvidersAreCacheable(): void
{
$this->pathProvider->templateRootPaths = [
20 => 'foo',
];

self::assertSame($expected, $this->subject->getPartialRootPaths());
$expected = [
10 => dirname(__DIR__, 2) . '/Fixtures/Templates',
20 => 'foo',
];

self::assertSame($expected, $this->subject->getTemplateRootPaths());

$this->pathProvider->templateRootPaths = [
20 => 'baz',
];

self::assertSame($expected, $this->subject->getTemplateRootPaths());
}

#[Framework\Attributes\Test]
public function getTemplateRootPathsDoesNotCacheRootPathsIfAnyPathProviderIsNonCacheable(): void
{
$this->pathProvider->cacheable = false;

$expected = [
10 => dirname(__DIR__, 2) . '/Fixtures/Templates',
];

self::assertSame($expected, $this->subject->getTemplateRootPaths());

$this->pathProvider->templateRootPaths = [
20 => 'foo',
];

$expected = [
10 => dirname(__DIR__, 2) . '/Fixtures/Templates',
20 => 'foo',
];

self::assertSame($expected, $this->subject->getTemplateRootPaths());
}

/**
* @return \Generator<string, array{array<int, string>, array<int, string>}>
*/
public static function getTemplateRootPathsMergesConfigurationFromPathProvidersDataProvider(): \Generator
public static function getPartialRootPathsMergesConfigurationFromPathProvidersDataProvider(): \Generator
{
yield 'no view configuration' => [
[],
[
10 => dirname(__DIR__, 2) . '/Fixtures/Templates',
10 => dirname(__DIR__, 2) . '/Fixtures/Partials',
],
];
yield 'view configuration with identical keys' => [
Expand All @@ -117,12 +205,12 @@ public static function getTemplateRootPathsMergesConfigurationFromPathProvidersD
/**
* @return \Generator<string, array{array<int, string>, array<int, string>}>
*/
public static function getPartialRootPathsMergesConfigurationFromPathProvidersDataProvider(): \Generator
public static function getTemplateRootPathsMergesConfigurationFromPathProvidersDataProvider(): \Generator
{
yield 'no view configuration' => [
[],
[
10 => dirname(__DIR__, 2) . '/Fixtures/Partials',
10 => dirname(__DIR__, 2) . '/Fixtures/Templates',
],
];
yield 'view configuration with identical keys' => [
Expand Down

0 comments on commit 6546a4c

Please sign in to comment.