mirror of
https://github.com/danog/amp.git
synced 2024-11-26 20:15:00 +01:00
Add timeoutWithDefault function (#245)
This commit is contained in:
parent
b73e03bfe2
commit
4a98cc4fde
@ -96,12 +96,14 @@ namespace Amp\Promise
|
|||||||
{
|
{
|
||||||
|
|
||||||
use Amp\Deferred;
|
use Amp\Deferred;
|
||||||
|
use Amp\Failure;
|
||||||
use Amp\Loop;
|
use Amp\Loop;
|
||||||
use Amp\MultiReasonException;
|
use Amp\MultiReasonException;
|
||||||
use Amp\Promise;
|
use Amp\Promise;
|
||||||
use Amp\Success;
|
use Amp\Success;
|
||||||
use Amp\TimeoutException;
|
use Amp\TimeoutException;
|
||||||
use React\Promise\PromiseInterface as ReactPromise;
|
use React\Promise\PromiseInterface as ReactPromise;
|
||||||
|
use function Amp\call;
|
||||||
use function Amp\Internal\createTypeError;
|
use function Amp\Internal\createTypeError;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -223,6 +225,34 @@ namespace Amp\Promise
|
|||||||
return $deferred->promise();
|
return $deferred->promise();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates an artificial timeout for any `Promise`.
|
||||||
|
*
|
||||||
|
* If the promise is resolved before the timeout expires, the result is returned
|
||||||
|
*
|
||||||
|
* If the timeout expires before the promise is resolved, a default value is returned
|
||||||
|
*
|
||||||
|
* @param \Amp\Promise|\React\Promise\PromiseInterface $promise Promise to which the timeout is applied.
|
||||||
|
* @param int $timeout Timeout in milliseconds.
|
||||||
|
* @param mixed $default
|
||||||
|
*
|
||||||
|
* @return \Amp\Promise
|
||||||
|
*
|
||||||
|
* @throws \TypeError If $promise is not an instance of \Amp\Promise or \React\Promise\PromiseInterface.
|
||||||
|
*/
|
||||||
|
function timeoutWithDefault($promise, int $timeout, $default = null): Promise
|
||||||
|
{
|
||||||
|
$promise = timeout($promise, $timeout);
|
||||||
|
|
||||||
|
return call(function () use ($promise, $default) {
|
||||||
|
try {
|
||||||
|
return yield $promise;
|
||||||
|
} catch (TimeoutException $exception) {
|
||||||
|
return $default;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Adapts any object with a done(callable $onFulfilled, callable $onRejected) or then(callable $onFulfilled,
|
* Adapts any object with a done(callable $onFulfilled, callable $onRejected) or then(callable $onFulfilled,
|
||||||
* callable $onRejected) method to a promise usable by components depending on placeholders implementing
|
* callable $onRejected) method to a promise usable by components depending on placeholders implementing
|
||||||
|
132
test/TimeoutWithDefaultTest.php
Normal file
132
test/TimeoutWithDefaultTest.php
Normal file
@ -0,0 +1,132 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Amp\Test;
|
||||||
|
|
||||||
|
use Amp\Delayed;
|
||||||
|
use Amp\Failure;
|
||||||
|
use Amp\Loop;
|
||||||
|
use Amp\Promise;
|
||||||
|
use Amp\Success;
|
||||||
|
use function React\Promise\resolve;
|
||||||
|
|
||||||
|
class TimeoutWithDefaultTest extends \PHPUnit\Framework\TestCase
|
||||||
|
{
|
||||||
|
public function testSuccessfulPromise()
|
||||||
|
{
|
||||||
|
Loop::run(function () {
|
||||||
|
$value = 1;
|
||||||
|
|
||||||
|
$promise = new Success($value);
|
||||||
|
|
||||||
|
$promise = Promise\timeoutWithDefault($promise, 100, 2);
|
||||||
|
$this->assertInstanceOf(Promise::class, $promise);
|
||||||
|
|
||||||
|
$callback = function ($exception, $value) use (&$result) {
|
||||||
|
$result = $value;
|
||||||
|
};
|
||||||
|
|
||||||
|
$promise->onResolve($callback);
|
||||||
|
|
||||||
|
$this->assertSame($value, $result);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testFailedPromise()
|
||||||
|
{
|
||||||
|
Loop::run(function () {
|
||||||
|
$exception = new \Exception;
|
||||||
|
|
||||||
|
$promise = new Failure($exception);
|
||||||
|
|
||||||
|
$promise = Promise\timeoutWithDefault($promise, 100, 2);
|
||||||
|
$this->assertInstanceOf(Promise::class, $promise);
|
||||||
|
|
||||||
|
$callback = function ($exception, $value) use (&$actual) {
|
||||||
|
if ($exception) {
|
||||||
|
$actual = $exception;
|
||||||
|
} else {
|
||||||
|
$actual = $value;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
$promise->onResolve($callback);
|
||||||
|
|
||||||
|
$this->assertSame($exception, $actual);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @depends testSuccessfulPromise
|
||||||
|
*/
|
||||||
|
public function testFastPending()
|
||||||
|
{
|
||||||
|
$value = 1;
|
||||||
|
|
||||||
|
Loop::run(function () use (&$result, $value) {
|
||||||
|
$promise = new Delayed(50, $value);
|
||||||
|
|
||||||
|
$promise = Promise\timeoutWithDefault($promise, 100);
|
||||||
|
$this->assertInstanceOf(Promise::class, $promise);
|
||||||
|
|
||||||
|
$callback = function ($exception, $value) use (&$result) {
|
||||||
|
$result = $value;
|
||||||
|
};
|
||||||
|
|
||||||
|
$promise->onResolve($callback);
|
||||||
|
});
|
||||||
|
|
||||||
|
$this->assertSame($value, $result);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @depends testSuccessfulPromise
|
||||||
|
*/
|
||||||
|
public function testSlowPending()
|
||||||
|
{
|
||||||
|
$expected = 2;
|
||||||
|
|
||||||
|
Loop::run(function () use (&$actual, $expected) {
|
||||||
|
$promise = new Delayed(200);
|
||||||
|
|
||||||
|
$promise = Promise\timeoutWithDefault($promise, 100, $expected);
|
||||||
|
$this->assertInstanceOf(Promise::class, $promise);
|
||||||
|
|
||||||
|
$callback = function ($exception, $value) use (&$actual) {
|
||||||
|
$actual = $value;
|
||||||
|
};
|
||||||
|
|
||||||
|
$promise->onResolve($callback);
|
||||||
|
});
|
||||||
|
|
||||||
|
$this->assertEquals($expected, $actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @depends testSuccessfulPromise
|
||||||
|
*/
|
||||||
|
public function testReactPromise()
|
||||||
|
{
|
||||||
|
Loop::run(function () {
|
||||||
|
$value = 1;
|
||||||
|
|
||||||
|
$promise = resolve($value);
|
||||||
|
|
||||||
|
$promise = Promise\timeoutWithDefault($promise, 100, 2);
|
||||||
|
$this->assertInstanceOf(Promise::class, $promise);
|
||||||
|
|
||||||
|
$callback = function ($exception, $value) use (&$result) {
|
||||||
|
$result = $value;
|
||||||
|
};
|
||||||
|
|
||||||
|
$promise->onResolve($callback);
|
||||||
|
|
||||||
|
$this->assertSame($value, $result);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testNonPromise()
|
||||||
|
{
|
||||||
|
$this->expectException(\TypeError::class);
|
||||||
|
Promise\timeoutWithDefault(42, 42);
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user