eventmachine 0.12.8-x86-mswin32-60 → 0.12.10-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.
Files changed (66) hide show
  1. data/.gitignore +14 -13
  2. data/Rakefile +374 -264
  3. data/eventmachine.gemspec +4 -5
  4. data/ext/binder.cpp +125 -126
  5. data/ext/binder.h +46 -48
  6. data/ext/cmain.cpp +184 -42
  7. data/ext/cplusplus.cpp +202 -202
  8. data/ext/ed.cpp +242 -81
  9. data/ext/ed.h +39 -22
  10. data/ext/em.cpp +127 -108
  11. data/ext/em.h +27 -18
  12. data/ext/emwin.cpp +3 -3
  13. data/ext/eventmachine.h +49 -38
  14. data/ext/eventmachine_cpp.h +96 -96
  15. data/ext/extconf.rb +147 -132
  16. data/ext/fastfilereader/extconf.rb +82 -76
  17. data/ext/project.h +151 -140
  18. data/ext/rubymain.cpp +222 -103
  19. data/ext/ssl.cpp +460 -460
  20. data/ext/ssl.h +94 -94
  21. data/java/src/com/rubyeventmachine/EmReactor.java +570 -423
  22. data/java/src/com/rubyeventmachine/EventableChannel.java +69 -57
  23. data/java/src/com/rubyeventmachine/EventableDatagramChannel.java +189 -171
  24. data/java/src/com/rubyeventmachine/EventableSocketChannel.java +364 -244
  25. data/java/src/com/rubyeventmachine/{Application.java → application/Application.java} +194 -200
  26. data/java/src/com/rubyeventmachine/{Connection.java → application/Connection.java} +74 -74
  27. data/java/src/com/rubyeventmachine/{ConnectionFactory.java → application/ConnectionFactory.java} +36 -36
  28. data/java/src/com/rubyeventmachine/{DefaultConnectionFactory.java → application/DefaultConnectionFactory.java} +46 -46
  29. data/java/src/com/rubyeventmachine/{PeriodicTimer.java → application/PeriodicTimer.java} +38 -38
  30. data/java/src/com/rubyeventmachine/{Timer.java → application/Timer.java} +54 -54
  31. data/java/src/com/rubyeventmachine/tests/ApplicationTest.java +109 -108
  32. data/java/src/com/rubyeventmachine/tests/ConnectTest.java +148 -146
  33. data/java/src/com/rubyeventmachine/tests/TestDatagrams.java +53 -53
  34. data/java/src/com/rubyeventmachine/tests/TestServers.java +75 -74
  35. data/java/src/com/rubyeventmachine/tests/TestTimers.java +90 -89
  36. data/lib/em/connection.rb +71 -12
  37. data/lib/em/deferrable.rb +191 -186
  38. data/lib/em/protocols.rb +36 -35
  39. data/lib/em/protocols/httpclient2.rb +590 -582
  40. data/lib/em/protocols/line_and_text.rb +125 -126
  41. data/lib/em/protocols/linetext2.rb +161 -160
  42. data/lib/em/protocols/object_protocol.rb +45 -39
  43. data/lib/em/protocols/smtpclient.rb +357 -331
  44. data/lib/em/protocols/socks4.rb +66 -0
  45. data/lib/em/queue.rb +60 -60
  46. data/lib/em/timers.rb +56 -55
  47. data/lib/em/version.rb +1 -1
  48. data/lib/eventmachine.rb +125 -169
  49. data/lib/jeventmachine.rb +257 -142
  50. data/tasks/{cpp.rake → cpp.rake_example} +76 -76
  51. data/tests/test_attach.rb +125 -100
  52. data/tests/test_basic.rb +1 -2
  53. data/tests/test_connection_count.rb +34 -44
  54. data/tests/test_epoll.rb +0 -2
  55. data/tests/test_get_sock_opt.rb +30 -0
  56. data/tests/test_httpclient2.rb +3 -3
  57. data/tests/test_inactivity_timeout.rb +21 -1
  58. data/tests/test_ltp.rb +182 -188
  59. data/tests/test_next_tick.rb +0 -2
  60. data/tests/test_pause.rb +70 -0
  61. data/tests/test_pending_connect_timeout.rb +48 -0
  62. data/tests/test_ssl_args.rb +78 -67
  63. data/tests/test_timers.rb +162 -141
  64. metadata +13 -11
  65. data/tasks/project.rake +0 -79
  66. data/tasks/tests.rake +0 -193
