123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726 |
- <?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\Transaction;
- use \PHPUnit_Framework_TestCase as StandardTestCase;
- use Predis\Client;
- use Predis\ResponseQueued;
- use Predis\ServerException;
- use Predis\Commands\ICommand;
- /**
- * @group realm-transaction
- */
- class MultiExecContextTest extends StandardTestCase
- {
- /**
- * @group disconnected
- * @expectedException Predis\NotSupportedException
- * @expectedExceptionMessage The current profile does not support MULTI, EXEC and DISCARD
- */
- public function testThrowsExceptionOnUnsupportedMultiExecInProfile()
- {
- $connection = $this->getMock('Predis\Network\IConnectionSingle');
- $client = new Client($connection, '1.2');
- $tx = new MultiExecContext($client);
- }
- /**
- * @group disconnected
- * @expectedException Predis\NotSupportedException
- * @expectedExceptionMessage The current profile does not support WATCH and UNWATCH
- */
- public function testThrowsExceptionOnUnsupportedWatchUnwatchInProfile()
- {
- $connection = $this->getMock('Predis\Network\IConnectionSingle');
- $client = new Client($connection, '2.0');
- $tx = new MultiExecContext($client, array('options' => 'cas'));
- $tx->watch('foo');
- }
- /**
- * @group disconnected
- */
- public function testExecutionWithFluentInterface()
- {
- $commands = array();
- $expected = array('one', 'two', 'three');
- $callback = $this->getExecuteCallback($expected, $commands);
- $tx = $this->getMockedTransaction($callback);
- $this->assertSame($expected, $tx->echo('one')->echo('two')->echo('three')->execute());
- $this->assertSame(array('MULTI', 'ECHO', 'ECHO', 'ECHO', 'EXEC'), self::commandsToIDs($commands));
- }
- /**
- * @group disconnected
- */
- public function testExecutionWithCallable()
- {
- $commands = array();
- $expected = array('one', 'two', 'three');
- $callback = $this->getExecuteCallback($expected, $commands);
- $tx = $this->getMockedTransaction($callback);
- $replies = $tx->execute(function($tx) {
- $tx->echo('one');
- $tx->echo('two');
- $tx->echo('three');
- });
- $this->assertSame($expected, $replies);
- $this->assertSame(array('MULTI', 'ECHO', 'ECHO', 'ECHO', 'EXEC'), self::commandsToIDs($commands));
- }
- /**
- * @group disconnected
- */
- public function testCannotMixExecutionWithFluentInterfaceAndCallable()
- {
- $commands = array();
- $callback = $this->getExecuteCallback(null, $commands);
- $tx = $this->getMockedTransaction($callback);
- $exception = null;
- try {
- $tx->echo('foo')->execute(function($tx) { $tx->echo('bar'); });
- }
- catch (\Exception $ex) {
- $exception = $ex;
- }
- $this->assertInstanceOf('Predis\ClientException', $exception);
- $this->assertSame(array('MULTI', 'ECHO', 'DISCARD'), self::commandsToIDs($commands));
- }
- /**
- * @group disconnected
- */
- public function testEmptyTransactionDoesNotSendMultiExecCommands()
- {
- $commands = array();
- $callback = $this->getExecuteCallback(null, $commands);
- $tx = $this->getMockedTransaction($callback);
- $replies = $tx->execute(function($tx) {
- // NOOP
- });
- $this->assertNull($replies);
- $this->assertSame(array(), self::commandsToIDs($commands));
- }
- /**
- * @group disconnected
- * @expectedException Predis\ClientException
- * @expectedExceptionMessage Cannot invoke 'execute' or 'exec' inside an active client transaction block
- */
- public function testThrowsExceptionOnExecInsideTransactionBlock()
- {
- $commands = array();
- $callback = $this->getExecuteCallback(null, $commands);
- $tx = $this->getMockedTransaction($callback);
- $replies = $tx->execute(function($tx) {
- $tx->exec();
- });
- $this->assertNull($replies);
- $this->assertSame(array(), self::commandsToIDs($commands));
- }
- /**
- * @group disconnected
- */
- public function testEmptyTransactionIgnoresDiscard()
- {
- $commands = array();
- $callback = $this->getExecuteCallback(null, $commands);
- $tx = $this->getMockedTransaction($callback);
- $replies = $tx->execute(function($tx) {
- $tx->discard();
- });
- $this->assertNull($replies);
- $this->assertSame(array(), self::commandsToIDs($commands));
- }
- /**
- * @group disconnected
- */
- public function testTransactionWithCommandsSendsDiscard()
- {
- $commands = array();
- $callback = $this->getExecuteCallback(null, $commands);
- $tx = $this->getMockedTransaction($callback);
- $replies = $tx->execute(function($tx) {
- $tx->set('foo', 'bar');
- $tx->get('foo');
- $tx->discard();
- });
- $this->assertNull($replies);
- $this->assertSame(array('MULTI', 'SET', 'GET', 'DISCARD'), self::commandsToIDs($commands));
- }
- /**
- * @group disconnected
- */
- public function testSendMultiOnCommandsFollowingDiscard()
- {
- $commands = array();
- $expected = array('after DISCARD');
- $callback = $this->getExecuteCallback($expected, $commands);
- $tx = $this->getMockedTransaction($callback);
- $replies = $tx->execute(function($tx) {
- $tx->echo('before DISCARD');
- $tx->discard();
- $tx->echo('after DISCARD');
- });
- $this->assertSame($replies, $expected);
- $this->assertSame(array('MULTI', 'ECHO', 'DISCARD', 'MULTI', 'ECHO', 'EXEC'), self::commandsToIDs($commands));
- }
- /**
- * @group disconnected
- * @expectedException Predis\ClientException
- */
- public function testThrowsExceptionOnWatchInsideMulti()
- {
- $callback = $this->getExecuteCallback();
- $tx = $this->getMockedTransaction($callback);
- $tx->echo('foobar')->watch('foo')->execute();
- }
- /**
- * @group disconnected
- */
- public function testUnwatchInsideMulti()
- {
- $commands = array();
- $expected = array('foobar', true);
- $callback = $this->getExecuteCallback($expected, $commands);
- $tx = $this->getMockedTransaction($callback);
- $replies = $tx->echo('foobar')->unwatch('foo')->execute();
- $this->assertSame($replies, $expected);
- $this->assertSame(array('MULTI', 'ECHO', 'UNWATCH', 'EXEC'), self::commandsToIDs($commands));
- }
- /**
- * @group disconnected
- */
- public function testAutomaticWatchInOptions()
- {
- $txCommands = $casCommands = array();
- $expected = array('bar', 'piyo');
- $options = array('watch' => array('foo', 'hoge'));
- $callback = $this->getExecuteCallback($expected, $txCommands, $casCommands);
- $tx = $this->getMockedTransaction($callback, $options);
- $replies = $tx->execute(function($tx) {
- $tx->get('foo');
- $tx->get('hoge');
- });
- $this->assertSame($replies, $expected);
- $this->assertSame(array('WATCH'), self::commandsToIDs($casCommands));
- $this->assertSame(array('foo', 'hoge'), $casCommands[0]->getArguments());
- $this->assertSame(array('MULTI', 'GET', 'GET', 'EXEC'), self::commandsToIDs($txCommands));
- }
- /**
- * @group disconnected
- */
- public function testCheckAndSetWithFluentInterface()
- {
- $txCommands = $casCommands = array();
- $expected = array('bar', 'piyo');
- $options = array('cas' => true, 'watch' => array('foo', 'hoge'));
- $callback = $this->getExecuteCallback($expected, $txCommands, $casCommands);
- $tx = $this->getMockedTransaction($callback, $options);
- $tx->watch('foobar');
- $this->assertSame('DUMMY_REPLY', $tx->get('foo'));
- $this->assertSame('DUMMY_REPLY', $tx->get('hoge'));
- $replies = $tx->multi()
- ->get('foo')
- ->get('hoge')
- ->execute();
- $this->assertSame($replies, $expected);
- $this->assertSame(array('WATCH', 'WATCH', 'GET', 'GET'), self::commandsToIDs($casCommands));
- $this->assertSame(array('MULTI', 'GET', 'GET', 'EXEC'), self::commandsToIDs($txCommands));
- }
- /**
- * @group disconnected
- */
- public function testCheckAndSetWithBlock()
- {
- $txCommands = $casCommands = array();
- $expected = array('bar', 'piyo');
- $options = array('cas' => true, 'watch' => array('foo', 'hoge'));
- $callback = $this->getExecuteCallback($expected, $txCommands, $casCommands);
- $tx = $this->getMockedTransaction($callback, $options);
- $test = $this;
- $replies = $tx->execute(function($tx) use($test) {
- $tx->watch('foobar');
- $reply1 = $tx->get('foo');
- $reply2 = $tx->get('hoge');
- $test->assertSame('DUMMY_REPLY', $reply1);
- $test->assertSame('DUMMY_REPLY', $reply2);
- $tx->multi();
- $tx->get('foo');
- $tx->get('hoge');
- });
- $this->assertSame($replies, $expected);
- $this->assertSame(array('WATCH', 'WATCH', 'GET', 'GET'), self::commandsToIDs($casCommands));
- $this->assertSame(array('MULTI', 'GET', 'GET', 'EXEC'), self::commandsToIDs($txCommands));
- }
- /**
- * @group disconnected
- */
- public function testCheckAndSetWithEmptyBlock()
- {
- $txCommands = $casCommands = array();
- $options = array('cas' => true);
- $callback = $this->getExecuteCallback(array(), $txCommands, $casCommands);
- $tx = $this->getMockedTransaction($callback, $options);
- $tx->execute(function($tx) {
- $tx->multi();
- });
- $this->assertSame(array(), self::commandsToIDs($casCommands));
- $this->assertSame(array(), self::commandsToIDs($txCommands));
- }
- /**
- * @group disconnected
- */
- public function testCheckAndSetWithoutExec()
- {
- $txCommands = $casCommands = array();
- $options = array('cas' => true);
- $callback = $this->getExecuteCallback(array(), $txCommands, $casCommands);
- $tx = $this->getMockedTransaction($callback, $options);
- $tx->execute(function($tx) {
- $bar = $tx->get('foo');
- $tx->set('hoge', 'piyo');
- });
- $this->assertSame(array('GET', 'SET'), self::commandsToIDs($casCommands));
- $this->assertSame(array(), self::commandsToIDs($txCommands));
- }
- /**
- * @group disconnected
- * @expectedException InvalidArgumentException
- * @expectedExceptionMessage Automatic retries can be used only when a transaction block is provided
- */
- public function testThrowsExceptionOnAutomaticRetriesWithFluentInterface()
- {
- $options = array('retry' => 1);
- $callback = $this->getExecuteCallback();
- $tx = $this->getMockedTransaction($callback, $options);
- $tx->echo('message')->execute();
- }
- /**
- * @group disconnected
- */
- public function testAutomaticRetryOnServerSideTransactionAbort()
- {
- $casCommands = $txCommands = array();
- $expected = array('bar');
- $options = array('watch' => array('foo', 'bar'), 'retry' => ($attempts = 2) + 1);
- $sentinel = $this->getMock('stdClass', array('signal'));
- $sentinel->expects($this->exactly($attempts))->method('signal');
- $callback = $this->getExecuteCallback($expected, $txCommands, $casCommands);
- $tx = $this->getMockedTransaction($callback, $options);
- $replies = $tx->execute(function($tx) use($sentinel, &$attempts) {
- $tx->get('foo');
- if ($attempts > 0) {
- $attempts -= 1;
- $sentinel->signal();
- $tx->echo('!!ABORT!!');
- }
- });
- $this->assertSame($replies, $expected);
- $this->assertSame(array('WATCH'), self::commandsToIDs($casCommands));
- $this->assertSame(array('foo', 'bar'), $casCommands[0]->getArguments());
- $this->assertSame(array('MULTI', 'GET', 'EXEC'), self::commandsToIDs($txCommands));
- }
- /**
- * @group disconnected
- * @expectedException Predis\Transaction\AbortedMultiExecException
- */
- public function testThrowsExceptionOnServerSideTransactionAbort()
- {
- $callback = $this->getExecuteCallback();
- $tx = $this->getMockedTransaction($callback);
- $replies = $tx->execute(function($tx) {
- $tx->echo('!!ABORT!!');
- });
- }
- /**
- * @group disconnected
- */
- public function testHandlesStandardExceptionsInBlock()
- {
- $commands = array();
- $expected = array('foobar', true);
- $callback = $this->getExecuteCallback($expected, $commands);
- $tx = $this->getMockedTransaction($callback);
- $replies = null;
- try {
- $replies = $tx->execute(function($tx) {
- $tx->set('foo', 'bar');
- $tx->get('foo');
- throw new \RuntimeException('TEST');
- });
- }
- catch (\Exception $ex) {
- // NOOP
- }
- $this->assertNull($replies, $expected);
- $this->assertSame(array('MULTI', 'SET', 'GET', 'DISCARD'), self::commandsToIDs($commands));
- }
- /**
- * @group disconnected
- */
- public function testHandlesServerExceptionsInBlock()
- {
- $commands = array();
- $expected = array('foobar', true);
- $callback = $this->getExecuteCallback($expected, $commands);
- $tx = $this->getMockedTransaction($callback);
- $replies = null;
- try {
- $replies = $tx->execute(function($tx) {
- $tx->set('foo', 'bar');
- $tx->echo('ERR Invalid operation');
- $tx->get('foo');
- });
- }
- catch (ServerException $ex) {
- $tx->discard();
- }
- $this->assertNull($replies);
- $this->assertSame(array('MULTI', 'SET', 'ECHO', 'DISCARD'), self::commandsToIDs($commands));
- }
- // ******************************************************************** //
- // ---- INTEGRATION TESTS --------------------------------------------- //
- // ******************************************************************** //
- /**
- * @group connected
- */
- public function testIntegrationHandlesStandardExceptionsInBlock()
- {
- $client = $this->getClient();
- $exception = null;
- try {
- $client->multiExec(function($tx) {
- $tx->set('foo', 'bar');
- throw new \RuntimeException("TEST");
- });
- }
- catch (\Exception $ex) {
- $exception = $ex;
- }
- $this->assertInstanceOf('RuntimeException', $exception);
- $this->assertFalse($client->exists('foo'));
- }
- /**
- * @group connected
- */
- public function testIntegrationSendMultiOnCommandsAfterDiscard()
- {
- $client = $this->getClient();
- $replies = $client->multiExec(function($tx) {
- $tx->set('foo', 'bar');
- $tx->discard();
- $tx->set('hoge', 'piyo');
- });
- $this->assertSame(1, count($replies));
- $this->assertFalse($client->exists('foo'));
- $this->assertTrue($client->exists('hoge'));
- }
- /**
- * @group connected
- */
- public function testIntegrationWritesOnWatchedKeysAbortTransaction()
- {
- $exception = null;
- $client1 = $this->getClient();
- $client2 = $this->getClient();
- try {
- $client1->multiExec(array('watch' => 'sentinel'), function($tx) use($client2) {
- $tx->set('sentinel', 'client1');
- $tx->get('sentinel');
- $client2->set('sentinel', 'client2');
- });
- }
- catch (AbortedMultiExecException $ex) {
- $exception = $ex;
- }
- $this->assertInstanceOf('Predis\Transaction\AbortedMultiExecException', $exception);
- $this->assertSame('client2', $client1->get('sentinel'));
- }
- /**
- * @group connected
- */
- public function testIntegrationCheckAndSetWithDiscardAndRetry()
- {
- $client = $this->getClient();
- $client->set('foo', 'bar');
- $options = array('watch' => 'foo', 'cas' => true);
- $replies = $client->multiExec($options, function($tx) {
- $tx->watch('foobar');
- $foo = $tx->get('foo');
- $tx->multi();
- $tx->set('foobar', $foo);
- $tx->discard();
- $tx->mget('foo', 'foobar');
- });
- $this->assertInternalType('array', $replies);
- $this->assertSame(array(array('bar', null)), $replies);
- $hijack = true;
- $client2 = $this->getClient();
- $client->set('foo', 'bar');
- $options = array('watch' => 'foo', 'cas' => true, 'retry' => 1);
- $replies = $client->multiExec($options, function($tx) use($client2, &$hijack) {
- $foo = $tx->get('foo');
- $tx->multi();
- $tx->set('foobar', $foo);
- $tx->discard();
- if ($hijack) {
- $hijack = false;
- $client2->set('foo', 'hijacked!');
- }
- $tx->mget('foo', 'foobar');
- });
- $this->assertInternalType('array', $replies);
- $this->assertSame(array(array('hijacked!', null)), $replies);
- }
- // ******************************************************************** //
- // ---- HELPER METHODS ------------------------------------------------ //
- // ******************************************************************** //
- /**
- * Returns a mocked instance of Predis\Network\IConnectionSingle using
- * the specified callback to return values from the executeCommand method.
- *
- * @param \Closure $executeCallback
- * @return \Predis\Network\IConnectionSingle
- */
- protected function getMockedConnection($executeCallback)
- {
- $connection = $this->getMock('Predis\Network\IConnectionSingle');
- $connection->expects($this->any())
- ->method('executeCommand')
- ->will($this->returnCallback($executeCallback));
- return $connection;
- }
- /**
- * Returns a mocked instance of Predis\Transaction\MultiExecContext using
- * the specified callback to return values from the executeCommand method
- * of the underlying connection.
- *
- * @param \Closure $executeCallback
- * @return MultiExecContext
- */
- protected function getMockedTransaction($executeCallback, $options = array())
- {
- $connection = $this->getMockedConnection($executeCallback);
- $client = new Client($connection);
- $transaction = new MultiExecContext($client, $options);
- return $transaction;
- }
- /**
- * Returns a callback that emulates a server-side MULTI/EXEC transaction context.
- *
- * @param array $expected Expected replies.
- * @param array $commands Reference to an array that stores the whole flow of commands.
- * @return \Closure
- */
- protected function getExecuteCallback($expected = array(), &$commands = array(), &$cas = array())
- {
- $multi = $watch = $abort = false;
- return function(ICommand $command) use(&$expected, &$commands, &$cas, &$multi, &$watch, &$abort) {
- $cmd = $command->getId();
- if ($multi || $cmd === 'MULTI') {
- $commands[] = $command;
- }
- else {
- $cas[] = $command;
- }
- switch ($cmd) {
- case 'WATCH':
- if ($multi) {
- throw new ServerException("ERR $cmd inside MULTI is not allowed");
- }
- return $watch = true;
- case 'MULTI':
- if ($multi) {
- throw new ServerException("ERR MULTI calls can not be nested");
- }
- return $multi = true;
- case 'EXEC':
- if (!$multi) {
- throw new ServerException("ERR $cmd without MULTI");
- }
- $watch = $multi = false;
- if ($abort) {
- $commands = $cas = array();
- $abort = false;
- return null;
- }
- return $expected;
- case 'DISCARD':
- if (!$multi) {
- throw new ServerException("ERR $cmd without MULTI");
- }
- $watch = $multi = false;
- return true;
- case 'ECHO':
- @list($trigger) = $command->getArguments();
- if (strpos($trigger, 'ERR ') === 0) {
- throw new ServerException($trigger);
- }
- if ($trigger === '!!ABORT!!' && $multi) {
- $abort = true;
- }
- return new ResponseQueued();
- case 'UNWATCH':
- $watch = false;
- default:
- return $multi ? new ResponseQueued() : 'DUMMY_REPLY';
- }
- };
- }
- /**
- * Converts an array of instances of Predis\Commands\ICommand and
- * returns an array containing their IDs.
- *
- * @param array $commands List of commands instances.
- * @return array
- */
- protected static function commandsToIDs($commands) {
- return array_map(function($cmd) { return $cmd->getId(); }, $commands);
- }
- /**
- * Returns a client instance connected to the specified Redis
- * server instance to perform integration tests.
- *
- * @return array Additional connection parameters.
- * @return Client client instance.
- */
- protected function getClient(Array $parameters = array())
- {
- $parameters = array_merge(array(
- 'scheme' => 'tcp',
- 'host' => REDIS_SERVER_HOST,
- 'port' => REDIS_SERVER_PORT,
- 'database' => REDIS_SERVER_DBNUM,
- ), $parameters);
- $client = new Client($parameters, array('profile' => REDIS_SERVER_VERSION));
- $client->connect();
- $client->flushdb();
- return $client;
- }
- }
|