2015-08-27 09:10:08 -05:00
|
|
|
<?php
|
|
|
|
namespace Icicle\Concurrent\Process;
|
|
|
|
|
2015-08-27 13:06:39 -05:00
|
|
|
use Icicle\Concurrent\Exception\InvalidArgumentError;
|
|
|
|
use Icicle\Concurrent\Exception\StatusError;
|
2015-08-27 09:10:08 -05:00
|
|
|
use Icicle\Concurrent\Exception\SynchronizationError;
|
2015-10-20 00:06:43 -05:00
|
|
|
use Icicle\Concurrent\ProcessInterface;
|
2015-08-27 09:10:08 -05:00
|
|
|
use Icicle\Concurrent\Sync\Channel;
|
2015-10-20 00:06:43 -05:00
|
|
|
use Icicle\Concurrent\Sync\ChannelInterface;
|
2015-08-29 02:13:14 -05:00
|
|
|
use Icicle\Concurrent\Sync\Internal\ExitStatusInterface;
|
2015-08-27 09:10:08 -05:00
|
|
|
|
2015-10-20 00:06:43 -05:00
|
|
|
class ChannelledProcess implements ChannelInterface, ProcessInterface
|
2015-08-27 09:10:08 -05:00
|
|
|
{
|
|
|
|
/**
|
|
|
|
* @var \Icicle\Concurrent\Process\Process
|
|
|
|
*/
|
|
|
|
private $process;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var \Icicle\Concurrent\Sync\Channel
|
|
|
|
*/
|
|
|
|
private $channel;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $path Path to PHP script.
|
|
|
|
* @param string $cwd Working directory.
|
|
|
|
* @param mixed[] $env Array of environment variables.
|
|
|
|
*/
|
|
|
|
public function __construct($path, $cwd = '', array $env = [])
|
|
|
|
{
|
|
|
|
$command = PHP_BINARY . ' ' . $path;
|
|
|
|
|
|
|
|
$this->process = new Process($command, $cwd, $env);
|
|
|
|
}
|
|
|
|
|
2015-08-27 13:06:39 -05:00
|
|
|
/**
|
|
|
|
* Resets process values.
|
|
|
|
*/
|
|
|
|
public function __clone()
|
|
|
|
{
|
|
|
|
$this->process = clone $this->process;
|
|
|
|
$this->channel = null;
|
|
|
|
}
|
|
|
|
|
2015-08-27 09:10:08 -05:00
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
public function start()
|
|
|
|
{
|
|
|
|
$this->process->start();
|
|
|
|
|
|
|
|
$this->channel = new Channel($this->process->getStdOut(), $this->process->getStdIn());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
public function isRunning()
|
|
|
|
{
|
|
|
|
return $this->process->isRunning();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
public function receive()
|
|
|
|
{
|
2015-09-19 21:34:41 -05:00
|
|
|
if (null === $this->channel) {
|
2015-08-27 13:06:39 -05:00
|
|
|
throw new StatusError('The process has not been started.');
|
2015-08-27 09:10:08 -05:00
|
|
|
}
|
|
|
|
|
2015-08-27 13:06:39 -05:00
|
|
|
$data = (yield $this->channel->receive());
|
|
|
|
|
|
|
|
if ($data instanceof ExitStatusInterface) {
|
|
|
|
$data = $data->getResult();
|
|
|
|
throw new SynchronizationError(sprintf(
|
|
|
|
'Thread unexpectedly exited with result of type: %s',
|
|
|
|
is_object($data) ? get_class($data) : gettype($data)
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
yield $data;
|
2015-08-27 09:10:08 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
public function send($data)
|
|
|
|
{
|
2015-09-19 21:34:41 -05:00
|
|
|
if (null === $this->channel) {
|
2015-08-27 13:06:39 -05:00
|
|
|
throw new StatusError('The process has not been started.');
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($data instanceof ExitStatusInterface) {
|
|
|
|
throw new InvalidArgumentError('Cannot send exit status objects.');
|
2015-08-27 09:10:08 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
yield $this->channel->send($data);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
public function join()
|
|
|
|
{
|
2015-09-19 21:34:41 -05:00
|
|
|
if (null === $this->channel) {
|
2015-08-27 13:06:39 -05:00
|
|
|
throw new StatusError('The process has not been started.');
|
|
|
|
}
|
|
|
|
|
2015-10-18 01:54:09 -05:00
|
|
|
$response = (yield $this->channel->receive());
|
2015-08-27 13:06:39 -05:00
|
|
|
|
2015-10-18 01:54:09 -05:00
|
|
|
yield $this->process->join();
|
2015-08-27 13:06:39 -05:00
|
|
|
|
2015-10-18 01:54:09 -05:00
|
|
|
if (!$response instanceof ExitStatusInterface) {
|
|
|
|
throw new SynchronizationError('Did not receive an exit status from thread.');
|
2015-08-27 13:06:39 -05:00
|
|
|
}
|
2015-10-18 01:54:09 -05:00
|
|
|
|
|
|
|
yield $response->getResult();
|
2015-08-27 09:10:08 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
public function kill()
|
|
|
|
{
|
|
|
|
$this->process->kill();
|
2015-10-18 01:54:09 -05:00
|
|
|
$this->channel = null;
|
2015-08-27 09:10:08 -05:00
|
|
|
}
|
2015-10-20 00:06:43 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
public function getPid()
|
|
|
|
{
|
|
|
|
return $this->process->getPid();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
public function signal($signo)
|
|
|
|
{
|
|
|
|
$this->process->signal($signo);
|
|
|
|
}
|
2015-08-27 09:10:08 -05:00
|
|
|
}
|