123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580 |
- <?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\Network;
- use \PHPUnit_Framework_TestCase as StandardTestCase;
- use Predis\ConnectionParameters;
- use Predis\Profiles\ServerProfile;
- /**
- *
- */
- class MasterSlaveReplicationTest extends StandardTestCase
- {
- /**
- * @group disconnected
- */
- public function testAddingConnectionsToReplication()
- {
- $master = $this->getMockConnection('tcp://host1?alias=master');
- $slave1 = $this->getMockConnection('tcp://host2?alias=slave1');
- $slave2 = $this->getMockConnection('tcp://host3?alias=slave2');
- $replication = new MasterSlaveReplication();
- $replication->add($master);
- $replication->add($slave1);
- $replication->add($slave2);
- $this->assertSame($master, $replication->getConnectionById('master'));
- $this->assertSame($slave1, $replication->getConnectionById('slave1'));
- $this->assertSame($slave2, $replication->getConnectionById('slave2'));
- $this->assertSame($master, $replication->getMaster());
- $this->assertSame(array($slave1, $slave2), $replication->getSlaves());
- }
- /**
- * @group disconnected
- */
- public function testRemovingConnectionsFromReplication()
- {
- $master = $this->getMockConnection('tcp://host1?alias=master');
- $slave1 = $this->getMockConnection('tcp://host2?alias=slave1');
- $slave2 = $this->getMockConnection('tcp://host3?alias=slave2');
- $replication = new MasterSlaveReplication();
- $replication->add($master);
- $replication->add($slave1);
- $this->assertTrue($replication->remove($slave1));
- $this->assertFalse($replication->remove($slave2));
- $this->assertSame($master, $replication->getMaster());
- $this->assertSame(array(), $replication->getSlaves());
- }
- /**
- * @group disconnected
- * @expectedException RuntimeException
- * @expectedExceptionMessage Replication needs a master and at least one slave
- */
- public function testThrowsExceptionOnEmptyReplication()
- {
- $replication = new MasterSlaveReplication();
- $replication->connect();
- }
- /**
- * @group disconnected
- * @expectedException RuntimeException
- * @expectedExceptionMessage Replication needs a master and at least one slave
- */
- public function testThrowsExceptionOnMissingMaster()
- {
- $replication = new MasterSlaveReplication();
- $replication->add($this->getMockConnection('tcp://host2?alias=slave1'));
- $replication->connect();
- }
- /**
- * @group disconnected
- * @expectedException RuntimeException
- * @expectedExceptionMessage Replication needs a master and at least one slave
- */
- public function testThrowsExceptionOnMissingSlave()
- {
- $replication = new MasterSlaveReplication();
- $replication->add($this->getMockConnection('tcp://host1?alias=master'));
- $replication->connect();
- }
- /**
- * @group disconnected
- */
- public function testConnectForcesConnectionToOneOfSlaves()
- {
- $master = $this->getMockConnection('tcp://host1?alias=master');
- $master->expects($this->never())->method('connect');
- $slave = $this->getMockConnection('tcp://host2?alias=slave1');
- $slave->expects($this->once())->method('connect');
- $replication = new MasterSlaveReplication();
- $replication->add($master);
- $replication->add($slave);
- $replication->connect();
- }
- /**
- * @group disconnected
- */
- public function testIsConnectedReturnsTrueIfAtLeastOneConnectionIsOpen()
- {
- $master = $this->getMockConnection('tcp://host1?alias=master');
- $master->expects($this->never())->method('isConnected')->will($this->returnValue(false));
- $slave = $this->getMockConnection('tcp://host2?alias=slave1');
- $slave->expects($this->once())->method('isConnected')->will($this->returnValue(true));
- $replication = new MasterSlaveReplication();
- $replication->add($master);
- $replication->add($slave);
- $replication->connect();
- $this->assertTrue($replication->isConnected());
- }
- /**
- * @group disconnected
- */
- public function testIsConnectedReturnsFalseIfAllConnectionsAreClosed()
- {
- $master = $this->getMockConnection('tcp://host1?alias=master');
- $master->expects($this->any())->method('isConnected')->will($this->returnValue(false));
- $slave = $this->getMockConnection('tcp://host2?alias=slave1');
- $slave->expects($this->any())->method('isConnected')->will($this->returnValue(false));
- $replication = new MasterSlaveReplication();
- $replication->add($master);
- $replication->add($slave);
- $this->assertFalse($replication->isConnected());
- $replication->connect();
- $replication->disconnect();
- $this->assertFalse($replication->isConnected());
- }
- /**
- * @group disconnected
- */
- public function testDisconnectForcesCurrentConnectionToDisconnect()
- {
- $master = $this->getMockConnection('tcp://host1?alias=master');
- $master->expects($this->once())->method('disconnect');
- $slave = $this->getMockConnection('tcp://host2?alias=slave1');
- $slave->expects($this->once())->method('disconnect');
- $replication = new MasterSlaveReplication();
- $replication->add($master);
- $replication->add($slave);
- $replication->disconnect();
- }
- /**
- * @group disconnected
- */
- public function testCanSwitchConnectionByAlias()
- {
- $master = $this->getMockConnection('tcp://host1?alias=master');
- $slave1 = $this->getMockConnection('tcp://host2?alias=slave1');
- $replication = new MasterSlaveReplication();
- $replication->add($master);
- $replication->add($slave1);
- $this->assertNull($replication->getCurrent());
- $replication->switchTo('master');
- $this->assertSame($master, $replication->getCurrent());
- $replication->switchTo('slave1');
- $this->assertSame($slave1, $replication->getCurrent());
- }
- /**
- * @group disconnected
- * @expectedException InvalidArgumentException
- * @expectedExceptionMessage The specified connection is not valid
- */
- public function testThrowsErrorWhenSwitchingToUnknownConnection()
- {
- $replication = new MasterSlaveReplication();
- $replication->add($this->getMockConnection('tcp://host1?alias=master'));
- $replication->add($this->getMockConnection('tcp://host2?alias=slave1'));
- $replication->switchTo('unknown');
- }
- /**
- * @group disconnected
- */
- public function testUsesSlavesOnReadOnlyCommands()
- {
- $profile = ServerProfile::getDefault();
- $master = $this->getMockConnection('tcp://host1?alias=master');
- $slave1 = $this->getMockConnection('tcp://host2?alias=slave1');
- $replication = new MasterSlaveReplication();
- $replication->add($master);
- $replication->add($slave1);
- $cmd = $profile->createCommand('exists', array('foo'));
- $this->assertSame($slave1, $replication->getConnection($cmd));
- $cmd = $profile->createCommand('get', array('foo'));
- $this->assertSame($slave1, $replication->getConnection($cmd));
- }
- /**
- * @group disconnected
- */
- public function testUsesMasterOnWriteCommands()
- {
- $profile = ServerProfile::getDefault();
- $master = $this->getMockConnection('tcp://host1?alias=master');
- $slave1 = $this->getMockConnection('tcp://host2?alias=slave1');
- $replication = new MasterSlaveReplication();
- $replication->add($master);
- $replication->add($slave1);
- $cmd = $profile->createCommand('set', array('foo', 'bar'));
- $this->assertSame($master, $replication->getConnection($cmd));
- $cmd = $profile->createCommand('get', array('foo'));
- $this->assertSame($master, $replication->getConnection($cmd));
- }
- /**
- * @group disconnected
- */
- public function testSwitchesFromSlaveToMasterOnWriteCommands()
- {
- $profile = ServerProfile::getDefault();
- $master = $this->getMockConnection('tcp://host1?alias=master');
- $slave1 = $this->getMockConnection('tcp://host2?alias=slave1');
- $replication = new MasterSlaveReplication();
- $replication->add($master);
- $replication->add($slave1);
- $cmd = $profile->createCommand('exists', array('foo'));
- $this->assertSame($slave1, $replication->getConnection($cmd));
- $cmd = $profile->createCommand('set', array('foo', 'bar'));
- $this->assertSame($master, $replication->getConnection($cmd));
- $cmd = $profile->createCommand('exists', array('foo'));
- $this->assertSame($master, $replication->getConnection($cmd));
- }
- /**
- * @group disconnected
- */
- public function testWritesCommandToCorrectConnection()
- {
- $profile = ServerProfile::getDefault();
- $cmdExists = $profile->createCommand('exists', array('foo'));
- $cmdSet = $profile->getDefault()->createCommand('set', array('foo', 'bar'));
- $master = $this->getMockConnection('tcp://host1?alias=master');
- $master->expects($this->once())->method('writeCommand')->with($cmdSet);
- $slave1 = $this->getMockConnection('tcp://host2?alias=slave1');
- $slave1->expects($this->once())->method('writeCommand')->with($cmdExists);
- $replication = new MasterSlaveReplication();
- $replication->add($master);
- $replication->add($slave1);
- $replication->writeCommand($cmdExists);
- $replication->writeCommand($cmdSet);
- }
- /**
- * @group disconnected
- */
- public function testReadsCommandFromCorrectConnection()
- {
- $profile = ServerProfile::getDefault();
- $cmdExists = $profile->createCommand('exists', array('foo'));
- $cmdSet = $profile->getDefault()->createCommand('set', array('foo', 'bar'));
- $master = $this->getMockConnection('tcp://host1?alias=master');
- $master->expects($this->once())->method('readResponse')->with($cmdSet);
- $slave1 = $this->getMockConnection('tcp://host2?alias=slave1');
- $slave1->expects($this->once())->method('readResponse')->with($cmdExists);
- $replication = new MasterSlaveReplication();
- $replication->add($master);
- $replication->add($slave1);
- $replication->readResponse($cmdExists);
- $replication->readResponse($cmdSet);
- }
- /**
- * @group disconnected
- */
- public function testExecutesCommandOnCorrectConnection()
- {
- $profile = ServerProfile::getDefault();
- $cmdExists = $profile->createCommand('exists', array('foo'));
- $cmdSet = $profile->getDefault()->createCommand('set', array('foo', 'bar'));
- $master = $this->getMockConnection('tcp://host1?alias=master');
- $master->expects($this->once())->method('executeCommand')->with($cmdSet);
- $slave1 = $this->getMockConnection('tcp://host2?alias=slave1');
- $slave1->expects($this->once())->method('executeCommand')->with($cmdExists);
- $replication = new MasterSlaveReplication();
- $replication->add($master);
- $replication->add($slave1);
- $replication->executeCommand($cmdExists);
- $replication->executeCommand($cmdSet);
- }
- /**
- * @group disconnected
- */
- public function testWatchTriggersSwitchToMasterConnection()
- {
- $profile = ServerProfile::getDefault();
- $cmdWatch = $profile->createCommand('watch', array('foo'));
- $master = $this->getMockConnection('tcp://host1?alias=master');
- $master->expects($this->once())->method('executeCommand')->with($cmdWatch);
- $slave1 = $this->getMockConnection('tcp://host2?alias=slave1');
- $slave1->expects($this->never())->method('executeCommand');
- $replication = new MasterSlaveReplication();
- $replication->add($master);
- $replication->add($slave1);
- $replication->executeCommand($cmdWatch);
- }
- /**
- * @group disconnected
- */
- public function testMultiTriggersSwitchToMasterConnection()
- {
- $profile = ServerProfile::getDefault();
- $cmdMulti = $profile->createCommand('multi');
- $master = $this->getMockConnection('tcp://host1?alias=master');
- $master->expects($this->once())->method('executeCommand')->with($cmdMulti);
- $slave1 = $this->getMockConnection('tcp://host2?alias=slave1');
- $slave1->expects($this->never())->method('executeCommand');
- $replication = new MasterSlaveReplication();
- $replication->add($master);
- $replication->add($slave1);
- $replication->executeCommand($cmdMulti);
- }
- /**
- * @group disconnected
- */
- public function testEvalTriggersSwitchToMasterConnection()
- {
- $profile = ServerProfile::get('dev');
- $cmdEval = $profile->createCommand('eval', array("return redis.call('info')"));
- $master = $this->getMockConnection('tcp://host1?alias=master');
- $master->expects($this->once())->method('executeCommand')->with($cmdEval);
- $slave1 = $this->getMockConnection('tcp://host2?alias=slave1');
- $slave1->expects($this->never())->method('executeCommand');
- $replication = new MasterSlaveReplication();
- $replication->add($master);
- $replication->add($slave1);
- $replication->executeCommand($cmdEval);
- }
- /**
- * @group disconnected
- */
- public function testSortTriggersSwitchToMasterConnectionOnStoreModifier()
- {
- $profile = ServerProfile::get('dev');
- $cmdSortNormal = $profile->createCommand('sort', array('key'));
- $cmdSortStore = $profile->createCommand('sort', array('key', array('store' => 'key:store')));
- $master = $this->getMockConnection('tcp://host1?alias=master');
- $master->expects($this->once())->method('executeCommand')->with($cmdSortStore);
- $slave1 = $this->getMockConnection('tcp://host2?alias=slave1');
- $slave1->expects($this->once())->method('executeCommand')->with($cmdSortNormal);
- $replication = new MasterSlaveReplication();
- $replication->add($master);
- $replication->add($slave1);
- $replication->executeCommand($cmdSortNormal);
- $replication->executeCommand($cmdSortStore);
- }
- /**
- * @group disconnected
- * @expectedException Predis\NotSupportedException
- * @expectedExceptionMessage The command INFO is not allowed in replication mode
- */
- public function testThrowsExceptionOnNonSupportedCommand()
- {
- $cmd = ServerProfile::getDefault()->createCommand('info');
- $replication = new MasterSlaveReplication();
- $replication->add($this->getMockConnection('tcp://host1?alias=master'));
- $replication->add($this->getMockConnection('tcp://host2?alias=slave1'));
- $replication->getConnection($cmd);
- }
- /**
- * @group disconnected
- */
- public function testCanOverrideReadOnlyFlagForCommands()
- {
- $profile = ServerProfile::getDefault();
- $cmdSet = $profile->createCommand('set', array('foo', 'bar'));
- $cmdGet = $profile->createCommand('get', array('foo'));
- $master = $this->getMockConnection('tcp://host1?alias=master');
- $master->expects($this->once())->method('executeCommand')->with($cmdGet);
- $slave1 = $this->getMockConnection('tcp://host2?alias=slave1');
- $slave1->expects($this->once())->method('executeCommand')->with($cmdSet);
- $replication = new MasterSlaveReplication();
- $replication->add($master);
- $replication->add($slave1);
- $replication->setCommandReadOnly($cmdSet->getId(), true);
- $replication->setCommandReadOnly($cmdGet->getId(), false);
- $replication->executeCommand($cmdSet);
- $replication->executeCommand($cmdGet);
- }
- /**
- * @group disconnected
- */
- public function testAcceptsCallableToOverrideReadOnlyFlagForCommands()
- {
- $profile = ServerProfile::getDefault();
- $cmdExistsFoo = $profile->createCommand('exists', array('foo'));
- $cmdExistsBar = $profile->createCommand('exists', array('bar'));
- $master = $this->getMockConnection('tcp://host1?alias=master');
- $master->expects($this->once())->method('executeCommand')->with($cmdExistsBar);
- $slave1 = $this->getMockConnection('tcp://host2?alias=slave1');
- $slave1->expects($this->once())->method('executeCommand')->with($cmdExistsFoo);
- $replication = new MasterSlaveReplication();
- $replication->add($master);
- $replication->add($slave1);
- $replication->setCommandReadOnly('exists', function($cmd) {
- list($arg1) = $cmd->getArguments();
- return $arg1 === 'foo';
- });
- $replication->executeCommand($cmdExistsFoo);
- $replication->executeCommand($cmdExistsBar);
- }
- /**
- * @group disconnected
- */
- public function testCanSetReadOnlyFlagForEvalScripts()
- {
- $profile = ServerProfile::get('dev');
- $cmdEval = $profile->createCommand('eval', array($script = "return redis.call('info');"));
- $cmdEvalSha = $profile->createCommand('evalsha', array($scriptSHA1 = sha1($script)));
- $master = $this->getMockConnection('tcp://host1?alias=master');
- $master->expects($this->never())->method('executeCommand');
- $slave1 = $this->getMockConnection('tcp://host2?alias=slave1');
- $slave1->expects($this->exactly(2))
- ->method('executeCommand')
- ->with($this->logicalOr($cmdEval, $cmdEvalSha));
- $replication = new MasterSlaveReplication();
- $replication->add($master);
- $replication->add($slave1);
- $replication->setScriptReadOnly($script);
- $replication->executeCommand($cmdEval);
- $replication->executeCommand($cmdEvalSha);
- }
- /**
- * @group disconnected
- */
- public function testCanBeSerialized()
- {
- $master = $this->getMockConnection('tcp://host1?alias=master');
- $slave1 = $this->getMockConnection('tcp://host2?alias=slave1');
- $replication = new MasterSlaveReplication();
- $replication->add($master);
- $replication->add($slave1);
- $unserialized = unserialize(serialize($replication));
- $this->assertEquals($master, $unserialized->getConnectionById('master'));
- $this->assertEquals($slave1, $unserialized->getConnectionById('slave1'));
- }
- // ******************************************************************** //
- // ---- HELPER METHODS ------------------------------------------------ //
- // ******************************************************************** //
- /**
- * Returns a base mocked connection from Predis\Network\IConnectionSingle.
- *
- * @param mixed $parameters Optional parameters.
- * @return mixed
- */
- protected function getMockConnection($parameters = null)
- {
- $connection = $this->getMock('Predis\Network\IConnectionSingle');
- if ($parameters) {
- $parameters = new ConnectionParameters($parameters);
- $hash = "{$parameters->host}:{$parameters->port}";
- $connection->expects($this->any())
- ->method('getParameters')
- ->will($this->returnValue($parameters));
- $connection->expects($this->any())
- ->method('__toString')
- ->will($this->returnValue($hash));
- }
- return $connection;
- }
- }
|