2016-12-29 21:09:49 +01:00
|
|
|
<?php
|
2016-08-16 23:39:25 +02:00
|
|
|
|
2016-08-17 06:27:10 +02:00
|
|
|
namespace Amp\Test;
|
2016-07-12 18:20:06 +02:00
|
|
|
|
2017-03-10 21:31:57 +01:00
|
|
|
use Amp\Coroutine;
|
|
|
|
use Amp\Failure;
|
|
|
|
use Amp\InvalidYieldError;
|
|
|
|
use Amp\Loop;
|
|
|
|
use Amp\Pause;
|
2017-03-10 21:58:46 +01:00
|
|
|
use Amp\Promise;
|
2017-04-23 14:39:19 +02:00
|
|
|
use Amp\Success;
|
2017-03-14 22:15:36 +01:00
|
|
|
use PHPUnit\Framework\TestCase;
|
|
|
|
use React\Promise\FulfilledPromise as FulfilledReactPromise;
|
2017-02-20 21:53:58 +01:00
|
|
|
use React\Promise\Promise as ReactPromise;
|
2016-07-12 18:20:06 +02:00
|
|
|
|
2017-03-14 22:15:36 +01:00
|
|
|
class CoroutineTest extends TestCase {
|
2016-08-16 23:39:25 +02:00
|
|
|
const TIMEOUT = 100;
|
2016-07-12 18:20:06 +02:00
|
|
|
|
2016-11-14 20:59:21 +01:00
|
|
|
public function testYieldSuccessfulPromise() {
|
2016-07-12 18:20:06 +02:00
|
|
|
$value = 1;
|
|
|
|
|
|
|
|
$generator = function () use (&$yielded, $value) {
|
2017-01-07 22:55:34 +01:00
|
|
|
$yielded = yield new Success($value);
|
2016-07-12 18:20:06 +02:00
|
|
|
};
|
|
|
|
|
2017-01-07 22:55:34 +01:00
|
|
|
new Coroutine($generator());
|
2016-07-12 18:20:06 +02:00
|
|
|
|
|
|
|
$this->assertSame($value, $yielded);
|
|
|
|
}
|
|
|
|
|
2016-11-14 20:59:21 +01:00
|
|
|
public function testYieldFailedPromise() {
|
2016-07-12 18:20:06 +02:00
|
|
|
$exception = new \Exception;
|
|
|
|
|
|
|
|
$generator = function () use (&$yielded, $exception) {
|
2017-01-07 22:55:34 +01:00
|
|
|
$yielded = yield new Failure($exception);
|
2016-07-12 18:20:06 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
$coroutine = new Coroutine($generator());
|
|
|
|
|
|
|
|
$this->assertNull($yielded);
|
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$coroutine->onResolve(function ($exception) use (&$reason) {
|
2016-07-12 18:20:06 +02:00
|
|
|
$reason = $exception;
|
|
|
|
});
|
|
|
|
|
|
|
|
$this->assertSame($exception, $reason);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2016-11-14 20:59:21 +01:00
|
|
|
* @depends testYieldSuccessfulPromise
|
2016-07-12 18:20:06 +02:00
|
|
|
*/
|
2016-11-14 20:59:21 +01:00
|
|
|
public function testYieldPendingPromise() {
|
2016-07-12 18:20:06 +02:00
|
|
|
$value = 1;
|
|
|
|
|
2017-03-10 21:31:57 +01:00
|
|
|
Loop::run(function () use (&$yielded, $value) {
|
2016-07-12 18:20:06 +02:00
|
|
|
$generator = function () use (&$yielded, $value) {
|
2017-01-07 22:55:34 +01:00
|
|
|
$yielded = yield new Pause(self::TIMEOUT, $value);
|
2016-07-12 18:20:06 +02:00
|
|
|
};
|
|
|
|
|
2017-01-07 22:55:34 +01:00
|
|
|
new Coroutine($generator());
|
2016-07-12 18:20:06 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
$this->assertSame($value, $yielded);
|
|
|
|
}
|
|
|
|
|
2017-03-12 19:38:36 +01:00
|
|
|
public function testYieldPromiseArray() {
|
|
|
|
Loop::run(function () {
|
|
|
|
$value = 1;
|
|
|
|
|
|
|
|
$generator = function () use (&$yielded, $value) {
|
|
|
|
list($yielded) = yield [
|
|
|
|
new Success($value)
|
|
|
|
];
|
|
|
|
};
|
|
|
|
|
|
|
|
yield new Coroutine($generator());
|
|
|
|
|
|
|
|
$this->assertSame($value, $yielded);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testYieldNonPromiseArray() {
|
|
|
|
$this->expectException(InvalidYieldError::class);
|
|
|
|
|
|
|
|
Loop::run(function () {
|
|
|
|
$value = 1;
|
|
|
|
|
|
|
|
$generator = function () use (&$yielded, $value) {
|
|
|
|
list($yielded) = yield [
|
|
|
|
$value
|
|
|
|
];
|
|
|
|
};
|
|
|
|
|
|
|
|
yield new Coroutine($generator());
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-03-14 22:32:14 +01:00
|
|
|
public function testYieldPromiseArrayAfterPendingPromise() {
|
|
|
|
Loop::run(function () {
|
|
|
|
$value = 1;
|
|
|
|
|
|
|
|
$generator = function () use (&$yielded, $value) {
|
|
|
|
yield new Pause(10);
|
|
|
|
list($yielded) = yield [
|
|
|
|
new Success($value)
|
|
|
|
];
|
|
|
|
};
|
|
|
|
|
|
|
|
yield new Coroutine($generator());
|
|
|
|
|
|
|
|
$this->assertSame($value, $yielded);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testYieldNonPromiseArrayAfterPendingPromise() {
|
|
|
|
$this->expectException(InvalidYieldError::class);
|
|
|
|
|
|
|
|
Loop::run(function () {
|
|
|
|
$value = 1;
|
|
|
|
|
|
|
|
$generator = function () use (&$yielded, $value) {
|
|
|
|
yield new Pause(10);
|
|
|
|
list($yielded) = yield [
|
|
|
|
$value
|
|
|
|
];
|
|
|
|
};
|
|
|
|
|
|
|
|
yield new Coroutine($generator());
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2016-07-12 18:20:06 +02:00
|
|
|
/**
|
2016-11-14 20:59:21 +01:00
|
|
|
* @depends testYieldFailedPromise
|
2016-07-12 18:20:06 +02:00
|
|
|
*/
|
2016-11-14 20:59:21 +01:00
|
|
|
public function testCatchingFailedPromiseException() {
|
2016-07-12 18:20:06 +02:00
|
|
|
$exception = new \Exception;
|
|
|
|
|
|
|
|
$fail = false;
|
|
|
|
$generator = function () use (&$fail, &$result, $exception) {
|
|
|
|
try {
|
|
|
|
yield new Failure($exception);
|
|
|
|
} catch (\Exception $exception) {
|
|
|
|
$result = $exception;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$fail = true;
|
|
|
|
};
|
|
|
|
|
2017-01-07 22:55:34 +01:00
|
|
|
new Coroutine($generator());
|
2016-07-12 18:20:06 +02:00
|
|
|
|
2017-03-14 00:52:57 +01:00
|
|
|
$this->assertFalse($fail);
|
2016-07-12 18:20:06 +02:00
|
|
|
}
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-07-12 18:20:06 +02:00
|
|
|
public function testInvalidYield() {
|
|
|
|
$generator = function () {
|
|
|
|
yield 1;
|
|
|
|
};
|
|
|
|
|
|
|
|
$coroutine = new Coroutine($generator());
|
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$coroutine->onResolve(function ($exception) use (&$reason) {
|
2016-07-12 18:20:06 +02:00
|
|
|
$reason = $exception;
|
|
|
|
});
|
|
|
|
|
2016-08-11 21:35:58 +02:00
|
|
|
$this->assertInstanceOf(InvalidYieldError::class, $reason);
|
2016-07-12 18:20:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @depends testInvalidYield
|
|
|
|
*/
|
2016-11-14 20:59:21 +01:00
|
|
|
public function testInvalidYieldAfterYieldPromise() {
|
2016-07-12 18:20:06 +02:00
|
|
|
$generator = function () {
|
|
|
|
yield new Success;
|
|
|
|
yield 1;
|
|
|
|
};
|
|
|
|
|
|
|
|
$coroutine = new Coroutine($generator());
|
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$coroutine->onResolve(function ($exception) use (&$reason) {
|
2016-07-12 18:20:06 +02:00
|
|
|
$reason = $exception;
|
|
|
|
});
|
|
|
|
|
2016-08-11 21:35:58 +02:00
|
|
|
$this->assertInstanceOf(InvalidYieldError::class, $reason);
|
2016-07-12 18:20:06 +02:00
|
|
|
}
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-12-17 15:16:17 +01:00
|
|
|
/**
|
|
|
|
* @depends testInvalidYield
|
|
|
|
*/
|
|
|
|
public function testInvalidYieldCatchingThrownException() {
|
|
|
|
$generator = function () {
|
|
|
|
try {
|
|
|
|
yield 1;
|
|
|
|
} catch (\Error $exception) {
|
|
|
|
// No further yields.
|
|
|
|
}
|
|
|
|
};
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-12-17 15:16:17 +01:00
|
|
|
$coroutine = new Coroutine($generator());
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$coroutine->onResolve(function ($exception) use (&$reason) {
|
2016-12-17 15:16:17 +01:00
|
|
|
$reason = $exception;
|
|
|
|
});
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-12-17 15:16:17 +01:00
|
|
|
$this->assertInstanceOf(InvalidYieldError::class, $reason);
|
|
|
|
}
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-12-17 15:16:17 +01:00
|
|
|
/**
|
|
|
|
* @depends testInvalidYieldCatchingThrownException
|
|
|
|
*/
|
|
|
|
public function testInvalidYieldCatchingThrownExceptionAndYieldingAgain() {
|
|
|
|
$generator = function () {
|
|
|
|
try {
|
|
|
|
yield 1;
|
|
|
|
} catch (\Error $exception) {
|
|
|
|
yield new Success;
|
|
|
|
}
|
|
|
|
};
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-12-17 15:16:17 +01:00
|
|
|
$coroutine = new Coroutine($generator());
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$coroutine->onResolve(function ($exception) use (&$reason) {
|
2016-12-17 15:16:17 +01:00
|
|
|
$reason = $exception;
|
|
|
|
});
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-12-17 15:16:17 +01:00
|
|
|
$this->assertInstanceOf(InvalidYieldError::class, $reason);
|
|
|
|
}
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-12-17 15:16:17 +01:00
|
|
|
/**
|
|
|
|
* @depends testInvalidYieldCatchingThrownException
|
|
|
|
*/
|
|
|
|
public function testInvalidYieldCatchingThrownExceptionAndThrowing() {
|
|
|
|
$exception = new \Exception;
|
|
|
|
$generator = function () use ($exception) {
|
|
|
|
try {
|
|
|
|
yield 1;
|
|
|
|
} catch (\Error $error) {
|
|
|
|
throw $exception;
|
|
|
|
}
|
|
|
|
};
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-12-17 15:16:17 +01:00
|
|
|
$coroutine = new Coroutine($generator());
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$coroutine->onResolve(function ($exception) use (&$reason) {
|
2016-12-17 15:16:17 +01:00
|
|
|
$reason = $exception;
|
|
|
|
});
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-12-17 15:16:17 +01:00
|
|
|
$this->assertInstanceOf(InvalidYieldError::class, $reason);
|
|
|
|
$this->assertSame($exception, $reason->getPrevious());
|
|
|
|
}
|
2016-07-12 18:20:06 +02:00
|
|
|
|
|
|
|
/**
|
2016-11-14 20:59:21 +01:00
|
|
|
* @depends testYieldFailedPromise
|
2016-07-12 18:20:06 +02:00
|
|
|
*/
|
2016-11-14 20:59:21 +01:00
|
|
|
public function testCatchingFailedPromiseExceptionWithNoFurtherYields() {
|
2016-07-12 18:20:06 +02:00
|
|
|
$exception = new \Exception;
|
|
|
|
|
|
|
|
$generator = function () use ($exception) {
|
|
|
|
try {
|
|
|
|
yield new Failure($exception);
|
|
|
|
} catch (\Exception $exception) {
|
|
|
|
// No further yields in generator.
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
$coroutine = new Coroutine($generator());
|
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$coroutine->onResolve(function ($exception, $value) use (&$reason, &$result) {
|
2017-01-07 22:55:34 +01:00
|
|
|
$reason = $exception;
|
2016-07-12 18:20:06 +02:00
|
|
|
$result = $value;
|
|
|
|
});
|
|
|
|
|
2017-01-07 22:55:34 +01:00
|
|
|
$this->assertNull($reason);
|
2016-07-12 18:20:06 +02:00
|
|
|
$this->assertNull($result);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testGeneratorThrowingExceptionFailsCoroutine() {
|
|
|
|
$exception = new \Exception;
|
|
|
|
|
|
|
|
$generator = function () use ($exception) {
|
|
|
|
throw $exception;
|
|
|
|
yield;
|
|
|
|
};
|
|
|
|
|
|
|
|
$coroutine = new Coroutine($generator());
|
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$coroutine->onResolve(function ($exception) use (&$reason) {
|
2016-07-12 18:20:06 +02:00
|
|
|
$reason = $exception;
|
|
|
|
});
|
|
|
|
|
|
|
|
$this->assertSame($exception, $reason);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @depends testGeneratorThrowingExceptionFailsCoroutine
|
|
|
|
*/
|
|
|
|
public function testGeneratorThrowingExceptionWithFinallyFailsCoroutine() {
|
|
|
|
$exception = new \Exception;
|
|
|
|
|
|
|
|
$invoked = false;
|
|
|
|
$generator = function () use (&$invoked, $exception) {
|
|
|
|
try {
|
|
|
|
throw $exception;
|
|
|
|
yield;
|
|
|
|
} finally {
|
|
|
|
$invoked = true;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
$coroutine = new Coroutine($generator());
|
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$coroutine->onResolve(function ($exception) use (&$reason) {
|
2016-07-12 18:20:06 +02:00
|
|
|
$reason = $exception;
|
|
|
|
});
|
|
|
|
|
|
|
|
$this->assertSame($exception, $reason);
|
|
|
|
$this->assertTrue($invoked);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2016-11-14 20:59:21 +01:00
|
|
|
* @depends testYieldFailedPromise
|
2016-07-12 18:20:06 +02:00
|
|
|
* @depends testGeneratorThrowingExceptionWithFinallyFailsCoroutine
|
|
|
|
*/
|
2016-11-14 20:59:21 +01:00
|
|
|
public function testGeneratorYieldingFailedPromiseWithFinallyFailsCoroutine() {
|
2016-07-12 18:20:06 +02:00
|
|
|
$exception = new \Exception;
|
|
|
|
|
|
|
|
$invoked = false;
|
|
|
|
$generator = function () use (&$invoked, $exception) {
|
|
|
|
try {
|
|
|
|
yield new Failure($exception);
|
|
|
|
} finally {
|
|
|
|
$invoked = true;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
$coroutine = new Coroutine($generator());
|
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$coroutine->onResolve(function ($exception) use (&$reason) {
|
2016-07-12 18:20:06 +02:00
|
|
|
$reason = $exception;
|
|
|
|
});
|
|
|
|
|
|
|
|
$this->assertSame($exception, $reason);
|
|
|
|
$this->assertTrue($invoked);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @depends testGeneratorThrowingExceptionFailsCoroutine
|
|
|
|
*/
|
2016-11-14 20:59:21 +01:00
|
|
|
public function testGeneratorThrowingExceptionAfterPendingPromiseWithFinallyFailsCoroutine() {
|
2016-07-12 18:20:06 +02:00
|
|
|
$exception = new \Exception;
|
|
|
|
$value = 1;
|
|
|
|
|
2017-03-10 21:31:57 +01:00
|
|
|
Loop::run(function () use (&$yielded, &$invoked, &$reason, $exception, $value) {
|
2016-07-12 18:20:06 +02:00
|
|
|
$invoked = false;
|
|
|
|
$generator = function () use (&$yielded, &$invoked, $exception, $value) {
|
|
|
|
try {
|
|
|
|
$yielded = (yield new Pause(self::TIMEOUT, $value));
|
|
|
|
throw $exception;
|
|
|
|
} finally {
|
|
|
|
$invoked = true;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
$coroutine = new Coroutine($generator());
|
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$coroutine->onResolve(function ($exception) use (&$reason) {
|
2016-07-12 18:20:06 +02:00
|
|
|
$reason = $exception;
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
$this->assertSame($exception, $reason);
|
|
|
|
$this->assertTrue($invoked);
|
|
|
|
$this->assertSame($value, $yielded);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Note that yielding in a finally block is not recommended.
|
|
|
|
*
|
2016-11-14 20:59:21 +01:00
|
|
|
* @depends testYieldPendingPromise
|
2016-07-12 18:20:06 +02:00
|
|
|
* @depends testGeneratorThrowingExceptionWithFinallyFailsCoroutine
|
|
|
|
*/
|
2016-11-14 20:59:21 +01:00
|
|
|
public function testGeneratorThrowingExceptionWithFinallyYieldingPendingPromise() {
|
2016-07-12 18:20:06 +02:00
|
|
|
$exception = new \Exception;
|
|
|
|
$value = 1;
|
|
|
|
|
2017-03-10 21:31:57 +01:00
|
|
|
Loop::run(function () use (&$yielded, &$reason, $exception, $value) {
|
2016-07-12 18:20:06 +02:00
|
|
|
$generator = function () use (&$yielded, $exception, $value) {
|
|
|
|
try {
|
|
|
|
throw $exception;
|
|
|
|
} finally {
|
2016-12-17 15:16:17 +01:00
|
|
|
$yielded = yield new Pause(self::TIMEOUT, $value);
|
2016-07-12 18:20:06 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
$coroutine = new Coroutine($generator());
|
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$coroutine->onResolve(function ($exception) use (&$reason) {
|
2016-07-12 18:20:06 +02:00
|
|
|
$reason = $exception;
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
$this->assertSame($value, $yielded);
|
|
|
|
$this->assertSame($exception, $reason);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2016-11-14 20:59:21 +01:00
|
|
|
* @depends testYieldPendingPromise
|
2016-07-12 18:20:06 +02:00
|
|
|
* @depends testGeneratorThrowingExceptionWithFinallyFailsCoroutine
|
|
|
|
*/
|
|
|
|
public function testGeneratorThrowingExceptionWithFinallyBlockThrowing() {
|
|
|
|
$exception = new \Exception;
|
|
|
|
|
|
|
|
$generator = function () use ($exception) {
|
|
|
|
try {
|
|
|
|
throw new \Exception;
|
|
|
|
} finally {
|
|
|
|
throw $exception;
|
|
|
|
}
|
|
|
|
|
|
|
|
yield; // Unreachable, but makes function a generator.
|
|
|
|
};
|
|
|
|
|
|
|
|
$coroutine = new Coroutine($generator());
|
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$coroutine->onResolve(function ($exception) use (&$reason) {
|
2016-07-12 18:20:06 +02:00
|
|
|
$reason = $exception;
|
|
|
|
});
|
|
|
|
|
|
|
|
$this->assertSame($exception, $reason);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2016-11-14 20:59:21 +01:00
|
|
|
* @depends testYieldSuccessfulPromise
|
2016-07-12 18:20:06 +02:00
|
|
|
*/
|
|
|
|
public function testFastInvalidGenerator() {
|
|
|
|
$generator = function () {
|
|
|
|
if (false) {
|
|
|
|
yield new Success;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
$coroutine = new Coroutine($generator());
|
|
|
|
|
|
|
|
$invoked = false;
|
2017-03-21 17:23:37 +01:00
|
|
|
$coroutine->onResolve(function () use (&$invoked) {
|
2016-07-12 18:20:06 +02:00
|
|
|
$invoked = true;
|
|
|
|
});
|
|
|
|
|
|
|
|
$this->assertTrue($invoked);
|
|
|
|
}
|
|
|
|
|
2017-04-23 14:09:16 +02:00
|
|
|
public function testCreateCallableFunction() {
|
|
|
|
$callable = \Amp\createCallable(function () {
|
2016-07-12 18:20:06 +02:00
|
|
|
yield;
|
|
|
|
});
|
|
|
|
|
|
|
|
$this->assertInstanceOf(Coroutine::class, $callable());
|
|
|
|
}
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-08-12 23:56:03 +02:00
|
|
|
/**
|
2017-04-23 14:09:16 +02:00
|
|
|
* @depends testCreateCallableFunction
|
2016-08-12 23:56:03 +02:00
|
|
|
*/
|
2017-04-23 14:09:16 +02:00
|
|
|
public function testCreateCallableFunctionWithCallbackReturningPromise() {
|
2016-08-12 23:56:03 +02:00
|
|
|
$value = 1;
|
2016-11-14 20:59:21 +01:00
|
|
|
$promise = new Success($value);
|
2017-04-23 14:09:16 +02:00
|
|
|
$callable = \Amp\createCallable(function ($value) {
|
2016-08-12 23:56:03 +02:00
|
|
|
return $value;
|
|
|
|
});
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2017-01-07 22:55:34 +01:00
|
|
|
/** @var Promise $promise */
|
2016-11-14 20:59:21 +01:00
|
|
|
$promise = $callable($promise);
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-11-14 20:59:21 +01:00
|
|
|
$this->assertInstanceOf(Promise::class, $promise);
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$promise->onResolve(function ($exception, $value) use (&$reason, &$result) {
|
2017-01-07 22:55:34 +01:00
|
|
|
$reason = $exception;
|
2016-08-12 23:56:03 +02:00
|
|
|
$result = $value;
|
|
|
|
});
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2017-01-07 22:55:34 +01:00
|
|
|
$this->assertNull($reason);
|
2016-08-12 23:56:03 +02:00
|
|
|
$this->assertSame($value, $result);
|
|
|
|
}
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-07-12 18:20:06 +02:00
|
|
|
/**
|
2017-04-23 14:09:16 +02:00
|
|
|
* @depends testCreateCallableFunction
|
2016-07-12 18:20:06 +02:00
|
|
|
*/
|
2017-04-23 14:09:16 +02:00
|
|
|
public function testCreateCallableFunctionWithNonGeneratorCallback() {
|
2016-08-12 23:56:03 +02:00
|
|
|
$value = 1;
|
2017-04-23 14:09:16 +02:00
|
|
|
$callable = \Amp\createCallable(function ($value) {
|
2016-08-12 23:56:03 +02:00
|
|
|
return $value;
|
|
|
|
});
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2017-01-07 22:55:34 +01:00
|
|
|
/** @var Promise $promise */
|
2016-11-14 20:59:21 +01:00
|
|
|
$promise = $callable($value);
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-11-14 20:59:21 +01:00
|
|
|
$this->assertInstanceOf(Promise::class, $promise);
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$promise->onResolve(function ($exception, $value) use (&$reason, &$result) {
|
2017-01-07 22:55:34 +01:00
|
|
|
$reason = $exception;
|
2016-08-12 23:56:03 +02:00
|
|
|
$result = $value;
|
|
|
|
});
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2017-01-07 22:55:34 +01:00
|
|
|
$this->assertNull($reason);
|
2016-08-12 23:56:03 +02:00
|
|
|
$this->assertSame($value, $result);
|
|
|
|
}
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-08-12 23:56:03 +02:00
|
|
|
/**
|
2017-04-23 14:09:16 +02:00
|
|
|
* @depends testCreateCallableFunction
|
2016-08-12 23:56:03 +02:00
|
|
|
*/
|
2017-04-23 14:09:16 +02:00
|
|
|
public function testCreateCallableFunctionWithThrowingCallback() {
|
2016-08-12 23:56:03 +02:00
|
|
|
$exception = new \Exception;
|
2017-04-23 14:09:16 +02:00
|
|
|
$callable = \Amp\createCallable(function () use ($exception) {
|
2016-08-12 23:56:03 +02:00
|
|
|
throw $exception;
|
|
|
|
});
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2017-01-07 22:55:34 +01:00
|
|
|
/** @var Promise $promise */
|
2016-11-14 20:59:21 +01:00
|
|
|
$promise = $callable();
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-11-14 20:59:21 +01:00
|
|
|
$this->assertInstanceOf(Promise::class, $promise);
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$promise->onResolve(function ($exception, $value) use (&$reason, &$result) {
|
2016-08-12 23:56:03 +02:00
|
|
|
$reason = $exception;
|
2017-01-07 22:55:34 +01:00
|
|
|
$result = $value;
|
2016-08-12 23:56:03 +02:00
|
|
|
});
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-08-12 23:56:03 +02:00
|
|
|
$this->assertSame($exception, $reason);
|
2017-01-07 22:55:34 +01:00
|
|
|
$this->assertNull($result);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-04-23 14:09:16 +02:00
|
|
|
* @depends testCreateCallableFunction
|
2017-01-07 22:55:34 +01:00
|
|
|
*/
|
2017-04-23 14:09:16 +02:00
|
|
|
public function testCreateCallableFunctionWithSuccessReturnCallback() {
|
|
|
|
$callable = \Amp\createCallable(function () {
|
2017-01-07 22:55:34 +01:00
|
|
|
return new Success(42);
|
|
|
|
});
|
|
|
|
|
|
|
|
/** @var Promise $promise */
|
|
|
|
$promise = $callable();
|
|
|
|
|
|
|
|
$this->assertInstanceOf(Promise::class, $promise);
|
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$promise->onResolve(function ($exception, $value) use (&$reason, &$result) {
|
2017-01-07 22:55:34 +01:00
|
|
|
$reason = $exception;
|
|
|
|
$result = $value;
|
|
|
|
});
|
|
|
|
|
|
|
|
$this->assertNull($reason);
|
|
|
|
$this->assertSame(42, $result);
|
2016-07-12 18:20:06 +02:00
|
|
|
}
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2017-04-23 14:09:16 +02:00
|
|
|
public function testCreateCallableFunctionWithReactPromise() {
|
|
|
|
$callable = \Amp\createCallable(function () {
|
2017-03-14 22:15:36 +01:00
|
|
|
return new FulfilledReactPromise(42);
|
|
|
|
});
|
|
|
|
|
|
|
|
/** @var Promise $promise */
|
|
|
|
$promise = $callable();
|
|
|
|
|
|
|
|
$this->assertInstanceOf(Promise::class, $promise);
|
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$promise->onResolve(function ($exception, $value) use (&$reason, &$result) {
|
2017-03-14 22:15:36 +01:00
|
|
|
$reason = $exception;
|
|
|
|
$result = $value;
|
|
|
|
});
|
|
|
|
|
|
|
|
$this->assertNull($reason);
|
|
|
|
$this->assertSame(42, $result);
|
|
|
|
}
|
|
|
|
|
2016-08-11 21:35:58 +02:00
|
|
|
public function testCoroutineResolvedWithReturn() {
|
|
|
|
$value = 1;
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-08-11 21:35:58 +02:00
|
|
|
$generator = function () use ($value) {
|
|
|
|
return $value;
|
|
|
|
yield; // Unreachable, but makes function a coroutine.
|
|
|
|
};
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-08-11 21:35:58 +02:00
|
|
|
$coroutine = new Coroutine($generator());
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$coroutine->onResolve(function ($exception, $value) use (&$reason, &$result) {
|
2017-01-07 22:55:34 +01:00
|
|
|
$reason = $exception;
|
2016-08-11 21:35:58 +02:00
|
|
|
$result = $value;
|
|
|
|
});
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2017-01-07 22:55:34 +01:00
|
|
|
$this->assertNull($reason);
|
2016-08-11 21:35:58 +02:00
|
|
|
$this->assertSame($value, $result);
|
|
|
|
}
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-08-11 21:35:58 +02:00
|
|
|
/**
|
|
|
|
* @depends testCoroutineResolvedWithReturn
|
|
|
|
*/
|
|
|
|
public function testYieldFromGenerator() {
|
|
|
|
$value = 1;
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-08-11 21:35:58 +02:00
|
|
|
$generator = function () use ($value) {
|
|
|
|
$generator = function () use ($value) {
|
|
|
|
return yield new Success($value);
|
|
|
|
};
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-08-11 21:35:58 +02:00
|
|
|
return yield from $generator();
|
|
|
|
};
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-08-11 21:35:58 +02:00
|
|
|
$coroutine = new Coroutine($generator());
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$coroutine->onResolve(function ($exception, $value) use (&$reason, &$result) {
|
2017-01-07 22:55:34 +01:00
|
|
|
$reason = $exception;
|
2016-08-11 21:35:58 +02:00
|
|
|
$result = $value;
|
|
|
|
});
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2017-01-07 22:55:34 +01:00
|
|
|
$this->assertNull($reason);
|
2016-08-11 21:35:58 +02:00
|
|
|
$this->assertSame($value, $result);
|
|
|
|
}
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-08-11 21:35:58 +02:00
|
|
|
/**
|
|
|
|
* @depends testCoroutineResolvedWithReturn
|
|
|
|
*/
|
|
|
|
public function testFastReturningGenerator()
|
|
|
|
{
|
|
|
|
$value = 1;
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-08-11 21:35:58 +02:00
|
|
|
$generator = function () use ($value) {
|
|
|
|
if (true) {
|
|
|
|
return $value;
|
|
|
|
}
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-08-11 21:35:58 +02:00
|
|
|
yield;
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-08-11 21:35:58 +02:00
|
|
|
return -$value;
|
|
|
|
};
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2016-08-11 21:35:58 +02:00
|
|
|
$coroutine = new Coroutine($generator());
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$coroutine->onResolve(function ($exception, $value) use (&$reason, &$result) {
|
2017-01-07 22:55:34 +01:00
|
|
|
$reason = $exception;
|
2016-08-11 21:35:58 +02:00
|
|
|
$result = $value;
|
|
|
|
});
|
2017-01-07 13:47:45 +01:00
|
|
|
|
2017-01-07 22:55:34 +01:00
|
|
|
$this->assertNull($reason);
|
2016-08-11 21:35:58 +02:00
|
|
|
$this->assertSame($value, $result);
|
|
|
|
}
|
2017-02-20 21:53:58 +01:00
|
|
|
|
|
|
|
public function testYieldingFulfilledReactPromise() {
|
|
|
|
$value = 1;
|
|
|
|
$promise = new ReactPromise(function ($resolve, $reject) use ($value) {
|
|
|
|
$resolve($value);
|
|
|
|
});
|
|
|
|
|
|
|
|
$generator = function () use ($promise) {
|
|
|
|
return yield $promise;
|
|
|
|
};
|
|
|
|
|
|
|
|
$coroutine = new Coroutine($generator());
|
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$coroutine->onResolve(function ($exception, $value) use (&$reason, &$result) {
|
2017-02-20 21:53:58 +01:00
|
|
|
$reason = $exception;
|
|
|
|
$result = $value;
|
|
|
|
});
|
|
|
|
|
|
|
|
$this->assertNull($reason);
|
|
|
|
$this->assertSame($value, $result);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testYieldingFulfilledReactPromiseAfterInteropPromise() {
|
|
|
|
$value = 1;
|
|
|
|
$promise = new ReactPromise(function ($resolve, $reject) use ($value) {
|
|
|
|
$resolve($value);
|
|
|
|
});
|
|
|
|
|
|
|
|
$generator = function () use ($promise) {
|
|
|
|
$value = yield new Success(-1);
|
|
|
|
return yield $promise;
|
|
|
|
};
|
|
|
|
|
|
|
|
$coroutine = new Coroutine($generator());
|
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$coroutine->onResolve(function ($exception, $value) use (&$reason, &$result) {
|
2017-02-20 21:53:58 +01:00
|
|
|
$reason = $exception;
|
|
|
|
$result = $value;
|
|
|
|
});
|
|
|
|
|
|
|
|
$this->assertNull($reason);
|
|
|
|
$this->assertSame($value, $result);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testYieldingRejectedReactPromise() {
|
|
|
|
$exception = new \Exception;
|
|
|
|
$promise = new ReactPromise(function ($resolve, $reject) use ($exception) {
|
|
|
|
$reject($exception);
|
|
|
|
});
|
|
|
|
|
|
|
|
$generator = function () use ($promise) {
|
|
|
|
return yield $promise;
|
|
|
|
};
|
|
|
|
|
|
|
|
$coroutine = new Coroutine($generator());
|
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$coroutine->onResolve(function ($exception, $value) use (&$reason, &$result) {
|
2017-02-20 21:53:58 +01:00
|
|
|
$reason = $exception;
|
|
|
|
$result = $value;
|
|
|
|
});
|
|
|
|
|
|
|
|
$this->assertSame($reason, $exception);
|
|
|
|
$this->assertNull($result);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testYieldingRejectedReactPromiseAfterInteropPromise() {
|
|
|
|
$exception = new \Exception;
|
|
|
|
$promise = new ReactPromise(function ($resolve, $reject) use ($exception) {
|
|
|
|
$reject($exception);
|
|
|
|
});
|
|
|
|
|
|
|
|
$generator = function () use ($promise) {
|
|
|
|
$value = yield new Success(-1);
|
|
|
|
return yield $promise;
|
|
|
|
};
|
|
|
|
|
|
|
|
$coroutine = new Coroutine($generator());
|
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$coroutine->onResolve(function ($exception, $value) use (&$reason, &$result) {
|
2017-02-20 21:53:58 +01:00
|
|
|
$reason = $exception;
|
|
|
|
$result = $value;
|
|
|
|
});
|
|
|
|
|
|
|
|
$this->assertSame($reason, $exception);
|
|
|
|
$this->assertNull($result);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testReturnFulfilledReactPromise() {
|
|
|
|
$value = 1;
|
|
|
|
$promise = new ReactPromise(function ($resolve, $reject) use ($value) {
|
|
|
|
$resolve($value);
|
|
|
|
});
|
|
|
|
|
|
|
|
$generator = function () use ($promise) {
|
|
|
|
return $promise;
|
|
|
|
yield; // Unreachable, but makes function a generator.
|
|
|
|
};
|
|
|
|
|
|
|
|
$coroutine = new Coroutine($generator());
|
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$coroutine->onResolve(function ($exception, $value) use (&$reason, &$result) {
|
2017-02-20 21:53:58 +01:00
|
|
|
$reason = $exception;
|
|
|
|
$result = $value;
|
|
|
|
});
|
|
|
|
|
|
|
|
$this->assertNull($reason);
|
|
|
|
$this->assertSame($value, $result);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testReturningRejectedReactPromise() {
|
|
|
|
$exception = new \Exception;
|
|
|
|
$promise = new ReactPromise(function ($resolve, $reject) use ($exception) {
|
|
|
|
$reject($exception);
|
|
|
|
});
|
|
|
|
|
|
|
|
$generator = function () use ($promise) {
|
|
|
|
return $promise;
|
|
|
|
yield; // Unreachable, but makes function a generator.
|
|
|
|
};
|
|
|
|
|
|
|
|
$coroutine = new Coroutine($generator());
|
|
|
|
|
2017-03-21 17:23:37 +01:00
|
|
|
$coroutine->onResolve(function ($exception, $value) use (&$reason, &$result) {
|
2017-02-20 21:53:58 +01:00
|
|
|
$reason = $exception;
|
|
|
|
$result = $value;
|
|
|
|
});
|
|
|
|
|
|
|
|
$this->assertSame($reason, $exception);
|
|
|
|
$this->assertNull($result);
|
|
|
|
}
|
2016-07-12 18:20:06 +02:00
|
|
|
}
|