eventmachine 1.0.3-x86-mingw32 → 1.2.0.dev.2-x86-mingw32
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 +7 -0
- data/CHANGELOG.md +84 -1
- data/README.md +6 -7
- data/ext/binder.cpp +10 -10
- data/ext/binder.h +5 -5
- data/ext/cmain.cpp +173 -61
- data/ext/ed.cpp +262 -127
- data/ext/ed.h +50 -30
- data/ext/em.cpp +491 -445
- data/ext/em.h +101 -36
- data/ext/eventmachine.h +67 -51
- data/ext/extconf.rb +124 -31
- data/ext/fastfilereader/extconf.rb +9 -2
- data/ext/fastfilereader/mapper.cpp +3 -1
- data/ext/fastfilereader/rubymain.cpp +7 -7
- data/ext/kb.cpp +1 -1
- data/ext/pipe.cpp +11 -4
- data/ext/project.h +26 -6
- data/ext/rubymain.cpp +408 -201
- data/ext/ssl.cpp +167 -20
- data/ext/ssl.h +11 -2
- data/java/src/com/rubyeventmachine/EmReactor.java +16 -0
- data/java/src/com/rubyeventmachine/EventableChannel.java +2 -0
- data/java/src/com/rubyeventmachine/EventableDatagramChannel.java +6 -0
- data/java/src/com/rubyeventmachine/EventableSocketChannel.java +55 -10
- data/lib/1.9/fastfilereaderext.so +0 -0
- data/lib/1.9/rubyeventmachine.so +0 -0
- data/lib/2.0/fastfilereaderext.so +0 -0
- data/lib/2.0/rubyeventmachine.so +0 -0
- data/lib/2.1/fastfilereaderext.so +0 -0
- data/lib/2.1/rubyeventmachine.so +0 -0
- data/lib/2.2/fastfilereaderext.so +0 -0
- data/lib/2.2/rubyeventmachine.so +0 -0
- data/lib/2.3/fastfilereaderext.so +0 -0
- data/lib/2.3/rubyeventmachine.so +0 -0
- data/lib/em/buftok.rb +34 -85
- data/lib/em/channel.rb +5 -0
- data/lib/em/completion.rb +2 -2
- data/lib/em/connection.rb +62 -4
- data/lib/em/iterator.rb +30 -48
- data/lib/em/pool.rb +1 -1
- data/lib/em/protocols/httpclient.rb +31 -11
- data/lib/em/protocols/line_and_text.rb +4 -4
- data/lib/em/protocols/linetext2.rb +44 -39
- data/lib/em/protocols/smtpclient.rb +60 -31
- data/lib/em/protocols/smtpserver.rb +32 -9
- data/lib/em/pure_ruby.rb +8 -3
- data/lib/em/queue.rb +16 -7
- data/lib/em/resolver.rb +64 -24
- data/lib/em/threaded_resource.rb +2 -2
- data/lib/em/tick_loop.rb +19 -19
- data/lib/em/version.rb +1 -1
- data/lib/eventmachine.rb +96 -49
- data/lib/jeventmachine.rb +17 -0
- data/rakelib/package.rake +31 -4
- data/tests/dhparam.pem +13 -0
- data/tests/em_test_helper.rb +87 -0
- data/tests/test_attach.rb +25 -0
- data/tests/test_basic.rb +27 -38
- data/tests/test_channel.rb +14 -1
- data/tests/test_completion.rb +1 -0
- data/tests/test_connection_count.rb +22 -1
- data/tests/test_connection_write.rb +35 -0
- data/tests/test_defer.rb +17 -0
- data/tests/test_epoll.rb +26 -14
- data/tests/test_file_watch.rb +1 -0
- data/tests/test_fork.rb +75 -0
- data/tests/test_httpclient.rb +43 -0
- data/tests/test_idle_connection.rb +6 -4
- data/tests/test_ipv4.rb +125 -0
- data/tests/test_ipv6.rb +131 -0
- data/tests/test_iterator.rb +115 -0
- data/tests/test_kb.rb +19 -25
- data/tests/test_ltp2.rb +20 -0
- data/tests/test_many_fds.rb +22 -0
- data/tests/test_pause.rb +29 -0
- data/tests/test_pool.rb +2 -0
- data/tests/test_process_watch.rb +2 -0
- data/tests/test_processes.rb +7 -7
- data/tests/test_queue.rb +14 -0
- data/tests/test_resolver.rb +56 -7
- data/tests/test_set_sock_opt.rb +2 -0
- data/tests/test_smtpclient.rb +20 -0
- data/tests/test_ssl_args.rb +2 -2
- data/tests/test_ssl_dhparam.rb +83 -0
- data/tests/test_ssl_ecdh_curve.rb +79 -0
- data/tests/test_ssl_extensions.rb +49 -0
- data/tests/test_ssl_methods.rb +22 -5
- data/tests/test_ssl_protocols.rb +246 -0
- data/tests/test_ssl_verify.rb +103 -59
- data/tests/test_system.rb +4 -0
- data/tests/test_threaded_resource.rb +8 -0
- data/tests/test_unbind_reason.rb +5 -1
- metadata +173 -107
- data/.gitignore +0 -21
- data/.travis.yml +0 -12
- data/.yardopts +0 -7
- data/Gemfile +0 -2
- data/Rakefile +0 -20
- data/eventmachine.gemspec +0 -36
- data/rakelib/cpp.rake_example +0 -77
@@ -32,20 +32,20 @@ module EventMachine
|
|
32
32
|
# for a version which is optimized for correctness with regard to binary text blocks
|
33
33
|
# that can switch back to line mode.
|
34
34
|
class LineAndTextProtocol < Connection
|
35
|
-
MaxLineLength = 16*1024
|
36
35
|
MaxBinaryLength = 32*1024*1024
|
37
36
|
|
38
37
|
def initialize *args
|
39
38
|
super
|
40
39
|
lbp_init_line_state
|
41
40
|
end
|
41
|
+
|
42
42
|
def receive_data data
|
43
43
|
if @lbp_mode == :lines
|
44
44
|
begin
|
45
|
-
@lpb_buffer.extract(data).each do |line|
|
45
|
+
@lpb_buffer.extract(data).each do |line|
|
46
46
|
receive_line(line.chomp) if respond_to?(:receive_line)
|
47
47
|
end
|
48
|
-
rescue
|
48
|
+
rescue
|
49
49
|
receive_error('overlength line') if respond_to?(:receive_error)
|
50
50
|
close_connection
|
51
51
|
return
|
@@ -116,7 +116,7 @@ module EventMachine
|
|
116
116
|
#--
|
117
117
|
# For internal use, establish protocol baseline for handling lines.
|
118
118
|
def lbp_init_line_state
|
119
|
-
@lpb_buffer = BufferedTokenizer.new("\n"
|
119
|
+
@lpb_buffer = BufferedTokenizer.new("\n")
|
120
120
|
@lbp_mode = :lines
|
121
121
|
end
|
122
122
|
private :lbp_init_line_state
|
@@ -37,11 +37,10 @@ module EventMachine
|
|
37
37
|
# When we get around to that, call #receive_error if the user defined it, otherwise
|
38
38
|
# throw exceptions.
|
39
39
|
|
40
|
-
MaxLineLength = 16*1024
|
41
40
|
MaxBinaryLength = 32*1024*1024
|
42
41
|
|
43
42
|
#--
|
44
|
-
# Will
|
43
|
+
# Will loop internally until there's no data left to read.
|
45
44
|
# That way the user-defined handlers we call can modify the
|
46
45
|
# handling characteristics on a per-token basis.
|
47
46
|
#
|
@@ -53,45 +52,51 @@ module EventMachine
|
|
53
52
|
@lt2_delimiter ||= "\n"
|
54
53
|
@lt2_linebuffer ||= []
|
55
54
|
|
56
|
-
|
57
|
-
|
58
|
-
|
59
|
-
|
60
|
-
@
|
61
|
-
|
62
|
-
ln.
|
55
|
+
remaining_data = data
|
56
|
+
|
57
|
+
while remaining_data.length > 0
|
58
|
+
if @lt2_mode == :lines
|
59
|
+
if ix = remaining_data.index( @lt2_delimiter )
|
60
|
+
@lt2_linebuffer << remaining_data[0...ix]
|
61
|
+
ln = @lt2_linebuffer.join
|
62
|
+
@lt2_linebuffer.clear
|
63
|
+
if @lt2_delimiter == "\n"
|
64
|
+
ln.chomp!
|
65
|
+
end
|
66
|
+
receive_line ln
|
67
|
+
remaining_data = remaining_data[(ix+@lt2_delimiter.length)..-1]
|
68
|
+
else
|
69
|
+
@lt2_linebuffer << remaining_data
|
70
|
+
remaining_data = ""
|
63
71
|
end
|
64
|
-
|
65
|
-
|
66
|
-
|
67
|
-
|
68
|
-
|
69
|
-
|
70
|
-
|
71
|
-
|
72
|
-
|
73
|
-
|
74
|
-
|
75
|
-
|
76
|
-
|
77
|
-
|
78
|
-
|
79
|
-
|
80
|
-
|
81
|
-
|
82
|
-
|
83
|
-
|
84
|
-
|
85
|
-
|
86
|
-
|
87
|
-
|
88
|
-
|
89
|
-
|
72
|
+
elsif @lt2_mode == :text
|
73
|
+
if @lt2_textsize
|
74
|
+
needed = @lt2_textsize - @lt2_textpos
|
75
|
+
will_take = if remaining_data.length > needed
|
76
|
+
needed
|
77
|
+
else
|
78
|
+
remaining_data.length
|
79
|
+
end
|
80
|
+
|
81
|
+
@lt2_textbuffer << remaining_data[0...will_take]
|
82
|
+
tail = remaining_data[will_take..-1]
|
83
|
+
|
84
|
+
@lt2_textpos += will_take
|
85
|
+
if @lt2_textpos >= @lt2_textsize
|
86
|
+
# Reset line mode (the default behavior) BEFORE calling the
|
87
|
+
# receive_binary_data. This makes it possible for user code
|
88
|
+
# to call set_text_mode, enabling chains of text blocks
|
89
|
+
# (which can possibly be of different sizes).
|
90
|
+
set_line_mode
|
91
|
+
receive_binary_data @lt2_textbuffer.join
|
92
|
+
receive_end_of_binary_data
|
93
|
+
end
|
94
|
+
|
95
|
+
remaining_data = tail
|
96
|
+
else
|
97
|
+
receive_binary_data remaining_data
|
98
|
+
remaining_data = ""
|
90
99
|
end
|
91
|
-
|
92
|
-
receive_data tail
|
93
|
-
else
|
94
|
-
receive_binary_data data
|
95
100
|
end
|
96
101
|
end
|
97
102
|
end
|
@@ -48,20 +48,20 @@ module EventMachine
|
|
48
48
|
# puts 'Email failed!'
|
49
49
|
# }
|
50
50
|
#
|
51
|
-
# Sending generated emails (using
|
51
|
+
# Sending generated emails (using Mail)
|
52
52
|
#
|
53
|
-
# mail =
|
54
|
-
#
|
55
|
-
#
|
56
|
-
#
|
57
|
-
#
|
58
|
-
#
|
53
|
+
# mail = Mail.new do
|
54
|
+
# from 'alice@example.com'
|
55
|
+
# to 'bob@example.com'
|
56
|
+
# subject 'This is a test email'
|
57
|
+
# body 'Hello, world!'
|
58
|
+
# end
|
59
59
|
#
|
60
60
|
# email = EM::P::SmtpClient.send(
|
61
|
-
# :domain=>'
|
62
|
-
# :from=>mail.from,
|
61
|
+
# :domain=>'example.com',
|
62
|
+
# :from=>mail.from.first,
|
63
63
|
# :to=>mail.to,
|
64
|
-
# :
|
64
|
+
# :message=>mail.to_s
|
65
65
|
# )
|
66
66
|
#
|
67
67
|
class SmtpClient < Connection
|
@@ -108,25 +108,29 @@ module EventMachine
|
|
108
108
|
# of each requested recipient is available after the call completes. TODO, we should define
|
109
109
|
# an overridable stub that will be called on rejection of a recipient or a sender, giving
|
110
110
|
# user code the chance to try again or abort the connection.
|
111
|
-
#
|
112
|
-
#
|
113
|
-
#
|
114
|
-
#
|
111
|
+
#
|
112
|
+
# One of either :message, :content, or :header and :body is required:
|
113
|
+
#
|
114
|
+
# :message => String
|
115
|
+
# A valid RFC2822 Internet Message.
|
116
|
+
# :content => String
|
117
|
+
# Raw data which MUST be in correct SMTP body format, with escaped leading dots and a trailing
|
118
|
+
# dot line.
|
119
|
+
# :header => String or Hash of values to be transmitted in the header of the message.
|
120
|
+
# The hash keys are the names of the headers (do NOT append a trailing colon), and the values
|
121
|
+
# are strings containing the header values. TODO, support Arrays of header values, which would
|
122
|
+
# cause us to send that specific header line more than once.
|
115
123
|
#
|
116
124
|
# @example
|
117
125
|
# :header => {"Subject" => "Bogus", "CC" => "myboss@example.com"}
|
118
126
|
#
|
119
|
-
# :body => Optional
|
127
|
+
# :body => Optional String or Array of Strings, defaults blank.
|
120
128
|
# This will be passed as the body of the email message.
|
121
129
|
# TODO, this needs to be significantly beefed up. As currently written, this requires the caller
|
122
130
|
# to properly format the input into CRLF-delimited lines of 7-bit characters in the standard
|
123
131
|
# SMTP transmission format. We need to be able to automatically convert binary data, and add
|
124
|
-
# correct line-breaks to text data.
|
125
|
-
#
|
126
|
-
# Then we can check if either :body or :content is present and do the right thing.
|
127
|
-
# :content => Optional array or string
|
128
|
-
# Alternative to providing header and body, an array or string of raw data which MUST be in
|
129
|
-
# correct SMTP body format, including a trailing dot line
|
132
|
+
# correct line-breaks to text data.
|
133
|
+
#
|
130
134
|
# :verbose => Optional.
|
131
135
|
# If true, will cause a lot of information (including the server-side of the
|
132
136
|
# conversation) to be dumped to $>.
|
@@ -233,12 +237,15 @@ module EventMachine
|
|
233
237
|
close_connection_after_writing
|
234
238
|
end
|
235
239
|
|
240
|
+
def send_ehlo
|
241
|
+
send_data "EHLO #{@args[:domain]}\r\n"
|
242
|
+
end
|
243
|
+
|
236
244
|
def receive_signon
|
237
245
|
return invoke_error unless @range == 2
|
238
|
-
|
246
|
+
send_ehlo
|
239
247
|
@responder = :receive_ehlo_response
|
240
248
|
end
|
241
|
-
|
242
249
|
def receive_ehlo_response
|
243
250
|
return invoke_error unless @range == 2
|
244
251
|
@server_caps = @msg
|
@@ -258,6 +265,15 @@ module EventMachine
|
|
258
265
|
def receive_starttls_response
|
259
266
|
return invoke_error unless @range == 2
|
260
267
|
start_tls
|
268
|
+
invoke_ehlo_over_tls
|
269
|
+
end
|
270
|
+
|
271
|
+
def invoke_ehlo_over_tls
|
272
|
+
send_ehlo
|
273
|
+
@responder = :receive_ehlo_over_tls_response
|
274
|
+
end
|
275
|
+
def receive_ehlo_over_tls_response
|
276
|
+
return invoke_error unless @range == 2
|
261
277
|
invoke_auth
|
262
278
|
end
|
263
279
|
|
@@ -316,6 +332,10 @@ module EventMachine
|
|
316
332
|
invoke_rcpt_to
|
317
333
|
end
|
318
334
|
|
335
|
+
def escape_leading_dots(s)
|
336
|
+
s.gsub(/^\./, '..')
|
337
|
+
end
|
338
|
+
|
319
339
|
def invoke_data
|
320
340
|
send_data "DATA\r\n"
|
321
341
|
@responder = :receive_data_response
|
@@ -323,25 +343,34 @@ module EventMachine
|
|
323
343
|
def receive_data_response
|
324
344
|
return invoke_error unless @range == 3
|
325
345
|
|
326
|
-
# The data to send can be given either
|
327
|
-
#
|
328
|
-
#
|
346
|
+
# The data to send can be given in either @args[:message], @args[:content], or the
|
347
|
+
# combination of @args[:header] and @args[:body].
|
348
|
+
#
|
349
|
+
# - @args[:message] (String) MUST be a valid RFC2822 Internet Message
|
329
350
|
#
|
330
|
-
|
351
|
+
# - @args[:content] (String) MUST be in correct SMTP body format, with escaped
|
352
|
+
# leading dots and a trailing dot line
|
353
|
+
#
|
354
|
+
# - @args[:header] (Hash or String)
|
355
|
+
# - @args[:body] (Array or String)
|
356
|
+
if @args[:message]
|
357
|
+
send_data escape_leading_dots(@args[:message].to_s)
|
358
|
+
send_data "\r\n.\r\n"
|
359
|
+
elsif @args[:content]
|
331
360
|
send_data @args[:content].to_s
|
332
361
|
else
|
333
362
|
# The header can be a hash or an array.
|
334
363
|
if @args[:header].is_a?(Hash)
|
335
|
-
(@args[:header] || {}).each {|k,v| send_data "#{k}: #{v}\r\n" }
|
364
|
+
(@args[:header] || {}).each {|k,v| send_data escape_leading_dots("#{k}: #{v}\r\n") }
|
336
365
|
else
|
337
|
-
send_data @args[:header].to_s
|
366
|
+
send_data escape_leading_dots(@args[:header].to_s)
|
338
367
|
end
|
339
368
|
send_data "\r\n"
|
340
369
|
|
341
370
|
if @args[:body].is_a?(Array)
|
342
|
-
@args[:body].each {|e| send_data e}
|
371
|
+
@args[:body].each {|e| send_data escape_leading_dots(e)}
|
343
372
|
else
|
344
|
-
send_data @args[:body].to_s
|
373
|
+
send_data escape_leading_dots(@args[:body].to_s)
|
345
374
|
end
|
346
375
|
|
347
376
|
send_data "\r\n.\r\n"
|
@@ -227,18 +227,26 @@ module EventMachine
|
|
227
227
|
process_unknown
|
228
228
|
end
|
229
229
|
end
|
230
|
-
|
230
|
+
|
231
231
|
# TODO - implement this properly, the implementation is a stub!
|
232
|
-
def
|
232
|
+
def process_help
|
233
233
|
send_data "250 Ok, but unimplemented\r\n"
|
234
234
|
end
|
235
|
+
|
236
|
+
# RFC2821, 3.5.3 Meaning of VRFY or EXPN Success Response:
|
237
|
+
# A server MUST NOT return a 250 code in response to a VRFY or EXPN
|
238
|
+
# command unless it has actually verified the address. In particular,
|
239
|
+
# a server MUST NOT return 250 if all it has done is to verify that the
|
240
|
+
# syntax given is valid. In that case, 502 (Command not implemented)
|
241
|
+
# or 500 (Syntax error, command unrecognized) SHOULD be returned.
|
242
|
+
#
|
235
243
|
# TODO - implement this properly, the implementation is a stub!
|
236
|
-
def
|
237
|
-
send_data "
|
244
|
+
def process_vrfy
|
245
|
+
send_data "502 Command not implemented\r\n"
|
238
246
|
end
|
239
247
|
# TODO - implement this properly, the implementation is a stub!
|
240
248
|
def process_expn
|
241
|
-
send_data "
|
249
|
+
send_data "502 Command not implemented\r\n"
|
242
250
|
end
|
243
251
|
|
244
252
|
#--
|
@@ -358,12 +366,23 @@ module EventMachine
|
|
358
366
|
def process_auth_line(line)
|
359
367
|
plain = line.unpack("m").first
|
360
368
|
_,user,psw = plain.split("\000")
|
361
|
-
|
369
|
+
|
370
|
+
succeeded = proc {
|
362
371
|
send_data "235 authentication ok\r\n"
|
363
372
|
@state << :auth
|
364
|
-
|
373
|
+
}
|
374
|
+
failed = proc {
|
365
375
|
send_data "535 invalid authentication\r\n"
|
376
|
+
}
|
377
|
+
auth = receive_plain_auth user,psw
|
378
|
+
|
379
|
+
if auth.respond_to?(:callback)
|
380
|
+
auth.callback(&succeeded)
|
381
|
+
auth.errback(&failed)
|
382
|
+
else
|
383
|
+
(auth ? succeeded : failed).call
|
366
384
|
end
|
385
|
+
|
367
386
|
@state.delete :auth_incomplete
|
368
387
|
end
|
369
388
|
|
@@ -409,8 +428,12 @@ module EventMachine
|
|
409
428
|
#--
|
410
429
|
# STARTTLS may not be issued before EHLO, or unless the user has chosen
|
411
430
|
# to support it.
|
412
|
-
# TODO, must support user-supplied certificates.
|
413
431
|
#
|
432
|
+
# If :starttls_options is present and :starttls is set in the parms
|
433
|
+
# pass the options in :starttls_options to start_tls. Do this if you want to use
|
434
|
+
# your own certificate
|
435
|
+
# e.g. {:cert_chain_file => "/etc/ssl/cert.pem", :private_key_file => "/etc/ssl/private/cert.key"}
|
436
|
+
|
414
437
|
def process_starttls
|
415
438
|
if @@parms[:starttls]
|
416
439
|
if @state.include?(:starttls)
|
@@ -419,7 +442,7 @@ module EventMachine
|
|
419
442
|
send_data "503 EHLO required before STARTTLS\r\n"
|
420
443
|
else
|
421
444
|
send_data "220 Start TLS negotiation\r\n"
|
422
|
-
start_tls
|
445
|
+
start_tls(@@parms[:starttls_options] || {})
|
423
446
|
@state << :starttls
|
424
447
|
end
|
425
448
|
else
|
data/lib/em/pure_ruby.rb
CHANGED
@@ -66,6 +66,11 @@ module EventMachine
|
|
66
66
|
def release_machine
|
67
67
|
end
|
68
68
|
|
69
|
+
|
70
|
+
def stopping?
|
71
|
+
return Reactor.instance.stop_scheduled
|
72
|
+
end
|
73
|
+
|
69
74
|
# @private
|
70
75
|
def stop
|
71
76
|
Reactor.instance.stop
|
@@ -273,7 +278,7 @@ module EventMachine
|
|
273
278
|
|
274
279
|
HeartbeatInterval = 2
|
275
280
|
|
276
|
-
attr_reader :current_loop_time
|
281
|
+
attr_reader :current_loop_time, :stop_scheduled
|
277
282
|
|
278
283
|
def initialize
|
279
284
|
initialize_for_run
|
@@ -393,7 +398,7 @@ module EventMachine
|
|
393
398
|
100.times {
|
394
399
|
@loopbreak_port = rand(10000) + 40000
|
395
400
|
begin
|
396
|
-
@loopbreak_reader.bind "
|
401
|
+
@loopbreak_reader.bind "127.0.0.1", @loopbreak_port
|
397
402
|
bound = true
|
398
403
|
break
|
399
404
|
rescue
|
@@ -410,7 +415,7 @@ module EventMachine
|
|
410
415
|
|
411
416
|
def signal_loopbreak
|
412
417
|
#@loopbreak_writer.write '+' if @loopbreak_writer
|
413
|
-
@loopbreak_writer.send('+',0,"
|
418
|
+
@loopbreak_writer.send('+',0,"127.0.0.1",@loopbreak_port) if @loopbreak_writer
|
414
419
|
end
|
415
420
|
|
416
421
|
def set_timer_quantum interval_in_seconds
|
data/lib/em/queue.rb
CHANGED
@@ -17,7 +17,8 @@ module EventMachine
|
|
17
17
|
#
|
18
18
|
class Queue
|
19
19
|
def initialize
|
20
|
-
@
|
20
|
+
@sink = []
|
21
|
+
@drain = []
|
21
22
|
@popq = []
|
22
23
|
end
|
23
24
|
|
@@ -29,10 +30,14 @@ module EventMachine
|
|
29
30
|
def pop(*a, &b)
|
30
31
|
cb = EM::Callback(*a, &b)
|
31
32
|
EM.schedule do
|
32
|
-
if @
|
33
|
+
if @drain.empty?
|
34
|
+
@drain = @sink
|
35
|
+
@sink = []
|
36
|
+
end
|
37
|
+
if @drain.empty?
|
33
38
|
@popq << cb
|
34
39
|
else
|
35
|
-
cb.call @
|
40
|
+
cb.call @drain.shift
|
36
41
|
end
|
37
42
|
end
|
38
43
|
nil # Always returns nil
|
@@ -43,8 +48,12 @@ module EventMachine
|
|
43
48
|
# next reactor tick.
|
44
49
|
def push(*items)
|
45
50
|
EM.schedule do
|
46
|
-
@
|
47
|
-
|
51
|
+
@sink.push(*items)
|
52
|
+
unless @popq.empty?
|
53
|
+
@drain = @sink
|
54
|
+
@sink = []
|
55
|
+
@popq.shift.call @drain.shift until @drain.empty? || @popq.empty?
|
56
|
+
end
|
48
57
|
end
|
49
58
|
end
|
50
59
|
alias :<< :push
|
@@ -52,13 +61,13 @@ module EventMachine
|
|
52
61
|
# @return [Boolean]
|
53
62
|
# @note This is a peek, it's not thread safe, and may only tend toward accuracy.
|
54
63
|
def empty?
|
55
|
-
@
|
64
|
+
@drain.empty? && @sink.empty?
|
56
65
|
end
|
57
66
|
|
58
67
|
# @return [Integer] Queue size
|
59
68
|
# @note This is a peek, it's not thread safe, and may only tend toward accuracy.
|
60
69
|
def size
|
61
|
-
@
|
70
|
+
@drain.size + @sink.size
|
62
71
|
end
|
63
72
|
|
64
73
|
# @return [Integer] Waiting size
|