171 lines
4.6 KiB
PHP

<?php
/**
*
* This file is part of phpFastCache.
*
* @license MIT License (MIT)
*
* For full copyright and license information, please see the docs/CREDITS.txt file.
*
* @author Georges.L (Geolim4) <contact@geolim4.com>
*
*/
declare(strict_types=1);
namespace Phpfastcache\Cluster\Drivers\MasterSlaveReplication;
use Phpfastcache\Cluster\ClusterPoolAbstract;
use Phpfastcache\Core\Pool\ExtendedCacheItemPoolInterface;
use Phpfastcache\Exceptions\PhpfastcacheDriverCheckException;
use Phpfastcache\Exceptions\PhpfastcacheDriverConnectException;
use Phpfastcache\Exceptions\PhpfastcacheExceptionInterface;
use Phpfastcache\Exceptions\PhpfastcacheInvalidArgumentException;
use Phpfastcache\Exceptions\PhpfastcacheInvalidConfigurationException;
use Phpfastcache\Exceptions\PhpfastcacheReplicationException;
use Psr\Cache\CacheItemInterface;
use ReflectionException;
/**
* Class MasterSlaveReplicationCluster
* @package Phpfastcache\Cluster\Drivers\MasterSlaveReplication
*/
class MasterSlaveReplicationCluster extends ClusterPoolAbstract
{
/**
* MasterSlaveReplicationCluster constructor.
* @param string $clusterName
* @param ExtendedCacheItemPoolInterface ...$driverPools
* @throws PhpfastcacheInvalidArgumentException
* @throws PhpfastcacheDriverCheckException
* @throws PhpfastcacheDriverConnectException
* @throws PhpfastcacheInvalidConfigurationException
* @throws ReflectionException
*/
public function __construct(string $clusterName, ExtendedCacheItemPoolInterface ...$driverPools)
{
if (\count($driverPools) !== 2) {
throw new PhpfastcacheInvalidArgumentException('A "master/slave" cluster requires exactly two pools to be working.');
}
parent::__construct($clusterName, ...$driverPools);
}
/**
* @inheritDoc
*/
public function getItem($key)
{
return $this->getStandardizedItem(
$this->makeOperation(
static function (ExtendedCacheItemPoolInterface $pool) use ($key) {
return $pool->getItem($key);
}
) ?? new Item($this, $key),
$this
);
}
/**
* @param callable $operation
* @return mixed
* @throws PhpfastcacheReplicationException
*/
protected function makeOperation(callable $operation)
{
try {
return $operation($this->getMasterPool());
} catch (PhpfastcacheExceptionInterface $e) {
try {
$this->eventManager->dispatch(
'CacheReplicationSlaveFallback',
$this,
\debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS, 2)[1]['function']
);
return $operation($this->getSlavePool());
} catch (PhpfastcacheExceptionInterface $e) {
throw new PhpfastcacheReplicationException('Master and Slave thrown an exception !');
}
}
}
/**
* @return ExtendedCacheItemPoolInterface
*/
protected function getMasterPool(): ExtendedCacheItemPoolInterface
{
return $this->clusterPools[0];
}
/**
* @return ExtendedCacheItemPoolInterface
*/
protected function getSlavePool(): ExtendedCacheItemPoolInterface
{
return $this->clusterPools[1];
}
/**
* @inheritDoc
*/
public function hasItem($key)
{
return $this->makeOperation(
static function (ExtendedCacheItemPoolInterface $pool) use ($key) {
return $pool->hasItem($key);
}
);
}
/**
* @inheritDoc
*/
public function clear()
{
return $this->makeOperation(
static function (ExtendedCacheItemPoolInterface $pool) {
return $pool->clear();
}
);
}
/**
* @inheritDoc
*/
public function deleteItem($key)
{
return $this->makeOperation(
static function (ExtendedCacheItemPoolInterface $pool) use ($key) {
return $pool->deleteItem($key);
}
);
}
/**
* @inheritDoc
*/
public function save(CacheItemInterface $item)
{
return $this->makeOperation(
function (ExtendedCacheItemPoolInterface $pool) use ($item) {
$item->setHit(true);
return $pool->save($this->getStandardizedItem($item, $pool));
}
);
}
/**
* @inheritDoc
*/
public function commit()
{
return $this->makeOperation(
static function (ExtendedCacheItemPoolInterface $pool) {
return $pool->commit();
}
);
}
}