2016-12-30 02:16:04 +01:00
|
|
|
<?php
|
2015-08-03 07:20:06 +02:00
|
|
|
|
2016-08-23 23:47:40 +02:00
|
|
|
namespace Amp\Parallel\Test\Sync;
|
2016-08-18 18:04:48 +02:00
|
|
|
|
2016-08-23 23:47:40 +02:00
|
|
|
use Amp\Parallel\Sync\ChannelledStream;
|
2017-01-09 18:11:46 +01:00
|
|
|
use Amp\Stream\ByteStream;
|
2016-08-19 00:36:58 +02:00
|
|
|
use Amp\Stream\ClosedException;
|
2016-08-23 23:47:40 +02:00
|
|
|
use Amp\Parallel\Test\TestCase;
|
2016-08-19 00:36:58 +02:00
|
|
|
use Amp\Success;
|
2017-01-09 18:11:25 +01:00
|
|
|
use AsyncInterop\Loop;
|
2016-08-19 00:36:58 +02:00
|
|
|
|
|
|
|
class ChannelledStreamTest extends TestCase {
|
2015-09-27 18:15:47 +02:00
|
|
|
/**
|
2017-01-09 18:11:46 +01:00
|
|
|
* @return \Amp\Stream\ByteStream|\PHPUnit_Framework_MockObject_MockObject
|
2015-09-27 18:15:47 +02:00
|
|
|
*/
|
2016-08-19 00:36:58 +02:00
|
|
|
protected function createMockStream() {
|
2017-01-09 18:11:46 +01:00
|
|
|
$mock = $this->createMock(ByteStream::class);
|
2015-09-27 18:15:47 +02:00
|
|
|
|
|
|
|
$buffer = '';
|
|
|
|
|
|
|
|
$mock->method('write')
|
|
|
|
->will($this->returnCallback(function ($data) use (&$buffer) {
|
|
|
|
$buffer .= $data;
|
2016-08-19 00:36:58 +02:00
|
|
|
return new Success(\strlen($data));
|
2015-09-27 18:15:47 +02:00
|
|
|
}));
|
|
|
|
|
|
|
|
$mock->method('read')
|
|
|
|
->will($this->returnCallback(function ($length, $byte = null, $timeout = 0) use (&$buffer) {
|
2016-08-19 00:36:58 +02:00
|
|
|
$result = \substr($buffer, 0, $length);
|
|
|
|
$buffer = \substr($buffer, $length);
|
|
|
|
return new Success($result);
|
2015-09-27 18:15:47 +02:00
|
|
|
}));
|
|
|
|
|
|
|
|
return $mock;
|
|
|
|
}
|
2016-08-19 00:36:58 +02:00
|
|
|
|
|
|
|
public function testSendReceive() {
|
2016-12-30 02:17:26 +01:00
|
|
|
Loop::execute(\Amp\wrap(function () {
|
2015-09-27 18:15:47 +02:00
|
|
|
$mock = $this->createMockStream();
|
2015-12-06 07:32:06 +01:00
|
|
|
$a = new ChannelledStream($mock);
|
|
|
|
$b = new ChannelledStream($mock);
|
2015-08-03 07:20:06 +02:00
|
|
|
|
2015-09-27 18:15:47 +02:00
|
|
|
$message = 'hello';
|
|
|
|
|
2016-08-19 00:36:58 +02:00
|
|
|
yield $a->send($message);
|
|
|
|
$data = yield $b->receive();
|
2015-09-27 18:15:47 +02:00
|
|
|
$this->assertSame($message, $data);
|
2016-12-30 02:17:26 +01:00
|
|
|
}));
|
2015-09-27 18:15:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @depends testSendReceive
|
|
|
|
*/
|
2016-08-19 00:36:58 +02:00
|
|
|
public function testSendReceiveLongData() {
|
2016-12-30 02:17:26 +01:00
|
|
|
Loop::execute(\Amp\wrap(function () {
|
2015-09-27 18:15:47 +02:00
|
|
|
$mock = $this->createMockStream();
|
2015-12-06 07:32:06 +01:00
|
|
|
$a = new ChannelledStream($mock);
|
|
|
|
$b = new ChannelledStream($mock);
|
2015-09-27 18:15:47 +02:00
|
|
|
|
|
|
|
$length = 0xffff;
|
|
|
|
$message = '';
|
|
|
|
for ($i = 0; $i < $length; ++$i) {
|
|
|
|
$message .= chr(mt_rand(0, 255));
|
|
|
|
}
|
|
|
|
|
2016-08-19 00:36:58 +02:00
|
|
|
yield $a->send($message);
|
|
|
|
$data = yield $b->receive();
|
2015-09-27 18:15:47 +02:00
|
|
|
$this->assertSame($message, $data);
|
2016-12-30 02:17:26 +01:00
|
|
|
}));
|
2015-08-03 07:20:06 +02:00
|
|
|
|
|
|
|
}
|
2015-09-03 01:29:48 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @depends testSendReceive
|
2016-08-23 23:47:40 +02:00
|
|
|
* @expectedException \Amp\Parallel\ChannelException
|
2015-09-03 01:29:48 +02:00
|
|
|
*/
|
2016-08-19 00:36:58 +02:00
|
|
|
public function testInvalidDataReceived() {
|
2016-12-30 02:17:26 +01:00
|
|
|
Loop::execute(\Amp\wrap(function () {
|
2015-09-27 18:15:47 +02:00
|
|
|
$mock = $this->createMockStream();
|
2015-12-06 07:32:06 +01:00
|
|
|
$a = new ChannelledStream($mock);
|
|
|
|
$b = new ChannelledStream($mock);
|
2015-09-03 01:29:48 +02:00
|
|
|
|
|
|
|
// Close $a. $b should close on next read...
|
2016-08-19 00:36:58 +02:00
|
|
|
yield $mock->write(pack('L', 10) . '1234567890');
|
|
|
|
$data = yield $b->receive();
|
2016-12-30 02:17:26 +01:00
|
|
|
}));
|
2015-09-03 01:29:48 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @depends testSendReceive
|
2016-08-23 23:47:40 +02:00
|
|
|
* @expectedException \Amp\Parallel\ChannelException
|
2015-09-03 01:29:48 +02:00
|
|
|
*/
|
2016-08-19 00:36:58 +02:00
|
|
|
public function testSendUnserializableData() {
|
2016-12-30 02:17:26 +01:00
|
|
|
Loop::execute(\Amp\wrap(function () {
|
2015-09-27 18:15:47 +02:00
|
|
|
$mock = $this->createMockStream();
|
2015-12-06 07:32:06 +01:00
|
|
|
$a = new ChannelledStream($mock);
|
|
|
|
$b = new ChannelledStream($mock);
|
2015-09-03 01:29:48 +02:00
|
|
|
|
|
|
|
// Close $a. $b should close on next read...
|
2016-08-19 00:36:58 +02:00
|
|
|
yield $a->send(function () {});
|
|
|
|
$data = yield $b->receive();
|
2016-12-30 02:17:26 +01:00
|
|
|
}));
|
2015-09-03 01:29:48 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @depends testSendReceive
|
2016-08-23 23:47:40 +02:00
|
|
|
* @expectedException \Amp\Parallel\ChannelException
|
2015-09-03 01:29:48 +02:00
|
|
|
*/
|
2016-08-19 00:36:58 +02:00
|
|
|
public function testSendAfterClose() {
|
2016-12-30 02:17:26 +01:00
|
|
|
Loop::execute(\Amp\wrap(function () {
|
2017-01-09 18:11:46 +01:00
|
|
|
$mock = $this->createMock(ByteStream::class);
|
2015-09-27 18:15:47 +02:00
|
|
|
$mock->expects($this->once())
|
|
|
|
->method('write')
|
2016-08-19 00:36:58 +02:00
|
|
|
->will($this->throwException(new ClosedException));
|
2015-09-03 01:29:48 +02:00
|
|
|
|
2015-12-06 07:32:06 +01:00
|
|
|
$a = new ChannelledStream($mock);
|
2017-01-09 18:11:46 +01:00
|
|
|
$b = new ChannelledStream($this->createMock(ByteStream::class));
|
2015-09-03 01:29:48 +02:00
|
|
|
|
2016-08-19 00:36:58 +02:00
|
|
|
yield $a->send('hello');
|
2016-12-30 02:17:26 +01:00
|
|
|
}));
|
2015-09-03 01:29:48 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @depends testSendReceive
|
2016-08-23 23:47:40 +02:00
|
|
|
* @expectedException \Amp\Parallel\ChannelException
|
2015-09-03 01:29:48 +02:00
|
|
|
*/
|
2016-08-19 00:36:58 +02:00
|
|
|
public function testReceiveAfterClose() {
|
2016-12-30 02:17:26 +01:00
|
|
|
Loop::execute(\Amp\wrap(function () {
|
2017-01-09 18:11:46 +01:00
|
|
|
$mock = $this->createMock(ByteStream::class);
|
2015-09-27 18:15:47 +02:00
|
|
|
$mock->expects($this->once())
|
|
|
|
->method('read')
|
2016-08-19 00:36:58 +02:00
|
|
|
->will($this->throwException(new ClosedException));
|
2015-09-03 01:29:48 +02:00
|
|
|
|
2015-12-06 07:32:06 +01:00
|
|
|
$a = new ChannelledStream($mock);
|
2015-09-03 01:29:48 +02:00
|
|
|
|
2016-08-19 00:36:58 +02:00
|
|
|
$data = yield $a->receive();
|
2016-12-30 02:17:26 +01:00
|
|
|
}));
|
2015-09-03 01:29:48 +02:00
|
|
|
|
|
|
|
}
|
2015-08-03 07:20:06 +02:00
|
|
|
}
|