1
0
mirror of https://github.com/danog/parallel.git synced 2025-01-22 22:11:11 +01:00
parallel/lib/Forking/Fork.php
2016-08-28 23:47:15 -05:00

351 lines
10 KiB
PHP

<?php declare(strict_types = 1);
namespace Amp\Parallel\Forking;
use Amp\Coroutine;
use Amp\Parallel\{
ContextException,
ChannelException,
Process,
SerializationException,
StatusError,
Strand,
SynchronizationError
};
use Amp\Parallel\Sync\{ Channel, ChannelledStream };
use Amp\Parallel\Sync\Internal\{ ExitFailure, ExitStatus, ExitSuccess };
use Amp\Socket\Socket;
use Interop\Async\Awaitable;
/**
* Implements a UNIX-compatible context using forked processes.
*/
class Fork implements Process, Strand {
/** @var \Amp\Parallel\Sync\Channel A channel for communicating with the child. */
private $channel;
/** @var \Amp\Socket\Socket */
private $pipe;
/** @var int */
private $pid = 0;
/** @var callable */
private $function;
/** @var mixed[] */
private $args;
/** @var int */
private $oid = 0;
/**
* Checks if forking is enabled.
*
* @return bool True if forking is enabled, otherwise false.
*/
public static function supported(): bool {
return \extension_loaded('pcntl');
}
/**
* Spawns a new forked process and runs it.
*
* @param callable $function A callable to invoke in the process.
*
* @return \Amp\Parallel\Forking\Fork The process object that was spawned.
*/
public static function spawn(callable $function, ...$args): self {
$fork = new self($function, ...$args);
$fork->start();
return $fork;
}
public function __construct(callable $function, ...$args) {
if (!self::supported()) {
throw new \Error("The pcntl extension is required to create forks.");
}
$this->function = $function;
$this->args = $args;
}
public function __clone() {
$this->pid = 0;
$this->oid = 0;
$this->pipe = null;
$this->channel = null;
}
public function __destruct() {
if (0 !== $this->pid && \posix_getpid() === $this->oid) { // Only kill in owner process.
$this->kill(); // Will only terminate if the process is still running.
}
}
/**
* Checks if the context is running.
*
* @return bool True if the context is running, otherwise false.
*/
public function isRunning(): bool {
return 0 !== $this->pid && false !== \posix_getpgid($this->pid);
}
/**
* Gets the forked process's process ID.
*
* @return int The process ID.
*/
public function getPid(): int {
return $this->pid;
}
/**
* Gets the fork's scheduling priority as a percentage.
*
* The priority is a float between 0 and 1 that indicates the relative priority for the forked process, where 0 is
* very low priority, 1 is very high priority, and 0.5 is considered a "normal" priority. The value is based on the
* forked process's "nice" value. The priority affects the operating system's scheduling of processes. How much the
* priority actually affects the amount of CPU time the process gets is ultimately system-specific.
*
* @return float A priority value between 0 and 1.
*
* @throws ContextException If the operation failed.
*
* @see Fork::setPriority()
* @see http://linux.die.net/man/2/getpriority
*/
public function getPriority(): float {
if (($nice = \pcntl_getpriority($this->pid)) === false) {
throw new ContextException('Failed to get the fork\'s priority.');
}
return (19 - $nice) / 39;
}
/**
* Sets the fork's scheduling priority as a percentage.
*
* Note that on many systems, only the superuser can increase the priority of a process.
*
* @param float $priority A priority value between 0 and 1.
*
* @throws \Error If the given priority is an invalid value.
* @throws ContextException If the operation failed.
*
* @see Fork::getPriority()
*/
public function setPriority(float $priority): float {
if ($priority < 0 || $priority > 1) {
throw new \Error('Priority value must be between 0.0 and 1.0.');
}
$nice = \round(19 - ($priority * 39));
if (!\pcntl_setpriority($nice, $this->pid, \PRIO_PROCESS)) {
throw new ContextException('Failed to set the fork\'s priority.');
}
}
/**
* Starts the context execution.
*
* @throws \Amp\Parallel\ContextException If forking fails.
* @throws \Amp\Socket\SocketException If creating a socket pair fails.
*/
public function start() {
if (0 !== $this->oid) {
throw new StatusError('The context has already been started.');
}
list($parent, $child) = \Amp\Socket\pair();
switch ($pid = \pcntl_fork()) {
case -1: // Failure
throw new ContextException('Could not fork process!');
case 0: // Child
// @codeCoverageIgnoreStart
\fclose($child);
try {
\Amp\execute(function () use ($parent) {
$channel = new ChannelledStream(new Socket($parent));
return $this->execute($channel);
});
$code = 0;
} catch (\Throwable $exception) {
$code = 1;
}
exit($code);
// @codeCoverageIgnoreEnd
default: // Parent
$this->pid = $pid;
$this->oid = \posix_getpid();
$this->channel = new ChannelledStream($this->pipe = new Socket($child));
\fclose($parent);
}
}
/**
* @coroutine
*
* This method is run only on the child.
*
* @param \Amp\Parallel\Sync\Channel $channel
*
* @return \Generator
*
* @codeCoverageIgnore Only executed in the child.
*/
private function execute(Channel $channel): \Generator {
try {
if ($this->function instanceof \Closure) {
$function = $this->function->bindTo($channel, Channel::class);
}
if (empty($function)) {
$function = $this->function;
}
$result = $function(...$this->args);
if ($result instanceof \Generator) {
$result = new Coroutine($result);
}
if ($result instanceof Awaitable) {
$result = yield $result;
}
$result = new ExitSuccess($result);
} catch (\Throwable $exception) {
$result = new ExitFailure($exception);
}
// Attempt to return the result.
try {
try {
return yield $channel->send($result);
} catch (SerializationException $exception) {
// Serializing the result failed. Send the reason why.
return yield $channel->send(new ExitFailure($exception));
}
} catch (ChannelException $exception) {
// The result was not sendable! The parent context must have died or killed the context.
return 0;
}
}
/**
* {@inheritdoc}
*/
public function kill() {
if ($this->isRunning()) {
// Forcefully kill the process using SIGKILL.
\posix_kill($this->pid, SIGKILL);
}
if (null !== $this->pipe && $this->pipe->isReadable()) {
$this->pipe->close();
}
// "Detach" from the process and let it die asynchronously.
$this->pid = 0;
$this->channel = null;
}
/**
* @param int $signo
*
* @throws \Amp\Parallel\StatusError
*/
public function signal(int $signo) {
if (0 === $this->pid) {
throw new StatusError('The fork has not been started or has already finished.');
}
\posix_kill($this->pid, (int) $signo);
}
/**
* Gets a promise that resolves when the context ends and joins with the
* parent context.
*
* @return \Interop\Async\Awaitable<int>
*
* @throws \Amp\Parallel\StatusError Thrown if the context has not been started.
* @throws \Amp\Parallel\SynchronizationError Thrown if an exit status object is not received.
*/
public function join(): Awaitable {
if (null === $this->channel) {
throw new StatusError('The fork has not been started or has already finished.');
}
return new Coroutine($this->doJoin());
}
/**
* @coroutine
*
* @return \Generator
*
* @throws \Amp\Parallel\SynchronizationError
*/
private function doJoin(): \Generator {
try {
$response = yield $this->channel->receive();
if (!$response instanceof ExitStatus) {
throw new SynchronizationError(\sprintf(
'Did not receive an exit status from fork. Instead received data of type %s',
\is_object($response) ? \get_class($response) : \gettype($response)
));
}
return $response->getResult();
} finally {
$this->kill();
}
}
/**
* {@inheritdoc}
*/
public function receive(): Awaitable {
if (null === $this->channel) {
throw new StatusError('The process has not been started.');
}
return \Amp\pipe($this->channel->receive(), static function ($data) {
if ($data instanceof ExitStatus) {
$data = $data->getResult();
throw new SynchronizationError(\sprintf(
'Forked process unexpectedly exited with result of type: %s',
\is_object($data) ? \get_class($data) : \gettype($data)
));
}
return $data;
});
}
/**
* {@inheritdoc}
*/
public function send($data): Awaitable {
if (null === $this->channel) {
throw new StatusError('The fork has not been started or has already finished.');
}
if ($data instanceof ExitStatus) {
throw new \Error('Cannot send exit status objects.');
}
return $this->channel->send($data);
}
}