eventmachine 1.0.0.beta.3-x86-mswin32-60 → 1.0.0.beta.4.1-x86-mswin32-60
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.
- data/.gitignore +5 -0
- data/.yardopts +5 -1
- data/{docs/GNU → GNU} +0 -0
- data/Gemfile +1 -0
- data/{docs/COPYING → LICENSE} +0 -0
- data/README.md +109 -0
- data/Rakefile +8 -0
- data/docs/DocumentationGuidesIndex.md +27 -0
- data/docs/GettingStarted.md +521 -0
- data/docs/{ChangeLog → old/ChangeLog} +0 -0
- data/docs/{DEFERRABLES → old/DEFERRABLES} +0 -0
- data/docs/{EPOLL → old/EPOLL} +0 -0
- data/docs/{INSTALL → old/INSTALL} +0 -0
- data/docs/{KEYBOARD → old/KEYBOARD} +0 -0
- data/docs/{LEGAL → old/LEGAL} +0 -0
- data/docs/{LIGHTWEIGHT_CONCURRENCY → old/LIGHTWEIGHT_CONCURRENCY} +0 -0
- data/docs/{PURE_RUBY → old/PURE_RUBY} +0 -0
- data/docs/{RELEASE_NOTES → old/RELEASE_NOTES} +0 -0
- data/docs/{SMTP → old/SMTP} +0 -0
- data/docs/{SPAWNED_PROCESSES → old/SPAWNED_PROCESSES} +0 -0
- data/docs/{TODO → old/TODO} +0 -0
- data/eventmachine.gemspec +5 -2
- data/examples/guides/getting_started/01_eventmachine_echo_server.rb +18 -0
- data/examples/guides/getting_started/02_eventmachine_echo_server_that_recognizes_exit_command.rb +22 -0
- data/examples/guides/getting_started/03_simple_chat_server.rb +149 -0
- data/examples/guides/getting_started/04_simple_chat_server_step_one.rb +27 -0
- data/examples/guides/getting_started/05_simple_chat_server_step_two.rb +43 -0
- data/examples/guides/getting_started/06_simple_chat_server_step_three.rb +98 -0
- data/examples/guides/getting_started/07_simple_chat_server_step_four.rb +121 -0
- data/examples/guides/getting_started/08_simple_chat_server_step_five.rb +141 -0
- data/examples/{ex_channel.rb → old/ex_channel.rb} +3 -3
- data/examples/{ex_queue.rb → old/ex_queue.rb} +0 -0
- data/examples/{ex_tick_loop_array.rb → old/ex_tick_loop_array.rb} +0 -0
- data/examples/{ex_tick_loop_counter.rb → old/ex_tick_loop_counter.rb} +0 -0
- data/examples/{helper.rb → old/helper.rb} +0 -0
- data/ext/cmain.cpp +3 -3
- data/ext/ed.cpp +90 -15
- data/ext/ed.h +5 -5
- data/ext/em.cpp +48 -56
- data/ext/em.h +12 -2
- data/ext/extconf.rb +3 -3
- data/ext/fastfilereader/extconf.rb +1 -1
- data/ext/pipe.cpp +2 -2
- data/ext/project.h +1 -1
- data/ext/rubymain.cpp +48 -3
- data/ext/ssl.cpp +5 -0
- data/java/src/com/rubyeventmachine/EmReactor.java +2 -2
- data/lib/em/buftok.rb +35 -63
- data/lib/em/callback.rb +43 -11
- data/lib/em/channel.rb +21 -14
- data/lib/em/completion.rb +304 -0
- data/lib/em/connection.rb +339 -209
- data/lib/em/deferrable.rb +4 -0
- data/lib/em/deferrable/pool.rb +2 -0
- data/lib/em/file_watch.rb +37 -18
- data/lib/em/iterator.rb +42 -42
- data/lib/em/pool.rb +146 -0
- data/lib/em/process_watch.rb +5 -4
- data/lib/em/processes.rb +8 -4
- data/lib/em/protocols/httpclient.rb +22 -11
- data/lib/em/protocols/httpclient2.rb +15 -5
- data/lib/em/protocols/line_protocol.rb +2 -1
- data/lib/em/protocols/memcache.rb +17 -9
- data/lib/em/protocols/object_protocol.rb +2 -1
- data/lib/em/protocols/postgres3.rb +8 -9
- data/lib/em/protocols/smtpclient.rb +19 -11
- data/lib/em/protocols/smtpserver.rb +1 -1
- data/lib/em/protocols/stomp.rb +8 -6
- data/lib/em/protocols/tcptest.rb +3 -2
- data/lib/em/pure_ruby.rb +212 -208
- data/lib/em/queue.rb +22 -13
- data/lib/em/resolver.rb +70 -64
- data/lib/em/spawnable.rb +6 -3
- data/lib/em/streamer.rb +33 -45
- data/lib/em/threaded_resource.rb +90 -0
- data/lib/em/timers.rb +6 -2
- data/lib/em/version.rb +1 -1
- data/lib/eventmachine.rb +538 -602
- data/lib/jeventmachine.rb +22 -1
- data/tasks/package.rake +12 -2
- data/tasks/test.rake +1 -0
- data/tests/em_test_helper.rb +12 -3
- data/tests/test_completion.rb +177 -0
- data/tests/test_epoll.rb +2 -2
- data/tests/test_httpclient.rb +9 -9
- data/tests/test_httpclient2.rb +11 -9
- data/tests/test_ltp.rb +2 -10
- data/tests/test_pool.rb +128 -0
- data/tests/test_processes.rb +20 -2
- data/tests/test_queue.rb +8 -0
- data/tests/test_resolver.rb +1 -1
- data/tests/test_set_sock_opt.rb +37 -0
- data/tests/test_shutdown_hooks.rb +23 -0
- data/tests/test_threaded_resource.rb +53 -0
- data/tests/test_unbind_reason.rb +31 -0
- metadata +96 -32
- data/README +0 -81
- data/tasks/doc.rake +0 -30
data/lib/em/queue.rb
CHANGED
@@ -1,29 +1,31 @@
|
|
1
1
|
module EventMachine
|
2
2
|
# A cross thread, reactor scheduled, linear queue.
|
3
3
|
#
|
4
|
-
# This class provides a simple
|
4
|
+
# This class provides a simple queue abstraction on top of the reactor
|
5
5
|
# scheduler. It services two primary purposes:
|
6
|
+
#
|
6
7
|
# * API sugar for stateful protocols
|
7
|
-
# * Pushing processing onto the
|
8
|
+
# * Pushing processing onto the reactor thread
|
8
9
|
#
|
9
|
-
#
|
10
|
+
# @example
|
10
11
|
#
|
11
12
|
# q = EM::Queue.new
|
12
13
|
# q.push('one', 'two', 'three')
|
13
14
|
# 3.times do
|
14
|
-
# q.pop{ |msg| puts(msg) }
|
15
|
+
# q.pop { |msg| puts(msg) }
|
15
16
|
# end
|
16
17
|
#
|
17
18
|
class Queue
|
18
|
-
# Create a new queue
|
19
19
|
def initialize
|
20
20
|
@items = []
|
21
21
|
@popq = []
|
22
22
|
end
|
23
23
|
|
24
24
|
# Pop items off the queue, running the block on the reactor thread. The pop
|
25
|
-
# will not happen immediately, but at some point in the future, either in
|
25
|
+
# will not happen immediately, but at some point in the future, either in
|
26
26
|
# the next tick, if the queue has data, or when the queue is populated.
|
27
|
+
#
|
28
|
+
# @return [NilClass] nil
|
27
29
|
def pop(*a, &b)
|
28
30
|
cb = EM::Callback(*a, &b)
|
29
31
|
EM.schedule do
|
@@ -37,7 +39,7 @@ module EventMachine
|
|
37
39
|
end
|
38
40
|
|
39
41
|
# Push items onto the queue in the reactor thread. The items will not appear
|
40
|
-
# in the queue immediately, but will be scheduled for addition during the
|
42
|
+
# in the queue immediately, but will be scheduled for addition during the
|
41
43
|
# next reactor tick.
|
42
44
|
def push(*items)
|
43
45
|
EM.schedule do
|
@@ -47,16 +49,23 @@ module EventMachine
|
|
47
49
|
end
|
48
50
|
alias :<< :push
|
49
51
|
|
50
|
-
#
|
51
|
-
# accuracy.
|
52
|
+
# @return [Boolean]
|
53
|
+
# @note This is a peek, it's not thread safe, and may only tend toward accuracy.
|
52
54
|
def empty?
|
53
55
|
@items.empty?
|
54
56
|
end
|
55
57
|
|
56
|
-
#
|
57
|
-
# accuracy.
|
58
|
+
# @return [Integer] Queue size
|
59
|
+
# @note This is a peek, it's not thread safe, and may only tend toward accuracy.
|
58
60
|
def size
|
59
61
|
@items.size
|
60
62
|
end
|
61
|
-
|
62
|
-
|
63
|
+
|
64
|
+
# @return [Integer] Waiting size
|
65
|
+
# @note This is a peek at the number of jobs that are currently waiting on the Queue
|
66
|
+
def num_waiting
|
67
|
+
@popq.size
|
68
|
+
end
|
69
|
+
|
70
|
+
end # Queue
|
71
|
+
end # EventMachine
|
data/lib/em/resolver.rb
CHANGED
@@ -6,6 +6,8 @@ module EventMachine
|
|
6
6
|
Request.new(socket, hostname)
|
7
7
|
end
|
8
8
|
|
9
|
+
@socket = @nameservers = nil
|
10
|
+
|
9
11
|
def self.socket
|
10
12
|
if !@socket || (@socket && @socket.error?)
|
11
13
|
@socket = Socket.open
|
@@ -58,6 +60,10 @@ module EventMachine
|
|
58
60
|
EventMachine::open_datagram_socket('0.0.0.0', 0, self)
|
59
61
|
end
|
60
62
|
|
63
|
+
def initialize
|
64
|
+
@nameserver = nil
|
65
|
+
end
|
66
|
+
|
61
67
|
def post_init
|
62
68
|
@requests = {}
|
63
69
|
EM.add_periodic_timer(0.1, &method(:tick))
|
@@ -99,88 +105,88 @@ module EventMachine
|
|
99
105
|
begin
|
100
106
|
msg = Resolv::DNS::Message.decode data
|
101
107
|
rescue
|
102
|
-
|
103
|
-
|
104
|
-
|
105
|
-
|
106
|
-
|
107
|
-
end
|
108
|
+
else
|
109
|
+
req = @requests[msg.id]
|
110
|
+
if req
|
111
|
+
@requests.delete(msg.id)
|
112
|
+
req.receive_answer(msg)
|
108
113
|
end
|
109
114
|
end
|
110
115
|
end
|
116
|
+
end
|
111
117
|
|
112
|
-
|
113
|
-
|
114
|
-
|
118
|
+
class Request
|
119
|
+
include Deferrable
|
120
|
+
attr_accessor :retry_interval, :max_tries
|
115
121
|
|
116
|
-
|
117
|
-
|
118
|
-
|
119
|
-
|
120
|
-
|
121
|
-
|
122
|
-
|
122
|
+
def initialize(socket, hostname)
|
123
|
+
@socket = socket
|
124
|
+
@hostname = hostname
|
125
|
+
@tries = 0
|
126
|
+
@last_send = Time.at(0)
|
127
|
+
@retry_interval = 3
|
128
|
+
@max_tries = 5
|
123
129
|
|
124
|
-
|
125
|
-
|
126
|
-
|
127
|
-
|
128
|
-
end
|
130
|
+
if addrs = Resolver.hosts[hostname]
|
131
|
+
succeed addrs
|
132
|
+
else
|
133
|
+
EM.next_tick { tick }
|
129
134
|
end
|
135
|
+
end
|
130
136
|
|
131
|
-
|
132
|
-
|
133
|
-
|
134
|
-
|
135
|
-
|
136
|
-
|
137
|
-
|
138
|
-
end
|
137
|
+
def tick
|
138
|
+
# Break early if nothing to do
|
139
|
+
return if @last_send + @retry_interval > Time.now
|
140
|
+
if @tries < @max_tries
|
141
|
+
send
|
142
|
+
else
|
143
|
+
fail 'retries exceeded'
|
139
144
|
end
|
145
|
+
end
|
140
146
|
|
141
|
-
|
142
|
-
|
143
|
-
|
144
|
-
|
145
|
-
|
146
|
-
|
147
|
-
end
|
147
|
+
def receive_answer(msg)
|
148
|
+
addrs = []
|
149
|
+
msg.each_answer do |name,ttl,data|
|
150
|
+
if data.kind_of?(Resolv::DNS::Resource::IN::A) ||
|
151
|
+
data.kind_of?(Resolv::DNS::Resource::IN::AAAA)
|
152
|
+
addrs << data.address.to_s
|
148
153
|
end
|
154
|
+
end
|
149
155
|
|
150
|
-
|
151
|
-
|
152
|
-
|
153
|
-
|
154
|
-
end
|
156
|
+
if addrs.empty?
|
157
|
+
fail "rcode=#{msg.rcode}"
|
158
|
+
else
|
159
|
+
succeed addrs
|
155
160
|
end
|
161
|
+
end
|
156
162
|
|
157
|
-
|
163
|
+
private
|
158
164
|
|
159
|
-
|
160
|
-
|
161
|
-
|
162
|
-
|
163
|
-
|
165
|
+
def send
|
166
|
+
@tries += 1
|
167
|
+
@last_send = Time.now
|
168
|
+
@socket.send_packet(packet.encode)
|
169
|
+
end
|
164
170
|
|
165
|
-
|
166
|
-
|
167
|
-
|
168
|
-
|
169
|
-
|
170
|
-
|
171
|
-
|
171
|
+
def id
|
172
|
+
begin
|
173
|
+
@id = rand(65535)
|
174
|
+
@socket.register_request(@id, self)
|
175
|
+
rescue RequestIdAlreadyUsed
|
176
|
+
retry
|
177
|
+
end unless defined?(@id)
|
172
178
|
|
173
|
-
|
174
|
-
|
179
|
+
@id
|
180
|
+
end
|
175
181
|
|
176
|
-
|
177
|
-
|
178
|
-
|
179
|
-
|
180
|
-
|
181
|
-
|
182
|
-
|
182
|
+
def packet
|
183
|
+
msg = Resolv::DNS::Message.new
|
184
|
+
msg.id = id
|
185
|
+
msg.rd = 1
|
186
|
+
msg.add_question @hostname, Resolv::DNS::Resource::IN::A
|
187
|
+
msg
|
188
|
+
end
|
183
189
|
|
184
|
-
end
|
185
190
|
end
|
186
191
|
end
|
192
|
+
end
|
data/lib/em/spawnable.rb
CHANGED
@@ -55,7 +55,8 @@ module EventMachine
|
|
55
55
|
|
56
56
|
end
|
57
57
|
|
58
|
-
|
58
|
+
# @private
|
59
|
+
class YieldBlockFromSpawnedProcess
|
59
60
|
def initialize block, notify
|
60
61
|
@block = [block,notify]
|
61
62
|
end
|
@@ -71,11 +72,13 @@ module EventMachine
|
|
71
72
|
s
|
72
73
|
end
|
73
74
|
|
74
|
-
|
75
|
+
# @private
|
76
|
+
def self.yield &block
|
75
77
|
return YieldBlockFromSpawnedProcess.new( block, false )
|
76
78
|
end
|
77
79
|
|
78
|
-
|
80
|
+
# @private
|
81
|
+
def self.yield_and_notify &block
|
79
82
|
return YieldBlockFromSpawnedProcess.new( block, true )
|
80
83
|
end
|
81
84
|
end
|
data/lib/em/streamer.rb
CHANGED
@@ -1,30 +1,24 @@
|
|
1
|
-
#--
|
2
|
-
#
|
3
|
-
# Author:: Francis Cianfrocca (gmail: blackhedd)
|
4
|
-
# Homepage:: http://rubyeventmachine.com
|
5
|
-
# Date:: 16 Jul 2006
|
6
|
-
#
|
7
|
-
# See EventMachine and EventMachine::Connection for documentation and
|
8
|
-
# usage examples.
|
9
|
-
#
|
10
|
-
#----------------------------------------------------------------------------
|
11
|
-
#
|
12
|
-
# Copyright (C) 2006-07 by Francis Cianfrocca. All Rights Reserved.
|
13
|
-
# Gmail: blackhedd
|
14
|
-
#
|
15
|
-
# This program is free software; you can redistribute it and/or modify
|
16
|
-
# it under the terms of either: 1) the GNU General Public License
|
17
|
-
# as published by the Free Software Foundation; either version 2 of the
|
18
|
-
# License, or (at your option) any later version; or 2) Ruby's License.
|
19
|
-
#
|
20
|
-
# See the file COPYING for complete licensing information.
|
21
|
-
#
|
22
|
-
#---------------------------------------------------------------------------
|
23
|
-
#
|
24
|
-
#
|
25
|
-
|
26
|
-
|
27
1
|
module EventMachine
|
2
|
+
# Streams a file over a given connection. Streaming begins once the object is
|
3
|
+
# instantiated. Typically FileStreamer instances are not reused.
|
4
|
+
#
|
5
|
+
# Streaming uses buffering for files larger than 16K and uses so-called fast file reader (a C++ extension)
|
6
|
+
# if available (it is part of eventmachine gem itself).
|
7
|
+
#
|
8
|
+
# @example
|
9
|
+
#
|
10
|
+
# module FileSender
|
11
|
+
# def post_init
|
12
|
+
# streamer = EventMachine::FileStreamer.new(self, '/tmp/bigfile.tar')
|
13
|
+
# streamer.callback{
|
14
|
+
# # file was sent successfully
|
15
|
+
# close_connection_after_writing
|
16
|
+
# }
|
17
|
+
# end
|
18
|
+
# end
|
19
|
+
#
|
20
|
+
#
|
21
|
+
# @author Francis Cianfrocca
|
28
22
|
class FileStreamer
|
29
23
|
include Deferrable
|
30
24
|
|
@@ -35,19 +29,10 @@ module EventMachine
|
|
35
29
|
# Send 16k chunks at a time
|
36
30
|
ChunkSize = 16384
|
37
31
|
|
38
|
-
#
|
39
|
-
#
|
40
|
-
#
|
41
|
-
# module FileSender
|
42
|
-
# def post_init
|
43
|
-
# streamer = EventMachine::FileStreamer.new(self, '/tmp/bigfile.tar')
|
44
|
-
# streamer.callback{
|
45
|
-
# # file was sent successfully
|
46
|
-
# close_connection_after_writing
|
47
|
-
# }
|
48
|
-
# end
|
49
|
-
# end
|
32
|
+
# @param [EventMachine::Connection] connection
|
33
|
+
# @param [String] filename File path
|
50
34
|
#
|
35
|
+
# @option args [Boolean] :http_chunks (false) Use HTTP 1.1 style chunked-encoding semantics.
|
51
36
|
def initialize connection, filename, args = {}
|
52
37
|
@connection = connection
|
53
38
|
@http_chunks = args[:http_chunks]
|
@@ -64,7 +49,8 @@ module EventMachine
|
|
64
49
|
end
|
65
50
|
end
|
66
51
|
|
67
|
-
|
52
|
+
# @private
|
53
|
+
def stream_without_mapping filename
|
68
54
|
if @http_chunks
|
69
55
|
@connection.send_data "#{@size.to_s(16)}\r\n"
|
70
56
|
@connection.send_file_data filename
|
@@ -76,7 +62,8 @@ module EventMachine
|
|
76
62
|
end
|
77
63
|
private :stream_without_mapping
|
78
64
|
|
79
|
-
|
65
|
+
# @private
|
66
|
+
def stream_with_mapping filename
|
80
67
|
ensure_mapping_extension_is_present
|
81
68
|
|
82
69
|
@position = 0
|
@@ -86,6 +73,7 @@ module EventMachine
|
|
86
73
|
private :stream_with_mapping
|
87
74
|
|
88
75
|
# Used internally to stream one chunk at a time over multiple reactor ticks
|
76
|
+
# @private
|
89
77
|
def stream_one_chunk
|
90
78
|
loop {
|
91
79
|
if @position < @size
|
@@ -111,7 +99,7 @@ module EventMachine
|
|
111
99
|
}
|
112
100
|
end
|
113
101
|
|
114
|
-
|
102
|
+
#
|
115
103
|
# We use an outboard extension class to get memory-mapped files.
|
116
104
|
# It's outboard to avoid polluting the core distro, but that means
|
117
105
|
# there's a "hidden" dependency on it. The first time we get here in
|
@@ -120,11 +108,11 @@ module EventMachine
|
|
120
108
|
# mapped files will work fine without it. This is a somewhat difficult
|
121
109
|
# compromise between usability and proper modularization.
|
122
110
|
#
|
123
|
-
|
111
|
+
# @private
|
112
|
+
def ensure_mapping_extension_is_present
|
124
113
|
@@fastfilereader ||= (require 'fastfilereaderext')
|
125
114
|
end
|
126
115
|
private :ensure_mapping_extension_is_present
|
127
116
|
|
128
|
-
end
|
129
|
-
end
|
130
|
-
|
117
|
+
end # FileStreamer
|
118
|
+
end # EventMachine
|
@@ -0,0 +1,90 @@
|
|
1
|
+
module EventMachine
|
2
|
+
# = EventMachine::ThreadedResource
|
3
|
+
#
|
4
|
+
# A threaded resource is a "quick and dirty" wrapper around the concept of
|
5
|
+
# wiring up synchronous code into a standard EM::Pool. This is useful to keep
|
6
|
+
# interfaces coherent and provide a simple approach at "making an interface
|
7
|
+
# async-ish".
|
8
|
+
#
|
9
|
+
# General usage is to wrap libraries that do not support EventMachine, or to
|
10
|
+
# have a specific number of dedicated high-cpu worker resources.
|
11
|
+
#
|
12
|
+
# == Basic Usage example
|
13
|
+
#
|
14
|
+
# This example requires the cassandra gem. The cassandra gem contains an
|
15
|
+
# EventMachine interface, but it's sadly Fiber based and thus only works on
|
16
|
+
# 1.9. It also requires (potentially) complex stack switching logic to reach
|
17
|
+
# completion of nested operations. By contrast this approach provides a block
|
18
|
+
# in which normal synchronous code can occur, but makes no attempt to wire the
|
19
|
+
# IO into EventMachines C++ IO implementations, instead relying on the reactor
|
20
|
+
# pattern in rb_thread_select.
|
21
|
+
#
|
22
|
+
# cassandra_dispatcher = ThreadedResource.new do
|
23
|
+
# Cassandra.new('allthethings', '127.0.0.1:9160')
|
24
|
+
# end
|
25
|
+
#
|
26
|
+
# pool = EM::Pool.new
|
27
|
+
#
|
28
|
+
# pool.add cassandra_dispatcher
|
29
|
+
#
|
30
|
+
# # If we don't care about the result:
|
31
|
+
# pool.perform do |dispatcher|
|
32
|
+
# # The following blcok executes inside a dedicated thread, and should not
|
33
|
+
# # access EventMachine things:
|
34
|
+
# dispatcher.dispatch do |cassandra|
|
35
|
+
# cassandra.insert(:Things, '10', 'stuff' => 'things')
|
36
|
+
# end
|
37
|
+
# end
|
38
|
+
#
|
39
|
+
# # Example where we care about the result:
|
40
|
+
# pool.perform do |dispatcher|
|
41
|
+
# # The dispatch block is executed in the resources thread.
|
42
|
+
# completion = dispatcher.dispatch do |cassandra|
|
43
|
+
# cassandra.get(:Things, '10', 'stuff')
|
44
|
+
# end
|
45
|
+
#
|
46
|
+
# # This block will be yielded on the EM thread:
|
47
|
+
# completion.callback do |result|
|
48
|
+
# EM.do_something_with(result)
|
49
|
+
# end
|
50
|
+
#
|
51
|
+
# completion
|
52
|
+
# end
|
53
|
+
class ThreadedResource
|
54
|
+
|
55
|
+
# The block should return the resource that will be yielded in a dispatch.
|
56
|
+
def initialize
|
57
|
+
@resource = yield
|
58
|
+
|
59
|
+
@running = true
|
60
|
+
@queue = ::Queue.new
|
61
|
+
@thread = Thread.new do
|
62
|
+
@queue.pop.call while @running
|
63
|
+
end
|
64
|
+
end
|
65
|
+
|
66
|
+
# Called on the EM thread, generally in a perform block to return a
|
67
|
+
# completion for the work.
|
68
|
+
def dispatch
|
69
|
+
completion = EM::Completion.new
|
70
|
+
@queue << lambda do
|
71
|
+
begin
|
72
|
+
result = yield @resource
|
73
|
+
completion.succeed result
|
74
|
+
rescue Exception => e
|
75
|
+
completion.fail e
|
76
|
+
end
|
77
|
+
end
|
78
|
+
completion
|
79
|
+
end
|
80
|
+
|
81
|
+
# Kill the internal thread. should only be used to cleanup - generally
|
82
|
+
# only required for tests.
|
83
|
+
def shutdown
|
84
|
+
@running = false
|
85
|
+
@queue << lambda {}
|
86
|
+
@thread.join
|
87
|
+
end
|
88
|
+
|
89
|
+
end
|
90
|
+
end
|