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,15 @@ 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
278
+ def ns_remove_ready_worker ( worker )
279
+ if index = @ready . index { |rw , _ | rw == worker }
280
+ @ready . delete_at ( index )
291
281
end
282
+ true
283
+ end
292
284
293
- @next_gc_time = Concurrent . monotonic_time + @gc_interval
285
+ def ns_prunable_capacity
286
+ [ @pool . size - @min_length , @ready . size ] . min
294
287
end
295
288
296
289
def ns_reset_if_forked
@@ -312,7 +305,7 @@ class Worker
312
305
313
306
def initialize ( pool , id )
314
307
# instance variables accessed only under pool's lock so no need to sync here again
315
- @queue = Queue . new
308
+ @queue = Collection :: TimeoutQueue . new
316
309
@pool = pool
317
310
@thread = create_worker @queue , pool , pool . idletime
318
311
@@ -338,17 +331,29 @@ def kill
338
331
def create_worker ( queue , pool , idletime )
339
332
Thread . new ( queue , pool , idletime ) do |my_queue , my_pool , my_idletime |
340
333
catch ( :stop ) do
341
- loop do
334
+ not_prunable = false
342
335
343
- case message = my_queue . pop
336
+ loop do
337
+ timeout = not_prunable ? nil : my_idletime
338
+ case message = my_queue . pop ( timeout : timeout )
339
+ when nil
340
+ unless my_pool . prunable_capacity . positive?
341
+ not_prunable = true
342
+ next
343
+ end
344
+
345
+ my_pool . remove_ready_worker ( self )
346
+ my_pool . remove_busy_worker ( self )
347
+ throw :stop
344
348
when :stop
349
+ my_pool . remove_ready_worker ( self )
345
350
my_pool . remove_busy_worker ( self )
346
351
throw :stop
347
-
348
352
else
349
353
task , args = message
350
354
run_task my_pool , task , args
351
355
my_pool . ready_worker ( self , Concurrent . monotonic_time )
356
+ not_prunable = false
352
357
end
353
358
end
354
359
end
0 commit comments