2016-01-20 12:01:40 +01:00
|
|
|
<?php
|
|
|
|
|
2017-03-10 19:19:32 +01:00
|
|
|
namespace Amp\Loop;
|
|
|
|
|
2017-03-10 23:03:41 +01:00
|
|
|
use Amp\Coroutine;
|
|
|
|
use Amp\Promise;
|
2017-03-10 21:31:57 +01:00
|
|
|
use Amp\Internal\Watcher;
|
2017-03-11 07:13:03 +01:00
|
|
|
use React\Promise\PromiseInterface as ReactPromise;
|
|
|
|
use function Amp\adapt;
|
2017-03-10 23:03:41 +01:00
|
|
|
use function Amp\rethrow;
|
2016-01-20 12:01:40 +01:00
|
|
|
|
2016-09-04 22:45:37 +02:00
|
|
|
/**
|
|
|
|
* Event loop driver which implements all basic operations to allow interoperability.
|
2016-12-28 14:18:14 +01:00
|
|
|
*
|
2017-02-09 09:50:23 +01:00
|
|
|
* Watchers (enabled or new watchers) MUST immediately be marked as enabled, but only be activated (i.e. callbacks can
|
|
|
|
* be called) right before the next tick. Callbacks of watchers MUST NOT be called in the tick they were enabled.
|
|
|
|
*
|
|
|
|
* All registered callbacks MUST NOT be called from a file with strict types enabled (`declare(strict_types=1)`).
|
2016-09-04 22:45:37 +02:00
|
|
|
*/
|
2017-03-10 21:58:46 +01:00
|
|
|
abstract class Driver {
|
2017-03-10 19:19:32 +01:00
|
|
|
// Don't use 1e3 / 1e6, they result in a float instead of int
|
|
|
|
const MILLISEC_PER_SEC = 1000;
|
|
|
|
const MICROSEC_PER_SEC = 1000000;
|
|
|
|
|
|
|
|
/** @var string */
|
|
|
|
private $nextId = "a";
|
|
|
|
|
2017-03-10 22:46:12 +01:00
|
|
|
/** @var \Amp\Internal\Watcher[] */
|
2017-03-10 19:19:32 +01:00
|
|
|
private $watchers = [];
|
|
|
|
|
2017-03-10 22:46:12 +01:00
|
|
|
/** @var \Amp\Internal\Watcher[] */
|
2017-03-10 19:19:32 +01:00
|
|
|
private $enableQueue = [];
|
|
|
|
|
2017-03-10 22:46:12 +01:00
|
|
|
/** @var \Amp\Internal\Watcher[] */
|
2017-03-10 19:19:32 +01:00
|
|
|
private $deferQueue = [];
|
|
|
|
|
2017-03-10 22:46:12 +01:00
|
|
|
/** @var \Amp\Internal\Watcher[] */
|
2017-03-10 19:19:32 +01:00
|
|
|
private $nextTickQueue = [];
|
|
|
|
|
|
|
|
/** @var callable|null */
|
|
|
|
private $errorHandler;
|
|
|
|
|
|
|
|
/** @var int */
|
|
|
|
private $running = 0;
|
|
|
|
|
|
|
|
/** @var array */
|
2016-08-04 21:08:52 +02:00
|
|
|
private $registry = [];
|
|
|
|
|
2016-02-17 16:25:39 +01:00
|
|
|
/**
|
2017-01-13 15:15:55 +01:00
|
|
|
* Run the event loop.
|
2017-01-13 15:14:01 +01:00
|
|
|
*
|
|
|
|
* One iteration of the loop is called one "tick". A tick covers the following steps:
|
|
|
|
*
|
2017-01-20 16:20:23 +01:00
|
|
|
* 1. Activate watchers created / enabled in the last tick / before `run()`.
|
2017-01-13 15:14:01 +01:00
|
|
|
* 2. Execute all enabled defer watchers.
|
2017-01-22 17:09:35 +01:00
|
|
|
* 3. Execute all due timer, pending signal and actionable stream callbacks, each only once per tick.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2016-05-20 12:49:31 +02:00
|
|
|
* The loop MUST continue to run until it is either stopped explicitly, no referenced watchers exist anymore, or an
|
|
|
|
* exception is thrown that cannot be handled. Exceptions that cannot be handled are exceptions thrown from an
|
2016-05-22 19:36:13 +02:00
|
|
|
* error handler or exceptions that would be passed to an error handler but none exists to handle them.
|
2016-05-20 12:49:31 +02:00
|
|
|
*
|
2016-02-17 16:25:39 +01:00
|
|
|
* @return void
|
|
|
|
*/
|
2017-03-10 19:19:32 +01:00
|
|
|
public function run() {
|
|
|
|
$previous = $this->running;
|
|
|
|
++$this->running;
|
|
|
|
|
|
|
|
try {
|
|
|
|
while ($this->running > $previous) {
|
|
|
|
if ($this->isEmpty()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
$this->tick();
|
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
$this->running = $previous;
|
|
|
|
}
|
|
|
|
}
|
2016-02-17 16:25:39 +01:00
|
|
|
|
2017-03-10 21:58:46 +01:00
|
|
|
/**
|
|
|
|
* @return bool True if no enabled and referenced watchers remain in the loop.
|
|
|
|
*/
|
|
|
|
private function isEmpty() {
|
|
|
|
foreach ($this->watchers as $watcher) {
|
|
|
|
if ($watcher->enabled && $watcher->referenced) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Executes a single tick of the event loop.
|
|
|
|
*/
|
|
|
|
private function tick() {
|
|
|
|
$this->deferQueue = \array_merge($this->deferQueue, $this->nextTickQueue);
|
|
|
|
$this->nextTickQueue = [];
|
|
|
|
|
|
|
|
$this->activate($this->enableQueue);
|
|
|
|
$this->enableQueue = [];
|
|
|
|
|
|
|
|
try {
|
|
|
|
foreach ($this->deferQueue as $watcher) {
|
|
|
|
if (!isset($this->deferQueue[$watcher->id])) {
|
|
|
|
continue; // Watcher disabled by another defer watcher.
|
|
|
|
}
|
|
|
|
|
|
|
|
unset($this->watchers[$watcher->id], $this->deferQueue[$watcher->id]);
|
|
|
|
|
|
|
|
$callback = $watcher->callback;
|
2017-03-10 23:03:41 +01:00
|
|
|
$result = $callback($watcher->id, $watcher->data);
|
|
|
|
|
2017-03-12 18:03:13 +01:00
|
|
|
if ($result === null) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-03-10 23:03:41 +01:00
|
|
|
if ($result instanceof \Generator) {
|
|
|
|
$result = new Coroutine($result);
|
2017-03-11 07:13:03 +01:00
|
|
|
} elseif ($result instanceof ReactPromise) {
|
|
|
|
$result = adapt($result);
|
2017-03-10 23:03:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if ($result instanceof Promise) {
|
|
|
|
rethrow($result);
|
|
|
|
}
|
2017-03-10 21:58:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
$this->dispatch(empty($this->nextTickQueue) && empty($this->enableQueue) && $this->running);
|
|
|
|
|
|
|
|
} catch (\Throwable $exception) {
|
|
|
|
if (null === $this->errorHandler) {
|
|
|
|
throw $exception;
|
|
|
|
}
|
|
|
|
|
|
|
|
$errorHandler = $this->errorHandler;
|
|
|
|
$errorHandler($exception);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Activates (enables) all the given watchers.
|
|
|
|
*
|
2017-03-10 22:46:12 +01:00
|
|
|
* @param \Amp\Internal\Watcher[] $watchers
|
2017-03-10 21:58:46 +01:00
|
|
|
*/
|
|
|
|
abstract protected function activate(array $watchers);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Dispatches any pending read/write, timer, and signal events.
|
|
|
|
*
|
|
|
|
* @param bool $blocking
|
|
|
|
*/
|
2017-03-10 22:46:12 +01:00
|
|
|
abstract protected function dispatch(bool $blocking);
|
2017-03-10 21:58:46 +01:00
|
|
|
|
2016-02-17 16:25:39 +01:00
|
|
|
/**
|
2016-03-14 11:56:31 +01:00
|
|
|
* Stop the event loop.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2016-05-20 12:49:31 +02:00
|
|
|
* When an event loop is stopped, it continues with its current tick and exits the loop afterwards. Multiple calls
|
|
|
|
* to stop MUST be ignored and MUST NOT raise an exception.
|
|
|
|
*
|
2016-02-17 16:25:39 +01:00
|
|
|
* @return void
|
|
|
|
*/
|
2017-03-10 19:19:32 +01:00
|
|
|
public function stop() {
|
|
|
|
--$this->running > 0 ?: $this->running = 0;
|
|
|
|
}
|
2016-02-17 16:25:39 +01:00
|
|
|
|
|
|
|
/**
|
2016-03-14 11:56:31 +01:00
|
|
|
* Defer the execution of a callback.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2017-02-09 15:47:44 +01:00
|
|
|
* The deferred callable MUST be executed before any other type of watcher in a tick. Order of enabling MUST be
|
|
|
|
* preserved when executing the callbacks.
|
|
|
|
*
|
|
|
|
* The created watcher MUST immediately be marked as enabled, but only be activated (i.e. callback can be called)
|
|
|
|
* right before the next tick. Callbacks of watchers MUST NOT be called in the tick they were enabled.
|
2016-05-20 12:49:31 +02:00
|
|
|
*
|
2017-03-10 21:58:46 +01:00
|
|
|
* @param callable (string $watcherId, mixed $data) $callback The callback to defer. The `$watcherId` will be
|
2016-06-26 16:15:01 +02:00
|
|
|
* invalidated before the callback call.
|
2016-09-25 13:22:01 +02:00
|
|
|
* @param mixed $data Arbitrary data given to the callback function as the `$data` parameter.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2016-05-28 15:57:07 +02:00
|
|
|
* @return string An unique identifier that can be used to cancel, enable or disable the watcher.
|
2016-02-17 16:25:39 +01:00
|
|
|
*/
|
2017-03-10 22:46:12 +01:00
|
|
|
public function defer(callable $callback, $data = null): string {
|
2017-03-10 19:19:32 +01:00
|
|
|
$watcher = new Watcher;
|
|
|
|
$watcher->type = Watcher::DEFER;
|
|
|
|
$watcher->id = $this->nextId++;
|
|
|
|
$watcher->callback = $callback;
|
|
|
|
$watcher->data = $data;
|
|
|
|
|
|
|
|
$this->watchers[$watcher->id] = $watcher;
|
|
|
|
$this->nextTickQueue[$watcher->id] = $watcher;
|
|
|
|
|
|
|
|
return $watcher->id;
|
|
|
|
}
|
2016-02-17 16:25:39 +01:00
|
|
|
|
|
|
|
/**
|
2016-05-20 12:49:31 +02:00
|
|
|
* Delay the execution of a callback.
|
|
|
|
*
|
2016-06-26 16:15:01 +02:00
|
|
|
* The delay is a minimum and approximate, accuracy is not guaranteed. Order of calls MUST be determined by which
|
2016-09-25 13:22:01 +02:00
|
|
|
* timers expire first, but timers with the same expiration time MAY be executed in any order.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2017-02-09 15:47:44 +01:00
|
|
|
* The created watcher MUST immediately be marked as enabled, but only be activated (i.e. callback can be called)
|
|
|
|
* right before the next tick. Callbacks of watchers MUST NOT be called in the tick they were enabled.
|
|
|
|
*
|
2017-03-10 21:58:46 +01:00
|
|
|
* @param int $delay The amount of time, in milliseconds, to delay the execution for.
|
|
|
|
* @param callable (string $watcherId, mixed $data) $callback The callback to delay. The `$watcherId` will be
|
2016-06-26 16:15:01 +02:00
|
|
|
* invalidated before the callback call.
|
2016-09-25 13:22:01 +02:00
|
|
|
* @param mixed $data Arbitrary data given to the callback function as the `$data` parameter.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2016-05-28 15:57:07 +02:00
|
|
|
* @return string An unique identifier that can be used to cancel, enable or disable the watcher.
|
2016-02-17 16:25:39 +01:00
|
|
|
*/
|
2017-03-10 22:46:12 +01:00
|
|
|
public function delay(int $delay, callable $callback, $data = null): string {
|
2017-03-10 19:19:32 +01:00
|
|
|
if ($delay < 0) {
|
|
|
|
throw new \InvalidArgumentException("Delay must be greater than or equal to zero");
|
|
|
|
}
|
|
|
|
|
|
|
|
$watcher = new Watcher;
|
|
|
|
$watcher->type = Watcher::DELAY;
|
|
|
|
$watcher->id = $this->nextId++;
|
|
|
|
$watcher->callback = $callback;
|
|
|
|
$watcher->value = $delay;
|
|
|
|
$watcher->data = $data;
|
|
|
|
|
|
|
|
$this->watchers[$watcher->id] = $watcher;
|
|
|
|
$this->enableQueue[$watcher->id] = $watcher;
|
|
|
|
|
|
|
|
return $watcher->id;
|
|
|
|
}
|
2016-02-17 16:25:39 +01:00
|
|
|
|
|
|
|
/**
|
2016-05-20 12:49:31 +02:00
|
|
|
* Repeatedly execute a callback.
|
|
|
|
*
|
2016-06-26 16:15:01 +02:00
|
|
|
* The interval between executions is a minimum and approximate, accuracy is not guaranteed. Order of calls MUST be
|
2016-10-25 17:29:09 +02:00
|
|
|
* determined by which timers expire first, but timers with the same expiration time MAY be executed in any order.
|
2016-05-23 20:29:10 +02:00
|
|
|
* The first execution is scheduled after the first interval period.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2017-02-09 15:47:44 +01:00
|
|
|
* The created watcher MUST immediately be marked as enabled, but only be activated (i.e. callback can be called)
|
|
|
|
* right before the next tick. Callbacks of watchers MUST NOT be called in the tick they were enabled.
|
|
|
|
*
|
2017-03-10 21:58:46 +01:00
|
|
|
* @param int $interval The time interval, in milliseconds, to wait between executions.
|
|
|
|
* @param callable (string $watcherId, mixed $data) $callback The callback to repeat.
|
2016-09-25 13:22:01 +02:00
|
|
|
* @param mixed $data Arbitrary data given to the callback function as the `$data` parameter.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2016-05-28 15:57:07 +02:00
|
|
|
* @return string An unique identifier that can be used to cancel, enable or disable the watcher.
|
2016-02-17 16:25:39 +01:00
|
|
|
*/
|
2017-03-10 22:46:12 +01:00
|
|
|
public function repeat(int $interval, callable $callback, $data = null): string {
|
2017-03-10 19:19:32 +01:00
|
|
|
if ($interval < 0) {
|
|
|
|
throw new \InvalidArgumentException("Interval must be greater than or equal to zero");
|
|
|
|
}
|
|
|
|
|
|
|
|
$watcher = new Watcher;
|
|
|
|
$watcher->type = Watcher::REPEAT;
|
|
|
|
$watcher->id = $this->nextId++;
|
|
|
|
$watcher->callback = $callback;
|
|
|
|
$watcher->value = $interval;
|
|
|
|
$watcher->data = $data;
|
|
|
|
|
|
|
|
$this->watchers[$watcher->id] = $watcher;
|
|
|
|
$this->enableQueue[$watcher->id] = $watcher;
|
|
|
|
|
|
|
|
return $watcher->id;
|
|
|
|
}
|
2016-02-17 16:25:39 +01:00
|
|
|
|
|
|
|
/**
|
2016-09-23 16:02:23 +02:00
|
|
|
* Execute a callback when a stream resource becomes readable or is closed for reading.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2016-10-24 17:08:53 +02:00
|
|
|
* Warning: Closing resources locally, e.g. with `fclose`, might not invoke the callback. Be sure to `cancel` the
|
2016-10-25 17:29:09 +02:00
|
|
|
* watcher when closing the resource locally. Drivers MAY choose to notify the user if there are watchers on invalid
|
2016-10-24 17:21:12 +02:00
|
|
|
* resources, but are not required to, due to the high performance impact. Watchers on closed resources are
|
|
|
|
* therefore undefined behavior.
|
2016-10-24 17:08:53 +02:00
|
|
|
*
|
2016-10-25 17:29:09 +02:00
|
|
|
* Multiple watchers on the same stream MAY be executed in any order.
|
2016-06-04 18:27:30 +02:00
|
|
|
*
|
2017-02-09 15:47:44 +01:00
|
|
|
* The created watcher MUST immediately be marked as enabled, but only be activated (i.e. callback can be called)
|
|
|
|
* right before the next tick. Callbacks of watchers MUST NOT be called in the tick they were enabled.
|
|
|
|
*
|
2016-03-14 11:56:31 +01:00
|
|
|
* @param resource $stream The stream to monitor.
|
2017-03-10 21:58:46 +01:00
|
|
|
* @param callable (string $watcherId, resource $stream, mixed $data) $callback The callback to execute.
|
|
|
|
* @param mixed $data Arbitrary data given to the callback function as the `$data` parameter.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2016-05-28 15:57:07 +02:00
|
|
|
* @return string An unique identifier that can be used to cancel, enable or disable the watcher.
|
2016-02-17 16:25:39 +01:00
|
|
|
*/
|
2017-03-10 22:46:12 +01:00
|
|
|
public function onReadable($stream, callable $callback, $data = null): string {
|
2017-03-10 19:19:32 +01:00
|
|
|
$watcher = new Watcher;
|
|
|
|
$watcher->type = Watcher::READABLE;
|
|
|
|
$watcher->id = $this->nextId++;
|
|
|
|
$watcher->callback = $callback;
|
|
|
|
$watcher->value = $stream;
|
|
|
|
$watcher->data = $data;
|
|
|
|
|
|
|
|
$this->watchers[$watcher->id] = $watcher;
|
|
|
|
$this->enableQueue[$watcher->id] = $watcher;
|
|
|
|
|
|
|
|
return $watcher->id;
|
|
|
|
}
|
2016-02-17 16:25:39 +01:00
|
|
|
|
|
|
|
/**
|
2016-09-23 16:02:23 +02:00
|
|
|
* Execute a callback when a stream resource becomes writable or is closed for writing.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2016-10-24 17:08:53 +02:00
|
|
|
* Warning: Closing resources locally, e.g. with `fclose`, might not invoke the callback. Be sure to `cancel` the
|
2016-10-25 17:29:09 +02:00
|
|
|
* watcher when closing the resource locally. Drivers MAY choose to notify the user if there are watchers on invalid
|
2016-10-24 17:21:12 +02:00
|
|
|
* resources, but are not required to, due to the high performance impact. Watchers on closed resources are
|
|
|
|
* therefore undefined behavior.
|
2016-10-24 17:08:53 +02:00
|
|
|
*
|
2016-10-25 17:29:09 +02:00
|
|
|
* Multiple watchers on the same stream MAY be executed in any order.
|
2016-06-04 18:27:30 +02:00
|
|
|
*
|
2017-02-09 15:47:44 +01:00
|
|
|
* The created watcher MUST immediately be marked as enabled, but only be activated (i.e. callback can be called)
|
|
|
|
* right before the next tick. Callbacks of watchers MUST NOT be called in the tick they were enabled.
|
|
|
|
*
|
2016-03-14 11:56:31 +01:00
|
|
|
* @param resource $stream The stream to monitor.
|
2017-03-10 21:58:46 +01:00
|
|
|
* @param callable (string $watcherId, resource $stream, mixed $data) $callback The callback to execute.
|
|
|
|
* @param mixed $data Arbitrary data given to the callback function as the `$data` parameter.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2016-05-28 15:57:07 +02:00
|
|
|
* @return string An unique identifier that can be used to cancel, enable or disable the watcher.
|
2016-02-17 16:25:39 +01:00
|
|
|
*/
|
2017-03-10 22:46:12 +01:00
|
|
|
public function onWritable($stream, callable $callback, $data = null): string {
|
2017-03-10 19:19:32 +01:00
|
|
|
$watcher = new Watcher;
|
|
|
|
$watcher->type = Watcher::WRITABLE;
|
|
|
|
$watcher->id = $this->nextId++;
|
|
|
|
$watcher->callback = $callback;
|
|
|
|
$watcher->value = $stream;
|
|
|
|
$watcher->data = $data;
|
|
|
|
|
|
|
|
$this->watchers[$watcher->id] = $watcher;
|
|
|
|
$this->enableQueue[$watcher->id] = $watcher;
|
|
|
|
|
|
|
|
return $watcher->id;
|
|
|
|
}
|
2016-02-17 16:25:39 +01:00
|
|
|
|
|
|
|
/**
|
2016-03-14 11:56:31 +01:00
|
|
|
* Execute a callback when a signal is received.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2016-10-24 17:08:53 +02:00
|
|
|
* Warning: Installing the same signal on different instances of this interface is deemed undefined behavior.
|
2016-10-25 17:29:09 +02:00
|
|
|
* Implementations MAY try to detect this, if possible, but are not required to. This is due to technical
|
2016-09-04 22:45:37 +02:00
|
|
|
* limitations of the signals being registered globally per process.
|
2016-07-19 18:10:51 +02:00
|
|
|
*
|
2016-10-25 17:29:09 +02:00
|
|
|
* Multiple watchers on the same signal MAY be executed in any order.
|
2016-10-24 17:08:53 +02:00
|
|
|
*
|
2017-02-09 15:47:44 +01:00
|
|
|
* The created watcher MUST immediately be marked as enabled, but only be activated (i.e. callback can be called)
|
|
|
|
* right before the next tick. Callbacks of watchers MUST NOT be called in the tick they were enabled.
|
|
|
|
*
|
2017-03-10 21:58:46 +01:00
|
|
|
* @param int $signo The signal number to monitor.
|
|
|
|
* @param callable (string $watcherId, int $signo, mixed $data) $callback The callback to execute.
|
2016-05-15 00:13:35 +02:00
|
|
|
* @param mixed $data Arbitrary data given to the callback function as the $data parameter.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2016-05-28 15:57:07 +02:00
|
|
|
* @return string An unique identifier that can be used to cancel, enable or disable the watcher.
|
2016-05-26 17:43:03 +02:00
|
|
|
*
|
2016-07-15 22:17:24 +02:00
|
|
|
* @throws UnsupportedFeatureException If signal handling is not supported.
|
2016-02-17 16:25:39 +01:00
|
|
|
*/
|
2017-03-10 22:46:12 +01:00
|
|
|
public function onSignal(int $signo, callable $callback, $data = null): string {
|
2017-03-10 19:19:32 +01:00
|
|
|
$watcher = new Watcher;
|
|
|
|
$watcher->type = Watcher::SIGNAL;
|
|
|
|
$watcher->id = $this->nextId++;
|
|
|
|
$watcher->callback = $callback;
|
|
|
|
$watcher->value = $signo;
|
|
|
|
$watcher->data = $data;
|
|
|
|
|
|
|
|
$this->watchers[$watcher->id] = $watcher;
|
|
|
|
$this->enableQueue[$watcher->id] = $watcher;
|
|
|
|
|
|
|
|
return $watcher->id;
|
|
|
|
}
|
2016-02-17 16:25:39 +01:00
|
|
|
|
|
|
|
/**
|
2017-02-09 15:47:44 +01:00
|
|
|
* Enable a watcher to be active starting in the next tick.
|
2016-07-15 22:17:24 +02:00
|
|
|
*
|
2017-02-09 09:50:23 +01:00
|
|
|
* Watchers MUST immediately be marked as enabled, but only be activated (i.e. callbacks can be called) right before
|
|
|
|
* the next tick. Callbacks of watchers MUST NOT be called in the tick they were enabled.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2016-05-19 17:21:26 +02:00
|
|
|
* @param string $watcherId The watcher identifier.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2016-03-01 01:52:59 +01:00
|
|
|
* @return void
|
2016-05-28 15:57:07 +02:00
|
|
|
*
|
2016-07-15 22:17:24 +02:00
|
|
|
* @throws InvalidWatcherException If the watcher identifier is invalid.
|
2016-02-17 16:25:39 +01:00
|
|
|
*/
|
2017-03-10 19:19:32 +01:00
|
|
|
public function enable(string $watcherId) {
|
|
|
|
if (!isset($this->watchers[$watcherId])) {
|
|
|
|
throw new InvalidWatcherException($watcherId, "Cannot enable an invalid watcher identifier: '{$watcherId}'");
|
|
|
|
}
|
|
|
|
|
|
|
|
$watcher = $this->watchers[$watcherId];
|
|
|
|
|
|
|
|
if ($watcher->enabled) {
|
|
|
|
return; // Watcher already enabled.
|
|
|
|
}
|
|
|
|
|
|
|
|
$watcher->enabled = true;
|
|
|
|
|
|
|
|
switch ($watcher->type) {
|
|
|
|
case Watcher::DEFER:
|
|
|
|
$this->nextTickQueue[$watcher->id] = $watcher;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
$this->enableQueue[$watcher->id] = $watcher;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2016-03-01 01:52:59 +01:00
|
|
|
|
2017-03-10 21:58:46 +01:00
|
|
|
/**
|
|
|
|
* Cancel a watcher.
|
|
|
|
*
|
|
|
|
* This will detatch the event loop from all resources that are associated to the watcher. After this operation the
|
|
|
|
* watcher is permanently invalid. Calling this function MUST NOT fail, even if passed an invalid watcher.
|
|
|
|
*
|
|
|
|
* @param string $watcherId The watcher identifier.
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function cancel(string $watcherId) {
|
|
|
|
$this->disable($watcherId);
|
|
|
|
unset($this->watchers[$watcherId]);
|
|
|
|
}
|
|
|
|
|
2016-03-01 01:52:59 +01:00
|
|
|
/**
|
2017-02-09 15:47:44 +01:00
|
|
|
* Disable a watcher immediately.
|
|
|
|
*
|
|
|
|
* A watcher MUST be disabled immediately, e.g. if a defer watcher disables a later defer watcher, the second defer
|
|
|
|
* watcher isn't executed in this tick.
|
2016-09-25 13:22:01 +02:00
|
|
|
*
|
|
|
|
* Disabling a watcher MUST NOT invalidate the watcher. Calling this function MUST NOT fail, even if passed an
|
|
|
|
* invalid watcher.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2016-05-19 17:21:26 +02:00
|
|
|
* @param string $watcherId The watcher identifier.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2016-03-01 01:52:59 +01:00
|
|
|
* @return void
|
|
|
|
*/
|
2017-03-10 19:19:32 +01:00
|
|
|
public function disable(string $watcherId) {
|
|
|
|
if (!isset($this->watchers[$watcherId])) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$watcher = $this->watchers[$watcherId];
|
|
|
|
|
|
|
|
if (!$watcher->enabled) {
|
|
|
|
return; // Watcher already disabled.
|
|
|
|
}
|
|
|
|
|
|
|
|
$watcher->enabled = false;
|
|
|
|
$id = $watcher->id;
|
|
|
|
|
|
|
|
switch ($watcher->type) {
|
|
|
|
case Watcher::DEFER:
|
|
|
|
if (isset($this->nextTickQueue[$id])) {
|
|
|
|
// Watcher was only queued to be enabled.
|
|
|
|
unset($this->nextTickQueue[$id]);
|
|
|
|
} else {
|
|
|
|
unset($this->deferQueue[$id]);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
if (isset($this->enableQueue[$id])) {
|
|
|
|
// Watcher was only queued to be enabled.
|
|
|
|
unset($this->enableQueue[$id]);
|
|
|
|
} else {
|
|
|
|
$this->deactivate($watcher);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2016-03-01 01:52:59 +01:00
|
|
|
|
|
|
|
/**
|
2017-03-10 21:58:46 +01:00
|
|
|
* Deactivates (disables) the given watcher.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2017-03-10 22:46:12 +01:00
|
|
|
* @param \Amp\Internal\Watcher $watcher
|
2016-03-01 01:52:59 +01:00
|
|
|
*/
|
2017-03-10 21:58:46 +01:00
|
|
|
abstract protected function deactivate(Watcher $watcher);
|
2016-03-23 10:47:18 +01:00
|
|
|
|
|
|
|
/**
|
2016-05-15 00:19:05 +02:00
|
|
|
* Reference a watcher.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2016-05-20 12:49:31 +02:00
|
|
|
* This will keep the event loop alive whilst the watcher is still being monitored. Watchers have this state by
|
|
|
|
* default.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2016-05-19 17:21:26 +02:00
|
|
|
* @param string $watcherId The watcher identifier.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2016-03-23 10:47:18 +01:00
|
|
|
* @return void
|
2016-05-28 15:57:07 +02:00
|
|
|
*
|
2016-07-15 22:17:24 +02:00
|
|
|
* @throws InvalidWatcherException If the watcher identifier is invalid.
|
2016-03-23 10:47:18 +01:00
|
|
|
*/
|
2017-03-10 19:19:32 +01:00
|
|
|
public function reference(string $watcherId) {
|
|
|
|
if (!isset($this->watchers[$watcherId])) {
|
|
|
|
throw new InvalidWatcherException($watcherId, "Cannot reference an invalid watcher identifier: '{$watcherId}'");
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->watchers[$watcherId]->referenced = true;
|
|
|
|
}
|
2016-03-23 10:47:18 +01:00
|
|
|
|
|
|
|
/**
|
2016-05-15 00:19:05 +02:00
|
|
|
* Unreference a watcher.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2016-05-20 12:49:31 +02:00
|
|
|
* The event loop should exit the run method when only unreferenced watchers are still being monitored. Watchers
|
|
|
|
* are all referenced by default.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2016-05-19 17:21:26 +02:00
|
|
|
* @param string $watcherId The watcher identifier.
|
2016-05-12 17:57:02 +02:00
|
|
|
*
|
2016-03-23 10:47:18 +01:00
|
|
|
* @return void
|
2016-05-28 15:57:07 +02:00
|
|
|
*
|
2016-07-15 22:17:24 +02:00
|
|
|
* @throws InvalidWatcherException If the watcher identifier is invalid.
|
2016-03-23 10:47:18 +01:00
|
|
|
*/
|
2017-03-10 19:19:32 +01:00
|
|
|
public function unreference(string $watcherId) {
|
|
|
|
if (!isset($this->watchers[$watcherId])) {
|
|
|
|
throw new InvalidWatcherException($watcherId, "Cannot unreference an invalid watcher identifier: '{$watcherId}'");
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->watchers[$watcherId]->referenced = false;
|
|
|
|
}
|
2016-07-15 22:17:24 +02:00
|
|
|
|
2016-05-27 17:45:45 +02:00
|
|
|
/**
|
2016-09-25 13:22:01 +02:00
|
|
|
* Stores information in the loop bound registry.
|
2016-05-27 17:45:45 +02:00
|
|
|
*
|
2016-09-25 13:22:01 +02:00
|
|
|
* This can be used to store loop bound information. Stored information is package private. Packages MUST NOT
|
|
|
|
* retrieve the stored state of other packages. Packages MUST use the following prefix for keys: `vendor.package.`
|
2016-05-27 17:45:45 +02:00
|
|
|
*
|
2016-09-04 22:45:37 +02:00
|
|
|
* @param string $key The namespaced storage key.
|
2017-03-10 21:58:46 +01:00
|
|
|
* @param mixed $value The value to be stored.
|
2016-05-27 17:45:45 +02:00
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
2017-03-10 21:58:46 +01:00
|
|
|
final public function setState(string $key, $value) {
|
2016-08-04 21:08:52 +02:00
|
|
|
if ($value === null) {
|
|
|
|
unset($this->registry[$key]);
|
|
|
|
} else {
|
|
|
|
$this->registry[$key] = $value;
|
|
|
|
}
|
|
|
|
}
|
2016-05-27 17:45:45 +02:00
|
|
|
|
|
|
|
/**
|
2016-09-25 13:22:01 +02:00
|
|
|
* Gets information stored bound to the loop.
|
2016-05-27 17:45:45 +02:00
|
|
|
*
|
2016-09-25 13:22:01 +02:00
|
|
|
* Stored information is package private. Packages MUST NOT retrieve the stored state of other packages. Packages
|
|
|
|
* MUST use the following prefix for keys: `vendor.package.`
|
2016-05-27 17:45:45 +02:00
|
|
|
*
|
2016-09-04 22:45:37 +02:00
|
|
|
* @param string $key The namespaced storage key.
|
2016-05-27 17:45:45 +02:00
|
|
|
*
|
2016-09-25 13:22:01 +02:00
|
|
|
* @return mixed The previously stored value or `null` if it doesn't exist.
|
2016-05-27 17:45:45 +02:00
|
|
|
*/
|
2017-03-10 21:58:46 +01:00
|
|
|
final public function getState(string $key) {
|
2016-08-04 21:08:52 +02:00
|
|
|
return isset($this->registry[$key]) ? $this->registry[$key] : null;
|
|
|
|
}
|
2016-05-13 23:41:54 +02:00
|
|
|
|
2016-05-18 13:12:42 +02:00
|
|
|
/**
|
|
|
|
* Set a callback to be executed when an error occurs.
|
|
|
|
*
|
2016-10-24 20:43:21 +02:00
|
|
|
* The callback receives the error as the first and only parameter. The return value of the callback gets ignored.
|
|
|
|
* If it can't handle the error, it MUST throw the error. Errors thrown by the callback or during its invocation
|
|
|
|
* MUST be thrown into the `run` loop and stop the driver.
|
|
|
|
*
|
2016-05-18 13:12:42 +02:00
|
|
|
* Subsequent calls to this method will overwrite the previous handler.
|
|
|
|
*
|
2017-03-10 21:58:46 +01:00
|
|
|
* @param callable (\Throwable|\Exception $error)|null $callback The callback to execute. `null` will clear the
|
2016-09-25 13:22:01 +02:00
|
|
|
* current handler.
|
2016-05-18 13:12:42 +02:00
|
|
|
*
|
2016-12-22 17:50:25 +01:00
|
|
|
* @return callable(\Throwable|\Exception $error)|null The previous handler, `null` if there was none.
|
2016-05-18 13:12:42 +02:00
|
|
|
*/
|
2017-03-10 19:19:32 +01:00
|
|
|
public function setErrorHandler(callable $callback = null) {
|
|
|
|
$previous = $this->errorHandler;
|
|
|
|
$this->errorHandler = $callback;
|
|
|
|
return $previous;
|
|
|
|
}
|
2016-05-18 13:12:42 +02:00
|
|
|
|
2017-03-10 21:58:46 +01:00
|
|
|
/**
|
|
|
|
* Get the underlying loop handle.
|
|
|
|
*
|
|
|
|
* Example: the `uv_loop` resource for `libuv` or the `EvLoop` object for `libev` or `null` for a native driver.
|
|
|
|
*
|
|
|
|
* Note: This function is *not* exposed in the `Loop` class. Users shall access it directly on the respective loop
|
|
|
|
* instance.
|
|
|
|
*
|
|
|
|
* @return null|object|resource The loop handle the event loop operates on. `null` if there is none.
|
|
|
|
*/
|
|
|
|
abstract public function getHandle();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the same array of data as getInfo().
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function __debugInfo() {
|
|
|
|
return $this->getInfo();
|
|
|
|
}
|
|
|
|
|
2016-05-21 11:00:01 +02:00
|
|
|
/**
|
|
|
|
* Retrieve an associative array of information about the event loop driver.
|
|
|
|
*
|
2016-07-15 22:17:24 +02:00
|
|
|
* The returned array MUST contain the following data describing the driver's currently registered watchers:
|
2016-05-21 11:00:01 +02:00
|
|
|
*
|
2016-09-25 13:22:01 +02:00
|
|
|
* [
|
2017-01-08 11:40:11 +01:00
|
|
|
* "defer" => ["enabled" => int, "disabled" => int],
|
|
|
|
* "delay" => ["enabled" => int, "disabled" => int],
|
|
|
|
* "repeat" => ["enabled" => int, "disabled" => int],
|
|
|
|
* "on_readable" => ["enabled" => int, "disabled" => int],
|
|
|
|
* "on_writable" => ["enabled" => int, "disabled" => int],
|
|
|
|
* "on_signal" => ["enabled" => int, "disabled" => int],
|
|
|
|
* "enabled_watchers" => ["referenced" => int, "unreferenced" => int],
|
2017-03-10 19:19:32 +01:00
|
|
|
* "running" => bool
|
2016-09-25 13:22:01 +02:00
|
|
|
* ];
|
|
|
|
*
|
|
|
|
* Implementations MAY optionally add more information in the array but at minimum the above `key => value` format
|
2016-07-15 22:17:24 +02:00
|
|
|
* MUST always be provided.
|
2016-05-21 11:00:01 +02:00
|
|
|
*
|
2016-12-23 11:05:37 +01:00
|
|
|
* @return array Statistics about the loop in the described format.
|
2016-05-21 11:00:01 +02:00
|
|
|
*/
|
2017-03-10 22:46:12 +01:00
|
|
|
public function getInfo(): array {
|
2017-03-10 19:19:32 +01:00
|
|
|
$watchers = [
|
2017-03-10 21:58:46 +01:00
|
|
|
"referenced" => 0,
|
2017-03-10 19:19:32 +01:00
|
|
|
"unreferenced" => 0,
|
|
|
|
];
|
|
|
|
|
|
|
|
$defer = $delay = $repeat = $onReadable = $onWritable = $onSignal = [
|
2017-03-10 21:58:46 +01:00
|
|
|
"enabled" => 0,
|
2017-03-10 19:19:32 +01:00
|
|
|
"disabled" => 0,
|
|
|
|
];
|
|
|
|
|
|
|
|
foreach ($this->watchers as $watcher) {
|
|
|
|
switch ($watcher->type) {
|
2017-03-10 21:58:46 +01:00
|
|
|
case Watcher::READABLE:
|
|
|
|
$array = &$onReadable;
|
|
|
|
break;
|
|
|
|
case Watcher::WRITABLE:
|
|
|
|
$array = &$onWritable;
|
|
|
|
break;
|
|
|
|
case Watcher::SIGNAL:
|
|
|
|
$array = &$onSignal;
|
|
|
|
break;
|
|
|
|
case Watcher::DEFER:
|
|
|
|
$array = &$defer;
|
|
|
|
break;
|
|
|
|
case Watcher::DELAY:
|
|
|
|
$array = &$delay;
|
|
|
|
break;
|
|
|
|
case Watcher::REPEAT:
|
|
|
|
$array = &$repeat;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
throw new \DomainException("Unknown watcher type");
|
2017-03-10 19:19:32 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if ($watcher->enabled) {
|
|
|
|
++$array["enabled"];
|
|
|
|
|
|
|
|
if ($watcher->referenced) {
|
|
|
|
++$watchers["referenced"];
|
|
|
|
} else {
|
|
|
|
++$watchers["unreferenced"];
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
++$array["disabled"];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return [
|
|
|
|
"enabled_watchers" => $watchers,
|
2017-03-10 21:58:46 +01:00
|
|
|
"defer" => $defer,
|
|
|
|
"delay" => $delay,
|
|
|
|
"repeat" => $repeat,
|
|
|
|
"on_readable" => $onReadable,
|
|
|
|
"on_writable" => $onWritable,
|
|
|
|
"on_signal" => $onSignal,
|
|
|
|
"running" => (bool) $this->running,
|
2017-03-10 19:19:32 +01:00
|
|
|
];
|
|
|
|
}
|
|
|
|
}
|