[Server] Tests

Increased unit test code coverage
Replaced concrete mock objects with PHPUnit mocks
This commit is contained in:
Chris Boden 2012-07-19 09:03:07 -04:00
parent bd5b7d09aa
commit 4a4e80a945
6 changed files with 176 additions and 57 deletions

View File

@ -11,6 +11,7 @@ interface ConnectionInterface {
/** /**
* Send data to the connection * Send data to the connection
* @param string * @param string
* @return ConnectionInterface
*/ */
function send($data); function send($data);

View File

@ -13,14 +13,16 @@ class IoConnection implements ConnectionInterface {
protected $conn; protected $conn;
public function __construct(ReactConn $conn) { public function __construct(ReactConn $conn) {
$this->conn = $conn; $this->conn = $conn;
} }
/** /**
* {@inheritdoc} * {@inheritdoc}
*/ */
public function send($data) { public function send($data) {
return $this->conn->write($data); $this->conn->write($data);
return $this;
} }
/** /**

View File

@ -17,6 +17,7 @@ class FlashPolicyTest extends \PHPUnit_Framework_TestCase {
$this->_policy->setSiteControl('all'); $this->_policy->setSiteControl('all');
$this->_policy->addAllowedAccess('example.com', '*'); $this->_policy->addAllowedAccess('example.com', '*');
$this->_policy->addAllowedAccess('dev.example.com', '*'); $this->_policy->addAllowedAccess('dev.example.com', '*');
$this->assertInstanceOf('SimpleXMLElement', $this->_policy->renderPolicy()); $this->assertInstanceOf('SimpleXMLElement', $this->_policy->renderPolicy());
} }
@ -108,4 +109,32 @@ class FlashPolicyTest extends \PHPUnit_Framework_TestCase {
, array(false, '838*') , array(false, '838*')
); );
} }
public function testAddAllowedAccessOnlyAcceptsValidPorts() {
$this->setExpectedException('UnexpectedValueException');
$this->_policy->addAllowedAccess('*', 'nope');
}
public function testSetSiteControlThrowsException() {
$this->setExpectedException('UnexpectedValueException');
$this->_policy->setSiteControl('nope');
}
public function testErrorClosesConnection() {
$conn = $this->getMock('\\Ratchet\\ConnectionInterface');
$conn->expects($this->once())->method('close');
$this->_policy->onError($conn, new \Exception);
}
public function testOnMessageSendsString() {
$this->_policy->addAllowedAccess('*', '*');
$conn = $this->getMock('\\Ratchet\\ConnectionInterface');
$conn->expects($this->once())->method('send')->with($this->isType('string'));
$this->_policy->onMessage($conn, ' ');
}
} }

View File

@ -0,0 +1,32 @@
<?php
namespace Ratchet\Tests\Application\Server;
use Ratchet\Server\IoConnection;
/**
* @covers Ratchet\Server\IoConnection
*/
class IoConnectionTest extends \PHPUnit_Framework_TestCase {
protected $sock;
protected $conn;
public function setUp() {
$this->sock = $this->getMock('\\React\\Socket\\ConnectionInterface');
$this->conn = new IoConnection($this->sock);
}
public function testCloseBubbles() {
$this->sock->expects($this->once())->method('end');
$this->conn->close();
}
public function testSendBubbles() {
$msg = '6 hour rides are productive';
$this->sock->expects($this->once())->method('write')->with($msg);
$this->conn->send($msg);
}
public function testSendReturnsSelf() {
$this->assertSame($this->conn, $this->conn->send('fluent interface'));
}
}

View File

@ -3,7 +3,6 @@ namespace Ratchet\Tests\Server;
use Ratchet\Server\IoServer; use Ratchet\Server\IoServer;
use React\EventLoop\StreamSelectLoop; use React\EventLoop\StreamSelectLoop;
use React\Socket\Server; use React\Socket\Server;
use Ratchet\Tests\Mock\Component;
/** /**
* @covers Ratchet\Server\IoServer * @covers Ratchet\Server\IoServer
@ -18,7 +17,7 @@ class IoServerTest extends \PHPUnit_Framework_TestCase {
protected $reactor; protected $reactor;
public function setUp() { public function setUp() {
$this->app = new Component; $this->app = $this->getMock('\\Ratchet\\MessageComponentInterface');
$loop = new StreamSelectLoop; $loop = new StreamSelectLoop;
$this->reactor = new Server($loop); $this->reactor = new Server($loop);
@ -29,18 +28,24 @@ class IoServerTest extends \PHPUnit_Framework_TestCase {
} }
public function testOnOpen() { public function testOnOpen() {
$this->app->expects($this->once())->method('onOpen')->with($this->isInstanceOf('\\Ratchet\\ConnectionInterface'));
$client = stream_socket_client("tcp://localhost:{$this->port}"); $client = stream_socket_client("tcp://localhost:{$this->port}");
$this->server->loop->tick(); $this->server->loop->tick();
$this->assertInstanceOf('\\Ratchet\\ConnectionInterface', $this->app->last['onOpen'][0]); //$this->assertTrue(is_string($this->app->last['onOpen'][0]->remoteAddress));
$this->assertTrue(is_string($this->app->last['onOpen'][0]->remoteAddress)); //$this->assertTrue(is_int($this->app->last['onOpen'][0]->resourceId));
$this->assertTrue(is_int($this->app->last['onOpen'][0]->resourceId));
} }
public function testOnData() { public function testOnData() {
$msg = 'Hello World!'; $msg = 'Hello World!';
$this->app->expects($this->once())->method('onMessage')->with(
$this->isInstanceOf('\\Ratchet\\ConnectionInterface')
, $msg
);
$client = socket_create(AF_INET, SOCK_STREAM, SOL_TCP); $client = socket_create(AF_INET, SOCK_STREAM, SOL_TCP);
socket_set_option($client, SOL_SOCKET, SO_REUSEADDR, 1); socket_set_option($client, SOL_SOCKET, SO_REUSEADDR, 1);
socket_set_option($client, SOL_SOCKET, SO_SNDBUF, 4096); socket_set_option($client, SOL_SOCKET, SO_SNDBUF, 4096);
@ -56,14 +61,12 @@ class IoServerTest extends \PHPUnit_Framework_TestCase {
socket_shutdown($client, 0); socket_shutdown($client, 0);
socket_close($client); socket_close($client);
usleep(5000);
$this->server->loop->tick(); $this->server->loop->tick();
$this->assertEquals($msg, $this->app->last['onMessage'][1]);
} }
public function testOnClose() { public function testOnClose() {
$this->app->expects($this->once())->method('onClose')->with($this->isInstanceOf('\\Ratchet\\ConnectionInterface'));
$client = socket_create(AF_INET, SOCK_STREAM, SOL_TCP); $client = socket_create(AF_INET, SOCK_STREAM, SOL_TCP);
socket_set_option($client, SOL_SOCKET, SO_REUSEADDR, 1); socket_set_option($client, SOL_SOCKET, SO_REUSEADDR, 1);
socket_set_option($client, SOL_SOCKET, SO_SNDBUF, 4096); socket_set_option($client, SOL_SOCKET, SO_SNDBUF, 4096);
@ -77,10 +80,6 @@ class IoServerTest extends \PHPUnit_Framework_TestCase {
socket_close($client); socket_close($client);
$this->server->loop->tick(); $this->server->loop->tick();
usleep(5000);
$this->assertSame($this->app->last['onOpen'][0], $this->app->last['onClose'][0]);
} }
public function testFactory() { public function testFactory() {
@ -88,10 +87,32 @@ class IoServerTest extends \PHPUnit_Framework_TestCase {
} }
public function testNoLoopProvidedError() { public function testNoLoopProvidedError() {
$loop = new StreamSelectLoop;
$io = new IoServer(new Component, new Server($loop));
$this->setExpectedException('RuntimeException'); $this->setExpectedException('RuntimeException');
$io = new IoServer($this->app, $this->reactor);
$io->run(); $io->run();
} }
public function testOnErrorPassesException() {
$conn = $this->getMock('\\React\\Socket\\ConnectionInterface');
$conn->decor = $this->getMock('\\Ratchet\\ConnectionInterface');
$err = new \Exception("Nope");
$this->app->expects($this->once())->method('onError')->with($conn->decor, $err);
$this->server->handleError($err, $conn);
}
public function onErrorCalledWhenExceptionThrown() {
$this->markTestIncomplete("Need to learn how to throw an exception from a mock");
$conn = $this->getMock('\\React\\Socket\\ConnectionInterface');
$this->server->handleConnect($conn);
$e = new \Exception;
$this->app->expects($this->once())->method('onMessage')->with($this->isInstanceOf('\\Ratchet\\ConnectionInterface'), 'f')->will($e);
$this->app->expects($this->once())->method('onError')->with($this->instanceOf('\\Ratchet\\ConnectionInterface', $e));
$this->server->handleData('f', $conn);
}
} }

View File

@ -1,29 +1,75 @@
<?php <?php
namespace Ratchet\Tests\Server; namespace Ratchet\Tests\Server;
use Ratchet\Server\IpBlackList; use Ratchet\Server\IpBlackList;
use Ratchet\Tests\Mock\Connection;
use Ratchet\Tests\Mock\Component as MockComponent;
/** /**
* @covers Ratchet\Server\IpBlackList * @covers Ratchet\Server\IpBlackList
*/ */
class IpBlackListTest extends \PHPUnit_Framework_TestCase { class IpBlackListTest extends \PHPUnit_Framework_TestCase {
protected $_comp; protected $blocker;
protected $_mock; protected $mock;
public function setUp() { public function setUp() {
$this->_mock = new MockComponent; $this->mock = $this->getMock('\\Ratchet\\MessageComponentInterface');
$this->_comp = new IpBlackList($this->_mock); $this->blocker = new IpBlackList($this->mock);
} }
public function testBlockAndCloseOnOpen() { public function testOnOpen() {
$conn = new Connection; $this->mock->expects($this->exactly(3))->method('onOpen');
$this->_comp->blockAddress($conn->remoteAddress); $conn1 = $this->newConn();
$conn2 = $this->newConn();
$conn3 = $this->newConn();
$ret = $this->_comp->onOpen($conn); $this->blocker->onOpen($conn1);
$this->blocker->onOpen($conn3);
$this->blocker->onOpen($conn2);
}
$this->assertTrue($conn->last['close']); public function testBlockDoesNotTriggerOnOpen() {
$conn = $this->newConn();
$this->blocker->blockAddress($conn->remoteAddress);
$this->mock->expects($this->never())->method('onOpen');
$ret = $this->blocker->onOpen($conn);
}
public function testBlockDoesNotTriggerOnClose() {
$conn = $this->newConn();
$this->blocker->blockAddress($conn->remoteAddress);
$this->mock->expects($this->never())->method('onClose');
$ret = $this->blocker->onOpen($conn);
}
public function testOnMessageDecoration() {
$conn = $this->newConn();
$msg = 'Hello not being blocked';
$this->mock->expects($this->once())->method('onMessage')->with($conn, $msg);
$this->blocker->onMessage($conn, $msg);
}
public function testOnCloseDecoration() {
$conn = $this->newConn();
$this->mock->expects($this->once())->method('onClose')->with($conn);
$this->blocker->onClose($conn);
}
public function testBlockClosesConnection() {
$conn = $this->newConn();
$this->blocker->blockAddress($conn->remoteAddress);
$conn->expects($this->once())->method('close');
$this->blocker->onOpen($conn);
} }
public function testAddAndRemoveWithFluentInterfaces() { public function testAddAndRemoveWithFluentInterfaces() {
@ -31,42 +77,23 @@ class IpBlackListTest extends \PHPUnit_Framework_TestCase {
$blockTwo = '192.168.1.1'; $blockTwo = '192.168.1.1';
$unblock = '75.119.207.140'; $unblock = '75.119.207.140';
$this->_comp $this->blocker
->blockAddress($unblock) ->blockAddress($unblock)
->blockAddress($blockOne) ->blockAddress($blockOne)
->unblockAddress($unblock) ->unblockAddress($unblock)
->blockAddress($blockTwo) ->blockAddress($blockTwo)
; ;
$this->assertEquals(array($blockOne, $blockTwo), $this->_comp->getBlockedAddresses()); $this->assertEquals(array($blockOne, $blockTwo), $this->blocker->getBlockedAddresses());
} }
public function testDecoratingMethods() { public function testDecoratorPassesErrors() {
$conn1 = new Connection; $conn = $this->newConn();
$conn2 = new Connection; $e = new \Exception('I threw an error');
$conn3 = new Connection;
$this->_comp->onOpen($conn1); $this->mock->expects($this->once())->method('onError')->with($conn, $e);
$this->_comp->onOpen($conn3);
$this->_comp->onOpen($conn2);
$this->assertSame($conn2, $this->_mock->last['onOpen'][0]);
$msg = 'Hello World!'; $this->blocker->onError($conn, $e);
$this->_comp->onMessage($conn1, $msg);
$this->assertSame($conn1, $this->_mock->last['onMessage'][0]);
$this->assertEquals($msg, $this->_mock->last['onMessage'][1]);
$this->_comp->onClose($conn3);
$this->assertSame($conn3, $this->_mock->last['onClose'][0]);
try {
throw new \Exception('I threw an error');
} catch (\Exception $e) {
}
$this->_comp->onError($conn2, $e);
$this->assertEquals($conn2, $this->_mock->last['onError'][0]);
$this->assertEquals($e, $this->_mock->last['onError'][1]);
} }
public function addressProvider() { public function addressProvider() {
@ -82,10 +109,17 @@ class IpBlackListTest extends \PHPUnit_Framework_TestCase {
* @dataProvider addressProvider * @dataProvider addressProvider
*/ */
public function testFilterAddress($expected, $input) { public function testFilterAddress($expected, $input) {
$this->assertEquals($expected, $this->_comp->filterAddress($input)); $this->assertEquals($expected, $this->blocker->filterAddress($input));
} }
public function testUnblockingSilentlyFails() { public function testUnblockingSilentlyFails() {
$this->assertInstanceOf('\\Ratchet\\Server\\IpBlackList', $this->_comp->unblockAddress('localhost')); $this->assertInstanceOf('\\Ratchet\\Server\\IpBlackList', $this->blocker->unblockAddress('localhost'));
}
protected function newConn() {
$conn = $this->getMock('\\Ratchet\\ConnectionInterface');
$conn->remoteAddress = '127.0.0.1';
return $conn;
} }
} }