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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 537c28ec35d12c724a5f5ef4117ada1641d0b49b780757965615665bddbe995a
4
- data.tar.gz: 5c11137a7a7dc6774ec815ec8b38791752f1beb939311b5968f0d0b242c7f8ad
3
+ metadata.gz: 8ec9aae8d4c83ff8dd3332187fdd12fc598596dc4bbab11eaad6400e61769684
4
+ data.tar.gz: a42cedfc33172dbacc4f61e16fa0a6fce3eb772e78e8829a2dfe744601c77b1c
5
5
  SHA512:
6
- metadata.gz: 912c4a3cea0a183350c939e588bba70656fcd14a07af6bd5912e3c2208cbb286150477d050482b030c494b0f87f21da7ec350c50f5bfee8e563bc21211143adc
7
- data.tar.gz: c72b01fcef19a525cef49b277f0f1fe949956499be51f34f75fe1e87fc6efe9222a43ac96d282460610c3a82989934a0f661e78a27736120117c1ef391036dd6
6
+ metadata.gz: 2f6c4a70a56c854dfaa707baeac665ed04adcf04a0f67c25419b2de50d367b155ed6aad3e6bb3dcdc6391af08f4a4dafc3afcadecc499f26cb4b5a06a5d0c69e
7
+ data.tar.gz: e8b74bd9dbec6a0d05e5510514ccff97792e04c0742e85624087eb92fe55d195b553288c0e3132ab468ba7a28a921ead91ce20ce26d6dab93f5b5a467f5f61bd
data/.yardopts CHANGED
@@ -18,6 +18,7 @@
18
18
  ./ext/polyphony
19
19
  -
20
20
  docs/readme.md
21
+ docs/installation.md
21
22
  docs/overview.md
22
23
  docs/tutorial.md
23
24
  docs/cancellation.md
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://ruby-doc.org/core-2.5.1/Fiber.html)
27
- to provide a cooperative, sequential coroutine-based concurrency model. Under
28
- the hood, Polyphony uses
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
- * Co-operative scheduling of concurrent tasks using Ruby fibers.
35
- * High-performance event reactor for handling I/O events and timers.
36
- * Natural, sequential programming style that makes it easy to reason about
37
- concurrent code.
38
- * Abstractions and constructs for controlling the execution of concurrent code:
39
- supervisors, cancel scopes, throttling, resource pools etc.
40
- * Code can use native networking classes and libraries, growing support for
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
- - Add support for `close` to io_uring backend
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://ruby-doc.org/core-2.5.1/Fiber.html)
29
- to provide a cooperative, sequential coroutine-based concurrency model. Under
30
- the hood, Polyphony uses
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
- * Co-operative scheduling of concurrent tasks using Ruby fibers.
37
- * High-performance event reactor for handling I/O events and timers.
38
- * Natural, sequential programming style that makes it easy to reason about
39
- concurrent code.
40
- * Abstractions and constructs for controlling the execution of concurrent code:
41
- supervisors, cancel scopes, throttling, resource pools etc.
42
- * Code can use native networking classes and libraries, growing support for
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
- def handle_client(conn)
15
- spin do
16
- parser = H1P::Parser.new(conn, :server)
14
+ module ::Kernel
15
+ def trace(*args)
16
+ STDOUT << format_trace(args)
17
+ end
17
18
 
18
- while true # assuming persistent connection
19
- headers = parser.parse_headers
20
- break unless headers
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
- parser.read_body unless parser.complete?
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).accept_loop { |c| handle_client(c) }
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 self;
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"