2018-10-17 21:52:26 +02:00
|
|
|
<?php
|
|
|
|
declare(strict_types = 1);
|
|
|
|
|
2018-11-06 03:57:36 +01:00
|
|
|
namespace Psalm\Internal\LanguageServer;
|
2018-10-17 21:52:26 +02:00
|
|
|
|
|
|
|
use AdvancedJsonRpc\Message as MessageBody;
|
2018-11-18 00:00:28 +01:00
|
|
|
use Exception;
|
|
|
|
use Psalm\Internal\LanguageServer\Message;
|
|
|
|
use Sabre\Event\Emitter;
|
|
|
|
use Sabre\Event\Loop;
|
2018-10-17 21:52:26 +02:00
|
|
|
|
2018-11-18 00:00:28 +01:00
|
|
|
/**
|
|
|
|
* Source: https://github.com/felixfbecker/php-language-server/tree/master/src/ProtocolStreamReader.php
|
|
|
|
*/
|
2018-10-17 21:52:26 +02:00
|
|
|
class ProtocolStreamReader extends Emitter implements ProtocolReader
|
|
|
|
{
|
|
|
|
const PARSE_HEADERS = 1;
|
|
|
|
const PARSE_BODY = 2;
|
|
|
|
|
|
|
|
/** @var resource */
|
|
|
|
private $input;
|
2018-11-18 00:00:28 +01:00
|
|
|
/**
|
|
|
|
* This is checked by ProtocolStreamReader so that it will stop reading from streams in the forked process.
|
|
|
|
* There could be buffered bytes in stdin/over TCP, those would be processed by TCP if it were not for this check.
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
private $is_accepting_new_requests = true;
|
2018-10-17 21:52:26 +02:00
|
|
|
/** @var int */
|
2018-11-18 00:00:28 +01:00
|
|
|
private $parsing_mode = self::PARSE_HEADERS;
|
2018-10-17 21:52:26 +02:00
|
|
|
/** @var string */
|
|
|
|
private $buffer = '';
|
2018-11-18 00:00:28 +01:00
|
|
|
/** @var string[] */
|
2018-10-17 21:52:26 +02:00
|
|
|
private $headers = [];
|
|
|
|
/** @var ?int */
|
2018-11-18 00:00:28 +01:00
|
|
|
private $content_length = null;
|
|
|
|
/** @var bool */
|
|
|
|
private $did_emit_close = false;
|
2018-10-17 21:52:26 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param resource $input
|
|
|
|
*/
|
|
|
|
public function __construct($input)
|
|
|
|
{
|
|
|
|
$this->input = $input;
|
|
|
|
|
|
|
|
$this->on(
|
|
|
|
'close',
|
|
|
|
/** @return void */
|
|
|
|
function () {
|
|
|
|
Loop\removeReadStream($this->input);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
Loop\addReadStream(
|
|
|
|
$this->input,
|
|
|
|
/** @return void */
|
|
|
|
function () {
|
|
|
|
if (feof($this->input)) {
|
|
|
|
// If stream_select reported a status change for this stream,
|
|
|
|
// but the stream is EOF, it means it was closed.
|
2018-11-18 00:00:28 +01:00
|
|
|
$this->emitClose();
|
2018-10-17 21:52:26 +02:00
|
|
|
return;
|
|
|
|
}
|
2018-11-18 00:00:28 +01:00
|
|
|
if (!$this->is_accepting_new_requests) {
|
|
|
|
// If we fork, don't read any bytes in the input buffer from the worker process.
|
|
|
|
$this->emitClose();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
$emitted_messages = $this->readMessages();
|
|
|
|
if ($emitted_messages > 0) {
|
|
|
|
$this->emit('readMessageGroup');
|
2018-10-17 21:52:26 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
2018-11-18 00:00:28 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @return int
|
|
|
|
*/
|
|
|
|
private function readMessages() : int
|
|
|
|
{
|
|
|
|
$emitted_messages = 0;
|
|
|
|
while (($c = fgetc($this->input)) !== false && $c !== '') {
|
|
|
|
$this->buffer .= $c;
|
|
|
|
switch ($this->parsing_mode) {
|
|
|
|
case self::PARSE_HEADERS:
|
|
|
|
if ($this->buffer === "\r\n") {
|
|
|
|
$this->parsing_mode = self::PARSE_BODY;
|
|
|
|
$this->content_length = (int)$this->headers['Content-Length'];
|
|
|
|
$this->buffer = '';
|
|
|
|
} elseif (substr($this->buffer, -2) === "\r\n") {
|
|
|
|
$parts = explode(':', $this->buffer);
|
|
|
|
$this->headers[$parts[0]] = trim($parts[1]);
|
|
|
|
$this->buffer = '';
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case self::PARSE_BODY:
|
|
|
|
if (strlen($this->buffer) === $this->content_length) {
|
|
|
|
if (!$this->is_accepting_new_requests) {
|
|
|
|
// If we fork, don't read any bytes in the input buffer from the worker process.
|
|
|
|
$this->emitClose();
|
|
|
|
return $emitted_messages;
|
|
|
|
}
|
|
|
|
// MessageBody::parse can throw an Error, maybe log an error?
|
|
|
|
try {
|
|
|
|
$msg = new Message(MessageBody::parse($this->buffer), $this->headers);
|
|
|
|
} catch (Exception $_) {
|
|
|
|
$msg = null;
|
|
|
|
}
|
|
|
|
if ($msg) {
|
|
|
|
$emitted_messages++;
|
|
|
|
$this->emit('message', [$msg]);
|
|
|
|
/** @psalm-suppress DocblockTypeContradiction */
|
|
|
|
if (!$this->is_accepting_new_requests) {
|
|
|
|
// If we fork, don't read any bytes in the input buffer from the worker process.
|
|
|
|
$this->emitClose();
|
|
|
|
return $emitted_messages;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$this->parsing_mode = self::PARSE_HEADERS;
|
|
|
|
$this->headers = [];
|
|
|
|
$this->buffer = '';
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $emitted_messages;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
private function emitClose()
|
|
|
|
{
|
|
|
|
if ($this->did_emit_close) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
$this->did_emit_close = true;
|
|
|
|
$this->emit('close');
|
|
|
|
}
|
2018-10-17 21:52:26 +02:00
|
|
|
}
|