|
@@ -0,0 +1,374 @@
|
|
|
|
+<?php
|
|
|
|
+
|
|
|
|
+/*
|
|
|
|
+ * This file is part of the Predis package.
|
|
|
|
+ *
|
|
|
|
+ * (c) Daniele Alessandri <suppakilla@gmail.com>
|
|
|
|
+ *
|
|
|
|
+ * For the full copyright and license information, please view the LICENSE
|
|
|
|
+ * file that was distributed with this source code.
|
|
|
|
+ */
|
|
|
|
+
|
|
|
|
+namespace Predis\Replication;
|
|
|
|
+
|
|
|
|
+use \PHPUnit_Framework_TestCase as StandardTestCase;
|
|
|
|
+
|
|
|
|
+use Predis\Command\CommandInterface;
|
|
|
|
+use Predis\Profile\ServerProfile;
|
|
|
|
+
|
|
|
|
+/**
|
|
|
|
+ *
|
|
|
|
+ */
|
|
|
|
+class ReplicationStrategyTest extends StandardTestCase
|
|
|
|
+{
|
|
|
|
+ /**
|
|
|
|
+ * @group disconnected
|
|
|
|
+ */
|
|
|
|
+ public function testReadCommands()
|
|
|
|
+ {
|
|
|
|
+ $profile = ServerProfile::getDevelopment();
|
|
|
|
+ $strategy = new ReplicationStrategy();
|
|
|
|
+
|
|
|
|
+ foreach ($this->getExpectedCommands('read') as $commandId) {
|
|
|
|
+ $command = $profile->createCommand($commandId);
|
|
|
|
+ $this->assertTrue($strategy->isReadOperation($command));
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /**
|
|
|
|
+ * @group disconnected
|
|
|
|
+ */
|
|
|
|
+ public function testWriteCommands()
|
|
|
|
+ {
|
|
|
|
+ $profile = ServerProfile::getDevelopment();
|
|
|
|
+ $strategy = new ReplicationStrategy();
|
|
|
|
+
|
|
|
|
+ foreach ($this->getExpectedCommands('write') as $commandId) {
|
|
|
|
+ $command = $profile->createCommand($commandId);
|
|
|
|
+ $this->assertFalse($strategy->isReadOperation($command), $commandId);
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /**
|
|
|
|
+ * @group disconnected
|
|
|
|
+ */
|
|
|
|
+ public function testDisallowedCommands()
|
|
|
|
+ {
|
|
|
|
+ $profile = ServerProfile::getDevelopment();
|
|
|
|
+ $strategy = new ReplicationStrategy();
|
|
|
|
+
|
|
|
|
+ foreach ($this->getExpectedCommands('disallowed') as $commandId) {
|
|
|
|
+ $command = $profile->createCommand($commandId);
|
|
|
|
+ $this->assertTrue($strategy->isDisallowedOperation($command), $commandId);
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /**
|
|
|
|
+ * @group disconnected
|
|
|
|
+ */
|
|
|
|
+ public function testSortCommand()
|
|
|
|
+ {
|
|
|
|
+ $profile = ServerProfile::getDevelopment();
|
|
|
|
+ $strategy = new ReplicationStrategy();
|
|
|
|
+
|
|
|
|
+ $cmdReadSort = $profile->createCommand('SORT', array('key:list'));
|
|
|
|
+ $this->assertTrue($strategy->isReadOperation($cmdReadSort), 'SORT [read-only]');
|
|
|
|
+
|
|
|
|
+ $cmdWriteSort = $profile->createCommand('SORT', array('key:list', array('store' => 'key:stored')));
|
|
|
|
+ $this->assertFalse($strategy->isReadOperation($cmdWriteSort), 'SORT [write with STORE]');
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /**
|
|
|
|
+ * @group disconnected
|
|
|
|
+ * @expectedException Predis\NotSupportedException
|
|
|
|
+ * @expectedExceptionMessage The command INFO is not allowed in replication mode
|
|
|
|
+ */
|
|
|
|
+ public function testUsingDisallowedCommandThrowsException()
|
|
|
|
+ {
|
|
|
|
+ $profile = ServerProfile::getDevelopment();
|
|
|
|
+ $strategy = new ReplicationStrategy();
|
|
|
|
+
|
|
|
|
+ $command = $profile->createCommand('INFO');
|
|
|
|
+ $strategy->isReadOperation($command);
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /**
|
|
|
|
+ * @group disconnected
|
|
|
|
+ */
|
|
|
|
+ public function testDefaultIsWriteOperation()
|
|
|
|
+ {
|
|
|
|
+ $strategy = new ReplicationStrategy();
|
|
|
|
+
|
|
|
|
+ $command = $this->getMock('Predis\Command\CommandInterface');
|
|
|
|
+ $command->expects($this->any())
|
|
|
|
+ ->method('getId')
|
|
|
|
+ ->will($this->returnValue('CMDTEST'));
|
|
|
|
+
|
|
|
|
+ $this->assertFalse($strategy->isReadOperation($command));
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /**
|
|
|
|
+ * @group disconnected
|
|
|
|
+ */
|
|
|
|
+ public function testCanSetCommandAsReadOperation()
|
|
|
|
+ {
|
|
|
|
+ $strategy = new ReplicationStrategy();
|
|
|
|
+
|
|
|
|
+ $command = $this->getMock('Predis\Command\CommandInterface');
|
|
|
|
+ $command->expects($this->any())
|
|
|
|
+ ->method('getId')
|
|
|
|
+ ->will($this->returnValue('CMDTEST'));
|
|
|
|
+
|
|
|
|
+
|
|
|
|
+ $strategy->setCommandReadOnly('CMDTEST', true);
|
|
|
|
+ $this->assertTrue($strategy->isReadOperation($command));
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /**
|
|
|
|
+ * @group disconnected
|
|
|
|
+ */
|
|
|
|
+ public function testCanSetCommandAsWriteOperation()
|
|
|
|
+ {
|
|
|
|
+ $strategy = new ReplicationStrategy();
|
|
|
|
+
|
|
|
|
+ $command = $this->getMock('Predis\Command\CommandInterface');
|
|
|
|
+ $command->expects($this->any())
|
|
|
|
+ ->method('getId')
|
|
|
|
+ ->will($this->returnValue('CMDTEST'));
|
|
|
|
+
|
|
|
|
+
|
|
|
|
+ $strategy->setCommandReadOnly('CMDTEST', false);
|
|
|
|
+ $this->assertFalse($strategy->isReadOperation($command));
|
|
|
|
+
|
|
|
|
+ $strategy->setCommandReadOnly('GET', false);
|
|
|
|
+ $this->assertFalse($strategy->isReadOperation($command));
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /**
|
|
|
|
+ * @group disconnected
|
|
|
|
+ */
|
|
|
|
+ public function testCanUseCallableToCheckCommand()
|
|
|
|
+ {
|
|
|
|
+ $strategy = new ReplicationStrategy();
|
|
|
|
+ $profile = ServerProfile::getDevelopment();
|
|
|
|
+
|
|
|
|
+ $strategy->setCommandReadOnly('SET', function ($command) {
|
|
|
|
+ return $command->getArgument(1) === true;
|
|
|
|
+ });
|
|
|
|
+
|
|
|
|
+ $command = $profile->createCommand('SET', array('trigger', false));
|
|
|
|
+ $this->assertFalse($strategy->isReadOperation($command));
|
|
|
|
+
|
|
|
|
+ $command = $profile->createCommand('SET', array('trigger', true));
|
|
|
|
+ $this->assertTrue($strategy->isReadOperation($command));
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /**
|
|
|
|
+ * @group disconnected
|
|
|
|
+ */
|
|
|
|
+ public function testSetLuaScriptAsReadOperation()
|
|
|
|
+ {
|
|
|
|
+ $strategy = new ReplicationStrategy();
|
|
|
|
+ $profile = ServerProfile::getDevelopment();
|
|
|
|
+
|
|
|
|
+ $writeScript = 'redis.call("set", "foo", "bar")';
|
|
|
|
+ $readScript = 'return true';
|
|
|
|
+
|
|
|
|
+ $strategy->setScriptReadOnly($readScript, true);
|
|
|
|
+
|
|
|
|
+ $cmdEval = $profile->createCommand('EVAL', array($writeScript));
|
|
|
|
+ $cmdEvalSHA = $profile->createCommand('EVALSHA', array(sha1($writeScript)));
|
|
|
|
+ $this->assertFalse($strategy->isReadOperation($cmdEval));
|
|
|
|
+ $this->assertFalse($strategy->isReadOperation($cmdEvalSHA));
|
|
|
|
+
|
|
|
|
+ $cmdEval = $profile->createCommand('EVAL', array($readScript));
|
|
|
|
+ $cmdEvalSHA = $profile->createCommand('EVALSHA', array(sha1($readScript)));
|
|
|
|
+ $this->assertTrue($strategy->isReadOperation($cmdEval));
|
|
|
|
+ $this->assertTrue($strategy->isReadOperation($cmdEvalSHA));
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /**
|
|
|
|
+ * @group disconnected
|
|
|
|
+ */
|
|
|
|
+ public function testSetLuaScriptAsReadOperationWorksWithScriptedCommand()
|
|
|
|
+ {
|
|
|
|
+ $strategy = new ReplicationStrategy();
|
|
|
|
+
|
|
|
|
+ $command = $this->getMock('Predis\Command\ScriptedCommand', array('getScript'));
|
|
|
|
+ $command->expects($this->any())
|
|
|
|
+ ->method('getScript')
|
|
|
|
+ ->will($this->returnValue($script = 'return true'));
|
|
|
|
+
|
|
|
|
+ $strategy->setScriptReadOnly($script, function ($command) {
|
|
|
|
+ return $command->getArgument(2) === true;
|
|
|
|
+ });
|
|
|
|
+
|
|
|
|
+ $command->setArguments(array(false));
|
|
|
|
+ $this->assertFalse($strategy->isReadOperation($command));
|
|
|
|
+
|
|
|
|
+ $command->setArguments(array(true));
|
|
|
|
+ $this->assertTrue($strategy->isReadOperation($command));
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /**
|
|
|
|
+ * @group disconnected
|
|
|
|
+ */
|
|
|
|
+ public function testSetLuaScriptAsReadOperationWorksWithScriptedCommandAndCallableCheck()
|
|
|
|
+ {
|
|
|
|
+ $strategy = new ReplicationStrategy();
|
|
|
|
+
|
|
|
|
+ $command = $this->getMock('Predis\Command\ScriptedCommand', array('getScript'));
|
|
|
|
+ $command->expects($this->any())
|
|
|
|
+ ->method('getScript')
|
|
|
|
+ ->will($this->returnValue($script = 'return true'));
|
|
|
|
+
|
|
|
|
+ $command->setArguments(array('trigger', false));
|
|
|
|
+
|
|
|
|
+ $strategy->setScriptReadOnly($script, true);
|
|
|
|
+
|
|
|
|
+ $this->assertTrue($strategy->isReadOperation($command));
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ // ******************************************************************** //
|
|
|
|
+ // ---- HELPER METHODS ------------------------------------------------ //
|
|
|
|
+ // ******************************************************************** //
|
|
|
|
+
|
|
|
|
+ /**
|
|
|
|
+ * Returns the list of expected supported commands.
|
|
|
|
+ *
|
|
|
|
+ * @param string $type Optional type of command (based on its keys)
|
|
|
|
+ * @return array
|
|
|
|
+ */
|
|
|
|
+ protected function getExpectedCommands($type = null)
|
|
|
|
+ {
|
|
|
|
+ $commands = array(
|
|
|
|
+ /* commands operating on the connection */
|
|
|
|
+ 'EXISTS' => 'read',
|
|
|
|
+ 'AUTH' => 'read',
|
|
|
|
+ 'SELECT' => 'read',
|
|
|
|
+ 'ECHO' => 'read',
|
|
|
|
+ 'QUIT' => 'read',
|
|
|
|
+ 'OBJECT' => 'read',
|
|
|
|
+ 'BITCOUNT' => 'read',
|
|
|
|
+ 'TIME' => 'read',
|
|
|
|
+ 'SHUTDOWN' => 'disallowed',
|
|
|
|
+ 'INFO' => 'disallowed',
|
|
|
|
+ 'DBSIZE' => 'disallowed',
|
|
|
|
+ 'LASTSAVE' => 'disallowed',
|
|
|
|
+ 'CONFIG' => 'disallowed',
|
|
|
|
+ 'MONITOR' => 'disallowed',
|
|
|
|
+ 'SLAVEOF' => 'disallowed',
|
|
|
|
+ 'SAVE' => 'disallowed',
|
|
|
|
+ 'BGSAVE' => 'disallowed',
|
|
|
|
+ 'BGREWRITEAOF' => 'disallowed',
|
|
|
|
+ 'SLOWLOG' => 'disallowed',
|
|
|
|
+
|
|
|
|
+ /* commands operating on the key space */
|
|
|
|
+ 'EXISTS' => 'read',
|
|
|
|
+ 'DEL' => 'write',
|
|
|
|
+ 'TYPE' => 'read',
|
|
|
|
+ 'EXPIRE' => 'write',
|
|
|
|
+ 'EXPIREAT' => 'write',
|
|
|
|
+ 'PERSIST' => 'write',
|
|
|
|
+ 'PEXPIRE' => 'write',
|
|
|
|
+ 'PEXPIREAT' => 'write',
|
|
|
|
+ 'TTL' => 'read',
|
|
|
|
+ 'PTTL' => 'write',
|
|
|
|
+ 'SORT' => 'variable',
|
|
|
|
+ 'KEYS' => 'read',
|
|
|
|
+ 'RANDOMKEY' => 'read',
|
|
|
|
+
|
|
|
|
+ /* commands operating on string values */
|
|
|
|
+ 'APPEND' => 'write',
|
|
|
|
+ 'DECR' => 'write',
|
|
|
|
+ 'DECRBY' => 'write',
|
|
|
|
+ 'GET' => 'read',
|
|
|
|
+ 'GETBIT' => 'read',
|
|
|
|
+ 'MGET' => 'read',
|
|
|
|
+ 'SET' => 'write',
|
|
|
|
+ 'GETRANGE' => 'read',
|
|
|
|
+ 'GETSET' => 'write',
|
|
|
|
+ 'INCR' => 'write',
|
|
|
|
+ 'INCRBY' => 'write',
|
|
|
|
+ 'SETBIT' => 'write',
|
|
|
|
+ 'SETEX' => 'write',
|
|
|
|
+ 'MSET' => 'write',
|
|
|
|
+ 'MSETNX' => 'write',
|
|
|
|
+ 'SETNX' => 'write',
|
|
|
|
+ 'SETRANGE' => 'write',
|
|
|
|
+ 'STRLEN' => 'read',
|
|
|
|
+ 'SUBSTR' => 'read',
|
|
|
|
+
|
|
|
|
+ /* commands operating on lists */
|
|
|
|
+ 'LINSERT' => 'write',
|
|
|
|
+ 'LINDEX' => 'read',
|
|
|
|
+ 'LLEN' => 'read',
|
|
|
|
+ 'LPOP' => 'write',
|
|
|
|
+ 'RPOP' => 'write',
|
|
|
|
+ 'BLPOP' => 'write',
|
|
|
|
+ 'BRPOP' => 'write',
|
|
|
|
+ 'LPUSH' => 'write',
|
|
|
|
+ 'LPUSHX' => 'write',
|
|
|
|
+ 'RPUSH' => 'write',
|
|
|
|
+ 'RPUSHX' => 'write',
|
|
|
|
+ 'LRANGE' => 'read',
|
|
|
|
+ 'LREM' => 'write',
|
|
|
|
+ 'LSET' => 'write',
|
|
|
|
+ 'LTRIM' => 'write',
|
|
|
|
+
|
|
|
|
+ /* commands operating on sets */
|
|
|
|
+ 'SADD' => 'write',
|
|
|
|
+ 'SCARD' => 'read',
|
|
|
|
+ 'SISMEMBER' => 'read',
|
|
|
|
+ 'SMEMBERS' => 'read',
|
|
|
|
+ 'SPOP' => 'write',
|
|
|
|
+ 'SRANDMEMBER' => 'read',
|
|
|
|
+ 'SREM' => 'write',
|
|
|
|
+ 'SINTER' => 'read',
|
|
|
|
+ 'SUNION' => 'read',
|
|
|
|
+ 'SDIFF' => 'read',
|
|
|
|
+
|
|
|
|
+ /* commands operating on sorted sets */
|
|
|
|
+ 'ZADD' => 'write',
|
|
|
|
+ 'ZCARD' => 'read',
|
|
|
|
+ 'ZCOUNT' => 'read',
|
|
|
|
+ 'ZINCRBY' => 'write',
|
|
|
|
+ 'ZRANGE' => 'read',
|
|
|
|
+ 'ZRANGEBYSCORE' => 'read',
|
|
|
|
+ 'ZRANK' => 'read',
|
|
|
|
+ 'ZREM' => 'write',
|
|
|
|
+ 'ZREMRANGEBYRANK' => 'write',
|
|
|
|
+ 'ZREMRANGEBYSCORE' => 'write',
|
|
|
|
+ 'ZREVRANGE' => 'read',
|
|
|
|
+ 'ZREVRANGEBYSCORE' => 'read',
|
|
|
|
+ 'ZREVRANK' => 'read',
|
|
|
|
+ 'ZSCORE' => 'read',
|
|
|
|
+
|
|
|
|
+ /* commands operating on hashes */
|
|
|
|
+ 'HDEL' => 'write',
|
|
|
|
+ 'HEXISTS' => 'read',
|
|
|
|
+ 'HGET' => 'read',
|
|
|
|
+ 'HGETALL' => 'read',
|
|
|
|
+ 'HMGET' => 'read',
|
|
|
|
+ 'HINCRBY' => 'write',
|
|
|
|
+ 'HINCRBYFLOAT' => 'write',
|
|
|
|
+ 'HKEYS' => 'read',
|
|
|
|
+ 'HLEN' => 'read',
|
|
|
|
+ 'HSET' => 'write',
|
|
|
|
+ 'HSETNX' => 'write',
|
|
|
|
+ 'HVALS' => 'read',
|
|
|
|
+
|
|
|
|
+ /* scripting */
|
|
|
|
+ 'EVAL' => 'write',
|
|
|
|
+ 'EVALSHA' => 'write',
|
|
|
|
+ );
|
|
|
|
+
|
|
|
|
+ if (isset($type)) {
|
|
|
|
+ $commands = array_filter($commands, function ($expectedType) use ($type) {
|
|
|
|
+ return $expectedType === $type;
|
|
|
|
+ });
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ return array_keys($commands);
|
|
|
|
+ }
|
|
|
|
+}
|