Initial commit
This commit is contained in:
commit
6a7ea39f5c
2
.gitignore
vendored
Normal file
2
.gitignore
vendored
Normal file
@ -0,0 +1,2 @@
|
||||
/.idea/
|
||||
/vendor/
|
21
composer.json
Normal file
21
composer.json
Normal file
@ -0,0 +1,21 @@
|
||||
{
|
||||
"name": "joopschilder/reactphp-input-stream",
|
||||
"description": "Wraps a non-blocking input to behave like a stream so it can be used in a ReactPHP EventLoop",
|
||||
"type": "library",
|
||||
"license": "MIT",
|
||||
"authors": [
|
||||
{
|
||||
"name": "Joop Schilder",
|
||||
"email": "jnmschilder@protonmail.com"
|
||||
}
|
||||
],
|
||||
"require": {
|
||||
"react/event-loop": "^1.1",
|
||||
"react/stream": "^1.1"
|
||||
},
|
||||
"autoload": {
|
||||
"psr-4": {
|
||||
"JoopSchilder\\React\\Stream\\NonBlockingInput\\": "src"
|
||||
}
|
||||
}
|
||||
}
|
149
composer.lock
generated
Normal file
149
composer.lock
generated
Normal file
@ -0,0 +1,149 @@
|
||||
{
|
||||
"_readme": [
|
||||
"This file locks the dependencies of your project to a known state",
|
||||
"Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies",
|
||||
"This file is @generated automatically"
|
||||
],
|
||||
"content-hash": "8cb0549e866a693d6b61e7e7f9a75862",
|
||||
"packages": [
|
||||
{
|
||||
"name": "evenement/evenement",
|
||||
"version": "v3.0.1",
|
||||
"source": {
|
||||
"type": "git",
|
||||
"url": "https://github.com/igorw/evenement.git",
|
||||
"reference": "531bfb9d15f8aa57454f5f0285b18bec903b8fb7"
|
||||
},
|
||||
"dist": {
|
||||
"type": "zip",
|
||||
"url": "https://api.github.com/repos/igorw/evenement/zipball/531bfb9d15f8aa57454f5f0285b18bec903b8fb7",
|
||||
"reference": "531bfb9d15f8aa57454f5f0285b18bec903b8fb7",
|
||||
"shasum": ""
|
||||
},
|
||||
"require": {
|
||||
"php": ">=7.0"
|
||||
},
|
||||
"require-dev": {
|
||||
"phpunit/phpunit": "^6.0"
|
||||
},
|
||||
"type": "library",
|
||||
"autoload": {
|
||||
"psr-0": {
|
||||
"Evenement": "src"
|
||||
}
|
||||
},
|
||||
"notification-url": "https://packagist.org/downloads/",
|
||||
"license": [
|
||||
"MIT"
|
||||
],
|
||||
"authors": [
|
||||
{
|
||||
"name": "Igor Wiedler",
|
||||
"email": "igor@wiedler.ch"
|
||||
}
|
||||
],
|
||||
"description": "Événement is a very simple event dispatching library for PHP",
|
||||
"keywords": [
|
||||
"event-dispatcher",
|
||||
"event-emitter"
|
||||
],
|
||||
"time": "2017-07-23T21:35:13+00:00"
|
||||
},
|
||||
{
|
||||
"name": "react/event-loop",
|
||||
"version": "v1.1.1",
|
||||
"source": {
|
||||
"type": "git",
|
||||
"url": "https://github.com/reactphp/event-loop.git",
|
||||
"reference": "6d24de090cd59cfc830263cfba965be77b563c13"
|
||||
},
|
||||
"dist": {
|
||||
"type": "zip",
|
||||
"url": "https://api.github.com/repos/reactphp/event-loop/zipball/6d24de090cd59cfc830263cfba965be77b563c13",
|
||||
"reference": "6d24de090cd59cfc830263cfba965be77b563c13",
|
||||
"shasum": ""
|
||||
},
|
||||
"require": {
|
||||
"php": ">=5.3.0"
|
||||
},
|
||||
"require-dev": {
|
||||
"phpunit/phpunit": "^7.0 || ^6.4 || ^5.7 || ^4.8.35"
|
||||
},
|
||||
"suggest": {
|
||||
"ext-event": "~1.0 for ExtEventLoop",
|
||||
"ext-pcntl": "For signal handling support when using the StreamSelectLoop",
|
||||
"ext-uv": "* for ExtUvLoop"
|
||||
},
|
||||
"type": "library",
|
||||
"autoload": {
|
||||
"psr-4": {
|
||||
"React\\EventLoop\\": "src"
|
||||
}
|
||||
},
|
||||
"notification-url": "https://packagist.org/downloads/",
|
||||
"license": [
|
||||
"MIT"
|
||||
],
|
||||
"description": "ReactPHP's core reactor event loop that libraries can use for evented I/O.",
|
||||
"keywords": [
|
||||
"asynchronous",
|
||||
"event-loop"
|
||||
],
|
||||
"time": "2020-01-01T18:39:52+00:00"
|
||||
},
|
||||
{
|
||||
"name": "react/stream",
|
||||
"version": "v1.1.0",
|
||||
"source": {
|
||||
"type": "git",
|
||||
"url": "https://github.com/reactphp/stream.git",
|
||||
"reference": "50426855f7a77ddf43b9266c22320df5bf6c6ce6"
|
||||
},
|
||||
"dist": {
|
||||
"type": "zip",
|
||||
"url": "https://api.github.com/repos/reactphp/stream/zipball/50426855f7a77ddf43b9266c22320df5bf6c6ce6",
|
||||
"reference": "50426855f7a77ddf43b9266c22320df5bf6c6ce6",
|
||||
"shasum": ""
|
||||
},
|
||||
"require": {
|
||||
"evenement/evenement": "^3.0 || ^2.0 || ^1.0",
|
||||
"php": ">=5.3.8",
|
||||
"react/event-loop": "^1.0 || ^0.5 || ^0.4 || ^0.3.5"
|
||||
},
|
||||
"require-dev": {
|
||||
"clue/stream-filter": "~1.2",
|
||||
"phpunit/phpunit": "^6.4 || ^5.7 || ^4.8.35"
|
||||
},
|
||||
"type": "library",
|
||||
"autoload": {
|
||||
"psr-4": {
|
||||
"React\\Stream\\": "src"
|
||||
}
|
||||
},
|
||||
"notification-url": "https://packagist.org/downloads/",
|
||||
"license": [
|
||||
"MIT"
|
||||
],
|
||||
"description": "Event-driven readable and writable streams for non-blocking I/O in ReactPHP",
|
||||
"keywords": [
|
||||
"event-driven",
|
||||
"io",
|
||||
"non-blocking",
|
||||
"pipe",
|
||||
"reactphp",
|
||||
"readable",
|
||||
"stream",
|
||||
"writable"
|
||||
],
|
||||
"time": "2019-01-01T16:15:09+00:00"
|
||||
}
|
||||
],
|
||||
"packages-dev": [],
|
||||
"aliases": [],
|
||||
"minimum-stability": "stable",
|
||||
"stability-flags": [],
|
||||
"prefer-stable": false,
|
||||
"prefer-lowest": false,
|
||||
"platform": [],
|
||||
"platform-dev": []
|
||||
}
|
22
examples/basic_example.php
Normal file
22
examples/basic_example.php
Normal file
@ -0,0 +1,22 @@
|
||||
<?php
|
||||
|
||||
use JoopSchilder\React\Stream\NonBlockingInput\ReadableNonBlockingInputStream;
|
||||
use React\EventLoop\Factory;
|
||||
|
||||
require_once __DIR__ . '/../vendor/autoload.php';
|
||||
require_once __DIR__ . '/basic_example_classes.php';
|
||||
|
||||
/**
|
||||
* The input will have data available every second.
|
||||
* After 4 seconds have passed, it will generate an error.
|
||||
*/
|
||||
$input = new DemoNonBlockingInput();
|
||||
|
||||
$loop = Factory::create();
|
||||
$stream = new ReadableNonBlockingInputStream($input, $loop);
|
||||
$stream->on('data', fn() => print('m'));
|
||||
$stream->on('error', fn() => print('e'));
|
||||
$stream->on('close', fn() => print('c'));
|
||||
$loop->addPeriodicTimer(0.2, fn() => print('.'));
|
||||
$loop->run();
|
||||
|
46
examples/basic_example_classes.php
Normal file
46
examples/basic_example_classes.php
Normal file
@ -0,0 +1,46 @@
|
||||
<?php
|
||||
|
||||
use JoopSchilder\React\Stream\NonBlockingInput\NonBlockingInputInterface;
|
||||
use JoopSchilder\React\Stream\NonBlockingInput\PayloadInterface;
|
||||
|
||||
final class DemoEmptyPayload implements PayloadInterface
|
||||
{
|
||||
}
|
||||
|
||||
final class DemoNonBlockingInput implements NonBlockingInputInterface
|
||||
{
|
||||
private const DATA_AVAILABLE_INTERVAL_S = 1.0;
|
||||
private const ERROR_AFTER_S = 4.0;
|
||||
|
||||
private float $lastEmission = 0;
|
||||
|
||||
private float $initialEmission = 0;
|
||||
|
||||
|
||||
public function open(): void
|
||||
{
|
||||
$this->initialEmission = microtime(true);
|
||||
}
|
||||
|
||||
|
||||
public function select(): ?PayloadInterface
|
||||
{
|
||||
$now = microtime(true);
|
||||
if ($now - $this->initialEmission > self::ERROR_AFTER_S) {
|
||||
throw new Exception('Oh no!');
|
||||
}
|
||||
if ($now - $this->lastEmission > self::DATA_AVAILABLE_INTERVAL_S) {
|
||||
$this->lastEmission = $now;
|
||||
|
||||
return new DemoEmptyPayload();
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
|
||||
public function close(): void
|
||||
{
|
||||
}
|
||||
|
||||
}
|
16
src/NonBlockingInputInterface.php
Normal file
16
src/NonBlockingInputInterface.php
Normal file
@ -0,0 +1,16 @@
|
||||
<?php
|
||||
|
||||
namespace JoopSchilder\React\Stream\NonBlockingInput;
|
||||
|
||||
interface NonBlockingInputInterface
|
||||
{
|
||||
|
||||
function open(): void;
|
||||
|
||||
|
||||
function select(): ?PayloadInterface;
|
||||
|
||||
|
||||
function close(): void;
|
||||
|
||||
}
|
10
src/PayloadInterface.php
Normal file
10
src/PayloadInterface.php
Normal file
@ -0,0 +1,10 @@
|
||||
<?php
|
||||
|
||||
namespace JoopSchilder\React\Stream\NonBlockingInput;
|
||||
|
||||
/**
|
||||
* Marker interface.
|
||||
*/
|
||||
interface PayloadInterface
|
||||
{
|
||||
}
|
140
src/ReadableNonBlockingInputStream.php
Normal file
140
src/ReadableNonBlockingInputStream.php
Normal file
@ -0,0 +1,140 @@
|
||||
<?php
|
||||
|
||||
namespace JoopSchilder\React\Stream\NonBlockingInput;
|
||||
|
||||
use Evenement\EventEmitter;
|
||||
use JoopSchilder\React\Stream\NonBlockingInput\ValueObject\PollingInterval;
|
||||
use React\EventLoop\LoopInterface;
|
||||
use React\EventLoop\TimerInterface;
|
||||
use React\Stream\ReadableResourceStream;
|
||||
use React\Stream\ReadableStreamInterface;
|
||||
use React\Stream\Util;
|
||||
use React\Stream\WritableStreamInterface;
|
||||
use RuntimeException;
|
||||
use Throwable;
|
||||
|
||||
/**
|
||||
* @see ReadableStreamInterface
|
||||
* @see ReadableResourceStream
|
||||
*/
|
||||
final class ReadableNonBlockingInputStream extends EventEmitter implements ReadableStreamInterface
|
||||
{
|
||||
private NonBlockingInputInterface $input;
|
||||
|
||||
private LoopInterface $loop;
|
||||
|
||||
private PollingInterval $interval;
|
||||
|
||||
private ?TimerInterface $periodicTimer = null;
|
||||
|
||||
private bool $closed = false;
|
||||
|
||||
private bool $listening = false;
|
||||
|
||||
|
||||
/**
|
||||
* @see ReadableStreamInterface
|
||||
* @see ReadableResourceStream for an example
|
||||
*/
|
||||
public function __construct(NonBlockingInputInterface $input, LoopInterface $loop, ?PollingInterval $interval = null)
|
||||
{
|
||||
$this->input = $input;
|
||||
$this->loop = $loop;
|
||||
$this->interval = $interval ?? new PollingInterval();
|
||||
|
||||
$this->resume();
|
||||
}
|
||||
|
||||
|
||||
public function isReadable()
|
||||
{
|
||||
return !$this->closed;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @see ReadableResourceStream::pause()
|
||||
* Pause consumption of the AMQP queue but do not mark the stream as closed
|
||||
*/
|
||||
public function pause()
|
||||
{
|
||||
if (!$this->listening) {
|
||||
return;
|
||||
}
|
||||
|
||||
$this->input->close();
|
||||
$this->loop->cancelTimer($this->periodicTimer);
|
||||
$this->listening = false;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @see ReadableResourceStream::resume()
|
||||
* Register the consumer with the broker and add consumer again
|
||||
*/
|
||||
public function resume()
|
||||
{
|
||||
if ($this->listening || $this->closed) {
|
||||
return;
|
||||
}
|
||||
|
||||
$this->input->open();
|
||||
$this->periodicTimer = $this->loop->addPeriodicTimer(
|
||||
$this->interval->getInterval(),
|
||||
function () {
|
||||
if ($data = $this->read()) {
|
||||
$this->emit('data', [$data]);
|
||||
}
|
||||
}
|
||||
);
|
||||
|
||||
$this->listening = true;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param WritableStreamInterface $dest
|
||||
* @param array $options
|
||||
* @return WritableStreamInterface
|
||||
* @see ReadableResourceStream::pipe()
|
||||
*/
|
||||
public function pipe(WritableStreamInterface $dest, array $options = [])
|
||||
{
|
||||
return Util::pipe($this, $dest, $options);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @see ReadableResourceStream::close()
|
||||
*/
|
||||
public function close()
|
||||
{
|
||||
if ($this->closed) {
|
||||
return;
|
||||
}
|
||||
|
||||
$this->closed = true;
|
||||
|
||||
$this->emit('close');
|
||||
$this->pause();
|
||||
$this->removeAllListeners();
|
||||
|
||||
$this->input->close();
|
||||
}
|
||||
|
||||
|
||||
private function read(): ?PayloadInterface
|
||||
{
|
||||
try {
|
||||
return $this->input->select();
|
||||
} catch (Throwable $t) {
|
||||
$this->emit('error', [
|
||||
new RuntimeException('Unable to read data from input: ' . $t->getMessage(), 0, $t),
|
||||
]);
|
||||
$this->close();
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
}
|
29
src/ValueObject/PollingInterval.php
Normal file
29
src/ValueObject/PollingInterval.php
Normal file
@ -0,0 +1,29 @@
|
||||
<?php
|
||||
|
||||
namespace JoopSchilder\React\Stream\NonBlockingInput\ValueObject;
|
||||
|
||||
use InvalidArgumentException;
|
||||
|
||||
final class PollingInterval
|
||||
{
|
||||
private const DEFAULT_INTERVAL = 0.05;
|
||||
|
||||
private float $interval;
|
||||
|
||||
|
||||
public function __construct(float $interval = self::DEFAULT_INTERVAL)
|
||||
{
|
||||
if ($interval < 0.0) {
|
||||
throw new InvalidArgumentException('Interval must be greater than 0');
|
||||
}
|
||||
|
||||
$this->interval = $interval;
|
||||
}
|
||||
|
||||
|
||||
public function getInterval(): float
|
||||
{
|
||||
return $this->interval;
|
||||
}
|
||||
|
||||
}
|
Loading…
Reference in New Issue
Block a user