eventmachine 1.0.3-x86-mingw32 → 1.2.0.dev.2-x86-mingw32
Sign up to get free protection for your applications and to get access to all the features.
- 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
Binary file
|
Binary file
|
Binary file
|
Binary file
|
Binary file
|
Binary file
|
Binary file
|
Binary file
|
Binary file
|
Binary file
|
data/lib/em/buftok.rb
CHANGED
@@ -1,110 +1,59 @@
|
|
1
1
|
# BufferedTokenizer takes a delimiter upon instantiation, or acts line-based
|
2
2
|
# by default. It allows input to be spoon-fed from some outside source which
|
3
3
|
# receives arbitrary length datagrams which may-or-may-not contain the token
|
4
|
-
# by which entities are delimited.
|
5
|
-
#
|
6
|
-
# By default, new BufferedTokenizers will operate on lines delimited by "\n" by default
|
7
|
-
# or allow you to specify any delimiter token you so choose, which will then
|
8
|
-
# be used by String#split to tokenize the input data
|
9
|
-
#
|
10
|
-
# @example Using BufferedTokernizer to parse lines out of incoming data
|
11
|
-
#
|
12
|
-
# module LineBufferedConnection
|
13
|
-
# def receive_data(data)
|
14
|
-
# (@buffer ||= BufferedTokenizer.new).extract(data).each do |line|
|
15
|
-
# receive_line(line)
|
16
|
-
# end
|
17
|
-
# end
|
18
|
-
# end
|
19
|
-
#
|
20
|
-
# @author Tony Arcieri
|
21
|
-
# @author Martin Emde
|
4
|
+
# by which entities are delimited. In this respect it's ideally paired with
|
5
|
+
# something like EventMachine (http://rubyeventmachine.com/).
|
22
6
|
class BufferedTokenizer
|
23
|
-
#
|
24
|
-
#
|
25
|
-
|
26
|
-
|
27
|
-
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
33
|
-
# number of objects required for the operation.
|
7
|
+
# New BufferedTokenizers will operate on lines delimited by a delimiter,
|
8
|
+
# which is by default the global input delimiter $/ ("\n").
|
9
|
+
#
|
10
|
+
# The input buffer is stored as an array. This is by far the most efficient
|
11
|
+
# approach given language constraints (in C a linked list would be a more
|
12
|
+
# appropriate data structure). Segments of input data are stored in a list
|
13
|
+
# which is only joined when a token is reached, substantially reducing the
|
14
|
+
# number of objects required for the operation.
|
15
|
+
def initialize(delimiter = $/)
|
16
|
+
@delimiter = delimiter
|
34
17
|
@input = []
|
35
|
-
|
36
|
-
|
37
|
-
@input_size = 0
|
18
|
+
@tail = ''
|
19
|
+
@trim = @delimiter.length - 1
|
38
20
|
end
|
39
21
|
|
40
22
|
# Extract takes an arbitrary string of input data and returns an array of
|
41
|
-
# tokenized entities, provided there were any available to extract.
|
42
|
-
#
|
43
|
-
# @example
|
23
|
+
# tokenized entities, provided there were any available to extract. This
|
24
|
+
# makes for easy processing of datagrams using a pattern like:
|
44
25
|
#
|
45
|
-
# tokenizer.extract(data).
|
46
|
-
# map { |entity| Decode(entity) }.each { ... }
|
26
|
+
# tokenizer.extract(data).map { |entity| Decode(entity) }.each do ...
|
47
27
|
#
|
48
|
-
#
|
28
|
+
# Using -1 makes split to return "" if the token is at the end of
|
29
|
+
# the string, meaning the last element is the start of the next chunk.
|
49
30
|
def extract(data)
|
50
|
-
|
51
|
-
|
52
|
-
|
53
|
-
# input buffer or not (i.e. a literal edge case) Specifying -1 forces split to
|
54
|
-
# return "" in this case, meaning that the last entry in the list represents a
|
55
|
-
# new segment of data where the token has not been encountered
|
56
|
-
entities = data.split @delimiter, -1
|
57
|
-
|
58
|
-
# Check to see if the buffer has exceeded capacity, if we're imposing a limit
|
59
|
-
if @size_limit
|
60
|
-
raise 'input buffer full' if @input_size + entities.first.size > @size_limit
|
61
|
-
@input_size += entities.first.size
|
31
|
+
if @trim > 0
|
32
|
+
tail_end = @tail.slice!(-@trim, @trim) # returns nil if string is too short
|
33
|
+
data = tail_end + data if tail_end
|
62
34
|
end
|
63
35
|
|
64
|
-
|
65
|
-
|
66
|
-
@
|
67
|
-
|
68
|
-
# If the resulting array from the split is empty, the token was not encountered
|
69
|
-
# (not even at the end of the buffer). Since we've encountered no token-delimited
|
70
|
-
# entities this go-around, return an empty array.
|
71
|
-
return [] if entities.empty?
|
72
|
-
|
73
|
-
# At this point, we've hit a token, or potentially multiple tokens. Now we can bring
|
74
|
-
# together all the data we've buffered from earlier calls without hitting a token,
|
75
|
-
# and add it to our list of discovered entities.
|
76
|
-
entities.unshift @input.join
|
36
|
+
@input << @tail
|
37
|
+
entities = data.split(@delimiter, -1)
|
38
|
+
@tail = entities.shift
|
77
39
|
|
78
|
-
|
79
|
-
|
80
|
-
|
81
|
-
|
82
|
-
|
83
|
-
|
84
|
-
# passed to split. It represents the beginning of a new list of as-yet-untokenized
|
85
|
-
# data, so we add it to the start of the list.
|
86
|
-
@input << entities.pop
|
87
|
-
|
88
|
-
# Set the new input buffer size, provided we're keeping track
|
89
|
-
@input_size = @input.first.size if @size_limit
|
40
|
+
unless entities.empty?
|
41
|
+
@input << @tail
|
42
|
+
entities.unshift @input.join
|
43
|
+
@input.clear
|
44
|
+
@tail = entities.pop
|
45
|
+
end
|
90
46
|
|
91
|
-
# Now we're left with the list of extracted token-delimited entities we wanted
|
92
|
-
# in the first place. Hooray!
|
93
47
|
entities
|
94
48
|
end
|
95
49
|
|
96
50
|
# Flush the contents of the input buffer, i.e. return the input buffer even though
|
97
|
-
# a token has not yet been encountered
|
98
|
-
#
|
99
|
-
# @return [String]
|
51
|
+
# a token has not yet been encountered
|
100
52
|
def flush
|
53
|
+
@input << @tail
|
101
54
|
buffer = @input.join
|
102
55
|
@input.clear
|
56
|
+
@tail = "" # @tail.clear is slightly faster, but not supported on 1.8.7
|
103
57
|
buffer
|
104
58
|
end
|
105
|
-
|
106
|
-
# @return [Boolean]
|
107
|
-
def empty?
|
108
|
-
@input.empty?
|
109
|
-
end
|
110
59
|
end
|
data/lib/em/channel.rb
CHANGED
@@ -17,6 +17,11 @@ module EventMachine
|
|
17
17
|
@uid = 0
|
18
18
|
end
|
19
19
|
|
20
|
+
# Return the number of current subscribers.
|
21
|
+
def num_subscribers
|
22
|
+
return @subs.size
|
23
|
+
end
|
24
|
+
|
20
25
|
# Takes any arguments suitable for EM::Callback() and returns a subscriber
|
21
26
|
# id for use when unsubscribing.
|
22
27
|
#
|
data/lib/em/completion.rb
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
# = EM::Completion
|
2
2
|
#
|
3
3
|
# A completion is a callback container for various states of completion. In
|
4
|
-
#
|
4
|
+
# its most basic form it has a start state and a finish state.
|
5
5
|
#
|
6
6
|
# This implementation includes some hold-back from the EM::Deferrable
|
7
7
|
# interface in order to be compatible - but it has a much cleaner
|
@@ -50,7 +50,7 @@
|
|
50
50
|
# @completion.fail :unknown, line
|
51
51
|
# end
|
52
52
|
# end
|
53
|
-
#
|
53
|
+
#
|
54
54
|
# def unbind
|
55
55
|
# @completion.fail :disconnected unless @completion.completed?
|
56
56
|
# end
|
data/lib/em/connection.rb
CHANGED
@@ -376,10 +376,21 @@ module EventMachine
|
|
376
376
|
#
|
377
377
|
# @option args [String] :private_key_file (nil) local path of a readable file that must contain a private key in the [PEM format](http://en.wikipedia.org/wiki/Privacy_Enhanced_Mail).
|
378
378
|
#
|
379
|
-
# @option args [
|
379
|
+
# @option args [Boolean] :verify_peer (false) indicates whether a server should request a certificate from a peer, to be verified by user code.
|
380
380
|
# If true, the {#ssl_verify_peer} callback on the {EventMachine::Connection} object is called with each certificate
|
381
381
|
# in the certificate chain provided by the peer. See documentation on {#ssl_verify_peer} for how to use this.
|
382
382
|
#
|
383
|
+
# @option args [Boolean] :fail_if_no_peer_cert (false) Used in conjunction with verify_peer. If set the SSL handshake will be terminated if the peer does not provide a certificate.
|
384
|
+
#
|
385
|
+
#
|
386
|
+
# @option args [String] :cipher_list ("ALL:!ADH:!LOW:!EXP:!DES-CBC3-SHA:@STRENGTH") indicates the available SSL cipher values. Default value is "ALL:!ADH:!LOW:!EXP:!DES-CBC3-SHA:@STRENGTH". Check the format of the OpenSSL cipher string at http://www.openssl.org/docs/apps/ciphers.html#CIPHER_LIST_FORMAT.
|
387
|
+
#
|
388
|
+
# @option args [String] :ecdh_curve (nil) The curve for ECDHE ciphers. See available ciphers with 'openssl ecparam -list_curves'
|
389
|
+
#
|
390
|
+
# @option args [String] :dhparam (nil) The local path of a file containing DH parameters for EDH ciphers in [PEM format](http://en.wikipedia.org/wiki/Privacy_Enhanced_Mail) See: 'openssl dhparam'
|
391
|
+
#
|
392
|
+
# @option args [Array] :ssl_version (TLSv1 TLSv1_1 TLSv1_2) indicates the allowed SSL/TLS versions. Possible values are: {SSLv2}, {SSLv3}, {TLSv1}, {TLSv1_1}, {TLSv1_2}.
|
393
|
+
#
|
383
394
|
# @example Using TLS with EventMachine
|
384
395
|
#
|
385
396
|
# require 'rubygems'
|
@@ -404,15 +415,47 @@ module EventMachine
|
|
404
415
|
#
|
405
416
|
# @see #ssl_verify_peer
|
406
417
|
def start_tls args={}
|
407
|
-
priv_key
|
418
|
+
priv_key = args[:private_key_file]
|
419
|
+
cert_chain = args[:cert_chain_file]
|
420
|
+
verify_peer = args[:verify_peer]
|
421
|
+
sni_hostname = args[:sni_hostname]
|
422
|
+
cipher_list = args[:cipher_list]
|
423
|
+
ssl_version = args[:ssl_version]
|
424
|
+
ecdh_curve = args[:ecdh_curve]
|
425
|
+
dhparam = args[:dhparam]
|
426
|
+
fail_if_no_peer_cert = args[:fail_if_no_peer_cert]
|
408
427
|
|
409
428
|
[priv_key, cert_chain].each do |file|
|
410
429
|
next if file.nil? or file.empty?
|
411
430
|
raise FileNotFoundException,
|
412
|
-
"Could not find #{file} for start_tls" unless File.
|
431
|
+
"Could not find #{file} for start_tls" unless File.exist? file
|
432
|
+
end
|
433
|
+
|
434
|
+
protocols_bitmask = 0
|
435
|
+
if ssl_version.nil?
|
436
|
+
protocols_bitmask |= EventMachine::EM_PROTO_TLSv1
|
437
|
+
protocols_bitmask |= EventMachine::EM_PROTO_TLSv1_1
|
438
|
+
protocols_bitmask |= EventMachine::EM_PROTO_TLSv1_2
|
439
|
+
else
|
440
|
+
[ssl_version].flatten.each do |p|
|
441
|
+
case p.to_s.downcase
|
442
|
+
when 'sslv2'
|
443
|
+
protocols_bitmask |= EventMachine::EM_PROTO_SSLv2
|
444
|
+
when 'sslv3'
|
445
|
+
protocols_bitmask |= EventMachine::EM_PROTO_SSLv3
|
446
|
+
when 'tlsv1'
|
447
|
+
protocols_bitmask |= EventMachine::EM_PROTO_TLSv1
|
448
|
+
when 'tlsv1_1'
|
449
|
+
protocols_bitmask |= EventMachine::EM_PROTO_TLSv1_1
|
450
|
+
when 'tlsv1_2'
|
451
|
+
protocols_bitmask |= EventMachine::EM_PROTO_TLSv1_2
|
452
|
+
else
|
453
|
+
raise("Unrecognized SSL/TLS Protocol: #{p}")
|
454
|
+
end
|
455
|
+
end
|
413
456
|
end
|
414
457
|
|
415
|
-
EventMachine::set_tls_parms(@signature, priv_key || '', cert_chain || '', verify_peer)
|
458
|
+
EventMachine::set_tls_parms(@signature, priv_key || '', cert_chain || '', verify_peer, fail_if_no_peer_cert, sni_hostname || '', cipher_list || '', ecdh_curve || '', dhparam || '', protocols_bitmask)
|
416
459
|
EventMachine::start_tls @signature
|
417
460
|
end
|
418
461
|
|
@@ -488,6 +531,21 @@ module EventMachine
|
|
488
531
|
EventMachine::get_peer_cert @signature
|
489
532
|
end
|
490
533
|
|
534
|
+
def get_cipher_bits
|
535
|
+
EventMachine::get_cipher_bits @signature
|
536
|
+
end
|
537
|
+
|
538
|
+
def get_cipher_name
|
539
|
+
EventMachine::get_cipher_name @signature
|
540
|
+
end
|
541
|
+
|
542
|
+
def get_cipher_protocol
|
543
|
+
EventMachine::get_cipher_protocol @signature
|
544
|
+
end
|
545
|
+
|
546
|
+
def get_sni_hostname
|
547
|
+
EventMachine::get_sni_hostname @signature
|
548
|
+
end
|
491
549
|
|
492
550
|
# Sends UDP messages.
|
493
551
|
#
|
data/lib/em/iterator.rb
CHANGED
@@ -41,6 +41,7 @@ module EventMachine
|
|
41
41
|
# end
|
42
42
|
#
|
43
43
|
class Iterator
|
44
|
+
Stop = "EM::Stop"
|
44
45
|
# Create a new parallel async iterator with specified concurrency.
|
45
46
|
#
|
46
47
|
# i = EM::Iterator.new(1..100, 10)
|
@@ -48,9 +49,21 @@ module EventMachine
|
|
48
49
|
# will create an iterator over the range that processes 10 items at a time. Iteration
|
49
50
|
# is started via #each, #map or #inject
|
50
51
|
#
|
52
|
+
# The list may either be an array-like object, or a proc that returns a new object
|
53
|
+
# to be processed each time it is called. If a proc is used, it must return
|
54
|
+
# EventMachine::Iterator::Stop to signal the end of the iterations.
|
55
|
+
#
|
51
56
|
def initialize(list, concurrency = 1)
|
52
|
-
raise ArgumentError, '
|
53
|
-
|
57
|
+
raise ArgumentError, 'concurrency must be bigger than zero' unless (concurrency > 0)
|
58
|
+
if list.respond_to?(:call)
|
59
|
+
@list = nil
|
60
|
+
@list_proc = list
|
61
|
+
elsif list.respond_to?(:to_a)
|
62
|
+
@list = list.to_a.dup
|
63
|
+
@list_proc = nil
|
64
|
+
else
|
65
|
+
raise ArgumentError, 'argument must be a proc or an array'
|
66
|
+
end
|
54
67
|
@concurrency = concurrency
|
55
68
|
|
56
69
|
@started = false
|
@@ -97,12 +110,12 @@ module EventMachine
|
|
97
110
|
@process_next = proc{
|
98
111
|
# p [:process_next, :pending=, @pending, :workers=, @workers, :ended=, @ended, :concurrency=, @concurrency, :list=, @list]
|
99
112
|
unless @ended or @workers > @concurrency
|
100
|
-
|
113
|
+
item = next_item()
|
114
|
+
if item.equal?(Stop)
|
101
115
|
@ended = true
|
102
116
|
@workers -= 1
|
103
117
|
all_done.call
|
104
118
|
else
|
105
|
-
item = @list.shift
|
106
119
|
@pending += 1
|
107
120
|
|
108
121
|
is_done = false
|
@@ -221,50 +234,19 @@ module EventMachine
|
|
221
234
|
})
|
222
235
|
nil
|
223
236
|
end
|
237
|
+
|
238
|
+
# Return the next item from @list or @list_proc.
|
239
|
+
# Once items have run out, will return EM::Iterator::Stop. Procs must supply this themselves
|
240
|
+
def next_item
|
241
|
+
if @list_proc
|
242
|
+
@list_proc.call
|
243
|
+
else
|
244
|
+
@list.empty? ? Stop : @list.shift
|
245
|
+
end
|
246
|
+
end
|
224
247
|
end
|
225
248
|
end
|
226
249
|
|
227
|
-
|
228
|
-
|
229
|
-
|
230
|
-
|
231
|
-
# TODO: real tests
|
232
|
-
# TODO: pass in one object instead of two? .each{ |iter| puts iter.current; iter.next }
|
233
|
-
# TODO: support iter.pause/resume/stop/break/continue?
|
234
|
-
# TODO: create some exceptions instead of using RuntimeError
|
235
|
-
# TODO: support proc instead of enumerable? EM::Iterator.new(proc{ return queue.pop })
|
236
|
-
|
237
|
-
EM.run{
|
238
|
-
EM::Iterator.new(1..50).each{ |num,iter| p num; iter.next }
|
239
|
-
EM::Iterator.new([1,2,3], 10).each{ |num,iter| p num; iter.next }
|
240
|
-
|
241
|
-
i = EM::Iterator.new(1..100, 5)
|
242
|
-
i.each(proc{|num,iter|
|
243
|
-
p num.to_s
|
244
|
-
iter.next
|
245
|
-
}, proc{
|
246
|
-
p :done
|
247
|
-
})
|
248
|
-
EM.add_timer(0.03){
|
249
|
-
i.concurrency = 1
|
250
|
-
}
|
251
|
-
EM.add_timer(0.04){
|
252
|
-
i.concurrency = 3
|
253
|
-
}
|
254
|
-
|
255
|
-
EM::Iterator.new(100..150).map(proc{ |num,iter|
|
256
|
-
EM.add_timer(0.01){ iter.return(num) }
|
257
|
-
}, proc{ |results|
|
258
|
-
p results
|
259
|
-
})
|
260
|
-
|
261
|
-
EM::Iterator.new(%w[ pwd uptime uname date ], 2).inject({}, proc{ |hash,cmd,iter|
|
262
|
-
EM.system(cmd){ |output,status|
|
263
|
-
hash[cmd] = status.exitstatus == 0 ? output.strip : nil
|
264
|
-
iter.return(hash)
|
265
|
-
}
|
266
|
-
}, proc{ |results|
|
267
|
-
p results
|
268
|
-
})
|
269
|
-
}
|
270
|
-
end
|
250
|
+
# TODO: pass in one object instead of two? .each{ |iter| puts iter.current; iter.next }
|
251
|
+
# TODO: support iter.pause/resume/stop/break/continue?
|
252
|
+
# TODO: create some exceptions instead of using RuntimeError
|
data/lib/em/pool.rb
CHANGED
@@ -23,8 +23,6 @@
|
|
23
23
|
#
|
24
24
|
#
|
25
25
|
|
26
|
-
|
27
|
-
|
28
26
|
module EventMachine
|
29
27
|
module Protocols
|
30
28
|
|
@@ -52,7 +50,6 @@ module EventMachine
|
|
52
50
|
# DNS: Some way to cache DNS lookups for hostnames we connect to. Ruby's
|
53
51
|
# DNS lookups are unbelievably slow.
|
54
52
|
# HEAD requests.
|
55
|
-
# Chunked transfer encoding.
|
56
53
|
# Convenience methods for requests. get, post, url, etc.
|
57
54
|
# SSL.
|
58
55
|
# Handle status codes like 304, 100, etc.
|
@@ -184,6 +181,8 @@ module EventMachine
|
|
184
181
|
@content_length = nil # not zero
|
185
182
|
@content = ""
|
186
183
|
@status = nil
|
184
|
+
@chunked = false
|
185
|
+
@chunk_length = nil
|
187
186
|
@read_state = :header
|
188
187
|
@connection_close = nil
|
189
188
|
when :header
|
@@ -191,7 +190,7 @@ module EventMachine
|
|
191
190
|
if ary.length == 2
|
192
191
|
data = ary.last
|
193
192
|
if ary.first == ""
|
194
|
-
if (@content_length and @content_length > 0) || @connection_close
|
193
|
+
if (@content_length and @content_length > 0) || @chunked || @connection_close
|
195
194
|
@read_state = :content
|
196
195
|
else
|
197
196
|
dispatch_response
|
@@ -211,6 +210,8 @@ module EventMachine
|
|
211
210
|
@content_length ||= $'.to_i
|
212
211
|
elsif ary.first =~ /\Aconnection:\s*close/i
|
213
212
|
@connection_close = true
|
213
|
+
elsif ary.first =~ /\Atransfer-encoding:\s*chunked/i
|
214
|
+
@chunked = true
|
214
215
|
end
|
215
216
|
end
|
216
217
|
else
|
@@ -218,12 +219,32 @@ module EventMachine
|
|
218
219
|
data = ""
|
219
220
|
end
|
220
221
|
when :content
|
221
|
-
|
222
|
-
|
223
|
-
|
224
|
-
|
225
|
-
|
226
|
-
|
222
|
+
if @chunked && @chunk_length
|
223
|
+
bytes_needed = @chunk_length - @chunk_read
|
224
|
+
new_data = data[0, bytes_needed]
|
225
|
+
@chunk_read += new_data.length
|
226
|
+
@content += new_data
|
227
|
+
data = data[bytes_needed..-1] || ""
|
228
|
+
if @chunk_length == @chunk_read && data[0,2] == "\r\n"
|
229
|
+
@chunk_length = nil
|
230
|
+
data = data[2..-1]
|
231
|
+
end
|
232
|
+
elsif @chunked
|
233
|
+
if (m = data.match(/\A(\S*)\r\n/m))
|
234
|
+
data = data[m[0].length..-1]
|
235
|
+
@chunk_length = m[1].to_i(16)
|
236
|
+
@chunk_read = 0
|
237
|
+
if @chunk_length == 0
|
238
|
+
dispatch_response
|
239
|
+
@read_state = :base
|
240
|
+
end
|
241
|
+
end
|
242
|
+
elsif @content_length
|
243
|
+
# If there was no content-length header, we have to wait until the connection
|
244
|
+
# closes. Everything we get until that point is content.
|
245
|
+
# TODO: Must impose a content-size limit, and also must implement chunking.
|
246
|
+
# Also, must support either temporary files for large content, or calling
|
247
|
+
# a content-consumer block supplied by the user.
|
227
248
|
bytes_needed = @content_length - @content.length
|
228
249
|
@content += data[0, bytes_needed]
|
229
250
|
data = data[bytes_needed..-1] || ""
|
@@ -274,6 +295,5 @@ module EventMachine
|
|
274
295
|
end
|
275
296
|
end
|
276
297
|
end
|
277
|
-
|
278
298
|
end
|
279
299
|
end
|