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();
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 |