<?php
|
|
require_once(dirname($_SERVER['PHP_SELF'])."/test.php");
|
echo "Redis Array tests.\n\n";
|
|
function custom_hash($str) {
|
// str has the following format: $APPID_fb$FACEBOOKID_$key.
|
$pos = strpos($str, '_fb');
|
if(preg_match("#\w+_fb(?<facebook_id>\d+)_\w+#", $str, $out)) {
|
return $out['facebook_id'];
|
}
|
return $str;
|
}
|
|
class Redis_Array_Test extends TestSuite
|
{
|
private $strings;
|
public $ra = NULL;
|
private $data = NULL;
|
|
public function setUp() {
|
|
// initialize strings.
|
$n = REDIS_ARRAY_DATA_SIZE;
|
$this->strings = array();
|
for($i = 0; $i < $n; $i++) {
|
$this->strings['key-'.$i] = 'val-'.$i;
|
}
|
|
global $newRing, $oldRing, $useIndex;
|
$this->ra = new RedisArray($newRing, array('previous' => $oldRing, 'index' => $useIndex));
|
}
|
|
public function testMSet() {
|
// run mset
|
$this->assertTrue(TRUE === $this->ra->mset($this->strings));
|
|
// check each key individually using the array
|
foreach($this->strings as $k => $v) {
|
$this->assertTrue($v === $this->ra->get($k));
|
}
|
|
// check each key individually using a new connection
|
foreach($this->strings as $k => $v) {
|
list($host, $port) = split(':', $this->ra->_target($k));
|
|
$r = new Redis;
|
$r->pconnect($host, (int)$port);
|
$this->assertTrue($v === $r->get($k));
|
}
|
}
|
|
public function testMGet() {
|
$this->assertTrue(array_values($this->strings) === $this->ra->mget(array_keys($this->strings)));
|
}
|
|
private function addData($commonString) {
|
$this->data = array();
|
for($i = 0; $i < REDIS_ARRAY_DATA_SIZE; $i++) {
|
$k = rand().'_'.$commonString.'_'.rand();
|
$this->data[$k] = rand();
|
}
|
$this->ra->mset($this->data);
|
}
|
|
private function checkCommonLocality() {
|
// check that they're all on the same node.
|
$lastNode = NULL;
|
foreach($this->data as $k => $v) {
|
$node = $this->ra->_target($k);
|
if($lastNode) {
|
$this->assertTrue($node === $lastNode);
|
}
|
$this->assertTrue($this->ra->get($k) == $v);
|
$lastNode = $node;
|
}
|
}
|
|
public function testKeyLocality() {
|
|
// basic key locality with default hash
|
$this->addData('{hashed part of the key}');
|
$this->checkCommonLocality();
|
|
// with common hashing function
|
global $newRing, $oldRing, $useIndex;
|
$this->ra = new RedisArray($newRing, array('previous' => $oldRing,
|
'index' => $useIndex,
|
'function' => 'custom_hash'));
|
|
// basic key locality with custom hash
|
$this->addData('fb'.rand());
|
$this->checkCommonLocality();
|
}
|
|
public function customDistributor($key)
|
{
|
$a = unpack("N*", md5($key, true));
|
global $newRing;
|
$pos = abs($a[1]) % count($newRing);
|
|
return $pos;
|
}
|
|
public function testKeyDistributor()
|
{
|
global $newRing, $useIndex;
|
$this->ra = new RedisArray($newRing, array(
|
'index' => $useIndex,
|
'function' => 'custom_hash',
|
'distributor' => array($this, "customDistributor")));
|
|
// custom key distribution function.
|
$this->addData('fb'.rand());
|
|
// check that they're all on the expected node.
|
$lastNode = NULL;
|
foreach($this->data as $k => $v) {
|
$node = $this->ra->_target($k);
|
$pos = $this->customDistributor($k);
|
$this->assertTrue($node === $newRing[$pos]);
|
}
|
}
|
|
}
|
|
class Redis_Rehashing_Test extends TestSuite
|
{
|
|
public $ra = NULL;
|
private $useIndex;
|
|
// data
|
private $strings;
|
private $sets;
|
private $lists;
|
private $hashes;
|
private $zsets;
|
|
public function setUp() {
|
|
// initialize strings.
|
$n = REDIS_ARRAY_DATA_SIZE;
|
$this->strings = array();
|
for($i = 0; $i < $n; $i++) {
|
$this->strings['key-'.$i] = 'val-'.$i;
|
}
|
|
// initialize sets
|
for($i = 0; $i < $n; $i++) {
|
// each set has 20 elements
|
$this->sets['set-'.$i] = range($i, $i+20);
|
}
|
|
// initialize lists
|
for($i = 0; $i < $n; $i++) {
|
// each list has 20 elements
|
$this->lists['list-'.$i] = range($i, $i+20);
|
}
|
|
// initialize hashes
|
for($i = 0; $i < $n; $i++) {
|
// each hash has 5 keys
|
$this->hashes['hash-'.$i] = array('A' => $i, 'B' => $i+1, 'C' => $i+2, 'D' => $i+3, 'E' => $i+4);
|
}
|
|
// initialize sorted sets
|
for($i = 0; $i < $n; $i++) {
|
// each sorted sets has 5 elements
|
$this->zsets['zset-'.$i] = array($i, 'A', $i+1, 'B', $i+2, 'C', $i+3, 'D', $i+4, 'E');
|
}
|
|
global $newRing, $oldRing, $useIndex;
|
|
// create array
|
$this->ra = new RedisArray($newRing, array('previous' => $oldRing, 'index' => $useIndex));
|
}
|
|
public function testFlush() {
|
|
// flush all servers first.
|
global $serverList;
|
foreach($serverList as $s) {
|
list($host, $port) = explode(':', $s);
|
|
$r = new Redis;
|
$r->pconnect($host, (int)$port);
|
$r->flushdb();
|
}
|
}
|
|
|
private function distributeKeys() {
|
|
// strings
|
foreach($this->strings as $k => $v) {
|
$this->ra->set($k, $v);
|
}
|
|
// sets
|
foreach($this->sets as $k => $v) {
|
call_user_func_array(array($this->ra, 'sadd'), array_merge(array($k), $v));
|
}
|
|
// lists
|
foreach($this->lists as $k => $v) {
|
call_user_func_array(array($this->ra, 'rpush'), array_merge(array($k), $v));
|
}
|
|
// hashes
|
foreach($this->hashes as $k => $v) {
|
$this->ra->hmset($k, $v);
|
}
|
|
// sorted sets
|
foreach($this->zsets as $k => $v) {
|
call_user_func_array(array($this->ra, 'zadd'), array_merge(array($k), $v));
|
}
|
}
|
|
public function testDistribution() {
|
|
$this->distributeKeys();
|
}
|
|
public function testSimpleRead() {
|
|
$this->readAllvalues();
|
}
|
|
private function readAllvalues() {
|
|
// strings
|
foreach($this->strings as $k => $v) {
|
$this->assertTrue($this->ra->get($k) === $v);
|
}
|
|
// sets
|
foreach($this->sets as $k => $v) {
|
$ret = $this->ra->smembers($k); // get values
|
|
// sort sets
|
sort($v);
|
sort($ret);
|
|
$this->assertTrue($ret == $v);
|
}
|
|
// lists
|
foreach($this->lists as $k => $v) {
|
$ret = $this->ra->lrange($k, 0, -1);
|
$this->assertTrue($ret == $v);
|
}
|
|
// hashes
|
foreach($this->hashes as $k => $v) {
|
$ret = $this->ra->hgetall($k); // get values
|
$this->assertTrue($ret == $v);
|
}
|
|
// sorted sets
|
foreach($this->zsets as $k => $v) {
|
$ret = $this->ra->zrange($k, 0, -1, TRUE); // get values with scores
|
|
// create assoc array from local dataset
|
$tmp = array();
|
for($i = 0; $i < count($v); $i += 2) {
|
$tmp[$v[$i+1]] = $v[$i];
|
}
|
|
// compare to RA value
|
$this->assertTrue($ret == $tmp);
|
}
|
}
|
|
// add a new node.
|
public function testCreateSecondRing() {
|
|
global $newRing, $oldRing, $serverList;
|
$oldRing = $newRing; // back up the original.
|
$newRing = $serverList; // add a new node to the main ring.
|
}
|
|
public function testReadUsingFallbackMechanism() {
|
$this->readAllvalues(); // some of the reads will fail and will go to another target node.
|
}
|
|
public function testRehash() {
|
$this->ra->_rehash(); // this will redistribute the keys
|
}
|
|
public function testRehashWithCallback() {
|
$total = 0;
|
$this->ra->_rehash(function ($host, $count) use (&$total) {
|
$total += $count;
|
});
|
$this->assertTrue($total > 0);
|
}
|
|
public function testReadRedistributedKeys() {
|
$this->readAllvalues(); // we shouldn't have any missed reads now.
|
}
|
}
|
|
// Test auto-migration of keys
|
class Redis_Auto_Rehashing_Test extends TestSuite {
|
|
public $ra = NULL;
|
|
// data
|
private $strings;
|
|
public function setUp() {
|
|
// initialize strings.
|
$n = REDIS_ARRAY_DATA_SIZE;
|
$this->strings = array();
|
for($i = 0; $i < $n; $i++) {
|
$this->strings['key-'.$i] = 'val-'.$i;
|
}
|
|
global $newRing, $oldRing, $useIndex;
|
|
// create array
|
$this->ra = new RedisArray($newRing, array('previous' => $oldRing, 'index' => $useIndex, 'autorehash' => TRUE));
|
}
|
|
public function testDistribute() {
|
// strings
|
foreach($this->strings as $k => $v) {
|
$this->ra->set($k, $v);
|
}
|
}
|
|
private function readAllvalues() {
|
foreach($this->strings as $k => $v) {
|
$this->assertTrue($this->ra->get($k) === $v);
|
}
|
}
|
|
|
public function testReadAll() {
|
$this->readAllvalues();
|
}
|
|
// add a new node.
|
public function testCreateSecondRing() {
|
|
global $newRing, $oldRing, $serverList;
|
$oldRing = $newRing; // back up the original.
|
$newRing = $serverList; // add a new node to the main ring.
|
}
|
|
// Read and migrate keys on fallback, causing the whole ring to be rehashed.
|
public function testReadAndMigrateAll() {
|
$this->readAllvalues();
|
}
|
|
// Read and migrate keys on fallback, causing the whole ring to be rehashed.
|
public function testAllKeysHaveBeenMigrated() {
|
foreach($this->strings as $k => $v) {
|
// get the target for each key
|
$target = $this->ra->_target($k);
|
|
// connect to the target host
|
list($host,$port) = split(':', $target);
|
$r = new Redis;
|
$r->pconnect($host, $port);
|
|
$this->assertTrue($v === $r->get($k)); // check that the key has actually been migrated to the new node.
|
}
|
}
|
}
|
|
// Test node-specific multi/exec
|
class Redis_Multi_Exec_Test extends TestSuite {
|
|
public $ra = NULL;
|
|
public function setUp() {
|
|
global $newRing, $oldRing, $useIndex;
|
// create array
|
$this->ra = new RedisArray($newRing, array('previous' => $oldRing, 'index' => $useIndex));
|
}
|
|
public function testInit() {
|
$this->ra->set('{groups}:managers', 2);
|
$this->ra->set('{groups}:executives', 3);
|
|
$this->ra->set('1_{employee:joe}_name', 'joe');
|
$this->ra->set('1_{employee:joe}_group', 2);
|
$this->ra->set('1_{employee:joe}_salary', 2000);
|
}
|
|
public function testKeyDistribution() {
|
// check that all of joe's keys are on the same instance
|
$lastNode = NULL;
|
foreach(array('name', 'group', 'salary') as $field) {
|
$node = $this->ra->_target('1_{employee:joe}_'.$field);
|
if($lastNode) {
|
$this->assertTrue($node === $lastNode);
|
}
|
$lastNode = $node;
|
}
|
}
|
|
public function testMultiExec() {
|
|
// Joe gets a promotion
|
$newGroup = $this->ra->get('{groups}:executives');
|
$newSalary = 4000;
|
|
// change both in a transaction.
|
$host = $this->ra->_target('{employee:joe}'); // transactions are per-node, so we need a reference to it.
|
$tr = $this->ra->multi($host)
|
->set('1_{employee:joe}_group', $newGroup)
|
->set('1_{employee:joe}_salary', $newSalary)
|
->exec();
|
|
// check that the group and salary have been changed
|
$this->assertTrue($this->ra->get('1_{employee:joe}_group') === $newGroup);
|
$this->assertTrue($this->ra->get('1_{employee:joe}_salary') == $newSalary);
|
|
}
|
|
public function testMultiExecMSet() {
|
|
global $newGroup, $newSalary;
|
$newGroup = 1;
|
$newSalary = 10000;
|
|
// test MSET, making Joe a top-level executive
|
$out = $this->ra->multi($this->ra->_target('{employee:joe}'))
|
->mset(array('1_{employee:joe}_group' => $newGroup, '1_{employee:joe}_salary' => $newSalary))
|
->exec();
|
|
$this->assertTrue($out[0] === TRUE);
|
}
|
|
public function testMultiExecMGet() {
|
|
global $newGroup, $newSalary;
|
|
// test MGET
|
$out = $this->ra->multi($this->ra->_target('{employee:joe}'))
|
->mget(array('1_{employee:joe}_group', '1_{employee:joe}_salary'))
|
->exec();
|
|
$this->assertTrue($out[0][0] == $newGroup);
|
$this->assertTrue($out[0][1] == $newSalary);
|
}
|
|
public function testMultiExecDel() {
|
|
// test DEL
|
$out = $this->ra->multi($this->ra->_target('{employee:joe}'))
|
->del('1_{employee:joe}_group', '1_{employee:joe}_salary')
|
->exec();
|
|
$this->assertTrue($out[0] === 2);
|
$this->assertTrue($this->ra->exists('1_{employee:joe}_group') === FALSE);
|
$this->assertTrue($this->ra->exists('1_{employee:joe}_salary') === FALSE);
|
}
|
|
public function testDiscard() {
|
/* phpredis issue #87 */
|
$key = 'test_err';
|
|
$this->assertTrue($this->ra->set($key, 'test'));
|
$this->assertTrue('test' === $this->ra->get($key));
|
|
$this->ra->watch($key);
|
|
// After watch, same
|
$this->assertTrue('test' === $this->ra->get($key));
|
|
// change in a multi/exec block.
|
$ret = $this->ra->multi($this->ra->_target($key))->set($key, 'test1')->exec();
|
$this->assertTrue($ret === array(true));
|
|
// Get after exec, 'test1':
|
$this->assertTrue($this->ra->get($key) === 'test1');
|
|
$this->ra->watch($key);
|
|
// After second watch, still test1.
|
$this->assertTrue($this->ra->get($key) === 'test1');
|
|
$ret = $this->ra->multi($this->ra->_target($key))->set($key, 'test2')->discard();
|
// Ret after discard: NULL";
|
$this->assertTrue($ret === NULL);
|
|
// Get after discard, unchanged:
|
$this->assertTrue($this->ra->get($key) === 'test1');
|
}
|
|
}
|
|
// Test custom distribution function
|
class Redis_Distributor_Test extends TestSuite {
|
|
public $ra = NULL;
|
|
public function setUp() {
|
|
global $newRing, $oldRing, $useIndex;
|
// create array
|
$this->ra = new RedisArray($newRing, array('previous' => $oldRing, 'index' => $useIndex, 'distributor' => array($this, 'distribute')));
|
}
|
|
public function testInit() {
|
$this->ra->set('{uk}test', 'joe');
|
$this->ra->set('{us}test', 'bob');
|
}
|
|
public function distribute($key) {
|
$matches = array();
|
if (preg_match('/{([^}]+)}.*/', $key, $matches) == 1) {
|
$countries = array('uk' => 0, 'us' => 1);
|
if (array_key_exists($matches[1], $countries)) {
|
return $countries[$matches[1]];
|
}
|
}
|
return 2; // default server
|
}
|
|
public function testDistribution() {
|
$ukServer = $this->ra->_target('{uk}test');
|
$usServer = $this->ra->_target('{us}test');
|
$deServer = $this->ra->_target('{de}test');
|
$defaultServer = $this->ra->_target('unknown');
|
|
$nodes = $this->ra->_hosts();
|
$this->assertTrue($ukServer === $nodes[0]);
|
$this->assertTrue($usServer === $nodes[1]);
|
$this->assertTrue($deServer === $nodes[2]);
|
$this->assertTrue($defaultServer === $nodes[2]);
|
}
|
}
|
|
function run_tests($className) {
|
// reset rings
|
global $newRing, $oldRing, $serverList;
|
$newRing = array('localhost:6379', 'localhost:6380', 'localhost:6381');
|
$oldRing = array();
|
$serverList = array('localhost:6379', 'localhost:6380', 'localhost:6381', 'localhost:6382');
|
|
// run
|
TestSuite::run($className);
|
}
|
|
define('REDIS_ARRAY_DATA_SIZE', 1000);
|
|
global $useIndex;
|
foreach(array(true, false) as $useIndex) {
|
|
echo "\n".($useIndex?"WITH":"WITHOUT"). " per-node index:\n";
|
|
run_tests('Redis_Array_Test');
|
run_tests('Redis_Rehashing_Test');
|
run_tests('Redis_Auto_Rehashing_Test');
|
run_tests('Redis_Multi_Exec_Test');
|
run_tests('Redis_Distributor_Test');
|
}
|
|
?>
|