Files
wy/ZeedFramework/library/Zeed/Cache/Backend/Mongo.php
2026-01-07 11:40:41 +08:00

460 lines
15 KiB
PHP

<?php
/**
* @see Zend_Cache_Backend
*/
/**
* @see Zend_Cache_Backend_ExtendedInterface
*/
/**
* @author Olivier Bregeras (Stunti) (olivier.bregeras@gmail.com)
* @category Stunti
* @package Stunti_Cache
* @subpackage Stunti_Cache_Backend
* @copyright Copyright (c) 2009 Stunti. (http://www.stunti.org)
* @license http://stunti.org/license/new-bsd New BSD License
*/
class Zeed_Cache_Backend_Mongo extends Zend_Cache_Backend implements Zend_Cache_Backend_ExtendedInterface
{
const DEFAULT_HOST = '127.0.0.1';
const DEFAULT_PORT = 27017;
const DEFAULT_PERSISTENT = true;
const DEFAULT_DBNAME = 'Db_Cache';
const DEFAULT_COLLECTION = 'C_Cache';
protected $_conn;
protected $_db;
protected $_collection;
/**
* Available options
*
* =====> (array) servers :
* an array of mongodb server ; each mongodb server is described by an associative array :
* 'host' => (string) : the name of the mongodb server
* 'port' => (int) : the port of the mongodb server
* 'persistent' => (bool) : use or not persistent connections to this mongodb server
* 'collection' => (string) : name of the collection to use
* 'dbname' => (string) : name of the database to use
*
* @var array available options
*/
protected $_options = array(
'host' => self::DEFAULT_HOST,
'port' => self::DEFAULT_PORT,
'persistent' => self::DEFAULT_PERSISTENT,
'collection' => self::DEFAULT_COLLECTION,
'dbname' => self::DEFAULT_DBNAME,
);
/**
* @return void
*/
public function __construct($options)
{
if (!extension_loaded('mongo')) {
Zend_Cache::throwException('The MongoDB extension must be loaded for using this backend !');
}
parent::__construct($options);
// Merge the options passed in; overridding any default options
$this->_options = array_merge($this->_options, $options);
$this->_conn = new Mongo($this->_options['host'], $this->_options['port'], $this->_options['persistent']);
$this->_db = $this->_conn->selectDB($this->_options['dbname']);
$this->_collection = $this->_db->selectCollection($this->_options['collection']);
}
/**
* Expires a record (mostly used for testing purposes)
* @param string $id
* @return void
*/
public function ___expire($id)
{
$cursor = $this->get($id);
if ($tmp = $cursor->getNext()) {
$tmp['l'] = -10;
$this->_collection->save($tmp);
}
}
/**
* Test if a cache is available for the given id and (if yes) return it (false else)
*
* @param string $id Cache id
* @param boolean $doNotTestCacheValidity If set to true, the cache validity won't be tested
* @return string|false cached datas
*/
public function load($id, $doNotTestCacheValidity = false)
{
$cursor = $this->get($id);
if ($tmp = $cursor->getNext()) {
if ($doNotTestCacheValidity || !$doNotTestCacheValidity && ($tmp['created_at'] + $tmp['l'])>=time()) {
return $tmp['d'];
}
return false;
}
return false;
}
/**
* Test if a cache is available or not (for the given id)
*
* @param string $id Cache id
* @return mixed|false (a cache is not available) or "last modified" timestamp (int) of the available cache record
*/
public function test($id)
{
$cursor = $this->get($id);
if ($tmp = $cursor->getNext()) {
return $tmp['created_at'];
}
return false;
}
/**
* Save some string datas into a cache record
*
* Note : $data is always "string" (serialization is done by the
* core not by the backend)
*
* @param string $data Datas to cache
* @param string $id Cache id
* @param array $tags Array of strings, the cache record will be tagged by each string entry
* @param int $specificLifetime If != false, set a specific lifetime for this cache record (null => infinite lifetime)
* @return boolean True if no problem
*/
public function save($data, $id, $tags = array(), $specificLifetime = false)
{
$lifetime = $this->getLifetime($specificLifetime);
$flag = 0;
// #ZF-5702 : we try add() first becase set() seems to be slower
$result = $this->set($id, $data, $lifetime,$tags);
return $result;
}
/**
* Remove a cache record
*
* @param string $id Cache id
* @return boolean True if no problem
*/
public function remove($id)
{
return $this->_collection->remove(array('_id' => $id));
}
/**
* Clean some cache records (protected method used for recursive stuff)
*
* Available modes are :
* Zend_Cache::CLEANING_MODE_ALL (default) => remove all cache entries ($tags is not used)
* Zend_Cache::CLEANING_MODE_OLD => remove too old cache entries ($tags is not used)
* Zend_Cache::CLEANING_MODE_MATCHING_TAG => remove cache entries matching all given tags
* ($tags can be an array of strings or a single string)
* Zend_Cache::CLEANING_MODE_NOT_MATCHING_TAG => remove cache entries not {matching one of the given tags}
* ($tags can be an array of strings or a single string)
* Zend_Cache::CLEANING_MODE_MATCHING_ANY_TAG => remove cache entries matching any given tags
* ($tags can be an array of strings or a single string)
*
* @param string $dir Directory to clean
* @param string $mode Clean mode
* @param array $tags Array of tags
* @throws Zend_Cache_Exception
* @return boolean True if no problem
*/
public function clean($mode = Zend_Cache::CLEANING_MODE_ALL, $tags = array())
{
switch ($mode) {
case Zend_Cache::CLEANING_MODE_ALL:
return $this->_collection->remove();
break;
case Zend_Cache::CLEANING_MODE_OLD:
//$res = $this->_instance->findOneCond(array('$where' => new MongoCode('function() { return (this.l + this.created_at) < '.(time()-1).'; }')));
//var_dump($res);exit;
return $this->_collection->remove(array('$where' => new MongoCode('function() { return (this.l + this.created_at) < '.(time()-1).'; }')));
break;
case Zend_Cache::CLEANING_MODE_MATCHING_TAG:
return $this->_collection->remove(array( 't' => array( '$all' => $tags ) ));
break;
case Zend_Cache::CLEANING_MODE_NOT_MATCHING_TAG:
return $this->_collection->remove(array( 't' => array( '$nin' => $tags ) ));
break;
case Zend_Cache::CLEANING_MODE_MATCHING_ANY_TAG:
//find all tags and remove them
//$this->_log(self::TAGS_UNSUPPORTED_BY_CLEAN_OF_MEMCACHED_BACKEND);
return $this->_collection->remove(array( 't' => array( '$in' => $tags ) ));
break;
default:
Zend_Cache::throwException('Invalid mode for clean() method');
break;
}
}
/**
* Return true if the automatic cleaning is available for the backend
*
* @return boolean
*/
public function isAutomaticCleaningAvailable()
{
return false;
}
/**
* Set the frontend directives
*
* @param array $directives Assoc of directives
* @throws Zend_Cache_Exception
* @return void
*/
public function setDirectives($directives)
{
parent::setDirectives($directives);
$lifetime = $this->getLifetime(false);
if ($lifetime === null) {
// #ZF-4614 : we tranform null to zero to get the maximal lifetime
parent::setDirectives(array('lifetime' => 0));
}
}
/**
* Return an array of stored cache ids
*
* @return array array of stored cache ids (string)
*/
public function getIds() {
$cursor = $this->_collection->find();
$ret = array();
while ($tmp = $cursor->getNext()) {
$ret[] = $tmp['_id'];
}
return $ret;
}
/**
* Return an array of stored tags
*
* @return array array of stored tags (string)
*/
public function getTags() {
//might have to use map reduce for that (example on Mongodb doc)
$cmd['mapreduce'] = $this->_options['collection'];
//$cmd['verbose'] = true;
$cmd['map'] = 'function(){
this.t.forEach(
function(z){
emit( z , { count : 1 } );
}
);
};';
$cmd['reduce'] = 'function( key , values ){
var total = 0;
for ( var i=0; i<values.length; i++ )
total += values[i].count;
return { count : total };
};
';
$res2 = $this->_db->command($cmd);
$res3 = $this->_db->selectCollection($res2['result'])->find();
$res = array();
foreach ($res3 as $key => $val) {
$res[] = $key;
}
$this->_db->dropCollection($res2['result']);
return $res;
}
public function drop()
{
return $this->_collection->drop();
}
/**
* Return an array of stored cache ids which match given tags
*
* In case of multiple tags, a logical AND is made between tags
*
* @param array $tags array of tags
* @return array array of matching cache ids (string)
*/
public function getIdsMatchingTags($tags = array()) {
$cursor = $this->_collection->find(array( 't' => array( '$all' => $tags )));
$ret = array();
while ($tmp = $cursor->getNext()) {
$ret[] = $tmp['_id'];
}
return $ret;
}
/**
* Return an array of stored cache ids which don't match given tags
*
* In case of multiple tags, a logical OR is made between tags
*
* @param array $tags array of tags
* @return array array of not matching cache ids (string)
*/
public function getIdsNotMatchingTags($tags = array()) {
$cursor = $this->_collection->find(array( 't' => array( '$nin' => $tags ) ));
$ret = array();
while ($tmp = $cursor->getNext()) {
$ret[] = $tmp['_id'];
}
return $ret;
}
/**
* Return an array of stored cache ids which match any given tags
*
* In case of multiple tags, a logical AND is made between tags
*
* @param array $tags array of tags
* @return array array of any matching cache ids (string)
*/
public function getIdsMatchingAnyTags($tags = array()) {
$res = $this->_collection->find(array( 't' => array( '$in' => $tags ) ));
$ret = array();
while ($tmp = $cursor->getNext()) {
$ret[] = $tmp['_id'];
}
return $ret;
}
/**
* No way to find the remaining space right now. So retrun 0.
*
* @throws Zend_Cache_Exception
* @return int integer between 0 and 100
*/
public function getFillingPercentage() {
return 1;
}
/**
* Return an array of metadatas for the given cache id
*
* The array must include these keys :
* - expire : the expire timestamp
* - tags : a string array of tags
* - mtime : timestamp of last modification time
*
* @param string $id cache id
* @return array array of metadatas (false if the cache id is not found)
*/
public function getMetadatas($id)
{
$cursor = $this->get($id);
if ($tmp = $cursor->getNext()) {
$data = $tmp['d'];
$mtime = $tmp['created_at'];
$lifetime = $tmp['l'];
return array(
'expire' => $mtime + $lifetime,
'tags' => $tmp['t'],
'mtime' => $mtime
);
}
return false;
}
/**
* Give (if possible) an extra lifetime to the given cache id
*
* @param string $id cache id
* @param int $extraLifetime
* @return boolean true if ok
*/
public function touch($id, $extraLifetime)
{
$cursor = $this->get($id);
if ($tmp = $cursor->getNext()) {
$data = $tmp['d'];
$mtime = $tmp['created_at'];
$lifetime = $tmp['l'];
$tags = $tmp['t'];
$newLifetime = $lifetime - (time() - $mtime) + $extraLifetime;
if ($newLifetime <=0) {
return false;
}
// #ZF-5702 : we try replace() first becase set() seems to be slower
$result = $this->set($id, $data, $newLifetime,$tags);
return $result;
}
return false;
}
/**
* Return an associative array of capabilities (booleans) of the backend
*
* The array must include these keys :
* - automatic_cleaning (is automating cleaning necessary)
* - tags (are tags supported)
* - expired_read (is it possible to read expired cache records
* (for doNotTestCacheValidity option for example))
* - priority does the backend deal with priority when saving
* - infinite_lifetime (is infinite lifetime can work with this backend)
* - get_list (is it possible to get the list of cache ids and the complete list of tags)
*
* @return array associative of with capabilities
*/
public function getCapabilities()
{
return array(
'automatic_cleaning' => true,
'tags' => true,
'expired_read' => true,
'priority' => false,
'infinite_lifetime' => true,
'get_list' => true
);
}
/**
* @param int $id
* @param array $data
* @param int $lifetime
* @param mixed $tags
* @return boolean
*/
function set($id, $data, $lifetime, $tags)
{
return $this->_collection->save(array('_id' => $id,
'd' => $data,
'created_at' => time(),
'l' => $lifetime,
't' => $tags
));
}
/**
* @param int $id
* @return array|false
*/
function get($id)
{
return $this->_collection->find(array('_id' => $id));
}
}