mirror of
https://github.com/danog/process.git
synced 2024-11-26 12:14:43 +01:00
118 lines
3.1 KiB
PHP
118 lines
3.1 KiB
PHP
<?php
|
|
|
|
namespace Amp\Process;
|
|
|
|
use Amp\ByteStream\ClosedException;
|
|
use Amp\ByteStream\OutputStream;
|
|
use Amp\ByteStream\ResourceOutputStream;
|
|
use Amp\ByteStream\StreamException;
|
|
use Amp\Deferred;
|
|
use Amp\Failure;
|
|
use Amp\Promise;
|
|
|
|
final class ProcessOutputStream implements OutputStream
|
|
{
|
|
/** @var \SplQueue */
|
|
private $queuedWrites;
|
|
|
|
/** @var bool */
|
|
private $shouldClose = false;
|
|
|
|
/** @var ResourceOutputStream */
|
|
private $resourceStream;
|
|
|
|
/** @var StreamException|null */
|
|
private $error;
|
|
|
|
public function __construct(Promise $resourceStreamPromise)
|
|
{
|
|
$this->queuedWrites = new \SplQueue;
|
|
$resourceStreamPromise->onResolve(function ($error, $resourceStream) {
|
|
if ($error) {
|
|
$this->error = new StreamException("Failed to launch process", 0, $error);
|
|
|
|
while (!$this->queuedWrites->isEmpty()) {
|
|
list(, $deferred) = $this->queuedWrites->shift();
|
|
$deferred->fail($this->error);
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
while (!$this->queuedWrites->isEmpty()) {
|
|
/**
|
|
* @var string $data
|
|
* @var \Amp\Deferred $deferred
|
|
*/
|
|
list($data, $deferred) = $this->queuedWrites->shift();
|
|
$deferred->resolve($resourceStream->write($data));
|
|
}
|
|
|
|
$this->resourceStream = $resourceStream;
|
|
|
|
if ($this->shouldClose) {
|
|
$this->resourceStream->close();
|
|
}
|
|
});
|
|
}
|
|
|
|
/** @inheritdoc */
|
|
public function write(string $data): Promise
|
|
{
|
|
if ($this->resourceStream) {
|
|
return $this->resourceStream->write($data);
|
|
}
|
|
|
|
if ($this->error) {
|
|
return new Failure($this->error);
|
|
}
|
|
|
|
if ($this->shouldClose) {
|
|
throw new ClosedException("Stream has already been closed.");
|
|
}
|
|
|
|
$deferred = new Deferred;
|
|
$this->queuedWrites->push([$data, $deferred]);
|
|
|
|
return $deferred->promise();
|
|
}
|
|
|
|
/** @inheritdoc */
|
|
public function end(string $finalData = ""): Promise
|
|
{
|
|
if ($this->resourceStream) {
|
|
return $this->resourceStream->end($finalData);
|
|
}
|
|
|
|
if ($this->error) {
|
|
return new Failure($this->error);
|
|
}
|
|
|
|
if ($this->shouldClose) {
|
|
throw new ClosedException("Stream has already been closed.");
|
|
}
|
|
|
|
$deferred = new Deferred;
|
|
$this->queuedWrites->push([$finalData, $deferred]);
|
|
|
|
$this->shouldClose = true;
|
|
|
|
return $deferred->promise();
|
|
}
|
|
|
|
public function close()
|
|
{
|
|
$this->shouldClose = true;
|
|
|
|
if ($this->resourceStream) {
|
|
$this->resourceStream->close();
|
|
} elseif (!$this->queuedWrites->isEmpty()) {
|
|
$error = new ClosedException("Stream closed.");
|
|
do {
|
|
list(, $deferred) = $this->queuedWrites->shift();
|
|
$deferred->fail($error);
|
|
} while (!$this->queuedWrites->isEmpty());
|
|
}
|
|
}
|
|
}
|