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.
Files changed (150) hide show
  1. data/.gitignore +2 -0
  2. data/Gemfile +1 -0
  3. data/README +80 -81
  4. data/Rakefile +7 -370
  5. data/docs/COPYING +60 -60
  6. data/docs/ChangeLog +211 -211
  7. data/docs/DEFERRABLES +246 -133
  8. data/docs/EPOLL +141 -141
  9. data/docs/GNU +281 -281
  10. data/docs/INSTALL +13 -13
  11. data/docs/KEYBOARD +42 -38
  12. data/docs/LEGAL +25 -25
  13. data/docs/LIGHTWEIGHT_CONCURRENCY +130 -70
  14. data/docs/PURE_RUBY +75 -75
  15. data/docs/RELEASE_NOTES +94 -94
  16. data/docs/SMTP +4 -2
  17. data/docs/SPAWNED_PROCESSES +148 -89
  18. data/docs/TODO +8 -8
  19. data/eventmachine.gemspec +19 -26
  20. data/examples/ex_channel.rb +42 -42
  21. data/examples/ex_queue.rb +2 -2
  22. data/examples/ex_tick_loop_array.rb +15 -0
  23. data/examples/ex_tick_loop_counter.rb +32 -0
  24. data/examples/helper.rb +1 -1
  25. data/ext/binder.cpp +0 -1
  26. data/ext/cmain.cpp +36 -25
  27. data/ext/ed.cpp +104 -113
  28. data/ext/ed.h +24 -30
  29. data/ext/em.cpp +349 -283
  30. data/ext/em.h +25 -29
  31. data/ext/eventmachine.h +5 -4
  32. data/ext/extconf.rb +58 -49
  33. data/ext/fastfilereader/extconf.rb +5 -3
  34. data/ext/fastfilereader/mapper.cpp +214 -214
  35. data/ext/fastfilereader/mapper.h +59 -59
  36. data/ext/fastfilereader/rubymain.cpp +127 -127
  37. data/ext/kb.cpp +1 -3
  38. data/ext/page.cpp +107 -107
  39. data/ext/page.h +51 -51
  40. data/ext/pipe.cpp +9 -11
  41. data/ext/project.h +12 -8
  42. data/ext/rubymain.cpp +138 -104
  43. data/java/.classpath +8 -8
  44. data/java/.project +17 -17
  45. data/java/src/com/rubyeventmachine/EmReactor.java +1 -0
  46. data/java/src/com/rubyeventmachine/EmReactorException.java +40 -40
  47. data/lib/em/buftok.rb +138 -138
  48. data/lib/em/callback.rb +25 -25
  49. data/lib/em/channel.rb +1 -1
  50. data/lib/em/connection.rb +6 -1
  51. data/lib/em/deferrable.rb +16 -2
  52. data/lib/em/file_watch.rb +53 -53
  53. data/lib/em/future.rb +61 -61
  54. data/lib/em/iterator.rb +270 -0
  55. data/lib/em/messages.rb +66 -66
  56. data/lib/em/process_watch.rb +43 -43
  57. data/lib/em/protocols.rb +1 -1
  58. data/lib/em/protocols/header_and_content.rb +138 -138
  59. data/lib/em/protocols/httpclient.rb +267 -262
  60. data/lib/em/protocols/line_protocol.rb +28 -0
  61. data/lib/em/protocols/memcache.rb +322 -322
  62. data/lib/em/protocols/postgres3.rb +247 -247
  63. data/lib/em/protocols/saslauth.rb +175 -175
  64. data/lib/em/protocols/smtpserver.rb +640 -547
  65. data/lib/em/protocols/stomp.rb +200 -200
  66. data/lib/em/protocols/tcptest.rb +52 -52
  67. data/lib/{pr_eventmachine.rb → em/pure_ruby.rb} +1013 -1022
  68. data/lib/em/queue.rb +1 -0
  69. data/lib/em/spawnable.rb +85 -85
  70. data/lib/em/streamer.rb +130 -130
  71. data/lib/em/tick_loop.rb +85 -0
  72. data/lib/em/timers.rb +2 -1
  73. data/lib/em/version.rb +1 -1
  74. data/lib/eventmachine.rb +40 -84
  75. data/lib/jeventmachine.rb +2 -1
  76. data/lib/rubyeventmachine.rb +2 -0
  77. data/setup.rb +1585 -1585
  78. data/tasks/doc.rake +30 -0
  79. data/tasks/package.rake +85 -0
  80. data/tasks/test.rake +6 -0
  81. data/tests/client.crt +31 -31
  82. data/tests/client.key +51 -51
  83. data/tests/test_attach.rb +13 -3
  84. data/tests/test_basic.rb +60 -95
  85. data/tests/test_channel.rb +3 -2
  86. data/tests/test_defer.rb +49 -47
  87. data/tests/test_deferrable.rb +35 -0
  88. data/tests/test_error_handler.rb +35 -35
  89. data/tests/test_errors.rb +82 -82
  90. data/tests/test_exc.rb +55 -55
  91. data/tests/test_file_watch.rb +49 -49
  92. data/tests/test_futures.rb +198 -198
  93. data/tests/test_handler_check.rb +36 -36
  94. data/tests/test_hc.rb +190 -218
  95. data/tests/test_httpclient.rb +227 -218
  96. data/tests/test_httpclient2.rb +3 -2
  97. data/tests/test_inactivity_timeout.rb +3 -3
  98. data/tests/test_kb.rb +60 -60
  99. data/tests/test_ltp.rb +13 -5
  100. data/tests/test_ltp2.rb +317 -317
  101. data/tests/test_next_tick.rb +1 -1
  102. data/tests/test_object_protocol.rb +36 -36
  103. data/tests/test_pending_connect_timeout.rb +2 -2
  104. data/tests/test_process_watch.rb +50 -48
  105. data/tests/test_proxy_connection.rb +52 -0
  106. data/tests/test_pure.rb +134 -125
  107. data/tests/test_queue.rb +44 -44
  108. data/tests/test_running.rb +42 -42
  109. data/tests/test_sasl.rb +72 -72
  110. data/tests/test_send_file.rb +251 -242
  111. data/tests/test_servers.rb +76 -76
  112. data/tests/test_smtpclient.rb +83 -83
  113. data/tests/test_smtpserver.rb +85 -85
  114. data/tests/test_spawn.rb +322 -322
  115. data/tests/test_ssl_methods.rb +49 -49
  116. data/tests/test_ssl_verify.rb +82 -82
  117. data/tests/test_tick_loop.rb +59 -0
  118. data/tests/test_timers.rb +13 -15
  119. data/tests/test_ud.rb +36 -36
  120. data/tests/testem.rb +31 -31
  121. metadata +66 -51
  122. data/ext/cplusplus.cpp +0 -202
  123. data/ext/emwin.cpp +0 -300
  124. data/ext/emwin.h +0 -94
  125. data/ext/epoll.cpp +0 -26
  126. data/ext/epoll.h +0 -25
  127. data/ext/eventmachine_cpp.h +0 -96
  128. data/ext/files.cpp +0 -94
  129. data/ext/files.h +0 -65
  130. data/ext/sigs.cpp +0 -89
  131. data/ext/sigs.h +0 -32
  132. data/java/src/com/rubyeventmachine/application/Application.java +0 -194
  133. data/java/src/com/rubyeventmachine/application/Connection.java +0 -74
  134. data/java/src/com/rubyeventmachine/application/ConnectionFactory.java +0 -37
  135. data/java/src/com/rubyeventmachine/application/DefaultConnectionFactory.java +0 -46
  136. data/java/src/com/rubyeventmachine/application/PeriodicTimer.java +0 -38
  137. data/java/src/com/rubyeventmachine/application/Timer.java +0 -54
  138. data/java/src/com/rubyeventmachine/tests/ApplicationTest.java +0 -109
  139. data/java/src/com/rubyeventmachine/tests/ConnectTest.java +0 -148
  140. data/java/src/com/rubyeventmachine/tests/EMTest.java +0 -80
  141. data/java/src/com/rubyeventmachine/tests/TestDatagrams.java +0 -53
  142. data/java/src/com/rubyeventmachine/tests/TestServers.java +0 -75
  143. data/java/src/com/rubyeventmachine/tests/TestTimers.java +0 -90
  144. data/lib/evma.rb +0 -32
  145. data/lib/evma/callback.rb +0 -32
  146. data/lib/evma/container.rb +0 -75
  147. data/lib/evma/factory.rb +0 -77
  148. data/lib/evma/protocol.rb +0 -87
  149. data/lib/evma/reactor.rb +0 -48
  150. data/web/whatis +0 -7
