334 lines
		
	
	
		
			8.8 KiB
		
	
	
	
		
			PHP
		
	
	
			
		
		
	
	
			334 lines
		
	
	
		
			8.8 KiB
		
	
	
	
		
			PHP
		
	
	
<?php
 | 
						|
 | 
						|
/*
 | 
						|
 * This file is part of the Symfony package.
 | 
						|
 *
 | 
						|
 * (c) Fabien Potencier <fabien@symfony.com>
 | 
						|
 *
 | 
						|
 * For the full copyright and license information, please view the LICENSE
 | 
						|
 * file that was distributed with this source code.
 | 
						|
 */
 | 
						|
 | 
						|
namespace Symfony\Component\Cache\Adapter;
 | 
						|
 | 
						|
use Psr\Cache\CacheItemInterface;
 | 
						|
use Psr\Cache\CacheItemPoolInterface;
 | 
						|
use Symfony\Component\Cache\CacheItem;
 | 
						|
use Symfony\Component\Cache\Exception\InvalidArgumentException;
 | 
						|
use Symfony\Component\Cache\PruneableInterface;
 | 
						|
use Symfony\Component\Cache\ResettableInterface;
 | 
						|
use Symfony\Component\Cache\Traits\ContractsTrait;
 | 
						|
use Symfony\Contracts\Cache\CacheInterface;
 | 
						|
use Symfony\Contracts\Service\ResetInterface;
 | 
						|
 | 
						|
/**
 | 
						|
 * Chains several adapters together.
 | 
						|
 *
 | 
						|
 * Cached items are fetched from the first adapter having them in its data store.
 | 
						|
 * They are saved and deleted in all adapters at once.
 | 
						|
 *
 | 
						|
 * @author Kévin Dunglas <dunglas@gmail.com>
 | 
						|
 */
 | 
						|
