Skip to content

Commit f3d097c

Browse files
bensheldonioquatix
authored andcommitted
Allow TimerTask to be safely restarted after shutdown and avoid duplicate tasks
Creates a new @running object on shutdown while the ScheduledTask is checking the previously set @running object which will remain false
1 parent 25ccddc commit f3d097c

File tree

2 files changed

+19
-4
lines changed

2 files changed

+19
-4
lines changed

lib/concurrent-ruby/concurrent/timer_task.rb

+5-3
Original file line numberDiff line numberDiff line change
@@ -278,24 +278,26 @@ def ns_initialize(opts, &task)
278278
# @!visibility private
279279
def ns_shutdown_execution
280280
@running.make_false
281+
@running = Concurrent::AtomicBoolean.new(false)
281282
super
282283
end
283284

284285
# @!visibility private
285286
def ns_kill_execution
286287
@running.make_false
288+
@running = Concurrent::AtomicBoolean.new(false)
287289
super
288290
end
289291

290292
# @!visibility private
291293
def schedule_next_task(interval = execution_interval)
292-
ScheduledTask.execute(interval, args: [Concurrent::Event.new], &method(:execute_task))
294+
ScheduledTask.execute(interval, args: [Concurrent::Event.new, @running], &method(:execute_task))
293295
nil
294296
end
295297

296298
# @!visibility private
297-
def execute_task(completion)
298-
return nil unless @running.true?
299+
def execute_task(completion, continue_running)
300+
return nil unless continue_running.true?
299301
_success, value, reason = @executor.execute(self)
300302
if completion.try?
301303
self.value = value

spec/concurrent/timer_task_spec.rb

+14-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
require_relative 'concern/dereferenceable_shared'
22
require_relative 'concern/observable_shared'
3+
require 'concurrent/atomic/atomic_fixnum'
34
require 'concurrent/timer_task'
45

56
module Concurrent
@@ -94,12 +95,24 @@ def trigger_observable(observable)
9495
end
9596

9697
context '#shutdown' do
97-
9898
it 'returns true on success' do
9999
task = TimerTask.execute(run_now: false) { nil }
100100
sleep(0.1)
101101
expect(task.shutdown).to be_truthy
102102
end
103+
104+
it 'will cancel pre-shutdown task even if restarted to avoid double-runs' do
105+
counter = Concurrent::AtomicFixnum.new(0)
106+
task = TimerTask.execute(execution_interval: 0.2, run_now: true) { counter.increment }
107+
sleep 0.05
108+
expect(counter.value).to eq 1
109+
110+
task.shutdown
111+
task.execute
112+
113+
sleep 0.25
114+
expect(counter.value).to eq 3
115+
end
103116
end
104117
end
105118

0 commit comments

Comments
 (0)