2015-11-07 16:59:20 +01:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* EGroupware API: Caching provider storing data in memcached via PHP's memcached extension
|
|
|
|
*
|
|
|
|
* @link http://www.egroupware.org
|
|
|
|
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
|
|
|
|
* @package api
|
|
|
|
* @subpackage cache
|
|
|
|
* @author Ralf Becker <RalfBecker-AT-outdoor-training.de>
|
2016-02-28 10:38:36 +01:00
|
|
|
* @copyright (c) 2009-16 by Ralf Becker <RalfBecker-AT-outdoor-training.de>
|
2015-11-07 16:59:20 +01:00
|
|
|
* @version $Id$
|
|
|
|
*/
|
|
|
|
|
2016-02-28 10:38:36 +01:00
|
|
|
namespace EGroupware\Api\Cache;
|
|
|
|
|
2015-11-07 16:59:20 +01:00
|
|
|
/**
|
|
|
|
* Caching provider storing data in memcached via PHP's memcached extension
|
|
|
|
*
|
|
|
|
* The provider concats all $keys with '::' to get a single string.
|
|
|
|
*
|
|
|
|
* To use this provider set in your header.inc.php:
|
2016-02-28 10:38:36 +01:00
|
|
|
* $GLOBALS['egw_info']['server']['cache_provider_instance'] = array('EGroupware\Api\Cache\Memcached','localhost'[,'otherhost:port']);
|
2015-11-07 16:59:20 +01:00
|
|
|
* and optional also $GLOBALS['egw_info']['server']['cache_provider_tree'] (defaults to instance)
|
|
|
|
*
|
|
|
|
* You can set more then one server and specify a port, if it's not the default one 11211.
|
|
|
|
*
|
2015-11-07 18:30:21 +01:00
|
|
|
* It allows addtional named parameters "timeout" (default 20ms), "retry" (default not) and "prefix".
|
2015-11-07 16:59:20 +01:00
|
|
|
*
|
|
|
|
* If igbinary extension is available, it is prefered over PHP (un)serialize.
|
|
|
|
*/
|
2016-02-28 10:38:36 +01:00
|
|
|
class Memcached extends Base implements ProviderMultiple
|
2015-11-07 16:59:20 +01:00
|
|
|
{
|
|
|
|
/**
|
2016-08-28 11:08:10 +02:00
|
|
|
* Instance of \Memcached
|
2015-11-07 16:59:20 +01:00
|
|
|
*
|
2016-08-28 11:08:10 +02:00
|
|
|
* @var \Memcached
|
2015-11-07 16:59:20 +01:00
|
|
|
*/
|
|
|
|
private $memcache;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Timeout in ms
|
|
|
|
*/
|
|
|
|
private $timeout = 20;
|
|
|
|
|
2018-02-28 10:44:00 +01:00
|
|
|
/**
|
|
|
|
* Use Libketama consistent hashing
|
|
|
|
*
|
2021-05-27 12:29:51 +02:00
|
|
|
* Off by default as for just 2 Memcached servers it creates an extreme
|
2018-02-28 10:44:00 +01:00
|
|
|
* unbalanced distribution favoring the 2. server and has no benefits
|
|
|
|
* as requests to the failed node can only go to the other one anyway.
|
|
|
|
*
|
|
|
|
* @var boolean
|
|
|
|
*/
|
|
|
|
private $consistent = false;
|
|
|
|
|
2015-11-07 16:59:20 +01:00
|
|
|
/**
|
2021-05-27 12:29:51 +02:00
|
|
|
* Retry on node failure: 0: no retry, 1: retry on set/add/delete, 2: always retry
|
2015-11-07 16:59:20 +01:00
|
|
|
*
|
|
|
|
* @var retry
|
|
|
|
*/
|
|
|
|
private $retry = 0;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Constructor, eg. opens the connection to the backend
|
|
|
|
*
|
|
|
|
* @throws Exception if connection to backend could not be established
|
|
|
|
* @param array $params eg. array('localhost'[,'localhost:11211',...])
|
|
|
|
* "timeout" in ms, "retry" on node failure 0: no retry (default), 1: retry on set/add/delete, 2: allways retry
|
2018-02-28 10:44:00 +01:00
|
|
|
* "prefix" prefix for keys and "consistent=1" to enable (by default disabled) consistent caching
|
2015-11-07 16:59:20 +01:00
|
|
|
*/
|
|
|
|
function __construct(array $params=null)
|
|
|
|
{
|
|
|
|
$this->params = $params ? $params : array('localhost'); // some reasonable default
|
|
|
|
|
|
|
|
if (isset($params['timeout']))
|
|
|
|
{
|
|
|
|
$this->timeout = (int)$params['timeout'];
|
|
|
|
unset($params['timeout']);
|
|
|
|
}
|
|
|
|
if (isset($params['retry']))
|
|
|
|
{
|
|
|
|
$this->retry = (int)$params['retry'];
|
|
|
|
unset($params['retry']);
|
|
|
|
}
|
2015-11-07 18:30:21 +01:00
|
|
|
if (isset($params['prefix']))
|
|
|
|
{
|
|
|
|
$prefix = $params['prefix'];
|
|
|
|
unset($params['prefix']);
|
|
|
|
}
|
2018-02-28 10:44:00 +01:00
|
|
|
if (isset($params['consistent']))
|
|
|
|
{
|
|
|
|
$this->consistent = !empty($params['consistent']);
|
|
|
|
}
|
2015-11-07 16:59:20 +01:00
|
|
|
|
|
|
|
check_load_extension('memcached',true);
|
|
|
|
// using a persitent connection for identical $params
|
2016-02-29 09:53:05 +01:00
|
|
|
$this->memcache = new \Memcached(md5(serialize($params)));
|
2015-11-07 16:59:20 +01:00
|
|
|
|
|
|
|
$this->memcache->setOptions(array(
|
|
|
|
// setting a short timeout, to better kope with failed nodes
|
2016-02-29 09:53:05 +01:00
|
|
|
\Memcached::OPT_CONNECT_TIMEOUT => $this->timeout,
|
|
|
|
\Memcached::OPT_SEND_TIMEOUT => $this->timeout,
|
|
|
|
\Memcached::OPT_RECV_TIMEOUT => $this->timeout,
|
2018-02-28 10:44:00 +01:00
|
|
|
// use more efficient binary protocol (also required for consistent hashing)
|
2016-02-29 09:53:05 +01:00
|
|
|
\Memcached::OPT_BINARY_PROTOCOL => true,
|
2018-02-28 10:44:00 +01:00
|
|
|
// Libketama compatible consistent hashing
|
|
|
|
\Memcached::OPT_LIBKETAMA_COMPATIBLE => $this->consistent,
|
2015-11-07 16:59:20 +01:00
|
|
|
// automatic failover and disabling of failed nodes
|
2016-02-29 09:53:05 +01:00
|
|
|
\Memcached::OPT_SERVER_FAILURE_LIMIT => 2,
|
2015-11-07 18:30:21 +01:00
|
|
|
// setting a prefix for all keys
|
2016-02-29 09:53:05 +01:00
|
|
|
\Memcached::OPT_PREFIX_KEY => $prefix,
|
2015-11-07 16:59:20 +01:00
|
|
|
));
|
2016-07-28 13:51:47 +02:00
|
|
|
// automatic disabling of failed nodes
|
|
|
|
if (@constant('Memcached::OPT_AUTO_EJECT_HOSTS'))
|
|
|
|
{
|
|
|
|
$this->memcache->setOption(\Memcached::OPT_AUTO_EJECT_HOSTS, true);
|
|
|
|
}
|
2016-07-13 08:56:31 +02:00
|
|
|
// use igbinary, if available
|
|
|
|
if (\Memcached::HAVE_IGBINARY)
|
|
|
|
{
|
|
|
|
$this->memcache->setOption(\Memcached::OPT_SERIALIZER, \Memcached::SERIALIZER_IGBINARY);
|
|
|
|
}
|
|
|
|
elseif(\Memcached::HAVE_JSON)
|
|
|
|
{
|
|
|
|
$this->memcache->setOption(\Memcached::OPT_SERIALIZER, \Memcached::SERIALIZER_JSON);
|
|
|
|
}
|
2015-11-07 16:59:20 +01:00
|
|
|
|
|
|
|
// with persistent connections, only add servers, if they not already added!
|
|
|
|
if (!count($this->memcache->getServerList()))
|
|
|
|
{
|
|
|
|
$ok = false;
|
|
|
|
foreach($params as $host_port)
|
|
|
|
{
|
|
|
|
$parts = explode(':',$host_port);
|
|
|
|
$host = array_shift($parts);
|
|
|
|
$port = $parts ? array_shift($parts) : 11211; // default port
|
|
|
|
|
|
|
|
$ok = $this->memcache->addServer($host,$port) || $ok;
|
|
|
|
//error_log(__METHOD__."(".array2string($params).") memcache->addServer('$host',$port) = ".(int)$ok);
|
|
|
|
}
|
|
|
|
if (!$ok)
|
|
|
|
{
|
|
|
|
throw new Exception (__METHOD__.'('.array2string($params).") Can't open connection to any memcached server!");
|
|
|
|
}
|
2015-11-10 00:52:47 +01:00
|
|
|
//error_log(__METHOD__."(".array2string($params).") creating new pool / persitent connection");
|
2015-11-07 16:59:20 +01:00
|
|
|
}
|
2015-11-10 00:52:47 +01:00
|
|
|
//else error_log(__METHOD__."(".array2string($params).") using existing pool / persitent connection");
|
2015-11-07 16:59:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Stores some data in the cache, if it does NOT already exists there
|
|
|
|
*
|
|
|
|
* @param array $keys eg. array($level,$app,$location)
|
|
|
|
* @param mixed $data
|
|
|
|
* @param int $expiration =0
|
|
|
|
* @return boolean true on success, false on error, incl. key already exists in cache
|
|
|
|
*/
|
|
|
|
function add(array $keys,$data,$expiration=0)
|
|
|
|
{
|
|
|
|
return $this->memcache->add(self::key($keys), $data, $expiration) ||
|
|
|
|
// if we have multiple nodes, retry on error, but not on data exists
|
2016-02-29 09:53:05 +01:00
|
|
|
$this->retry > 0 && $this->memcache->getResultCode() !== \Memcached::RES_DATA_EXISTS &&
|
2015-11-07 16:59:20 +01:00
|
|
|
$this->memcache->add(self::key($keys), $data, $expiration);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Stores some data in the cache
|
|
|
|
*
|
|
|
|
* @param array $keys eg. array($level,$app,$location)
|
|
|
|
* @param mixed $data
|
|
|
|
* @param int $expiration =0
|
|
|
|
* @return boolean true on success, false on error
|
|
|
|
*/
|
|
|
|
function set(array $keys,$data,$expiration=0)
|
|
|
|
{
|
|
|
|
return $this->memcache->set(self::key($keys), $data, $expiration) ||
|
|
|
|
// if we have multiple nodes, retry on error
|
|
|
|
$this->retry > 0 && $this->memcache->set(self::key($keys), $data, $expiration);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get some data from the cache
|
|
|
|
*
|
|
|
|
* @param array $keys eg. array($level,$app,$location)
|
|
|
|
* @return mixed data stored or NULL if not found in cache
|
|
|
|
*/
|
|
|
|
function get(array $keys)
|
|
|
|
{
|
|
|
|
if (($data = $this->memcache->get($key=self::key($keys))) === false &&
|
2016-02-29 09:53:05 +01:00
|
|
|
$this->memcache->getResultCode() !== \Memcached::RES_SUCCESS ||
|
2015-11-07 16:59:20 +01:00
|
|
|
// if we have multiple nodes, retry on error, but not on not found
|
2016-02-29 09:53:05 +01:00
|
|
|
$this->retry > 1 && $this->memcache->getResultCode() !== \Memcached::RES_NOTFOUND &&
|
2015-11-07 16:59:20 +01:00
|
|
|
($data = $this->memcache->get($key=self::key($keys))) === false &&
|
2016-02-29 09:53:05 +01:00
|
|
|
$this->memcache->getResultCode() !== \Memcached::RES_SUCCESS)
|
2015-11-07 16:59:20 +01:00
|
|
|
{
|
|
|
|
//error_log(__METHOD__."(".array2string($keys).") key='$key' NOT found!".' $this->memcache->getResultCode()='.$this->memcache->getResultCode().')');
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
//error_log(__METHOD__."(".array2string($keys).") key='$key' found ".bytes($data)." bytes).");
|
|
|
|
return $data;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get multiple data from the cache
|
|
|
|
*
|
|
|
|
* @param array $keys eg. array of array($level,$app,array $locations)
|
|
|
|
* @return array key => data stored, not found keys are NOT returned
|
|
|
|
*/
|
|
|
|
function mget(array $keys)
|
|
|
|
{
|
|
|
|
$locations = array_pop($keys);
|
|
|
|
$prefix = self::key($keys);
|
|
|
|
foreach($locations as &$location)
|
|
|
|
{
|
|
|
|
$location = $prefix.'::'.$location;
|
|
|
|
}
|
|
|
|
if (($multiple = $this->memcache->getMulti($locations)) === false ||
|
|
|
|
// if we have multiple nodes, retry on error, but not on not found
|
2016-02-29 09:53:05 +01:00
|
|
|
$this->retry > 1 && $this->memcache->getResultCode() !== \Memcached::RES_NOTFOUND &&
|
2015-11-07 16:59:20 +01:00
|
|
|
($multiple = $this->memcache->getMulti($locations)) === false)
|
|
|
|
{
|
|
|
|
return array();
|
|
|
|
}
|
|
|
|
$ret = array();
|
|
|
|
$prefix_len = strlen($prefix)+2;
|
|
|
|
foreach($multiple as $location => $data)
|
|
|
|
{
|
|
|
|
$key = substr($location,$prefix_len);
|
|
|
|
//error_log(__METHOD__."(".array2string($locations).") key='$key' found ".bytes($data)." bytes).");
|
|
|
|
$ret[$key] = $data;
|
|
|
|
}
|
|
|
|
return $ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Delete some data from the cache
|
|
|
|
*
|
|
|
|
* @param array $keys eg. array($level,$app,$location)
|
|
|
|
* @return boolean true on success, false on error (eg. $key not set)
|
|
|
|
*/
|
|
|
|
function delete(array $keys)
|
|
|
|
{
|
|
|
|
return $this->memcache->delete(self::key($keys)) ||
|
2016-02-29 09:53:05 +01:00
|
|
|
$this->retry > 0 && $this->memcache->getResultCode() !== \Memcached::RES_NOTFOUND &&
|
2015-11-07 16:59:20 +01:00
|
|
|
$this->memcache->delete(self::key($keys));
|
|
|
|
}
|
|
|
|
|
2021-05-27 12:29:51 +02:00
|
|
|
/**
|
|
|
|
* Increments value in cache
|
|
|
|
*
|
|
|
|
* @param array $keys
|
|
|
|
* @param int $offset =1 how much to increment by
|
|
|
|
* @param int $intial_value =0 value to set and return, if not in cache
|
|
|
|
* @param int $expiration =0
|
|
|
|
* @return false|int new value on success, false on error
|
|
|
|
*/
|
|
|
|
function increment(array $keys, int $offset=1, int $intial_value=0, int $expiration=0)
|
|
|
|
{
|
|
|
|
$ret = $this->memcache->increment($key=self::key($keys), $offset, $intial_value, $expiration);
|
|
|
|
if ($ret === false && $this->retry > 0)
|
|
|
|
{
|
|
|
|
$ret = $this->memcache->increment($key, $offset, $intial_value, $expiration);
|
|
|
|
}
|
|
|
|
// fix memcached returning strings
|
|
|
|
return $ret !== false ? (int)$ret : $ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Decrements value in cache, but never below 0
|
|
|
|
*
|
|
|
|
* If new value would be below 0, 0 will be set as new value!
|
|
|
|
*
|
|
|
|
* @param array $keys
|
|
|
|
* @param int $offset =1 how much to increment by
|
|
|
|
* @param int $intial_value =0 value to set and return, if not in cache
|
|
|
|
* @param int $expiration =0
|
|
|
|
* @return false|int new value on success, false on error
|
|
|
|
*/
|
|
|
|
function decrement(array $keys, int $offset=1, int $intial_value=0, int $expiration=0)
|
|
|
|
{
|
|
|
|
$ret = $this->memcache->decrement($key=self::key($keys), $offset, $intial_value, $expiration);
|
|
|
|
if ($ret === false && $this->retry > 0)
|
|
|
|
{
|
|
|
|
$ret = $this->memcache->decrement($key, $offset, $intial_value, $expiration);
|
|
|
|
}
|
|
|
|
// fix memcached returning strings
|
|
|
|
return $ret !== false ? (int)$ret : $ret;
|
|
|
|
}
|
|
|
|
|
2015-11-07 16:59:20 +01:00
|
|
|
/**
|
|
|
|
* Create a single key from $keys
|
|
|
|
*
|
|
|
|
* @param array $keys
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
private function key(array $keys)
|
|
|
|
{
|
|
|
|
return implode('::',$keys);
|
|
|
|
}
|
|
|
|
}
|