2013-08-11 13:30:41 +02:00
|
|
|
<?php
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2023-02-15 21:22:37 +01:00
|
|
|
declare(strict_types=1);
|
2018-11-06 19:23:32 +01:00
|
|
|
|
2022-06-22 18:29:28 +02:00
|
|
|
class CacheFactory
|
|
|
|
{
|
2023-09-21 22:05:55 +02:00
|
|
|
private Logger $logger;
|
|
|
|
|
|
|
|
public function __construct(
|
|
|
|
Logger $logger
|
|
|
|
) {
|
|
|
|
$this->logger = $logger;
|
|
|
|
}
|
|
|
|
|
2022-07-05 13:20:01 +02:00
|
|
|
public function create(string $name = null): CacheInterface
|
2022-06-22 18:29:28 +02:00
|
|
|
{
|
2023-02-15 21:22:37 +01:00
|
|
|
$cacheNames = [];
|
|
|
|
foreach (scandir(PATH_LIB_CACHES) as $file) {
|
|
|
|
if (preg_match('/^([^.]+)Cache\.php$/U', $file, $m)) {
|
|
|
|
$cacheNames[] = $m[1];
|
|
|
|
}
|
2016-09-10 20:41:11 +02:00
|
|
|
}
|
2022-06-22 18:29:28 +02:00
|
|
|
// Trim trailing '.php' if exists
|
|
|
|
if (preg_match('/(.+)(?:\.php)/', $name, $matches)) {
|
|
|
|
$name = $matches[1];
|
2019-02-06 18:52:44 +01:00
|
|
|
}
|
2022-06-22 18:29:28 +02:00
|
|
|
// Trim trailing 'Cache' if exists
|
|
|
|
if (preg_match('/(.+)(?:Cache)$/i', $name, $matches)) {
|
|
|
|
$name = $matches[1];
|
2019-02-06 18:52:44 +01:00
|
|
|
}
|
2023-03-06 21:50:40 +01:00
|
|
|
|
|
|
|
$index = array_search(strtolower($name), array_map('strtolower', $cacheNames));
|
|
|
|
if ($index === false) {
|
2023-02-15 21:22:37 +01:00
|
|
|
throw new \InvalidArgumentException(sprintf('Invalid cache name: "%s"', $name));
|
|
|
|
}
|
2024-01-09 20:33:35 +01:00
|
|
|
|
2023-03-06 21:50:40 +01:00
|
|
|
$className = $cacheNames[$index] . 'Cache';
|
|
|
|
if (!preg_match('/^[A-Z][a-zA-Z0-9-]*$/', $className)) {
|
|
|
|
throw new \InvalidArgumentException(sprintf('Invalid cache classname: "%s"', $className));
|
2022-06-22 18:29:28 +02:00
|
|
|
}
|
2023-03-06 21:50:40 +01:00
|
|
|
|
|
|
|
switch ($className) {
|
|
|
|
case NullCache::class:
|
|
|
|
return new NullCache();
|
|
|
|
case FileCache::class:
|
2023-06-30 22:31:19 +02:00
|
|
|
$fileCacheConfig = [
|
|
|
|
// Intentionally checking for truthy value because the historic default value is the empty string
|
2023-03-20 19:10:01 +01:00
|
|
|
'path' => Configuration::getConfig('FileCache', 'path') ?: PATH_CACHE,
|
2023-03-06 21:50:40 +01:00
|
|
|
'enable_purge' => Configuration::getConfig('FileCache', 'enable_purge'),
|
2023-06-30 22:31:19 +02:00
|
|
|
];
|
|
|
|
if (!is_dir($fileCacheConfig['path'])) {
|
|
|
|
throw new \Exception(sprintf('The FileCache path does not exists: %s', $fileCacheConfig['path']));
|
|
|
|
}
|
|
|
|
if (!is_writable($fileCacheConfig['path'])) {
|
|
|
|
throw new \Exception(sprintf('The FileCache path is not writable: %s', $fileCacheConfig['path']));
|
|
|
|
}
|
2023-09-21 22:05:55 +02:00
|
|
|
return new FileCache($this->logger, $fileCacheConfig);
|
2023-03-06 21:50:40 +01:00
|
|
|
case SQLiteCache::class:
|
2023-07-06 15:59:38 +02:00
|
|
|
if (!extension_loaded('sqlite3')) {
|
|
|
|
throw new \Exception('"sqlite3" extension not loaded. Please check "php.ini"');
|
|
|
|
}
|
|
|
|
if (!is_writable(PATH_CACHE)) {
|
|
|
|
throw new \Exception('The cache folder is not writable');
|
|
|
|
}
|
|
|
|
$file = Configuration::getConfig('SQLiteCache', 'file');
|
|
|
|
if (!$file) {
|
|
|
|
throw new \Exception(sprintf('Configuration for %s missing.', 'SQLiteCache'));
|
|
|
|
}
|
|
|
|
if (dirname($file) == '.') {
|
|
|
|
$file = PATH_CACHE . $file;
|
|
|
|
} elseif (!is_dir(dirname($file))) {
|
|
|
|
throw new \Exception(sprintf('Invalid configuration for %s', 'SQLiteCache'));
|
|
|
|
}
|
2023-09-21 22:05:55 +02:00
|
|
|
return new SQLiteCache($this->logger, [
|
2023-07-06 15:59:38 +02:00
|
|
|
'file' => $file,
|
|
|
|
'timeout' => Configuration::getConfig('SQLiteCache', 'timeout'),
|
|
|
|
'enable_purge' => Configuration::getConfig('SQLiteCache', 'enable_purge'),
|
|
|
|
]);
|
2023-03-06 21:50:40 +01:00
|
|
|
case MemcachedCache::class:
|
2023-09-10 21:50:15 +02:00
|
|
|
if (!extension_loaded('memcached')) {
|
|
|
|
throw new \Exception('"memcached" extension not loaded. Please check "php.ini"');
|
|
|
|
}
|
|
|
|
$section = 'MemcachedCache';
|
|
|
|
$host = Configuration::getConfig($section, 'host');
|
|
|
|
$port = Configuration::getConfig($section, 'port');
|
|
|
|
if (empty($host) && empty($port)) {
|
|
|
|
throw new \Exception('Configuration for ' . $section . ' missing.');
|
|
|
|
}
|
|
|
|
if (empty($host)) {
|
|
|
|
throw new \Exception('"host" param is not set for ' . $section);
|
|
|
|
}
|
|
|
|
if (empty($port)) {
|
|
|
|
throw new \Exception('"port" param is not set for ' . $section);
|
|
|
|
}
|
|
|
|
if (!ctype_digit($port)) {
|
|
|
|
throw new \Exception('"port" param is invalid for ' . $section);
|
|
|
|
}
|
|
|
|
$port = intval($port);
|
|
|
|
if ($port < 1 || $port > 65535) {
|
|
|
|
throw new \Exception('"port" param is invalid for ' . $section);
|
|
|
|
}
|
2023-09-21 22:05:55 +02:00
|
|
|
return new MemcachedCache($this->logger, $host, $port);
|
2023-03-06 21:50:40 +01:00
|
|
|
default:
|
|
|
|
if (!file_exists(PATH_LIB_CACHES . $className . '.php')) {
|
|
|
|
throw new \Exception('Unable to find the cache file');
|
|
|
|
}
|
|
|
|
return new $className();
|
2023-02-15 21:22:37 +01:00
|
|
|
}
|
2019-02-06 18:52:44 +01:00
|
|
|
}
|
2015-12-04 09:19:05 +00:00
|
|
|
}
|