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
@@ -95,8 +96,16 @@ def remaining_capacity
95
96
end
96
97
97
98
# @!visibility private
98
- def remove_busy_worker ( worker )
99
- synchronize { ns_remove_busy_worker worker }
99
+ def prunable_capacity
100
+ synchronize { ns_prunable_capacity }
101
+ end
102
+
103
+ # @!visibility private
104
+ def remove_worker ( worker )
105
+ synchronize do
106
+ ns_remove_ready_worker worker
107
+ ns_remove_busy_worker worker
108
+ end
100
109
end
101
110
102
111
# @!visibility private
@@ -114,11 +123,6 @@ def worker_task_completed
114
123
synchronize { @completed_task_count += 1 }
115
124
end
116
125
117
- # @!macro thread_pool_executor_method_prune_pool
118
- def prune_pool
119
- synchronize { ns_prune_pool }
120
- end
121
-
122
126
private
123
127
124
128
# @!visibility private
@@ -146,9 +150,6 @@ def ns_initialize(opts)
146
150
@largest_length = 0
147
151
@workers_counter = 0
148
152
@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
153
end
153
154
154
155
# @!visibility private
@@ -162,12 +163,10 @@ def ns_execute(*args, &task)
162
163
163
164
if ns_assign_worker ( *args , &task ) || ns_enqueue ( *args , &task )
164
165
@scheduled_task_count += 1
166
+ nil
165
167
else
166
- return fallback_action ( *args , &task )
168
+ fallback_action ( *args , &task )
167
169
end
168
-
169
- ns_prune_pool if @next_gc_time < Concurrent . monotonic_time
170
- nil
171
170
end
172
171
173
172
# @!visibility private
@@ -218,7 +217,7 @@ def ns_assign_worker(*args, &task)
218
217
# @!visibility private
219
218
def ns_enqueue ( *args , &task )
220
219
return false if @synchronous
221
-
220
+
222
221
if !ns_limited_queue? || @queue . size < @max_queue
223
222
@queue << [ task , args ]
224
223
true
@@ -265,7 +264,7 @@ def ns_ready_worker(worker, last_message, success = true)
265
264
end
266
265
end
267
266
268
- # removes a worker which is not in not tracked in @ready
267
+ # removes a worker which is not tracked in @ready
269
268
#
270
269
# @!visibility private
271
270
def ns_remove_busy_worker ( worker )
@@ -274,23 +273,19 @@ def ns_remove_busy_worker(worker)
274
273
true
275
274
end
276
275
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
276
+ def ns_remove_ready_worker ( worker )
277
+ if index = @ready . index { |rw , _ | rw == worker }
278
+ @ready . delete_at ( index )
291
279
end
280
+ true
281
+ end
292
282
293
- @next_gc_time = Concurrent . monotonic_time + @gc_interval
283
+ def ns_prunable_capacity
284
+ if running?
285
+ [ @pool . size - @min_length , @ready . size ] . min
286
+ else
287
+ @pool . size
288
+ end
294
289
end
295
290
296
291
def ns_reset_if_forked
@@ -312,7 +307,7 @@ class Worker
312
307
313
308
def initialize ( pool , id )
314
309
# instance variables accessed only under pool's lock so no need to sync here again
315
- @queue = Queue . new
310
+ @queue = Collection :: TimeoutQueue . new
316
311
@pool = pool
317
312
@thread = create_worker @queue , pool , pool . idletime
318
313
@@ -338,17 +333,26 @@ def kill
338
333
def create_worker ( queue , pool , idletime )
339
334
Thread . new ( queue , pool , idletime ) do |my_queue , my_pool , my_idletime |
340
335
catch ( :stop ) do
341
- loop do
336
+ prunable = true
342
337
343
- case message = my_queue . pop
338
+ loop do
339
+ timeout = prunable && my_pool . running? ? my_idletime : nil
340
+ case message = my_queue . pop ( timeout : timeout )
341
+ when nil
342
+ if my_pool . prunable_capacity . positive?
343
+ my_pool . remove_worker ( self )
344
+ throw :stop
345
+ end
346
+
347
+ prunable = false
344
348
when :stop
345
- my_pool . remove_busy_worker ( self )
349
+ my_pool . remove_worker ( self )
346
350
throw :stop
347
-
348
351
else
349
352
task , args = message
350
353
run_task my_pool , task , args
351
354
my_pool . ready_worker ( self , Concurrent . monotonic_time )
355
+ prunable = true
352
356
end
353
357
end
354
358
end
0 commit comments