1
0
mirror of https://github.com/danog/amp.git synced 2025-01-23 05:41:25 +01:00
amp/src/Test.php

875 lines
22 KiB
PHP
Raw Normal View History

2016-05-27 03:25:55 +02:00
<?php
namespace Interop\Async\Loop;
if (!defined("SIGUSR1")) {
define("SIGUSR1", 10);
}
abstract class Test extends \PHPUnit_Framework_TestCase {
/**
* The DriverFactory to run this test on
*
* @return DriverFactory
*/
abstract function getFactory();
/** @var Driver */
public $loop;
function setUp()
{
2016-05-27 03:25:55 +02:00
$this->loop = $this->getFactory()->create();
if (!$this->loop instanceof Driver) {
$this->fail("Factory did not return a loop Driver");
}
}
function start($cb)
{
2016-05-27 03:25:55 +02:00
$cb($this->loop);
$this->loop->run();
}
function testEmptyLoop()
{
2016-05-27 03:25:55 +02:00
$this->loop->run();
}
function testStopWorksEvenIfNotCurrentlyRunning()
{
2016-05-27 03:25:55 +02:00
$this->loop->stop();
}
// Note: The running nesting is important for being able to continue actually still running loops (i.e. running flag set, if the driver has one) inside register_shutdown_function() for example
function testLoopRunsCanBeConsecutiveAndNested()
{
2016-05-27 03:25:55 +02:00
$this->expectOutputString("123456");
$this->start(function (Driver $loop) {
$loop->defer(function() {
echo 1;
});
$loop->run();
$loop->defer(function() use ($loop) {
$loop->run();
echo 4;
$loop->defer(function() use ($loop) {
echo 6;
$loop->stop();
$loop->defer(function() {
$this->fail("A loop stopped at all levels must not execute further deferreds");
});
});
});
$loop->defer(function() use ($loop) {
echo 2;
$loop->stop();
$loop->defer(function() use ($loop) {
echo 5;
});
});
$loop->defer(function() use ($loop) {
echo 3;
});
});
}
function testSignalCapability()
{
2016-05-27 03:25:55 +02:00
try {
$watcher = $this->loop->onSignal(SIGUSR1, function() {});
$this->loop->cancel($watcher);
} catch (UnsupportedFeatureException $e) {
2016-05-27 03:25:55 +02:00
$this->markTestSkipped("The loop is not capable of handling signals properly. Skipping.");
}
}
function testWatcherUnrefRerefRunResult()
{
2016-05-27 03:25:55 +02:00
$invoked = false;
$this->start(function(Driver $loop) use (&$invoked) {
$watcher = $loop->defer(function() use (&$invoked) {
$invoked = true;
});
$loop->unreference($watcher);
$loop->reference($watcher);
$loop->reference($watcher);
});
$this->assertTrue($invoked);
}
function testDeferWatcherUnrefRunResult()
{
$this->start(function(Driver $loop) {
$watcher = $loop->defer(function() {
$this->fail("Unreferenced defer watcher should not keep loop running");
2016-05-27 03:25:55 +02:00
});
$loop->unreference($watcher);
});
}
function testOnceWatcherUnrefRunResult()
{
2016-05-27 03:25:55 +02:00
$invoked = false;
$this->start(function(Driver $loop) use (&$invoked) {
$watcher = $loop->delay(2000, function() use (&$invoked) {
$invoked = true;
});
$loop->unreference($watcher);
$loop->unreference($watcher);
});
$this->assertFalse($invoked);
}
function testRepeatWatcherUnrefRunResult()
{
2016-05-27 03:25:55 +02:00
$invoked = false;
$this->start(function(Driver $loop) use (&$invoked) {
$watcher = $loop->repeat(2000, function() use (&$invoked) {
$invoked = true;
});
$loop->unreference($watcher);
});
$this->assertFalse($invoked);
}
function testOnReadableWatcherUnrefRunResult()
{
2016-05-27 03:25:55 +02:00
$this->start(function(Driver $loop) {
$watcher = $loop->onReadable(STDIN, function() {
// empty
});
$loop->unreference($watcher);
});
}
function testOnWritableWatcherKeepAliveRunResult()
{
2016-05-27 03:25:55 +02:00
$this->start(function(Driver $loop) {
$watcher = $loop->onWritable(STDOUT, function() {
// empty
});
$loop->unreference($watcher);
});
}
/** @depends testSignalCapability */
function testOnSignalWatcherKeepAliveRunResult()
{
2016-05-27 03:25:55 +02:00
$this->start(function(Driver $loop) {
$watcher = $loop->onSignal(SIGUSR1, function() {
// empty
});
$loop->unreference($watcher);
});
}
function testDisabledDeferReenableInSubsequentTick()
{
$this->expectOutputString("123");
2016-05-27 03:25:55 +02:00
$this->start(function(Driver $loop) {
$watcherId = $loop->defer(function ($watcherId) {
echo 3;
2016-05-27 03:25:55 +02:00
});
$loop->disable($watcherId);
$loop->defer(function () use ($loop, $watcherId) {
2016-05-27 03:25:55 +02:00
$loop->enable($watcherId);
echo 2;
});
echo 1;
2016-05-27 03:25:55 +02:00
});
}
function provideRegistrationArgs()
{
2016-05-27 03:25:55 +02:00
$args = [
["defer", [function() {}]],
["delay", [5, function() {}]],
["repeat", [5, function() {}]],
2016-05-27 03:25:55 +02:00
["onWritable", [\STDOUT, function() {}]],
["onReadable", [\STDIN, function() {}]],
["onSignal", [\SIGUSR1, function() {}]],
];
return $args;
}
/**
* @dataProvider provideRegistrationArgs
*/
function testDisableWithConsecutiveCancel($type, $args)
{
2016-05-27 03:25:55 +02:00
if ($type === "onSignal") {
$this->testSignalCapability();
}
$this->start(function(Driver $loop) use ($type, $args) {
$func = [$loop, $type];
$watcherId = \call_user_func_array($func, $args);
$loop->disable($watcherId);
2016-05-27 03:25:55 +02:00
$loop->defer(function() use ($loop, $watcherId) {
$loop->cancel($watcherId);
});
});
}
/**
* @dataProvider provideRegistrationArgs
*/
function testWatcherReferenceInfo($type, $args)
{
2016-05-27 03:25:55 +02:00
if ($type === "onSignal") {
$this->testSignalCapability();
}
$loop = $this->loop;
$func = [$loop, $type];
if (substr($type, 0, 2) === "on") {
$type = "on_" . lcfirst(substr($type, 2));
}
// being referenced is the default
$watcherId1 = \call_user_func_array($func, $args);
$info = $loop->info();
$expected = ["enabled" => 1, "disabled" => 0];
$this->assertSame($expected, $info[$type]);
$expected = ["referenced" => 1, "unreferenced" => 0];
$this->assertSame($expected, $info["watchers"]);
// explicitly reference() even though it's the default setting
$argsCopy = $args;
$watcherId2 = \call_user_func_array($func, $argsCopy);
$loop->reference($watcherId2);
$loop->reference($watcherId2);
$info = $loop->info();
$expected = ["enabled" => 2, "disabled" => 0];
$this->assertSame($expected, $info[$type]);
$expected = ["referenced" => 2, "unreferenced" => 0];
$this->assertSame($expected, $info["watchers"]);
// disabling a referenced watcher should decrement the referenced count
$loop->disable($watcherId2);
$loop->disable($watcherId2);
$loop->disable($watcherId2);
$info = $loop->info();
$expected = ["referenced" => 1, "unreferenced" => 0];
$this->assertSame($expected, $info["watchers"]);
// enabling a referenced watcher should increment the referenced count
$loop->enable($watcherId2);
$loop->enable($watcherId2);
$info = $loop->info();
$expected = ["referenced" => 2, "unreferenced" => 0];
$this->assertSame($expected, $info["watchers"]);
// cancelling an referenced watcher should decrement the referenced count
$loop->cancel($watcherId2);
$info = $loop->info();
$expected = ["referenced" => 1, "unreferenced" => 0];
$this->assertSame($expected, $info["watchers"]);
// unreference() should just increment unreferenced count
$watcherId2 = \call_user_func_array($func, $args);
$loop->unreference($watcherId2);
$loop->unreference($watcherId2);
$loop->unreference($watcherId2);
$info = $loop->info();
$expected = ["enabled" => 2, "disabled" => 0];
$this->assertSame($expected, $info[$type]);
$expected = ["referenced" => 1, "unreferenced" => 1];
$this->assertSame($expected, $info["watchers"]);
$loop->cancel($watcherId1);
$loop->cancel($watcherId2);
}
/**
* @dataProvider provideRegistrationArgs
*/
function testWatcherRegistrationAndCancellationInfo($type, $args)
{
2016-05-27 03:25:55 +02:00
if ($type === "onSignal") {
$this->testSignalCapability();
}
$loop = $this->loop;
$func = [$loop, $type];
if (substr($type, 0, 2) === "on") {
$type = "on_" . lcfirst(substr($type, 2));
}
$watcherId = \call_user_func_array($func, $args);
$this->assertInternalType("string", $watcherId);
$info = $loop->info();
$expected = ["enabled" => 1, "disabled" => 0];
$this->assertSame($expected, $info[$type]);
// invoke enable() on active watcher to ensure it has no side-effects
$loop->enable($watcherId);
$info = $loop->info();
$expected = ["enabled" => 1, "disabled" => 0];
$this->assertSame($expected, $info[$type]);
// invoke disable() twice to ensure it has no side-effects
$loop->disable($watcherId);
$loop->disable($watcherId);
$info = $loop->info();
$expected = ["enabled" => 0, "disabled" => 1];
$this->assertSame($expected, $info[$type]);
$loop->cancel($watcherId);
$info = $loop->info();
$expected = ["enabled" => 0, "disabled" => 0];
$this->assertSame($expected, $info[$type]);
$watcherId = \call_user_func_array($func, $args);
$info = $loop->info();
$expected = ["enabled" => 1, "disabled" => 0];
$this->assertSame($expected, $info[$type]);
$loop->disable($watcherId);
$info = $loop->info();
$expected = ["enabled" => 0, "disabled" => 1];
$this->assertSame($expected, $info[$type]);
$loop->enable($watcherId);
$info = $loop->info();
$expected = ["enabled" => 1, "disabled" => 0];
$this->assertSame($expected, $info[$type]);
$loop->cancel($watcherId);
$info = $loop->info();
$expected = ["enabled" => 0, "disabled" => 0];
$this->assertSame($expected, $info[$type]);
}
/**
* @dataProvider provideRegistrationArgs
*/
function testNoMemoryLeak($type, $args)
{
2016-05-27 03:25:55 +02:00
if ($type === "onSignal") {
$this->testSignalCapability();
}
2016-05-27 03:25:55 +02:00
$this->start(function(Driver $loop) use ($type, $args) {
$initialMem = memory_get_usage();
$cb = function ($runs) use ($loop, $type, $args) {
$func = [$loop, $type];
for ($watchers = [], $i = 0; $i < $runs; $i++) {
$watchers[] = \call_user_func_array($func, $args);
}
foreach ($watchers as $watcher) {
$loop->cancel($watcher);
}
for ($watchers = [], $i = 0; $i < $runs; $i++) {
$watchers[] = \call_user_func_array($func, $args);
}
foreach ($watchers as $watcher) {
$loop->disable($watcher);
$loop->cancel($watcher);
}
for ($watchers = [], $i = 0; $i < $runs; $i++) {
$watchers[] = \call_user_func_array($func, $args);
}
if ($type === "repeat") {
2016-05-27 03:25:55 +02:00
$loop->delay($msInterval = 7, function () use ($loop, $watchers) {
foreach ($watchers as $watcher) {
$loop->cancel($watcher);
}
});
} elseif ($type !== "defer" && $type !== "delay") {
2016-05-27 03:25:55 +02:00
$loop->defer(function () use ($loop, $watchers) {
foreach ($watchers as $watcher) {
$loop->cancel($watcher);
}
});
}
$loop->run();
if ($type === "defer") {
2016-05-27 03:25:55 +02:00
$loop->defer($fn = function ($watcherId, $i) use (&$fn, $loop) {
if ($i) {
$loop->defer($fn, --$i);
}
}, $runs);
$loop->run();
}
if ($type === "delay") {
2016-05-27 03:25:55 +02:00
$loop->delay($msDelay = 0, $fn = function ($watcherId, $i) use (&$fn, $loop) {
if ($i) {
$loop->delay($msDelay = 0, $fn, --$i);
}
}, $runs);
$loop->run();
}
if ($type === "repeat") {
2016-05-27 03:25:55 +02:00
$loop->repeat($msDelay = 0, $fn = function ($watcherId, $i) use (&$fn, $loop) {
$loop->cancel($watcherId);
if ($i) {
$loop->repeat($msDelay = 0, $fn, --$i);
}
}, $runs);
$loop->run();
}
if ($type === "onWritable") {
2016-05-27 03:25:55 +02:00
$loop->defer(function($watcherId, $runs) use ($loop) {
$fn = function ($watcherId, $socket, $i) use (&$fn, $loop) {
$loop->cancel($watcherId);
if ($socket) {
fwrite($socket, ".");
}
if ($i) {
// explicitly use *different* streams with *different* resource ids
$ends = stream_socket_pair(\stripos(PHP_OS, "win") === 0 ? STREAM_PF_INET : STREAM_PF_UNIX, STREAM_SOCK_STREAM, STREAM_IPPROTO_IP);
$loop->onWritable($ends[0], $fn, --$i);
$loop->onReadable($ends[1], function($watcherId) use ($loop) {
$loop->cancel($watcherId);
});
}
};
$fn($watcherId, null, $runs);
}, $runs + 1);
$loop->run();
}
if ($type === "onSignal") {
2016-05-27 03:25:55 +02:00
$watchers = [$loop->repeat(\SIGUSR1, $fn = function ($watcherId, $i) use (&$fn, $loop, &$watchers) {
if ($i) {
$watchers[] = $loop->onSignal(\SIGUSR1, $fn, --$i);
} else {
foreach ($watchers as $watcher) {
$loop->cancel($watcher);
}
}
}, $runs)];
$loop->run();
}
};
$closureMem = memory_get_usage() - $initialMem;
$cb(500); /* just to set up eventual structures inside loop without counting towards memory comparison */
gc_collect_cycles();
$initialMem = memory_get_usage() - $closureMem;
$cb(10000);
unset($cb);
2016-05-27 03:25:55 +02:00
gc_collect_cycles();
$endMem = memory_get_usage();
2016-05-27 03:25:55 +02:00
/* this is allowing some memory usage due to runtime caches etc., but nothing actually leaking */
$this->assertLessThan(40000, $endMem - $initialMem); // 4 * 10000, as 4 is minimal sizeof(void *)
2016-05-27 03:25:55 +02:00
});
}
/**
* @expectedException \LogicException
*/
function testSuccessOnEnableNonexistentWatcher()
{
2016-05-27 03:25:55 +02:00
$this->loop->enable("nonexistentWatcher");
}
function testSuccessOnDisableNonexistentWatcher()
{
2016-05-27 03:25:55 +02:00
$this->loop->disable("nonexistentWatcher");
}
function testSuccessOnCancelNonexistentWatcher()
{
2016-05-27 03:25:55 +02:00
$this->loop->cancel("nonexistentWatcher");
}
function testEnablingWatcherAllowsSubsequentInvocation()
{
2016-05-27 03:25:55 +02:00
$loop = $this->loop;
$increment = 0;
$watcherId = $loop->defer(function() use (&$increment) {
$increment++;
});
$loop->disable($watcherId);
$loop->delay($msDelay = 5, [$loop, "stop"]);
$loop->run();
$this->assertEquals(0, $increment);
$loop->enable($watcherId);
$loop->delay($msDelay = 5, [$loop, "stop"]);
$loop->run();
$this->assertEquals(1, $increment);
}
function testUnresolvedEventsAreReenabledOnRunFollowingPreviousStop()
{
2016-05-27 03:25:55 +02:00
$increment = 0;
$this->start(function(Driver $loop) use (&$increment) {
$loop->defer([$loop, "stop"]);
$loop->run();
$loop->defer(function () use (&$increment, $loop) {
$loop->delay($msDelay = 100, function () use ($loop, &$increment) {
$increment++;
$loop->stop();
});
});
2016-05-27 03:25:55 +02:00
$this->assertEquals(0, $increment);
\usleep(5000);
});
$this->assertEquals(1, $increment);
}
function testTimerWatcherParameterOrder()
{
2016-05-27 03:25:55 +02:00
$this->start(function(Driver $loop) {
$counter = 0;
$loop->defer(function ($watcherId) use ($loop, &$counter) {
$this->assertInternalType("string", $watcherId);
if (++$counter === 3) {
$loop->stop();
}
});
$loop->delay($msDelay = 5, function ($watcherId) use ($loop, &$counter) {
$this->assertInternalType("string", $watcherId);
if (++$counter === 3) {
$loop->stop();
}
});
$loop->repeat($msDelay = 5, function ($watcherId) use ($loop, &$counter) {
$this->assertInternalType("string", $watcherId);
$loop->cancel($watcherId);
if (++$counter === 3) {
$loop->stop();
}
});
});
}
function testStreamWatcherParameterOrder()
{
$this->start(function(Driver $loop) use (&$invoked) {
2016-05-27 03:25:55 +02:00
$invoked = 0;
$loop->onWritable(STDOUT, function ($watcherId, $stream) use ($loop, &$invoked) {
$this->assertInternalType("string", $watcherId);
$this->assertSame(STDOUT, $stream);
$invoked++;
$loop->cancel($watcherId);
});
});
$this->assertSame(1, $invoked);
2016-05-27 03:25:55 +02:00
}
function testDisablingWatcherPreventsSubsequentInvocation()
{
2016-05-27 03:25:55 +02:00
$this->start(function(Driver $loop) {
$increment = 0;
$watcherId = $loop->defer(function () use (&$increment) {
$increment++;
});
$loop->disable($watcherId);
$loop->delay($msDelay = 5, [$loop, "stop"]);
$this->assertEquals(0, $increment);
});
}
function testImmediateExecution()
{
2016-05-27 03:25:55 +02:00
$loop = $this->loop;
$increment = 0;
$this->start(function(Driver $loop) use (&$increment) {
$loop->defer(function () use (&$increment) {
$increment++;
});
$loop->defer([$loop, "stop"]);
});
$this->assertEquals(1, $increment);
}
function testImmediatelyCallbacksDoNotRecurseInSameTick()
{
2016-05-27 03:25:55 +02:00
$increment = 0;
$this->start(function(Driver $loop) use (&$increment) {
$loop->defer(function () use ($loop, &$increment) {
$increment++;
$loop->defer(function () use (&$increment) {
$increment++;
});
});
$loop->defer([$loop, "stop"]);
});
$this->assertEquals(1, $increment);
}
function testRunExecutesEventsUntilExplicitlyStopped()
{
2016-05-27 03:25:55 +02:00
$increment = 0;
$this->start(function(Driver $loop) use (&$increment) {
$loop->repeat($msInterval = 5, function ($watcherId) use ($loop, &$increment) {
$increment++;
if ($increment === 10) {
$loop->cancel($watcherId);
}
});
});
$this->assertEquals(10, $increment);
}
/**
* @expectedException \Exception
* @expectedExceptionMessage loop error
*/
function testLoopAllowsExceptionToBubbleUpDuringStart()
{
2016-05-27 03:25:55 +02:00
$this->start(function(Driver $loop) {
$loop->defer(function() {
throw new \Exception("loop error");
});
});
}
/**
* @expectedException \RuntimeException
* @expectedExceptionMessage test
*/
function testLoopAllowsExceptionToBubbleUpFromRepeatingAlarmDuringStart()
{
2016-05-27 03:25:55 +02:00
$this->start(function(Driver $loop) {
$loop->repeat($msInterval = 1, function () {
2016-05-27 03:25:55 +02:00
throw new \RuntimeException("test");
});
});
}
function testErrorHandlerCapturesUncaughtException()
{
2016-05-27 03:25:55 +02:00
$msg = "";
$this->loop->setErrorHandler(function(\Exception $error) use (&$msg) {
$msg = $error->getMessage();
});
$this->start(function(Driver $loop) {
$loop->defer(function() {
throw new \Exception("loop error");
});
});
$this->assertSame("loop error", $msg);
}
/**
* @expectedException \Exception
* @expectedExceptionMessage errorception
*/
function testOnErrorFailure()
{
2016-05-27 03:25:55 +02:00
$this->loop->setErrorHandler(function() {
throw new \Exception("errorception");
});
$this->start(function(Driver $loop) {
$loop->delay($msDelay = 5, function() {
throw new \Exception("error");
});
});
}
/**
* @expectedException \RuntimeException
* @expectedExceptionMessage test
*/
function testLoopException()
{
2016-05-27 04:17:30 +02:00
$this->start(function(Driver $loop) {
2016-05-27 03:25:55 +02:00
$loop->defer(function() use ($loop) {
// force next tick, outside of primary startup tick
$loop->defer(function() {
throw new \RuntimeException("test");
});
});
});
}
/**
* @depends testSignalCapability
*/
function testOnSignalWatcher()
{
2016-05-27 03:25:55 +02:00
if (!\extension_loaded("posix")) {
$this->markTestSkipped("ext/posix required to test signal handlers");
2016-05-27 03:25:55 +02:00
}
$this->expectOutputString("caught SIGUSR1");
$this->start(function(Driver $loop) {
$loop->delay($msDelay = 1, function() use ($loop) {
\posix_kill(\getmypid(), \SIGUSR1);
$loop->delay($msDelay = 10, function() use ($loop) {
$loop->stop();
});
});
$loop->onSignal(SIGUSR1, function($watcherId) use ($loop) {
$loop->cancel($watcherId);
echo "caught SIGUSR1";
});
});
}
function testInitiallyDisabledOnSignalWatcher()
{
2016-05-27 03:25:55 +02:00
if (!\extension_loaded("posix")) {
2016-05-27 04:17:30 +02:00
$this->markTestSkipped("ext/posix required to test signal handlers");
2016-05-27 03:25:55 +02:00
}
$this->expectOutputString("caught SIGUSR1");
$this->start(function(Driver $loop) {
$sigWatcherId = $loop->onSignal(SIGUSR1, function() use ($loop) {
echo "caught SIGUSR1";
$loop->stop();
}, $options = ["enable" => false]);
$loop->delay($msDelay = 10, function() use ($loop, $sigWatcherId) {
$loop->enable($sigWatcherId);
$loop->delay($msDelay = 10, function() use ($sigWatcherId) {
\posix_kill(\getmypid(), \SIGUSR1);
});
});
});
}
function testCancelRemovesWatcher()
{
2016-05-27 03:25:55 +02:00
$this->start(function(Driver $loop) {
$watcherId = $loop->delay($msDelay = 10, function () {
$this->fail('Watcher was not cancelled as expected');
});
$loop->defer(function () use ($loop, $watcherId) {
$loop->cancel($watcherId);
});
$loop->delay($msDelay = 5, [$loop, "stop"]);
});
}
function testOnWritableWatcher()
{
2016-05-27 03:25:55 +02:00
$flag = false;
$this->start(function(Driver $loop) use (&$flag) {
2016-05-27 03:25:55 +02:00
$loop->onWritable(STDOUT, function () use ($loop, &$flag) {
$flag = true;
$loop->stop();
});
$loop->delay($msDelay = 5, [$loop, "stop"]);
});
$this->assertTrue($flag);
}
function testInitiallyDisabledWriteWatcher()
{
2016-05-27 03:25:55 +02:00
$increment = 0;
$this->start(function(Driver $loop) {
$watcherId = $loop->onWritable(STDOUT, function () use (&$increment) {
$increment++;
});
$loop->disable($watcherId);
$loop->delay($msDelay = 5, [$loop, "stop"]);
});
$this->assertSame(0, $increment);
}
function testInitiallyDisabledWriteWatcherIsTriggeredOnceEnabled()
{
2016-05-27 03:25:55 +02:00
$this->expectOutputString("12");
$this->start(function (Driver $loop) {
$watcherId = $loop->onWritable(STDOUT, function () use ($loop) {
echo 2;
$loop->stop();
});
$loop->disable($watcherId);
$loop->defer(function () use ($loop, $watcherId) {
$loop->enable($watcherId);
echo 1;
});
});
}
/**
* @expectedException \RuntimeException
*/
function testStreamWatcherDoesntSwallowExceptions()
{
2016-05-27 03:25:55 +02:00
$this->start(function(Driver $loop) {
$loop->onWritable(STDOUT, function () {
throw new \RuntimeException;
});
$loop->delay($msDelay = 5, [$loop, "stop"]);
});
}
function testReactorRunsUntilNoWatchersRemain()
{
2016-05-27 03:25:55 +02:00
$var1 = $var2 = 0;
$this->start(function(Driver $loop) use (&$var1, &$var2) {
$loop->repeat($msDelay = 1, function ($watcherId) use ($loop, &$var1) {
2016-05-27 03:25:55 +02:00
if (++$var1 === 3) {
$loop->cancel($watcherId);
}
});
$loop->onWritable(STDOUT, function ($watcherId) use ($loop, &$var2) {
if (++$var2 === 4) {
$loop->cancel($watcherId);
}
});
});
$this->assertSame(3, $var1);
$this->assertSame(4, $var2);
}
function testReactorRunsUntilNoWatchersRemainWhenStartedDeferred()
{
2016-05-27 03:25:55 +02:00
$var1 = $var2 = 0;
$this->start(function(Driver $loop) use (&$var1, &$var2) {
$loop->defer(function() use ($loop, &$var1, &$var2) {
$loop->repeat($msDelay = 1, function ($watcherId) use ($loop, &$var1) {
2016-05-27 03:25:55 +02:00
if (++$var1 === 3) {
$loop->cancel($watcherId);
}
});
$loop->onWritable(STDOUT, function ($watcherId) use ($loop, &$var2) {
if (++$var2 === 4) {
$loop->cancel($watcherId);
}
});
});
});
$this->assertSame(3, $var1);
$this->assertSame(4, $var2);
}
function testOptionalCallbackDataPassedOnInvocation()
{
2016-05-27 03:25:55 +02:00
$callbackData = new \StdClass;
$this->start(function(Driver $loop) use ($callbackData) {
$loop->defer(function ($watcherId, $callbackData) use ($loop) {
$callbackData->defer = true;
}, $callbackData);
$loop->delay($msDelay = 1, function ($watcherId, $callbackData) use ($loop) {
$callbackData->delay = true;
}, $callbackData);
$loop->repeat($msDelay = 1, function ($watcherId, $callbackData) use ($loop) {
$callbackData->repeat = true;
$loop->cancel($watcherId);
}, $callbackData);
$loop->onWritable(STDERR, function ($watcherId, $stream, $callbackData) use ($loop) {
$callbackData->onWritable = true;
$loop->cancel($watcherId);
}, $callbackData);
});
$this->assertTrue($callbackData->defer);
$this->assertTrue($callbackData->delay);
$this->assertTrue($callbackData->repeat);
$this->assertTrue($callbackData->onWritable);
}
}