2014-09-22 22:47:48 +02:00
|
|
|
<?php
|
|
|
|
|
2014-09-23 04:38:32 +02:00
|
|
|
namespace Amp;
|
2014-09-22 22:47:48 +02:00
|
|
|
|
|
|
|
class Future implements Promisor, Promise {
|
|
|
|
private $isResolved = false;
|
|
|
|
private $watchers = [];
|
|
|
|
private $whens = [];
|
|
|
|
private $error;
|
|
|
|
private $result;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve the Promise placeholder for this deferred value
|
|
|
|
*
|
|
|
|
* This implementation acts as both Promisor and Promise so we simply return the
|
|
|
|
* current instance. If users require a Promisor that can only be resolved by code
|
2014-09-23 04:38:32 +02:00
|
|
|
* holding a reference to the Promisor they may instead use Amp\PrivateFuture.
|
2014-09-22 22:47:48 +02:00
|
|
|
*/
|
2015-03-19 16:14:21 +01:00
|
|
|
public function promise(): Future {
|
2014-09-22 22:47:48 +02:00
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Notify the $func callback when the promise resolves (whether successful or not)
|
|
|
|
*
|
|
|
|
* $func callbacks are invoked with parameters in error-first style.
|
2015-03-19 16:14:21 +01:00
|
|
|
*
|
|
|
|
* @return void
|
2014-09-22 22:47:48 +02:00
|
|
|
*/
|
|
|
|
public function when(callable $func) {
|
|
|
|
if ($this->isResolved) {
|
|
|
|
$func($this->error, $this->result);
|
|
|
|
} else {
|
|
|
|
$this->whens[] = $func;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Notify the $func callback when resolution progress events are emitted
|
2015-03-19 16:14:21 +01:00
|
|
|
*
|
|
|
|
* @return void
|
2014-09-22 22:47:48 +02:00
|
|
|
*/
|
|
|
|
public function watch(callable $func) {
|
|
|
|
if (!$this->isResolved) {
|
|
|
|
$this->watchers[] = $func;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2014-12-02 01:45:57 +01:00
|
|
|
* This method is deprecated. New code should use Amp\wait($promise) instead.
|
2014-09-22 22:47:48 +02:00
|
|
|
*/
|
|
|
|
public function wait() {
|
2014-12-02 01:45:57 +01:00
|
|
|
trigger_error(
|
|
|
|
'Amp\\Promise::wait() is deprecated and scheduled for removal. ' .
|
|
|
|
'Please update code to use Amp\\wait($promise) instead.',
|
|
|
|
E_USER_DEPRECATED
|
|
|
|
);
|
2014-09-22 22:47:48 +02:00
|
|
|
|
2014-12-02 01:45:57 +01:00
|
|
|
$isWaiting = true;
|
|
|
|
$resolvedError = $resolvedResult = null;
|
|
|
|
$this->when(function($error, $result) use (&$isWaiting, &$resolvedError, &$resolvedResult) {
|
|
|
|
$isWaiting = false;
|
2014-09-22 22:47:48 +02:00
|
|
|
$resolvedError = $error;
|
|
|
|
$resolvedResult = $result;
|
2014-12-02 01:45:57 +01:00
|
|
|
});
|
|
|
|
$reactor = getReactor();
|
|
|
|
while ($isWaiting) {
|
|
|
|
$reactor->tick();
|
2014-09-22 22:47:48 +02:00
|
|
|
}
|
|
|
|
if ($resolvedError) {
|
|
|
|
throw $resolvedError;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $resolvedResult;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Update watchers of resolution progress events
|
2015-03-19 16:14:21 +01:00
|
|
|
*
|
2014-09-22 22:47:48 +02:00
|
|
|
* @param mixed $progress
|
2015-03-19 16:14:21 +01:00
|
|
|
* @throws \LogicException if the promise has already resolved
|
2014-09-22 22:47:48 +02:00
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function update($progress) {
|
|
|
|
if ($this->isResolved) {
|
|
|
|
throw new \LogicException(
|
|
|
|
'Cannot update resolved promise'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($this->watchers as $watcher) {
|
|
|
|
$watcher($progress);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Resolve the promised value as a success
|
2015-03-19 16:14:21 +01:00
|
|
|
*
|
2014-09-22 22:47:48 +02:00
|
|
|
* @param mixed $result
|
2015-03-19 16:14:21 +01:00
|
|
|
* @throws \LogicException if the promise has already resolved or the result is the current instance
|
2014-09-22 22:47:48 +02:00
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function succeed($result = null) {
|
|
|
|
if ($this->isResolved) {
|
|
|
|
throw new \LogicException(
|
|
|
|
'Promise already resolved'
|
|
|
|
);
|
|
|
|
} elseif ($result === $this) {
|
|
|
|
throw new \LogicException(
|
|
|
|
'A Promise cannot act as its own resolution result'
|
|
|
|
);
|
|
|
|
} elseif ($result instanceof Promise) {
|
|
|
|
$result->when(function(\Exception $error = null, $result = null) {
|
|
|
|
if ($error) {
|
|
|
|
$this->fail($error);
|
|
|
|
} else {
|
|
|
|
$this->succeed($result);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
$this->isResolved = true;
|
|
|
|
$this->result = $result;
|
|
|
|
$error = null;
|
|
|
|
foreach ($this->whens as $when) {
|
|
|
|
$when($error, $result);
|
|
|
|
}
|
|
|
|
$this->whens = $this->watchers = [];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Resolve the promised value as a failure
|
2015-03-19 16:14:21 +01:00
|
|
|
*
|
|
|
|
* @throws \LogicException if the promise has already resolved
|
2014-09-22 22:47:48 +02:00
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function fail(\Exception $error) {
|
|
|
|
if ($this->isResolved) {
|
|
|
|
throw new \LogicException(
|
|
|
|
'Promise already resolved'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->isResolved = true;
|
|
|
|
$this->error = $error;
|
|
|
|
$result = null;
|
|
|
|
|
|
|
|
foreach ($this->whens as $when) {
|
|
|
|
$when($error, $result);
|
|
|
|
}
|
|
|
|
$this->whens = $this->watchers = [];
|
|
|
|
}
|
|
|
|
}
|