mirror of
https://github.com/yiisoft/yii2.git
synced 2025-11-01 20:19:42 +08:00
Redis cache implementation
This commit is contained in:
@ -21,7 +21,7 @@ use yii\base\InvalidConfigException;
|
||||
* MemCache can be configured with a list of memcache servers by settings its [[servers]] property.
|
||||
* By default, MemCache assumes there is a memcache server running on localhost at port 11211.
|
||||
*
|
||||
* See [[Cache]] for common cache operations that ApcCache supports.
|
||||
* See [[Cache]] for common cache operations that MemCache supports.
|
||||
*
|
||||
* Note, there is no security measure to protected data in memcache.
|
||||
* All data in memcache can be accessed by any process running in the system.
|
||||
@ -89,7 +89,7 @@ class MemCache extends Cache
|
||||
if (count($servers)) {
|
||||
foreach ($servers as $server) {
|
||||
if ($server->host === null) {
|
||||
throw new Exception("The 'host' property must be specified for every memcache server.");
|
||||
throw new InvalidConfigException("The 'host' property must be specified for every memcache server.");
|
||||
}
|
||||
if ($this->useMemcached) {
|
||||
$cache->addServer($server->host, $server->port, $server->weight);
|
||||
|
||||
191
framework/caching/RedisCache.php
Normal file
191
framework/caching/RedisCache.php
Normal file
@ -0,0 +1,191 @@
|
||||
<?php
|
||||
/**
|
||||
* @link http://www.yiiframework.com/
|
||||
* @copyright Copyright (c) 2008 Yii Software LLC
|
||||
* @license http://www.yiiframework.com/license/
|
||||
*/
|
||||
|
||||
namespace yii\caching;
|
||||
|
||||
use yii\db\redis\Connection;
|
||||
|
||||
/**
|
||||
* RedisCache implements a cache application component based on [redis](http://redis.io/).
|
||||
*
|
||||
* RedisCache needs to be configured with [[hostname]], [[port]] and [[database]] of the server
|
||||
* to connect to. By default RedisCache assumes there is a redis server running on localhost at
|
||||
* port 6379 and uses the database number 0.
|
||||
*
|
||||
* RedisCache also supports [the AUTH command](http://redis.io/commands/auth) of redis.
|
||||
* When the server needs authentication, you can set the [[password]] property to
|
||||
* authenticate with the server after connect.
|
||||
*
|
||||
* See [[Cache]] for common cache operations that RedisCache supports.
|
||||
*
|
||||
* To use RedisCache as the cache application component, configure the application as follows,
|
||||
*
|
||||
* ~~~
|
||||
* array(
|
||||
* 'components'=>array(
|
||||
* 'cache'=>array(
|
||||
* 'class'=>'RedisCache',
|
||||
* 'hostname'=>'localhost',
|
||||
* 'port'=>6379,
|
||||
* 'database'=>0,
|
||||
* ),
|
||||
* ),
|
||||
* )
|
||||
* ~~~
|
||||
*
|
||||
* In the above, two memcache servers are used: server1 and server2. You can configure more properties of
|
||||
* each server, such as `persistent`, `weight`, `timeout`. Please see [[MemCacheServer]] for available options.
|
||||
*
|
||||
* @property \Memcache|\Memcached $memCache The memcache instance (or memcached if [[useMemcached]] is true) used by this component.
|
||||
* @property MemCacheServer[] $servers List of memcache server configurations.
|
||||
*
|
||||
* @author Carsten Brandt <mail@cebe.cc>
|
||||
* @since 2.0
|
||||
*/
|
||||
class RedisCache extends Cache
|
||||
{
|
||||
/**
|
||||
* @var string hostname to use for connecting to the redis server. Defaults to 'localhost'.
|
||||
*/
|
||||
public $hostname = 'localhost';
|
||||
/**
|
||||
* @var int the to use for connecting to the redis server. Default port is 6379.
|
||||
*/
|
||||
public $port = 6379;
|
||||
/**
|
||||
* @var string the password to use to identify with the redis server. If not set, no AUTH command will be sent.
|
||||
*/
|
||||
public $password;
|
||||
/**
|
||||
* @var int the redis database to use. This is an integer value starting from 0. Defaults to 0.
|
||||
*/
|
||||
public $database = 0;
|
||||
/**
|
||||
* @var float timeout to use for connection to redis. If not set the timeout set in php.ini will be used: ini_get("default_socket_timeout")
|
||||
*/
|
||||
public $timeout = null;
|
||||
/**
|
||||
* @var \yii\db\redis\Connection the redis connection
|
||||
*/
|
||||
private $_connection;
|
||||
|
||||
|
||||
/**
|
||||
* Initializes the cache component by establishing a connection to the redis server.
|
||||
*/
|
||||
public function init()
|
||||
{
|
||||
parent::init();
|
||||
$this->getConnection();
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the redis connection object.
|
||||
* Establishes a connection to the redis server if it does not already exists.
|
||||
*
|
||||
* TODO throw exception on error
|
||||
* @return \yii\db\redis\Connection
|
||||
*/
|
||||
public function getConnection()
|
||||
{
|
||||
if ($this->_connection === null) {
|
||||
$this->_connection = new Connection(array(
|
||||
'dsn' => 'redis://' . $this->hostname . ':' . $this->port . '/' . $this->database,
|
||||
'password' => $this->password,
|
||||
'timeout' => $this->timeout,
|
||||
));
|
||||
}
|
||||
return $this->_connection;
|
||||
}
|
||||
|
||||
/**
|
||||
* Retrieves a value from cache with a specified key.
|
||||
* This is the implementation of the method declared in the parent class.
|
||||
* @param string $key a unique key identifying the cached value
|
||||
* @return string the value stored in cache, false if the value is not in the cache or expired.
|
||||
*/
|
||||
protected function getValue($key)
|
||||
{
|
||||
return $this->_connection->executeCommand('GET', $key);
|
||||
}
|
||||
|
||||
/**
|
||||
* Retrieves multiple values from cache with the specified keys.
|
||||
* @param array $keys a list of keys identifying the cached values
|
||||
* @return array a list of cached values indexed by the keys
|
||||
*/
|
||||
protected function getValues($keys)
|
||||
{
|
||||
return $this->_connection->executeCommand('MGET', $keys);
|
||||
}
|
||||
|
||||
/**
|
||||
* Stores a value identified by a key in cache.
|
||||
* This is the implementation of the method declared in the parent class.
|
||||
*
|
||||
* @param string $key the key identifying the value to be cached
|
||||
* @param string $value the value to be cached
|
||||
* @param float $expire the number of seconds in which the cached value will expire. 0 means never expire.
|
||||
* This can be a floating point number to specify the time in milliseconds.
|
||||
* @return boolean true if the value is successfully stored into cache, false otherwise
|
||||
*/
|
||||
protected function setValue($key,$value,$expire)
|
||||
{
|
||||
if ($expire == 0) {
|
||||
return (bool) $this->_connection->executeCommand('SET', array($key, $value));
|
||||
} else {
|
||||
$expire = (int) ($expire * 1000);
|
||||
return (bool) $this->_connection->executeCommand('PSETEX', array($key, $expire, $value));
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Stores a value identified by a key into cache if the cache does not contain this key.
|
||||
* This is the implementation of the method declared in the parent class.
|
||||
*
|
||||
* @param string $key the key identifying the value to be cached
|
||||
* @param string $value the value to be cached
|
||||
* @param float $expire the number of seconds in which the cached value will expire. 0 means never expire.
|
||||
* This can be a floating point number to specify the time in milliseconds.
|
||||
* @return boolean true if the value is successfully stored into cache, false otherwise
|
||||
*/
|
||||
protected function addValue($key,$value,$expire)
|
||||
{
|
||||
if ($expire == 0) {
|
||||
return (bool) $this->_connection->executeCommand('SETNX', array($key, $value));
|
||||
} else {
|
||||
// TODO consider requiring redis version >= 2.6.12 that supports this in one command
|
||||
$expire = (int) ($expire * 1000);
|
||||
$this->_connection->executeCommand('MULTI');
|
||||
$this->_connection->executeCommand('SETNX', array($key, $value));
|
||||
$this->_connection->executeCommand('PEXPIRE', array($key, $expire));
|
||||
$response = $this->_connection->executeCommand('EXEC');
|
||||
return (bool) $response[0];
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Deletes a value with the specified key from cache
|
||||
* This is the implementation of the method declared in the parent class.
|
||||
* @param string $key the key of the value to be deleted
|
||||
* @return boolean if no error happens during deletion
|
||||
*/
|
||||
protected function deleteValue($key)
|
||||
{
|
||||
return (bool) $this->_connection->executeCommand('DEL', array($key));
|
||||
}
|
||||
|
||||
/**
|
||||
* Deletes all values from cache.
|
||||
* This is the implementation of the method declared in the parent class.
|
||||
* @return boolean whether the flush operation was successful.
|
||||
*/
|
||||
protected function flushValues()
|
||||
{
|
||||
return $this->_connection->executeCommand('FLUSHDB');
|
||||
}
|
||||
}
|
||||
@ -45,6 +45,10 @@ class Connection extends Component
|
||||
* See http://redis.io/commands/auth
|
||||
*/
|
||||
public $password;
|
||||
/**
|
||||
* @var float timeout to use for connection to redis. If not set the timeout set in php.ini will be used: ini_get("default_socket_timeout")
|
||||
*/
|
||||
public $timeout = null;
|
||||
|
||||
/**
|
||||
* @var array List of available redis commands http://redis.io/commands
|
||||
@ -237,7 +241,12 @@ class Connection extends Component
|
||||
$db = isset($dsn[3]) ? $dsn[3] : 0;
|
||||
|
||||
\Yii::trace('Opening DB connection: ' . $this->dsn, __CLASS__);
|
||||
$this->_socket = @stream_socket_client($host, $errorNumber, $errorDescription);
|
||||
$this->_socket = @stream_socket_client(
|
||||
$host,
|
||||
$errorNumber,
|
||||
$errorDescription,
|
||||
$this->timeout ? $this->timeout : ini_get("default_socket_timeout")
|
||||
);
|
||||
if ($this->_socket) {
|
||||
if ($this->password !== null) {
|
||||
$this->executeCommand('AUTH', array($this->password));
|
||||
@ -337,6 +346,8 @@ class Connection extends Component
|
||||
* @param $name
|
||||
* @param $params
|
||||
* @return array|bool|null|string
|
||||
* Returns true on Status reply
|
||||
* TODO explain all reply types
|
||||
*/
|
||||
public function executeCommand($name, $params=array())
|
||||
{
|
||||
|
||||
@ -4,7 +4,7 @@ use yii\caching\MemCache;
|
||||
use yiiunit\TestCase;
|
||||
|
||||
/**
|
||||
* Class for testing memcache cache backend
|
||||
* Class for testing memcached cache backend
|
||||
*/
|
||||
class MemCachedTest extends CacheTest
|
||||
{
|
||||
|
||||
34
tests/unit/framework/caching/RedisCacheTest.php
Normal file
34
tests/unit/framework/caching/RedisCacheTest.php
Normal file
@ -0,0 +1,34 @@
|
||||
<?php
|
||||
namespace yiiunit\framework\caching;
|
||||
use yii\caching\MemCache;
|
||||
use yii\caching\RedisCache;
|
||||
use yiiunit\TestCase;
|
||||
|
||||
/**
|
||||
* Class for testing redis cache backend
|
||||
*/
|
||||
class RedisCacheTest extends CacheTest
|
||||
{
|
||||
private $_cacheInstance = null;
|
||||
|
||||
/**
|
||||
* @return MemCache
|
||||
*/
|
||||
protected function getCacheInstance()
|
||||
{
|
||||
$config = array(
|
||||
'hostname' => 'localhost',
|
||||
'port' => 6379,
|
||||
'database' => 0,
|
||||
);
|
||||
$dsn = $config['hostname'] . ':' .$config['port'];
|
||||
if(!@stream_socket_client($dsn, $errorNumber, $errorDescription, 0.5)) {
|
||||
$this->markTestSkipped('No redis server running at ' . $dsn .' : ' . $errorNumber . ' - ' . $errorDescription);
|
||||
}
|
||||
|
||||
if($this->_cacheInstance === null) {
|
||||
$this->_cacheInstance = new RedisCache($config);
|
||||
}
|
||||
return $this->_cacheInstance;
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user