-
-
Notifications
You must be signed in to change notification settings - Fork 206
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Fix cron double-enqueue because delay close to 0.01 and possibly clock-drift #1543
Conversation
@bensheldon thanks for your help. |
@ccouton doublecheck my math, but I believe this line should prevent Concurrent Ruby from ever immediately running the task: |
Yes but it will be scheludled 0.02s after so a risk to be before the next at and so at the same of the previous at |
oh right! we do want to run it immediately if it is in fact in the past. I'll tweak that. |
Sorry i'm not clear enough, we have the case when clock drift cause an execution just before the real next scheduled at. Here i think this issue is not fixed. And maybe not easy to fix. We have ti execute it immeditaly if it's the first execution but if it's an other if cron_at is leather than current for me there is no reason to execute it? |
@ccouton this should (theoretically, assuming I did it right) prevent the I don't have a strategy for preventing the case where some delay causes a previously scheduled time to run at about the same time as the next scheduled time. I would say that if your schedule is at such a cadence that you frequently run into that (like every second), cron is maybe the wrong tool for that and it would be better to manage your own sleep loop. |
maybe we can compare at the entry of the scheduled task that the thr_cron_at and previously_at are not equal ? and if it's the case we don't enqueue and reschedule the task again? wdyt? |
I tested it locally in my test.rb and it solves the issue def create_task(previous_cron_at = nil)
cron_at = @fugit.next_time.to_t
puts Time.now.to_f
delay = [(cron_at - Time.now).to_f, 0].max
puts "delay = #{delay}"
future = Concurrent::ScheduledTask.new(delay, args: [self, cron_at, previous_cron_at]) do |tf, cron_at, previous_cron_at|
puts "previous_cron_at / cron_at (#{previous_cron_at} / #{cron_at})"
if previous_cron_at == cron_at
puts "!!!! previous_cron_at == cron_at !!!! (#{previous_cron_at} #{cron_at})"
tf.create_task(previous_cron_at)
else
tf.create_task(cron_at)
puts "Exec task #{Time.now.to_f}"
end
end
future.execute
end |
I don't like guarding against I wanted to ask, do you not like what I've proposed:
This is the version of your script with my strategy (though I have still haven't been able to reproduce the behavior you're seeing when I run the script): require 'concurrent-ruby'
require 'time'
require 'fugit'
class TestFugit
def initialize
@fugit = Fugit.parse('*/1 * * * *')
end
def create_task(at: nil)
cron_at = at || @fugit.next_time.to_t
puts Time.now.to_f
delay = cron_at <= Time.now ? 0.0 : [(cron_at - Time.now).to_f, 0.02].max
puts "delay = #{delay}"
future = Concurrent::ScheduledTask.new(delay, args: [self, cron_at]) do |tf, thr_cron_at|
if thr_cron_at > Time.now
puts "Too soon, rescheduling"
tf.create_task(at: thr_cron_at)
else
tf.create_task
puts "Exec task #{Time.now.to_f}"
end
end
future.execute
end
end
TestFugit.new.create_task
sleep 1000 |
seems working ok for that solution |
when you will merge it? |
Fixes one problem in #1536
Case of clock drift and delay calculation result close to 0.001 so task is executed immediately which causes an unnecessary
duplicate key value violates unique constraint
-error even on single-process installations.