eventmachine 0.12.10-x86-mswin32-60 → 1.0.0.beta.2-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 +2 -0
- data/Gemfile +1 -0
- data/README +80 -81
- data/Rakefile +7 -370
- data/docs/COPYING +60 -60
- data/docs/ChangeLog +211 -211
- data/docs/DEFERRABLES +246 -133
- data/docs/EPOLL +141 -141
- data/docs/GNU +281 -281
- data/docs/INSTALL +13 -13
- data/docs/KEYBOARD +42 -38
- data/docs/LEGAL +25 -25
- data/docs/LIGHTWEIGHT_CONCURRENCY +130 -70
- data/docs/PURE_RUBY +75 -75
- data/docs/RELEASE_NOTES +94 -94
- data/docs/SMTP +4 -2
- data/docs/SPAWNED_PROCESSES +148 -89
- data/docs/TODO +8 -8
- data/eventmachine.gemspec +19 -26
- data/examples/ex_channel.rb +42 -42
- data/examples/ex_queue.rb +2 -2
- data/examples/ex_tick_loop_array.rb +15 -0
- data/examples/ex_tick_loop_counter.rb +32 -0
- data/examples/helper.rb +1 -1
- data/ext/binder.cpp +0 -1
- data/ext/cmain.cpp +36 -25
- data/ext/ed.cpp +104 -113
- data/ext/ed.h +24 -30
- data/ext/em.cpp +349 -283
- data/ext/em.h +25 -29
- data/ext/eventmachine.h +5 -4
- data/ext/extconf.rb +58 -49
- data/ext/fastfilereader/extconf.rb +5 -3
- data/ext/fastfilereader/mapper.cpp +214 -214
- data/ext/fastfilereader/mapper.h +59 -59
- data/ext/fastfilereader/rubymain.cpp +127 -127
- data/ext/kb.cpp +1 -3
- data/ext/page.cpp +107 -107
- data/ext/page.h +51 -51
- data/ext/pipe.cpp +9 -11
- data/ext/project.h +12 -8
- data/ext/rubymain.cpp +138 -104
- data/java/.classpath +8 -8
- data/java/.project +17 -17
- data/java/src/com/rubyeventmachine/EmReactor.java +1 -0
- data/java/src/com/rubyeventmachine/EmReactorException.java +40 -40
- data/lib/em/buftok.rb +138 -138
- data/lib/em/callback.rb +25 -25
- data/lib/em/channel.rb +1 -1
- data/lib/em/connection.rb +6 -1
- data/lib/em/deferrable.rb +16 -2
- data/lib/em/file_watch.rb +53 -53
- data/lib/em/future.rb +61 -61
- data/lib/em/iterator.rb +270 -0
- data/lib/em/messages.rb +66 -66
- data/lib/em/process_watch.rb +43 -43
- data/lib/em/protocols.rb +1 -1
- data/lib/em/protocols/header_and_content.rb +138 -138
- data/lib/em/protocols/httpclient.rb +267 -262
- data/lib/em/protocols/line_protocol.rb +28 -0
- data/lib/em/protocols/memcache.rb +322 -322
- data/lib/em/protocols/postgres3.rb +247 -247
- data/lib/em/protocols/saslauth.rb +175 -175
- data/lib/em/protocols/smtpserver.rb +640 -547
- data/lib/em/protocols/stomp.rb +200 -200
- data/lib/em/protocols/tcptest.rb +52 -52
- data/lib/{pr_eventmachine.rb → em/pure_ruby.rb} +1013 -1022
- data/lib/em/queue.rb +1 -0
- data/lib/em/spawnable.rb +85 -85
- data/lib/em/streamer.rb +130 -130
- data/lib/em/tick_loop.rb +85 -0
- data/lib/em/timers.rb +2 -1
- data/lib/em/version.rb +1 -1
- data/lib/eventmachine.rb +40 -84
- data/lib/jeventmachine.rb +2 -1
- data/lib/rubyeventmachine.rb +2 -0
- data/setup.rb +1585 -1585
- data/tasks/doc.rake +30 -0
- data/tasks/package.rake +85 -0
- data/tasks/test.rake +6 -0
- data/tests/client.crt +31 -31
- data/tests/client.key +51 -51
- data/tests/test_attach.rb +13 -3
- data/tests/test_basic.rb +60 -95
- data/tests/test_channel.rb +3 -2
- data/tests/test_defer.rb +49 -47
- data/tests/test_deferrable.rb +35 -0
- data/tests/test_error_handler.rb +35 -35
- data/tests/test_errors.rb +82 -82
- data/tests/test_exc.rb +55 -55
- data/tests/test_file_watch.rb +49 -49
- data/tests/test_futures.rb +198 -198
- data/tests/test_handler_check.rb +36 -36
- data/tests/test_hc.rb +190 -218
- data/tests/test_httpclient.rb +227 -218
- data/tests/test_httpclient2.rb +3 -2
- data/tests/test_inactivity_timeout.rb +3 -3
- data/tests/test_kb.rb +60 -60
- data/tests/test_ltp.rb +13 -5
- data/tests/test_ltp2.rb +317 -317
- data/tests/test_next_tick.rb +1 -1
- data/tests/test_object_protocol.rb +36 -36
- data/tests/test_pending_connect_timeout.rb +2 -2
- data/tests/test_process_watch.rb +50 -48
- data/tests/test_proxy_connection.rb +52 -0
- data/tests/test_pure.rb +134 -125
- data/tests/test_queue.rb +44 -44
- data/tests/test_running.rb +42 -42
- data/tests/test_sasl.rb +72 -72
- data/tests/test_send_file.rb +251 -242
- data/tests/test_servers.rb +76 -76
- data/tests/test_smtpclient.rb +83 -83
- data/tests/test_smtpserver.rb +85 -85
- data/tests/test_spawn.rb +322 -322
- data/tests/test_ssl_methods.rb +49 -49
- data/tests/test_ssl_verify.rb +82 -82
- data/tests/test_tick_loop.rb +59 -0
- data/tests/test_timers.rb +13 -15
- data/tests/test_ud.rb +36 -36
- data/tests/testem.rb +31 -31
- metadata +66 -51
- data/ext/cplusplus.cpp +0 -202
- data/ext/emwin.cpp +0 -300
- data/ext/emwin.h +0 -94
- data/ext/epoll.cpp +0 -26
- data/ext/epoll.h +0 -25
- data/ext/eventmachine_cpp.h +0 -96
- data/ext/files.cpp +0 -94
- data/ext/files.h +0 -65
- data/ext/sigs.cpp +0 -89
- data/ext/sigs.h +0 -32
- data/java/src/com/rubyeventmachine/application/Application.java +0 -194
- data/java/src/com/rubyeventmachine/application/Connection.java +0 -74
- data/java/src/com/rubyeventmachine/application/ConnectionFactory.java +0 -37
- data/java/src/com/rubyeventmachine/application/DefaultConnectionFactory.java +0 -46
- data/java/src/com/rubyeventmachine/application/PeriodicTimer.java +0 -38
- data/java/src/com/rubyeventmachine/application/Timer.java +0 -54
- data/java/src/com/rubyeventmachine/tests/ApplicationTest.java +0 -109
- data/java/src/com/rubyeventmachine/tests/ConnectTest.java +0 -148
- data/java/src/com/rubyeventmachine/tests/EMTest.java +0 -80
- data/java/src/com/rubyeventmachine/tests/TestDatagrams.java +0 -53
- data/java/src/com/rubyeventmachine/tests/TestServers.java +0 -75
- data/java/src/com/rubyeventmachine/tests/TestTimers.java +0 -90
- data/lib/evma.rb +0 -32
- data/lib/evma/callback.rb +0 -32
- data/lib/evma/container.rb +0 -75
- data/lib/evma/factory.rb +0 -77
- data/lib/evma/protocol.rb +0 -87
- data/lib/evma/reactor.rb +0 -48
- data/web/whatis +0 -7
data/lib/em/queue.rb
CHANGED
data/lib/em/spawnable.rb
CHANGED
@@ -1,85 +1,85 @@
|
|
1
|
-
#--
|
2
|
-
#
|
3
|
-
# Author:: Francis Cianfrocca (gmail: blackhedd)
|
4
|
-
# Homepage:: http://rubyeventmachine.com
|
5
|
-
# Date:: 25 Aug 2007
|
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
|
-
module EventMachine
|
27
|
-
# Support for Erlang-style processes.
|
28
|
-
#
|
29
|
-
class SpawnedProcess
|
30
|
-
# Send a message to the spawned process
|
31
|
-
def notify *x
|
32
|
-
me = self
|
33
|
-
EM.next_tick {
|
34
|
-
# A notification executes in the context of this
|
35
|
-
# SpawnedProcess object. That makes self and notify
|
36
|
-
# work as one would expect.
|
37
|
-
#
|
38
|
-
y = me.call(*x)
|
39
|
-
if y and y.respond_to?(:pull_out_yield_block)
|
40
|
-
a,b = y.pull_out_yield_block
|
41
|
-
set_receiver a
|
42
|
-
self.notify if b
|
43
|
-
end
|
44
|
-
}
|
45
|
-
end
|
46
|
-
alias_method :resume, :notify
|
47
|
-
alias_method :run, :notify # for formulations like (EM.spawn {xxx}).run
|
48
|
-
#attr_accessor :receiver
|
49
|
-
|
50
|
-
#--
|
51
|
-
# I know I'm missing something stupid, but the inside of class << s
|
52
|
-
# can't see locally-bound values. It can see globals, though.
|
53
|
-
def set_receiver blk
|
54
|
-
$em______tmpglobal = blk
|
55
|
-
class << self
|
56
|
-
define_method :call, $em______tmpglobal.dup
|
57
|
-
end
|
58
|
-
end
|
59
|
-
|
60
|
-
end
|
61
|
-
|
62
|
-
class YieldBlockFromSpawnedProcess # :nodoc:
|
63
|
-
def initialize block, notify
|
64
|
-
@block = [block,notify]
|
65
|
-
end
|
66
|
-
def pull_out_yield_block
|
67
|
-
@block
|
68
|
-
end
|
69
|
-
end
|
70
|
-
|
71
|
-
# Spawn an erlang-style process
|
72
|
-
def self.spawn &block
|
73
|
-
s = SpawnedProcess.new
|
74
|
-
s.set_receiver block
|
75
|
-
s
|
76
|
-
end
|
77
|
-
|
78
|
-
def self.yield &block # :nodoc:
|
79
|
-
return YieldBlockFromSpawnedProcess.new( block, false )
|
80
|
-
end
|
81
|
-
|
82
|
-
def self.yield_and_notify &block # :nodoc:
|
83
|
-
return YieldBlockFromSpawnedProcess.new( block, true )
|
84
|
-
end
|
85
|
-
end
|
1
|
+
#--
|
2
|
+
#
|
3
|
+
# Author:: Francis Cianfrocca (gmail: blackhedd)
|
4
|
+
# Homepage:: http://rubyeventmachine.com
|
5
|
+
# Date:: 25 Aug 2007
|
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
|
+
module EventMachine
|
27
|
+
# Support for Erlang-style processes.
|
28
|
+
#
|
29
|
+
class SpawnedProcess
|
30
|
+
# Send a message to the spawned process
|
31
|
+
def notify *x
|
32
|
+
me = self
|
33
|
+
EM.next_tick {
|
34
|
+
# A notification executes in the context of this
|
35
|
+
# SpawnedProcess object. That makes self and notify
|
36
|
+
# work as one would expect.
|
37
|
+
#
|
38
|
+
y = me.call(*x)
|
39
|
+
if y and y.respond_to?(:pull_out_yield_block)
|
40
|
+
a,b = y.pull_out_yield_block
|
41
|
+
set_receiver a
|
42
|
+
self.notify if b
|
43
|
+
end
|
44
|
+
}
|
45
|
+
end
|
46
|
+
alias_method :resume, :notify
|
47
|
+
alias_method :run, :notify # for formulations like (EM.spawn {xxx}).run
|
48
|
+
#attr_accessor :receiver
|
49
|
+
|
50
|
+
#--
|
51
|
+
# I know I'm missing something stupid, but the inside of class << s
|
52
|
+
# can't see locally-bound values. It can see globals, though.
|
53
|
+
def set_receiver blk
|
54
|
+
$em______tmpglobal = blk
|
55
|
+
class << self
|
56
|
+
define_method :call, $em______tmpglobal.dup
|
57
|
+
end
|
58
|
+
end
|
59
|
+
|
60
|
+
end
|
61
|
+
|
62
|
+
class YieldBlockFromSpawnedProcess # :nodoc:
|
63
|
+
def initialize block, notify
|
64
|
+
@block = [block,notify]
|
65
|
+
end
|
66
|
+
def pull_out_yield_block
|
67
|
+
@block
|
68
|
+
end
|
69
|
+
end
|
70
|
+
|
71
|
+
# Spawn an erlang-style process
|
72
|
+
def self.spawn &block
|
73
|
+
s = SpawnedProcess.new
|
74
|
+
s.set_receiver block
|
75
|
+
s
|
76
|
+
end
|
77
|
+
|
78
|
+
def self.yield &block # :nodoc:
|
79
|
+
return YieldBlockFromSpawnedProcess.new( block, false )
|
80
|
+
end
|
81
|
+
|
82
|
+
def self.yield_and_notify &block # :nodoc:
|
83
|
+
return YieldBlockFromSpawnedProcess.new( block, true )
|
84
|
+
end
|
85
|
+
end
|
data/lib/em/streamer.rb
CHANGED
@@ -1,130 +1,130 @@
|
|
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
|
-
module EventMachine
|
28
|
-
class FileStreamer
|
29
|
-
include Deferrable
|
30
|
-
|
31
|
-
# Use mapped streamer for files bigger than 16k
|
32
|
-
MappingThreshold = 16384
|
33
|
-
# Wait until next tick to send more data when 50k is still in the outgoing buffer
|
34
|
-
BackpressureLevel = 50000
|
35
|
-
# Send 16k chunks at a time
|
36
|
-
ChunkSize = 16384
|
37
|
-
|
38
|
-
# Stream a file over a given connection. An optional :http_chunks => true argument will
|
39
|
-
# use HTTP 1.1 style chunked-encoding semantics.
|
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
|
50
|
-
#
|
51
|
-
def initialize connection, filename, args = {}
|
52
|
-
@connection = connection
|
53
|
-
@http_chunks = args[:http_chunks]
|
54
|
-
|
55
|
-
if File.exist?(filename)
|
56
|
-
@size = File.size
|
57
|
-
if @size <= MappingThreshold
|
58
|
-
stream_without_mapping filename
|
59
|
-
else
|
60
|
-
stream_with_mapping filename
|
61
|
-
end
|
62
|
-
else
|
63
|
-
fail "file not found"
|
64
|
-
end
|
65
|
-
end
|
66
|
-
|
67
|
-
def stream_without_mapping filename # :nodoc:
|
68
|
-
if @http_chunks
|
69
|
-
@connection.send_data "#{@size.to_s(16)}\r\n"
|
70
|
-
@connection.send_file_data filename
|
71
|
-
@connection.send_data "\r\n0\r\n\r\n"
|
72
|
-
else
|
73
|
-
@connection.send_file_data filename
|
74
|
-
end
|
75
|
-
succeed
|
76
|
-
end
|
77
|
-
private :stream_without_mapping
|
78
|
-
|
79
|
-
def stream_with_mapping filename # :nodoc:
|
80
|
-
ensure_mapping_extension_is_present
|
81
|
-
|
82
|
-
@position = 0
|
83
|
-
@mapping = EventMachine::FastFileReader::Mapper.new filename
|
84
|
-
stream_one_chunk
|
85
|
-
end
|
86
|
-
private :stream_with_mapping
|
87
|
-
|
88
|
-
# Used internally to stream one chunk at a time over multiple reactor ticks
|
89
|
-
def stream_one_chunk
|
90
|
-
loop {
|
91
|
-
if @position < @size
|
92
|
-
if @connection.get_outbound_data_size > BackpressureLevel
|
93
|
-
EventMachine::next_tick {stream_one_chunk}
|
94
|
-
break
|
95
|
-
else
|
96
|
-
len = @size - @position
|
97
|
-
len = ChunkSize if (len > ChunkSize)
|
98
|
-
|
99
|
-
@connection.send_data( "#{len.to_s(16)}\r\n" ) if @http_chunks
|
100
|
-
@connection.send_data( @mapping.get_chunk( @position, len ))
|
101
|
-
@connection.send_data("\r\n") if @http_chunks
|
102
|
-
|
103
|
-
@position += len
|
104
|
-
end
|
105
|
-
else
|
106
|
-
@connection.send_data "0\r\n\r\n" if @http_chunks
|
107
|
-
@mapping.close
|
108
|
-
succeed
|
109
|
-
break
|
110
|
-
end
|
111
|
-
}
|
112
|
-
end
|
113
|
-
|
114
|
-
#--
|
115
|
-
# We use an outboard extension class to get memory-mapped files.
|
116
|
-
# It's outboard to avoid polluting the core distro, but that means
|
117
|
-
# there's a "hidden" dependency on it. The first time we get here in
|
118
|
-
# any run, try to load up the dependency extension. User code will see
|
119
|
-
# a LoadError if it's not available, but code that doesn't require
|
120
|
-
# mapped files will work fine without it. This is a somewhat difficult
|
121
|
-
# compromise between usability and proper modularization.
|
122
|
-
#
|
123
|
-
def ensure_mapping_extension_is_present # :nodoc:
|
124
|
-
@@fastfilereader ||= (require 'fastfilereaderext')
|
125
|
-
end
|
126
|
-
private :ensure_mapping_extension_is_present
|
127
|
-
|
128
|
-
end
|
129
|
-
end
|
130
|
-
|
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
|
+
module EventMachine
|
28
|
+
class FileStreamer
|
29
|
+
include Deferrable
|
30
|
+
|
31
|
+
# Use mapped streamer for files bigger than 16k
|
32
|
+
MappingThreshold = 16384
|
33
|
+
# Wait until next tick to send more data when 50k is still in the outgoing buffer
|
34
|
+
BackpressureLevel = 50000
|
35
|
+
# Send 16k chunks at a time
|
36
|
+
ChunkSize = 16384
|
37
|
+
|
38
|
+
# Stream a file over a given connection. An optional :http_chunks => true argument will
|
39
|
+
# use HTTP 1.1 style chunked-encoding semantics.
|
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
|
50
|
+
#
|
51
|
+
def initialize connection, filename, args = {}
|
52
|
+
@connection = connection
|
53
|
+
@http_chunks = args[:http_chunks]
|
54
|
+
|
55
|
+
if File.exist?(filename)
|
56
|
+
@size = File.size(filename)
|
57
|
+
if @size <= MappingThreshold
|
58
|
+
stream_without_mapping filename
|
59
|
+
else
|
60
|
+
stream_with_mapping filename
|
61
|
+
end
|
62
|
+
else
|
63
|
+
fail "file not found"
|
64
|
+
end
|
65
|
+
end
|
66
|
+
|
67
|
+
def stream_without_mapping filename # :nodoc:
|
68
|
+
if @http_chunks
|
69
|
+
@connection.send_data "#{@size.to_s(16)}\r\n"
|
70
|
+
@connection.send_file_data filename
|
71
|
+
@connection.send_data "\r\n0\r\n\r\n"
|
72
|
+
else
|
73
|
+
@connection.send_file_data filename
|
74
|
+
end
|
75
|
+
succeed
|
76
|
+
end
|
77
|
+
private :stream_without_mapping
|
78
|
+
|
79
|
+
def stream_with_mapping filename # :nodoc:
|
80
|
+
ensure_mapping_extension_is_present
|
81
|
+
|
82
|
+
@position = 0
|
83
|
+
@mapping = EventMachine::FastFileReader::Mapper.new filename
|
84
|
+
stream_one_chunk
|
85
|
+
end
|
86
|
+
private :stream_with_mapping
|
87
|
+
|
88
|
+
# Used internally to stream one chunk at a time over multiple reactor ticks
|
89
|
+
def stream_one_chunk
|
90
|
+
loop {
|
91
|
+
if @position < @size
|
92
|
+
if @connection.get_outbound_data_size > BackpressureLevel
|
93
|
+
EventMachine::next_tick {stream_one_chunk}
|
94
|
+
break
|
95
|
+
else
|
96
|
+
len = @size - @position
|
97
|
+
len = ChunkSize if (len > ChunkSize)
|
98
|
+
|
99
|
+
@connection.send_data( "#{len.to_s(16)}\r\n" ) if @http_chunks
|
100
|
+
@connection.send_data( @mapping.get_chunk( @position, len ))
|
101
|
+
@connection.send_data("\r\n") if @http_chunks
|
102
|
+
|
103
|
+
@position += len
|
104
|
+
end
|
105
|
+
else
|
106
|
+
@connection.send_data "0\r\n\r\n" if @http_chunks
|
107
|
+
@mapping.close
|
108
|
+
succeed
|
109
|
+
break
|
110
|
+
end
|
111
|
+
}
|
112
|
+
end
|
113
|
+
|
114
|
+
#--
|
115
|
+
# We use an outboard extension class to get memory-mapped files.
|
116
|
+
# It's outboard to avoid polluting the core distro, but that means
|
117
|
+
# there's a "hidden" dependency on it. The first time we get here in
|
118
|
+
# any run, try to load up the dependency extension. User code will see
|
119
|
+
# a LoadError if it's not available, but code that doesn't require
|
120
|
+
# mapped files will work fine without it. This is a somewhat difficult
|
121
|
+
# compromise between usability and proper modularization.
|
122
|
+
#
|
123
|
+
def ensure_mapping_extension_is_present # :nodoc:
|
124
|
+
@@fastfilereader ||= (require 'fastfilereaderext')
|
125
|
+
end
|
126
|
+
private :ensure_mapping_extension_is_present
|
127
|
+
|
128
|
+
end
|
129
|
+
end
|
130
|
+
|
data/lib/em/tick_loop.rb
ADDED
@@ -0,0 +1,85 @@
|
|
1
|
+
module EventMachine
|
2
|
+
# Creates and immediately starts an EventMachine::TickLoop
|
3
|
+
def self.tick_loop(*a, &b)
|
4
|
+
TickLoop.new(*a, &b).start
|
5
|
+
end
|
6
|
+
|
7
|
+
# A TickLoop is useful when one needs to distribute amounts of work
|
8
|
+
# throughout ticks in order to maintain response times. It is also useful for
|
9
|
+
# simple repeated checks and metrics.
|
10
|
+
#
|
11
|
+
# # Here we run through an array one item per tick until it is empty,
|
12
|
+
# # printing each element.
|
13
|
+
# # When the array is empty, we return :stop from the callback, and the
|
14
|
+
# # loop will terminate.
|
15
|
+
# # When the loop terminates, the on_stop callbacks will be called.
|
16
|
+
# EM.run do
|
17
|
+
# array = (1..100).to_a
|
18
|
+
#
|
19
|
+
# tickloop = EM.tick_loop do
|
20
|
+
# if array.empty?
|
21
|
+
# :stop
|
22
|
+
# else
|
23
|
+
# puts array.shift
|
24
|
+
# end
|
25
|
+
# end
|
26
|
+
#
|
27
|
+
# tickloop.on_stop { EM.stop }
|
28
|
+
# end
|
29
|
+
#
|
30
|
+
class TickLoop
|
31
|
+
|
32
|
+
# Arguments: A callback (EM::Callback) to call each tick. If the call
|
33
|
+
# returns +:stop+ then the loop will be stopped. Any other value is
|
34
|
+
# ignored.
|
35
|
+
def initialize(*a, &b)
|
36
|
+
@work = EM::Callback(*a, &b)
|
37
|
+
@stops = []
|
38
|
+
@stopped = true
|
39
|
+
end
|
40
|
+
|
41
|
+
# Arguments: A callback (EM::Callback) to call once on the next stop (or
|
42
|
+
# immediately if already stopped).
|
43
|
+
def on_stop(*a, &b)
|
44
|
+
if @stopped
|
45
|
+
EM::Callback(*a, &b).call
|
46
|
+
else
|
47
|
+
@stops << EM::Callback(*a, &b)
|
48
|
+
end
|
49
|
+
end
|
50
|
+
|
51
|
+
# Stop the tick loop immediately, and call it's on_stop callbacks.
|
52
|
+
def stop
|
53
|
+
@stopped = true
|
54
|
+
until @stops.empty?
|
55
|
+
@stops.shift.call
|
56
|
+
end
|
57
|
+
end
|
58
|
+
|
59
|
+
# Query if the loop is stopped.
|
60
|
+
def stopped?
|
61
|
+
@stopped
|
62
|
+
end
|
63
|
+
|
64
|
+
# Start the tick loop, will raise argument error if the loop is already
|
65
|
+
# running.
|
66
|
+
def start
|
67
|
+
raise ArgumentError, "double start" unless @stopped
|
68
|
+
@stopped = false
|
69
|
+
schedule
|
70
|
+
end
|
71
|
+
|
72
|
+
private
|
73
|
+
def schedule
|
74
|
+
EM.next_tick do
|
75
|
+
next if @stopped
|
76
|
+
if @work.call == :stop
|
77
|
+
stop
|
78
|
+
else
|
79
|
+
schedule
|
80
|
+
end
|
81
|
+
end
|
82
|
+
self
|
83
|
+
end
|
84
|
+
end
|
85
|
+
end
|