data/ext/ssl.h CHANGED
@@ -1,94 +1,94 @@
1
- /*****************************************************************************
2
-
3
- $Id$
4
-
5
- File: ssl.h
6
- Date: 30Apr06
7
-
8
- Copyright (C) 2006-07 by Francis Cianfrocca. All Rights Reserved.
9
- Gmail: blackhedd
10
-
11
- This program is free software; you can redistribute it and/or modify
12
- it under the terms of either: 1) the GNU General Public License
13
- as published by the Free Software Foundation; either version 2 of the
14
- License, or (at your option) any later version; or 2) Ruby's License.
15
-
16
- See the file COPYING for complete licensing information.
17
-
18
- *****************************************************************************/
19
-
20
-
21
- #ifndef __SslBox__H_
22
- #define __SslBox__H_
23
-
24
-
25
-
26
-
27
- #ifdef WITH_SSL
28
-
29
- /******************
30
- class SslContext_t
31
- ******************/
32
-
33
- class SslContext_t
34
- {
35
- public:
36
- SslContext_t (bool is_server, const string &privkeyfile, const string &certchainfile);
37
- virtual ~SslContext_t();
38
-
39
- private:
40
- static bool bLibraryInitialized;
41
-
42
- private:
43
- bool bIsServer;
44
- SSL_CTX *pCtx;
45
-
46
- EVP_PKEY *PrivateKey;
47
- X509 *Certificate;
48
-
49
- friend class SslBox_t;
50
- };
51
-
52
-
53
- /**************
54
- class SslBox_t
55
- **************/
56
-
57
- class SslBox_t
58
- {
59
- public:
60
- SslBox_t (bool is_server, const string &privkeyfile, const string &certchainfile, bool verify_peer, const char *binding);
61
- virtual ~SslBox_t();
62
-
63
- int PutPlaintext (const char*, int);
64
- int GetPlaintext (char*, int);
65
-
66
- bool PutCiphertext (const char*, int);
67
- bool CanGetCiphertext();
68
- int GetCiphertext (char*, int);
69
- bool IsHandshakeCompleted() {return bHandshakeCompleted;}
70
-
71
- X509 *GetPeerCert();
72
-
73
- void Shutdown();
74
-
75
- protected:
76
- SslContext_t *Context;
77
-
78
- bool bVerifyPeer;
79
- bool bIsServer;
80
- bool bHandshakeCompleted;
81
- SSL *pSSL;
82
- BIO *pbioRead;
83
- BIO *pbioWrite;
84
-
85
- PageList OutboundQ;
86
- };
87
-
88
- extern "C" int ssl_verify_wrapper(int, X509_STORE_CTX*);
89
-
90
- #endif // WITH_SSL
91
-
92
-
93
- #endif // __SslBox__H_
94
-
1
+ /*****************************************************************************
2
+
3
+ $Id$
4
+
5
+ File: ssl.h
6
+ Date: 30Apr06
7
+
8
+ Copyright (C) 2006-07 by Francis Cianfrocca. All Rights Reserved.
9
+ Gmail: blackhedd
10
+
11
+ This program is free software; you can redistribute it and/or modify
12
+ it under the terms of either: 1) the GNU General Public License
13
+ as published by the Free Software Foundation; either version 2 of the
14
+ License, or (at your option) any later version; or 2) Ruby's License.
15
+
16
+ See the file COPYING for complete licensing information.
17
+
18
+ *****************************************************************************/
19
+
20
+
21
+ #ifndef __SslBox__H_
22
+ #define __SslBox__H_
23
+
24
+
25
+
26
+
27
+ #ifdef WITH_SSL
28
+
29
+ /******************
30
+ class SslContext_t
31
+ ******************/
32
+
33
+ class SslContext_t
34
+ {
35
+ public:
36
+ SslContext_t (bool is_server, const string &privkeyfile, const string &certchainfile);
37
+ virtual ~SslContext_t();
38
+
39
+ private:
40
+ static bool bLibraryInitialized;
41
+
42
+ private:
43
+ bool bIsServer;
44
+ SSL_CTX *pCtx;
45
+
46
+ EVP_PKEY *PrivateKey;
47
+ X509 *Certificate;
48
+
49
+ friend class SslBox_t;
50
+ };
51
+
52
+
53
+ /**************
54
+ class SslBox_t
55
+ **************/
56
+
57
+ class SslBox_t
58
+ {
59
+ public:
60
+ SslBox_t (bool is_server, const string &privkeyfile, const string &certchainfile, bool verify_peer, const unsigned long binding);
61
+ virtual ~SslBox_t();
62
+
63
+ int PutPlaintext (const char*, int);
64
+ int GetPlaintext (char*, int);
65
+
66
+ bool PutCiphertext (const char*, int);
67
+ bool CanGetCiphertext();
68
+ int GetCiphertext (char*, int);
69
+ bool IsHandshakeCompleted() {return bHandshakeCompleted;}
70
+
71
+ X509 *GetPeerCert();
72
+
73
+ void Shutdown();
74
+
75
+ protected:
76
+ SslContext_t *Context;
77
+
78
+ bool bIsServer;
79
+ bool bHandshakeCompleted;
80
+ bool bVerifyPeer;
81
+ SSL *pSSL;
82
+ BIO *pbioRead;
83
+ BIO *pbioWrite;
84
+
85
+ PageList OutboundQ;
86
+ };
87
+
88
+ extern "C" int ssl_verify_wrapper(int, X509_STORE_CTX*);
89
+
90
+ #endif // WITH_SSL
91
+
92
+
93
+ #endif // __SslBox__H_
94
+
@@ -1,423 +1,570 @@
1
- /**
2
- * $Id$
3
- *
4
- * Author:: Francis Cianfrocca (gmail: blackhedd)
5
- * Homepage:: http://rubyeventmachine.com
6
- * Date:: 15 Jul 2007
7
- *
8
- * See EventMachine and EventMachine::Connection for documentation and
9
- * usage examples.
10
- *
11
- *
12
- *----------------------------------------------------------------------------
13
- *
14
- * Copyright (C) 2006-07 by Francis Cianfrocca. All Rights Reserved.
15
- * Gmail: blackhedd
16
- *
17
- * This program is free software; you can redistribute it and/or modify
18
- * it under the terms of either: 1) the GNU General Public License
19
- * as published by the Free Software Foundation; either version 2 of the
20
- * License, or (at your option) any later version; or 2) Ruby's License.
21
- *
22
- * See the file COPYING for complete licensing information.
23
- *
24
- *---------------------------------------------------------------------------
25
- *
26
- *
27
- */
28
-
29
- package com.rubyeventmachine;
30
-
31
- import java.io.*;
32
- import java.nio.channels.*;
33
- import java.util.*;
34
- import java.nio.*;
35
- import java.net.*;
36
- import java.util.concurrent.atomic.*;
37
- import java.security.*;
38
-
39
- public class EmReactor {
40
-
41
-
42
- public final int EM_TIMER_FIRED = 100;
43
- public final int EM_CONNECTION_READ = 101;
44
- public final int EM_CONNECTION_UNBOUND = 102;
45
- public final int EM_CONNECTION_ACCEPTED = 103;
46
- public final int EM_CONNECTION_COMPLETED = 104;
47
- public final int EM_LOOPBREAK_SIGNAL = 105;
48
-
49
- private Selector mySelector;
50
- private TreeMap<Long, String> Timers;
51
- private TreeMap<String, EventableChannel> Connections;
52
- private TreeMap<String, ServerSocketChannel> Acceptors;
53
-
54
- private boolean bRunReactor;
55
- private long BindingIndex;
56
- private ByteBuffer EmptyByteBuffer;
57
- private AtomicBoolean loopBreaker;
58
- private ByteBuffer myReadBuffer;
59
- private int timerQuantum;
60
-
61
- public EmReactor() {
62
- Timers = new TreeMap<Long, String>();
63
- Connections = new TreeMap<String, EventableChannel>();
64
- Acceptors = new TreeMap<String, ServerSocketChannel>();
65
-
66
- BindingIndex = 100000;
67
- EmptyByteBuffer = ByteBuffer.allocate(0);
68
- loopBreaker = new AtomicBoolean();
69
- loopBreaker.set(false);
70
- myReadBuffer = ByteBuffer.allocate(32*1024); // don't use a direct buffer. Ruby doesn't seem to like them.
71
- timerQuantum = 98;
72
- }
73
-
74
- /**
75
- * Intended to be overridden in languages (like Ruby) that can't handle ByteBuffer. This is a stub.
76
- * Obsolete now that I figured out how to make Ruby deal with ByteBuffers.
77
- * @param sig
78
- * @param eventType
79
- * @param data
80
- */
81
- /*
82
- public void stringEventCallback (String sig, int eventType, String data) {
83
- System.out.println ("Default event callback: " + sig + " " + eventType + " " + data);
84
- }
85
- */
86
-
87
- /**
88
- * This is a no-op stub, intended to be overridden in user code.
89
- * @param sig
90
- * @param eventType
91
- * @param data
92
- */
93
- public void eventCallback (String sig, int eventType, ByteBuffer data) {
94
- System.out.println ("Default callback: "+sig+" "+eventType+" "+data);
95
- //stringEventCallback (sig, eventType, new String (data.array(), data.position(), data.limit()));
96
-
97
- }
98
-
99
- public void run() throws IOException {
100
- mySelector = Selector.open();
101
- bRunReactor = true;
102
-
103
- //int n = 0;
104
- for (;;) {
105
- //System.out.println ("loop "+ (n++));
106
- if (!bRunReactor) break;
107
- runLoopbreaks();
108
- if (!bRunReactor) break;
109
- runTimers();
110
- if (!bRunReactor) break;
111
- mySelector.select(timerQuantum);
112
-
113
- Iterator<SelectionKey> it = mySelector.selectedKeys().iterator();
114
- while (it.hasNext()) {
115
- SelectionKey k = it.next();
116
- it.remove();
117
-
118
- try {
119
- if (k.isAcceptable()) {
120
- ServerSocketChannel ss = (ServerSocketChannel) k.channel();
121
- SocketChannel sn;
122
- while ((sn = ss.accept()) != null) {
123
- sn.configureBlocking(false);
124
- String b = createBinding();
125
- EventableSocketChannel ec = new EventableSocketChannel (sn, b, mySelector);
126
- Connections.put(b, ec);
127
- eventCallback ((String)k.attachment(), EM_CONNECTION_ACCEPTED, ByteBuffer.wrap(b.getBytes()));
128
- }
129
- }
130
-
131
- if (k.isReadable()) {
132
- EventableChannel ec = (EventableChannel)k.attachment();
133
- myReadBuffer.clear();
134
- ec.readInboundData (myReadBuffer);
135
- myReadBuffer.flip();
136
- String b = ec.getBinding();
137
- if (myReadBuffer.limit() > 0) {
138
- eventCallback (b, EM_CONNECTION_READ, myReadBuffer);
139
- }
140
- else {
141
- eventCallback (b, EM_CONNECTION_UNBOUND, EmptyByteBuffer);
142
- Connections.remove(b);
143
- k.channel().close();
144
- }
145
- /*
146
- System.out.println ("READABLE");
147
- SocketChannel sn = (SocketChannel) k.channel();
148
- //ByteBuffer bb = ByteBuffer.allocate(16 * 1024);
149
- // Obviously not thread-safe, since we're using the same buffer for every connection.
150
- // This should minimize the production of garbage, though.
151
- // TODO, we need somehow to make a call to the EventableChannel, so we can pass the
152
- // inbound data through an SSLEngine. Hope that won't break the strategy of using one
153
- // global read-buffer.
154
- myReadBuffer.clear();
155
- int r = sn.read(myReadBuffer);
156
- if (r > 0) {
157
- myReadBuffer.flip();
158
- //bb = ((EventableChannel)k.attachment()).dispatchInboundData (bb);
159
- eventCallback (((EventableChannel)k.attachment()).getBinding(), EM_CONNECTION_READ, myReadBuffer);
160
- }
161
- else {
162
- // TODO. Figure out if a socket that selects readable can ever return 0 bytes
163
- // without it being indicative of an error condition. If Java is like C, the answer is no.
164
- String b = ((EventableChannel)k.attachment()).getBinding();
165
- eventCallback (b, EM_CONNECTION_UNBOUND, EmptyByteBuffer);
166
- Connections.remove(b);
167
- sn.close();
168
- }
169
- */
170
- }
171
-
172
-
173
- if (k.isWritable()) {
174
- EventableChannel ec = (EventableChannel)k.attachment();
175
- if (!ec.writeOutboundData()) {
176
- eventCallback (ec.getBinding(), EM_CONNECTION_UNBOUND, EmptyByteBuffer);
177
- Connections.remove (ec.getBinding());
178
- k.channel().close();
179
- }
180
- }
181
-
182
- if (k.isConnectable()) {
183
- EventableSocketChannel ec = (EventableSocketChannel)k.attachment();
184
- if (ec.finishConnecting()) {
185
- eventCallback (ec.getBinding(), EM_CONNECTION_COMPLETED, EmptyByteBuffer);
186
- }
187
- else {
188
- Connections.remove (ec.getBinding());
189
- k.channel().close();
190
- eventCallback (ec.getBinding(), EM_CONNECTION_UNBOUND, EmptyByteBuffer);
191
- }
192
- }
193
- }
194
- catch (CancelledKeyException e) {
195
- // No-op. We can come here if a read-handler closes a socket before we fall through
196
- // to call isWritable.
197
- }
198
-
199
- }
200
- }
201
-
202
- close();
203
- }
204
-
205
- void close() throws IOException {
206
- mySelector.close();
207
- mySelector = null;
208
-
209
- // run down open connections and sockets.
210
- Iterator<ServerSocketChannel> i = Acceptors.values().iterator();
211
- while (i.hasNext()) {
212
- i.next().close();
213
- }
214
-
215
- Iterator<EventableChannel> i2 = Connections.values().iterator();
216
- while (i2.hasNext())
217
- i2.next().close();
218
- }
219
-
220
- void runLoopbreaks() {
221
- if (loopBreaker.getAndSet(false)) {
222
- eventCallback ("", EM_LOOPBREAK_SIGNAL, EmptyByteBuffer);
223
- }
224
- }
225
-
226
- public void stop() {
227
- bRunReactor = false;
228
- signalLoopbreak();
229
- }
230
-
231
- void runTimers() {
232
- long now = new Date().getTime();
233
- while (!Timers.isEmpty()) {
234
- long k = Timers.firstKey();
235
- //System.out.println (k - now);
236
- if (k > now)
237
- break;
238
- String s = Timers.remove(k);
239
- eventCallback ("", EM_TIMER_FIRED, ByteBuffer.wrap(s.getBytes()));
240
- }
241
- }
242
-
243
- public String installOneshotTimer (int milliseconds) {
244
- BindingIndex++;
245
- String s = createBinding();
246
- Timers.put(new Date().getTime() + milliseconds, s);
247
- return s;
248
- }
249
-
250
- public String startTcpServer (SocketAddress sa) throws EmReactorException {
251
- try {
252
- ServerSocketChannel server = ServerSocketChannel.open();
253
- server.configureBlocking(false);
254
- server.socket().bind (sa);
255
- String s = createBinding();
256
- Acceptors.put(s, server);
257
- server.register(mySelector, SelectionKey.OP_ACCEPT, s);
258
- return s;
259
- } catch (IOException e) {
260
- // TODO, should parameterize this exception better.
261
- throw new EmReactorException ("unable to open socket acceptor");
262
- }
263
- }
264
-
265
- public String startTcpServer (String address, int port) throws EmReactorException {
266
- return startTcpServer (new InetSocketAddress (address, port));
267
- /*
268
- ServerSocketChannel server = ServerSocketChannel.open();
269
- server.configureBlocking(false);
270
- server.socket().bind(new java.net.InetSocketAddress(address, port));
271
- String s = createBinding();
272
- Acceptors.put(s, server);
273
- server.register(mySelector, SelectionKey.OP_ACCEPT, s);
274
- return s;
275
- */
276
- }
277
-
278
- public void stopTcpServer (String signature) throws IOException {
279
- ServerSocketChannel server = Acceptors.remove(signature);
280
- if (server != null)
281
- server.close();
282
- else
283
- throw new RuntimeException ("failed to close unknown acceptor");
284
- }
285
-
286
-
287
- public String openUdpSocket (String address, int port) throws IOException {
288
- return openUdpSocket (new InetSocketAddress (address, port));
289
- }
290
- /**
291
- *
292
- * @param address
293
- * @param port
294
- * @return
295
- * @throws IOException
296
- */
297
- public String openUdpSocket (InetSocketAddress address) throws IOException {
298
- // TODO, don't throw an exception out of here.
299
- DatagramChannel dg = DatagramChannel.open();
300
- dg.configureBlocking(false);
301
- dg.socket().bind(address);
302
- String b = createBinding();
303
- EventableChannel ec = new EventableDatagramChannel (dg, b, mySelector);
304
- dg.register(mySelector, SelectionKey.OP_READ, ec);
305
- Connections.put(b, ec);
306
- return b;
307
- }
308
-
309
- public void sendData (String sig, ByteBuffer bb) throws IOException {
310
- (Connections.get(sig)).scheduleOutboundData( bb );
311
- }
312
- public void sendData (String sig, byte[] data) throws IOException {
313
- sendData (sig, ByteBuffer.wrap(data));
314
- //(Connections.get(sig)).scheduleOutboundData( ByteBuffer.wrap(data.getBytes()));
315
- }
316
- public void setCommInactivityTimeout (String sig, long mills) {
317
- (Connections.get(sig)).setCommInactivityTimeout (mills);
318
- }
319
-
320
- /**
321
- *
322
- * @param sig
323
- * @param data
324
- * @param length
325
- * @param recipAddress
326
- * @param recipPort
327
- */
328
- public void sendDatagram (String sig, String data, int length, String recipAddress, int recipPort) {
329
- sendDatagram (sig, ByteBuffer.wrap(data.getBytes()), recipAddress, recipPort);
330
- }
331
-
332
- /**
333
- *
334
- * @param sig
335
- * @param bb
336
- * @param recipAddress
337
- * @param recipPort
338
- */
339
- public void sendDatagram (String sig, ByteBuffer bb, String recipAddress, int recipPort) {
340
- (Connections.get(sig)).scheduleOutboundDatagram( bb, recipAddress, recipPort);
341
- }
342
-
343
- /**
344
- *
345
- * @param address
346
- * @param port
347
- * @return
348
- * @throws ClosedChannelException
349
- */
350
- public String connectTcpServer (String address, int port) throws ClosedChannelException {
351
- return connectTcpServer(null, 0, address, port);
352
- }
353
-
354
- /**
355
- *
356
- * @param bindAddr
357
- * @param bindPort
358
- * @param address
359
- * @param port
360
- * @return
361
- * @throws ClosedChannelException
362
- */
363
- public String connectTcpServer (String bindAddr, int bindPort, String address, int port) throws ClosedChannelException {
364
- String b = createBinding();
365
-
366
- try {
367
- SocketChannel sc = SocketChannel.open();
368
- sc.configureBlocking(false);
369
- if (bindAddr != null)
370
- sc.socket().bind(new InetSocketAddress (bindAddr, bindPort));
371
-
372
- EventableSocketChannel ec = new EventableSocketChannel (sc, b, mySelector);
373
-
374
- if (sc.connect (new InetSocketAddress (address, port))) {
375
- // Connection returned immediately. Can happen with localhost connections.
376
- // WARNING, this code is untested due to lack of available test conditions.
377
- // Ought to be be able to come here from a localhost connection, but that
378
- // doesn't happen on Linux. (Maybe on FreeBSD?)
379
- // The reason for not handling this until we can test it is that we
380
- // really need to return from this function WITHOUT triggering any EM events.
381
- // That's because until the user code has seen the signature we generated here,
382
- // it won't be able to properly dispatch them. The C++ EM deals with this
383
- // by setting pending mode as a flag in ALL eventable descriptors and making
384
- // the descriptor select for writable. Then, it can send UNBOUND and
385
- // CONNECTION_COMPLETED on the next pass through the loop, because writable will
386
- // fire.
387
- throw new RuntimeException ("immediate-connect unimplemented");
388
- }
389
- else {
390
- Connections.put (b, ec);
391
- ec.setConnectPending();
392
- }
393
- } catch (IOException e) {
394
- // Can theoretically come here if a connect failure can be determined immediately.
395
- // I don't know how to make that happen for testing purposes.
396
- throw new RuntimeException ("immediate-connect unimplemented");
397
- }
398
- return b;
399
- }
400
-
401
- public void closeConnection (String sig, boolean afterWriting) throws ClosedChannelException {
402
- Connections.get(sig).scheduleClose (afterWriting);
403
- }
404
-
405
- String createBinding() {
406
- return new String ("BND_" + (++BindingIndex));
407
- }
408
-
409
- public void signalLoopbreak() {
410
- loopBreaker.set(true);
411
- mySelector.wakeup();
412
- }
413
-
414
- public void startTls (String sig) throws NoSuchAlgorithmException, KeyManagementException {
415
- Connections.get(sig).startTls();
416
- }
417
-
418
- public void setTimerQuantum (int mills) {
419
- if (mills < 5 || mills > 2500)
420
- throw new RuntimeException ("attempt to set invalid timer-quantum value: "+mills);
421
- timerQuantum = mills;
422
- }
423
- }
1
+ /**
2
+ * $Id$
3
+ *
4
+ * Author:: Francis Cianfrocca (gmail: blackhedd)
5
+ * Homepage:: http://rubyeventmachine.com
6
+ * Date:: 15 Jul 2007
7
+ *
8
+ * See EventMachine and EventMachine::Connection for documentation and
9
+ * usage examples.
10
+ *
11
+ *
12
+ *----------------------------------------------------------------------------
13
+ *
14
+ * Copyright (C) 2006-07 by Francis Cianfrocca. All Rights Reserved.
15
+ * Gmail: blackhedd
16
+ *
17
+ * This program is free software; you can redistribute it and/or modify
18
+ * it under the terms of either: 1) the GNU General Public License
19
+ * as published by the Free Software Foundation; either version 2 of the
20
+ * License, or (at your option) any later version; or 2) Ruby's License.
21
+ *
22
+ * See the file COPYING for complete licensing information.
23
+ *
24
+ *---------------------------------------------------------------------------
25
+ *
26
+ *
27
+ */
28
+
29
+ package com.rubyeventmachine;
30
+
31
+ import java.io.*;
32
+ import java.nio.channels.*;
33
+ import java.util.*;
34
+ import java.nio.*;
35
+ import java.net.*;
36
+ import java.util.concurrent.atomic.*;
37
+ import java.security.*;
38
+
39
+ public class EmReactor {
40
+ public final int EM_TIMER_FIRED = 100;
41
+ public final int EM_CONNECTION_READ = 101;
42
+ public final int EM_CONNECTION_UNBOUND = 102;
43
+ public final int EM_CONNECTION_ACCEPTED = 103;
44
+ public final int EM_CONNECTION_COMPLETED = 104;
45
+ public final int EM_LOOPBREAK_SIGNAL = 105;
46
+ public final int EM_CONNECTION_NOTIFY_READABLE = 106;
47
+ public final int EM_CONNECTION_NOTIFY_WRITABLE = 107;
48
+ public final int EM_SSL_HANDSHAKE_COMPLETED = 108;
49
+ public final int EM_SSL_VERIFY = 109;
50
+ public final int EM_PROXY_TARGET_UNBOUND = 110;
51
+
52
+ private Selector mySelector;
53
+ private TreeMap<Long, ArrayList<Long>> Timers;
54
+ private HashMap<Long, EventableChannel> Connections;
55
+ private HashMap<Long, ServerSocketChannel> Acceptors;
56
+ private ArrayList<Long> NewConnections;
57
+ private ArrayList<Long> UnboundConnections;
58
+ private ArrayList<EventableSocketChannel> DetachedConnections;
59
+
60
+ private boolean bRunReactor;
61
+ private long BindingIndex;
62
+ private AtomicBoolean loopBreaker;
63
+ private ByteBuffer myReadBuffer;
64
+ private int timerQuantum;
65
+
66
+ public EmReactor() {
67
+ Timers = new TreeMap<Long, ArrayList<Long>>();
68
+ Connections = new HashMap<Long, EventableChannel>();
69
+ Acceptors = new HashMap<Long, ServerSocketChannel>();
70
+ NewConnections = new ArrayList<Long>();
71
+ UnboundConnections = new ArrayList<Long>();
72
+ DetachedConnections = new ArrayList<EventableSocketChannel>();
73
+
74
+ BindingIndex = 0;
75
+ loopBreaker = new AtomicBoolean();
76
+ loopBreaker.set(false);
77
+ myReadBuffer = ByteBuffer.allocate(32*1024); // don't use a direct buffer. Ruby doesn't seem to like them.
78
+ timerQuantum = 98;
79
+ }
80
+
81
+ /**
82
+ * This is a no-op stub, intended to be overridden in user code.
83
+ */
84
+ public void eventCallback (long sig, int eventType, ByteBuffer data, long data2) {
85
+ System.out.println ("Default callback: "+sig+" "+eventType+" "+data+" "+data2);
86
+ }
87
+ public void eventCallback (long sig, int eventType, ByteBuffer data) {
88
+ eventCallback (sig, eventType, data, 0);
89
+ }
90
+
91
+
92
+ public void run() {
93
+ try {
94
+ mySelector = Selector.open();
95
+ bRunReactor = true;
96
+ } catch (IOException e) {
97
+ throw new RuntimeException ("Could not open selector", e);
98
+ }
99
+
100
+ while (bRunReactor) {
101
+ runLoopbreaks();
102
+ if (!bRunReactor) break;
103
+
104
+ runTimers();
105
+ if (!bRunReactor) break;
106
+
107
+ removeUnboundConnections();
108
+ checkIO();
109
+ addNewConnections();
110
+ processIO();
111
+ }
112
+
113
+ close();
114
+ }
115
+
116
+ void addNewConnections() {
117
+ ListIterator<EventableSocketChannel> iter = DetachedConnections.listIterator(0);
118
+ while (iter.hasNext()) {
119
+ EventableSocketChannel ec = iter.next();
120
+ ec.cleanup();
121
+ }
122
+ DetachedConnections.clear();
123
+
124
+ ListIterator<Long> iter2 = NewConnections.listIterator(0);
125
+ while (iter2.hasNext()) {
126
+ long b = iter2.next();
127
+
128
+ EventableChannel ec = Connections.get(b);
129
+ if (ec != null) {
130
+ try {
131
+ ec.register();
132
+ } catch (ClosedChannelException e) {
133
+ UnboundConnections.add (ec.getBinding());
134
+ }
135
+ }
136
+ }
137
+ NewConnections.clear();
138
+ }
139
+
140
+ void removeUnboundConnections() {
141
+ ListIterator<Long> iter = UnboundConnections.listIterator(0);
142
+ while (iter.hasNext()) {
143
+ long b = iter.next();
144
+
145
+ EventableChannel ec = Connections.remove(b);
146
+ if (ec != null) {
147
+ eventCallback (b, EM_CONNECTION_UNBOUND, null);
148
+ ec.close();
149
+
150
+ EventableSocketChannel sc = (EventableSocketChannel) ec;
151
+ if (sc != null && sc.isAttached())
152
+ DetachedConnections.add (sc);
153
+ }
154
+ }
155
+ UnboundConnections.clear();
156
+ }
157
+
158
+ void checkIO() {
159
+ long timeout;
160
+
161
+ if (NewConnections.size() > 0) {
162
+ timeout = -1;
163
+ } else if (!Timers.isEmpty()) {
164
+ long now = new Date().getTime();
165
+ long k = Timers.firstKey();
166
+ long diff = k-now;
167
+
168
+ if (diff <= 0)
169
+ timeout = -1; // don't wait, just poll once
170
+ else
171
+ timeout = diff;
172
+ } else {
173
+ timeout = 0; // wait indefinitely
174
+ }
175
+
176
+ try {
177
+ if (timeout == -1)
178
+ mySelector.selectNow();
179
+ else
180
+ mySelector.select(timeout);
181
+ } catch (IOException e) {
182
+ e.printStackTrace();
183
+ }
184
+ }
185
+
186
+ void processIO() {
187
+ Iterator<SelectionKey> it = mySelector.selectedKeys().iterator();
188
+ while (it.hasNext()) {
189
+ SelectionKey k = it.next();
190
+ it.remove();
191
+
192
+ if (k.isConnectable())
193
+ isConnectable(k);
194
+
195
+ else if (k.isAcceptable())
196
+ isAcceptable(k);
197
+
198
+ else {
199
+ if (k.isWritable())
200
+ isWritable(k);
201
+
202
+ if (k.isReadable())
203
+ isReadable(k);
204
+ }
205
+ }
206
+ }
207
+
208
+ void isAcceptable (SelectionKey k) {
209
+ ServerSocketChannel ss = (ServerSocketChannel) k.channel();
210
+ SocketChannel sn;
211
+ long b;
212
+
213
+ for (int n = 0; n < 10; n++) {
214
+ try {
215
+ sn = ss.accept();
216
+ if (sn == null)
217
+ break;
218
+ } catch (IOException e) {
219
+ e.printStackTrace();
220
+ k.cancel();
221
+
222
+ ServerSocketChannel server = Acceptors.remove(k.attachment());
223
+ if (server != null)
224
+ try{ server.close(); } catch (IOException ex) {};
225
+ break;
226
+ }
227
+
228
+ try {
229
+ sn.configureBlocking(false);
230
+ } catch (IOException e) {
231
+ e.printStackTrace();
232
+ continue;
233
+ }
234
+
235
+ b = createBinding();
236
+ EventableSocketChannel ec = new EventableSocketChannel (sn, b, mySelector);
237
+ Connections.put (b, ec);
238
+ NewConnections.add (b);
239
+
240
+ eventCallback (((Long)k.attachment()).longValue(), EM_CONNECTION_ACCEPTED, null, b);
241
+ }
242
+ }
243
+
244
+ void isReadable (SelectionKey k) {
245
+ EventableChannel ec = (EventableChannel) k.attachment();
246
+ long b = ec.getBinding();
247
+
248
+ if (ec.isWatchOnly()) {
249
+ if (ec.isNotifyReadable())
250
+ eventCallback (b, EM_CONNECTION_NOTIFY_READABLE, null);
251
+ } else {
252
+ myReadBuffer.clear();
253
+
254
+ try {
255
+ ec.readInboundData (myReadBuffer);
256
+ myReadBuffer.flip();
257
+ if (myReadBuffer.limit() > 0)
258
+ eventCallback (b, EM_CONNECTION_READ, myReadBuffer);
259
+ } catch (IOException e) {
260
+ UnboundConnections.add (b);
261
+ }
262
+ }
263
+ }
264
+
265
+ void isWritable (SelectionKey k) {
266
+ EventableChannel ec = (EventableChannel) k.attachment();
267
+ long b = ec.getBinding();
268
+
269
+ if (ec.isWatchOnly()) {
270
+ if (ec.isNotifyWritable())
271
+ eventCallback (b, EM_CONNECTION_NOTIFY_WRITABLE, null);
272
+ }
273
+ else {
274
+ try {
275
+ if (!ec.writeOutboundData())
276
+ UnboundConnections.add (b);
277
+ } catch (IOException e) {
278
+ UnboundConnections.add (b);
279
+ }
280
+ }
281
+ }
282
+
283
+ void isConnectable (SelectionKey k) {
284
+ EventableSocketChannel ec = (EventableSocketChannel) k.attachment();
285
+ long b = ec.getBinding();
286
+
287
+ try {
288
+ if (ec.finishConnecting())
289
+ eventCallback (b, EM_CONNECTION_COMPLETED, null);
290
+ else
291
+ UnboundConnections.add (b);
292
+ } catch (IOException e) {
293
+ UnboundConnections.add (b);
294
+ }
295
+ }
296
+
297
+ void close() {
298
+ try {
299
+ if (mySelector != null)
300
+ mySelector.close();
301
+ } catch (IOException e) {}
302
+ mySelector = null;
303
+
304
+ // run down open connections and sockets.
305
+ Iterator<ServerSocketChannel> i = Acceptors.values().iterator();
306
+ while (i.hasNext()) {
307
+ try {
308
+ i.next().close();
309
+ } catch (IOException e) {}
310
+ }
311
+
312
+ // 29Sep09: We create an ArrayList of the existing connections, then iterate over
313
+ // that to call unbind on them. This is because an unbind can trigger a reconnect,
314
+ // which will add to the Connections HashMap, causing a ConcurrentModificationException.
315
+ // XXX: The correct behavior here would be to latch the various reactor methods to return
316
+ // immediately if the reactor is shutting down.
317
+ ArrayList<EventableChannel> conns = new ArrayList<EventableChannel>();
318
+ Iterator<EventableChannel> i2 = Connections.values().iterator();
319
+ while (i2.hasNext()) {
320
+ EventableChannel ec = i2.next();
321
+ if (ec != null) {
322
+ conns.add (ec);
323
+ }
324
+ }
325
+ Connections.clear();
326
+
327
+ ListIterator<EventableChannel> i3 = conns.listIterator(0);
328
+ while (i3.hasNext()) {
329
+ EventableChannel ec = i3.next();
330
+ eventCallback (ec.getBinding(), EM_CONNECTION_UNBOUND, null);
331
+ ec.close();
332
+
333
+ EventableSocketChannel sc = (EventableSocketChannel) ec;
334
+ if (sc != null && sc.isAttached())
335
+ DetachedConnections.add (sc);
336
+ }
337
+
338
+ ListIterator<EventableSocketChannel> i4 = DetachedConnections.listIterator(0);
339
+ while (i4.hasNext()) {
340
+ EventableSocketChannel ec = i4.next();
341
+ ec.cleanup();
342
+ }
343
+ DetachedConnections.clear();
344
+ }
345
+
346
+ void runLoopbreaks() {
347
+ if (loopBreaker.getAndSet(false)) {
348
+ eventCallback (0, EM_LOOPBREAK_SIGNAL, null);
349
+ }
350
+ }
351
+
352
+ public void stop() {
353
+ bRunReactor = false;
354
+ signalLoopbreak();
355
+ }
356
+
357
+ void runTimers() {
358
+ long now = new Date().getTime();
359
+ while (!Timers.isEmpty()) {
360
+ long k = Timers.firstKey();
361
+ if (k > now)
362
+ break;
363
+
364
+ ArrayList<Long> callbacks = Timers.get(k);
365
+ Timers.remove(k);
366
+
367
+ // Fire all timers at this timestamp
368
+ ListIterator<Long> iter = callbacks.listIterator(0);
369
+ while (iter.hasNext()) {
370
+ eventCallback (0, EM_TIMER_FIRED, null, iter.next().longValue());
371
+ }
372
+ }
373
+ }
374
+
375
+ public long installOneshotTimer (int milliseconds) {
376
+ long s = createBinding();
377
+ long deadline = new Date().getTime() + milliseconds;
378
+
379
+ if (Timers.containsKey(deadline)) {
380
+ Timers.get(deadline).add(s);
381
+ } else {
382
+ ArrayList<Long> callbacks = new ArrayList<Long>();
383
+ callbacks.add(s);
384
+ Timers.put(deadline, callbacks);
385
+ }
386
+
387
+ return s;
388
+ }
389
+
390
+ public long startTcpServer (SocketAddress sa) throws EmReactorException {
391
+ try {
392
+ ServerSocketChannel server = ServerSocketChannel.open();
393
+ server.configureBlocking(false);
394
+ server.socket().bind (sa);
395
+ long s = createBinding();
396
+ Acceptors.put(s, server);
397
+ server.register(mySelector, SelectionKey.OP_ACCEPT, s);
398
+ return s;
399
+ } catch (IOException e) {
400
+ throw new EmReactorException ("unable to open socket acceptor: " + e.toString());
401
+ }
402
+ }
403
+
404
+ public long startTcpServer (String address, int port) throws EmReactorException {
405
+ return startTcpServer (new InetSocketAddress (address, port));
406
+ }
407
+
408
+ public void stopTcpServer (long signature) throws IOException {
409
+ ServerSocketChannel server = Acceptors.remove(signature);
410
+ if (server != null)
411
+ server.close();
412
+ else
413
+ throw new RuntimeException ("failed to close unknown acceptor");
414
+ }
415
+
416
+ public long openUdpSocket (InetSocketAddress address) throws IOException {
417
+ // TODO, don't throw an exception out of here.
418
+ DatagramChannel dg = DatagramChannel.open();
419
+ dg.configureBlocking(false);
420
+ dg.socket().bind(address);
421
+ long b = createBinding();
422
+ EventableChannel ec = new EventableDatagramChannel (dg, b, mySelector);
423
+ dg.register(mySelector, SelectionKey.OP_READ, ec);
424
+ Connections.put(b, ec);
425
+ return b;
426
+ }
427
+
428
+ public long openUdpSocket (String address, int port) throws IOException {
429
+ return openUdpSocket (new InetSocketAddress (address, port));
430
+ }
431
+
432
+ public void sendData (long sig, ByteBuffer bb) throws IOException {
433
+ Connections.get(sig).scheduleOutboundData( bb );
434
+ }
435
+
436
+ public void sendData (long sig, byte[] data) throws IOException {
437
+ sendData (sig, ByteBuffer.wrap(data));
438
+ }
439
+
440
+ public void setCommInactivityTimeout (long sig, long mills) {
441
+ Connections.get(sig).setCommInactivityTimeout (mills);
442
+ }
443
+
444
+ public void sendDatagram (long sig, String data, int length, String recipAddress, int recipPort) {
445
+ sendDatagram (sig, ByteBuffer.wrap(data.getBytes()), recipAddress, recipPort);
446
+ }
447
+
448
+ public void sendDatagram (long sig, ByteBuffer bb, String recipAddress, int recipPort) {
449
+ (Connections.get(sig)).scheduleOutboundDatagram( bb, recipAddress, recipPort);
450
+ }
451
+
452
+ public long connectTcpServer (String address, int port) {
453
+ return connectTcpServer(null, 0, address, port);
454
+ }
455
+
456
+ public long connectTcpServer (String bindAddr, int bindPort, String address, int port) {
457
+ long b = createBinding();
458
+
459
+ try {
460
+ SocketChannel sc = SocketChannel.open();
461
+ sc.configureBlocking(false);
462
+ if (bindAddr != null)
463
+ sc.socket().bind(new InetSocketAddress (bindAddr, bindPort));
464
+
465
+ EventableSocketChannel ec = new EventableSocketChannel (sc, b, mySelector);
466
+
467
+ if (sc.connect (new InetSocketAddress (address, port))) {
468
+ // Connection returned immediately. Can happen with localhost connections.
469
+ // WARNING, this code is untested due to lack of available test conditions.
470
+ // Ought to be be able to come here from a localhost connection, but that
471
+ // doesn't happen on Linux. (Maybe on FreeBSD?)
472
+ // The reason for not handling this until we can test it is that we
473
+ // really need to return from this function WITHOUT triggering any EM events.
474
+ // That's because until the user code has seen the signature we generated here,
475
+ // it won't be able to properly dispatch them. The C++ EM deals with this
476
+ // by setting pending mode as a flag in ALL eventable descriptors and making
477
+ // the descriptor select for writable. Then, it can send UNBOUND and
478
+ // CONNECTION_COMPLETED on the next pass through the loop, because writable will
479
+ // fire.
480
+ throw new RuntimeException ("immediate-connect unimplemented");
481
+ }
482
+ else {
483
+ ec.setConnectPending();
484
+ Connections.put (b, ec);
485
+ NewConnections.add (b);
486
+ }
487
+ } catch (IOException e) {
488
+ // Can theoretically come here if a connect failure can be determined immediately.
489
+ // I don't know how to make that happen for testing purposes.
490
+ throw new RuntimeException ("immediate-connect unimplemented: " + e.toString());
491
+ }
492
+ return b;
493
+ }
494
+
495
+ public void closeConnection (long sig, boolean afterWriting) {
496
+ EventableChannel ec = Connections.get(sig);
497
+ if (ec != null)
498
+ if (ec.scheduleClose (afterWriting))
499
+ UnboundConnections.add (sig);
500
+ }
501
+
502
+ long createBinding() {
503
+ return ++BindingIndex;
504
+ }
505
+
506
+ public void signalLoopbreak() {
507
+ loopBreaker.set(true);
508
+ if (mySelector != null)
509
+ mySelector.wakeup();
510
+ }
511
+
512
+ public void startTls (long sig) throws NoSuchAlgorithmException, KeyManagementException {
513
+ Connections.get(sig).startTls();
514
+ }
515
+
516
+ public void setTimerQuantum (int mills) {
517
+ if (mills < 5 || mills > 2500)
518
+ throw new RuntimeException ("attempt to set invalid timer-quantum value: "+mills);
519
+ timerQuantum = mills;
520
+ }
521
+
522
+ public Object[] getPeerName (long sig) {
523
+ return Connections.get(sig).getPeerName();
524
+ }
525
+
526
+ public long attachChannel (SocketChannel sc, boolean watch_mode) {
527
+ long b = createBinding();
528
+
529
+ EventableSocketChannel ec = new EventableSocketChannel (sc, b, mySelector);
530
+
531
+ ec.setAttached();
532
+ if (watch_mode)
533
+ ec.setWatchOnly();
534
+
535
+ Connections.put (b, ec);
536
+ NewConnections.add (b);
537
+
538
+ return b;
539
+ }
540
+
541
+ public SocketChannel detachChannel (long sig) {
542
+ EventableSocketChannel ec = (EventableSocketChannel) Connections.get (sig);
543
+ if (ec != null) {
544
+ UnboundConnections.add (sig);
545
+ return ec.getChannel();
546
+ } else {
547
+ return null;
548
+ }
549
+ }
550
+
551
+ public void setNotifyReadable (long sig, boolean mode) {
552
+ ((EventableSocketChannel) Connections.get(sig)).setNotifyReadable(mode);
553
+ }
554
+
555
+ public void setNotifyWritable (long sig, boolean mode) {
556
+ ((EventableSocketChannel) Connections.get(sig)).setNotifyWritable(mode);
557
+ }
558
+
559
+ public boolean isNotifyReadable (long sig) {
560
+ return Connections.get(sig).isNotifyReadable();
561
+ }
562
+
563
+ public boolean isNotifyWritable (long sig) {
564
+ return Connections.get(sig).isNotifyWritable();
565
+ }
566
+
567
+ public int getConnectionCount() {
568
+ return Connections.size() + Acceptors.size();
569
+ }
570
+ }