1
0
mirror of https://github.com/danog/parallel.git synced 2024-12-03 18:17:52 +01:00
parallel/lib/Threading/Internal/Thread.php

160 lines
4.9 KiB
PHP
Raw Normal View History

2016-12-30 02:16:04 +01:00
<?php
2016-08-23 23:47:40 +02:00
namespace Amp\Parallel\Threading\Internal;
2016-08-18 18:04:48 +02:00
use Amp\Coroutine;
2016-08-23 23:47:40 +02:00
use Amp\Parallel\{ ChannelException, SerializationException };
2016-08-31 01:27:14 +02:00
use Amp\Parallel\Sync\{ Channel, ChannelledSocket, Internal\ExitFailure, Internal\ExitSuccess };
use AsyncInterop\{ Loop, Promise };
/**
* An internal thread that executes a given function concurrently.
*
* @internal
*/
2016-08-18 18:04:48 +02:00
class Thread extends \Thread {
const KILL_CHECK_FREQUENCY = 250;
2017-01-17 12:14:06 +01:00
const AUTOLOAD_FILENAME = "autoload.php";
/** @var string */
private static $autoloadPath;
2016-08-26 17:10:03 +02:00
/** @var callable The function to execute in the thread. */
private $function;
2016-08-26 17:10:03 +02:00
/** @var mixed[] Arguments to pass to the function. */
private $args;
2016-08-26 17:10:03 +02:00
/** @var resource */
private $socket;
2016-08-26 17:10:03 +02:00
/** @var bool */
private $killed = false;
/**
* Creates a new thread object.
*
* @param resource $socket IPC communication socket.
* @param callable $function The function to execute in the thread.
* @param mixed[] $args Arguments to pass to the function.
*/
2016-08-18 18:04:48 +02:00
public function __construct($socket, callable $function, array $args = []) {
$this->function = $function;
$this->args = $args;
$this->socket = $socket;
if (self::$autoloadPath === null) { // Determine path to composer autoload.php
2017-01-17 12:14:06 +01:00
$files = [
dirname(__DIR__, 3) . \DIRECTORY_SEPARATOR . "vendor" . \DIRECTORY_SEPARATOR . self::AUTOLOAD_FILENAME,
dirname(__DIR__, 5) . \DIRECTORY_SEPARATOR . self::AUTOLOAD_FILENAME,
];
foreach ($files as $file) {
foreach (\get_included_files() as $path) {
if (\substr($path, -\strlen($file)) === $file) {
self::$autoloadPath = $path;
return;
}
}
}
2017-01-16 20:45:32 +01:00
throw new \Error(\sprintf("Could not locate %s", self::AUTOLOAD_FILENAME));
}
}
/**
* Runs the thread code and the initialized function.
*
* @codeCoverageIgnore Only executed in thread.
*/
2016-08-18 18:04:48 +02:00
public function run() {
/* First thing we need to do is re-initialize the class autoloader. If
* we don't do this first, any object of a class that was loaded after
* the thread started will just be garbage data and unserializable
* values (like resources) will be lost. This happens even with
* thread-safe objects.
*/
// Protect scope by using an unbound closure (protects static access as well).
$autoloadPath = self::$autoloadPath;
(static function () use ($autoloadPath) { require $autoloadPath; })->bindTo(null, null)();
// At this point, the thread environment has been prepared so begin using the thread.
try {
Loop::execute(\Amp\wrap(function () {
2016-08-31 01:27:14 +02:00
$channel = new ChannelledSocket($this->socket, $this->socket, false);
$watcher = Loop::repeat(self::KILL_CHECK_FREQUENCY, function () {
2016-08-18 18:04:48 +02:00
if ($this->killed) {
Loop::stop();
2016-08-18 18:04:48 +02:00
}
});
Loop::unreference($watcher);
return $this->execute($channel);
}));
2016-01-27 08:03:37 +01:00
} catch (\Throwable $exception) {
2016-08-18 18:04:48 +02:00
return 1;
}
2016-08-18 18:04:48 +02:00
return 0;
}
/**
* Sets a local variable to true so the running event loop can check for a kill signal.
*/
2016-08-18 18:04:48 +02:00
public function kill() {
return $this->killed = true;
}
/**
* @coroutine
*
2016-08-23 23:47:40 +02:00
* @param \Amp\Parallel\Sync\Channel $channel
*
* @return \Generator
*
* @codeCoverageIgnore Only executed in thread.
*/
2016-08-18 18:04:48 +02:00
private function execute(Channel $channel): \Generator {
try {
2015-08-27 21:49:41 +02:00
if ($this->function instanceof \Closure) {
$function = $this->function->bindTo($channel, Channel::class);
2015-08-27 21:49:41 +02:00
}
if (empty($function)) {
$function = $this->function;
}
2016-08-18 18:04:48 +02:00
$result = $function(...$this->args);
2016-08-18 18:04:48 +02:00
if ($result instanceof \Generator) {
$result = new Coroutine($result);
}
2016-11-15 00:43:44 +01:00
if ($result instanceof Promise) {
2016-08-18 18:04:48 +02:00
$result = yield $result;
}
2016-08-18 18:04:48 +02:00
$result = new ExitSuccess($result);
2016-01-23 07:00:56 +01:00
} catch (\Throwable $exception) {
$result = new ExitFailure($exception);
}
2015-12-12 07:34:41 +01:00
// Attempt to return the result.
try {
2016-01-23 18:20:58 +01:00
try {
2016-08-18 18:04:48 +02:00
return yield $channel->send($result);
2016-01-23 18:20:58 +01:00
} catch (SerializationException $exception) {
// Serializing the result failed. Send the reason why.
2016-08-18 18:04:48 +02:00
return yield $channel->send(new ExitFailure($exception));
2016-01-23 18:20:58 +01:00
}
} catch (ChannelException $exception) {
// The result was not sendable! The parent context must have died or killed the context.
2016-01-23 18:29:22 +01:00
return 0;
2015-12-12 07:34:41 +01:00
}
}
}