polyphony 0.42 → 0.43.4
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/CHANGELOG.md +27 -0
- data/Gemfile.lock +2 -2
- data/README.md +0 -1
- data/TODO.md +1 -2
- data/docs/_config.yml +2 -2
- data/docs/_sass/custom/custom.scss +10 -0
- data/docs/favicon.ico +0 -0
- data/docs/getting-started/overview.md +3 -24
- data/docs/index.md +5 -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/performance/mem-usage.rb +34 -28
- data/examples/performance/thread-vs-fiber/polyphony_server_read_loop.rb +1 -1
- data/ext/polyphony/libev_agent.c +161 -146
- data/ext/polyphony/libev_queue.c +5 -4
- data/ext/polyphony/polyphony.c +0 -6
- data/ext/polyphony/polyphony_ext.c +0 -2
- data/ext/polyphony/thread.c +0 -6
- data/lib/polyphony.rb +26 -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/extensions/core.rb +31 -0
- data/lib/polyphony/extensions/fiber.rb +0 -12
- data/lib/polyphony/extensions/io.rb +5 -1
- data/lib/polyphony/extensions/openssl.rb +34 -10
- data/lib/polyphony/extensions/socket.rb +2 -2
- data/lib/polyphony/version.rb +1 -1
- data/test/test_agent.rb +13 -7
- data/test/test_fiber.rb +3 -3
- data/test/test_global_api.rb +48 -15
- data/test/test_resource_pool.rb +12 -0
- data/test/test_socket.rb +5 -4
- data/test/test_throttler.rb +6 -5
- metadata +7 -3
- data/ext/polyphony/socket.c +0 -213
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 9481b252526ebc3f8e0f5a0121eebb68492bc0c1502cfed447da7b0648e6095b
|
4
|
+
data.tar.gz: dfd2d5d0415833f14f9ebe2cc6336e87d0951ce86f07ebcdec0ad892f71fb74c
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 652633b1fc27623edb2e87b5657322ba4826ec176e8eb5131a43249e08340f793ba426743be343bfe0fce02fc60b73cb790afef816a4153585d68bdfeaa9c55d
|
7
|
+
data.tar.gz: 946abb7526324cae5f1d6b62e25c8410129b07ca75c82a6b7285d6269ed1911e161a6030a394ef29ba238a80da6f96787dd94fdbd5c05bd2a643379252ce80d0
|
data/CHANGELOG.md
CHANGED
@@ -1,3 +1,30 @@
|
|
1
|
+
## 0.43.4 2020-07-09
|
2
|
+
|
3
|
+
* Reimplement Kernel#trap
|
4
|
+
* Dynamically allocate read buffer if length not given (#23)
|
5
|
+
* Prevent CPU saturation on infinite sleep (#24)
|
6
|
+
|
7
|
+
## 0.43.3 2020-07-08
|
8
|
+
|
9
|
+
* Fix behaviour after call to `Process.daemon` (#8)
|
10
|
+
* Replace core `Queue` class with `Polyphony::Queue` (#22)
|
11
|
+
* Make `ResourcePool` reentrant (#1)
|
12
|
+
* Accept `:with_exception` argument in `cancel_after` (#16)
|
13
|
+
|
14
|
+
## 0.43.2 2020-07-07
|
15
|
+
|
16
|
+
* Fix sending Redis commands with array arguments (#21)
|
17
|
+
|
18
|
+
## 0.43.1 2020-06
|
19
|
+
|
20
|
+
* Fix compiling C-extension on MacOS (#20)
|
21
|
+
|
22
|
+
## 0.43 2020-07-05
|
23
|
+
|
24
|
+
* Add IO#read_loop
|
25
|
+
* Fix OpenSSL extension
|
26
|
+
* More work on docs
|
27
|
+
|
1
28
|
## 0.42 2020-07-03
|
2
29
|
|
3
30
|
* Improve documentation
|
data/Gemfile.lock
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
PATH
|
2
2
|
remote: .
|
3
3
|
specs:
|
4
|
-
polyphony (0.
|
4
|
+
polyphony (0.43.4)
|
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,6 +1,5 @@
|
|
1
1
|
# Polyphony - Fine-Grained Concurrency for Ruby
|
2
2
|
|
3
|
-
|
4
3
|
[](http://rubygems.org/gems/polyphony)
|
5
4
|
[](https://github.com/digital-fabric/polyphony/actions?query=workflow%3ATests)
|
6
5
|
[](https://github.com/digital-fabric/polyphony/blob/master/LICENSE)
|
data/TODO.md
CHANGED
@@ -1,11 +1,10 @@
|
|
1
|
-
## 0.
|
1
|
+
## 0.43
|
2
2
|
|
3
3
|
- Reimplement ResourcePool, Channel, Mutex using LibevQueue
|
4
4
|
-- Add `Fiber#schedule_with_priority` method, aliased by `Fiber#wakeup`
|
5
5
|
- Implement agent interface is virtual function table
|
6
6
|
- Implement proxy agent for plugging in a user-provided agent class
|
7
7
|
|
8
|
-
## 0.43
|
9
8
|
- Debugging
|
10
9
|
- Eat your own dogfood: need a good tool to check what's going on when some
|
11
10
|
test fails
|
data/docs/_config.yml
CHANGED
@@ -46,13 +46,13 @@ heading_anchors: true
|
|
46
46
|
back_to_top: true
|
47
47
|
back_to_top_text: "Back to top"
|
48
48
|
|
49
|
-
footer_content: "Copyright ©
|
49
|
+
footer_content: "Copyright © 2020 Sharon Rosner. Distributed by an <a href=\"https://github.com/digital-fabric/polyphony/tree/master/LICENSE\">MIT license.</a>"
|
50
50
|
|
51
51
|
# Footer "Edit this page on GitHub" link text
|
52
52
|
gh_edit_link: true # show or hide edit this page link
|
53
53
|
gh_edit_link_text: "Edit this page on GitHub"
|
54
54
|
gh_edit_repository: "https://github.com/digital-fabric/polyphony" # the github URL for your repo
|
55
|
-
gh_edit_branch: "master" # the branch that your docs is served from
|
55
|
+
gh_edit_branch: "master/docs" # the branch that your docs is served from
|
56
56
|
gh_edit_view_mode: "tree" # "tree" or "edit" if you want the user to jump into the editor immediately
|
57
57
|
|
58
58
|
compress_html:
|
data/docs/favicon.ico
ADDED
Binary file
|
@@ -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]
|
@@ -416,28 +416,7 @@ request handler after all request headers have been received. This allows the
|
|
416
416
|
application to better deal with slow client attacks, big file uploads, and also
|
417
417
|
to minimize costly memory allocation and GC'ing.
|
418
418
|
|
419
|
-
|
420
|
-
Puma and 20X as fast as Unicorn. With SSL termination and using HTTP 2, the
|
421
|
-
Polyphony web server is about 2X as fast as Falcon.
|
422
|
-
|
423
|
-
### HTTP/1
|
424
|
-
|
425
|
-
|Server|requests/sec|average latency|max latency|
|
426
|
-
|------|-----------:|--------------:|----------:|
|
427
|
-
|Puma|
|
428
|
-
|Unicorn|
|
429
|
-
|Agoo|
|
430
|
-
|Polyphony|
|
431
|
-
|
432
|
-
### HTTP/2 with SSL Termination
|
433
|
-
|
434
|
-
|Server|requests/sec|average latency|max latency|
|
435
|
-
|------|-----------:|--------------:|----------:|
|
436
|
-
|Falcon|
|
437
|
-
|Polyphony|
|
438
|
-
|
439
|
-
(*Non-official benchmark with a basic "Hello world" Rack application. The usual
|
440
|
-
caveats regarding benchmarks should be applied here.*)
|
419
|
+
Benchmarks will be included here at a later time.
|
441
420
|
|
442
421
|
## Integrating Polyphony with other Gems
|
443
422
|
|
@@ -504,4 +483,4 @@ reach version 1.0. Here are some of the exciting directions we're working on.
|
|
504
483
|
- Support for more core and stdlib APIs
|
505
484
|
- More adapters for gems with C-extensions, such as `mysql`, `sqlite3` etc
|
506
485
|
- Use `io_uring` agent as alternative to the libev agent
|
507
|
-
- More concurrency constructs for building highly concurrent applications
|
486
|
+
- More concurrency constructs for building highly concurrent applications
|
data/docs/index.md
CHANGED
@@ -6,7 +6,11 @@ permalink: /
|
|
6
6
|
next_title: Installing Polyphony
|
7
7
|
---
|
8
8
|
|
9
|
-
# Polyphony
|
9
|
+
# Polyphony
|
10
|
+
{:.text-center .logo-title}
|
11
|
+
|
12
|
+
## Fine-grained concurrency for Ruby
|
13
|
+
{:.text-center .logo-title}
|
10
14
|
|
11
15
|
Polyphony is a library for building concurrent applications in Ruby. Polyphony
|
12
16
|
implements a comprehensive
|
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
|
@@ -4,47 +4,53 @@ def mem_usage
|
|
4
4
|
`ps -o rss #{$$}`.split.last.to_i
|
5
5
|
end
|
6
6
|
|
7
|
-
def
|
7
|
+
def calculate_memory_cost(name, count, &block)
|
8
|
+
GC.enable
|
9
|
+
ObjectSpace.garbage_collect
|
10
|
+
sleep 0.5
|
8
11
|
GC.disable
|
9
12
|
rss0 = mem_usage
|
10
|
-
|
13
|
+
count0 = ObjectSpace.count_objects[:TOTAL] - ObjectSpace.count_objects[:FREE]
|
14
|
+
a = []
|
15
|
+
count.times { a << block.call }
|
11
16
|
rss1 = mem_usage
|
12
|
-
|
17
|
+
count1 = ObjectSpace.count_objects[:TOTAL] - ObjectSpace.count_objects[:FREE]
|
18
|
+
p [count0, count1]
|
19
|
+
# sleep 0.5
|
13
20
|
cost = (rss1 - rss0).to_f / count
|
21
|
+
count_delta = (count1 - count0) / count
|
14
22
|
|
15
|
-
puts "
|
23
|
+
puts "#{name} rss cost: #{cost}KB object count: #{count_delta}"
|
16
24
|
end
|
17
25
|
|
18
|
-
|
19
|
-
|
20
|
-
def calculate_thread_memory_cost(count)
|
21
|
-
GC.disable
|
22
|
-
rss0 = mem_usage
|
23
|
-
count.times { Thread.new { sleep 1 } }
|
24
|
-
sleep 0.5
|
25
|
-
rss1 = mem_usage
|
26
|
-
sleep 0.5
|
27
|
-
GC.start
|
28
|
-
cost = (rss1 - rss0).to_f / count
|
26
|
+
f = Fiber.new { |f| f.transfer }
|
27
|
+
f.transfer Fiber.current
|
29
28
|
|
30
|
-
|
29
|
+
calculate_memory_cost('fiber', 10000) do
|
30
|
+
f = Fiber.new { |f| f.transfer :foo }
|
31
|
+
f.transfer Fiber.current
|
32
|
+
f
|
31
33
|
end
|
32
34
|
|
33
|
-
|
35
|
+
t = Thread.new { sleep 1}
|
36
|
+
t.kill
|
37
|
+
t.join
|
38
|
+
|
39
|
+
calculate_memory_cost('thread', 500) do
|
40
|
+
t = Thread.new { sleep 1 }
|
41
|
+
sleep 0.001
|
42
|
+
t
|
43
|
+
end
|
44
|
+
(Thread.list - [Thread.current]).each(&:kill).each(&:join)
|
34
45
|
|
35
46
|
require 'bundler/setup'
|
36
47
|
require 'polyphony'
|
37
48
|
|
38
|
-
|
39
|
-
|
40
|
-
rss0 = mem_usage
|
41
|
-
count.times { spin { :foo } }
|
42
|
-
snooze
|
43
|
-
rss1 = mem_usage
|
44
|
-
GC.start
|
45
|
-
cost = (rss1 - rss0).to_f / count
|
49
|
+
f = spin { sleep 0.1 }
|
50
|
+
f.await
|
46
51
|
|
47
|
-
|
52
|
+
calculate_memory_cost('polyphony fiber', 10000) do
|
53
|
+
f = spin { :foo }
|
54
|
+
f.await
|
55
|
+
f
|
48
56
|
end
|
49
|
-
|
50
|
-
calculate_extended_fiber_memory_cost(10000)
|
@@ -13,7 +13,7 @@ def handle_client(socket)
|
|
13
13
|
parser.on_message_complete = proc do |env|
|
14
14
|
reqs << Object.new # parser
|
15
15
|
end
|
16
|
-
|
16
|
+
socket.read_loop do |data|
|
17
17
|
parser << data
|
18
18
|
while (req = reqs.shift)
|
19
19
|
handle_request(socket, req)
|
data/ext/polyphony/libev_agent.c
CHANGED
@@ -1,6 +1,8 @@
|
|
1
|
+
#include <netdb.h>
|
2
|
+
#include <sys/socket.h>
|
3
|
+
|
1
4
|
#include "polyphony.h"
|
2
5
|
#include "../libev/ev.h"
|
3
|
-
#include <sys/socket.h>
|
4
6
|
|
5
7
|
VALUE cLibevAgent = Qnil;
|
6
8
|
VALUE cTCPSocket;
|
@@ -128,7 +130,7 @@ VALUE LibevAgent_poll(VALUE self, VALUE nowait, VALUE current_fiber, VALUE queue
|
|
128
130
|
GetLibevAgent(self, agent);
|
129
131
|
|
130
132
|
if (is_nowait) {
|
131
|
-
|
133
|
+
long runnable_count = RARRAY_LEN(queue);
|
132
134
|
agent->run_no_wait_count++;
|
133
135
|
if (agent->run_no_wait_count < runnable_count || agent->run_no_wait_count < 10)
|
134
136
|
return self;
|
@@ -234,7 +236,7 @@ struct libev_io {
|
|
234
236
|
VALUE fiber;
|
235
237
|
};
|
236
238
|
|
237
|
-
|
239
|
+
void LibevAgent_io_callback(EV_P_ ev_io *w, int revents)
|
238
240
|
{
|
239
241
|
struct libev_io *watcher = (struct libev_io *)w;
|
240
242
|
Fiber_make_runnable(watcher->fiber, Qnil);
|
@@ -255,11 +257,32 @@ VALUE libev_agent_await(VALUE self) {
|
|
255
257
|
return libev_await(agent);
|
256
258
|
}
|
257
259
|
|
260
|
+
VALUE libev_io_wait(struct LibevAgent_t *agent, struct libev_io *watcher, rb_io_t *fptr, int flags) {
|
261
|
+
VALUE switchpoint_result;
|
262
|
+
|
263
|
+
if (watcher->fiber == Qnil) {
|
264
|
+
watcher->fiber = rb_fiber_current();
|
265
|
+
ev_io_init(&watcher->io, LibevAgent_io_callback, fptr->fd, flags);
|
266
|
+
}
|
267
|
+
ev_io_start(agent->ev_loop, &watcher->io);
|
268
|
+
switchpoint_result = libev_await(agent);
|
269
|
+
ev_io_stop(agent->ev_loop, &watcher->io);
|
270
|
+
|
271
|
+
RB_GC_GUARD(switchpoint_result);
|
272
|
+
return switchpoint_result;
|
273
|
+
}
|
274
|
+
|
275
|
+
VALUE libev_snooze() {
|
276
|
+
Fiber_make_runnable(rb_fiber_current(), Qnil);
|
277
|
+
return Thread_switch_fiber(rb_thread_current());
|
278
|
+
}
|
279
|
+
|
258
280
|
VALUE LibevAgent_read(VALUE self, VALUE io, VALUE str, VALUE length, VALUE to_eof) {
|
259
281
|
struct LibevAgent_t *agent;
|
260
282
|
struct libev_io watcher;
|
261
283
|
rb_io_t *fptr;
|
262
|
-
|
284
|
+
long dynamic_len = length == Qnil;
|
285
|
+
long len = dynamic_len ? 4096 : NUM2INT(length);
|
263
286
|
int shrinkable = io_setstrbuf(&str, len);
|
264
287
|
char *buf = RSTRING_PTR(str);
|
265
288
|
long total = 0;
|
@@ -276,44 +299,35 @@ VALUE LibevAgent_read(VALUE self, VALUE io, VALUE str, VALUE length, VALUE to_eo
|
|
276
299
|
|
277
300
|
OBJ_TAINT(str);
|
278
301
|
|
279
|
-
while (
|
280
|
-
|
281
|
-
if (n
|
282
|
-
break;
|
283
|
-
if (n > 0) {
|
284
|
-
total = total + n;
|
285
|
-
buf += n;
|
286
|
-
len -= n;
|
287
|
-
if (!read_to_eof || (len == 0)) break;
|
288
|
-
}
|
289
|
-
else {
|
302
|
+
while (1) {
|
303
|
+
ssize_t n = read(fptr->fd, buf, len - total);
|
304
|
+
if (n < 0) {
|
290
305
|
int e = errno;
|
291
|
-
if (
|
292
|
-
|
293
|
-
|
294
|
-
|
295
|
-
}
|
296
|
-
ev_io_start(agent->ev_loop, &watcher.io);
|
297
|
-
switchpoint_result = libev_await(agent);
|
298
|
-
ev_io_stop(agent->ev_loop, &watcher.io);
|
299
|
-
if (TEST_EXCEPTION(switchpoint_result)) {
|
300
|
-
goto error;
|
301
|
-
}
|
302
|
-
}
|
303
|
-
else
|
304
|
-
rb_syserr_fail(e, strerror(e));
|
305
|
-
// rb_syserr_fail_path(e, fptr->pathv);
|
306
|
+
if (e != EWOULDBLOCK && e != EAGAIN) rb_syserr_fail(e, strerror(e));
|
307
|
+
|
308
|
+
switchpoint_result = libev_io_wait(agent, &watcher, fptr, EV_READ);
|
309
|
+
if (TEST_EXCEPTION(switchpoint_result)) goto error;
|
306
310
|
}
|
307
|
-
|
311
|
+
else {
|
312
|
+
switchpoint_result = libev_snooze();
|
313
|
+
if (TEST_EXCEPTION(switchpoint_result)) goto error;
|
308
314
|
|
309
|
-
|
310
|
-
|
311
|
-
|
312
|
-
|
313
|
-
|
315
|
+
if (n == 0) break; // EOF
|
316
|
+
total = total + n;
|
317
|
+
if (!read_to_eof) break;
|
318
|
+
|
319
|
+
if (total == len) {
|
320
|
+
if (!dynamic_len) break;
|
321
|
+
|
322
|
+
rb_str_resize(str, total);
|
323
|
+
rb_str_modify_expand(str, len);
|
324
|
+
buf = RSTRING_PTR(str) + total;
|
325
|
+
shrinkable = 0;
|
326
|
+
len += len;
|
327
|
+
}
|
328
|
+
else buf += n;
|
314
329
|
}
|
315
330
|
}
|
316
|
-
|
317
331
|
if (total == 0) return Qnil;
|
318
332
|
|
319
333
|
io_set_read_length(str, total, shrinkable);
|
@@ -347,8 +361,8 @@ VALUE LibevAgent_read_loop(VALUE self, VALUE io) {
|
|
347
361
|
struct libev_io watcher;
|
348
362
|
rb_io_t *fptr;
|
349
363
|
VALUE str;
|
350
|
-
|
351
|
-
|
364
|
+
long total;
|
365
|
+
long len = 8192;
|
352
366
|
int shrinkable;
|
353
367
|
char *buf;
|
354
368
|
VALUE switchpoint_result = Qnil;
|
@@ -366,10 +380,20 @@ VALUE LibevAgent_read_loop(VALUE self, VALUE io) {
|
|
366
380
|
OBJ_TAINT(str);
|
367
381
|
|
368
382
|
while (1) {
|
369
|
-
|
370
|
-
if (n
|
371
|
-
|
372
|
-
|
383
|
+
ssize_t n = read(fptr->fd, buf, len);
|
384
|
+
if (n < 0) {
|
385
|
+
int e = errno;
|
386
|
+
if ((e != EWOULDBLOCK && e != EAGAIN)) rb_syserr_fail(e, strerror(e));
|
387
|
+
|
388
|
+
switchpoint_result = libev_io_wait(agent, &watcher, fptr, EV_READ);
|
389
|
+
if (TEST_EXCEPTION(switchpoint_result)) goto error;
|
390
|
+
}
|
391
|
+
else {
|
392
|
+
switchpoint_result = libev_snooze();
|
393
|
+
if (TEST_EXCEPTION(switchpoint_result)) goto error;
|
394
|
+
|
395
|
+
if (n == 0) break; // EOF
|
396
|
+
|
373
397
|
total = n;
|
374
398
|
YIELD_STR();
|
375
399
|
Fiber_make_runnable(rb_fiber_current(), Qnil);
|
@@ -378,24 +402,6 @@ VALUE LibevAgent_read_loop(VALUE self, VALUE io) {
|
|
378
402
|
goto error;
|
379
403
|
}
|
380
404
|
}
|
381
|
-
else {
|
382
|
-
int e = errno;
|
383
|
-
if ((e == EWOULDBLOCK || e == EAGAIN)) {
|
384
|
-
if (watcher.fiber == Qnil) {
|
385
|
-
watcher.fiber = rb_fiber_current();
|
386
|
-
ev_io_init(&watcher.io, LibevAgent_io_callback, fptr->fd, EV_READ);
|
387
|
-
}
|
388
|
-
ev_io_start(agent->ev_loop, &watcher.io);
|
389
|
-
switchpoint_result = libev_await(agent);
|
390
|
-
ev_io_stop(agent->ev_loop, &watcher.io);
|
391
|
-
if (TEST_EXCEPTION(switchpoint_result)) {
|
392
|
-
goto error;
|
393
|
-
}
|
394
|
-
}
|
395
|
-
else
|
396
|
-
rb_syserr_fail(e, strerror(e));
|
397
|
-
// rb_syserr_fail_path(e, fptr->pathv);
|
398
|
-
}
|
399
405
|
}
|
400
406
|
|
401
407
|
RB_GC_GUARD(str);
|
@@ -414,8 +420,8 @@ VALUE LibevAgent_write(VALUE self, VALUE io, VALUE str) {
|
|
414
420
|
VALUE switchpoint_result = Qnil;
|
415
421
|
|
416
422
|
char *buf = StringValuePtr(str);
|
417
|
-
|
418
|
-
|
423
|
+
long len = RSTRING_LEN(str);
|
424
|
+
long left = len;
|
419
425
|
|
420
426
|
VALUE underlying_io = rb_iv_get(io, "@io");
|
421
427
|
if (underlying_io != Qnil) io = underlying_io;
|
@@ -425,37 +431,20 @@ VALUE LibevAgent_write(VALUE self, VALUE io, VALUE str) {
|
|
425
431
|
watcher.fiber = Qnil;
|
426
432
|
|
427
433
|
while (left > 0) {
|
428
|
-
|
429
|
-
if (
|
434
|
+
ssize_t n = write(fptr->fd, buf, left);
|
435
|
+
if (n < 0) {
|
430
436
|
int e = errno;
|
431
|
-
if (e
|
432
|
-
|
433
|
-
|
434
|
-
|
435
|
-
}
|
436
|
-
ev_io_start(agent->ev_loop, &watcher.io);
|
437
|
-
switchpoint_result = libev_await(agent);
|
438
|
-
ev_io_stop(agent->ev_loop, &watcher.io);
|
439
|
-
if (TEST_EXCEPTION(switchpoint_result))
|
440
|
-
goto error;
|
441
|
-
}
|
442
|
-
else {
|
443
|
-
rb_syserr_fail(e, strerror(e));
|
444
|
-
// rb_syserr_fail_path(e, fptr->pathv);
|
445
|
-
|
446
|
-
}
|
437
|
+
if ((e != EWOULDBLOCK && e != EAGAIN)) rb_syserr_fail(e, strerror(e));
|
438
|
+
|
439
|
+
switchpoint_result = libev_io_wait(agent, &watcher, fptr, EV_WRITE);
|
440
|
+
if (TEST_EXCEPTION(switchpoint_result)) goto error;
|
447
441
|
}
|
448
442
|
else {
|
449
|
-
|
450
|
-
|
451
|
-
}
|
452
|
-
}
|
443
|
+
switchpoint_result = libev_snooze();
|
444
|
+
if (TEST_EXCEPTION(switchpoint_result)) goto error;
|
453
445
|
|
454
|
-
|
455
|
-
|
456
|
-
switchpoint_result = Thread_switch_fiber(rb_thread_current());
|
457
|
-
if (TEST_EXCEPTION(switchpoint_result)) {
|
458
|
-
goto error;
|
446
|
+
buf += n;
|
447
|
+
left -= n;
|
459
448
|
}
|
460
449
|
}
|
461
450
|
|
@@ -488,50 +477,39 @@ VALUE LibevAgent_accept(VALUE self, VALUE sock) {
|
|
488
477
|
fd = accept(fptr->fd, &addr, &len);
|
489
478
|
if (fd < 0) {
|
490
479
|
int e = errno;
|
491
|
-
if (e
|
492
|
-
|
493
|
-
|
494
|
-
|
495
|
-
}
|
496
|
-
ev_io_start(agent->ev_loop, &watcher.io);
|
497
|
-
switchpoint_result = libev_await(agent);
|
498
|
-
ev_io_stop(agent->ev_loop, &watcher.io);
|
499
|
-
|
500
|
-
TEST_RESUME_EXCEPTION(switchpoint_result);
|
501
|
-
RB_GC_GUARD(watcher.fiber);
|
502
|
-
RB_GC_GUARD(switchpoint_result);
|
503
|
-
}
|
504
|
-
else
|
505
|
-
rb_syserr_fail(e, strerror(e));
|
506
|
-
// rb_syserr_fail_path(e, fptr->pathv);
|
480
|
+
if ((e != EWOULDBLOCK && e != EAGAIN)) rb_syserr_fail(e, strerror(e));
|
481
|
+
|
482
|
+
switchpoint_result = libev_io_wait(agent, &watcher, fptr, EV_READ);
|
483
|
+
if (TEST_EXCEPTION(switchpoint_result)) goto error;
|
507
484
|
}
|
508
485
|
else {
|
509
|
-
VALUE
|
486
|
+
VALUE socket;
|
510
487
|
rb_io_t *fp;
|
511
|
-
|
488
|
+
switchpoint_result = libev_snooze();
|
489
|
+
if (TEST_EXCEPTION(switchpoint_result)) {
|
490
|
+
close(fd); // close fd since we're raising an exception
|
491
|
+
goto error;
|
492
|
+
}
|
493
|
+
|
494
|
+
socket = rb_obj_alloc(cTCPSocket);
|
495
|
+
MakeOpenFile(socket, fp);
|
512
496
|
rb_update_max_fd(fd);
|
513
497
|
fp->fd = fd;
|
514
498
|
fp->mode = FMODE_READWRITE | FMODE_DUPLEX;
|
515
|
-
rb_io_ascii8bit_binmode(
|
499
|
+
rb_io_ascii8bit_binmode(socket);
|
516
500
|
rb_io_set_nonblock(fp);
|
517
501
|
rb_io_synchronized(fp);
|
518
502
|
|
519
503
|
// if (rsock_do_not_reverse_lookup) {
|
520
504
|
// fp->mode |= FMODE_NOREVLOOKUP;
|
521
505
|
// }
|
522
|
-
|
523
|
-
if (watcher.fiber == Qnil) {
|
524
|
-
Fiber_make_runnable(rb_fiber_current(), Qnil);
|
525
|
-
switchpoint_result = Thread_switch_fiber(rb_thread_current());
|
526
|
-
if (TEST_EXCEPTION(switchpoint_result)) {
|
527
|
-
return rb_funcall(rb_mKernel, ID_raise, 1, switchpoint_result);
|
528
|
-
}
|
529
|
-
}
|
530
|
-
|
531
|
-
return connection;
|
506
|
+
return socket;
|
532
507
|
}
|
533
508
|
}
|
509
|
+
RB_GC_GUARD(switchpoint_result);
|
534
510
|
return Qnil;
|
511
|
+
error:
|
512
|
+
return rb_funcall(rb_mKernel, ID_raise, 1, switchpoint_result);
|
535
513
|
}
|
536
514
|
|
537
515
|
VALUE LibevAgent_accept_loop(VALUE self, VALUE sock) {
|
@@ -542,7 +520,7 @@ VALUE LibevAgent_accept_loop(VALUE self, VALUE sock) {
|
|
542
520
|
struct sockaddr addr;
|
543
521
|
socklen_t len = (socklen_t)sizeof addr;
|
544
522
|
VALUE switchpoint_result = Qnil;
|
545
|
-
VALUE
|
523
|
+
VALUE socket = Qnil;
|
546
524
|
VALUE underlying_sock = rb_iv_get(sock, "@io");
|
547
525
|
if (underlying_sock != Qnil) sock = underlying_sock;
|
548
526
|
|
@@ -555,46 +533,82 @@ VALUE LibevAgent_accept_loop(VALUE self, VALUE sock) {
|
|
555
533
|
fd = accept(fptr->fd, &addr, &len);
|
556
534
|
if (fd < 0) {
|
557
535
|
int e = errno;
|
558
|
-
if (e
|
559
|
-
|
560
|
-
|
561
|
-
|
562
|
-
}
|
563
|
-
ev_io_start(agent->ev_loop, &watcher.io);
|
564
|
-
switchpoint_result = libev_await(agent);
|
565
|
-
ev_io_stop(agent->ev_loop, &watcher.io);
|
566
|
-
|
567
|
-
TEST_RESUME_EXCEPTION(switchpoint_result);
|
568
|
-
}
|
569
|
-
else
|
570
|
-
rb_syserr_fail(e, strerror(e));
|
571
|
-
// rb_syserr_fail_path(e, fptr->pathv);
|
536
|
+
if ((e != EWOULDBLOCK && e != EAGAIN)) rb_syserr_fail(e, strerror(e));
|
537
|
+
|
538
|
+
switchpoint_result = libev_io_wait(agent, &watcher, fptr, EV_READ);
|
539
|
+
if (TEST_EXCEPTION(switchpoint_result)) goto error;
|
572
540
|
}
|
573
541
|
else {
|
574
542
|
rb_io_t *fp;
|
575
|
-
|
576
|
-
|
543
|
+
switchpoint_result = libev_snooze();
|
544
|
+
if (TEST_EXCEPTION(switchpoint_result)) {
|
545
|
+
close(fd); // close fd since we're raising an exception
|
546
|
+
goto error;
|
547
|
+
}
|
548
|
+
|
549
|
+
socket = rb_obj_alloc(cTCPSocket);
|
550
|
+
MakeOpenFile(socket, fp);
|
577
551
|
rb_update_max_fd(fd);
|
578
552
|
fp->fd = fd;
|
579
553
|
fp->mode = FMODE_READWRITE | FMODE_DUPLEX;
|
580
|
-
rb_io_ascii8bit_binmode(
|
554
|
+
rb_io_ascii8bit_binmode(socket);
|
581
555
|
rb_io_set_nonblock(fp);
|
582
556
|
rb_io_synchronized(fp);
|
583
557
|
|
584
|
-
rb_yield(
|
585
|
-
|
586
|
-
|
587
|
-
Fiber_make_runnable(rb_fiber_current(), Qnil);
|
588
|
-
switchpoint_result = Thread_switch_fiber(rb_thread_current());
|
589
|
-
TEST_RESUME_EXCEPTION(switchpoint_result);
|
558
|
+
rb_yield(socket);
|
559
|
+
socket = Qnil;
|
590
560
|
}
|
591
561
|
}
|
592
562
|
|
593
|
-
RB_GC_GUARD(
|
563
|
+
RB_GC_GUARD(socket);
|
594
564
|
RB_GC_GUARD(watcher.fiber);
|
595
565
|
RB_GC_GUARD(switchpoint_result);
|
566
|
+
return Qnil;
|
567
|
+
error:
|
568
|
+
return rb_funcall(rb_mKernel, ID_raise, 1, switchpoint_result);
|
596
569
|
}
|
597
570
|
|
571
|
+
// VALUE LibevAgent_connect(VALUE self, VALUE sock, VALUE host, VALUE port) {
|
572
|
+
// struct LibevAgent_t *agent;
|
573
|
+
// struct libev_io watcher;
|
574
|
+
// rb_io_t *fptr;
|
575
|
+
// struct sockaddr_in addr;
|
576
|
+
// char *host_buf = StringValueCStr(host);
|
577
|
+
// VALUE switchpoint_result = Qnil;
|
578
|
+
// VALUE underlying_sock = rb_iv_get(sock, "@io");
|
579
|
+
// if (underlying_sock != Qnil) sock = underlying_sock;
|
580
|
+
|
581
|
+
// GetLibevAgent(self, agent);
|
582
|
+
// GetOpenFile(sock, fptr);
|
583
|
+
// rb_io_set_nonblock(fptr);
|
584
|
+
// watcher.fiber = Qnil;
|
585
|
+
|
586
|
+
// addr.sin_family = AF_INET;
|
587
|
+
// addr.sin_addr.s_addr = inet_addr(host_buf);
|
588
|
+
// addr.sin_port = htons(NUM2INT(port));
|
589
|
+
|
590
|
+
// while (1) {
|
591
|
+
// int result = connect(fptr->fd, &addr, sizeof(addr));
|
592
|
+
// if (result < 0) {
|
593
|
+
// int e = errno;
|
594
|
+
// if ((e != EWOULDBLOCK && e != EAGAIN)) rb_syserr_fail(e, strerror(e));
|
595
|
+
|
596
|
+
// switchpoint_result = libev_io_wait(agent, &watcher, fptr, EV_WRITE);
|
597
|
+
// if (TEST_EXCEPTION(switchpoint_result)) goto error;
|
598
|
+
// }
|
599
|
+
// else {
|
600
|
+
// switchpoint_result = libev_snooze();
|
601
|
+
// if (TEST_EXCEPTION(switchpoint_result)) goto error;
|
602
|
+
|
603
|
+
// return sock;
|
604
|
+
// }
|
605
|
+
// }
|
606
|
+
// RB_GC_GUARD(switchpoint_result);
|
607
|
+
// return Qnil;
|
608
|
+
// error:
|
609
|
+
// return rb_funcall(rb_mKernel, ID_raise, 1, switchpoint_result);
|
610
|
+
// }
|
611
|
+
|
598
612
|
VALUE LibevAgent_wait_io(VALUE self, VALUE io, VALUE write) {
|
599
613
|
struct LibevAgent_t *agent;
|
600
614
|
struct libev_io watcher;
|
@@ -624,7 +638,7 @@ struct libev_timer {
|
|
624
638
|
VALUE fiber;
|
625
639
|
};
|
626
640
|
|
627
|
-
|
641
|
+
void LibevAgent_timer_callback(EV_P_ ev_timer *w, int revents)
|
628
642
|
{
|
629
643
|
struct libev_timer *watcher = (struct libev_timer *)w;
|
630
644
|
Fiber_make_runnable(watcher->fiber, Qnil);
|
@@ -654,7 +668,7 @@ struct libev_child {
|
|
654
668
|
VALUE fiber;
|
655
669
|
};
|
656
670
|
|
657
|
-
|
671
|
+
void LibevAgent_child_callback(EV_P_ ev_child *w, int revents)
|
658
672
|
{
|
659
673
|
struct libev_child *watcher = (struct libev_child *)w;
|
660
674
|
int exit_status = w->rstatus >> 8; // weird, why should we do this?
|
@@ -712,6 +726,7 @@ void Init_LibevAgent() {
|
|
712
726
|
rb_define_method(cLibevAgent, "write", LibevAgent_write, 2);
|
713
727
|
rb_define_method(cLibevAgent, "accept", LibevAgent_accept, 1);
|
714
728
|
rb_define_method(cLibevAgent, "accept_loop", LibevAgent_accept_loop, 1);
|
729
|
+
// rb_define_method(cLibevAgent, "connect", LibevAgent_accept, 3);
|
715
730
|
rb_define_method(cLibevAgent, "wait_io", LibevAgent_wait_io, 2);
|
716
731
|
rb_define_method(cLibevAgent, "sleep", LibevAgent_sleep, 1);
|
717
732
|
rb_define_method(cLibevAgent, "waitpid", LibevAgent_waitpid, 1);
|