mirror of
https://github.com/danog/amp.git
synced 2025-01-22 13:21:16 +01:00
Move promise documentation to own document + success / failure, fixes #97
This commit is contained in:
parent
9816bd9828
commit
3b6ca19dfe
@ -5,104 +5,6 @@ The weak link when managing concurrency is humans; we simply don't think asynchr
|
||||
1. Get smarter (not feasible);
|
||||
2. Abstract concurrent task execution to make it feel synchronous.
|
||||
|
||||
## Promises
|
||||
|
||||
The basic unit of concurrency in Amp applications is the `Amp\Promise`. These objects should be thought of as placeholders for values or tasks that aren't yet complete. By using placeholders we're able to reason about the results of concurrent operations as if they were already complete variables.
|
||||
|
||||
> **NOTE**
|
||||
>
|
||||
> Amp's `Promise` interface **does not** conform to the "Thenables" abstraction common in JavaScript promise implementations. Chaining `.then()` calls is a suboptimal method for avoiding callback hell in a world with generator coroutines. Instead, Amp utilizes PHP generators to "synchronize" concurrent task execution.
|
||||
>
|
||||
> However, as ReactPHP is another wide-spread implementation, we also accept any `React\Promise\PromiseInterface` where we accept instances of `Amp\Promise`. In case of custom implementations not implementing `React\Promise\PromiseInterface`, `Amp\adapt` can be used to adapt any object having a `then` or `done` method.
|
||||
|
||||
### The Promise API
|
||||
|
||||
```php
|
||||
interface Promise {
|
||||
public function when(callable $onResolve);
|
||||
}
|
||||
```
|
||||
|
||||
In its simplest form the `Amp\Promise` aggregates callbacks for dealing with computational results once they eventually resolve. While most code will not interact with this API directly thanks to the magic of [Generators](#generators), let's take a quick look at the one simple API method exposed on `Amp\Promise` implementations:
|
||||
|
||||
|
||||
| Method | Callback Signature |
|
||||
| --------- | ----------------------------------------------------------|
|
||||
| when | `function ($error = null, $result = null)` |
|
||||
|
||||
`Amp\Promise::when()` accepts an error-first callback. This callback is responsible for reacting to the eventual result of the computation represented by the promise placeholder. For example:
|
||||
|
||||
```php
|
||||
<?php
|
||||
|
||||
$promise = someFunctionThatReturnsAPromise();
|
||||
$promise->when(function (Throwable $error = null, $result = null) {
|
||||
if ($error) {
|
||||
printf(
|
||||
"Something went wrong:\n%s\n",
|
||||
$error->getMessage()
|
||||
);
|
||||
} else {
|
||||
printf(
|
||||
"Hurray! Our result is:\n%s\n",
|
||||
print_r($result, true)
|
||||
);
|
||||
}
|
||||
});
|
||||
```
|
||||
|
||||
Those familiar with JavaScript code generally reflect that the above interface quickly devolves into ["callback hell"](http://callbackhell.com/), and they're correct. We will shortly see how to avoid this problem in the [Generators](#generators) section.
|
||||
|
||||
## Deferred
|
||||
|
||||
`Amp\Deferred` is the abstraction responsible for resolving future values once they become available. A library that resolves values asynchronously creates an `Amp\Deferred` and uses it to return an `Amp\Promise` to API consumers. Once the async library determines that the value is ready it resolves the promise held by the API consumer using methods on the linked promisor.
|
||||
|
||||
### The Deferred API
|
||||
|
||||
```php
|
||||
final class Deferred {
|
||||
public function promise(): Promise;
|
||||
public function resolve($result = null);
|
||||
public function fail($error);
|
||||
}
|
||||
```
|
||||
|
||||
#### `promise()`
|
||||
|
||||
Returns the corresponding `Promise` instance. `Deferred` and `Promise` are separated, so the consumer of the promise can't fulfill it.
|
||||
|
||||
#### `resolve()`
|
||||
|
||||
Resolves the promise with the first parameter as value, otherwise `null`. If a `Amp\Promise` is passed, the resolution will wait until the passed promise has been resolved. Invokes all registered `Promise::when()` callbacks.
|
||||
|
||||
#### `fail()`
|
||||
|
||||
Makes the promise fail. Invokes all registered `Promise::when()` callbacks with the passed `Throwable` as `$error` argument.
|
||||
|
||||
Here's a simple example of an async value producer `asyncMultiply()` creating a deferred and returning the associated promise to its API consumer.
|
||||
|
||||
```php
|
||||
<?php // Example async producer using promisor
|
||||
|
||||
use Amp\Loop;
|
||||
|
||||
function asyncMultiply($x, $y) {
|
||||
// Create a new promisor
|
||||
$deferred = new Amp\Deferred;
|
||||
|
||||
// Resolve the async result one second from now
|
||||
Loop::delay($msDelay = 1000, function () use ($deferred, $x, $y) {
|
||||
$deferred->resolve($x * $y);
|
||||
});
|
||||
|
||||
return $deferred->promise();
|
||||
}
|
||||
|
||||
$promise = asyncMultiply(6, 7);
|
||||
$result = Amp\wait($promise);
|
||||
var_dump($result); // int(42)
|
||||
```
|
||||
|
||||
## Combinators
|
||||
|
||||
### `map()`
|
||||
|
@ -1,3 +1,101 @@
|
||||
# Promises
|
||||
|
||||
This is a documentation stub. Please help writing this documentation. See [#97](https://github.com/amphp/amp/issues/97).
|
||||
@TODO: Fix generator references once coroutines docs are finished.
|
||||
|
||||
The basic unit of concurrency in Amp applications is the `Amp\Promise`. These objects should be thought of as placeholders for values or tasks that aren't yet complete. By using placeholders we're able to reason about the results of concurrent operations as if they were already complete variables.
|
||||
|
||||
> **NOTE**
|
||||
>
|
||||
> Amp's `Promise` interface **does not** conform to the "Thenables" abstraction common in JavaScript promise implementations. Chaining `.then()` calls is a suboptimal method for avoiding callback hell in a world with generator coroutines. Instead, Amp utilizes PHP generators to "synchronize" concurrent task execution.
|
||||
>
|
||||
> However, as ReactPHP is another wide-spread implementation, we also accept any `React\Promise\PromiseInterface` where we accept instances of `Amp\Promise`. In case of custom implementations not implementing `React\Promise\PromiseInterface`, `Amp\adapt` can be used to adapt any object having a `then` or `done` method.
|
||||
|
||||
## Promise Consumption
|
||||
|
||||
```php
|
||||
interface Promise {
|
||||
public function when(callable $onResolve);
|
||||
}
|
||||
```
|
||||
|
||||
In its simplest form the `Amp\Promise` aggregates callbacks for dealing with computational results once they eventually resolve. While most code will not interact with this API directly thanks to the magic of [Generators](#generators), let's take a quick look at the one simple API method exposed on `Amp\Promise` implementations:
|
||||
|
||||
|
||||
| Method | Callback Signature |
|
||||
| --------- | ----------------------------------------------------------|
|
||||
| when | `function ($error = null, $result = null)` |
|
||||
|
||||
`Amp\Promise::when()` accepts an error-first callback. This callback is responsible for reacting to the eventual result of the computation represented by the promise placeholder. For example:
|
||||
|
||||
```php
|
||||
<?php
|
||||
|
||||
$promise = someFunctionThatReturnsAPromise();
|
||||
$promise->when(function (Throwable $error = null, $result = null) {
|
||||
if ($error) {
|
||||
printf(
|
||||
"Something went wrong:\n%s\n",
|
||||
$error->getMessage()
|
||||
);
|
||||
} else {
|
||||
printf(
|
||||
"Hurray! Our result is:\n%s\n",
|
||||
print_r($result, true)
|
||||
);
|
||||
}
|
||||
});
|
||||
```
|
||||
|
||||
Those familiar with JavaScript code generally reflect that the above interface quickly devolves into ["callback hell"](http://callbackhell.com/), and they're correct. We will shortly see how to avoid this problem in the [Generators](#generators) section.
|
||||
|
||||
## Promise Creation
|
||||
|
||||
`Amp\Deferred` is the abstraction responsible for resolving future values once they become available. A library that resolves values asynchronously creates an `Amp\Deferred` and uses it to return an `Amp\Promise` to API consumers. Once the async library determines that the value is ready it resolves the promise held by the API consumer using methods on the linked promisor.
|
||||
|
||||
```php
|
||||
final class Deferred {
|
||||
public function promise(): Promise;
|
||||
public function resolve($result = null);
|
||||
public function fail($error);
|
||||
}
|
||||
```
|
||||
|
||||
#### `promise()`
|
||||
|
||||
Returns the corresponding `Promise` instance. `Deferred` and `Promise` are separated, so the consumer of the promise can't fulfill it.
|
||||
|
||||
#### `resolve()`
|
||||
|
||||
Resolves the promise with the first parameter as value, otherwise `null`. If a `Amp\Promise` is passed, the resolution will wait until the passed promise has been resolved. Invokes all registered `Promise::when()` callbacks.
|
||||
|
||||
#### `fail()`
|
||||
|
||||
Makes the promise fail. Invokes all registered `Promise::when()` callbacks with the passed `Throwable` as `$error` argument.
|
||||
|
||||
Here's a simple example of an async value producer `asyncMultiply()` creating a deferred and returning the associated promise to its API consumer.
|
||||
|
||||
```php
|
||||
<?php // Example async producer using promisor
|
||||
|
||||
use Amp\Loop;
|
||||
|
||||
function asyncMultiply($x, $y) {
|
||||
// Create a new promisor
|
||||
$deferred = new Amp\Deferred;
|
||||
|
||||
// Resolve the async result one second from now
|
||||
Loop::delay($msDelay = 1000, function () use ($deferred, $x, $y) {
|
||||
$deferred->resolve($x * $y);
|
||||
});
|
||||
|
||||
return $deferred->promise();
|
||||
}
|
||||
|
||||
$promise = asyncMultiply(6, 7);
|
||||
$result = Amp\wait($promise);
|
||||
var_dump($result); // int(42)
|
||||
```
|
||||
|
||||
### Success and Failure
|
||||
|
||||
Sometimes values are immediately available. This might be due to them being cached, but can also be the case if an interface mandates a promise to be returned to allow for async I/O, but the specific implementation being always having directly the result available. In these cases `Amp\Success` and `Amp\Failure` can be used to construct a immediately resolved promise. `Amp\Success` accepts a resolution value. `Amp\Failure` accepts an exception as failure reason.
|
||||
|
Loading…
x
Reference in New Issue
Block a user