2018-03-24 18:39:13 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Friendica\Core\Cache;
|
|
|
|
|
2019-08-03 18:48:56 +00:00
|
|
|
use Friendica\Database\Database;
|
2018-03-24 18:39:13 +00:00
|
|
|
use Friendica\Util\DateTimeFormat;
|
|
|
|
|
|
|
|
/**
|
2019-08-04 08:26:53 +00:00
|
|
|
* Database Cache
|
2018-03-24 18:39:13 +00:00
|
|
|
*
|
2018-09-15 23:28:38 +00:00
|
|
|
* @author Hypolite Petovan <hypolite@mrpetovan.com>
|
2018-03-24 18:39:13 +00:00
|
|
|
*/
|
2019-08-04 13:51:49 +00:00
|
|
|
class DatabaseCache extends Cache implements ICache
|
2018-03-24 18:39:13 +00:00
|
|
|
{
|
2019-08-03 18:48:56 +00:00
|
|
|
/**
|
|
|
|
* @var Database
|
|
|
|
*/
|
|
|
|
private $dba;
|
|
|
|
|
|
|
|
public function __construct(string $hostname, Database $dba)
|
|
|
|
{
|
|
|
|
parent::__construct($hostname);
|
|
|
|
|
|
|
|
$this->dba = $dba;
|
|
|
|
}
|
|
|
|
|
2018-09-26 02:52:32 +00:00
|
|
|
/**
|
|
|
|
* (@inheritdoc)
|
|
|
|
*/
|
2018-10-06 22:27:54 +00:00
|
|
|
public function getAllKeys($prefix = null)
|
2018-09-26 02:52:32 +00:00
|
|
|
{
|
2018-10-06 22:27:54 +00:00
|
|
|
if (empty($prefix)) {
|
|
|
|
$where = ['`expires` >= ?', DateTimeFormat::utcNow()];
|
|
|
|
} else {
|
2018-10-07 20:14:05 +00:00
|
|
|
$where = ['`expires` >= ? AND `k` LIKE CONCAT(?, \'%\')', DateTimeFormat::utcNow(), $prefix];
|
2018-10-06 22:27:54 +00:00
|
|
|
}
|
2018-09-26 02:52:32 +00:00
|
|
|
|
2019-08-03 18:48:56 +00:00
|
|
|
$stmt = $this->dba->select('cache', ['k'], $where);
|
2018-10-06 22:27:54 +00:00
|
|
|
|
2018-10-07 20:14:05 +00:00
|
|
|
$keys = [];
|
2019-08-03 18:48:56 +00:00
|
|
|
while ($key = $this->dba->fetch($stmt)) {
|
2018-10-07 20:14:05 +00:00
|
|
|
array_push($keys, $key['k']);
|
2018-10-06 22:27:54 +00:00
|
|
|
}
|
2019-08-03 18:48:56 +00:00
|
|
|
$this->dba->close($stmt);
|
2018-10-06 22:27:54 +00:00
|
|
|
|
2018-10-07 20:14:05 +00:00
|
|
|
return $keys;
|
2018-09-26 02:52:32 +00:00
|
|
|
}
|
|
|
|
|
2018-09-26 02:51:41 +00:00
|
|
|
/**
|
|
|
|
* (@inheritdoc)
|
|
|
|
*/
|
2018-03-24 18:39:13 +00:00
|
|
|
public function get($key)
|
|
|
|
{
|
2019-08-03 18:48:56 +00:00
|
|
|
$cache = $this->dba->selectFirst('cache', ['v'], ['`k` = ? AND (`expires` >= ? OR `expires` = -1)', $key, DateTimeFormat::utcNow()]);
|
2018-03-24 18:39:13 +00:00
|
|
|
|
2019-08-03 18:48:56 +00:00
|
|
|
if ($this->dba->isResult($cache)) {
|
2018-03-24 18:39:13 +00:00
|
|
|
$cached = $cache['v'];
|
|
|
|
$value = @unserialize($cached);
|
|
|
|
|
|
|
|
// Only return a value if the serialized value is valid.
|
|
|
|
// We also check if the db entry is a serialized
|
|
|
|
// boolean 'false' value (which we want to return).
|
|
|
|
if ($cached === serialize(false) || $value !== false) {
|
|
|
|
return $value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2018-09-26 02:51:41 +00:00
|
|
|
/**
|
|
|
|
* (@inheritdoc)
|
|
|
|
*/
|
2018-07-04 21:37:22 +00:00
|
|
|
public function set($key, $value, $ttl = Cache::FIVE_MINUTES)
|
2018-03-24 18:39:13 +00:00
|
|
|
{
|
2018-10-29 09:16:07 +00:00
|
|
|
if ($ttl > 0) {
|
|
|
|
$fields = [
|
|
|
|
'v' => serialize($value),
|
|
|
|
'expires' => DateTimeFormat::utc('now + ' . $ttl . 'seconds'),
|
|
|
|
'updated' => DateTimeFormat::utcNow()
|
|
|
|
];
|
|
|
|
} else {
|
|
|
|
$fields = [
|
|
|
|
'v' => serialize($value),
|
|
|
|
'expires' => -1,
|
|
|
|
'updated' => DateTimeFormat::utcNow()
|
|
|
|
];
|
|
|
|
}
|
2018-03-24 18:39:13 +00:00
|
|
|
|
2019-08-03 18:48:56 +00:00
|
|
|
return $this->dba->update('cache', $fields, ['k' => $key], true);
|
2018-03-24 18:39:13 +00:00
|
|
|
}
|
|
|
|
|
2018-09-26 02:51:41 +00:00
|
|
|
/**
|
|
|
|
* (@inheritdoc)
|
|
|
|
*/
|
2018-03-24 18:39:13 +00:00
|
|
|
public function delete($key)
|
|
|
|
{
|
2019-08-03 18:48:56 +00:00
|
|
|
return $this->dba->delete('cache', ['k' => $key]);
|
2018-03-24 18:39:13 +00:00
|
|
|
}
|
|
|
|
|
2018-09-26 02:51:41 +00:00
|
|
|
/**
|
|
|
|
* (@inheritdoc)
|
|
|
|
*/
|
2018-07-07 17:46:16 +00:00
|
|
|
public function clear($outdated = true)
|
2018-03-24 18:39:13 +00:00
|
|
|
{
|
2018-07-07 17:46:16 +00:00
|
|
|
if ($outdated) {
|
2019-08-03 18:48:56 +00:00
|
|
|
return $this->dba->delete('cache', ['`expires` < NOW()']);
|
2018-07-07 17:46:16 +00:00
|
|
|
} else {
|
2019-08-03 18:48:56 +00:00
|
|
|
return $this->dba->delete('cache', ['`k` IS NOT NULL ']);
|
2018-07-07 17:46:16 +00:00
|
|
|
}
|
2018-03-24 18:39:13 +00:00
|
|
|
}
|
2019-08-04 13:42:39 +00:00
|
|
|
|
2019-08-04 14:13:53 +00:00
|
|
|
/**
|
|
|
|
* {@inheritDoc}
|
|
|
|
*/
|
|
|
|
public function getName()
|
2019-08-04 13:42:39 +00:00
|
|
|
{
|
|
|
|
return self::TYPE_DATABASE;
|
|
|
|
}
|
2018-03-24 18:39:13 +00:00
|
|
|
}
|