192 lines
7.3 KiB
PHP
192 lines
7.3 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\Log\LoggerAwareInterface;
|
||
|
use Psr\Log\LoggerInterface;
|
||
|
use Symfony\Component\Cache\CacheItem;
|
||
|
use Symfony\Component\Cache\Exception\InvalidArgumentException;
|
||
|
use Symfony\Component\Cache\ResettableInterface;
|
||
|
use Symfony\Component\Cache\Traits\AbstractAdapterTrait;
|
||
|
use Symfony\Component\Cache\Traits\ContractsTrait;
|
||
|
use Symfony\Contracts\Cache\CacheInterface;
|
||
|
|
||
|
/**
|
||
|
* @author Nicolas Grekas <p@tchwork.com>
|
||
|
*/
|
||
|
abstract class AbstractAdapter implements AdapterInterface, CacheInterface, LoggerAwareInterface, ResettableInterface
|
||
|
{
|
||
|
use AbstractAdapterTrait;
|
||
|
use ContractsTrait;
|
||
|
|
||
|
/**
|
||
|
* @internal
|
||
|
*/
|
||
|
protected const NS_SEPARATOR = ':';
|
||
|
|
||
|
private static bool $apcuSupported;
|
||
|
|
||
|
protected function __construct(string $namespace = '', int $defaultLifetime = 0)
|
||
|
{
|
||
|
$this->namespace = '' === $namespace ? '' : CacheItem::validateKey($namespace).static::NS_SEPARATOR;
|
||
|
$this->defaultLifetime = $defaultLifetime;
|
||
|
if (null !== $this->maxIdLength && \strlen($namespace) > $this->maxIdLength - 24) {
|
||
|
throw new InvalidArgumentException(sprintf('Namespace must be %d chars max, %d given ("%s").', $this->maxIdLength - 24, \strlen($namespace), $namespace));
|
||
|
}
|
||
|
self::$createCacheItem ??= \Closure::bind(
|
||
|
static function ($key, $value, $isHit) {
|
||
|
$item = new CacheItem();
|
||
|
$item->key = $key;
|
||
|
$item->value = $value;
|
||
|
$item->isHit = $isHit;
|
||
|
$item->unpack();
|
||
|
|
||
|
return $item;
|
||
|
},
|
||
|
null,
|
||
|
CacheItem::class
|
||
|
);
|
||
|
self::$mergeByLifetime ??= \Closure::bind(
|
||
|
static function ($deferred, $namespace, &$expiredIds, $getId, $defaultLifetime) {
|
||
|
$byLifetime = [];
|
||
|
$now = microtime(true);
|
||
|
$expiredIds = [];
|
||
|
|
||
|
foreach ($deferred as $key => $item) {
|
||
|
$key = (string) $key;
|
||
|
if (null === $item->expiry) {
|
||
|
$ttl = 0 < $defaultLifetime ? $defaultLifetime : 0;
|
||
|
} elseif (!$item->expiry) {
|
||
|
$ttl = 0;
|
||
|
} elseif (0 >= $ttl = (int) (0.1 + $item->expiry - $now)) {
|
||
|
$expiredIds[] = $getId($key);
|
||
|
continue;
|
||
|
}
|
||
|
$byLifetime[$ttl][$getId($key)] = $item->pack();
|
||
|
}
|
||
|
|
||
|
return $byLifetime;
|
||
|
},
|
||
|
null,
|
||
|
CacheItem::class
|
||
|
);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns the best possible adapter that your runtime supports.
|
||
|
*
|
||
|
* Using ApcuAdapter makes system caches compatible with read-only filesystems.
|
||
|
*/
|
||
|
public static function createSystemCache(string $namespace, int $defaultLifetime, string $version, string $directory, ?LoggerInterface $logger = null): AdapterInterface
|
||
|
{
|
||
|
$opcache = new PhpFilesAdapter($namespace, $defaultLifetime, $directory, true);
|
||
|
if (null !== $logger) {
|
||
|
$opcache->setLogger($logger);
|
||
|
}
|
||
|
|
||
|
if (!self::$apcuSupported ??= ApcuAdapter::isSupported()) {
|
||
|
return $opcache;
|
||
|
}
|
||
|
|
||
|
if ('cli' === \PHP_SAPI && !filter_var(\ini_get('apc.enable_cli'), \FILTER_VALIDATE_BOOL)) {
|
||
|
return $opcache;
|
||
|
}
|
||
|
|
||
|
$apcu = new ApcuAdapter($namespace, intdiv($defaultLifetime, 5), $version);
|
||
|
if (null !== $logger) {
|
||
|
$apcu->setLogger($logger);
|
||
|
}
|
||
|
|
||
|
return new ChainAdapter([$apcu, $opcache]);
|
||
|
}
|
||
|
|
||
|
public static function createConnection(#[\SensitiveParameter] string $dsn, array $options = []): mixed
|
||
|
{
|
||
|
if (str_starts_with($dsn, 'redis:') || str_starts_with($dsn, 'rediss:')) {
|
||
|
return RedisAdapter::createConnection($dsn, $options);
|
||
|
}
|
||
|
if (str_starts_with($dsn, 'memcached:')) {
|
||
|
return MemcachedAdapter::createConnection($dsn, $options);
|
||
|
}
|
||
|
if (str_starts_with($dsn, 'couchbase:')) {
|
||
|
if (class_exists('CouchbaseBucket') && CouchbaseBucketAdapter::isSupported()) {
|
||
|
return CouchbaseBucketAdapter::createConnection($dsn, $options);
|
||
|
}
|
||
|
|
||
|
return CouchbaseCollectionAdapter::createConnection($dsn, $options);
|
||
|
}
|
||
|
if (preg_match('/^(mysql|oci|pgsql|sqlsrv|sqlite):/', $dsn)) {
|
||
|
return PdoAdapter::createConnection($dsn, $options);
|
||
|
}
|
||
|
|
||
|
throw new InvalidArgumentException('Unsupported DSN: it does not start with "redis[s]:", "memcached:", "couchbase:", "mysql:", "oci:", "pgsql:", "sqlsrv:" nor "sqlite:".');
|
||
|
}
|
||
|
|
||
|
public function commit(): bool
|
||
|
{
|
||
|
$ok = true;
|
||
|
$byLifetime = (self::$mergeByLifetime)($this->deferred, $this->namespace, $expiredIds, $this->getId(...), $this->defaultLifetime);
|
||
|
$retry = $this->deferred = [];
|
||
|
|
||
|
if ($expiredIds) {
|
||
|
try {
|
||
|
$this->doDelete($expiredIds);
|
||
|
} catch (\Exception $e) {
|
||
|
$ok = false;
|
||
|
CacheItem::log($this->logger, 'Failed to delete expired items: '.$e->getMessage(), ['exception' => $e, 'cache-adapter' => get_debug_type($this)]);
|
||
|
}
|
||
|
}
|
||
|
foreach ($byLifetime as $lifetime => $values) {
|
||
|
try {
|
||
|
$e = $this->doSave($values, $lifetime);
|
||
|
} catch (\Exception $e) {
|
||
|
}
|
||
|
if (true === $e || [] === $e) {
|
||
|
continue;
|
||
|
}
|
||
|
if (\is_array($e) || 1 === \count($values)) {
|
||
|
foreach (\is_array($e) ? $e : array_keys($values) as $id) {
|
||
|
$ok = false;
|
||
|
$v = $values[$id];
|
||
|
$type = get_debug_type($v);
|
||
|
$message = sprintf('Failed to save key "{key}" of type %s%s', $type, $e instanceof \Exception ? ': '.$e->getMessage() : '.');
|
||
|
CacheItem::log($this->logger, $message, ['key' => substr($id, \strlen($this->namespace)), 'exception' => $e instanceof \Exception ? $e : null, 'cache-adapter' => get_debug_type($this)]);
|
||
|
}
|
||
|
} else {
|
||
|
foreach ($values as $id => $v) {
|
||
|
$retry[$lifetime][] = $id;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// When bulk-save failed, retry each item individually
|
||
|
foreach ($retry as $lifetime => $ids) {
|
||
|
foreach ($ids as $id) {
|
||
|
try {
|
||
|
$v = $byLifetime[$lifetime][$id];
|
||
|
$e = $this->doSave([$id => $v], $lifetime);
|
||
|
} catch (\Exception $e) {
|
||
|
}
|
||
|
if (true === $e || [] === $e) {
|
||
|
continue;
|
||
|
}
|
||
|
$ok = false;
|
||
|
$type = get_debug_type($v);
|
||
|
$message = sprintf('Failed to save key "{key}" of type %s%s', $type, $e instanceof \Exception ? ': '.$e->getMessage() : '.');
|
||
|
CacheItem::log($this->logger, $message, ['key' => substr($id, \strlen($this->namespace)), 'exception' => $e instanceof \Exception ? $e : null, 'cache-adapter' => get_debug_type($this)]);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $ok;
|
||
|
}
|
||
|
}
|