@@ -1,322 +1,322 @@
1
- # $Id$
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
- #
27
-
28
-
29
- $:.unshift "../lib"
30
- require 'eventmachine'
31
- require 'test/unit'
32
-
33
-
34
-
35
- class TestSpawn < Test::Unit::TestCase
36
-
37
- # Spawn a process that simply stops the reactor.
38
- # Assert that the notification runs after the block that calls it.
39
- #
40
- def test_stop
41
- x = nil
42
- EM.run {
43
- s = EM.spawn {EM.stop}
44
- s.notify
45
- x = true
46
- }
47
- assert x
48
- end
49
-
50
-
51
- # Pass a parameter to a spawned process.
52
- #
53
- def test_parms
54
- val = 5
55
- EM.run {
56
- s = EM.spawn {|v| val *= v; EM.stop}
57
- s.notify 3
58
- }
59
- assert_equal( 15, val )
60
- end
61
-
62
- # Pass multiple parameters to a spawned process.
63
- #
64
- def test_multiparms
65
- val = 5
66
- EM.run {
67
- s = EM.spawn {|v1,v2| val *= (v1 + v2); EM.stop}
68
- s.notify 3,4
69
- }
70
- assert_equal( 35, val )
71
- end
72
-
73
-
74
- # This test demonstrates that a notification does not happen immediately,
75
- # but rather is scheduled sometime after the current code path completes.
76
- #
77
- def test_race
78
- x = 0
79
- EM.run {
80
- s = EM.spawn {x *= 2; EM.stop}
81
- s.notify
82
- x = 2
83
- }
84
- assert_equal( 4, x)
85
- end
86
-
87
-
88
- # Spawn a process and notify it 25 times to run fibonacci
89
- # on a pair of global variables.
90
- #
91
- def test_fibonacci
92
- x = 1
93
- y = 1
94
- EM.run {
95
- s = EM.spawn {x,y = y,x+y}
96
- 25.times {s.notify}
97
-
98
- t = EM.spawn {EM.stop}
99
- t.notify
100
- }
101
- assert_equal( 121393, x)
102
- assert_equal( 196418, y)
103
- end
104
-
105
- # This one spawns 25 distinct processes, and notifies each one once,
106
- # rather than notifying a single process 25 times.
107
- #
108
- def test_another_fibonacci
109
- x = 1
110
- y = 1
111
- EM.run {
112
- 25.times {
113
- s = EM.spawn {x,y = y,x+y}
114
- s.notify
115
- }
116
-
117
- t = EM.spawn {EM.stop}
118
- t.notify
119
- }
120
- assert_equal( 121393, x)
121
- assert_equal( 196418, y)
122
- end
123
-
124
-
125
- # Make a chain of processes that notify each other in turn
126
- # with intermediate fibonacci results. The final process in
127
- # the chain stops the loop and returns the result.
128
- #
129
- def test_fibonacci_chain
130
- a,b = nil
131
-
132
- EM.run {
133
- nextpid = EM.spawn {|x,y|
134
- a,b = x,y
135
- EM.stop
136
- }
137
-
138
- 25.times {
139
- n = nextpid
140
- nextpid = EM.spawn {|x,y| n.notify( y, x+y )}
141
- }
142
-
143
- nextpid.notify( 1, 1 )
144
- }
145
-
146
- assert_equal( 121393, a)
147
- assert_equal( 196418, b)
148
- end
149
-
150
-
151
- # EM#yield gives a spawed process to yield control to other processes
152
- # (in other words, to stop running), and to specify a different code block
153
- # that will run on its next notification.
154
- #
155
- def test_yield
156
- a = 0
157
- EM.run {
158
- n = EM.spawn {
159
- a += 10
160
- EM.yield {
161
- a += 20
162
- EM.yield {
163
- a += 30
164
- EM.stop
165
- }
166
- }
167
- }
168
- n.notify
169
- n.notify
170
- n.notify
171
- }
172
- assert_equal( 60, a )
173
- end
174
-
175
- # EM#yield_and_notify behaves like EM#yield, except that it also notifies the
176
- # yielding process. This may sound trivial, since the yield block will run very
177
- # shortly after with no action by the program, but this actually can be very useful,
178
- # because it causes the reactor core to execute once before the yielding process
179
- # gets control back. So it can be used to allow heavily-used network connections
180
- # to clear buffers, or allow other processes to process their notifications.
181
- #
182
- # Notice in this test code that only a simple notify is needed at the bottom
183
- # of the initial block. Even so, all of the yielded blocks will execute.
184
- #
185
- def test_yield_and_notify
186
- a = 0
187
- EM.run {
188
- n = EM.spawn {
189
- a += 10
190
- EM.yield_and_notify {
191
- a += 20
192
- EM.yield_and_notify {
193
- a += 30
194
- EM.stop
195
- }
196
- }
197
- }
198
- n.notify
199
- }
200
- assert_equal( 60, a )
201
- end
202
-
203
- # resume is an alias for notify.
204
- #
205
- def test_resume
206
- EM.run {
207
- n = EM.spawn {EM.stop}
208
- n.resume
209
- }
210
- assert true
211
- end
212
-
213
- # run is an idiomatic alias for notify.
214
- #
215
- def test_run
216
- EM.run {
217
- (EM.spawn {EM.stop}).run
218
- }
219
- assert true
220
- end
221
-
222
-
223
- # Clones the ping-pong example from the Erlang tutorial, in much less code.
224
- # Illustrates that a spawned block executes in the context of a SpawnableObject.
225
- # (Meaning, we can pass self as a parameter to another process that can then
226
- # notify us.)
227
- #
228
- def test_ping_pong
229
- n_pongs = 0
230
- EM.run {
231
- pong = EM.spawn {|x, ping|
232
- n_pongs += 1
233
- ping.notify( x-1 )
234
- }
235
- ping = EM.spawn {|x|
236
- if x > 0
237
- pong.notify x, self
238
- else
239
- EM.stop
240
- end
241
- }
242
- ping.notify 3
243
- }
244
- assert_equal( 3, n_pongs )
245
- end
246
-
247
- # Illustrates that you can call notify inside a notification, and it will cause
248
- # the currently-executing process to be re-notified. Of course, the new notification
249
- # won't run until sometime after the current one completes.
250
- #
251
- def test_self_notify
252
- n = 0
253
- EM.run {
254
- pid = EM.spawn {|x|
255
- if x > 0
256
- n += x
257
- notify( x-1 )
258
- else
259
- EM.stop
260
- end
261
- }
262
- pid.notify 3
263
- }
264
- assert_equal( 6, n )
265
- end
266
-
267
-
268
- # Illustrates that the block passed to #spawn executes in the context of a
269
- # SpawnedProcess object, NOT in the local context. This can often be deceptive.
270
- #
271
- class BlockScopeTest
272
- attr_reader :var
273
- def run
274
- # The following line correctly raises a NameError.
275
- # The problem is that the programmer expected the spawned block to
276
- # execute in the local context, but it doesn't.
277
- #
278
- # (EM.spawn { do_something }).notify ### NO! BAD!
279
-
280
-
281
-
282
- # The following line correctly passes self as a parameter to the
283
- # notified process.
284
- #
285
- (EM.spawn {|obj| obj.do_something }).notify(self)
286
-
287
-
288
-
289
- # Here's another way to do it. This works because "myself" is bound
290
- # in the local scope, unlike "self," so the spawned block sees it.
291
- #
292
- myself = self
293
- (EM.spawn { myself.do_something }).notify
294
-
295
-
296
-
297
- # And we end the loop.
298
- # This is a tangential point, but observe that #notify never blocks.
299
- # It merely appends a message to the internal queue of a spawned process
300
- # and returns. As it turns out, the reactor processes notifications for ALL
301
- # spawned processes in the order that #notify is called. So there is a
302
- # reasonable expectation that the process which stops the reactor will
303
- # execute after the previous ones in this method. HOWEVER, this is NOT
304
- # a documented behavior and is subject to change.
305
- #
306
- (EM.spawn {EM.stop}).notify
307
- end
308
- def do_something
309
- @var ||= 0
310
- @var += 100
311
- end
312
- end
313
-
314
- def test_block_scope
315
- bs = BlockScopeTest.new
316
- EM.run {
317
- bs.run
318
- }
319
- assert_equal( 200, bs.var )
320
- end
321
-
322
- end
1
+ # $Id$
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
+ #
27
+
28
+
29
+ $:.unshift "../lib"
30
+ require 'eventmachine'
31
+ require 'test/unit'
32
+
33
+
34
+
35
+ class TestSpawn < Test::Unit::TestCase
36
+
37
+ # Spawn a process that simply stops the reactor.
38
+ # Assert that the notification runs after the block that calls it.
39
+ #
40
+ def test_stop
41
+ x = nil
42
+ EM.run {
43
+ s = EM.spawn {EM.stop}
44
+ s.notify
45
+ x = true
46
+ }
47
+ assert x
48
+ end
49
+
50
+
51
+ # Pass a parameter to a spawned process.
52
+ #
53
+ def test_parms
54
+ val = 5
55
+ EM.run {
56
+ s = EM.spawn {|v| val *= v; EM.stop}
57
+ s.notify 3
58
+ }
59
+ assert_equal( 15, val )
60
+ end
61
+
62
+ # Pass multiple parameters to a spawned process.
63
+ #
64
+ def test_multiparms
65
+ val = 5
66
+ EM.run {
67
+ s = EM.spawn {|v1,v2| val *= (v1 + v2); EM.stop}
68
+ s.notify 3,4
69
+ }
70
+ assert_equal( 35, val )
71
+ end
72
+
73
+
74
+ # This test demonstrates that a notification does not happen immediately,
75
+ # but rather is scheduled sometime after the current code path completes.
76
+ #
77
+ def test_race
78
+ x = 0
79
+ EM.run {
80
+ s = EM.spawn {x *= 2; EM.stop}
81
+ s.notify
82
+ x = 2
83
+ }
84
+ assert_equal( 4, x)
85
+ end
86
+
87
+
88
+ # Spawn a process and notify it 25 times to run fibonacci
89
+ # on a pair of global variables.
90
+ #
91
+ def test_fibonacci
92
+ x = 1
93
+ y = 1
94
+ EM.run {
95
+ s = EM.spawn {x,y = y,x+y}
96
+ 25.times {s.notify}
97
+
98
+ t = EM.spawn {EM.stop}
99
+ t.notify
100
+ }
101
+ assert_equal( 121393, x)
102
+ assert_equal( 196418, y)
103
+ end
104
+
105
+ # This one spawns 25 distinct processes, and notifies each one once,
106
+ # rather than notifying a single process 25 times.
107
+ #
108
+ def test_another_fibonacci
109
+ x = 1
110
+ y = 1
111
+ EM.run {
112
+ 25.times {
113
+ s = EM.spawn {x,y = y,x+y}
114
+ s.notify
115
+ }
116
+
117
+ t = EM.spawn {EM.stop}
118
+ t.notify
119
+ }
120
+ assert_equal( 121393, x)
121
+ assert_equal( 196418, y)
122
+ end
123
+
124
+
125
+ # Make a chain of processes that notify each other in turn
126
+ # with intermediate fibonacci results. The final process in
127
+ # the chain stops the loop and returns the result.
128
+ #
129
+ def test_fibonacci_chain
130
+ a,b = nil
131
+
132
+ EM.run {
133
+ nextpid = EM.spawn {|x,y|
134
+ a,b = x,y
135
+ EM.stop
136
+ }
137
+
138
+ 25.times {
139
+ n = nextpid
140
+ nextpid = EM.spawn {|x,y| n.notify( y, x+y )}
141
+ }
142
+
143
+ nextpid.notify( 1, 1 )
144
+ }
145
+
146
+ assert_equal( 121393, a)
147
+ assert_equal( 196418, b)
148
+ end
149
+
150
+
151
+ # EM#yield gives a spawed process to yield control to other processes
152
+ # (in other words, to stop running), and to specify a different code block
153
+ # that will run on its next notification.
154
+ #
155
+ def test_yield
156
+ a = 0
157
+ EM.run {
158
+ n = EM.spawn {
159
+ a += 10
160
+ EM.yield {
161
+ a += 20
162
+ EM.yield {
163
+ a += 30
164
+ EM.stop
165
+ }
166
+ }
167
+ }
168
+ n.notify
169
+ n.notify
170
+ n.notify
171
+ }
172
+ assert_equal( 60, a )
173
+ end
174
+
175
+ # EM#yield_and_notify behaves like EM#yield, except that it also notifies the
176
+ # yielding process. This may sound trivial, since the yield block will run very
177
+ # shortly after with no action by the program, but this actually can be very useful,
178
+ # because it causes the reactor core to execute once before the yielding process
179
+ # gets control back. So it can be used to allow heavily-used network connections
180
+ # to clear buffers, or allow other processes to process their notifications.
181
+ #
182
+ # Notice in this test code that only a simple notify is needed at the bottom
183
+ # of the initial block. Even so, all of the yielded blocks will execute.
184
+ #
185
+ def test_yield_and_notify
186
+ a = 0
187
+ EM.run {
188
+ n = EM.spawn {
189
+ a += 10
190
+ EM.yield_and_notify {
191
+ a += 20
192
+ EM.yield_and_notify {
193
+ a += 30
194
+ EM.stop
195
+ }
196
+ }
197
+ }
198
+ n.notify
199
+ }
200
+ assert_equal( 60, a )
201
+ end
202
+
203
+ # resume is an alias for notify.
204
+ #
205
+ def test_resume
206
+ EM.run {
207
+ n = EM.spawn {EM.stop}
208
+ n.resume
209
+ }
210
+ assert true
211
+ end
212
+
213
+ # run is an idiomatic alias for notify.
214
+ #
215
+ def test_run
216
+ EM.run {
217
+ (EM.spawn {EM.stop}).run
218
+ }
219
+ assert true
220
+ end
221
+
222
+
223
+ # Clones the ping-pong example from the Erlang tutorial, in much less code.
224
+ # Illustrates that a spawned block executes in the context of a SpawnableObject.
225
+ # (Meaning, we can pass self as a parameter to another process that can then
226
+ # notify us.)
227
+ #
228
+ def test_ping_pong
229
+ n_pongs = 0
230
+ EM.run {
231
+ pong = EM.spawn {|x, ping|
232
+ n_pongs += 1
233
+ ping.notify( x-1 )
234
+ }
235
+ ping = EM.spawn {|x|
236
+ if x > 0
237
+ pong.notify x, self
238
+ else
239
+ EM.stop
240
+ end
241
+ }
242
+ ping.notify 3
243
+ }
244
+ assert_equal( 3, n_pongs )
245
+ end
246
+
247
+ # Illustrates that you can call notify inside a notification, and it will cause
248
+ # the currently-executing process to be re-notified. Of course, the new notification
249
+ # won't run until sometime after the current one completes.
250
+ #
251
+ def test_self_notify
252
+ n = 0
253
+ EM.run {
254
+ pid = EM.spawn {|x|
255
+ if x > 0
256
+ n += x
257
+ notify( x-1 )
258
+ else
259
+ EM.stop
260
+ end
261
+ }
262
+ pid.notify 3
263
+ }
264
+ assert_equal( 6, n )
265
+ end
266
+
267
+
268
+ # Illustrates that the block passed to #spawn executes in the context of a
269
+ # SpawnedProcess object, NOT in the local context. This can often be deceptive.
270
+ #
271
+ class BlockScopeTest
272
+ attr_reader :var
273
+ def run
274
+ # The following line correctly raises a NameError.
275
+ # The problem is that the programmer expected the spawned block to
276
+ # execute in the local context, but it doesn't.
277
+ #
278
+ # (EM.spawn { do_something }).notify ### NO! BAD!
279
+
280
+
281
+
282
+ # The following line correctly passes self as a parameter to the
283
+ # notified process.
284
+ #
285
+ (EM.spawn {|obj| obj.do_something }).notify(self)
286
+
287
+
288
+
289
+ # Here's another way to do it. This works because "myself" is bound
290
+ # in the local scope, unlike "self," so the spawned block sees it.
291
+ #
292
+ myself = self
293
+ (EM.spawn { myself.do_something }).notify
294
+
295
+
296
+
297
+ # And we end the loop.
298
+ # This is a tangential point, but observe that #notify never blocks.
299
+ # It merely appends a message to the internal queue of a spawned process
300
+ # and returns. As it turns out, the reactor processes notifications for ALL
301
+ # spawned processes in the order that #notify is called. So there is a
302
+ # reasonable expectation that the process which stops the reactor will
303
+ # execute after the previous ones in this method. HOWEVER, this is NOT
304
+ # a documented behavior and is subject to change.
305
+ #
306
+ (EM.spawn {EM.stop}).notify
307
+ end
308
+ def do_something
309
+ @var ||= 0
310
+ @var += 100
311
+ end
312
+ end
313
+
314
+ def test_block_scope
315
+ bs = BlockScopeTest.new
316
+ EM.run {
317
+ bs.run
318
+ }
319
+ assert_equal( 200, bs.var )
320
+ end
321
+
322
+ end