2021-01-29 00:20:46 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
class MergeRequest::CleanupSchedule < ApplicationRecord
|
2021-09-30 23:02:18 +05:30
|
|
|
STATUSES = {
|
|
|
|
unstarted: 0,
|
|
|
|
running: 1,
|
|
|
|
completed: 2,
|
|
|
|
failed: 3
|
|
|
|
}.freeze
|
|
|
|
|
2021-01-29 00:20:46 +05:30
|
|
|
belongs_to :merge_request, inverse_of: :cleanup_schedule
|
|
|
|
|
|
|
|
validates :scheduled_at, presence: true
|
|
|
|
|
2021-09-30 23:02:18 +05:30
|
|
|
state_machine :status, initial: :unstarted do
|
|
|
|
state :unstarted, value: STATUSES[:unstarted]
|
|
|
|
state :running, value: STATUSES[:running]
|
|
|
|
state :completed, value: STATUSES[:completed]
|
|
|
|
state :failed, value: STATUSES[:failed]
|
|
|
|
|
|
|
|
event :run do
|
|
|
|
transition unstarted: :running
|
|
|
|
end
|
|
|
|
|
|
|
|
event :retry do
|
|
|
|
transition running: :unstarted
|
|
|
|
end
|
|
|
|
|
|
|
|
event :complete do
|
|
|
|
transition running: :completed
|
|
|
|
end
|
|
|
|
|
|
|
|
event :mark_as_failed do
|
|
|
|
transition running: :failed
|
|
|
|
end
|
|
|
|
|
|
|
|
before_transition to: [:completed] do |cleanup_schedule, _transition|
|
|
|
|
cleanup_schedule.completed_at = Time.current
|
|
|
|
end
|
|
|
|
|
|
|
|
before_transition from: :running, to: [:unstarted, :failed] do |cleanup_schedule, _transition|
|
|
|
|
cleanup_schedule.failed_count += 1
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
scope :scheduled_and_unstarted, -> {
|
|
|
|
where('completed_at IS NULL AND scheduled_at <= NOW() AND status = ?', STATUSES[:unstarted])
|
2021-01-29 00:20:46 +05:30
|
|
|
.order('scheduled_at DESC')
|
2021-09-30 23:02:18 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
def self.start_next
|
|
|
|
MergeRequest::CleanupSchedule.transaction do
|
|
|
|
cleanup_schedule = scheduled_and_unstarted.lock('FOR UPDATE SKIP LOCKED').first
|
|
|
|
|
|
|
|
next if cleanup_schedule.blank?
|
|
|
|
|
|
|
|
cleanup_schedule.run!
|
|
|
|
cleanup_schedule
|
|
|
|
end
|
2021-01-29 00:20:46 +05:30
|
|
|
end
|
|
|
|
end
|