class ChainAdapter implements AdapterInterface, CacheInterface, PruneableInterface, ResettableInterface
 | 
						|
{
 | 
						|
    use ContractsTrait;
 | 
						|
 | 
						|
    private $adapters = [];
 | 
						|
    private $adapterCount;
 | 
						|
    private $syncItem;
 | 
						|
 | 
						|
    /**
 | 
						|
     * @param CacheItemPoolInterface[] $adapters        The ordered list of adapters used to fetch cached items
 | 
						|
     * @param int                      $defaultLifetime The default lifetime of items propagated from lower adapters to upper ones
 | 
						|
     */
 | 
						|
    public function __construct(array $adapters, int $defaultLifetime = 0)
 | 
						|
    {
 | 
						|
        if (!$adapters) {
 | 
						|
            throw new InvalidArgumentException('At least one adapter must be specified.');
 | 
						|
        }
 | 
						|
 | 
						|
        foreach ($adapters as $adapter) {
 | 
						|
            if (!$adapter instanceof CacheItemPoolInterface) {
 | 
						|
                throw new InvalidArgumentException(sprintf('The class "%s" does not implement the "%s" interface.', \get_class($adapter), CacheItemPoolInterface::class));
 | 
						|
            }
 | 
						|
            if (\in_array(\PHP_SAPI, ['cli', 'phpdbg'], true) && $adapter instanceof ApcuAdapter && !filter_var(ini_get('apc.enable_cli'), \FILTER_VALIDATE_BOOLEAN)) {
 | 
						|
                continue; // skip putting APCu in the chain when the backend is disabled
 | 
						|
            }
 | 
						|
 | 
						|
            if ($adapter instanceof AdapterInterface) {
 | 
						|
                $this->adapters[] = $adapter;
 | 
						|
            } else {
 | 
						|
                $this->adapters[] = new ProxyAdapter($adapter);
 | 
						|
            }
 | 
						|
        }
 | 
						|
        $this->adapterCount = \count($this->adapters);
 | 
						|
 | 
						|
        $this->syncItem = \Closure::bind(
 | 
						|
            static function ($sourceItem, $item, $sourceMetadata = null) use ($defaultLifetime) {
 | 
						|
                $sourceItem->isTaggable = false;
 | 
						|
                $sourceMetadata = $sourceMetadata ?? $sourceItem->metadata;
 | 
						|
                unset($sourceMetadata[CacheItem::METADATA_TAGS]);
 | 
						|
 | 
						|
                $item->value = $sourceItem->value;
 | 
						|
                $item->isHit = $sourceItem->isHit;
 | 
						|
                $item->metadata = $item->newMetadata = $sourceItem->metadata = $sourceMetadata;
 | 
						|
 | 
						|
                if (isset($item->metadata[CacheItem::METADATA_EXPIRY])) {
 | 
						|
                    $item->expiresAt(\DateTime::createFromFormat('U.u', sprintf('%.6F', $item->metadata[CacheItem::METADATA_EXPIRY])));
 | 
						|
                } elseif (0 < $defaultLifetime) {
 | 
						|
                    $item->expiresAfter($defaultLifetime);
 | 
						|
                }
 | 
						|
 | 
						|
                return $item;
 | 
						|
            },
 | 
						|
            null,
 | 
						|
            CacheItem::class
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * {@inheritdoc}
 | 
						|
     */
 | 
						|
    public function get(string $key, callable $callback, float $beta = null, array &$metadata = null)
 | 
						|
    {
 | 
						|
        $lastItem = null;
 | 
						|
        $i = 0;
 | 
						|
        $wrap = function (CacheItem $item = null) use ($key, $callback, $beta, &$wrap, &$i, &$lastItem, &$metadata) {
 | 
						|
            $adapter = $this->adapters[$i];
 | 
						|
            if (isset($this->adapters[++$i])) {
 | 
						|
                $callback = $wrap;
 | 
						|
                $beta = \INF === $beta ? \INF : 0;
 | 
						|
            }
 | 
						|
            if ($adapter instanceof CacheInterface) {
 | 
						|
                $value = $adapter->get($key, $callback, $beta, $metadata);
 | 
						|
            } else {
 | 
						|
                $value = $this->doGet($adapter, $key, $callback, $beta, $metadata);
 | 
						|
            }
 | 
						|
            if (null !== $item) {
 | 
						|
                ($this->syncItem)($lastItem = $lastItem ?? $item, $item, $metadata);
 | 
						|
            }
 | 
						|
 | 
						|
            return $value;
 | 
						|
        };
 | 
						|
 | 
						|
        return $wrap();
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * {@inheritdoc}
 | 
						|
     */
 | 
						|
    public function getItem($key)
 | 
						|
    {
 | 
						|
        $syncItem = $this->syncItem;
 | 
						|
        $misses = [];
 | 
						|
 | 
						|
        foreach ($this->adapters as $i => $adapter) {
 | 
						|
            $item = $adapter->getItem($key);
 | 
						|
 | 
						|
            if ($item->isHit()) {
 | 
						|
                while (0 <= --$i) {
 | 
						|
                    $this->adapters[$i]->save($syncItem($item, $misses[$i]));
 | 
						|
                }
 | 
						|
 | 
						|
                return $item;
 | 
						|
            }
 | 
						|
 | 
						|
            $misses[$i] = $item;
 | 
						|
        }
 | 
						|
 | 
						|
        return $item;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * {@inheritdoc}
 | 
						|
     */
 | 
						|
    public function getItems(array $keys = [])
 | 
						|
    {
 | 
						|
        return $this->generateItems($this->adapters[0]->getItems($keys), 0);
 | 
						|
    }
 | 
						|
 | 
						|
    private function generateItems(iterable $items, int $adapterIndex)
 | 
						|
    {
 | 
						|
        $missing = [];
 | 
						|
        $misses = [];
 | 
						|
        $nextAdapterIndex = $adapterIndex + 1;
 | 
						|
        $nextAdapter = $this->adapters[$nextAdapterIndex] ?? null;
 | 
						|
 | 
						|
        foreach ($items as $k => $item) {
 | 
						|
            if (!$nextAdapter || $item->isHit()) {
 | 
						|
                yield $k => $item;
 | 
						|
            } else {
 | 
						|
                $missing[] = $k;
 | 
						|
                $misses[$k] = $item;
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        if ($missing) {
 | 
						|
            $syncItem = $this->syncItem;
 | 
						|
            $adapter = $this->adapters[$adapterIndex];
 | 
						|
            $items = $this->generateItems($nextAdapter->getItems($missing), $nextAdapterIndex);
 | 
						|
 | 
						|
            foreach ($items as $k => $item) {
 | 
						|
                if ($item->isHit()) {
 | 
						|
                    $adapter->save($syncItem($item, $misses[$k]));
 | 
						|
                }
 | 
						|
 | 
						|
                yield $k => $item;
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * {@inheritdoc}
 | 
						|
     *
 | 
						|
     * @return bool
 | 
						|
     */
 | 
						|
    public function hasItem($key)
 | 
						|
    {
 | 
						|
        foreach ($this->adapters as $adapter) {
 | 
						|
            if ($adapter->hasItem($key)) {
 | 
						|
                return true;
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        return false;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * {@inheritdoc}
 | 
						|
     *
 | 
						|
     * @param string $prefix
 | 
						|
     *
 | 
						|
     * @return bool
 | 
						|
     */
 | 
						|
    public function clear(/*string $prefix = ''*/)
 | 
						|
    {
 | 
						|
        $prefix = 0 < \func_num_args() ? (string) func_get_arg(0) : '';
 | 
						|
        $cleared = true;
 | 
						|
        $i = $this->adapterCount;
 | 
						|
 | 
						|
        while ($i--) {
 | 
						|
            if ($this->adapters[$i] instanceof AdapterInterface) {
 | 
						|
                $cleared = $this->adapters[$i]->clear($prefix) && $cleared;
 | 
						|
            } else {
 | 
						|
                $cleared = $this->adapters[$i]->clear() && $cleared;
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        return $cleared;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * {@inheritdoc}
 | 
						|
     *
 | 
						|
     * @return bool
 | 
						|
     */
 | 
						|
    public function deleteItem($key)
 | 
						|
    {
 | 
						|
        $deleted = true;
 | 
						|
        $i = $this->adapterCount;
 | 
						|
 | 
						|
        while ($i--) {
 | 
						|
            $deleted = $this->adapters[$i]->deleteItem($key) && $deleted;
 | 
						|
        }
 | 
						|
 | 
						|
        return $deleted;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * {@inheritdoc}
 | 
						|
     *
 | 
						|
     * @return bool
 | 
						|
     */
 | 
						|
    public function deleteItems(array $keys)
 | 
						|
    {
 | 
						|
        $deleted = true;
 | 
						|
        $i = $this->adapterCount;
 | 
						|
 | 
						|
        while ($i--) {
 | 
						|
            $deleted = $this->adapters[$i]->deleteItems($keys) && $deleted;
 | 
						|
        }
 | 
						|
 | 
						|
        return $deleted;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * {@inheritdoc}
 | 
						|
     *
 | 
						|
     * @return bool
 | 
						|
     */
 | 
						|
    public function save(CacheItemInterface $item)
 | 
						|
    {
 | 
						|
        $saved = true;
 | 
						|
        $i = $this->adapterCount;
 | 
						|
 | 
						|
        while ($i--) {
 | 
						|
            $saved = $this->adapters[$i]->save($item) && $saved;
 | 
						|
        }
 | 
						|
 | 
						|
        return $saved;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * {@inheritdoc}
 | 
						|
     *
 | 
						|
     * @return bool
 | 
						|
     */
 | 
						|
    public function saveDeferred(CacheItemInterface $item)
 | 
						|
    {
 | 
						|
        $saved = true;
 | 
						|
        $i = $this->adapterCount;
 | 
						|
 | 
						|
        while ($i--) {
 | 
						|
            $saved = $this->adapters[$i]->saveDeferred($item) && $saved;
 | 
						|
        }
 | 
						|
 | 
						|
        return $saved;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * {@inheritdoc}
 | 
						|
     *
 | 
						|
     * @return bool
 | 
						|
     */
 | 
						|
    public function commit()
 | 
						|
    {
 | 
						|
        $committed = true;
 | 
						|
        $i = $this->adapterCount;
 | 
						|
 | 
						|
        while ($i--) {
 | 
						|
            $committed = $this->adapters[$i]->commit() && $committed;
 | 
						|
        }
 | 
						|
 | 
						|
        return $committed;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * {@inheritdoc}
 | 
						|
     */
 | 
						|
    public function prune()
 | 
						|
    {
 | 
						|
        $pruned = true;
 | 
						|
 | 
						|
        foreach ($this->adapters as $adapter) {
 | 
						|
            if ($adapter instanceof PruneableInterface) {
 | 
						|
                $pruned = $adapter->prune() && $pruned;
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        return $pruned;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * {@inheritdoc}
 | 
						|
     */
 | 
						|
    public function reset()
 | 
						|
    {
 | 
						|
        foreach ($this->adapters as $adapter) {
 | 
						|
            if ($adapter instanceof ResetInterface) {
 | 
						|
                $adapter->reset();
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 |