goru 0.0.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/LICENSE +353 -0
- data/lib/goru/queue.rb +61 -0
- data/lib/goru/reactor.rb +153 -0
- data/lib/goru/routine.rb +79 -0
- data/lib/goru/routines/io.rb +42 -0
- data/lib/goru/scheduler.rb +79 -0
- data/lib/goru/version.rb +11 -0
- data/lib/goru.rb +16 -0
- metadata +106 -0
checksums.yaml
ADDED
@@ -0,0 +1,7 @@
|
|
1
|
+
---
|
2
|
+
SHA256:
|
3
|
+
metadata.gz: 2052d47d66797e7a2a268ec567209e6e5a41888f3f7e805ed4883f6534c61e72
|
4
|
+
data.tar.gz: c154d10da685ffc931d7e2c3e6385c18c4cd534f593aa286babd02c13296fd6f
|
5
|
+
SHA512:
|
6
|
+
metadata.gz: b82cfe65acf08ecb2f584b6eb5817c3dd20ad59474d86525e744e0745d34f46dacfbb66ae73e16897965e43e57d4ac8b39bcc6f61141a686314a5ad9a3f71c42
|
7
|
+
data.tar.gz: a2aae75a244ca2d69acd1102fc42da12519eea8221995ce49b7eb2ff00c4f19ad4a07362efc66d833b28f3335ecc12d26333fd0610278b3b04129a2a4fe4e306
|
data/LICENSE
ADDED
@@ -0,0 +1,353 @@
|
|
1
|
+
Mozilla Public License, version 2.0
|
2
|
+
|
3
|
+
1. Definitions
|
4
|
+
|
5
|
+
1.1. “Contributor”
|
6
|
+
|
7
|
+
means each individual or legal entity that creates, contributes to the
|
8
|
+
creation of, or owns Covered Software.
|
9
|
+
|
10
|
+
1.2. “Contributor Version”
|
11
|
+
|
12
|
+
means the combination of the Contributions of others (if any) used by a
|
13
|
+
Contributor and that particular Contributor’s Contribution.
|
14
|
+
|
15
|
+
1.3. “Contribution”
|
16
|
+
|
17
|
+
means Covered Software of a particular Contributor.
|
18
|
+
|
19
|
+
1.4. “Covered Software”
|
20
|
+
|
21
|
+
means Source Code Form to which the initial Contributor has attached the
|
22
|
+
notice in Exhibit A, the Executable Form of such Source Code Form, and
|
23
|
+
Modifications of such Source Code Form, in each case including portions
|
24
|
+
thereof.
|
25
|
+
|
26
|
+
1.5. “Incompatible With Secondary Licenses”
|
27
|
+
means
|
28
|
+
|
29
|
+
a. that the initial Contributor has attached the notice described in
|
30
|
+
Exhibit B to the Covered Software; or
|
31
|
+
|
32
|
+
b. that the Covered Software was made available under the terms of version
|
33
|
+
1.1 or earlier of the License, but not also under the terms of a
|
34
|
+
Secondary License.
|
35
|
+
|
36
|
+
1.6. “Executable Form”
|
37
|
+
|
38
|
+
means any form of the work other than Source Code Form.
|
39
|
+
|
40
|
+
1.7. “Larger Work”
|
41
|
+
|
42
|
+
means a work that combines Covered Software with other material, in a separate
|
43
|
+
file or files, that is not Covered Software.
|
44
|
+
|
45
|
+
1.8. “License”
|
46
|
+
|
47
|
+
means this document.
|
48
|
+
|
49
|
+
1.9. “Licensable”
|
50
|
+
|
51
|
+
means having the right to grant, to the maximum extent possible, whether at the
|
52
|
+
time of the initial grant or subsequently, any and all of the rights conveyed by
|
53
|
+
this License.
|
54
|
+
|
55
|
+
1.10. “Modifications”
|
56
|
+
|
57
|
+
means any of the following:
|
58
|
+
|
59
|
+
a. any file in Source Code Form that results from an addition to, deletion
|
60
|
+
from, or modification of the contents of Covered Software; or
|
61
|
+
|
62
|
+
b. any new file in Source Code Form that contains any Covered Software.
|
63
|
+
|
64
|
+
1.11. “Patent Claims” of a Contributor
|
65
|
+
|
66
|
+
means any patent claim(s), including without limitation, method, process,
|
67
|
+
and apparatus claims, in any patent Licensable by such Contributor that
|
68
|
+
would be infringed, but for the grant of the License, by the making,
|
69
|
+
using, selling, offering for sale, having made, import, or transfer of
|
70
|
+
either its Contributions or its Contributor Version.
|
71
|
+
|
72
|
+
1.12. “Secondary License”
|
73
|
+
|
74
|
+
means either the GNU General Public License, Version 2.0, the GNU Lesser
|
75
|
+
General Public License, Version 2.1, the GNU Affero General Public
|
76
|
+
License, Version 3.0, or any later versions of those licenses.
|
77
|
+
|
78
|
+
1.13. “Source Code Form”
|
79
|
+
|
80
|
+
means the form of the work preferred for making modifications.
|
81
|
+
|
82
|
+
1.14. “You” (or “Your”)
|
83
|
+
|
84
|
+
means an individual or a legal entity exercising rights under this
|
85
|
+
License. For legal entities, “You” includes any entity that controls, is
|
86
|
+
controlled by, or is under common control with You. For purposes of this
|
87
|
+
definition, “control” means (a) the power, direct or indirect, to cause
|
88
|
+
the direction or management of such entity, whether by contract or
|
89
|
+
otherwise, or (b) ownership of more than fifty percent (50%) of the
|
90
|
+
outstanding shares or beneficial ownership of such entity.
|
91
|
+
|
92
|
+
|
93
|
+
2. License Grants and Conditions
|
94
|
+
|
95
|
+
2.1. Grants
|
96
|
+
|
97
|
+
Each Contributor hereby grants You a world-wide, royalty-free,
|
98
|
+
non-exclusive license:
|
99
|
+
|
100
|
+
a. under intellectual property rights (other than patent or trademark)
|
101
|
+
Licensable by such Contributor to use, reproduce, make available,
|
102
|
+
modify, display, perform, distribute, and otherwise exploit its
|
103
|
+
Contributions, either on an unmodified basis, with Modifications, or as
|
104
|
+
part of a Larger Work; and
|
105
|
+
|
106
|
+
b. under Patent Claims of such Contributor to make, use, sell, offer for
|
107
|
+
sale, have made, import, and otherwise transfer either its Contributions
|
108
|
+
or its Contributor Version.
|
109
|
+
|
110
|
+
2.2. Effective Date
|
111
|
+
|
112
|
+
The licenses granted in Section 2.1 with respect to any Contribution become
|
113
|
+
effective for each Contribution on the date the Contributor first distributes
|
114
|
+
such Contribution.
|
115
|
+
|
116
|
+
2.3. Limitations on Grant Scope
|
117
|
+
|
118
|
+
The licenses granted in this Section 2 are the only rights granted under this
|
119
|
+
License. No additional rights or licenses will be implied from the distribution
|
120
|
+
or licensing of Covered Software under this License. Notwithstanding Section
|
121
|
+
2.1(b) above, no patent license is granted by a Contributor:
|
122
|
+
|
123
|
+
a. for any code that a Contributor has removed from Covered Software; or
|
124
|
+
|
125
|
+
b. for infringements caused by: (i) Your and any other third party’s
|
126
|
+
modifications of Covered Software, or (ii) the combination of its
|
127
|
+
Contributions with other software (except as part of its Contributor
|
128
|
+
Version); or
|
129
|
+
|
130
|
+
c. under Patent Claims infringed by Covered Software in the absence of its
|
131
|
+
Contributions.
|
132
|
+
|
133
|
+
This License does not grant any rights in the trademarks, service marks, or
|
134
|
+
logos of any Contributor (except as may be necessary to comply with the
|
135
|
+
notice requirements in Section 3.4).
|
136
|
+
|
137
|
+
2.4. Subsequent Licenses
|
138
|
+
|
139
|
+
No Contributor makes additional grants as a result of Your choice to
|
140
|
+
distribute the Covered Software under a subsequent version of this License
|
141
|
+
(see Section 10.2) or under the terms of a Secondary License (if permitted
|
142
|
+
under the terms of Section 3.3).
|
143
|
+
|
144
|
+
2.5. Representation
|
145
|
+
|
146
|
+
Each Contributor represents that the Contributor believes its Contributions
|
147
|
+
are its original creation(s) or it has sufficient rights to grant the
|
148
|
+
rights to its Contributions conveyed by this License.
|
149
|
+
|
150
|
+
2.6. Fair Use
|
151
|
+
|
152
|
+
This License is not intended to limit any rights You have under applicable
|
153
|
+
copyright doctrines of fair use, fair dealing, or other equivalents.
|
154
|
+
|
155
|
+
2.7. Conditions
|
156
|
+
|
157
|
+
Sections 3.1, 3.2, 3.3, and 3.4 are conditions of the licenses granted in
|
158
|
+
Section 2.1.
|
159
|
+
|
160
|
+
|
161
|
+
3. Responsibilities
|
162
|
+
|
163
|
+
3.1. Distribution of Source Form
|
164
|
+
|
165
|
+
All distribution of Covered Software in Source Code Form, including any
|
166
|
+
Modifications that You create or to which You contribute, must be under the
|
167
|
+
terms of this License. You must inform recipients that the Source Code Form
|
168
|
+
of the Covered Software is governed by the terms of this License, and how
|
169
|
+
they can obtain a copy of this License. You may not attempt to alter or
|
170
|
+
restrict the recipients’ rights in the Source Code Form.
|
171
|
+
|
172
|
+
3.2. Distribution of Executable Form
|
173
|
+
|
174
|
+
If You distribute Covered Software in Executable Form then:
|
175
|
+
|
176
|
+
a. such Covered Software must also be made available in Source Code Form,
|
177
|
+
as described in Section 3.1, and You must inform recipients of the
|
178
|
+
Executable Form how they can obtain a copy of such Source Code Form by
|
179
|
+
reasonable means in a timely manner, at a charge no more than the cost
|
180
|
+
of distribution to the recipient; and
|
181
|
+
|
182
|
+
b. You may distribute such Executable Form under the terms of this License,
|
183
|
+
or sublicense it under different terms, provided that the license for
|
184
|
+
the Executable Form does not attempt to limit or alter the recipients’
|
185
|
+
rights in the Source Code Form under this License.
|
186
|
+
|
187
|
+
3.3. Distribution of a Larger Work
|
188
|
+
|
189
|
+
You may create and distribute a Larger Work under terms of Your choice,
|
190
|
+
provided that You also comply with the requirements of this License for the
|
191
|
+
Covered Software. If the Larger Work is a combination of Covered Software
|
192
|
+
with a work governed by one or more Secondary Licenses, and the Covered
|
193
|
+
Software is not Incompatible With Secondary Licenses, this License permits
|
194
|
+
You to additionally distribute such Covered Software under the terms of
|
195
|
+
such Secondary License(s), so that the recipient of the Larger Work may, at
|
196
|
+
their option, further distribute the Covered Software under the terms of
|
197
|
+
either this License or such Secondary License(s).
|
198
|
+
|
199
|
+
3.4. Notices
|
200
|
+
|
201
|
+
You may not remove or alter the substance of any license notices (including
|
202
|
+
copyright notices, patent notices, disclaimers of warranty, or limitations
|
203
|
+
of liability) contained within the Source Code Form of the Covered
|
204
|
+
Software, except that You may alter any license notices to the extent
|
205
|
+
required to remedy known factual inaccuracies.
|
206
|
+
|
207
|
+
3.5. Application of Additional Terms
|
208
|
+
|
209
|
+
You may choose to offer, and to charge a fee for, warranty, support,
|
210
|
+
indemnity or liability obligations to one or more recipients of Covered
|
211
|
+
Software. However, You may do so only on Your own behalf, and not on behalf
|
212
|
+
of any Contributor. You must make it absolutely clear that any such
|
213
|
+
warranty, support, indemnity, or liability obligation is offered by You
|
214
|
+
alone, and You hereby agree to indemnify every Contributor for any
|
215
|
+
liability incurred by such Contributor as a result of warranty, support,
|
216
|
+
indemnity or liability terms You offer. You may include additional
|
217
|
+
disclaimers of warranty and limitations of liability specific to any
|
218
|
+
jurisdiction.
|
219
|
+
|
220
|
+
4. Inability to Comply Due to Statute or Regulation
|
221
|
+
|
222
|
+
If it is impossible for You to comply with any of the terms of this License
|
223
|
+
with respect to some or all of the Covered Software due to statute, judicial
|
224
|
+
order, or regulation then You must: (a) comply with the terms of this License
|
225
|
+
to the maximum extent possible; and (b) describe the limitations and the code
|
226
|
+
they affect. Such description must be placed in a text file included with all
|
227
|
+
distributions of the Covered Software under this License. Except to the
|
228
|
+
extent prohibited by statute or regulation, such description must be
|
229
|
+
sufficiently detailed for a recipient of ordinary skill to be able to
|
230
|
+
understand it.
|
231
|
+
|
232
|
+
5. Termination
|
233
|
+
|
234
|
+
5.1. The rights granted under this License will terminate automatically if You
|
235
|
+
fail to comply with any of its terms. However, if You become compliant,
|
236
|
+
then the rights granted under this License from a particular Contributor
|
237
|
+
are reinstated (a) provisionally, unless and until such Contributor
|
238
|
+
explicitly and finally terminates Your grants, and (b) on an ongoing basis,
|
239
|
+
if such Contributor fails to notify You of the non-compliance by some
|
240
|
+
reasonable means prior to 60 days after You have come back into compliance.
|
241
|
+
Moreover, Your grants from a particular Contributor are reinstated on an
|
242
|
+
ongoing basis if such Contributor notifies You of the non-compliance by
|
243
|
+
some reasonable means, this is the first time You have received notice of
|
244
|
+
non-compliance with this License from such Contributor, and You become
|
245
|
+
compliant prior to 30 days after Your receipt of the notice.
|
246
|
+
|
247
|
+
5.2. If You initiate litigation against any entity by asserting a patent
|
248
|
+
infringement claim (excluding declaratory judgment actions, counter-claims,
|
249
|
+
and cross-claims) alleging that a Contributor Version directly or
|
250
|
+
indirectly infringes any patent, then the rights granted to You by any and
|
251
|
+
all Contributors for the Covered Software under Section 2.1 of this License
|
252
|
+
shall terminate.
|
253
|
+
|
254
|
+
5.3. In the event of termination under Sections 5.1 or 5.2 above, all end user
|
255
|
+
license agreements (excluding distributors and resellers) which have been
|
256
|
+
validly granted by You or Your distributors under this License prior to
|
257
|
+
termination shall survive termination.
|
258
|
+
|
259
|
+
6. Disclaimer of Warranty
|
260
|
+
|
261
|
+
Covered Software is provided under this License on an “as is” basis, without
|
262
|
+
warranty of any kind, either expressed, implied, or statutory, including,
|
263
|
+
without limitation, warranties that the Covered Software is free of defects,
|
264
|
+
merchantable, fit for a particular purpose or non-infringing. The entire
|
265
|
+
risk as to the quality and performance of the Covered Software is with You.
|
266
|
+
Should any Covered Software prove defective in any respect, You (not any
|
267
|
+
Contributor) assume the cost of any necessary servicing, repair, or
|
268
|
+
correction. This disclaimer of warranty constitutes an essential part of this
|
269
|
+
License. No use of any Covered Software is authorized under this License
|
270
|
+
except under this disclaimer.
|
271
|
+
|
272
|
+
7. Limitation of Liability
|
273
|
+
|
274
|
+
Under no circumstances and under no legal theory, whether tort (including
|
275
|
+
negligence), contract, or otherwise, shall any Contributor, or anyone who
|
276
|
+
distributes Covered Software as permitted above, be liable to You for any
|
277
|
+
direct, indirect, special, incidental, or consequential damages of any
|
278
|
+
character including, without limitation, damages for lost profits, loss of
|
279
|
+
goodwill, work stoppage, computer failure or malfunction, or any and all
|
280
|
+
other commercial damages or losses, even if such party shall have been
|
281
|
+
informed of the possibility of such damages. This limitation of liability
|
282
|
+
shall not apply to liability for death or personal injury resulting from such
|
283
|
+
party’s negligence to the extent applicable law prohibits such limitation.
|
284
|
+
Some jurisdictions do not allow the exclusion or limitation of incidental or
|
285
|
+
consequential damages, so this exclusion and limitation may not apply to You.
|
286
|
+
|
287
|
+
8. Litigation
|
288
|
+
|
289
|
+
Any litigation relating to this License may be brought only in the courts of
|
290
|
+
a jurisdiction where the defendant maintains its principal place of business
|
291
|
+
and such litigation shall be governed by laws of that jurisdiction, without
|
292
|
+
reference to its conflict-of-law provisions. Nothing in this Section shall
|
293
|
+
prevent a party’s ability to bring cross-claims or counter-claims.
|
294
|
+
|
295
|
+
9. Miscellaneous
|
296
|
+
|
297
|
+
This License represents the complete agreement concerning the subject matter
|
298
|
+
hereof. If any provision of this License is held to be unenforceable, such
|
299
|
+
provision shall be reformed only to the extent necessary to make it
|
300
|
+
enforceable. Any law or regulation which provides that the language of a
|
301
|
+
contract shall be construed against the drafter shall not be used to construe
|
302
|
+
this License against a Contributor.
|
303
|
+
|
304
|
+
|
305
|
+
10. Versions of the License
|
306
|
+
|
307
|
+
10.1. New Versions
|
308
|
+
|
309
|
+
Mozilla Foundation is the license steward. Except as provided in Section
|
310
|
+
10.3, no one other than the license steward has the right to modify or
|
311
|
+
publish new versions of this License. Each version will be given a
|
312
|
+
distinguishing version number.
|
313
|
+
|
314
|
+
10.2. Effect of New Versions
|
315
|
+
|
316
|
+
You may distribute the Covered Software under the terms of the version of
|
317
|
+
the License under which You originally received the Covered Software, or
|
318
|
+
under the terms of any subsequent version published by the license
|
319
|
+
steward.
|
320
|
+
|
321
|
+
10.3. Modified Versions
|
322
|
+
|
323
|
+
If you create software not governed by this License, and you want to
|
324
|
+
create a new license for such software, you may create and use a modified
|
325
|
+
version of this License if you rename the license and remove any
|
326
|
+
references to the name of the license steward (except to note that such
|
327
|
+
modified license differs from this License).
|
328
|
+
|
329
|
+
10.4. Distributing Source Code Form that is Incompatible With Secondary Licenses
|
330
|
+
If You choose to distribute Source Code Form that is Incompatible With
|
331
|
+
Secondary Licenses under the terms of this version of the License, the
|
332
|
+
notice described in Exhibit B of this License must be attached.
|
333
|
+
|
334
|
+
Exhibit A - Source Code Form License Notice
|
335
|
+
|
336
|
+
This Source Code Form is subject to the
|
337
|
+
terms of the Mozilla Public License, v.
|
338
|
+
2.0. If a copy of the MPL was not
|
339
|
+
distributed with this file, You can
|
340
|
+
obtain one at
|
341
|
+
http://mozilla.org/MPL/2.0/.
|
342
|
+
|
343
|
+
If it is not possible or desirable to put the notice in a particular file, then
|
344
|
+
You may include the notice in a location (such as a LICENSE file in a relevant
|
345
|
+
directory) where a recipient would be likely to look for such a notice.
|
346
|
+
|
347
|
+
You may add additional accurate notices of copyright ownership.
|
348
|
+
|
349
|
+
Exhibit B - “Incompatible With Secondary Licenses” Notice
|
350
|
+
|
351
|
+
This Source Code Form is “Incompatible
|
352
|
+
With Secondary Licenses”, as defined by
|
353
|
+
the Mozilla Public License, v. 2.0.
|
data/lib/goru/queue.rb
ADDED
@@ -0,0 +1,61 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Goru
|
4
|
+
# [public] Based on: https://spin.atomicobject.com/2017/06/28/queue-pop-with-timeout-fixed/
|
5
|
+
#
|
6
|
+
class Queue
|
7
|
+
def initialize
|
8
|
+
@mutex = Mutex.new
|
9
|
+
@queue = []
|
10
|
+
@received = ConditionVariable.new
|
11
|
+
@closed = false
|
12
|
+
end
|
13
|
+
|
14
|
+
# [public]
|
15
|
+
#
|
16
|
+
def <<(x)
|
17
|
+
@mutex.synchronize do
|
18
|
+
@queue << x
|
19
|
+
@received.signal
|
20
|
+
end
|
21
|
+
end
|
22
|
+
|
23
|
+
# [public]
|
24
|
+
#
|
25
|
+
def pop(non_block = false)
|
26
|
+
pop_with_timeout(non_block ? 0 : nil)
|
27
|
+
end
|
28
|
+
|
29
|
+
# [public]
|
30
|
+
#
|
31
|
+
def pop_with_timeout(timeout = nil)
|
32
|
+
@mutex.synchronize do
|
33
|
+
if timeout.nil?
|
34
|
+
# wait indefinitely until there is an element in the queue
|
35
|
+
while @queue.empty? && !@closed
|
36
|
+
@received.wait(@mutex)
|
37
|
+
end
|
38
|
+
elsif @queue.empty? && !@closed && timeout != 0
|
39
|
+
# wait for element or timeout
|
40
|
+
timeout_time = timeout + Time.now.to_f
|
41
|
+
while @queue.empty? && (remaining_time = timeout_time - Time.now.to_f) > 0
|
42
|
+
@received.wait(@mutex, remaining_time)
|
43
|
+
end
|
44
|
+
end
|
45
|
+
return if @closed
|
46
|
+
# if we're still empty after the timeout, raise exception
|
47
|
+
raise ThreadError, "queue empty" if @queue.empty?
|
48
|
+
@queue.shift
|
49
|
+
end
|
50
|
+
end
|
51
|
+
|
52
|
+
# [public]
|
53
|
+
#
|
54
|
+
def close
|
55
|
+
@mutex.synchronize do
|
56
|
+
@closed = true
|
57
|
+
@received.broadcast
|
58
|
+
end
|
59
|
+
end
|
60
|
+
end
|
61
|
+
end
|
data/lib/goru/reactor.rb
ADDED
@@ -0,0 +1,153 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "nio"
|
4
|
+
|
5
|
+
require "timers/group"
|
6
|
+
require "timers/wait"
|
7
|
+
|
8
|
+
require_relative "routines/io"
|
9
|
+
|
10
|
+
module Goru
|
11
|
+
# [public]
|
12
|
+
#
|
13
|
+
class Reactor
|
14
|
+
def initialize(queue:, scheduler:)
|
15
|
+
@queue = queue
|
16
|
+
@scheduler = scheduler
|
17
|
+
@routines = []
|
18
|
+
@finished = []
|
19
|
+
@timers = Timers::Group.new
|
20
|
+
@selector = NIO::Selector.new
|
21
|
+
@status = nil
|
22
|
+
@stopped = false
|
23
|
+
end
|
24
|
+
|
25
|
+
# [public]
|
26
|
+
#
|
27
|
+
attr_reader :status
|
28
|
+
|
29
|
+
# [public]
|
30
|
+
#
|
31
|
+
def run
|
32
|
+
@status = :running
|
33
|
+
|
34
|
+
until @stopped
|
35
|
+
@routines.each do |routine|
|
36
|
+
call_routine(routine)
|
37
|
+
end
|
38
|
+
|
39
|
+
# TODO: Remove the monitor from the selector.
|
40
|
+
#
|
41
|
+
cleanup_finished_routines
|
42
|
+
|
43
|
+
begin
|
44
|
+
if (routine = @queue.pop(true))
|
45
|
+
adopt_routine(routine)
|
46
|
+
end
|
47
|
+
rescue ThreadError
|
48
|
+
interval = @timers.wait_interval
|
49
|
+
|
50
|
+
if interval.nil?
|
51
|
+
if @routines.empty?
|
52
|
+
if @selector.empty?
|
53
|
+
@status = :looking
|
54
|
+
@scheduler.signal(self)
|
55
|
+
if (routine = @queue.pop)
|
56
|
+
adopt_routine(routine)
|
57
|
+
end
|
58
|
+
else
|
59
|
+
# TODO: The issue doing this is that this reactor won't grab new routines. Will calling `@selector.wakeup`
|
60
|
+
# from the scheduler when a routine is added to the queue resolve this?
|
61
|
+
#
|
62
|
+
@selector.select do |monitor|
|
63
|
+
monitor.value.call
|
64
|
+
end
|
65
|
+
end
|
66
|
+
else
|
67
|
+
@selector.select(0) do |monitor|
|
68
|
+
monitor.value.call
|
69
|
+
end
|
70
|
+
end
|
71
|
+
elsif interval > 0
|
72
|
+
if @selector.empty?
|
73
|
+
Timers::Wait.for(interval) do |remaining|
|
74
|
+
if (routine = @queue.pop_with_timeout(remaining))
|
75
|
+
adopt_routine(routine)
|
76
|
+
break
|
77
|
+
end
|
78
|
+
rescue ThreadError
|
79
|
+
# nothing to do
|
80
|
+
end
|
81
|
+
else
|
82
|
+
@selector.select(interval) do |monitor|
|
83
|
+
monitor.value.call
|
84
|
+
end
|
85
|
+
end
|
86
|
+
end
|
87
|
+
|
88
|
+
@timers.fire
|
89
|
+
end
|
90
|
+
end
|
91
|
+
ensure
|
92
|
+
@selector.close
|
93
|
+
@status = :finished
|
94
|
+
end
|
95
|
+
|
96
|
+
# [public]
|
97
|
+
#
|
98
|
+
def stop
|
99
|
+
@stopped = true
|
100
|
+
|
101
|
+
unless @selector.closed?
|
102
|
+
@selector.wakeup
|
103
|
+
end
|
104
|
+
end
|
105
|
+
|
106
|
+
# [public]
|
107
|
+
#
|
108
|
+
def sleep(routine, seconds)
|
109
|
+
@timers.after(seconds) {
|
110
|
+
routine.wake
|
111
|
+
}
|
112
|
+
end
|
113
|
+
|
114
|
+
private def adopt_routine(routine)
|
115
|
+
routine.reactor = self
|
116
|
+
|
117
|
+
case routine
|
118
|
+
when Routines::IO
|
119
|
+
monitor = @selector.register(routine.io, routine.intent)
|
120
|
+
|
121
|
+
monitor.value = proc {
|
122
|
+
# TODO: Try to combine this with `call_routine` below.
|
123
|
+
#
|
124
|
+
case routine.status
|
125
|
+
when :selecting
|
126
|
+
routine.call
|
127
|
+
else
|
128
|
+
@finished << routine
|
129
|
+
end
|
130
|
+
}
|
131
|
+
else
|
132
|
+
@routines << routine
|
133
|
+
end
|
134
|
+
end
|
135
|
+
|
136
|
+
private def call_routine(routine)
|
137
|
+
case routine.status
|
138
|
+
when :running
|
139
|
+
routine.call
|
140
|
+
when :sleeping, :selecting
|
141
|
+
# ignore these
|
142
|
+
else
|
143
|
+
@finished << routine
|
144
|
+
end
|
145
|
+
end
|
146
|
+
|
147
|
+
private def cleanup_finished_routines
|
148
|
+
until @finished.empty?
|
149
|
+
@routines.delete(@finished.pop)
|
150
|
+
end
|
151
|
+
end
|
152
|
+
end
|
153
|
+
end
|
data/lib/goru/routine.rb
ADDED
@@ -0,0 +1,79 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Goru
|
4
|
+
# [public]
|
5
|
+
#
|
6
|
+
class Routine
|
7
|
+
def initialize(state = nil, &block)
|
8
|
+
@state = state
|
9
|
+
@block = block
|
10
|
+
@status = :running
|
11
|
+
@result, @error, @reactor = nil
|
12
|
+
end
|
13
|
+
|
14
|
+
# [public]
|
15
|
+
#
|
16
|
+
attr_reader :state, :status
|
17
|
+
|
18
|
+
# [public]
|
19
|
+
#
|
20
|
+
attr_writer :reactor
|
21
|
+
|
22
|
+
# [public]
|
23
|
+
#
|
24
|
+
def running?
|
25
|
+
@status == :running
|
26
|
+
end
|
27
|
+
|
28
|
+
# [public]
|
29
|
+
#
|
30
|
+
def call
|
31
|
+
@block.call(self)
|
32
|
+
rescue => error
|
33
|
+
puts "[routine error] #{error}"
|
34
|
+
puts error.backtrace
|
35
|
+
|
36
|
+
@error = error
|
37
|
+
@status = :errored
|
38
|
+
end
|
39
|
+
|
40
|
+
# [public]
|
41
|
+
#
|
42
|
+
def finished(result = nil)
|
43
|
+
unless @finished
|
44
|
+
@result = result
|
45
|
+
@status = :finished
|
46
|
+
end
|
47
|
+
end
|
48
|
+
|
49
|
+
# [public]
|
50
|
+
#
|
51
|
+
def update(state)
|
52
|
+
@state = state
|
53
|
+
end
|
54
|
+
|
55
|
+
# [public]
|
56
|
+
#
|
57
|
+
def result
|
58
|
+
case @status
|
59
|
+
when :errored
|
60
|
+
raise @error
|
61
|
+
else
|
62
|
+
@result
|
63
|
+
end
|
64
|
+
end
|
65
|
+
|
66
|
+
# [public]
|
67
|
+
#
|
68
|
+
def sleep(seconds)
|
69
|
+
@status = :sleeping
|
70
|
+
@reactor.sleep(self, seconds)
|
71
|
+
end
|
72
|
+
|
73
|
+
# [public]
|
74
|
+
#
|
75
|
+
def wake
|
76
|
+
@status = :running
|
77
|
+
end
|
78
|
+
end
|
79
|
+
end
|
@@ -0,0 +1,42 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require_relative "../routine"
|
4
|
+
|
5
|
+
module Goru
|
6
|
+
module Routines
|
7
|
+
# [public]
|
8
|
+
#
|
9
|
+
class IO < Routine
|
10
|
+
def initialize(state = nil, io:, intent: :rw, &block)
|
11
|
+
super(state, &block)
|
12
|
+
|
13
|
+
@io = io
|
14
|
+
@intent = intent
|
15
|
+
@status = :selecting
|
16
|
+
end
|
17
|
+
|
18
|
+
# [public]
|
19
|
+
#
|
20
|
+
attr_reader :io, :intent
|
21
|
+
|
22
|
+
# [public]
|
23
|
+
#
|
24
|
+
def accept
|
25
|
+
@io.accept_nonblock
|
26
|
+
end
|
27
|
+
|
28
|
+
# [public]
|
29
|
+
#
|
30
|
+
def read(bytes)
|
31
|
+
result = @io.read_nonblock(bytes, exception: false)
|
32
|
+
|
33
|
+
case result
|
34
|
+
when :wait_readable, nil
|
35
|
+
# nothing to do
|
36
|
+
else
|
37
|
+
result
|
38
|
+
end
|
39
|
+
end
|
40
|
+
end
|
41
|
+
end
|
42
|
+
end
|
@@ -0,0 +1,79 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "is/global"
|
4
|
+
|
5
|
+
require_relative "queue"
|
6
|
+
require_relative "reactor"
|
7
|
+
require_relative "routines/io"
|
8
|
+
|
9
|
+
module Goru
|
10
|
+
# [public]
|
11
|
+
#
|
12
|
+
class Scheduler
|
13
|
+
include Is::Global
|
14
|
+
include MonitorMixin
|
15
|
+
|
16
|
+
def initialize(...)
|
17
|
+
super
|
18
|
+
|
19
|
+
@stopping = false
|
20
|
+
@routines = Queue.new
|
21
|
+
@condition = new_cond
|
22
|
+
|
23
|
+
@reactors = 10.times.map {
|
24
|
+
Reactor.new(queue: @routines, scheduler: self)
|
25
|
+
}
|
26
|
+
|
27
|
+
@threads = @reactors.map { |reactor|
|
28
|
+
Thread.new {
|
29
|
+
Thread.handle_interrupt(Interrupt => :never) do
|
30
|
+
reactor.run
|
31
|
+
end
|
32
|
+
}
|
33
|
+
}
|
34
|
+
end
|
35
|
+
|
36
|
+
# [public]
|
37
|
+
#
|
38
|
+
def go(state = nil, io: nil, intent: :rw, &block)
|
39
|
+
@routines << if io
|
40
|
+
Routines::IO.new(state, io: io, intent: intent, &block)
|
41
|
+
else
|
42
|
+
Routine.new(state, &block)
|
43
|
+
end
|
44
|
+
end
|
45
|
+
|
46
|
+
# [public]
|
47
|
+
#
|
48
|
+
def wait
|
49
|
+
synchronize do
|
50
|
+
@condition.wait_until do
|
51
|
+
@stopping
|
52
|
+
end
|
53
|
+
end
|
54
|
+
ensure
|
55
|
+
stop
|
56
|
+
end
|
57
|
+
|
58
|
+
# [public]
|
59
|
+
#
|
60
|
+
def stop
|
61
|
+
@stopping = true
|
62
|
+
@routines.close
|
63
|
+
@reactors.each(&:stop)
|
64
|
+
@threads.each(&:join)
|
65
|
+
end
|
66
|
+
|
67
|
+
# [public]
|
68
|
+
#
|
69
|
+
def signal(reactor)
|
70
|
+
synchronize do
|
71
|
+
if @reactors.all? { |reactor| reactor.status == :looking || reactor.status == :stopped }
|
72
|
+
@stopping = true
|
73
|
+
end
|
74
|
+
|
75
|
+
@condition.signal
|
76
|
+
end
|
77
|
+
end
|
78
|
+
end
|
79
|
+
end
|
data/lib/goru/version.rb
ADDED
data/lib/goru.rb
ADDED
@@ -0,0 +1,16 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "is/extension"
|
4
|
+
|
5
|
+
module Goru
|
6
|
+
require_relative "goru/scheduler"
|
7
|
+
require_relative "goru/version"
|
8
|
+
|
9
|
+
extend Is::Extension
|
10
|
+
|
11
|
+
def go(state = nil, io: nil, intent: :rw, &block)
|
12
|
+
Scheduler.go(state, io: io, intent: intent, &block)
|
13
|
+
end
|
14
|
+
end
|
15
|
+
|
16
|
+
at_exit { Goru::Scheduler.wait }
|
metadata
ADDED
@@ -0,0 +1,106 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: goru
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 0.0.0
|
5
|
+
platform: ruby
|
6
|
+
authors:
|
7
|
+
- Bryan Powell
|
8
|
+
autorequire:
|
9
|
+
bindir: bin
|
10
|
+
cert_chain: []
|
11
|
+
date: 2022-01-06 00:00:00.000000000 Z
|
12
|
+
dependencies:
|
13
|
+
- !ruby/object:Gem::Dependency
|
14
|
+
name: core-extension
|
15
|
+
requirement: !ruby/object:Gem::Requirement
|
16
|
+
requirements:
|
17
|
+
- - "~>"
|
18
|
+
- !ruby/object:Gem::Version
|
19
|
+
version: '0.4'
|
20
|
+
type: :runtime
|
21
|
+
prerelease: false
|
22
|
+
version_requirements: !ruby/object:Gem::Requirement
|
23
|
+
requirements:
|
24
|
+
- - "~>"
|
25
|
+
- !ruby/object:Gem::Version
|
26
|
+
version: '0.4'
|
27
|
+
- !ruby/object:Gem::Dependency
|
28
|
+
name: core-global
|
29
|
+
requirement: !ruby/object:Gem::Requirement
|
30
|
+
requirements:
|
31
|
+
- - "~>"
|
32
|
+
- !ruby/object:Gem::Version
|
33
|
+
version: '0.1'
|
34
|
+
type: :runtime
|
35
|
+
prerelease: false
|
36
|
+
version_requirements: !ruby/object:Gem::Requirement
|
37
|
+
requirements:
|
38
|
+
- - "~>"
|
39
|
+
- !ruby/object:Gem::Version
|
40
|
+
version: '0.1'
|
41
|
+
- !ruby/object:Gem::Dependency
|
42
|
+
name: nio4r
|
43
|
+
requirement: !ruby/object:Gem::Requirement
|
44
|
+
requirements:
|
45
|
+
- - "~>"
|
46
|
+
- !ruby/object:Gem::Version
|
47
|
+
version: '2.5'
|
48
|
+
type: :runtime
|
49
|
+
prerelease: false
|
50
|
+
version_requirements: !ruby/object:Gem::Requirement
|
51
|
+
requirements:
|
52
|
+
- - "~>"
|
53
|
+
- !ruby/object:Gem::Version
|
54
|
+
version: '2.5'
|
55
|
+
- !ruby/object:Gem::Dependency
|
56
|
+
name: timers
|
57
|
+
requirement: !ruby/object:Gem::Requirement
|
58
|
+
requirements:
|
59
|
+
- - "~>"
|
60
|
+
- !ruby/object:Gem::Version
|
61
|
+
version: '4.3'
|
62
|
+
type: :runtime
|
63
|
+
prerelease: false
|
64
|
+
version_requirements: !ruby/object:Gem::Requirement
|
65
|
+
requirements:
|
66
|
+
- - "~>"
|
67
|
+
- !ruby/object:Gem::Version
|
68
|
+
version: '4.3'
|
69
|
+
description: Concurrent routines for Ruby.
|
70
|
+
email: bryan@metabahn.com
|
71
|
+
executables: []
|
72
|
+
extensions: []
|
73
|
+
extra_rdoc_files: []
|
74
|
+
files:
|
75
|
+
- LICENSE
|
76
|
+
- lib/goru.rb
|
77
|
+
- lib/goru/queue.rb
|
78
|
+
- lib/goru/reactor.rb
|
79
|
+
- lib/goru/routine.rb
|
80
|
+
- lib/goru/routines/io.rb
|
81
|
+
- lib/goru/scheduler.rb
|
82
|
+
- lib/goru/version.rb
|
83
|
+
homepage: https://github.com/metabahn/goru/
|
84
|
+
licenses:
|
85
|
+
- MPL-2.0
|
86
|
+
metadata: {}
|
87
|
+
post_install_message:
|
88
|
+
rdoc_options: []
|
89
|
+
require_paths:
|
90
|
+
- lib
|
91
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
92
|
+
requirements:
|
93
|
+
- - ">="
|
94
|
+
- !ruby/object:Gem::Version
|
95
|
+
version: 2.6.7
|
96
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
97
|
+
requirements:
|
98
|
+
- - ">="
|
99
|
+
- !ruby/object:Gem::Version
|
100
|
+
version: '0'
|
101
|
+
requirements: []
|
102
|
+
rubygems_version: 3.2.22
|
103
|
+
signing_key:
|
104
|
+
specification_version: 4
|
105
|
+
summary: Concurrent routines for Ruby.
|
106
|
+
test_files: []
|