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
9
10
# @!macro thread_pool_executor
10
11
# @!macro thread_pool_options
11
12
# @!visibility private
12
13
class RubyThreadPoolExecutor < RubyExecutorService
14
+ include Concern ::Deprecation
13
15
14
16
# @!macro thread_pool_executor_constant_default_max_pool_size
15
17
DEFAULT_MAX_POOL_SIZE = 2_147_483_647 # java.lang.Integer::MAX_VALUE
@@ -95,8 +97,16 @@ def remaining_capacity
95
97
end
96
98
97
99
# @!visibility private
98
- def remove_busy_worker ( worker )
99
- synchronize { ns_remove_busy_worker worker }
100
+ def prunable_capacity
101
+ synchronize { ns_prunable_capacity }
102
+ end
103
+
104
+ # @!visibility private
105
+ def remove_worker ( worker )
106
+ synchronize do
107
+ ns_remove_ready_worker worker
108
+ ns_remove_busy_worker worker
109
+ end
100
110
end
101
111
102
112
# @!visibility private
@@ -116,7 +126,7 @@ def worker_task_completed
116
126
117
127
# @!macro thread_pool_executor_method_prune_pool
118
128
def prune_pool
119
- synchronize { ns_prune_pool }
129
+ deprecated "#prune_pool has no effect and will be removed in next the release, see https://github.com/ruby-concurrency/concurrent-ruby/pull/1082."
120
130
end
121
131
122
132
private
@@ -146,9 +156,6 @@ def ns_initialize(opts)
146
156
@largest_length = 0
147
157
@workers_counter = 0
148
158
@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
159
end
153
160
154
161
# @!visibility private
@@ -162,12 +169,10 @@ def ns_execute(*args, &task)
162
169
163
170
if ns_assign_worker ( *args , &task ) || ns_enqueue ( *args , &task )
164
171
@scheduled_task_count += 1
172
+ nil
165
173
else
166
- return fallback_action ( *args , &task )
174
+ fallback_action ( *args , &task )
167
175
end
168
-
169
- ns_prune_pool if @next_gc_time < Concurrent . monotonic_time
170
- nil
171
176
end
172
177
173
178
# @!visibility private
@@ -218,7 +223,7 @@ def ns_assign_worker(*args, &task)
218
223
# @!visibility private
219
224
def ns_enqueue ( *args , &task )
220
225
return false if @synchronous
221
-
226
+
222
227
if !ns_limited_queue? || @queue . size < @max_queue
223
228
@queue << [ task , args ]
224
229
true
@@ -265,7 +270,7 @@ def ns_ready_worker(worker, last_message, success = true)
265
270
end
266
271
end
267
272
268
- # removes a worker which is not in not tracked in @ready
273
+ # removes a worker which is not tracked in @ready
269
274
#
270
275
# @!visibility private
271
276
def ns_remove_busy_worker ( worker )
@@ -274,23 +279,19 @@ def ns_remove_busy_worker(worker)
274
279
true
275
280
end
276
281
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
282
+ def ns_remove_ready_worker ( worker )
283
+ if index = @ready . index { |rw , _ | rw == worker }
284
+ @ready . delete_at ( index )
291
285
end
286
+ true
287
+ end
292
288
293
- @next_gc_time = Concurrent . monotonic_time + @gc_interval
289
+ def ns_prunable_capacity
290
+ if running?
291
+ [ @pool . size - @min_length , @ready . size ] . min
292
+ else
293
+ @pool . size
294
+ end
294
295
end
295
296
296
297
def ns_reset_if_forked
@@ -312,7 +313,7 @@ class Worker
312
313
313
314
def initialize ( pool , id )
314
315
# instance variables accessed only under pool's lock so no need to sync here again
315
- @queue = Queue . new
316
+ @queue = Collection :: TimeoutQueue . new
316
317
@pool = pool
317
318
@thread = create_worker @queue , pool , pool . idletime
318
319
@@ -338,17 +339,26 @@ def kill
338
339
def create_worker ( queue , pool , idletime )
339
340
Thread . new ( queue , pool , idletime ) do |my_queue , my_pool , my_idletime |
340
341
catch ( :stop ) do
341
- loop do
342
+ prunable = true
342
343
343
- case message = my_queue . pop
344
+ loop do
345
+ timeout = prunable && my_pool . running? ? my_idletime : nil
346
+ case message = my_queue . pop ( timeout : timeout )
347
+ when nil
348
+ if my_pool . prunable_capacity . positive?
349
+ my_pool . remove_worker ( self )
350
+ throw :stop
351
+ end
352
+
353
+ prunable = false
344
354
when :stop
345
- my_pool . remove_busy_worker ( self )
355
+ my_pool . remove_worker ( self )
346
356
throw :stop
347
-
348
357
else
349
358
task , args = message
350
359
run_task my_pool , task , args
351
360
my_pool . ready_worker ( self , Concurrent . monotonic_time )
361
+ prunable = true
352
362
end
353
363
end
354
364
end
0 commit comments