eventmachine 1.0.0.beta.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.
- data/.gitignore +16 -0
- data/Gemfile +1 -0
- data/README +81 -0
- data/Rakefile +11 -0
- data/docs/COPYING +60 -0
- data/docs/ChangeLog +211 -0
- data/docs/DEFERRABLES +246 -0
- data/docs/EPOLL +141 -0
- data/docs/GNU +281 -0
- data/docs/INSTALL +13 -0
- data/docs/KEYBOARD +42 -0
- data/docs/LEGAL +25 -0
- data/docs/LIGHTWEIGHT_CONCURRENCY +130 -0
- data/docs/PURE_RUBY +75 -0
- data/docs/RELEASE_NOTES +94 -0
- data/docs/SMTP +4 -0
- data/docs/SPAWNED_PROCESSES +148 -0
- data/docs/TODO +8 -0
- data/eventmachine.gemspec +33 -0
- data/examples/ex_channel.rb +43 -0
- data/examples/ex_queue.rb +2 -0
- data/examples/ex_tick_loop_array.rb +15 -0
- data/examples/ex_tick_loop_counter.rb +32 -0
- data/examples/helper.rb +2 -0
- data/ext/binder.cpp +124 -0
- data/ext/binder.h +46 -0
- data/ext/cmain.cpp +838 -0
- data/ext/ed.cpp +1884 -0
- data/ext/ed.h +418 -0
- data/ext/em.cpp +2348 -0
- data/ext/em.h +228 -0
- data/ext/eventmachine.h +123 -0
- data/ext/extconf.rb +157 -0
- data/ext/fastfilereader/extconf.rb +85 -0
- data/ext/fastfilereader/mapper.cpp +214 -0
- data/ext/fastfilereader/mapper.h +59 -0
- data/ext/fastfilereader/rubymain.cpp +127 -0
- data/ext/kb.cpp +79 -0
- data/ext/page.cpp +107 -0
- data/ext/page.h +51 -0
- data/ext/pipe.cpp +347 -0
- data/ext/project.h +155 -0
- data/ext/rubymain.cpp +1200 -0
- data/ext/ssl.cpp +460 -0
- data/ext/ssl.h +94 -0
- data/java/.classpath +8 -0
- data/java/.project +17 -0
- data/java/src/com/rubyeventmachine/EmReactor.java +571 -0
- data/java/src/com/rubyeventmachine/EmReactorException.java +40 -0
- data/java/src/com/rubyeventmachine/EventableChannel.java +69 -0
- data/java/src/com/rubyeventmachine/EventableDatagramChannel.java +189 -0
- data/java/src/com/rubyeventmachine/EventableSocketChannel.java +364 -0
- data/lib/em/buftok.rb +138 -0
- data/lib/em/callback.rb +26 -0
- data/lib/em/channel.rb +57 -0
- data/lib/em/connection.rb +569 -0
- data/lib/em/deferrable.rb +206 -0
- data/lib/em/file_watch.rb +54 -0
- data/lib/em/future.rb +61 -0
- data/lib/em/iterator.rb +270 -0
- data/lib/em/messages.rb +66 -0
- data/lib/em/process_watch.rb +44 -0
- data/lib/em/processes.rb +119 -0
- data/lib/em/protocols.rb +36 -0
- data/lib/em/protocols/header_and_content.rb +138 -0
- data/lib/em/protocols/httpclient.rb +268 -0
- data/lib/em/protocols/httpclient2.rb +590 -0
- data/lib/em/protocols/line_and_text.rb +125 -0
- data/lib/em/protocols/line_protocol.rb +28 -0
- data/lib/em/protocols/linetext2.rb +161 -0
- data/lib/em/protocols/memcache.rb +323 -0
- data/lib/em/protocols/object_protocol.rb +45 -0
- data/lib/em/protocols/postgres3.rb +247 -0
- data/lib/em/protocols/saslauth.rb +175 -0
- data/lib/em/protocols/smtpclient.rb +357 -0
- data/lib/em/protocols/smtpserver.rb +640 -0
- data/lib/em/protocols/socks4.rb +66 -0
- data/lib/em/protocols/stomp.rb +200 -0
- data/lib/em/protocols/tcptest.rb +53 -0
- data/lib/em/pure_ruby.rb +1013 -0
- data/lib/em/queue.rb +62 -0
- data/lib/em/spawnable.rb +85 -0
- data/lib/em/streamer.rb +130 -0
- data/lib/em/tick_loop.rb +85 -0
- data/lib/em/timers.rb +57 -0
- data/lib/em/version.rb +3 -0
- data/lib/eventmachine.rb +1548 -0
- data/lib/jeventmachine.rb +258 -0
- data/lib/rubyeventmachine.rb +2 -0
- data/setup.rb +1585 -0
- data/tasks/cpp.rake_example +77 -0
- data/tasks/doc.rake +30 -0
- data/tasks/package.rake +85 -0
- data/tasks/test.rake +6 -0
- data/tests/client.crt +31 -0
- data/tests/client.key +51 -0
- data/tests/test_attach.rb +136 -0
- data/tests/test_basic.rb +249 -0
- data/tests/test_channel.rb +64 -0
- data/tests/test_connection_count.rb +35 -0
- data/tests/test_defer.rb +49 -0
- data/tests/test_deferrable.rb +35 -0
- data/tests/test_epoll.rb +160 -0
- data/tests/test_error_handler.rb +35 -0
- data/tests/test_errors.rb +82 -0
- data/tests/test_exc.rb +55 -0
- data/tests/test_file_watch.rb +49 -0
- data/tests/test_futures.rb +198 -0
- data/tests/test_get_sock_opt.rb +30 -0
- data/tests/test_handler_check.rb +37 -0
- data/tests/test_hc.rb +190 -0
- data/tests/test_httpclient.rb +227 -0
- data/tests/test_httpclient2.rb +154 -0
- data/tests/test_inactivity_timeout.rb +50 -0
- data/tests/test_kb.rb +60 -0
- data/tests/test_ltp.rb +190 -0
- data/tests/test_ltp2.rb +317 -0
- data/tests/test_next_tick.rb +133 -0
- data/tests/test_object_protocol.rb +37 -0
- data/tests/test_pause.rb +70 -0
- data/tests/test_pending_connect_timeout.rb +48 -0
- data/tests/test_process_watch.rb +50 -0
- data/tests/test_processes.rb +128 -0
- data/tests/test_proxy_connection.rb +144 -0
- data/tests/test_pure.rb +134 -0
- data/tests/test_queue.rb +44 -0
- data/tests/test_running.rb +42 -0
- data/tests/test_sasl.rb +72 -0
- data/tests/test_send_file.rb +251 -0
- data/tests/test_servers.rb +76 -0
- data/tests/test_smtpclient.rb +83 -0
- data/tests/test_smtpserver.rb +85 -0
- data/tests/test_spawn.rb +322 -0
- data/tests/test_ssl_args.rb +79 -0
- data/tests/test_ssl_methods.rb +50 -0
- data/tests/test_ssl_verify.rb +82 -0
- data/tests/test_tick_loop.rb +59 -0
- data/tests/test_timers.rb +160 -0
- data/tests/test_ud.rb +36 -0
- data/tests/testem.rb +31 -0
- metadata +240 -0
@@ -0,0 +1,66 @@
|
|
1
|
+
module EventMachine
|
2
|
+
module Protocols
|
3
|
+
# Basic SOCKS v4 client implementation
|
4
|
+
#
|
5
|
+
# Use as you would any regular connection:
|
6
|
+
#
|
7
|
+
# class MyConn < EM::P::Socks4
|
8
|
+
# def post_init
|
9
|
+
# send_data("sup")
|
10
|
+
# end
|
11
|
+
#
|
12
|
+
# def receive_data(data)
|
13
|
+
# send_data("you said: #{data}")
|
14
|
+
# end
|
15
|
+
# end
|
16
|
+
#
|
17
|
+
# EM.connect socks_host, socks_port, MyConn, host, port
|
18
|
+
#
|
19
|
+
class Socks4 < Connection
|
20
|
+
def initialize(host, port)
|
21
|
+
@host = Socket.gethostbyname(host).last
|
22
|
+
@port = port
|
23
|
+
@socks_error_code = nil
|
24
|
+
@buffer = ''
|
25
|
+
setup_methods
|
26
|
+
end
|
27
|
+
|
28
|
+
def setup_methods
|
29
|
+
class << self
|
30
|
+
def post_init; socks_post_init; end
|
31
|
+
def receive_data(*a); socks_receive_data(*a); end
|
32
|
+
end
|
33
|
+
end
|
34
|
+
|
35
|
+
def restore_methods
|
36
|
+
class << self
|
37
|
+
remove_method :post_init
|
38
|
+
remove_method :receive_data
|
39
|
+
end
|
40
|
+
end
|
41
|
+
|
42
|
+
def socks_post_init
|
43
|
+
header = [4, 1, @port, @host, 0].flatten.pack("CCnA4C")
|
44
|
+
send_data(header)
|
45
|
+
end
|
46
|
+
|
47
|
+
def socks_receive_data(data)
|
48
|
+
@buffer << data
|
49
|
+
return if @buffer.size < 8
|
50
|
+
|
51
|
+
header_resp = @buffer.slice! 0, 8
|
52
|
+
_, r = header_resp.unpack("cc")
|
53
|
+
if r != 90
|
54
|
+
@socks_error_code = r
|
55
|
+
close_connection
|
56
|
+
return
|
57
|
+
end
|
58
|
+
|
59
|
+
restore_methods
|
60
|
+
|
61
|
+
post_init
|
62
|
+
receive_data(@buffer) unless @buffer.empty?
|
63
|
+
end
|
64
|
+
end
|
65
|
+
end
|
66
|
+
end
|
@@ -0,0 +1,200 @@
|
|
1
|
+
#--
|
2
|
+
#
|
3
|
+
# Author:: Francis Cianfrocca (gmail: blackhedd)
|
4
|
+
# Homepage:: http://rubyeventmachine.com
|
5
|
+
# Date:: 15 November 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
|
+
module Protocols
|
29
|
+
|
30
|
+
# Implements Stomp (http://docs.codehaus.org/display/STOMP/Protocol).
|
31
|
+
#
|
32
|
+
# == Usage example
|
33
|
+
#
|
34
|
+
# module StompClient
|
35
|
+
# include EM::Protocols::Stomp
|
36
|
+
#
|
37
|
+
# def connection_completed
|
38
|
+
# connect :login => 'guest', :passcode => 'guest'
|
39
|
+
# end
|
40
|
+
#
|
41
|
+
# def receive_msg msg
|
42
|
+
# if msg.command == "CONNECTED"
|
43
|
+
# subscribe '/some/topic'
|
44
|
+
# else
|
45
|
+
# p ['got a message', msg]
|
46
|
+
# puts msg.body
|
47
|
+
# end
|
48
|
+
# end
|
49
|
+
# end
|
50
|
+
#
|
51
|
+
# EM.run{
|
52
|
+
# EM.connect 'localhost', 61613, StompClient
|
53
|
+
# }
|
54
|
+
#
|
55
|
+
module Stomp
|
56
|
+
include LineText2
|
57
|
+
|
58
|
+
class Message
|
59
|
+
# The command associated with the message, usually 'CONNECTED' or 'MESSAGE'
|
60
|
+
attr_accessor :command
|
61
|
+
# Hash containing headers such as destination and message-id
|
62
|
+
attr_accessor :header
|
63
|
+
alias :headers :header
|
64
|
+
# Body of the message
|
65
|
+
attr_accessor :body
|
66
|
+
|
67
|
+
def initialize # :nodoc:
|
68
|
+
@header = {}
|
69
|
+
@state = :precommand
|
70
|
+
@content_length = nil
|
71
|
+
end
|
72
|
+
def consume_line line # :nodoc:
|
73
|
+
if @state == :precommand
|
74
|
+
unless line =~ /\A\s*\Z/
|
75
|
+
@command = line
|
76
|
+
@state = :headers
|
77
|
+
end
|
78
|
+
elsif @state == :headers
|
79
|
+
if line == ""
|
80
|
+
if @content_length
|
81
|
+
yield( [:sized_text, @content_length+1] )
|
82
|
+
else
|
83
|
+
@state = :body
|
84
|
+
yield( [:unsized_text] )
|
85
|
+
end
|
86
|
+
elsif line =~ /\A([^:]+):(.+)\Z/
|
87
|
+
k = $1.dup.strip
|
88
|
+
v = $2.dup.strip
|
89
|
+
@header[k] = v
|
90
|
+
if k == "content-length"
|
91
|
+
@content_length = v.to_i
|
92
|
+
end
|
93
|
+
else
|
94
|
+
# This is a protocol error. How to signal it?
|
95
|
+
end
|
96
|
+
elsif @state == :body
|
97
|
+
@body = line
|
98
|
+
yield( [:dispatch] )
|
99
|
+
end
|
100
|
+
end
|
101
|
+
end
|
102
|
+
|
103
|
+
# :stopdoc:
|
104
|
+
|
105
|
+
def send_frame verb, headers={}, body=""
|
106
|
+
ary = [verb, "\n"]
|
107
|
+
headers.each {|k,v| ary << "#{k}:#{v}\n" }
|
108
|
+
ary << "content-length: #{body.to_s.length}\n"
|
109
|
+
ary << "content-type: text/plain; charset=UTF-8\n" unless headers.has_key? 'content-type'
|
110
|
+
ary << "\n"
|
111
|
+
ary << body.to_s
|
112
|
+
ary << "\0"
|
113
|
+
send_data ary.join
|
114
|
+
end
|
115
|
+
|
116
|
+
def receive_line line
|
117
|
+
@stomp_initialized || init_message_reader
|
118
|
+
@stomp_message.consume_line(line) {|outcome|
|
119
|
+
if outcome.first == :sized_text
|
120
|
+
set_text_mode outcome[1]
|
121
|
+
elsif outcome.first == :unsized_text
|
122
|
+
set_delimiter "\0"
|
123
|
+
elsif outcome.first == :dispatch
|
124
|
+
receive_msg(@stomp_message) if respond_to?(:receive_msg)
|
125
|
+
init_message_reader
|
126
|
+
end
|
127
|
+
}
|
128
|
+
end
|
129
|
+
|
130
|
+
def receive_binary_data data
|
131
|
+
@stomp_message.body = data[0..-2]
|
132
|
+
receive_msg(@stomp_message) if respond_to?(:receive_msg)
|
133
|
+
init_message_reader
|
134
|
+
end
|
135
|
+
|
136
|
+
def init_message_reader
|
137
|
+
@stomp_initialized = true
|
138
|
+
set_delimiter "\n"
|
139
|
+
set_line_mode
|
140
|
+
@stomp_message = Message.new
|
141
|
+
end
|
142
|
+
|
143
|
+
# :startdoc:
|
144
|
+
|
145
|
+
# Invoked with an incoming Stomp::Message received from the STOMP server
|
146
|
+
def receive_msg msg
|
147
|
+
# stub, overwrite this in your handler
|
148
|
+
end
|
149
|
+
|
150
|
+
# CONNECT command, for authentication
|
151
|
+
#
|
152
|
+
# connect :login => 'guest', :passcode => 'guest'
|
153
|
+
#
|
154
|
+
def connect parms={}
|
155
|
+
send_frame "CONNECT", parms
|
156
|
+
end
|
157
|
+
|
158
|
+
# SEND command, for publishing messages to a topic
|
159
|
+
#
|
160
|
+
# send '/topic/name', 'some message here'
|
161
|
+
#
|
162
|
+
def send destination, body, parms={}
|
163
|
+
send_frame "SEND", parms.merge( :destination=>destination ), body.to_s
|
164
|
+
end
|
165
|
+
|
166
|
+
# SUBSCRIBE command, for subscribing to topics
|
167
|
+
#
|
168
|
+
# subscribe '/topic/name', false
|
169
|
+
#
|
170
|
+
def subscribe dest, ack=false
|
171
|
+
send_frame "SUBSCRIBE", {:destination=>dest, :ack=>(ack ? "client" : "auto")}
|
172
|
+
end
|
173
|
+
|
174
|
+
# ACK command, for acknowledging receipt of messages
|
175
|
+
#
|
176
|
+
# module StompClient
|
177
|
+
# include EM::P::Stomp
|
178
|
+
#
|
179
|
+
# def connection_completed
|
180
|
+
# connect :login => 'guest', :passcode => 'guest'
|
181
|
+
# # subscribe with ack mode
|
182
|
+
# subscribe '/some/topic', true
|
183
|
+
# end
|
184
|
+
#
|
185
|
+
# def receive_msg msg
|
186
|
+
# if msg.command == "MESSAGE"
|
187
|
+
# ack msg.headers['message-id']
|
188
|
+
# puts msg.body
|
189
|
+
# end
|
190
|
+
# end
|
191
|
+
# end
|
192
|
+
#
|
193
|
+
def ack msgid
|
194
|
+
send_frame "ACK", 'message-id'=> msgid
|
195
|
+
end
|
196
|
+
|
197
|
+
end
|
198
|
+
end
|
199
|
+
end
|
200
|
+
|
@@ -0,0 +1,53 @@
|
|
1
|
+
#--
|
2
|
+
#
|
3
|
+
# Author:: Francis Cianfrocca (gmail: blackhedd)
|
4
|
+
# Homepage:: http://rubyeventmachine.com
|
5
|
+
# Date:: 16 July 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
|
+
module Protocols
|
29
|
+
|
30
|
+
class TcpConnectTester < Connection # :nodoc:
|
31
|
+
include EventMachine::Deferrable
|
32
|
+
|
33
|
+
def self.test( host, port )
|
34
|
+
EventMachine.connect( host, port, self )
|
35
|
+
end
|
36
|
+
|
37
|
+
def post_init
|
38
|
+
@start_time = Time.now
|
39
|
+
end
|
40
|
+
|
41
|
+
def connection_completed
|
42
|
+
@completed = true
|
43
|
+
set_deferred_status :succeeded, (Time.now - @start_time)
|
44
|
+
close_connection
|
45
|
+
end
|
46
|
+
|
47
|
+
def unbind
|
48
|
+
set_deferred_status :failed, (Time.now - @start_time) unless @completed
|
49
|
+
end
|
50
|
+
end
|
51
|
+
|
52
|
+
end
|
53
|
+
end
|
data/lib/em/pure_ruby.rb
ADDED
@@ -0,0 +1,1013 @@
|
|
1
|
+
#--
|
2
|
+
#
|
3
|
+
# Author:: Francis Cianfrocca (gmail: blackhedd)
|
4
|
+
# Homepage:: http://rubyeventmachine.com
|
5
|
+
# Date:: 8 Apr 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
|
+
# TODO List:
|
27
|
+
# TCP-connects currently assume non-blocking connect is available- need to
|
28
|
+
# degrade automatically on versions of Ruby prior to June 2006.
|
29
|
+
#
|
30
|
+
|
31
|
+
require 'singleton'
|
32
|
+
require 'forwardable'
|
33
|
+
require 'socket'
|
34
|
+
require 'fcntl'
|
35
|
+
require 'set'
|
36
|
+
|
37
|
+
module EventMachine
|
38
|
+
class << self
|
39
|
+
# This is mostly useful for automated tests.
|
40
|
+
# Return a distinctive symbol so the caller knows whether he's dealing
|
41
|
+
# with an extension or with a pure-Ruby library.
|
42
|
+
def library_type
|
43
|
+
:pure_ruby
|
44
|
+
end
|
45
|
+
|
46
|
+
# #initialize_event_machine
|
47
|
+
def initialize_event_machine
|
48
|
+
Reactor.instance.initialize_for_run
|
49
|
+
end
|
50
|
+
|
51
|
+
# #add_oneshot_timer
|
52
|
+
#--
|
53
|
+
# Changed 04Oct06: intervals from the caller are now in milliseconds, but our native-ruby
|
54
|
+
# processor still wants them in seconds.
|
55
|
+
def add_oneshot_timer interval
|
56
|
+
Reactor.instance.install_oneshot_timer(interval / 1000)
|
57
|
+
end
|
58
|
+
|
59
|
+
# run_machine
|
60
|
+
def run_machine
|
61
|
+
Reactor.instance.run
|
62
|
+
end
|
63
|
+
|
64
|
+
# release_machine. Probably a no-op.
|
65
|
+
def release_machine
|
66
|
+
end
|
67
|
+
|
68
|
+
# #stop
|
69
|
+
def stop
|
70
|
+
Reactor.instance.stop
|
71
|
+
end
|
72
|
+
|
73
|
+
# #connect_server. Return a connection descriptor to the caller.
|
74
|
+
# TODO, what do we return here if we can't connect?
|
75
|
+
def connect_server host, port
|
76
|
+
bind_connect_server nil, nil, host, port
|
77
|
+
end
|
78
|
+
|
79
|
+
def bind_connect_server bind_addr, bind_port, host, port
|
80
|
+
EvmaTCPClient.connect(bind_addr, bind_port, host, port).uuid
|
81
|
+
end
|
82
|
+
|
83
|
+
# #send_data
|
84
|
+
def send_data target, data, datalength
|
85
|
+
selectable = Reactor.instance.get_selectable( target ) or raise "unknown send_data target"
|
86
|
+
selectable.send_data data
|
87
|
+
end
|
88
|
+
|
89
|
+
# #close_connection
|
90
|
+
# The extension version does NOT raise any kind of an error if an attempt is made
|
91
|
+
# to close a non-existent connection. Not sure whether we should. For now, we'll
|
92
|
+
# raise an error here in that case.
|
93
|
+
def close_connection target, after_writing
|
94
|
+
selectable = Reactor.instance.get_selectable( target ) or raise "unknown close_connection target"
|
95
|
+
selectable.schedule_close after_writing
|
96
|
+
end
|
97
|
+
|
98
|
+
# #start_tcp_server
|
99
|
+
def start_tcp_server host, port
|
100
|
+
(s = EvmaTCPServer.start_server host, port) or raise "no acceptor"
|
101
|
+
s.uuid
|
102
|
+
end
|
103
|
+
|
104
|
+
# #stop_tcp_server
|
105
|
+
def stop_tcp_server sig
|
106
|
+
s = Reactor.instance.get_selectable(sig)
|
107
|
+
s.schedule_close
|
108
|
+
end
|
109
|
+
|
110
|
+
# #start_unix_server
|
111
|
+
def start_unix_server chain
|
112
|
+
(s = EvmaUNIXServer.start_server chain) or raise "no acceptor"
|
113
|
+
s.uuid
|
114
|
+
end
|
115
|
+
|
116
|
+
# #connect_unix_server
|
117
|
+
def connect_unix_server chain
|
118
|
+
EvmaUNIXClient.connect(chain).uuid
|
119
|
+
end
|
120
|
+
|
121
|
+
# #signal_loopbreak
|
122
|
+
def signal_loopbreak
|
123
|
+
Reactor.instance.signal_loopbreak
|
124
|
+
end
|
125
|
+
|
126
|
+
# #get_peername
|
127
|
+
def get_peername sig
|
128
|
+
selectable = Reactor.instance.get_selectable( sig ) or raise "unknown get_peername target"
|
129
|
+
selectable.get_peername
|
130
|
+
end
|
131
|
+
|
132
|
+
# #open_udp_socket
|
133
|
+
def open_udp_socket host, port
|
134
|
+
EvmaUDPSocket.create(host, port).uuid
|
135
|
+
end
|
136
|
+
|
137
|
+
# #send_datagram. This is currently only for UDP!
|
138
|
+
# We need to make it work with unix-domain sockets as well.
|
139
|
+
def send_datagram target, data, datalength, host, port
|
140
|
+
selectable = Reactor.instance.get_selectable( target ) or raise "unknown send_data target"
|
141
|
+
selectable.send_datagram data, Socket::pack_sockaddr_in(port, host)
|
142
|
+
end
|
143
|
+
|
144
|
+
|
145
|
+
# #set_timer_quantum in milliseconds. The underlying Reactor function wants a (possibly
|
146
|
+
# fractional) number of seconds.
|
147
|
+
def set_timer_quantum interval
|
148
|
+
Reactor.instance.set_timer_quantum(( 1.0 * interval) / 1000.0)
|
149
|
+
end
|
150
|
+
|
151
|
+
# #epoll is a harmless no-op in the pure-Ruby implementation. This is intended to ensure
|
152
|
+
# that user code behaves properly across different EM implementations.
|
153
|
+
def epoll
|
154
|
+
end
|
155
|
+
|
156
|
+
# #ssl? is not implemented for pure-Ruby implementation
|
157
|
+
def ssl?
|
158
|
+
false
|
159
|
+
end
|
160
|
+
|
161
|
+
# #set_rlimit_nofile is a no-op in the pure-Ruby implementation. We simply return Ruby's built-in
|
162
|
+
# per-process file-descriptor limit.
|
163
|
+
def set_rlimit_nofile n
|
164
|
+
1024
|
165
|
+
end
|
166
|
+
|
167
|
+
# #set_max_timer_count is a harmless no-op in pure Ruby, which doesn't have a built-in limit
|
168
|
+
# on the number of available timers.
|
169
|
+
def set_max_timer_count n
|
170
|
+
end
|
171
|
+
|
172
|
+
# #send_file_data
|
173
|
+
def send_file_data sig, filename
|
174
|
+
sz = File.size(filename)
|
175
|
+
raise "file too large" if sz > 32*1024
|
176
|
+
data =
|
177
|
+
begin
|
178
|
+
File.read filename
|
179
|
+
rescue
|
180
|
+
""
|
181
|
+
end
|
182
|
+
send_data sig, data, data.length
|
183
|
+
end
|
184
|
+
|
185
|
+
# #get_outbound_data_size
|
186
|
+
#
|
187
|
+
def get_outbound_data_size sig
|
188
|
+
r = Reactor.instance.get_selectable( sig ) or raise "unknown get_outbound_data_size target"
|
189
|
+
r.get_outbound_data_size
|
190
|
+
end
|
191
|
+
|
192
|
+
# #read_keyboard
|
193
|
+
#
|
194
|
+
def read_keyboard
|
195
|
+
EvmaKeyboard.open.uuid
|
196
|
+
end
|
197
|
+
|
198
|
+
# #set_comm_inactivity_timeout
|
199
|
+
#
|
200
|
+
def set_comm_inactivity_timeout sig, tm
|
201
|
+
r = Reactor.instance.get_selectable( sig ) or raise "unknown set_comm_inactivity_timeout target"
|
202
|
+
r.set_inactivity_timeout tm
|
203
|
+
end
|
204
|
+
end
|
205
|
+
|
206
|
+
end
|
207
|
+
|
208
|
+
|
209
|
+
#-----------------------------------------------------------------
|
210
|
+
|
211
|
+
module EventMachine
|
212
|
+
|
213
|
+
class Error < Exception; end
|
214
|
+
|
215
|
+
end
|
216
|
+
|
217
|
+
#-----------------------------------------------------------------
|
218
|
+
|
219
|
+
module EventMachine
|
220
|
+
class Connection
|
221
|
+
def get_outbound_data_size
|
222
|
+
EventMachine::get_outbound_data_size @signature
|
223
|
+
end
|
224
|
+
end
|
225
|
+
end
|
226
|
+
|
227
|
+
#-----------------------------------------------------------------
|
228
|
+
|
229
|
+
module EventMachine
|
230
|
+
|
231
|
+
# Factored out so we can substitute other implementations
|
232
|
+
# here if desired, such as the one in ActiveRBAC.
|
233
|
+
module UuidGenerator
|
234
|
+
|
235
|
+
def self.generate
|
236
|
+
@ix ||= 0
|
237
|
+
@ix += 1
|
238
|
+
end
|
239
|
+
|
240
|
+
end
|
241
|
+
|
242
|
+
end
|
243
|
+
|
244
|
+
#-----------------------------------------------------------------
|
245
|
+
|
246
|
+
module EventMachine
|
247
|
+
|
248
|
+
TimerFired = 100
|
249
|
+
ConnectionData = 101
|
250
|
+
ConnectionUnbound = 102
|
251
|
+
ConnectionAccepted = 103
|
252
|
+
ConnectionCompleted = 104
|
253
|
+
LoopbreakSignalled = 105
|
254
|
+
|
255
|
+
end
|
256
|
+
|
257
|
+
#-----------------------------------------------------------------
|
258
|
+
|
259
|
+
module EventMachine
|
260
|
+
class Reactor
|
261
|
+
include Singleton
|
262
|
+
|
263
|
+
HeartbeatInterval = 2
|
264
|
+
|
265
|
+
attr_reader :current_loop_time
|
266
|
+
|
267
|
+
def initialize
|
268
|
+
initialize_for_run
|
269
|
+
end
|
270
|
+
|
271
|
+
#--
|
272
|
+
# Replaced original implementation 05Dec07, was way too slow because of the sort.
|
273
|
+
def install_oneshot_timer interval
|
274
|
+
uuid = UuidGenerator::generate
|
275
|
+
#@timers << [Time.now + interval, uuid]
|
276
|
+
#@timers.sort! {|a,b| a.first <=> b.first}
|
277
|
+
@timers.add([Time.now + interval, uuid])
|
278
|
+
uuid
|
279
|
+
end
|
280
|
+
|
281
|
+
# Called before run, this is a good place to clear out arrays
|
282
|
+
# with cruft that may be left over from a previous run.
|
283
|
+
def initialize_for_run
|
284
|
+
@running = false
|
285
|
+
@stop_scheduled = false
|
286
|
+
@selectables ||= {}; @selectables.clear
|
287
|
+
@timers = SortedSet.new # []
|
288
|
+
set_timer_quantum(0.1)
|
289
|
+
@current_loop_time = Time.now
|
290
|
+
@next_heartbeat = @current_loop_time + HeartbeatInterval
|
291
|
+
end
|
292
|
+
|
293
|
+
def add_selectable io
|
294
|
+
@selectables[io.uuid] = io
|
295
|
+
end
|
296
|
+
|
297
|
+
def get_selectable uuid
|
298
|
+
@selectables[uuid]
|
299
|
+
end
|
300
|
+
|
301
|
+
def run
|
302
|
+
raise Error.new( "already running" ) if @running
|
303
|
+
@running = true
|
304
|
+
|
305
|
+
begin
|
306
|
+
open_loopbreaker
|
307
|
+
|
308
|
+
loop {
|
309
|
+
@current_loop_time = Time.now
|
310
|
+
|
311
|
+
break if @stop_scheduled
|
312
|
+
run_timers
|
313
|
+
break if @stop_scheduled
|
314
|
+
crank_selectables
|
315
|
+
break if @stop_scheduled
|
316
|
+
run_heartbeats
|
317
|
+
}
|
318
|
+
ensure
|
319
|
+
close_loopbreaker
|
320
|
+
@selectables.each {|k, io| io.close}
|
321
|
+
@selectables.clear
|
322
|
+
|
323
|
+
@running = false
|
324
|
+
end
|
325
|
+
|
326
|
+
end
|
327
|
+
|
328
|
+
def run_timers
|
329
|
+
@timers.each {|t|
|
330
|
+
if t.first <= @current_loop_time
|
331
|
+
@timers.delete t
|
332
|
+
EventMachine::event_callback "", TimerFired, t.last
|
333
|
+
else
|
334
|
+
break
|
335
|
+
end
|
336
|
+
}
|
337
|
+
#while @timers.length > 0 and @timers.first.first <= now
|
338
|
+
# t = @timers.shift
|
339
|
+
# EventMachine::event_callback "", TimerFired, t.last
|
340
|
+
#end
|
341
|
+
end
|
342
|
+
|
343
|
+
def run_heartbeats
|
344
|
+
if @next_heartbeat <= @current_loop_time
|
345
|
+
@next_heartbeat = @current_loop_time + HeartbeatInterval
|
346
|
+
@selectables.each {|k,io| io.heartbeat}
|
347
|
+
end
|
348
|
+
end
|
349
|
+
|
350
|
+
def crank_selectables
|
351
|
+
#$stderr.write 'R'
|
352
|
+
|
353
|
+
readers = @selectables.values.select {|io| io.select_for_reading?}
|
354
|
+
writers = @selectables.values.select {|io| io.select_for_writing?}
|
355
|
+
|
356
|
+
s = select( readers, writers, nil, @timer_quantum)
|
357
|
+
|
358
|
+
s and s[1] and s[1].each {|w| w.eventable_write }
|
359
|
+
s and s[0] and s[0].each {|r| r.eventable_read }
|
360
|
+
|
361
|
+
@selectables.delete_if {|k,io|
|
362
|
+
if io.close_scheduled?
|
363
|
+
io.close
|
364
|
+
true
|
365
|
+
end
|
366
|
+
}
|
367
|
+
end
|
368
|
+
|
369
|
+
# #stop
|
370
|
+
def stop
|
371
|
+
raise Error.new( "not running") unless @running
|
372
|
+
@stop_scheduled = true
|
373
|
+
end
|
374
|
+
|
375
|
+
def open_loopbreaker
|
376
|
+
# Can't use an IO.pipe because they can't be set nonselectable in Windows.
|
377
|
+
# Pick a random localhost UDP port.
|
378
|
+
#@loopbreak_writer.close if @loopbreak_writer
|
379
|
+
#rd,@loopbreak_writer = IO.pipe
|
380
|
+
@loopbreak_reader = UDPSocket.new
|
381
|
+
@loopbreak_writer = UDPSocket.new
|
382
|
+
bound = false
|
383
|
+
100.times {
|
384
|
+
@loopbreak_port = rand(10000) + 40000
|
385
|
+
begin
|
386
|
+
@loopbreak_reader.bind "localhost", @loopbreak_port
|
387
|
+
bound = true
|
388
|
+
break
|
389
|
+
rescue
|
390
|
+
end
|
391
|
+
}
|
392
|
+
raise "Unable to bind Loopbreaker" unless bound
|
393
|
+
LoopbreakReader.new(@loopbreak_reader)
|
394
|
+
end
|
395
|
+
|
396
|
+
def close_loopbreaker
|
397
|
+
@loopbreak_writer.close
|
398
|
+
@loopbreak_writer = nil
|
399
|
+
end
|
400
|
+
|
401
|
+
def signal_loopbreak
|
402
|
+
#@loopbreak_writer.write '+' if @loopbreak_writer
|
403
|
+
@loopbreak_writer.send('+',0,"localhost",@loopbreak_port) if @loopbreak_writer
|
404
|
+
end
|
405
|
+
|
406
|
+
def set_timer_quantum interval_in_seconds
|
407
|
+
@timer_quantum = interval_in_seconds
|
408
|
+
end
|
409
|
+
|
410
|
+
end
|
411
|
+
|
412
|
+
end
|
413
|
+
|
414
|
+
|
415
|
+
#--------------------------------------------------------------
|
416
|
+
|
417
|
+
class IO
|
418
|
+
extend Forwardable
|
419
|
+
def_delegator :@my_selectable, :close_scheduled?
|
420
|
+
def_delegator :@my_selectable, :select_for_reading?
|
421
|
+
def_delegator :@my_selectable, :select_for_writing?
|
422
|
+
def_delegator :@my_selectable, :eventable_read
|
423
|
+
def_delegator :@my_selectable, :eventable_write
|
424
|
+
def_delegator :@my_selectable, :uuid
|
425
|
+
def_delegator :@my_selectable, :send_data
|
426
|
+
def_delegator :@my_selectable, :schedule_close
|
427
|
+
def_delegator :@my_selectable, :get_peername
|
428
|
+
def_delegator :@my_selectable, :send_datagram
|
429
|
+
def_delegator :@my_selectable, :get_outbound_data_size
|
430
|
+
def_delegator :@my_selectable, :set_inactivity_timeout
|
431
|
+
def_delegator :@my_selectable, :heartbeat
|
432
|
+
end
|
433
|
+
|
434
|
+
#--------------------------------------------------------------
|
435
|
+
|
436
|
+
module EventMachine
|
437
|
+
class Selectable
|
438
|
+
|
439
|
+
attr_reader :io, :uuid
|
440
|
+
|
441
|
+
def initialize io
|
442
|
+
@uuid = UuidGenerator.generate
|
443
|
+
@io = io
|
444
|
+
@last_activity = Reactor.instance.current_loop_time
|
445
|
+
|
446
|
+
if defined?(Fcntl::F_GETFL)
|
447
|
+
m = @io.fcntl(Fcntl::F_GETFL, 0)
|
448
|
+
@io.fcntl(Fcntl::F_SETFL, Fcntl::O_NONBLOCK | m)
|
449
|
+
else
|
450
|
+
# Windows doesn't define F_GETFL.
|
451
|
+
# It's not very reliable about setting descriptors nonblocking either.
|
452
|
+
begin
|
453
|
+
s = Socket.for_fd(@io.fileno)
|
454
|
+
s.fcntl( Fcntl::F_SETFL, Fcntl::O_NONBLOCK )
|
455
|
+
rescue Errno::EINVAL, Errno::EBADF
|
456
|
+
STDERR.puts "Serious error: unable to set descriptor non-blocking"
|
457
|
+
end
|
458
|
+
end
|
459
|
+
# TODO, should set CLOEXEC on Unix?
|
460
|
+
|
461
|
+
@close_scheduled = false
|
462
|
+
@close_requested = false
|
463
|
+
|
464
|
+
se = self; @io.instance_eval { @my_selectable = se }
|
465
|
+
Reactor.instance.add_selectable @io
|
466
|
+
end
|
467
|
+
|
468
|
+
def close_scheduled?
|
469
|
+
@close_scheduled
|
470
|
+
end
|
471
|
+
|
472
|
+
def select_for_reading?
|
473
|
+
false
|
474
|
+
end
|
475
|
+
|
476
|
+
def select_for_writing?
|
477
|
+
false
|
478
|
+
end
|
479
|
+
|
480
|
+
def get_peername
|
481
|
+
nil
|
482
|
+
end
|
483
|
+
|
484
|
+
def set_inactivity_timeout tm
|
485
|
+
@inactivity_timeout = tm
|
486
|
+
end
|
487
|
+
|
488
|
+
def heartbeat
|
489
|
+
end
|
490
|
+
end
|
491
|
+
|
492
|
+
end
|
493
|
+
|
494
|
+
#--------------------------------------------------------------
|
495
|
+
|
496
|
+
|
497
|
+
module EventMachine
|
498
|
+
|
499
|
+
class StreamObject < Selectable
|
500
|
+
def initialize io
|
501
|
+
super io
|
502
|
+
@outbound_q = []
|
503
|
+
end
|
504
|
+
|
505
|
+
# If we have to close, or a close-after-writing has been requested,
|
506
|
+
# then don't read any more data.
|
507
|
+
def select_for_reading?
|
508
|
+
true unless (@close_scheduled || @close_requested)
|
509
|
+
end
|
510
|
+
|
511
|
+
# If we have to close, don't select for writing.
|
512
|
+
# Otherwise, see if the protocol is ready to close.
|
513
|
+
# If not, see if he has data to send.
|
514
|
+
# If a close-after-writing has been requested and the outbound queue
|
515
|
+
# is empty, convert the status to close_scheduled.
|
516
|
+
def select_for_writing?
|
517
|
+
unless @close_scheduled
|
518
|
+
if @outbound_q.empty?
|
519
|
+
@close_scheduled = true if @close_requested
|
520
|
+
false
|
521
|
+
else
|
522
|
+
true
|
523
|
+
end
|
524
|
+
end
|
525
|
+
end
|
526
|
+
|
527
|
+
# Proper nonblocking I/O was added to Ruby 1.8.4 in May 2006.
|
528
|
+
# If we have it, then we can read multiple times safely to improve
|
529
|
+
# performance.
|
530
|
+
# The last-activity clock ASSUMES that we only come here when we
|
531
|
+
# have selected readable.
|
532
|
+
# TODO, coalesce multiple reads into a single event.
|
533
|
+
# TODO, do the function check somewhere else and cache it.
|
534
|
+
def eventable_read
|
535
|
+
@last_activity = Reactor.instance.current_loop_time
|
536
|
+
begin
|
537
|
+
if io.respond_to?(:read_nonblock)
|
538
|
+
10.times {
|
539
|
+
data = io.read_nonblock(4096)
|
540
|
+
EventMachine::event_callback uuid, ConnectionData, data
|
541
|
+
}
|
542
|
+
else
|
543
|
+
data = io.sysread(4096)
|
544
|
+
EventMachine::event_callback uuid, ConnectionData, data
|
545
|
+
end
|
546
|
+
rescue Errno::EAGAIN, Errno::EWOULDBLOCK
|
547
|
+
# no-op
|
548
|
+
rescue Errno::ECONNRESET, Errno::ECONNREFUSED, EOFError
|
549
|
+
@close_scheduled = true
|
550
|
+
EventMachine::event_callback uuid, ConnectionUnbound, nil
|
551
|
+
end
|
552
|
+
|
553
|
+
end
|
554
|
+
|
555
|
+
# Provisional implementation. Will be re-implemented in subclasses.
|
556
|
+
# TODO: Complete this implementation. As it stands, this only writes
|
557
|
+
# a single packet per cycle. Highly inefficient, but required unless
|
558
|
+
# we're running on a Ruby with proper nonblocking I/O (Ruby 1.8.4
|
559
|
+
# built from sources from May 25, 2006 or newer).
|
560
|
+
# We need to improve the loop so it writes multiple times, however
|
561
|
+
# not more than a certain number of bytes per cycle, otherwise
|
562
|
+
# one busy connection could hog output buffers and slow down other
|
563
|
+
# connections. Also we should coalesce small writes.
|
564
|
+
# URGENT TODO: Coalesce small writes. They are a performance killer.
|
565
|
+
# The last-activity recorder ASSUMES we'll only come here if we've
|
566
|
+
# selected writable.
|
567
|
+
def eventable_write
|
568
|
+
# coalesce the outbound array here, perhaps
|
569
|
+
@last_activity = Reactor.instance.current_loop_time
|
570
|
+
while data = @outbound_q.shift do
|
571
|
+
begin
|
572
|
+
data = data.to_s
|
573
|
+
w = if io.respond_to?(:write_nonblock)
|
574
|
+
io.write_nonblock data
|
575
|
+
else
|
576
|
+
io.syswrite data
|
577
|
+
end
|
578
|
+
|
579
|
+
if w < data.length
|
580
|
+
@outbound_q.unshift data[w..-1]
|
581
|
+
break
|
582
|
+
end
|
583
|
+
rescue Errno::EAGAIN
|
584
|
+
@outbound_q.unshift data
|
585
|
+
rescue EOFError, Errno::ECONNRESET, Errno::ECONNREFUSED
|
586
|
+
@close_scheduled = true
|
587
|
+
@outbound_q.clear
|
588
|
+
end
|
589
|
+
end
|
590
|
+
|
591
|
+
end
|
592
|
+
|
593
|
+
# #send_data
|
594
|
+
def send_data data
|
595
|
+
# TODO, coalesce here perhaps by being smarter about appending to @outbound_q.last?
|
596
|
+
unless @close_scheduled or @close_requested or !data or data.length <= 0
|
597
|
+
@outbound_q << data.to_s
|
598
|
+
end
|
599
|
+
end
|
600
|
+
|
601
|
+
# #schedule_close
|
602
|
+
# The application wants to close the connection.
|
603
|
+
def schedule_close after_writing
|
604
|
+
if after_writing
|
605
|
+
@close_requested = true
|
606
|
+
else
|
607
|
+
@close_scheduled = true
|
608
|
+
end
|
609
|
+
end
|
610
|
+
|
611
|
+
# #get_peername
|
612
|
+
# This is defined in the normal way on connected stream objects.
|
613
|
+
# Return an object that is suitable for passing to Socket#unpack_sockaddr_in or variants.
|
614
|
+
# We could also use a convenience method that did the unpacking automatically.
|
615
|
+
def get_peername
|
616
|
+
io.getpeername
|
617
|
+
end
|
618
|
+
|
619
|
+
# #get_outbound_data_size
|
620
|
+
def get_outbound_data_size
|
621
|
+
@outbound_q.inject(0) {|memo,obj| memo += (obj || "").length}
|
622
|
+
end
|
623
|
+
|
624
|
+
def heartbeat
|
625
|
+
if @inactivity_timeout and @inactivity_timeout > 0 and (@last_activity + @inactivity_timeout) < Reactor.instance.current_loop_time
|
626
|
+
schedule_close true
|
627
|
+
end
|
628
|
+
end
|
629
|
+
end
|
630
|
+
|
631
|
+
|
632
|
+
end
|
633
|
+
|
634
|
+
|
635
|
+
#--------------------------------------------------------------
|
636
|
+
|
637
|
+
|
638
|
+
|
639
|
+
module EventMachine
|
640
|
+
class EvmaTCPClient < StreamObject
|
641
|
+
|
642
|
+
def self.connect bind_addr, bind_port, host, port
|
643
|
+
sd = Socket.new( Socket::AF_INET, Socket::SOCK_STREAM, 0 )
|
644
|
+
sd.bind( Socket.pack_sockaddr_in( bind_port, bind_addr )) if bind_addr
|
645
|
+
|
646
|
+
begin
|
647
|
+
# TODO, this assumes a current Ruby snapshot.
|
648
|
+
# We need to degrade to a nonblocking connect otherwise.
|
649
|
+
sd.connect_nonblock( Socket.pack_sockaddr_in( port, host ))
|
650
|
+
rescue Errno::EINPROGRESS
|
651
|
+
end
|
652
|
+
EvmaTCPClient.new sd
|
653
|
+
end
|
654
|
+
|
655
|
+
|
656
|
+
def initialize io
|
657
|
+
super
|
658
|
+
@pending = true
|
659
|
+
end
|
660
|
+
|
661
|
+
|
662
|
+
def select_for_writing?
|
663
|
+
@pending ? true : super
|
664
|
+
end
|
665
|
+
|
666
|
+
def select_for_reading?
|
667
|
+
@pending ? false : super
|
668
|
+
end
|
669
|
+
|
670
|
+
def eventable_write
|
671
|
+
if @pending
|
672
|
+
@pending = false
|
673
|
+
if 0 == io.getsockopt(Socket::SOL_SOCKET, Socket::SO_ERROR).unpack("i").first
|
674
|
+
EventMachine::event_callback uuid, ConnectionCompleted, ""
|
675
|
+
end
|
676
|
+
else
|
677
|
+
super
|
678
|
+
end
|
679
|
+
end
|
680
|
+
|
681
|
+
|
682
|
+
|
683
|
+
end
|
684
|
+
end
|
685
|
+
|
686
|
+
#--------------------------------------------------------------
|
687
|
+
|
688
|
+
|
689
|
+
|
690
|
+
module EventMachine
|
691
|
+
class EvmaKeyboard < StreamObject
|
692
|
+
|
693
|
+
def self.open
|
694
|
+
EvmaKeyboard.new STDIN
|
695
|
+
end
|
696
|
+
|
697
|
+
|
698
|
+
def initialize io
|
699
|
+
super
|
700
|
+
end
|
701
|
+
|
702
|
+
|
703
|
+
def select_for_writing?
|
704
|
+
false
|
705
|
+
end
|
706
|
+
|
707
|
+
def select_for_reading?
|
708
|
+
true
|
709
|
+
end
|
710
|
+
|
711
|
+
|
712
|
+
end
|
713
|
+
end
|
714
|
+
|
715
|
+
|
716
|
+
#--------------------------------------------------------------
|
717
|
+
|
718
|
+
|
719
|
+
|
720
|
+
module EventMachine
|
721
|
+
class EvmaUNIXClient < StreamObject
|
722
|
+
|
723
|
+
def self.connect chain
|
724
|
+
sd = Socket.new( Socket::AF_LOCAL, Socket::SOCK_STREAM, 0 )
|
725
|
+
begin
|
726
|
+
# TODO, this assumes a current Ruby snapshot.
|
727
|
+
# We need to degrade to a nonblocking connect otherwise.
|
728
|
+
sd.connect_nonblock( Socket.pack_sockaddr_un( chain ))
|
729
|
+
rescue Errno::EINPROGRESS
|
730
|
+
end
|
731
|
+
EvmaUNIXClient.new sd
|
732
|
+
end
|
733
|
+
|
734
|
+
|
735
|
+
def initialize io
|
736
|
+
super
|
737
|
+
@pending = true
|
738
|
+
end
|
739
|
+
|
740
|
+
|
741
|
+
def select_for_writing?
|
742
|
+
@pending ? true : super
|
743
|
+
end
|
744
|
+
|
745
|
+
def select_for_reading?
|
746
|
+
@pending ? false : super
|
747
|
+
end
|
748
|
+
|
749
|
+
def eventable_write
|
750
|
+
if @pending
|
751
|
+
@pending = false
|
752
|
+
if 0 == io.getsockopt(Socket::SOL_SOCKET, Socket::SO_ERROR).unpack("i").first
|
753
|
+
EventMachine::event_callback uuid, ConnectionCompleted, ""
|
754
|
+
end
|
755
|
+
else
|
756
|
+
super
|
757
|
+
end
|
758
|
+
end
|
759
|
+
|
760
|
+
|
761
|
+
|
762
|
+
end
|
763
|
+
end
|
764
|
+
|
765
|
+
|
766
|
+
#--------------------------------------------------------------
|
767
|
+
|
768
|
+
module EventMachine
|
769
|
+
class EvmaTCPServer < Selectable
|
770
|
+
|
771
|
+
# TODO, refactor and unify with EvmaUNIXServer.
|
772
|
+
|
773
|
+
class << self
|
774
|
+
# Versions of ruby 1.8.4 later than May 26 2006 will work properly
|
775
|
+
# with an object of type TCPServer. Prior versions won't so we
|
776
|
+
# play it safe and just build a socket.
|
777
|
+
#
|
778
|
+
def start_server host, port
|
779
|
+
sd = Socket.new( Socket::AF_INET, Socket::SOCK_STREAM, 0 )
|
780
|
+
sd.setsockopt( Socket::SOL_SOCKET, Socket::SO_REUSEADDR, true )
|
781
|
+
sd.bind( Socket.pack_sockaddr_in( port, host ))
|
782
|
+
sd.listen( 50 ) # 5 is what you see in all the books. Ain't enough.
|
783
|
+
EvmaTCPServer.new sd
|
784
|
+
end
|
785
|
+
end
|
786
|
+
|
787
|
+
def initialize io
|
788
|
+
super io
|
789
|
+
end
|
790
|
+
|
791
|
+
|
792
|
+
def select_for_reading?
|
793
|
+
true
|
794
|
+
end
|
795
|
+
|
796
|
+
#--
|
797
|
+
# accept_nonblock returns an array consisting of the accepted
|
798
|
+
# socket and a sockaddr_in which names the peer.
|
799
|
+
# Don't accept more than 10 at a time.
|
800
|
+
def eventable_read
|
801
|
+
begin
|
802
|
+
10.times {
|
803
|
+
descriptor,peername = io.accept_nonblock
|
804
|
+
sd = StreamObject.new descriptor
|
805
|
+
EventMachine::event_callback uuid, ConnectionAccepted, sd.uuid
|
806
|
+
}
|
807
|
+
rescue Errno::EWOULDBLOCK, Errno::EAGAIN
|
808
|
+
end
|
809
|
+
end
|
810
|
+
|
811
|
+
#--
|
812
|
+
#
|
813
|
+
def schedule_close
|
814
|
+
@close_scheduled = true
|
815
|
+
end
|
816
|
+
|
817
|
+
end
|
818
|
+
end
|
819
|
+
|
820
|
+
|
821
|
+
#--------------------------------------------------------------
|
822
|
+
|
823
|
+
module EventMachine
|
824
|
+
class EvmaUNIXServer < Selectable
|
825
|
+
|
826
|
+
# TODO, refactor and unify with EvmaTCPServer.
|
827
|
+
|
828
|
+
class << self
|
829
|
+
# Versions of ruby 1.8.4 later than May 26 2006 will work properly
|
830
|
+
# with an object of type TCPServer. Prior versions won't so we
|
831
|
+
# play it safe and just build a socket.
|
832
|
+
#
|
833
|
+
def start_server chain
|
834
|
+
sd = Socket.new( Socket::AF_LOCAL, Socket::SOCK_STREAM, 0 )
|
835
|
+
sd.setsockopt( Socket::SOL_SOCKET, Socket::SO_REUSEADDR, true )
|
836
|
+
sd.bind( Socket.pack_sockaddr_un( chain ))
|
837
|
+
sd.listen( 50 ) # 5 is what you see in all the books. Ain't enough.
|
838
|
+
EvmaUNIXServer.new sd
|
839
|
+
end
|
840
|
+
end
|
841
|
+
|
842
|
+
def initialize io
|
843
|
+
super io
|
844
|
+
end
|
845
|
+
|
846
|
+
|
847
|
+
def select_for_reading?
|
848
|
+
true
|
849
|
+
end
|
850
|
+
|
851
|
+
#--
|
852
|
+
# accept_nonblock returns an array consisting of the accepted
|
853
|
+
# socket and a sockaddr_in which names the peer.
|
854
|
+
# Don't accept more than 10 at a time.
|
855
|
+
def eventable_read
|
856
|
+
begin
|
857
|
+
10.times {
|
858
|
+
descriptor,peername = io.accept_nonblock
|
859
|
+
sd = StreamObject.new descriptor
|
860
|
+
EventMachine::event_callback uuid, ConnectionAccepted, sd.uuid
|
861
|
+
}
|
862
|
+
rescue Errno::EWOULDBLOCK, Errno::EAGAIN
|
863
|
+
end
|
864
|
+
end
|
865
|
+
|
866
|
+
#--
|
867
|
+
#
|
868
|
+
def schedule_close
|
869
|
+
@close_scheduled = true
|
870
|
+
end
|
871
|
+
|
872
|
+
end
|
873
|
+
end
|
874
|
+
|
875
|
+
|
876
|
+
|
877
|
+
#--------------------------------------------------------------
|
878
|
+
|
879
|
+
module EventMachine
|
880
|
+
class LoopbreakReader < Selectable
|
881
|
+
|
882
|
+
def select_for_reading?
|
883
|
+
true
|
884
|
+
end
|
885
|
+
|
886
|
+
def eventable_read
|
887
|
+
io.sysread(128)
|
888
|
+
EventMachine::event_callback "", LoopbreakSignalled, ""
|
889
|
+
end
|
890
|
+
|
891
|
+
end
|
892
|
+
end
|
893
|
+
|
894
|
+
#--------------------------------------------------------------
|
895
|
+
|
896
|
+
|
897
|
+
module EventMachine
|
898
|
+
|
899
|
+
class DatagramObject < Selectable
|
900
|
+
def initialize io
|
901
|
+
super io
|
902
|
+
@outbound_q = []
|
903
|
+
end
|
904
|
+
|
905
|
+
# #send_datagram
|
906
|
+
def send_datagram data, target
|
907
|
+
# TODO, coalesce here perhaps by being smarter about appending to @outbound_q.last?
|
908
|
+
unless @close_scheduled or @close_requested
|
909
|
+
@outbound_q << [data.to_s, target]
|
910
|
+
end
|
911
|
+
end
|
912
|
+
|
913
|
+
# #select_for_writing?
|
914
|
+
def select_for_writing?
|
915
|
+
unless @close_scheduled
|
916
|
+
if @outbound_q.empty?
|
917
|
+
@close_scheduled = true if @close_requested
|
918
|
+
false
|
919
|
+
else
|
920
|
+
true
|
921
|
+
end
|
922
|
+
end
|
923
|
+
end
|
924
|
+
|
925
|
+
# #select_for_reading?
|
926
|
+
def select_for_reading?
|
927
|
+
true
|
928
|
+
end
|
929
|
+
|
930
|
+
# #get_outbound_data_size
|
931
|
+
def get_outbound_data_size
|
932
|
+
@outbound_q.inject(0) {|memo,obj| memo += (obj || "").length}
|
933
|
+
end
|
934
|
+
|
935
|
+
|
936
|
+
end
|
937
|
+
|
938
|
+
|
939
|
+
end
|
940
|
+
|
941
|
+
|
942
|
+
#--------------------------------------------------------------
|
943
|
+
|
944
|
+
module EventMachine
|
945
|
+
class EvmaUDPSocket < DatagramObject
|
946
|
+
|
947
|
+
class << self
|
948
|
+
def create host, port
|
949
|
+
sd = Socket.new( Socket::AF_INET, Socket::SOCK_DGRAM, 0 )
|
950
|
+
sd.bind Socket::pack_sockaddr_in( port, host )
|
951
|
+
EvmaUDPSocket.new sd
|
952
|
+
end
|
953
|
+
end
|
954
|
+
|
955
|
+
# #eventable_write
|
956
|
+
# This really belongs in DatagramObject, but there is some UDP-specific stuff.
|
957
|
+
def eventable_write
|
958
|
+
40.times {
|
959
|
+
break if @outbound_q.empty?
|
960
|
+
begin
|
961
|
+
data,target = @outbound_q.first
|
962
|
+
|
963
|
+
# This damn better be nonblocking.
|
964
|
+
io.send data.to_s, 0, target
|
965
|
+
|
966
|
+
@outbound_q.shift
|
967
|
+
rescue Errno::EAGAIN
|
968
|
+
# It's not been observed in testing that we ever get here.
|
969
|
+
# True to the definition, packets will be accepted and quietly dropped
|
970
|
+
# if the system is under pressure.
|
971
|
+
break
|
972
|
+
rescue EOFError, Errno::ECONNRESET
|
973
|
+
@close_scheduled = true
|
974
|
+
@outbound_q.clear
|
975
|
+
end
|
976
|
+
}
|
977
|
+
end
|
978
|
+
|
979
|
+
# Proper nonblocking I/O was added to Ruby 1.8.4 in May 2006.
|
980
|
+
# If we have it, then we can read multiple times safely to improve
|
981
|
+
# performance.
|
982
|
+
def eventable_read
|
983
|
+
begin
|
984
|
+
if io.respond_to?(:recvfrom_nonblock)
|
985
|
+
40.times {
|
986
|
+
data,@return_address = io.recvfrom_nonblock(16384)
|
987
|
+
EventMachine::event_callback uuid, ConnectionData, data
|
988
|
+
@return_address = nil
|
989
|
+
}
|
990
|
+
else
|
991
|
+
raise "unimplemented datagram-read operation on this Ruby"
|
992
|
+
end
|
993
|
+
rescue Errno::EAGAIN
|
994
|
+
# no-op
|
995
|
+
rescue Errno::ECONNRESET, EOFError
|
996
|
+
@close_scheduled = true
|
997
|
+
EventMachine::event_callback uuid, ConnectionUnbound, nil
|
998
|
+
end
|
999
|
+
|
1000
|
+
end
|
1001
|
+
|
1002
|
+
|
1003
|
+
def send_data data
|
1004
|
+
send_datagram data, @return_address
|
1005
|
+
end
|
1006
|
+
|
1007
|
+
end
|
1008
|
+
end
|
1009
|
+
|
1010
|
+
# load base EM api on top, now that we have the underlying pure ruby
|
1011
|
+
# implementation defined
|
1012
|
+
require 'eventmachine'
|
1013
|
+
|