polyphony 0.43.1 → 0.43.6
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/.github/workflows/test.yml +1 -1
- data/CHANGELOG.md +37 -0
- data/Gemfile.lock +2 -2
- data/README.md +2 -1
- data/docs/_includes/head.html +40 -0
- data/docs/_includes/title.html +1 -0
- data/docs/_user-guide/web-server.md +11 -11
- data/docs/getting-started/overview.md +2 -2
- data/docs/index.md +4 -3
- data/docs/main-concepts/design-principles.md +23 -34
- data/docs/main-concepts/fiber-scheduling.md +1 -1
- data/docs/polyphony-logo.png +0 -0
- data/examples/adapters/concurrent-ruby.rb +9 -0
- data/examples/adapters/redis_blpop.rb +12 -0
- data/examples/core/xx-daemon.rb +14 -0
- data/examples/io/xx-happy-eyeballs.rb +21 -22
- data/examples/io/xx-zip.rb +19 -0
- data/examples/performance/mem-usage.rb +34 -28
- data/examples/performance/messaging.rb +29 -0
- data/examples/performance/multi_snooze.rb +11 -9
- data/examples/xx-spin.rb +32 -0
- data/ext/polyphony/libev_agent.c +242 -145
- data/ext/polyphony/libev_queue.c +129 -57
- data/ext/polyphony/polyphony.h +12 -5
- data/ext/polyphony/ring_buffer.c +120 -0
- data/ext/polyphony/ring_buffer.h +28 -0
- data/ext/polyphony/thread.c +13 -7
- data/lib/polyphony.rb +29 -10
- data/lib/polyphony/adapters/redis.rb +3 -2
- data/lib/polyphony/core/global_api.rb +5 -3
- data/lib/polyphony/core/resource_pool.rb +19 -9
- data/lib/polyphony/core/thread_pool.rb +1 -1
- data/lib/polyphony/extensions/core.rb +40 -0
- data/lib/polyphony/extensions/fiber.rb +9 -14
- data/lib/polyphony/extensions/io.rb +17 -16
- data/lib/polyphony/extensions/openssl.rb +8 -0
- data/lib/polyphony/extensions/socket.rb +12 -0
- data/lib/polyphony/version.rb +1 -1
- data/test/q.rb +24 -0
- data/test/test_agent.rb +13 -7
- data/test/test_fiber.rb +3 -3
- data/test/test_global_api.rb +50 -17
- data/test/test_io.rb +10 -2
- data/test/test_queue.rb +26 -1
- data/test/test_resource_pool.rb +12 -0
- data/test/test_socket.rb +43 -0
- data/test/test_throttler.rb +6 -5
- metadata +13 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: b048cfa1e0d7cd542840ab9f2198349b2474d40a6d943b584d1bcb57593ca41d
|
4
|
+
data.tar.gz: 0d9ed9fc45ec2165018e61f8befbdefb5668cfcf4b7628a648a055f5b6052e4c
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: a962cb0db032fd58a8db4024d9a6ad2d1be5307caff72d9508bdf879bafbd6c55a5c79e9bc69a9d3bda96d2f72fda51ed2f978e96fa0ca636458284e9e9dbbaf
|
7
|
+
data.tar.gz: 2b419313309e898003399f780abcf7641de34c3350ae5d7c10c785209e3e6c12e4d741b01676a49ec1ec2f3f15ea8f9001636ec44880e13583d68168f82c9cb7
|
data/.github/workflows/test.yml
CHANGED
data/CHANGELOG.md
CHANGED
@@ -1,3 +1,40 @@
|
|
1
|
+
## 0.43.6 2020-07-18
|
2
|
+
|
3
|
+
* Allow brute-force interrupting with second Ctrl-C
|
4
|
+
* Fix outgoing SSL connections (#28)
|
5
|
+
* Improve Fiber#await_all_children with many children
|
6
|
+
* Use `writev` for writing multiple strings
|
7
|
+
* Add logo (thanks [Gerald](https://webocube.com/)!)
|
8
|
+
|
9
|
+
## 0.43.5 2020-07-13
|
10
|
+
|
11
|
+
* Fix `#read_nonblock`, `#write_nonblock` for `IO` and `Socket` (#27)
|
12
|
+
* Patch `Kernel#p`, `IO#puts` to issue single write call
|
13
|
+
* Add support for multiple arguments in `IO#write` and `LibevAgent#write`
|
14
|
+
* Use LibevQueue for fiber run queue
|
15
|
+
* Reimplement LibevQueue as ring buffer
|
16
|
+
|
17
|
+
## 0.43.4 2020-07-09
|
18
|
+
|
19
|
+
* Reimplement Kernel#trap
|
20
|
+
* Dynamically allocate read buffer if length not given (#23)
|
21
|
+
* Prevent CPU saturation on infinite sleep (#24)
|
22
|
+
|
23
|
+
## 0.43.3 2020-07-08
|
24
|
+
|
25
|
+
* Fix behaviour after call to `Process.daemon` (#8)
|
26
|
+
* Replace core `Queue` class with `Polyphony::Queue` (#22)
|
27
|
+
* Make `ResourcePool` reentrant (#1)
|
28
|
+
* Accept `:with_exception` argument in `cancel_after` (#16)
|
29
|
+
|
30
|
+
## 0.43.2 2020-07-07
|
31
|
+
|
32
|
+
* Fix sending Redis commands with array arguments (#21)
|
33
|
+
|
34
|
+
## 0.43.1 2020-06
|
35
|
+
|
36
|
+
* Fix compiling C-extension on MacOS (#20)
|
37
|
+
|
1
38
|
## 0.43 2020-07-05
|
2
39
|
|
3
40
|
* Add IO#read_loop
|
data/Gemfile.lock
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
PATH
|
2
2
|
remote: .
|
3
3
|
specs:
|
4
|
-
polyphony (0.43.
|
4
|
+
polyphony (0.43.6)
|
5
5
|
|
6
6
|
GEM
|
7
7
|
remote: https://rubygems.org/
|
@@ -12,7 +12,7 @@ GEM
|
|
12
12
|
ast (2.4.0)
|
13
13
|
builder (3.2.4)
|
14
14
|
colorator (1.1.0)
|
15
|
-
concurrent-ruby (1.1.
|
15
|
+
concurrent-ruby (1.1.6)
|
16
16
|
docile (1.3.2)
|
17
17
|
em-websocket (0.5.1)
|
18
18
|
eventmachine (>= 0.12.9)
|
data/README.md
CHANGED
@@ -1,5 +1,6 @@
|
|
1
|
-
|
1
|
+
<p align="center"><img src="docs/polyphony-logo.png" /></p>
|
2
2
|
|
3
|
+
# Polyphony - Fine-Grained Concurrency for Ruby
|
3
4
|
|
4
5
|
[](http://rubygems.org/gems/polyphony)
|
5
6
|
[](https://github.com/digital-fabric/polyphony/actions?query=workflow%3ATests)
|
@@ -0,0 +1,40 @@
|
|
1
|
+
<head>
|
2
|
+
<meta charset="UTF-8">
|
3
|
+
<meta http-equiv="X-UA-Compatible" content="IE=Edge">
|
4
|
+
|
5
|
+
{% unless site.plugins contains "jekyll-seo-tag" %}
|
6
|
+
<title>{{ page.title }} - {{ site.title }}</title>
|
7
|
+
|
8
|
+
{% if page.description %}
|
9
|
+
<meta name="Description" content="{{ page.description }}">
|
10
|
+
{% endif %}
|
11
|
+
{% endunless %}
|
12
|
+
|
13
|
+
<link rel="shortcut icon" href="{{ 'polyphony-logo.png' | absolute_url }}" type="image/png">
|
14
|
+
|
15
|
+
<link rel="stylesheet" href="{{ '/assets/css/just-the-docs-default.css' | absolute_url }}">
|
16
|
+
|
17
|
+
{% if site.ga_tracking != nil %}
|
18
|
+
<script async src="https://www.googletagmanager.com/gtag/js?id={{ site.ga_tracking }}"></script>
|
19
|
+
<script>
|
20
|
+
window.dataLayer = window.dataLayer || [];
|
21
|
+
function gtag(){dataLayer.push(arguments);}
|
22
|
+
gtag('js', new Date());
|
23
|
+
|
24
|
+
gtag('config', '{{ site.ga_tracking }}'{% unless site.ga_tracking_anonymize_ip == nil %}, { 'anonymize_ip': true }{% endunless %});
|
25
|
+
</script>
|
26
|
+
|
27
|
+
{% endif %}
|
28
|
+
|
29
|
+
{% if site.search_enabled != false %}
|
30
|
+
<script type="text/javascript" src="{{ '/assets/js/vendor/lunr.min.js' | absolute_url }}"></script>
|
31
|
+
{% endif %}
|
32
|
+
<script type="text/javascript" src="{{ '/assets/js/just-the-docs.js' | absolute_url }}"></script>
|
33
|
+
|
34
|
+
<meta name="viewport" content="width=device-width, initial-scale=1">
|
35
|
+
|
36
|
+
{% seo %}
|
37
|
+
|
38
|
+
{% include head_custom.html %}
|
39
|
+
|
40
|
+
</head>
|
@@ -0,0 +1 @@
|
|
1
|
+
<img src="{{ 'polyphony-logo.png' | absolute_url }}" style="height: 1.5em; margin-right: 0.5em">Polyphony
|
@@ -26,9 +26,9 @@ the entire request body.
|
|
26
26
|
## A basic web server
|
27
27
|
|
28
28
|
```ruby
|
29
|
-
require '
|
29
|
+
require 'tipi'
|
30
30
|
|
31
|
-
|
31
|
+
Tipi.serve('0.0.0.0', 1234) do |request|
|
32
32
|
request.respond("Hello world!\n")
|
33
33
|
end
|
34
34
|
```
|
@@ -55,13 +55,13 @@ TLS termination can be handled by passing a `secure_context` option to the
|
|
55
55
|
server:
|
56
56
|
|
57
57
|
```ruby
|
58
|
-
require '
|
58
|
+
require 'tipi'
|
59
59
|
require 'localhost/authority'
|
60
60
|
|
61
61
|
authority = Localhost::Authority.fetch
|
62
62
|
opts = { secure_context: authority.server_context }
|
63
63
|
|
64
|
-
|
64
|
+
Tipi.serve('0.0.0.0', 1234, opts) do |request|
|
65
65
|
request.respond("Hello world!\n")
|
66
66
|
end
|
67
67
|
```
|
@@ -72,8 +72,8 @@ Polyphony's web server makes it really easy to integrate websocket communication
|
|
72
72
|
with normal HTTP processing:
|
73
73
|
|
74
74
|
```ruby
|
75
|
-
require '
|
76
|
-
require '
|
75
|
+
require 'tipi'
|
76
|
+
require 'tipi/websocket'
|
77
77
|
|
78
78
|
ws_handler = Polyphony::Websocket.handler do |ws|
|
79
79
|
while (msg = ws.recv)
|
@@ -85,7 +85,7 @@ opts = {
|
|
85
85
|
upgrade: { websocket: ws_handler }
|
86
86
|
}
|
87
87
|
|
88
|
-
|
88
|
+
Tipi.serve('0.0.0.0', 1234, opts) do |request|
|
89
89
|
request.respond("Hello world!\n")
|
90
90
|
end
|
91
91
|
```
|
@@ -93,7 +93,7 @@ end
|
|
93
93
|
Polyphony also supports general-purpose HTTP upgrades using the same mechanism:
|
94
94
|
|
95
95
|
```ruby
|
96
|
-
require '
|
96
|
+
require 'tipi'
|
97
97
|
|
98
98
|
opts = {
|
99
99
|
upgrade: {
|
@@ -105,7 +105,7 @@ opts = {
|
|
105
105
|
}
|
106
106
|
}
|
107
107
|
|
108
|
-
|
108
|
+
Tipi.serve('0.0.0.0', 1234, opts) do |request|
|
109
109
|
request.respond("Hello world!\n")
|
110
110
|
end
|
111
111
|
```
|
@@ -117,7 +117,7 @@ and enables streaming (using chunked encoding for HTTP/1.1 connections). Here's
|
|
117
117
|
an example of an SSE response:
|
118
118
|
|
119
119
|
```ruby
|
120
|
-
require '
|
120
|
+
require 'tipi'
|
121
121
|
|
122
122
|
def sse_response(request)
|
123
123
|
request.send_headers('Content-Type': 'text/event-stream')
|
@@ -131,6 +131,6 @@ ensure
|
|
131
131
|
request.send_chunk("retry: 0\n\n", done: true)
|
132
132
|
end
|
133
133
|
|
134
|
-
|
134
|
+
Tipi.serve('0.0.0.0', 1234, &method(:sse_response))
|
135
135
|
```
|
136
136
|
|
@@ -279,7 +279,7 @@ def chat_user_handler(user_name, connection)
|
|
279
279
|
while command = connection.gets
|
280
280
|
case command
|
281
281
|
when /^connect (.+)/
|
282
|
-
room&.send [:
|
282
|
+
room&.send [:subscribe, message_subscriber]
|
283
283
|
room = CHAT_ROOMS[$1]
|
284
284
|
when "disconnect"
|
285
285
|
room&.send [:unsubscribe, message_subscriber]
|
@@ -483,4 +483,4 @@ reach version 1.0. Here are some of the exciting directions we're working on.
|
|
483
483
|
- Support for more core and stdlib APIs
|
484
484
|
- More adapters for gems with C-extensions, such as `mysql`, `sqlite3` etc
|
485
485
|
- Use `io_uring` agent as alternative to the libev agent
|
486
|
-
- More concurrency constructs for building highly concurrent applications
|
486
|
+
- More concurrency constructs for building highly concurrent applications
|
data/docs/index.md
CHANGED
@@ -6,6 +6,8 @@ permalink: /
|
|
6
6
|
next_title: Installing Polyphony
|
7
7
|
---
|
8
8
|
|
9
|
+
<p align="center"><img src="{{ 'polyphony-logo.png' | absolute_url }}" /></p>
|
10
|
+
|
9
11
|
# Polyphony
|
10
12
|
{:.text-center .logo-title}
|
11
13
|
|
@@ -18,11 +20,10 @@ implements a comprehensive
|
|
18
20
|
using [libev](https://github.com/enki/libev) as a high-performance event reactor
|
19
21
|
for I/O, timers, and other asynchronous events.
|
20
22
|
|
23
|
+
[Overview](getting-started/overview){: .btn .btn-green .text-gamma }
|
21
24
|
[Take the tutorial](getting-started/tutorial){: .btn .btn-blue .text-gamma }
|
22
|
-
[Main Concepts](main-concepts/concurrency/){: .btn .btn-green .text-gamma }
|
23
|
-
[FAQ](faq){: .btn .btn-green .text-gamma }
|
24
25
|
[Source code](https://github.com/digital-fabric/polyphony){: .btn .btn-purple .text-gamma target="_blank" }
|
25
|
-
{
|
26
|
+
{:.text-center .mt-6 .h-align-center }
|
26
27
|
|
27
28
|
## Focused on Developer Happiness
|
28
29
|
|
@@ -6,7 +6,7 @@ parent: Main Concepts
|
|
6
6
|
permalink: /main-concepts/design-principles/
|
7
7
|
prev_title: Extending Polyphony
|
8
8
|
---
|
9
|
-
# The Design of Polyphony
|
9
|
+
# The Design of Polyphony
|
10
10
|
|
11
11
|
Polyphony is a new gem that aims to enable developing high-performance
|
12
12
|
concurrent applications in Ruby using a fluent, compact syntax and API.
|
@@ -47,7 +47,7 @@ Nevertheless, while work is being done to harness fibers for providing a better
|
|
47
47
|
way to do concurrency in Ruby, fibers remain a mistery for most Ruby
|
48
48
|
programmers, a perplexing unfamiliar corner right at the heart of Ruby.
|
49
49
|
|
50
|
-
##
|
50
|
+
## The History of Polyphony
|
51
51
|
|
52
52
|
Polyphony started as an experiment, but over about two years of slow, jerky
|
53
53
|
evolution turned into something I'm really excited to share with the Ruby
|
@@ -58,31 +58,24 @@ Polyphony today as nothing like the way it began. A careful examination of the
|
|
58
58
|
[CHANGELOG](https://github.com/digital-fabric/polyphony/blob/master/CHANGELOG.md)
|
59
59
|
would show how Polyphony explored not only different event reactor designs, but
|
60
60
|
also different API designs incorporating various concurrent paradigms such as
|
61
|
-
promises, async/await, fibers, and finally structured concurrency.
|
62
|
-
|
63
|
-
While Polyphony, like nio4r or EventMachine, uses an event reactor to turn
|
64
|
-
blocking operations into non-blocking ones, it completely embraces fibers and in
|
65
|
-
fact does not provide any callback-based APIs. Furthermore, Polyphony provides
|
66
|
-
fullblown fiber-aware implementations of blocking operations, such as
|
67
|
-
`read/write`, `sleep` or `waitpid`, instead of just event watching primitives.
|
61
|
+
promises, async/await, fibers, and finally structured concurrency.
|
68
62
|
|
69
63
|
Throughout the development process, it was my intention to create a programming
|
70
|
-
interface that would make highly-concurrent
|
71
|
-
|
72
|
-
|
73
|
-
|
74
|
-
|
64
|
+
interface that would make it easy to author highly-concurrent Ruby programs.
|
75
65
|
|
66
|
+
## Design Principles
|
76
67
|
|
68
|
+
While Polyphony, like nio4r or EventMachine, uses an event reactor to turn
|
69
|
+
blocking operations into non-blocking ones, it completely embraces fibers and in
|
70
|
+
fact does not provide any callback-based APIs.
|
77
71
|
|
78
|
-
|
79
|
-
|
72
|
+
Furthermore, Polyphony provides fullblown fiber-aware implementations of
|
73
|
+
blocking operations, such as `read/write`, `sleep` or `waitpid`, instead of just
|
74
|
+
event watching primitives.
|
80
75
|
|
81
|
-
|
82
|
-
[libev](http://pod.tst.eu/http://cvs.schmorp.de/libev/ev.pod) event reactor
|
83
|
-
library. Polyphony's design is based on the following principles:
|
76
|
+
Polyphony's design is based on the following principles:
|
84
77
|
|
85
|
-
-
|
78
|
+
- The concurrency model should feel "baked-in". The API should allow
|
86
79
|
concurrency with minimal effort. Polyphony should facilitate writing both
|
87
80
|
large apps and small scripts with as little boilerplate code as possible.
|
88
81
|
There should be no calls to initialize the event reactor, or other ceremonial
|
@@ -91,8 +84,8 @@ library. Polyphony's design is based on the following principles:
|
|
91
84
|
```ruby
|
92
85
|
require 'polyphony'
|
93
86
|
|
94
|
-
# start 10 fibers, each sleeping for
|
95
|
-
10.times { spin { sleep
|
87
|
+
# start 10 fibers, each sleeping for 3 seconds
|
88
|
+
10.times { spin { sleep 3 } }
|
96
89
|
|
97
90
|
puts 'going to sleep now'
|
98
91
|
# wait for other fibers to terminate
|
@@ -106,14 +99,14 @@ library. Polyphony's design is based on the following principles:
|
|
106
99
|
```ruby
|
107
100
|
# in Polyphony, I/O ops might block the current fiber, but implicitly yield to
|
108
101
|
# other concurrent fibers:
|
109
|
-
clients.each
|
102
|
+
clients.each do |client|
|
110
103
|
spin { client.puts 'Elvis has left the chatroom' }
|
111
|
-
|
104
|
+
end
|
112
105
|
```
|
113
106
|
|
114
107
|
- Concurrency primitives should be accessible using idiomatic Ruby techniques
|
115
108
|
(blocks, method chaining...) and should feel as much as possible "part of the
|
116
|
-
language". The resulting API is based
|
109
|
+
language". The resulting API is fundamentally based on methods rather than classes,
|
117
110
|
for example `spin` or `move_on_after`, leading to a coding style that is both
|
118
111
|
more compact and more legible:
|
119
112
|
|
@@ -125,8 +118,6 @@ library. Polyphony's design is based on the following principles:
|
|
125
118
|
}
|
126
119
|
```
|
127
120
|
|
128
|
-
- Breaking up operations into
|
129
|
-
|
130
121
|
- Polyphony should embrace Ruby's standard `raise/rescue/ensure` exception
|
131
122
|
handling mechanism. Exception handling in a highly concurrent environment
|
132
123
|
should be robust and foolproof:
|
@@ -147,7 +138,8 @@ library. Polyphony's design is based on the following principles:
|
|
147
138
|
constructs through composition.
|
148
139
|
|
149
140
|
- The entire design should embrace fibers. There should be no callback-based
|
150
|
-
asynchronous APIs.
|
141
|
+
asynchronous APIs. The library and its ecosystem will foster the development
|
142
|
+
of techniques and tools for converting callback-based APIs to fiber-based ones.
|
151
143
|
|
152
144
|
- Use of extensive monkey patching of Ruby core modules and classes such as
|
153
145
|
`Kernel`, `Fiber`, `IO` and `Timeout`. This allows porting over non-Polyphony
|
@@ -160,13 +152,10 @@ library. Polyphony's design is based on the following principles:
|
|
160
152
|
# use TCPServer from Ruby's stdlib
|
161
153
|
server = TCPServer.open('127.0.0.1', 1234)
|
162
154
|
while (client = server.accept)
|
163
|
-
spin
|
155
|
+
spin {
|
164
156
|
while (data = client.gets)
|
165
|
-
client.write(
|
157
|
+
client.write("you said: #{ data.chomp }\n")
|
166
158
|
end
|
167
|
-
|
159
|
+
}
|
168
160
|
end
|
169
161
|
```
|
170
|
-
|
171
|
-
- Development of techniques and tools for converting callback-based APIs to
|
172
|
-
fiber-based ones.
|
@@ -44,7 +44,7 @@ pong = Fiber.new { loop { puts "pong"; ping.transfer } }
|
|
44
44
|
ping.transfer
|
45
45
|
```
|
46
46
|
|
47
|
-
`Fiber#
|
47
|
+
`Fiber#transfer` also allows using the main fiber as a general purpose
|
48
48
|
resumable execution context. For that reason, Polyphony uses `Fiber#transfer`
|
49
49
|
exclusively for scheduling fibers. Normally, however, applications based on
|
50
50
|
Polyphony will not use this API directly.
|
data/docs/polyphony-logo.png
CHANGED
Binary file
|
@@ -0,0 +1,12 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'bundler/setup'
|
4
|
+
require 'polyphony/adapters/redis'
|
5
|
+
# require 'redis'
|
6
|
+
|
7
|
+
redis = Redis.new(host: ENV['REDISHOST'] || 'localhost')
|
8
|
+
|
9
|
+
redis.lpush("queue_key", "omgvalue")
|
10
|
+
puts "len: #{redis.llen("queue_key")}"
|
11
|
+
result = redis.blpop("queue_key")
|
12
|
+
puts result.inspect
|
@@ -0,0 +1,14 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'bundler/setup'
|
4
|
+
require 'polyphony'
|
5
|
+
|
6
|
+
Exception.__disable_sanitized_backtrace__ = true
|
7
|
+
|
8
|
+
puts "pid: #{Process.pid}"
|
9
|
+
|
10
|
+
Process.daemon(true, true)
|
11
|
+
|
12
|
+
Polyphony::ThreadPool.process do
|
13
|
+
puts "Hello world from pid #{Process.pid}"
|
14
|
+
end
|
@@ -1,37 +1,36 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
# idea taken from the example given in trio:
|
4
|
-
# https://www.youtube.com/watch?v=oLkfnc_UMcE
|
5
|
-
|
6
|
-
require 'bundler/setup'
|
7
3
|
require 'polyphony'
|
8
4
|
|
9
|
-
def try_connect(
|
10
|
-
puts "trying #{
|
11
|
-
|
12
|
-
|
13
|
-
|
5
|
+
def try_connect(ip_address, port, supervisor)
|
6
|
+
puts "trying #{ip_address}"
|
7
|
+
sleep rand * 0.2
|
8
|
+
socket = TCPSocket.new(ip_address, port)
|
9
|
+
puts "connected to #{ip_address}"
|
10
|
+
supervisor.schedule [ip_address, socket]
|
14
11
|
rescue IOError, SystemCallError
|
15
12
|
# ignore error
|
16
13
|
end
|
17
14
|
|
18
|
-
def happy_eyeballs(hostname, port, max_wait_time: 0.
|
15
|
+
def happy_eyeballs(hostname, port, max_wait_time: 0.010)
|
19
16
|
targets = Socket.getaddrinfo(hostname, port, :INET, :STREAM)
|
20
17
|
t0 = Time.now
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
|
26
|
-
|
27
|
-
end
|
28
|
-
if success
|
29
|
-
puts format('success: %s (%.3fs)', success[0], Time.now - t0)
|
30
|
-
else
|
31
|
-
puts "timed out (#{Time.now - t0}s)"
|
18
|
+
fibers = []
|
19
|
+
supervisor = Fiber.current
|
20
|
+
spin do
|
21
|
+
targets.each do |t|
|
22
|
+
spin { try_connect(t[2], t[1], supervisor) }
|
23
|
+
sleep(max_wait_time)
|
32
24
|
end
|
25
|
+
suspend
|
26
|
+
end
|
27
|
+
target, socket = move_on_after(5) { suspend }
|
28
|
+
supervisor.shutdown_all_children
|
29
|
+
if target
|
30
|
+
puts format('success: %s (%.3fs)', target, Time.now - t0)
|
31
|
+
else
|
32
|
+
puts 'timed out'
|
33
33
|
end
|
34
34
|
end
|
35
35
|
|
36
|
-
# Let's try it out:
|
37
36
|
happy_eyeballs('debian.org', 'https')
|