2017-11-09 11:05:18 -05:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* @file src/Core/Cache.php
|
|
|
|
*/
|
|
|
|
namespace Friendica\Core;
|
|
|
|
|
|
|
|
use Friendica\Core\Config;
|
|
|
|
use Friendica\Database\DBM;
|
2018-01-26 21:38:34 -05:00
|
|
|
use Friendica\Util\DateTimeFormat;
|
2017-11-21 18:03:33 -05:00
|
|
|
use dba;
|
2018-01-24 21:08:45 -05:00
|
|
|
use Memcache;
|
2017-11-09 11:05:18 -05:00
|
|
|
|
2017-12-17 15:24:57 -05:00
|
|
|
require_once 'include/dba.php';
|
|
|
|
|
2017-11-09 11:05:18 -05:00
|
|
|
/**
|
|
|
|
* @brief Class for storing data for a short time
|
|
|
|
*/
|
|
|
|
class Cache
|
|
|
|
{
|
|
|
|
/**
|
2018-02-03 00:43:57 -05:00
|
|
|
* @brief Check for Memcache and open a connection if configured
|
2017-11-09 11:05:18 -05:00
|
|
|
*
|
2018-02-03 00:43:57 -05:00
|
|
|
* @return Memcache|boolean The Memcache object - or "false" if not successful
|
2017-11-09 11:05:18 -05:00
|
|
|
*/
|
|
|
|
public static function memcache()
|
|
|
|
{
|
2018-02-03 11:59:43 -05:00
|
|
|
if (!class_exists('Memcache', false)) {
|
2017-11-09 11:05:18 -05:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!Config::get('system', 'memcache')) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
$memcache_host = Config::get('system', 'memcache_host', '127.0.0.1');
|
|
|
|
$memcache_port = Config::get('system', 'memcache_port', 11211);
|
|
|
|
|
2018-02-03 11:59:43 -05:00
|
|
|
$memcache = new Memcache();
|
2017-11-09 11:05:18 -05:00
|
|
|
|
|
|
|
if (!$memcache->connect($memcache_host, $memcache_port)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $memcache;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Return the duration for a given cache level
|
|
|
|
*
|
|
|
|
* @param integer $level Cache level
|
|
|
|
*
|
|
|
|
* @return integer The cache duration in seconds
|
|
|
|
*/
|
|
|
|
private static function duration($level)
|
|
|
|
{
|
|
|
|
switch ($level) {
|
|
|
|
case CACHE_MONTH:
|
|
|
|
$seconds = 2592000;
|
|
|
|
break;
|
|
|
|
case CACHE_WEEK:
|
|
|
|
$seconds = 604800;
|
|
|
|
break;
|
|
|
|
case CACHE_DAY:
|
|
|
|
$seconds = 86400;
|
|
|
|
break;
|
|
|
|
case CACHE_HOUR:
|
|
|
|
$seconds = 3600;
|
|
|
|
break;
|
|
|
|
case CACHE_HALF_HOUR:
|
|
|
|
$seconds = 1800;
|
|
|
|
break;
|
|
|
|
case CACHE_QUARTER_HOUR:
|
|
|
|
$seconds = 900;
|
|
|
|
break;
|
|
|
|
case CACHE_FIVE_MINUTES:
|
|
|
|
$seconds = 300;
|
|
|
|
break;
|
|
|
|
case CACHE_MINUTE:
|
|
|
|
$seconds = 60;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return $seconds;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Fetch cached data according to the key
|
|
|
|
*
|
|
|
|
* @param string $key The key to the cached data
|
|
|
|
*
|
|
|
|
* @return mixed Cached $value or "null" if not found
|
|
|
|
*/
|
|
|
|
public static function get($key)
|
|
|
|
{
|
|
|
|
$memcache = self::memcache();
|
|
|
|
if (is_object($memcache)) {
|
|
|
|
// We fetch with the hostname as key to avoid problems with other applications
|
|
|
|
$cached = $memcache->get(get_app()->get_hostname().":".$key);
|
|
|
|
$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;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Frequently clear cache
|
2017-12-17 15:31:37 -05:00
|
|
|
self::clear();
|
2017-11-09 11:05:18 -05:00
|
|
|
|
2018-01-11 03:26:30 -05:00
|
|
|
$cache = dba::selectFirst('cache', ['v'], ['k' => $key]);
|
2017-11-09 11:05:18 -05:00
|
|
|
|
2018-01-11 03:26:30 -05:00
|
|
|
if (DBM::is_result($cache)) {
|
|
|
|
$cached = $cache['v'];
|
2017-11-09 11:05:18 -05:00
|
|
|
$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;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Put data in the cache according to the key
|
|
|
|
*
|
|
|
|
* The input $value can have multiple formats.
|
|
|
|
*
|
|
|
|
* @param string $key The key to the cached data
|
|
|
|
* @param mixed $value The value that is about to be stored
|
|
|
|
* @param integer $duration The cache lifespan
|
2017-11-19 14:15:25 -05:00
|
|
|
*
|
|
|
|
* @return void
|
2017-11-09 11:05:18 -05:00
|
|
|
*/
|
|
|
|
public static function set($key, $value, $duration = CACHE_MONTH)
|
|
|
|
{
|
|
|
|
// Do we have an installed memcache? Use it instead.
|
|
|
|
$memcache = self::memcache();
|
|
|
|
if (is_object($memcache)) {
|
|
|
|
// We store with the hostname as key to avoid problems with other applications
|
|
|
|
$memcache->set(get_app()->get_hostname().":".$key, serialize($value), MEMCACHE_COMPRESSED, self::duration($duration));
|
|
|
|
return;
|
|
|
|
}
|
2018-01-26 21:38:34 -05:00
|
|
|
$fields = ['v' => serialize($value), 'expire_mode' => $duration, 'updated' => DateTimeFormat::utcNow()];
|
2018-01-15 08:05:12 -05:00
|
|
|
$condition = ['k' => $key];
|
2017-11-21 18:03:33 -05:00
|
|
|
dba::update('cache', $fields, $condition, true);
|
2017-11-09 11:05:18 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Remove outdated data from the cache
|
|
|
|
*
|
|
|
|
* @param integer $max_level The maximum cache level that is to be cleared
|
2017-11-19 14:15:25 -05:00
|
|
|
*
|
|
|
|
* @return void
|
2017-11-09 11:05:18 -05:00
|
|
|
*/
|
|
|
|
public static function clear($max_level = CACHE_MONTH)
|
|
|
|
{
|
|
|
|
// Clear long lasting cache entries only once a day
|
|
|
|
if (Config::get("system", "cache_cleared_day") < time() - self::duration(CACHE_DAY)) {
|
|
|
|
if ($max_level == CACHE_MONTH) {
|
2018-01-15 08:05:12 -05:00
|
|
|
$condition = ["`updated` < ? AND `expire_mode` = ?",
|
2018-01-26 21:38:34 -05:00
|
|
|
DateTimeFormat::utc("now - 30 days"),
|
2018-01-15 08:05:12 -05:00
|
|
|
CACHE_MONTH];
|
2017-11-21 18:03:33 -05:00
|
|
|
dba::delete('cache', $condition);
|
2017-11-09 11:05:18 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
if ($max_level <= CACHE_WEEK) {
|
2018-01-15 08:05:12 -05:00
|
|
|
$condition = ["`updated` < ? AND `expire_mode` = ?",
|
2018-01-26 21:38:34 -05:00
|
|
|
DateTimeFormat::utc("now - 7 days"),
|
2018-01-15 08:05:12 -05:00
|
|
|
CACHE_WEEK];
|
2017-11-21 18:03:33 -05:00
|
|
|
dba::delete('cache', $condition);
|
2017-11-09 11:05:18 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
if ($max_level <= CACHE_DAY) {
|
2018-01-15 08:05:12 -05:00
|
|
|
$condition = ["`updated` < ? AND `expire_mode` = ?",
|
2018-01-26 21:38:34 -05:00
|
|
|
DateTimeFormat::utc("now - 1 days"),
|
2018-01-15 08:05:12 -05:00
|
|
|
CACHE_DAY];
|
2017-11-21 18:03:33 -05:00
|
|
|
dba::delete('cache', $condition);
|
2017-11-09 11:05:18 -05:00
|
|
|
}
|
|
|
|
Config::set("system", "cache_cleared_day", time());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (($max_level <= CACHE_HOUR) && (Config::get("system", "cache_cleared_hour")) < time() - self::duration(CACHE_HOUR)) {
|
2018-01-15 08:05:12 -05:00
|
|
|
$condition = ["`updated` < ? AND `expire_mode` = ?",
|
2018-01-26 21:38:34 -05:00
|
|
|
DateTimeFormat::utc("now - 1 hours"),
|
2018-01-15 08:05:12 -05:00
|
|
|
CACHE_HOUR];
|
2017-11-21 18:03:33 -05:00
|
|
|
dba::delete('cache', $condition);
|
2017-11-09 11:05:18 -05:00
|
|
|
|
|
|
|
Config::set("system", "cache_cleared_hour", time());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (($max_level <= CACHE_HALF_HOUR) && (Config::get("system", "cache_cleared_half_hour")) < time() - self::duration(CACHE_HALF_HOUR)) {
|
2018-01-15 08:05:12 -05:00
|
|
|
$condition = ["`updated` < ? AND `expire_mode` = ?",
|
2018-01-26 21:38:34 -05:00
|
|
|
DateTimeFormat::utc("now - 30 minutes"),
|
2018-01-15 08:05:12 -05:00
|
|
|
CACHE_HALF_HOUR];
|
2017-11-21 18:03:33 -05:00
|
|
|
dba::delete('cache', $condition);
|
2017-11-09 11:05:18 -05:00
|
|
|
|
|
|
|
Config::set("system", "cache_cleared_half_hour", time());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (($max_level <= CACHE_QUARTER_HOUR) && (Config::get("system", "cache_cleared_quarter_hour")) < time() - self::duration(CACHE_QUARTER_HOUR)) {
|
2018-01-15 08:05:12 -05:00
|
|
|
$condition = ["`updated` < ? AND `expire_mode` = ?",
|
2018-01-26 21:38:34 -05:00
|
|
|
DateTimeFormat::utc("now - 15 minutes"),
|
2018-01-15 08:05:12 -05:00
|
|
|
CACHE_QUARTER_HOUR];
|
2017-11-21 18:03:33 -05:00
|
|
|
dba::delete('cache', $condition);
|
2017-11-09 11:05:18 -05:00
|
|
|
|
|
|
|
Config::set("system", "cache_cleared_quarter_hour", time());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (($max_level <= CACHE_FIVE_MINUTES) && (Config::get("system", "cache_cleared_five_minute")) < time() - self::duration(CACHE_FIVE_MINUTES)) {
|
2018-01-15 08:05:12 -05:00
|
|
|
$condition = ["`updated` < ? AND `expire_mode` = ?",
|
2018-01-26 21:38:34 -05:00
|
|
|
DateTimeFormat::utc("now - 5 minutes"),
|
2018-01-15 08:05:12 -05:00
|
|
|
CACHE_FIVE_MINUTES];
|
2017-11-21 18:03:33 -05:00
|
|
|
dba::delete('cache', $condition);
|
2017-11-09 11:05:18 -05:00
|
|
|
|
|
|
|
Config::set("system", "cache_cleared_five_minute", time());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (($max_level <= CACHE_MINUTE) && (Config::get("system", "cache_cleared_minute")) < time() - self::duration(CACHE_MINUTE)) {
|
2018-01-15 08:05:12 -05:00
|
|
|
$condition = ["`updated` < ? AND `expire_mode` = ?",
|
2018-01-26 21:38:34 -05:00
|
|
|
DateTimeFormat::utc("now - 1 minutes"),
|
2018-01-15 08:05:12 -05:00
|
|
|
CACHE_MINUTE];
|
2017-11-21 18:03:33 -05:00
|
|
|
dba::delete('cache', $condition);
|
2017-11-09 11:05:18 -05:00
|
|
|
|
|
|
|
Config::set("system", "cache_cleared_minute", time());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|