123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424 |
- <?php
- namespace React\Tests\Socket;
- use React\EventLoop\Loop;
- use React\Socket\UnixServer;
- use React\Stream\DuplexResourceStream;
- class UnixServerTest extends TestCase
- {
- /** @var ?UnixServer */
- private $server;
- /** @var ?string */
- private $uds;
- /**
- * @before
- * @covers React\Socket\UnixServer::__construct
- * @covers React\Socket\UnixServer::getAddress
- */
- public function setUpServer()
- {
- if (!in_array('unix', stream_get_transports())) {
- $this->markTestSkipped('Unix domain sockets (UDS) not supported on your platform (Windows?)');
- }
- $this->uds = $this->getRandomSocketUri();
- $this->server = new UnixServer($this->uds);
- }
- public function testConstructWithoutLoopAssignsLoopAutomatically()
- {
- unlink(str_replace('unix://', '', $this->uds));
- $this->uds = $this->getRandomSocketUri();
- $server = new UnixServer($this->uds);
- $ref = new \ReflectionProperty($server, 'loop');
- $ref->setAccessible(true);
- $loop = $ref->getValue($server);
- $this->assertInstanceOf('React\EventLoop\LoopInterface', $loop);
- $server->close();
- }
- /**
- * @covers React\Socket\UnixServer::handleConnection
- */
- public function testConnection()
- {
- $client = stream_socket_client($this->uds);
- assert(is_resource($client));
- $this->server->on('connection', $this->expectCallableOnce());
- $this->tick();
- $this->tick();
- }
- /**
- * @covers React\Socket\UnixServer::handleConnection
- */
- public function testConnectionWithManyClients()
- {
- $client1 = stream_socket_client($this->uds);
- assert(is_resource($client1));
- $client2 = stream_socket_client($this->uds);
- assert(is_resource($client2));
- $client3 = stream_socket_client($this->uds);
- assert(is_resource($client3));
- $this->server->on('connection', $this->expectCallableExactly(3));
- $this->tick();
- $this->tick();
- $this->tick();
- }
- public function testDataEventWillNotBeEmittedWhenClientSendsNoData()
- {
- $client = stream_socket_client($this->uds);
- assert(is_resource($client));
- $mock = $this->expectCallableNever();
- $this->server->on('connection', function ($conn) use ($mock) {
- $conn->on('data', $mock);
- });
- $this->tick();
- $this->tick();
- }
- public function testDataWillBeEmittedWithDataClientSends()
- {
- $client = stream_socket_client($this->uds);
- assert(is_resource($client));
- fwrite($client, "foo\n");
- $mock = $this->expectCallableOnceWith("foo\n");
- $this->server->on('connection', function ($conn) use ($mock) {
- $conn->on('data', $mock);
- });
- $this->tick();
- $this->tick();
- }
- public function testDataWillBeEmittedEvenWhenClientShutsDownAfterSending()
- {
- $client = stream_socket_client($this->uds);
- fwrite($client, "foo\n");
- stream_socket_shutdown($client, STREAM_SHUT_WR);
- $mock = $this->expectCallableOnceWith("foo\n");
- $this->server->on('connection', function ($conn) use ($mock) {
- $conn->on('data', $mock);
- });
- $this->tick();
- $this->tick();
- }
- public function testLoopWillEndWhenServerIsClosed()
- {
- // explicitly unset server because we already call close()
- $this->server->close();
- $this->server = null;
- Loop::run();
- // if we reach this, then everything is good
- $this->assertNull(null);
- }
- public function testCloseTwiceIsNoOp()
- {
- $this->server->close();
- $this->server->close();
- // if we reach this, then everything is good
- $this->assertNull(null);
- }
- public function testGetAddressAfterCloseReturnsNull()
- {
- $this->server->close();
- $this->assertNull($this->server->getAddress());
- }
- public function testLoopWillEndWhenServerIsClosedAfterSingleConnection()
- {
- $client = stream_socket_client($this->uds);
- assert(is_resource($client));
- // explicitly unset server because we only accept a single connection
- // and then already call close()
- $server = $this->server;
- $this->server = null;
- $server->on('connection', function ($conn) use ($server) {
- $conn->close();
- $server->close();
- });
- Loop::run();
- // if we reach this, then everything is good
- $this->assertNull(null);
- }
- public function testDataWillBeEmittedInMultipleChunksWhenClientSendsExcessiveAmounts()
- {
- $client = stream_socket_client($this->uds);
- $stream = new DuplexResourceStream($client);
- $bytes = 1024 * 1024;
- $stream->end(str_repeat('*', $bytes));
- $mock = $this->expectCallableOnce();
- // explicitly unset server because we only accept a single connection
- // and then already call close()
- $server = $this->server;
- $this->server = null;
- $received = 0;
- $server->on('connection', function ($conn) use ($mock, &$received, $server) {
- // count number of bytes received
- $conn->on('data', function ($data) use (&$received) {
- $received += strlen($data);
- });
- $conn->on('end', $mock);
- // do not await any further connections in order to let the loop terminate
- $server->close();
- });
- Loop::run();
- $this->assertEquals($bytes, $received);
- }
- public function testConnectionDoesNotEndWhenClientDoesNotClose()
- {
- $client = stream_socket_client($this->uds);
- assert(is_resource($client));
- $mock = $this->expectCallableNever();
- $this->server->on('connection', function ($conn) use ($mock) {
- $conn->on('end', $mock);
- });
- $this->tick();
- $this->tick();
- }
- /**
- * @covers React\Socket\Connection::end
- */
- public function testConnectionDoesEndWhenClientCloses()
- {
- $client = stream_socket_client($this->uds);
- fclose($client);
- $mock = $this->expectCallableOnce();
- $this->server->on('connection', function ($conn) use ($mock) {
- $conn->on('end', $mock);
- });
- $this->tick();
- $this->tick();
- }
- public function testCtorAddsResourceToLoop()
- {
- unlink(str_replace('unix://', '', $this->uds));
- $this->uds = $this->getRandomSocketUri();
- $loop = $this->getMockBuilder('React\EventLoop\LoopInterface')->getMock();
- $loop->expects($this->once())->method('addReadStream');
- new UnixServer($this->uds, $loop);
- }
- public function testCtorThrowsForInvalidAddressScheme()
- {
- $loop = $this->getMockBuilder('React\EventLoop\LoopInterface')->getMock();
- $this->setExpectedException(
- 'InvalidArgumentException',
- 'Given URI "tcp://localhost:0" is invalid (EINVAL)',
- defined('SOCKET_EINVAL') ? SOCKET_EINVAL : (defined('PCNTL_EINVAL') ? PCNTL_EINVAL : 22)
- );
- new UnixServer('tcp://localhost:0', $loop);
- }
- public function testCtorThrowsWhenPathIsNotWritableWithoutCallingCustomErrorHandler()
- {
- $loop = $this->getMockBuilder('React\EventLoop\LoopInterface')->getMock();
- $error = null;
- set_error_handler(function ($_, $errstr) use (&$error) {
- $error = $errstr;
- });
- $this->setExpectedException('RuntimeException');
- try {
- new UnixServer('/dev/null', $loop);
- restore_error_handler();
- } catch (\Exception $e) {
- restore_error_handler();
- $this->assertNull($error);
- throw $e;
- }
- }
- public function testResumeWithoutPauseIsNoOp()
- {
- unlink(str_replace('unix://', '', $this->uds));
- $this->uds = $this->getRandomSocketUri();
- $loop = $this->getMockBuilder('React\EventLoop\LoopInterface')->getMock();
- $loop->expects($this->once())->method('addReadStream');
- $server = new UnixServer($this->uds, $loop);
- $server->resume();
- }
- public function testPauseRemovesResourceFromLoop()
- {
- unlink(str_replace('unix://', '', $this->uds));
- $this->uds = $this->getRandomSocketUri();
- $loop = $this->getMockBuilder('React\EventLoop\LoopInterface')->getMock();
- $loop->expects($this->once())->method('removeReadStream');
- $server = new UnixServer($this->uds, $loop);
- $server->pause();
- }
- public function testPauseAfterPauseIsNoOp()
- {
- unlink(str_replace('unix://', '', $this->uds));
- $this->uds = $this->getRandomSocketUri();
- $loop = $this->getMockBuilder('React\EventLoop\LoopInterface')->getMock();
- $loop->expects($this->once())->method('removeReadStream');
- $server = new UnixServer($this->uds, $loop);
- $server->pause();
- $server->pause();
- }
- public function testCloseRemovesResourceFromLoop()
- {
- unlink(str_replace('unix://', '', $this->uds));
- $this->uds = $this->getRandomSocketUri();
- $loop = $this->getMockBuilder('React\EventLoop\LoopInterface')->getMock();
- $loop->expects($this->once())->method('removeReadStream');
- $server = new UnixServer($this->uds, $loop);
- $server->close();
- }
- public function testEmitsErrorWhenAcceptListenerFailsWithoutCallingCustomErrorHandler()
- {
- unlink(str_replace('unix://', '', $this->uds));
- $this->uds = $this->getRandomSocketUri();
- $listener = null;
- $loop = $this->getMockBuilder('React\EventLoop\LoopInterface')->getMock();
- $loop->expects($this->once())->method('addReadStream')->with($this->anything(), $this->callback(function ($cb) use (&$listener) {
- $listener = $cb;
- return true;
- }));
- $server = new UnixServer($this->uds, $loop);
- $exception = null;
- $server->on('error', function ($e) use (&$exception) {
- $exception = $e;
- });
- $this->assertNotNull($listener);
- $socket = stream_socket_server('tcp://127.0.0.1:0');
- $error = null;
- set_error_handler(function ($_, $errstr) use (&$error) {
- $error = $errstr;
- });
- $time = microtime(true);
- $listener($socket);
- $time = microtime(true) - $time;
- restore_error_handler();
- $this->assertNull($error);
- $this->assertLessThan(1, $time);
- $this->assertInstanceOf('RuntimeException', $exception);
- assert($exception instanceof \RuntimeException);
- $this->assertStringStartsWith('Unable to accept new connection: ', $exception->getMessage());
- return $exception;
- }
- /**
- * @param \RuntimeException $e
- * @requires extension sockets
- * @depends testEmitsErrorWhenAcceptListenerFailsWithoutCallingCustomErrorHandler
- */
- public function testEmitsTimeoutErrorWhenAcceptListenerFails(\RuntimeException $exception)
- {
- if (defined('HHVM_VERSION')) {
- $this->markTestSkipped('not supported on HHVM');
- }
- $this->assertEquals('Unable to accept new connection: ' . socket_strerror(SOCKET_ETIMEDOUT) . ' (ETIMEDOUT)', $exception->getMessage());
- $this->assertEquals(SOCKET_ETIMEDOUT, $exception->getCode());
- }
- public function testListenOnBusyPortThrows()
- {
- if (DIRECTORY_SEPARATOR === '\\') {
- $this->markTestSkipped('Windows supports listening on same port multiple times');
- }
- $this->setExpectedException('RuntimeException');
- new UnixServer($this->uds);
- }
- /**
- * @after
- * @covers React\Socket\UnixServer::close
- */
- public function tearDownServer()
- {
- if ($this->server) {
- $this->server->close();
- $this->server = null;
- }
- assert(is_string($this->uds));
- unlink(str_replace('unix://', '', $this->uds));
- }
- private function getRandomSocketUri()
- {
- return "unix://" . sys_get_temp_dir() . DIRECTORY_SEPARATOR . uniqid(rand(), true) . '.sock';
- }
- private function tick()
- {
- \React\Async\await(\React\Promise\Timer\sleep(0.0));
- }
- }
|