Files
michaelschiemer/src/Framework/View/Caching/CacheManager.php
Michael Schiemer 5050c7d73a docs: consolidate documentation into organized structure
- Move 12 markdown files from root to docs/ subdirectories
- Organize documentation by category:
  • docs/troubleshooting/ (1 file)  - Technical troubleshooting guides
  • docs/deployment/      (4 files) - Deployment and security documentation
  • docs/guides/          (3 files) - Feature-specific guides
  • docs/planning/        (4 files) - Planning and improvement proposals

Root directory cleanup:
- Reduced from 16 to 4 markdown files in root
- Only essential project files remain:
  • CLAUDE.md (AI instructions)
  • README.md (Main project readme)
  • CLEANUP_PLAN.md (Current cleanup plan)
  • SRC_STRUCTURE_IMPROVEMENTS.md (Structure improvements)

This improves:
 Documentation discoverability
 Logical organization by purpose
 Clean root directory
 Better maintainability
2025-10-05 11:05:04 +02:00

165 lines
5.8 KiB
PHP

<?php
declare(strict_types=1);
namespace App\Framework\View\Caching;
use App\Framework\Cache\Cache;
use App\Framework\Cache\CacheItem;
use App\Framework\View\Caching\Analysis\CacheStrategy;
use App\Framework\View\Caching\Analysis\TemplateAnalysis;
use App\Framework\View\Caching\Analysis\TemplateAnalyzer;
use App\Framework\View\Caching\Strategies\ComponentCacheStrategy;
use App\Framework\View\Caching\Strategies\FragmentCacheStrategy;
use App\Framework\View\Caching\Strategies\FullPageCacheStrategy;
use App\Framework\View\Caching\Strategies\NoCacheStrategy;
use App\Framework\View\Caching\Strategies\ViewCacheStrategy;
use App\Framework\View\Exceptions\TemplateCacheException;
class CacheManager
{
private array $strategies = [];
private ?TemplateAnalysis $lastAnalysis = null;
public function __construct(
private Cache $cache,
private TemplateAnalyzer $analyzer,
private FragmentCache $fragmentCache,
private array $strategyMapping = []
) {
$this->initializeStrategies();
}
public function render(TemplateContext $context, callable $renderer): string
{
// DEBUG: Log cache analysis for routes template
if ($context->template === 'routes') {
error_log("CacheManager::render - Template: routes");
}
// 1. Template analysieren für optimale Strategy
$analysis = $this->analyzer->analyze($context->template);
$this->lastAnalysis = $analysis;
// DEBUG: Log analysis results for routes template
if ($context->template === 'routes') {
error_log("CacheManager::render - Strategy: " . $analysis->recommendedStrategy->name);
error_log("CacheManager::render - Cacheable: " . ($analysis->cacheability->isCacheable() ? 'YES' : 'NO'));
error_log("CacheManager::render - Static ratio: " . $analysis->cacheability->staticContentRatio);
}
// 2. Passende Strategy auswählen
$strategy = $this->selectStrategy($analysis);
if (! $strategy->shouldCache($context)) {
$content = $renderer();
if (! is_string($content)) {
throw TemplateCacheException::invalidCacheFormat(
'non_cached_content',
'string',
get_debug_type($content)
);
}
return $content;
}
// 3. Cache-Key generieren
$cacheKey = $strategy->generateKey($context);
// 4. Cache-Lookup
$result = $this->cache->get($cacheKey);
$cached = $result->getItem($cacheKey);
if ($cached->isHit && is_string($cached->value)) {
// DEBUG: Log cache hit for routes template
if ($context->template === 'routes') {
error_log("CacheManager::render - CACHE HIT! Returning cached content (length: " . strlen($cached->value) . ")");
error_log("CacheManager::render - First 200 chars: " . substr($cached->value, 0, 200));
}
return $cached->value;
}
// 5. Rendern und cachen
if ($context->template === 'routes') {
error_log("CacheManager::render - CACHE MISS! Calling renderer()");
}
$content = $renderer();
if (! is_string($content)) {
throw TemplateCacheException::invalidCacheFormat(
$cacheKey,
'string',
get_debug_type($content)
);
}
$ttl = $strategy->getTtl($context);
$this->cache->set(CacheItem::forSet($cacheKey, $content, \App\Framework\Core\ValueObjects\Duration::fromSeconds($ttl)));
return $content;
}
public function invalidateTemplate(string $template): int
{
$invalidated = 0;
// Invalidiere alle Strategies für dieses Template
foreach ($this->strategies as $strategy) {
if ($strategy->canInvalidate($template)) {
// Pattern-basierte Invalidierung je nach Strategy
$pattern = $this->buildInvalidationPattern($strategy, $template);
$invalidated += $this->invalidateByPattern($pattern);
}
}
return $invalidated;
}
private function selectStrategy(TemplateAnalysis $analysis): ViewCacheStrategy
{
return match($analysis->recommendedStrategy) {
CacheStrategy::FULL_PAGE => $this->strategies['full_page'],
CacheStrategy::COMPONENT => $this->strategies['component'],
CacheStrategy::FRAGMENT => $this->strategies['fragment'],
CacheStrategy::USER_AWARE => $this->strategies['user_aware'],
default => $this->strategies['no_cache']
};
}
private function initializeStrategies(): void
{
$this->strategies = [
'full_page' => new FullPageCacheStrategy($this->cache),
'component' => new ComponentCacheStrategy($this->cache),
'fragment' => new FragmentCacheStrategy($this->cache),
'no_cache' => new NoCacheStrategy(),
];
}
private function buildInvalidationPattern(mixed $strategy, string $template): string
{
return match(get_class($strategy)) {
FullPageCacheStrategy::class => "page:{$template}:*",
ComponentCacheStrategy::class => "component:*{$template}*",
FragmentCacheStrategy::class => "fragment:{$template}:*",
default => "*{$template}*"
};
}
private function invalidateByPattern(string $pattern): int
{
// Vereinfachte Implementation - in Realität müsste das der Cache-Driver unterstützen
// Für jetzt: Cache komplett leeren bei Pattern-Match
$invalidated = 0;
if (str_contains($pattern, '*')) {
$this->cache->clear();
$invalidated = 1;
}
return $invalidated;
}
}