mirror of
https://github.com/danog/amp.git
synced 2025-01-22 21:31:18 +01:00
256 lines
6.5 KiB
PHP
256 lines
6.5 KiB
PHP
<?php
|
|
|
|
namespace Amp\Test;
|
|
|
|
use Amp;
|
|
use Amp\Producer;
|
|
use Amp\Listener;
|
|
use Amp\Pause;
|
|
use Amp\Emitter;
|
|
use Amp\Loop;
|
|
|
|
class ListenerTest extends \PHPUnit\Framework\TestCase {
|
|
const TIMEOUT = 10;
|
|
|
|
public function testSingleEmittingStream() {
|
|
Loop::run(function () {
|
|
$value = 1;
|
|
$stream = new Producer(function (callable $emit) use ($value) {
|
|
yield $emit($value);
|
|
return $value;
|
|
});
|
|
|
|
$listener = new Listener($stream);
|
|
|
|
while (yield $listener->advance()) {
|
|
$this->assertSame($listener->getCurrent(), $value);
|
|
}
|
|
|
|
$this->assertSame($listener->getResult(), $value);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* @depends testSingleEmittingStream
|
|
*/
|
|
public function testFastEmittingStream() {
|
|
Loop::run(function () {
|
|
$count = 10;
|
|
|
|
$emitter = new Emitter;
|
|
|
|
$listener = new Listener($emitter->stream());
|
|
|
|
for ($i = 0; $i < $count; ++$i) {
|
|
$promises[] = $emitter->emit($i);
|
|
}
|
|
|
|
$emitter->resolve($i);
|
|
|
|
for ($i = 0; yield $listener->advance(); ++$i) {
|
|
$this->assertSame($listener->getCurrent(), $i);
|
|
}
|
|
|
|
$this->assertSame($count, $i);
|
|
$this->assertSame($listener->getResult(), $i);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* @depends testSingleEmittingStream
|
|
*/
|
|
public function testSlowEmittingStream() {
|
|
Loop::run(function () {
|
|
$count = 10;
|
|
$stream = new Producer(function (callable $emit) use ($count) {
|
|
for ($i = 0; $i < $count; ++$i) {
|
|
yield new Pause(self::TIMEOUT);
|
|
yield $emit($i);
|
|
}
|
|
return $i;
|
|
});
|
|
|
|
$listener = new Listener($stream);
|
|
|
|
for ($i = 0; yield $listener->advance(); ++$i) {
|
|
$this->assertSame($listener->getCurrent(), $i);
|
|
}
|
|
|
|
$this->assertSame($count, $i);
|
|
$this->assertSame($listener->getResult(), $i);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* @depends testFastEmittingStream
|
|
*/
|
|
public function testDrain() {
|
|
Loop::run(function () {
|
|
$count = 10;
|
|
$expected = \range(0, $count - 1);
|
|
|
|
$emitter = new Emitter;
|
|
|
|
$listener = new Listener($emitter->stream());
|
|
|
|
for ($i = 0; $i < $count; ++$i) {
|
|
$promises[] = $emitter->emit($i);
|
|
}
|
|
|
|
$value = null;
|
|
if (yield $listener->advance()) {
|
|
$value = $listener->getCurrent();
|
|
}
|
|
|
|
$this->assertSame(reset($expected), $value);
|
|
unset($expected[0]);
|
|
|
|
$emitter->resolve($i);
|
|
|
|
$values = $listener->drain();
|
|
|
|
$this->assertSame($expected, $values);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* @expectedException \Error
|
|
* @expectedExceptionMessage The stream has not resolved
|
|
*/
|
|
public function testDrainBeforeResolution() {
|
|
$emitter = new Emitter;
|
|
|
|
$listener = new Listener($emitter->stream());
|
|
|
|
$listener->drain();
|
|
}
|
|
|
|
public function testFailingStream() {
|
|
Loop::run(function () {
|
|
$exception = new \Exception;
|
|
|
|
$emitter = new Emitter;
|
|
|
|
$listener = new Listener($emitter->stream());
|
|
|
|
$emitter->fail($exception);
|
|
|
|
try {
|
|
while (yield $listener->advance());
|
|
$this->fail("Listener::advance() should throw stream failure reason");
|
|
} catch (\Exception $reason) {
|
|
$this->assertSame($exception, $reason);
|
|
}
|
|
|
|
try {
|
|
$result = $listener->getResult();
|
|
$this->fail("Listener::getResult() should throw stream failure reason");
|
|
} catch (\Exception $reason) {
|
|
$this->assertSame($exception, $reason);
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* @expectedException \Error
|
|
* @expectedExceptionMessage Promise returned from advance() must resolve before calling this method
|
|
*/
|
|
public function testGetCurrentBeforeAdvanceResolves() {
|
|
$emitter = new Emitter;
|
|
|
|
$listener = new Listener($emitter->stream());
|
|
|
|
$promise = $listener->advance();
|
|
|
|
$listener->getCurrent();
|
|
}
|
|
|
|
/**
|
|
* @expectedException \Error
|
|
* @expectedExceptionMessage The stream has resolved
|
|
*/
|
|
public function testGetCurrentAfterResolution() {
|
|
$emitter = new Emitter;
|
|
|
|
$listener = new Listener($emitter->stream());
|
|
|
|
$emitter->resolve();
|
|
|
|
$listener->getCurrent();
|
|
}
|
|
|
|
/**
|
|
* @expectedException \Error
|
|
* @expectedExceptionMessage The stream has not resolved
|
|
*/
|
|
public function testGetResultBeforeResolution() {
|
|
Loop::run(Amp\wrap(function () {
|
|
$emitter = new Emitter;
|
|
|
|
$listener = new Listener($emitter->stream());
|
|
|
|
$listener->getResult();
|
|
}));
|
|
}
|
|
|
|
/**
|
|
* @expectedException \Error
|
|
* @expectedExceptionMessage The prior promise returned must resolve before invoking this method again
|
|
*/
|
|
public function testConsecutiveAdvanceCalls() {
|
|
$emitter = new Emitter;
|
|
$listener = new Listener($emitter->stream());
|
|
$listener->advance();
|
|
$listener->advance();
|
|
}
|
|
|
|
public function testListenerDestroyedAfterEmits() {
|
|
$emitter = new Emitter;
|
|
$listener = new Listener($emitter->stream());
|
|
|
|
$promise = $emitter->emit(1);
|
|
|
|
unset($listener);
|
|
|
|
$invoked = false;
|
|
$promise->onResolve(function () use (&$invoked) {
|
|
$invoked = true;
|
|
});
|
|
|
|
$this->assertTrue($invoked);
|
|
}
|
|
|
|
public function testListenerDestroyedThenStreamEmits() {
|
|
$emitter = new Emitter;
|
|
$listener = new Listener($emitter->stream());
|
|
|
|
$emitter->emit(1);
|
|
|
|
unset($listener);
|
|
|
|
$promise = $emitter->emit(2);
|
|
|
|
$invoked = false;
|
|
$promise->onResolve(function () use (&$invoked) {
|
|
$invoked = true;
|
|
});
|
|
|
|
$this->assertTrue($invoked);
|
|
}
|
|
|
|
public function testStreamFailsWhenListenerWaiting() {
|
|
$exception = new \Exception;
|
|
$emitter = new Emitter;
|
|
$listener = new Listener($emitter->stream());
|
|
|
|
$promise = $listener->advance();
|
|
$promise->onResolve(function ($exception, $value) use (&$reason) {
|
|
$reason = $exception;
|
|
});
|
|
|
|
$emitter->fail($exception);
|
|
|
|
$this->assertSame($exception, $reason);
|
|
}
|
|
}
|