3
3
require 'concurrent/concern/logging'
4
4
require 'concurrent/executor/ruby_executor_service'
5
5
require 'concurrent/utility/monotonic_time'
6
+ require 'concurrent/collection/timeout_queue'
6
7
7
8
module Concurrent
8
9
@@ -104,6 +105,11 @@ def ready_worker(worker, last_message)
104
105
synchronize { ns_ready_worker worker , last_message }
105
106
end
106
107
108
+ # @!visibility private
109
+ def remove_ready_worker ( worker )
110
+ synchronize { ns_remove_ready_worker worker }
111
+ end
112
+
107
113
# @!visibility private
108
114
def worker_died ( worker )
109
115
synchronize { ns_worker_died worker }
@@ -114,9 +120,9 @@ def worker_task_completed
114
120
synchronize { @completed_task_count += 1 }
115
121
end
116
122
117
- # @!macro thread_pool_executor_method_prune_pool
118
- def prune_pool
119
- synchronize { ns_prune_pool }
123
+ # @!visibility private
124
+ def prunable_capacity
125
+ synchronize { ns_prunable_capacity }
120
126
end
121
127
122
128
private
@@ -146,9 +152,6 @@ def ns_initialize(opts)
146
152
@largest_length = 0
147
153
@workers_counter = 0
148
154
@ruby_pid = $$ # detects if Ruby has forked
149
-
150
- @gc_interval = opts . fetch ( :gc_interval , @idletime / 2.0 ) . to_i # undocumented
151
- @next_gc_time = Concurrent . monotonic_time + @gc_interval
152
155
end
153
156
154
157
# @!visibility private
@@ -162,12 +165,10 @@ def ns_execute(*args, &task)
162
165
163
166
if ns_assign_worker ( *args , &task ) || ns_enqueue ( *args , &task )
164
167
@scheduled_task_count += 1
168
+ nil
165
169
else
166
- return fallback_action ( *args , &task )
170
+ fallback_action ( *args , &task )
167
171
end
168
-
169
- ns_prune_pool if @next_gc_time < Concurrent . monotonic_time
170
- nil
171
172
end
172
173
173
174
# @!visibility private
@@ -218,7 +219,7 @@ def ns_assign_worker(*args, &task)
218
219
# @!visibility private
219
220
def ns_enqueue ( *args , &task )
220
221
return false if @synchronous
221
-
222
+
222
223
if !ns_limited_queue? || @queue . size < @max_queue
223
224
@queue << [ task , args ]
224
225
true
@@ -265,7 +266,7 @@ def ns_ready_worker(worker, last_message, success = true)
265
266
end
266
267
end
267
268
268
- # removes a worker which is not in not tracked in @ready
269
+ # removes a worker which is not tracked in @ready
269
270
#
270
271
# @!visibility private
271
272
def ns_remove_busy_worker ( worker )
@@ -274,23 +275,13 @@ def ns_remove_busy_worker(worker)
274
275
true
275
276
end
276
277
277
- # try oldest worker if it is idle for enough time, it's returned back at the start
278
- #
279
- # @!visibility private
280
- def ns_prune_pool
281
- now = Concurrent . monotonic_time
282
- stopped_workers = 0
283
- while !@ready . empty? && ( @pool . size - stopped_workers > @min_length )
284
- worker , last_message = @ready . first
285
- if now - last_message > self . idletime
286
- stopped_workers += 1
287
- @ready . shift
288
- worker << :stop
289
- else break
290
- end
291
- end
278
+ def ns_remove_ready_worker ( worker )
279
+ @ready . delete_if { |rw , _ | rw == worker }
280
+ true
281
+ end
292
282
293
- @next_gc_time = Concurrent . monotonic_time + @gc_interval
283
+ def ns_prunable_capacity
284
+ [ @pool . size - @min_length , @ready . size ] . min
294
285
end
295
286
296
287
def ns_reset_if_forked
@@ -312,7 +303,7 @@ class Worker
312
303
313
304
def initialize ( pool , id )
314
305
# instance variables accessed only under pool's lock so no need to sync here again
315
- @queue = Queue . new
306
+ @queue = Collection :: TimeoutQueue . new
316
307
@pool = pool
317
308
@thread = create_worker @queue , pool , pool . idletime
318
309
@@ -338,17 +329,29 @@ def kill
338
329
def create_worker ( queue , pool , idletime )
339
330
Thread . new ( queue , pool , idletime ) do |my_queue , my_pool , my_idletime |
340
331
catch ( :stop ) do
341
- loop do
332
+ not_prunable = false
342
333
343
- case message = my_queue . pop
334
+ loop do
335
+ timeout = not_prunable ? nil : my_idletime
336
+ case message = my_queue . pop ( timeout : timeout )
337
+ when nil
338
+ unless my_pool . prunable_capacity . positive?
339
+ not_prunable = true
340
+ next
341
+ end
342
+
343
+ my_pool . remove_ready_worker ( self )
344
+ my_pool . remove_busy_worker ( self )
345
+ throw :stop
344
346
when :stop
347
+ my_pool . remove_ready_worker ( self )
345
348
my_pool . remove_busy_worker ( self )
346
349
throw :stop
347
-
348
350
else
349
351
task , args = message
350
352
run_task my_pool , task , args
351
353
my_pool . ready_worker ( self , Concurrent . monotonic_time )
354
+ not_prunable = false
352
355
end
353
356
end
354
357
end
0 commit comments