| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339 | 
							- <?php
 
- /**
 
-  * Credis, a Redis interface for the modest
 
-  *
 
-  * @author Justin Poliey <jdp34@njit.edu>
 
-  * @copyright 2009 Justin Poliey <jdp34@njit.edu>
 
-  * @license http://www.opensource.org/licenses/mit-license.php The MIT License
 
-  * @package Credis
 
-  */
 
- /**
 
-  * A generalized Credis_Client interface for a cluster of Redis servers
 
-  *
 
-  * @deprecated
 
-  */
 
- class Credis_Cluster
 
- {
 
-   /**
 
-    * Collection of Credis_Client objects attached to Redis servers
 
-    * @var Credis_Client[]
 
-    */
 
-   protected $clients;
 
-   /**
 
-    * If a server is set as master, all write commands go to that one
 
-    * @var Credis_Client
 
-    */
 
-   protected $masterClient;
 
-   /**
 
-    * Aliases of Credis_Client objects attached to Redis servers, used to route commands to specific servers
 
-    * @see Credis_Cluster::to
 
-    * @var array
 
-    */
 
-   protected $aliases;
 
-   /**
 
-    * Hash ring of Redis server nodes
 
-    * @var array
 
-    */
 
-   protected $ring;
 
-   /**
 
-    * Individual nodes of pointers to Redis servers on the hash ring
 
-    * @var array
 
-    */
 
-   protected $nodes;
 
-   /**
 
-    * The commands that are not subject to hashing
 
-    * @var array
 
-    * @access protected
 
-    */
 
-   protected $dont_hash;
 
-   /**
 
-    * Currently working cluster-wide database number.
 
-    * @var int
 
-    */
 
-   protected $selectedDb = 0;
 
-   /**
 
-    * Creates an interface to a cluster of Redis servers
 
-    * Each server should be in the format:
 
-    *  array(
 
-    *   'host' => hostname,
 
-    *   'port' => port,
 
-    *   'db' => db,
 
-    *   'password' => password,
 
-    *   'timeout' => timeout,
 
-    *   'alias' => alias,
 
-    *   'persistent' => persistence_identifier,
 
-    *   'master' => master
 
-    *   'write_only'=> true/false
 
-    * )
 
-    *
 
-    * @param array $servers The Redis servers in the cluster.
 
-    * @param int $replicas
 
-    * @param bool $standAlone
 
-    * @throws CredisException
 
-    */
 
-   public function __construct($servers, $replicas = 128, $standAlone = false)
 
-   {
 
-     $this->clients = array();
 
-     $this->masterClient = null;
 
-     $this->aliases = array();
 
-     $this->ring = array();
 
-     $this->replicas = (int)$replicas;
 
-     $client = null;
 
-     foreach ($servers as $server)
 
-     {
 
-       if(is_array($server)){
 
-           $client = new Credis_Client(
 
-             $server['host'],
 
-             $server['port'],
 
-             isset($server['timeout']) ? $server['timeout'] : 2.5,
 
-             isset($server['persistent']) ? $server['persistent'] : '',
 
-             isset($server['db']) ? $server['db'] : 0,
 
-             isset($server['password']) ? $server['password'] : null
 
-           );
 
-           if (isset($server['alias'])) {
 
-             $this->aliases[$server['alias']] = $client;
 
-           }
 
-           if(isset($server['master']) && $server['master'] === true){
 
-             $this->masterClient = $client;
 
-             if(isset($server['write_only']) && $server['write_only'] === true){
 
-                 continue;
 
-             }
 
-           }
 
-       } elseif($server instanceof Credis_Client){
 
-         $client = $server;
 
-       } else {
 
-           throw new CredisException('Server should either be an array or an instance of Credis_Client');
 
-       }
 
-       if($standAlone) {
 
-           $client->forceStandalone();
 
-       }
 
-       $this->clients[] = $client;
 
-       for ($replica = 0; $replica <= $this->replicas; $replica++) {
 
-           $md5num = hexdec(substr(md5($client->getHost().':'.$client->getPort().'-'.$replica),0,7));
 
-           $this->ring[$md5num] = count($this->clients)-1;
 
-       }
 
-     }
 
-     ksort($this->ring, SORT_NUMERIC);
 
-     $this->nodes = array_keys($this->ring);
 
-     $this->dont_hash = array_flip(array(
 
-       'RANDOMKEY', 'DBSIZE', 'PIPELINE', 'EXEC',
 
-       'SELECT',    'MOVE',    'FLUSHDB',  'FLUSHALL',
 
-       'SAVE',      'BGSAVE',  'LASTSAVE', 'SHUTDOWN',
 
-       'INFO',      'MONITOR', 'SLAVEOF'
 
-     ));
 
-     if($this->masterClient !== null && count($this->clients()) == 0){
 
-         $this->clients[] = $this->masterClient;
 
-         for ($replica = 0; $replica <= $this->replicas; $replica++) {
 
-             $md5num = hexdec(substr(md5($this->masterClient->getHost().':'.$this->masterClient->getHost().'-'.$replica),0,7));
 
-             $this->ring[$md5num] = count($this->clients)-1;
 
-         }
 
-         $this->nodes = array_keys($this->ring);
 
-     }
 
-   }
 
-   /**
 
-    * @param Credis_Client $masterClient
 
-    * @param bool $writeOnly
 
-    * @return Credis_Cluster
 
-    */
 
-   public function setMasterClient(Credis_Client $masterClient, $writeOnly=false)
 
-   {
 
-     if(!$masterClient instanceof Credis_Client){
 
-         throw new CredisException('Master client should be an instance of Credis_Client');
 
-     }
 
-     $this->masterClient = $masterClient;
 
-     if (!isset($this->aliases['master'])) {
 
-         $this->aliases['master'] = $masterClient;
 
-     }
 
-     if(!$writeOnly){
 
-         $this->clients[] = $this->masterClient;
 
-         for ($replica = 0; $replica <= $this->replicas; $replica++) {
 
-             $md5num = hexdec(substr(md5($this->masterClient->getHost().':'.$this->masterClient->getHost().'-'.$replica),0,7));
 
-             $this->ring[$md5num] = count($this->clients)-1;
 
-         }
 
-         $this->nodes = array_keys($this->ring);
 
-     }
 
-     return $this;
 
-   }
 
-   /**
 
-    * Get a client by index or alias.
 
-    *
 
-    * @param string|int $alias
 
-    * @throws CredisException
 
-    * @return Credis_Client
 
-    */
 
-   public function client($alias)
 
-   {
 
-     if (is_int($alias) && isset($this->clients[$alias])) {
 
-       return $this->clients[$alias];
 
-     }
 
-     else if (isset($this->aliases[$alias])) {
 
-       return $this->aliases[$alias];
 
-     }
 
-     throw new CredisException("Client $alias does not exist.");
 
-   }
 
-   /**
 
-    * Get an array of all clients
 
-    *
 
-    * @return array|Credis_Client[]
 
-    */
 
-   public function clients()
 
-   {
 
-     return $this->clients;
 
-   }
 
-   /**
 
-    * Execute a command on all clients
 
-    *
 
-    * @return array
 
-    */
 
-   public function all()
 
-   {
 
-     $args = func_get_args();
 
-     $name = array_shift($args);
 
-     $results = array();
 
-     foreach($this->clients as $client) {
 
-       $results[] = call_user_func_array([$client, $name], $args);
 
-     }
 
-     return $results;
 
-   }
 
-   /**
 
-    * Get the client that the key would hash to.
 
-    *
 
-    * @param string $key
 
-    * @return \Credis_Client
 
-    */
 
-   public function byHash($key)
 
-   {
 
-     return $this->clients[$this->hash($key)];
 
-   }
 
-   /**
 
-    * @param int $index
 
-    * @return void
 
-    */
 
-   public function select($index)
 
-   {
 
-       $this->selectedDb = (int) $index;
 
-   }
 
-   /**
 
-    * Execute a Redis command on the cluster with automatic consistent hashing and read/write splitting
 
-    *
 
-    * @param string $name
 
-    * @param array $args
 
-    * @return mixed
 
-    */
 
-   public function __call($name, $args)
 
-   {
 
-     if($this->masterClient !== null && !$this->isReadOnlyCommand($name)){
 
-         $client = $this->masterClient;
 
-     }elseif (count($this->clients()) == 1 || isset($this->dont_hash[strtoupper($name)]) || !isset($args[0])) {
 
-       $client = $this->clients[0];
 
-     }
 
-     else {
 
-       $client = $this->byHash($args[0]);
 
-     }
 
-     // Ensure that current client is working on the same database as expected.
 
-     if ($client->getSelectedDb() != $this->selectedDb) {
 
-       $client->select($this->selectedDb);
 
-     }
 
-     return call_user_func_array([$client, $name], $args);
 
-   }
 
-   /**
 
-    * Get client index for a key by searching ring with binary search
 
-    *
 
-    * @param string $key The key to hash
 
-    * @return int The index of the client object associated with the hash of the key
 
-    */
 
-   public function hash($key)
 
-   {
 
-     $needle = hexdec(substr(md5($key),0,7));
 
-     $server = $min = 0;
 
-     $max = count($this->nodes) - 1;
 
-     while ($max >= $min) {
 
-       $position = (int) (($min + $max) / 2);
 
-       $server = $this->nodes[$position];
 
-       if ($needle < $server) {
 
-         $max = $position - 1;
 
-       }
 
-       else if ($needle > $server) {
 
-         $min = $position + 1;
 
-       }
 
-       else {
 
-         break;
 
-       }
 
-     }
 
-     return $this->ring[$server];
 
-   }
 
-   public function isReadOnlyCommand($command)
 
-   {
 
-       static $readOnlyCommands = array(
 
-           'DBSIZE' => true,
 
-           'INFO' => true,
 
-           'MONITOR' => true,
 
-           'EXISTS' => true,
 
-           'TYPE' => true,
 
-           'KEYS' => true,
 
-           'SCAN' => true,
 
-           'RANDOMKEY' => true,
 
-           'TTL' => true,
 
-           'GET' => true,
 
-           'MGET' => true,
 
-           'SUBSTR' => true,
 
-           'STRLEN' => true,
 
-           'GETRANGE' => true,
 
-           'GETBIT' => true,
 
-           'LLEN' => true,
 
-           'LRANGE' => true,
 
-           'LINDEX' => true,
 
-           'SCARD' => true,
 
-           'SISMEMBER' => true,
 
-           'SINTER' => true,
 
-           'SUNION' => true,
 
-           'SDIFF' => true,
 
-           'SMEMBERS' => true,
 
-           'SSCAN' => true,
 
-           'SRANDMEMBER' => true,
 
-           'ZRANGE' => true,
 
-           'ZREVRANGE' => true,
 
-           'ZRANGEBYSCORE' => true,
 
-           'ZREVRANGEBYSCORE' => true,
 
-           'ZCARD' => true,
 
-           'ZSCORE' => true,
 
-           'ZCOUNT' => true,
 
-           'ZRANK' => true,
 
-           'ZREVRANK' => true,
 
-           'ZSCAN' => true,
 
-           'HGET' => true,
 
-           'HMGET' => true,
 
-           'HEXISTS' => true,
 
-           'HLEN' => true,
 
-           'HKEYS' => true,
 
-           'HVALS' => true,
 
-           'HGETALL' => true,
 
-           'HSCAN' => true,
 
-           'PING' => true,
 
-           'AUTH' => true,
 
-           'SELECT' => true,
 
-           'ECHO' => true,
 
-           'QUIT' => true,
 
-           'OBJECT' => true,
 
-           'BITCOUNT' => true,
 
-           'TIME' => true,
 
-           'SORT' => true,
 
-       );
 
-       return array_key_exists(strtoupper($command), $readOnlyCommands);
 
-   }
 
- }
 
 
  |