mirror of
https://github.com/danog/amp.git
synced 2025-01-22 13:21:16 +01:00
118 lines
3.4 KiB
PHP
Executable File
118 lines
3.4 KiB
PHP
Executable File
<?php
|
|
|
|
namespace Alert;
|
|
|
|
/**
|
|
* A Promise is a writable container used to complete a future once its value is resolved.
|
|
*
|
|
* The Alert\Promise is NOT the same as the common javascript "promise" idiom. Instead,
|
|
* Alert defines a "Promise" as an internal agreement made by producers of asynchronous
|
|
* results to fulfill a placeholder (Future) value at some point in the future. In this
|
|
* regard an Alert\Promise has more in common with the Scala promise API than javascript
|
|
* implementations.
|
|
*
|
|
* A Promise resolves its associated Future placeholder with a value using the succeed()
|
|
* method. Conversely, a promise can also report future failure by passing an Exception
|
|
* via the failure() method.
|
|
*
|
|
* Example:
|
|
*
|
|
* class MyAsyncProducer {
|
|
* public function retrieveValueAsynchronously() {
|
|
* // Create a new promise that needs to be resolved
|
|
* $promise = new Alert\Promise;
|
|
*
|
|
* $future = $promise->getFuture();
|
|
*
|
|
* // When we finish non-blocking value resolution we
|
|
* // simply call the relevant Promise method depending
|
|
* // on whether or not retrieval succeeded:
|
|
* //
|
|
* // $promise->succeed($value)
|
|
* // $promise->fail($error)
|
|
*
|
|
* return $future;
|
|
* }
|
|
* }
|
|
*/
|
|
class Promise {
|
|
private $value;
|
|
private $error;
|
|
private $future;
|
|
private $futureResolver;
|
|
|
|
public function __construct() {
|
|
$futureResolver = function(\Exception $error = NULL, $value = NULL) {
|
|
$this->resolve($error, $value);
|
|
};
|
|
$future = new Unresolved;
|
|
$this->futureResolver = $futureResolver->bindTo($future, $future);
|
|
$this->future = $future;
|
|
}
|
|
|
|
/**
|
|
* Retrieve the Future value associated with this Promise
|
|
*
|
|
* @return \Alert\Future
|
|
*/
|
|
public function getFuture() {
|
|
return $this->future;
|
|
}
|
|
|
|
/**
|
|
* Fufill the Promise's Future value with a successful result
|
|
*
|
|
* @param mixed $value
|
|
* @return void
|
|
*/
|
|
public function succeed($value = NULL) {
|
|
if ($value instanceof Future) {
|
|
$value->onComplete(function(Future $f) {
|
|
$this->resolve($f->getError(), $f->getValue());
|
|
});
|
|
} else {
|
|
call_user_func($this->futureResolver, $error = NULL, $value);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Fail the Promise's associated Future
|
|
*
|
|
* @param \Exception $error
|
|
* @return void
|
|
*/
|
|
public function fail(\Exception $error) {
|
|
call_user_func($this->futureResolver, $error, $value = NULL);
|
|
}
|
|
|
|
/**
|
|
* Resolve the Promise's associated Future
|
|
*
|
|
* @param \Exception $error
|
|
* @param mixed $value
|
|
* @return void
|
|
*/
|
|
public function resolve(\Exception $error = NULL, $value = NULL) {
|
|
call_user_func($this->futureResolver, $error, $value);
|
|
}
|
|
|
|
/**
|
|
* Resolve the associated Future but only if it has not previously completed
|
|
*
|
|
* @param \Exception $error
|
|
* @param mixed $value
|
|
* @return bool Returns TRUE if the Future was resolved by this operation or FALSE if the
|
|
* relevant Future was previously resolved
|
|
*/
|
|
public function resolveSafely(\Exception $error = NULL, $value = NULL) {
|
|
if ($this->future->isComplete()) {
|
|
$didResolve = FALSE;
|
|
} else {
|
|
$this->resolve($error, $value);
|
|
$didResolve = TRUE;
|
|
}
|
|
|
|
return $didResolve;
|
|
}
|
|
}
|