2016-12-29 19:16:04 -06:00
|
|
|
<?php
|
2015-08-24 10:40:36 -05:00
|
|
|
|
2017-11-29 15:01:32 -06:00
|
|
|
namespace Amp\Parallel\Context\Internal;
|
2016-08-18 11:04:48 -05:00
|
|
|
|
2017-05-18 09:51:31 +02:00
|
|
|
use Amp\Loop;
|
|
|
|
use Amp\Parallel\Sync\Channel;
|
|
|
|
use Amp\Parallel\Sync\ChannelException;
|
|
|
|
use Amp\Parallel\Sync\ChannelledSocket;
|
2017-07-28 17:34:24 -05:00
|
|
|
use Amp\Parallel\Sync\ExitFailure;
|
|
|
|
use Amp\Parallel\Sync\ExitSuccess;
|
2017-05-18 09:51:31 +02:00
|
|
|
use Amp\Parallel\Sync\SerializationException;
|
2017-07-18 23:44:25 -05:00
|
|
|
use function Amp\call;
|
2015-08-24 10:40:36 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
* An internal thread that executes a given function concurrently.
|
|
|
|
*
|
|
|
|
* @internal
|
|
|
|
*/
|
2018-10-21 10:54:46 -05:00
|
|
|
final class Thread extends \Thread
|
2018-10-07 09:50:45 -05:00
|
|
|
{
|
2016-08-18 11:04:48 -05:00
|
|
|
const KILL_CHECK_FREQUENCY = 250;
|
2017-01-16 13:30:16 -06:00
|
|
|
|
2019-02-20 17:31:22 -06:00
|
|
|
private $id;
|
|
|
|
|
2016-08-26 10:10:03 -05:00
|
|
|
/** @var callable The function to execute in the thread. */
|
2015-08-24 10:40:36 -05:00
|
|
|
private $function;
|
|
|
|
|
2016-08-26 10:10:03 -05:00
|
|
|
/** @var mixed[] Arguments to pass to the function. */
|
2015-08-24 10:40:36 -05:00
|
|
|
private $args;
|
|
|
|
|
2016-08-26 10:10:03 -05:00
|
|
|
/** @var resource */
|
2015-08-24 10:40:36 -05:00
|
|
|
private $socket;
|
|
|
|
|
2016-08-26 10:10:03 -05:00
|
|
|
/** @var bool */
|
2015-09-05 12:52:56 -05:00
|
|
|
private $killed = false;
|
|
|
|
|
2015-08-24 10:40:36 -05:00
|
|
|
/**
|
|
|
|
* Creates a new thread object.
|
|
|
|
*
|
2019-02-20 17:31:22 -06:00
|
|
|
* @param int $id Thread ID.
|
2015-08-24 10:40:36 -05:00
|
|
|
* @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.
|
|
|
|
*/
|
2019-02-20 17:31:22 -06:00
|
|
|
public function __construct(int $id, $socket, callable $function, array $args = [])
|
2018-10-07 09:50:45 -05:00
|
|
|
{
|
2019-02-20 17:31:22 -06:00
|
|
|
$this->id = $id;
|
2015-08-24 10:40:36 -05:00
|
|
|
$this->function = $function;
|
|
|
|
$this->args = $args;
|
|
|
|
$this->socket = $socket;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Runs the thread code and the initialized function.
|
2015-09-03 18:10:19 -05:00
|
|
|
*
|
|
|
|
* @codeCoverageIgnore Only executed in thread.
|
2015-08-24 10:40:36 -05:00
|
|
|
*/
|
2018-10-07 09:50:45 -05:00
|
|
|
public function run()
|
|
|
|
{
|
2019-02-13 14:15:16 -06:00
|
|
|
\define("AMP_CONTEXT", "thread");
|
2019-02-20 17:31:22 -06:00
|
|
|
\define("AMP_CONTEXT_ID", $this->id);
|
2019-02-13 14:15:16 -06:00
|
|
|
|
2015-08-24 10:40:36 -05:00
|
|
|
/* 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.
|
|
|
|
*/
|
2017-01-13 11:00:35 +01:00
|
|
|
|
2017-01-16 13:30:16 -06:00
|
|
|
// Protect scope by using an unbound closure (protects static access as well).
|
2017-11-25 08:59:07 -06:00
|
|
|
(static function () {
|
|
|
|
$paths = [
|
|
|
|
\dirname(__DIR__, 3) . \DIRECTORY_SEPARATOR . "vendor" . \DIRECTORY_SEPARATOR . "autoload.php",
|
|
|
|
\dirname(__DIR__, 5) . \DIRECTORY_SEPARATOR . "autoload.php",
|
|
|
|
];
|
|
|
|
|
|
|
|
foreach ($paths as $path) {
|
|
|
|
if (\file_exists($path)) {
|
|
|
|
$autoloadPath = $path;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isset($autoloadPath)) {
|
2017-12-07 21:26:55 -06:00
|
|
|
throw new \Error("Could not locate autoload.php");
|
2017-11-25 08:59:07 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
require $autoloadPath;
|
|
|
|
})->bindTo(null, null)();
|
2017-01-13 11:00:35 +01:00
|
|
|
|
2015-08-24 10:40:36 -05:00
|
|
|
// At this point, the thread environment has been prepared so begin using the thread.
|
2016-01-27 00:54:51 -06:00
|
|
|
|
2017-07-19 00:14:16 -05:00
|
|
|
if ($this->killed) {
|
|
|
|
return; // Thread killed while requiring autoloader, simply exit.
|
|
|
|
}
|
|
|
|
|
2017-07-19 00:08:55 -05:00
|
|
|
Loop::run(function () {
|
|
|
|
$watcher = Loop::repeat(self::KILL_CHECK_FREQUENCY, function () {
|
|
|
|
if ($this->killed) {
|
|
|
|
Loop::stop();
|
|
|
|
}
|
2017-03-16 17:03:59 -05:00
|
|
|
});
|
2017-07-19 00:08:55 -05:00
|
|
|
Loop::unreference($watcher);
|
2017-01-13 11:00:35 +01:00
|
|
|
|
2017-11-29 21:50:04 -06:00
|
|
|
try {
|
|
|
|
$channel = new ChannelledSocket($this->socket, $this->socket);
|
2017-12-26 23:16:44 -06:00
|
|
|
yield from $this->execute($channel);
|
2017-11-29 21:50:04 -06:00
|
|
|
} catch (\Throwable $exception) {
|
|
|
|
return; // Parent context exited or destroyed thread, no need to continue.
|
2017-12-26 23:16:44 -06:00
|
|
|
} finally {
|
|
|
|
Loop::cancel($watcher);
|
2017-11-29 21:50:04 -06:00
|
|
|
}
|
2017-07-19 00:08:55 -05:00
|
|
|
});
|
2015-09-05 12:52:56 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets a local variable to true so the running event loop can check for a kill signal.
|
|
|
|
*/
|
2018-10-07 09:50:45 -05:00
|
|
|
public function kill()
|
|
|
|
{
|
2016-01-23 10:52:18 -06:00
|
|
|
return $this->killed = true;
|
2015-08-24 10:40:36 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2016-08-23 16:47:40 -05:00
|
|
|
* @param \Amp\Parallel\Sync\Channel $channel
|
2015-08-24 10:40:36 -05:00
|
|
|
*
|
|
|
|
* @return \Generator
|
|
|
|
*
|
2015-09-03 18:10:19 -05:00
|
|
|
* @codeCoverageIgnore Only executed in thread.
|
2015-08-24 10:40:36 -05:00
|
|
|
*/
|
2018-10-07 09:50:45 -05:00
|
|
|
private function execute(Channel $channel): \Generator
|
|
|
|
{
|
2015-08-24 10:40:36 -05:00
|
|
|
try {
|
2017-12-10 16:43:19 -06:00
|
|
|
$result = new ExitSuccess(yield call($this->function, $channel, ...$this->args));
|
2016-01-23 00:00:56 -06:00
|
|
|
} catch (\Throwable $exception) {
|
2015-08-24 10:40:36 -05:00
|
|
|
$result = new ExitFailure($exception);
|
|
|
|
}
|
|
|
|
|
2017-07-18 23:44:25 -05:00
|
|
|
if ($this->killed) {
|
|
|
|
return; // Parent is not listening for a result.
|
|
|
|
}
|
|
|
|
|
2015-12-12 00:34:41 -06:00
|
|
|
// Attempt to return the result.
|
|
|
|
try {
|
2016-01-23 11:20:58 -06:00
|
|
|
try {
|
2017-07-18 23:44:25 -05:00
|
|
|
yield $channel->send($result);
|
2016-01-23 11:20:58 -06:00
|
|
|
} catch (SerializationException $exception) {
|
|
|
|
// Serializing the result failed. Send the reason why.
|
2017-07-18 23:44:25 -05:00
|
|
|
yield $channel->send(new ExitFailure($exception));
|
2016-01-23 11:20:58 -06:00
|
|
|
}
|
|
|
|
} catch (ChannelException $exception) {
|
2016-01-23 10:57:10 -06:00
|
|
|
// The result was not sendable! The parent context must have died or killed the context.
|
2015-12-12 00:34:41 -06:00
|
|
|
}
|
2015-08-24 10:40:36 -05:00
|
|
|
}
|
|
|
|
}
|