1
0
mirror of https://github.com/danog/amp.git synced 2025-01-09 06:28:17 +01:00
amp/lib/Future.php

126 lines
3.4 KiB
PHP
Raw Normal View History

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;
}
}
/**
* 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) {
2014-09-22 22:47:48 +02:00
if ($this->isResolved) {
throw new \LogicException(
'Cannot update resolved promise'
);
}
foreach ($this->watchers as $watcher) {
$watcher(...$progress);
2014-09-22 22:47:48 +02:00
}
}
/**
* 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 = [];
}
}