polyphony 1.3 → 1.5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/.yardopts +1 -0
- data/CHANGELOG.md +14 -0
- data/README.md +11 -45
- data/TODO.md +5 -10
- data/docs/installation.md +30 -0
- data/docs/readme.md +11 -45
- data/examples/core/stages.rb +99 -0
- data/examples/pipes/http_server.rb +42 -12
- data/examples/pipes/http_server2.rb +45 -0
- data/ext/polyphony/backend_common.c +1 -1
- data/ext/polyphony/backend_common.h +5 -0
- data/ext/polyphony/backend_io_uring.c +163 -144
- data/ext/polyphony/backend_libev.c +68 -15
- data/ext/polyphony/extconf.rb +6 -2
- data/ext/polyphony/pipe.c +1 -1
- data/ext/polyphony/polyphony.c +22 -25
- data/ext/polyphony/polyphony.h +1 -6
- data/ext/polyphony/win_uio.h +18 -0
- data/lib/polyphony/core/debug.rb +8 -8
- data/lib/polyphony/extensions/io.rb +17 -4
- data/lib/polyphony/extensions/kernel.rb +16 -0
- data/lib/polyphony/extensions/openssl.rb +11 -11
- data/lib/polyphony/extensions/pipe.rb +5 -5
- data/lib/polyphony/extensions/socket.rb +18 -32
- data/lib/polyphony/extensions/timeout.rb +5 -1
- data/lib/polyphony/version.rb +1 -1
- data/test/stress.rb +6 -1
- data/test/test_backend.rb +11 -2
- data/test/test_ext.rb +14 -0
- data/test/test_global_api.rb +5 -5
- data/test/test_io.rb +1 -1
- data/test/test_signal.rb +15 -6
- data/test/test_socket.rb +3 -102
- data/test/test_timer.rb +1 -1
- metadata +10 -6
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 8ec9aae8d4c83ff8dd3332187fdd12fc598596dc4bbab11eaad6400e61769684
|
4
|
+
data.tar.gz: a42cedfc33172dbacc4f61e16fa0a6fce3eb772e78e8829a2dfe744601c77b1c
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 2f6c4a70a56c854dfaa707baeac665ed04adcf04a0f67c25419b2de50d367b155ed6aad3e6bb3dcdc6391af08f4a4dafc3afcadecc499f26cb4b5a06a5d0c69e
|
7
|
+
data.tar.gz: e8b74bd9dbec6a0d05e5510514ccff97792e04c0742e85624087eb92fe55d195b553288c0e3132ab468ba7a28a921ead91ce20ce26d6dab93f5b5a467f5f61bd
|
data/.yardopts
CHANGED
data/CHANGELOG.md
CHANGED
@@ -1,3 +1,17 @@
|
|
1
|
+
## 1.5 2023-07-28
|
2
|
+
|
3
|
+
- Refactor backend_await in io_uring backend
|
4
|
+
- Fix calling `Timeout.timeout` with `nil` or `0` (#114)
|
5
|
+
- Rework support for io_uring multishot accept
|
6
|
+
- Combine SQE submission and waiting for CQE into a single syscall
|
7
|
+
- Use io_uring for closing a `Polyphony::Pipe`, removing call to `close()`
|
8
|
+
|
9
|
+
## 1.4 2023-07-01
|
10
|
+
|
11
|
+
- Implement concurrent `IO#close`
|
12
|
+
- Improve docs
|
13
|
+
- Use only positional arguments in `IO#read` and `IO#readpartial` (#109 @floriandejonckheere)
|
14
|
+
|
1
15
|
## 1.3 2023-06-23
|
2
16
|
|
3
17
|
- Improve cancellation doc page
|
data/README.md
CHANGED
@@ -23,58 +23,24 @@
|
|
23
23
|
## What is Polyphony?
|
24
24
|
|
25
25
|
Polyphony is a library for building concurrent applications in Ruby. Polyphony
|
26
|
-
harnesses the power of [Ruby fibers](https://
|
27
|
-
|
28
|
-
|
29
|
-
[io_uring](https://unixism.net/loti/what_is_io_uring.html) or
|
26
|
+
harnesses the power of [Ruby fibers](https://rubyapi.org/3.2/o/fiber) to provide
|
27
|
+
a cooperative, sequential coroutine-based concurrency model. Under the hood,
|
28
|
+
Polyphony uses [io_uring](https://unixism.net/loti/what_is_io_uring.html) or
|
30
29
|
[libev](https://github.com/enki/libev) to maximize I/O performance.
|
31
30
|
|
32
31
|
## Features
|
33
32
|
|
34
|
-
*
|
35
|
-
*
|
36
|
-
|
37
|
-
|
38
|
-
*
|
39
|
-
|
40
|
-
|
41
|
-
third-party gems such as `pg` and `redis`.
|
42
|
-
* Use stdlib classes such as `TCPServer`, `TCPSocket` and
|
43
|
-
`OpenSSL::SSL::SSLSocket`.
|
44
|
-
* Competitive performance and scalability characteristics, in terms of both
|
45
|
-
throughput and memory consumption.
|
46
|
-
|
47
|
-
## Installing
|
48
|
-
|
49
|
-
### System Requirements
|
50
|
-
|
51
|
-
In order to use Polyphony you need to have:
|
52
|
-
|
53
|
-
- Linux or MacOS (support for Windows will come at a later stage)
|
54
|
-
- Ruby (MRI) 3.1 or newer
|
55
|
-
|
56
|
-
### Installing the Polyphony Gem
|
57
|
-
|
58
|
-
Add this line to your application's Gemfile:
|
59
|
-
|
60
|
-
```ruby
|
61
|
-
gem 'polyphony'
|
62
|
-
```
|
63
|
-
|
64
|
-
And then execute:
|
65
|
-
|
66
|
-
```bash
|
67
|
-
$ bundle
|
68
|
-
```
|
69
|
-
|
70
|
-
Or install it yourself as:
|
71
|
-
|
72
|
-
```bash
|
73
|
-
$ gem install polyphony
|
74
|
-
```
|
33
|
+
* Ruby fibers as the main unit of concurrency.
|
34
|
+
* [Structured concurrency](https://en.wikipedia.org/wiki/Structured_concurrency)
|
35
|
+
coupled with robust exception handling.
|
36
|
+
* Message passing between fibers, even across threads!
|
37
|
+
* High-performance I/O using the core Ruby I/O classes and
|
38
|
+
[io_uring](https://unixism.net/loti/what_is_io_uring.html) with support for
|
39
|
+
[advanced I/O patterns](docs/advanced-io.md).
|
75
40
|
|
76
41
|
## Usage
|
77
42
|
|
43
|
+
- [Installation](docs/installation.md)
|
78
44
|
- [Overview](docs/overview.md)
|
79
45
|
- [Tutorial](docs/tutorial.md)
|
80
46
|
- [All About Cancellation: How to Stop Concurrent Operations](docs/cancellation.md)
|
data/TODO.md
CHANGED
@@ -5,6 +5,10 @@
|
|
5
5
|
- if `io_uring_get_sqe` returns null, call `io_uring_submit`, (snooze fiber)?
|
6
6
|
and try again
|
7
7
|
|
8
|
+
- closing and shutdown:
|
9
|
+
- `Pipe_free()` - can we use the backend to close the pipe fds?
|
10
|
+
- Implement `BasicSocket#shutdown`, add `Backend_shutdown` API.
|
11
|
+
|
8
12
|
- Tracing:
|
9
13
|
- Emit events on I/O ops, e.g.:
|
10
14
|
- [:op_read_submit, id, io, len]
|
@@ -19,16 +23,7 @@
|
|
19
23
|
- More tight loops
|
20
24
|
- `IO#gets_loop`, `Socket#gets_loop`, `OpenSSL::Socket#gets_loop` (medium effort)
|
21
25
|
|
22
|
-
|
23
|
-
|
24
|
-
## Roadmap for Polyphony 1.1
|
25
|
-
|
26
|
-
- io_uring
|
27
|
-
- Use playground.c to find out why we when submitting and waiting for
|
28
|
-
completion in single syscall signals seem to be blocked until the syscall
|
29
|
-
returns. Is this a bug in io_uring/liburing?
|
30
|
-
|
31
|
-
-----------------------------------------------------
|
26
|
+
## Roadmap for Polyphony 2
|
32
27
|
|
33
28
|
- allow backend selection at runtime?
|
34
29
|
- Debugging
|
@@ -0,0 +1,30 @@
|
|
1
|
+
# @title Installation
|
2
|
+
|
3
|
+
# Installation
|
4
|
+
|
5
|
+
## System Requirements
|
6
|
+
|
7
|
+
In order to use Polyphony you need to have:
|
8
|
+
|
9
|
+
- Linux or MacOS (support for Windows will come at a later stage)
|
10
|
+
- Ruby (MRI) 3.1 or newer
|
11
|
+
|
12
|
+
## Installing the Polyphony Gem
|
13
|
+
|
14
|
+
Add this line to your application's Gemfile:
|
15
|
+
|
16
|
+
```ruby
|
17
|
+
gem 'polyphony'
|
18
|
+
```
|
19
|
+
|
20
|
+
And then execute:
|
21
|
+
|
22
|
+
```bash
|
23
|
+
$ bundle
|
24
|
+
```
|
25
|
+
|
26
|
+
Or install it yourself as:
|
27
|
+
|
28
|
+
```bash
|
29
|
+
$ gem install polyphony
|
30
|
+
```
|
data/docs/readme.md
CHANGED
@@ -25,58 +25,24 @@
|
|
25
25
|
## What is Polyphony?
|
26
26
|
|
27
27
|
Polyphony is a library for building concurrent applications in Ruby. Polyphony
|
28
|
-
harnesses the power of [Ruby fibers](https://
|
29
|
-
|
30
|
-
|
31
|
-
[io_uring](https://unixism.net/loti/what_is_io_uring.html) or
|
28
|
+
harnesses the power of [Ruby fibers](https://rubyapi.org/3.2/o/fiber) to provide
|
29
|
+
a cooperative, sequential coroutine-based concurrency model. Under the hood,
|
30
|
+
Polyphony uses [io_uring](https://unixism.net/loti/what_is_io_uring.html) or
|
32
31
|
[libev](https://github.com/enki/libev) to maximize I/O performance.
|
33
32
|
|
34
33
|
## Features
|
35
34
|
|
36
|
-
*
|
37
|
-
*
|
38
|
-
|
39
|
-
|
40
|
-
*
|
41
|
-
|
42
|
-
|
43
|
-
third-party gems such as `pg` and `redis`.
|
44
|
-
* Use stdlib classes such as `TCPServer`, `TCPSocket` and
|
45
|
-
`OpenSSL::SSL::SSLSocket`.
|
46
|
-
* Competitive performance and scalability characteristics, in terms of both
|
47
|
-
throughput and memory consumption.
|
48
|
-
|
49
|
-
## Installing
|
50
|
-
|
51
|
-
### System Requirements
|
52
|
-
|
53
|
-
In order to use Polyphony you need to have:
|
54
|
-
|
55
|
-
- Linux or MacOS (support for Windows will come at a later stage)
|
56
|
-
- Ruby (MRI) 3.1 or newer
|
57
|
-
|
58
|
-
### Installing the Polyphony Gem
|
59
|
-
|
60
|
-
Add this line to your application's Gemfile:
|
61
|
-
|
62
|
-
```ruby
|
63
|
-
gem 'polyphony'
|
64
|
-
```
|
65
|
-
|
66
|
-
And then execute:
|
67
|
-
|
68
|
-
```bash
|
69
|
-
$ bundle
|
70
|
-
```
|
71
|
-
|
72
|
-
Or install it yourself as:
|
73
|
-
|
74
|
-
```bash
|
75
|
-
$ gem install polyphony
|
76
|
-
```
|
35
|
+
* Ruby fibers as the main unit of concurrency.
|
36
|
+
* [Structured concurrency](https://en.wikipedia.org/wiki/Structured_concurrency)
|
37
|
+
coupled with robust exception handling.
|
38
|
+
* Message passing between fibers, even across threads!
|
39
|
+
* High-performance I/O using the core Ruby I/O classes and
|
40
|
+
[io_uring](https://unixism.net/loti/what_is_io_uring.html) with support for
|
41
|
+
{file:/docs/advanced-io.md advanced I/O patterns}.
|
77
42
|
|
78
43
|
## Usage
|
79
44
|
|
45
|
+
- {file:/docs/installation.md Installation}
|
80
46
|
- {file:/docs/overview.md Overview}
|
81
47
|
- {file:/docs/tutorial.md Tutorial}
|
82
48
|
- {file:/docs/cancellation.md All About Cancellation: How to Stop Concurrent Operations}
|
@@ -0,0 +1,99 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'bundler/setup'
|
4
|
+
require 'polyphony'
|
5
|
+
|
6
|
+
# Based on the design of Elixir's GenStage
|
7
|
+
|
8
|
+
class Producer
|
9
|
+
def initialize(mod, *a, **b)
|
10
|
+
extend(mod)
|
11
|
+
setup(*a, **b)
|
12
|
+
@_fiber = spin do
|
13
|
+
receive_loop do |msg|
|
14
|
+
case msg[:kind]
|
15
|
+
when :demand
|
16
|
+
items = handle_demand(msg[:limit])
|
17
|
+
msg[:peer] << items
|
18
|
+
end
|
19
|
+
end
|
20
|
+
end
|
21
|
+
end
|
22
|
+
|
23
|
+
def <<(msg)
|
24
|
+
@_fiber << msg
|
25
|
+
end
|
26
|
+
end
|
27
|
+
|
28
|
+
module Counter
|
29
|
+
def setup(counter = 0)
|
30
|
+
@counter = counter
|
31
|
+
end
|
32
|
+
|
33
|
+
def handle_demand(demand)
|
34
|
+
events = (@counter...@counter + demand).to_a
|
35
|
+
@counter += demand
|
36
|
+
events
|
37
|
+
end
|
38
|
+
end
|
39
|
+
|
40
|
+
counter = Producer.new(Counter, 0)
|
41
|
+
|
42
|
+
class Consumer
|
43
|
+
def initialize(mod, *a, **b)
|
44
|
+
extend(mod)
|
45
|
+
setup(*a, **b) if respond_to?(:setup)
|
46
|
+
@_fiber = spin do
|
47
|
+
while true
|
48
|
+
items = get_items
|
49
|
+
handle_items(items)
|
50
|
+
end
|
51
|
+
end
|
52
|
+
|
53
|
+
@max_demand = 10
|
54
|
+
@min_demand = 5
|
55
|
+
end
|
56
|
+
|
57
|
+
def subscribe(upstream)
|
58
|
+
@upstream = upstream
|
59
|
+
end
|
60
|
+
|
61
|
+
private
|
62
|
+
|
63
|
+
def get_items
|
64
|
+
send_demand(@max_demand) if !@sent_demand
|
65
|
+
items = receive
|
66
|
+
send_demand(@min_demand)
|
67
|
+
items
|
68
|
+
end
|
69
|
+
|
70
|
+
def send_demand(demand)
|
71
|
+
if @upstream
|
72
|
+
@upstream << { peer: Fiber.current, kind: :demand, limit: demand }
|
73
|
+
@sent_demand = true
|
74
|
+
else
|
75
|
+
sleep 0.1
|
76
|
+
end
|
77
|
+
end
|
78
|
+
end
|
79
|
+
|
80
|
+
module Printer
|
81
|
+
def handle_items(items)
|
82
|
+
sleep 1
|
83
|
+
puts "got: #{items.join(' ')}"
|
84
|
+
end
|
85
|
+
end
|
86
|
+
|
87
|
+
# counter << { peer: Fiber.current, kind: :demand, limit: 10 }
|
88
|
+
# r = receive
|
89
|
+
|
90
|
+
# p r: r
|
91
|
+
|
92
|
+
# counter << { peer: Fiber.current, kind: :demand, limit: 10 }
|
93
|
+
# r = receive
|
94
|
+
# p r: r
|
95
|
+
|
96
|
+
printer = Consumer.new(Printer)
|
97
|
+
printer.subscribe(counter)
|
98
|
+
|
99
|
+
sleep
|
@@ -4,33 +4,63 @@ require 'bundler/inline'
|
|
4
4
|
|
5
5
|
gemfile do
|
6
6
|
source 'https://rubygems.org'
|
7
|
-
gem 'h1p'
|
8
|
-
gem 'polyphony', path: '.'
|
7
|
+
gem 'h1p', path: '../h1p'
|
8
|
+
# gem 'polyphony', path: '.'
|
9
9
|
end
|
10
10
|
|
11
|
-
require 'polyphony'
|
11
|
+
# require 'polyphony'
|
12
12
|
require 'h1p'
|
13
13
|
|
14
|
-
|
15
|
-
|
16
|
-
|
14
|
+
module ::Kernel
|
15
|
+
def trace(*args)
|
16
|
+
STDOUT << format_trace(args)
|
17
|
+
end
|
17
18
|
|
18
|
-
|
19
|
-
|
20
|
-
|
19
|
+
def format_trace(args)
|
20
|
+
if args.first.is_a?(String)
|
21
|
+
if args.size > 1
|
22
|
+
format("%s: %p\n", args.shift, args)
|
23
|
+
else
|
24
|
+
format("%s\n", args.first)
|
25
|
+
end
|
26
|
+
else
|
27
|
+
format("%p\n", args.size == 1 ? args.first : args)
|
28
|
+
end
|
29
|
+
end
|
21
30
|
|
22
|
-
|
31
|
+
def monotonic_clock
|
32
|
+
::Process.clock_gettime(::Process::CLOCK_MONOTONIC)
|
33
|
+
end
|
34
|
+
end
|
23
35
|
|
36
|
+
def handle_client(conn)
|
37
|
+
Thread.new do
|
38
|
+
reader = proc do |len, buf, buf_pos|
|
39
|
+
trace(len:, buf:, buf_pos:)
|
40
|
+
s = conn.readpartial(len)
|
41
|
+
buf ? (buf << s) : +s
|
42
|
+
rescue EOFError
|
43
|
+
nil
|
44
|
+
end
|
45
|
+
parser = H1P::Parser.new(reader, :server)
|
46
|
+
# parser = H1P::Parser.new(conn, :server)
|
47
|
+
while (headers = parser.parse_headers)
|
48
|
+
parser.read_body unless parser.complete?
|
24
49
|
conn << "HTTP/1.1 200 OK\r\nContent-Length: 14\r\n\r\nHello, world!\n"
|
25
50
|
end
|
26
|
-
rescue Errno::ECONNRESET
|
51
|
+
rescue Errno::ECONNRESET, Errno::EPIPE
|
27
52
|
# ignore
|
28
53
|
rescue H1P::Error
|
29
54
|
puts 'Got invalid request, closing connection...'
|
30
55
|
ensure
|
56
|
+
parser = nil
|
31
57
|
conn.close rescue nil
|
32
58
|
end
|
33
59
|
end
|
34
60
|
|
35
61
|
puts "Serving HTTP on port 1234..."
|
36
|
-
TCPServer.new('0.0.0.0', 1234)
|
62
|
+
s = TCPServer.new('0.0.0.0', 1234)
|
63
|
+
while true
|
64
|
+
c = s.accept
|
65
|
+
handle_client(c)
|
66
|
+
end
|
@@ -0,0 +1,45 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'bundler/inline'
|
4
|
+
|
5
|
+
gemfile do
|
6
|
+
source 'https://rubygems.org'
|
7
|
+
gem 'http_parser.rb'
|
8
|
+
gem 'polyphony', path: '.'
|
9
|
+
end
|
10
|
+
|
11
|
+
require 'polyphony'
|
12
|
+
require 'http_parser.rb'
|
13
|
+
|
14
|
+
def handle_client(conn)
|
15
|
+
spin do
|
16
|
+
parser = Http::Parser.new
|
17
|
+
done = false
|
18
|
+
headers = nil
|
19
|
+
parser.on_headers_complete = proc do |h|
|
20
|
+
headers = h
|
21
|
+
headers[':method'] = parser.http_method
|
22
|
+
headers[':path'] = parser.request_url
|
23
|
+
end
|
24
|
+
parser.on_message_complete = proc { done = true }
|
25
|
+
|
26
|
+
while true # assuming persistent connection
|
27
|
+
conn.read_loop do |msg|
|
28
|
+
parser << msg
|
29
|
+
break if done
|
30
|
+
end
|
31
|
+
|
32
|
+
conn << "HTTP/1.1 200 OK\r\nContent-Length: 14\r\n\r\nHello, world!\n"
|
33
|
+
done = false
|
34
|
+
headers = nil
|
35
|
+
end
|
36
|
+
rescue Errno::ECONNRESET, Errno::EPIPE
|
37
|
+
# ignore
|
38
|
+
ensure
|
39
|
+
parser = nil
|
40
|
+
conn.close rescue nil
|
41
|
+
end
|
42
|
+
end
|
43
|
+
|
44
|
+
puts "Serving HTTP on port 1234..."
|
45
|
+
TCPServer.new('0.0.0.0', 1234).accept_loop { |c| handle_client(c) }
|
@@ -458,7 +458,7 @@ VALUE Backend_stats(VALUE self) {
|
|
458
458
|
|
459
459
|
VALUE Backend_verify_blocking_mode(VALUE self, VALUE io, VALUE blocking) {
|
460
460
|
io_verify_blocking_mode(io, rb_io_descriptor(io), blocking);
|
461
|
-
return
|
461
|
+
return io;
|
462
462
|
}
|
463
463
|
|
464
464
|
void backend_setup_stats_symbols(void) {
|
@@ -2,9 +2,14 @@
|
|
2
2
|
#define BACKEND_COMMON_H
|
3
3
|
|
4
4
|
#include <sys/types.h>
|
5
|
+
#ifdef POLYPHONY_WINDOWS
|
6
|
+
#include <winsock2.h>
|
7
|
+
#else
|
5
8
|
#include <arpa/inet.h>
|
6
9
|
#include <netinet/in.h>
|
7
10
|
#include <netdb.h>
|
11
|
+
#include <sys/socket.h>
|
12
|
+
#endif
|
8
13
|
|
9
14
|
#include "ruby.h"
|
10
15
|
#include "ruby/io.h"
|