mirror of
https://github.com/php-win-ext/phpredis.git
synced 2026-03-29 20:42:13 +02:00
The php7 api has substantial changes in the way one goes about interacting with the zval type (php's internal structure for dealing with dynamically typed variables). Most notably, most everything is dereferenced by one pointer. So, where you once used zval** you typically use zval*, and where you used zval* you now just use a zval struct on the stack. In addition, the api changed in how you return a string. Previously you had the option of whether or not to "duplicate" the memory returned (or in other words, pass ownership of the pointer to PHP or not). Because phpredis sometimes buffers commands to the server (say in the case of a pipeline, or a MULTI/EXEC) transaction, we had several stack violations and/or memory corruption which resulted from keeping track of the address of a stack allocated structure, which was accessed at a later date (sometimes causing segmentation faults, bus errors, etc). Also, there were a few places where this pattern was being used: zval **ptr = emalloc(sizeof(zval*) * count); /* stuff */ ZVAL_STRING(ptr[0], "hello world"); Given that ZVAL_STRING() thinks it's writing to an allocated structure it would cause various random issues (because we were blowing through the data segment and writing into the code section). This commit (so far) fixes all of the segmentation faults and memory errors but there are still leaks (specifically in RedisArray) that need to be solved. Addresses #727
543 lines
16 KiB
PHP
543 lines
16 KiB
PHP
<?php defined('PHPREDIS_TESTRUN') or die("Use TestRedis.php to run tests!\n");
|
|
require_once(dirname($_SERVER['PHP_SELF'])."/TestSuite.php");
|
|
|
|
define('REDIS_ARRAY_DATA_SIZE', 1000);
|
|
|
|
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) = explode(':', $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, 0);
|
|
$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) = explode(':', $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, $str_filter) {
|
|
// 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, $str_filter);
|
|
}
|
|
|
|
?>
|