1
0
mirror of https://github.com/danog/amp.git synced 2024-11-30 04:29:08 +01:00
amp/test/CoroutineTest.php

773 lines
21 KiB
PHP
Raw Normal View History

<?php
2016-08-17 06:27:10 +02:00
namespace Amp\Test;
2016-07-12 18:20:06 +02:00
use Amp\Coroutine;
use Amp\Failure;
use Amp\InvalidYieldError;
use Amp\Loop;
use Amp\Pause;
use Amp\Success;
use Amp\Promise;
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 {
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) {
$yielded = yield new Success($value);
2016-07-12 18:20:06 +02: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) {
$yielded = yield new Failure($exception);
2016-07-12 18:20:06 +02:00
};
$coroutine = new Coroutine($generator());
$this->assertNull($yielded);
$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;
Loop::run(function () use (&$yielded, $value) {
2016-07-12 18:20:06 +02:00
$generator = function () use (&$yielded, $value) {
$yielded = yield new Pause(self::TIMEOUT, $value);
2016-07-12 18:20:06 +02:00
};
new Coroutine($generator());
2016-07-12 18:20:06 +02:00
});
$this->assertSame($value, $yielded);
}
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;
};
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
}
2016-07-12 18:20:06 +02:00
public function testInvalidYield() {
$generator = function () {
yield 1;
};
$coroutine = new Coroutine($generator());
$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());
$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
}
2016-12-17 15:16:17 +01:00
/**
* @depends testInvalidYield
*/
public function testInvalidYieldCatchingThrownException() {
$generator = function () {
try {
yield 1;
} catch (\Error $exception) {
// No further yields.
}
};
2016-12-17 15:16:17 +01:00
$coroutine = new Coroutine($generator());
$coroutine->onResolve(function ($exception) use (&$reason) {
2016-12-17 15:16:17 +01:00
$reason = $exception;
});
2016-12-17 15:16:17 +01:00
$this->assertInstanceOf(InvalidYieldError::class, $reason);
}
2016-12-17 15:16:17 +01:00
/**
* @depends testInvalidYieldCatchingThrownException
*/
public function testInvalidYieldCatchingThrownExceptionAndYieldingAgain() {
$generator = function () {
try {
yield 1;
} catch (\Error $exception) {
yield new Success;
}
};
2016-12-17 15:16:17 +01:00
$coroutine = new Coroutine($generator());
$coroutine->onResolve(function ($exception) use (&$reason) {
2016-12-17 15:16:17 +01:00
$reason = $exception;
});
2016-12-17 15:16:17 +01:00
$this->assertInstanceOf(InvalidYieldError::class, $reason);
}
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;
}
};
2016-12-17 15:16:17 +01:00
$coroutine = new Coroutine($generator());
$coroutine->onResolve(function ($exception) use (&$reason) {
2016-12-17 15:16:17 +01:00
$reason = $exception;
});
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());
$coroutine->onResolve(function ($exception, $value) use (&$reason, &$result) {
$reason = $exception;
2016-07-12 18:20:06 +02:00
$result = $value;
});
$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());
$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());
$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());
$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;
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());
$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;
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());
$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());
$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;
$coroutine->onResolve(function () use (&$invoked) {
2016-07-12 18:20:06 +02:00
$invoked = true;
});
$this->assertTrue($invoked);
}
public function testCreateCallableFunction() {
$callable = \Amp\createCallable(function () {
2016-07-12 18:20:06 +02:00
yield;
});
$this->assertInstanceOf(Coroutine::class, $callable());
}
2016-08-12 23:56:03 +02:00
/**
* @depends testCreateCallableFunction
2016-08-12 23:56:03 +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);
$callable = \Amp\createCallable(function ($value) {
2016-08-12 23:56:03 +02:00
return $value;
});
/** @var Promise $promise */
2016-11-14 20:59:21 +01:00
$promise = $callable($promise);
2016-11-14 20:59:21 +01:00
$this->assertInstanceOf(Promise::class, $promise);
$promise->onResolve(function ($exception, $value) use (&$reason, &$result) {
$reason = $exception;
2016-08-12 23:56:03 +02:00
$result = $value;
});
$this->assertNull($reason);
2016-08-12 23:56:03 +02:00
$this->assertSame($value, $result);
}
2016-07-12 18:20:06 +02:00
/**
* @depends testCreateCallableFunction
2016-07-12 18:20:06 +02:00
*/
public function testCreateCallableFunctionWithNonGeneratorCallback() {
2016-08-12 23:56:03 +02:00
$value = 1;
$callable = \Amp\createCallable(function ($value) {
2016-08-12 23:56:03 +02:00
return $value;
});
/** @var Promise $promise */
2016-11-14 20:59:21 +01:00
$promise = $callable($value);
2016-11-14 20:59:21 +01:00
$this->assertInstanceOf(Promise::class, $promise);
$promise->onResolve(function ($exception, $value) use (&$reason, &$result) {
$reason = $exception;
2016-08-12 23:56:03 +02:00
$result = $value;
});
$this->assertNull($reason);
2016-08-12 23:56:03 +02:00
$this->assertSame($value, $result);
}
2016-08-12 23:56:03 +02:00
/**
* @depends testCreateCallableFunction
2016-08-12 23:56:03 +02:00
*/
public function testCreateCallableFunctionWithThrowingCallback() {
2016-08-12 23:56:03 +02:00
$exception = new \Exception;
$callable = \Amp\createCallable(function () use ($exception) {
2016-08-12 23:56:03 +02:00
throw $exception;
});
/** @var Promise $promise */
2016-11-14 20:59:21 +01:00
$promise = $callable();
2016-11-14 20:59:21 +01:00
$this->assertInstanceOf(Promise::class, $promise);
$promise->onResolve(function ($exception, $value) use (&$reason, &$result) {
2016-08-12 23:56:03 +02:00
$reason = $exception;
$result = $value;
2016-08-12 23:56:03 +02:00
});
2016-08-12 23:56:03 +02:00
$this->assertSame($exception, $reason);
$this->assertNull($result);
}
/**
* @depends testCreateCallableFunction
*/
public function testCreateCallableFunctionWithSuccessReturnCallback() {
$callable = \Amp\createCallable(function () {
return new Success(42);
});
/** @var Promise $promise */
$promise = $callable();
$this->assertInstanceOf(Promise::class, $promise);
$promise->onResolve(function ($exception, $value) use (&$reason, &$result) {
$reason = $exception;
$result = $value;
});
$this->assertNull($reason);
$this->assertSame(42, $result);
2016-07-12 18:20:06 +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);
$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;
2016-08-11 21:35:58 +02:00
$generator = function () use ($value) {
return $value;
yield; // Unreachable, but makes function a coroutine.
};
2016-08-11 21:35:58 +02:00
$coroutine = new Coroutine($generator());
$coroutine->onResolve(function ($exception, $value) use (&$reason, &$result) {
$reason = $exception;
2016-08-11 21:35:58 +02:00
$result = $value;
});
$this->assertNull($reason);
2016-08-11 21:35:58 +02:00
$this->assertSame($value, $result);
}
2016-08-11 21:35:58 +02:00
/**
* @depends testCoroutineResolvedWithReturn
*/
public function testYieldFromGenerator() {
$value = 1;
2016-08-11 21:35:58 +02:00
$generator = function () use ($value) {
$generator = function () use ($value) {
return yield new Success($value);
};
2016-08-11 21:35:58 +02:00
return yield from $generator();
};
2016-08-11 21:35:58 +02:00
$coroutine = new Coroutine($generator());
$coroutine->onResolve(function ($exception, $value) use (&$reason, &$result) {
$reason = $exception;
2016-08-11 21:35:58 +02:00
$result = $value;
});
$this->assertNull($reason);
2016-08-11 21:35:58 +02:00
$this->assertSame($value, $result);
}
2016-08-11 21:35:58 +02:00
/**
* @depends testCoroutineResolvedWithReturn
*/
public function testFastReturningGenerator()
{
$value = 1;
2016-08-11 21:35:58 +02:00
$generator = function () use ($value) {
if (true) {
return $value;
}
2016-08-11 21:35:58 +02:00
yield;
2016-08-11 21:35:58 +02:00
return -$value;
};
2016-08-11 21:35:58 +02:00
$coroutine = new Coroutine($generator());
$coroutine->onResolve(function ($exception, $value) use (&$reason, &$result) {
$reason = $exception;
2016-08-11 21:35:58 +02:00
$result = $value;
});
$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());
$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());
$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());
$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());
$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());
$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());
$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
}