1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- module MonitoringHelper
- class HealthChecker
- class Scheduler < Backend
- include ActionView::Helpers::DateHelper
- LAST_EXECUTION_TOLERANCE = 8.minutes
- def run_health_check
- last_execution
- none_running
- failed_jobs
- end
- private
- def last_execution
- last_execution_scope.each do |scheduler|
- next if last_execution_on_time?(scheduler)
- response.issues.push "scheduler may not run (last execution of #{scheduler.method} #{time_ago_in_words(scheduler.last_run)} ago) - please contact your system administrator"
- break
- end
- end
- def last_execution_scope
- ::Scheduler
- .where('active = ? AND period > 300', true)
- .where.not(last_run: nil)
- .reorder(last_run: :asc, period: :asc)
- end
- def last_execution_deadline(scheduler)
- return scheduler.last_run if scheduler.timeplan.blank?
- calculator = TimeplanCalculation.new(scheduler.timeplan, Setting.get('timezone_default'))
- intermediary = calculator.next_at(scheduler.last_run + 10.minutes)
- calculator.next_at(intermediary + 10.minutes)
- end
- def last_execution_on_time?(scheduler)
- return false if scheduler.last_run.blank?
- last_execution_deadline(scheduler) + scheduler.period.seconds >= LAST_EXECUTION_TOLERANCE.ago
- end
- def none_running
- return if ::Scheduler.where(active: true).where.not(last_run: nil).exists?
- response.issues.push 'scheduler not running'
- end
- def failed_jobs
- ::Scheduler.failed_jobs.each do |job|
- response.issues.push "Failed to run scheduled job '#{job.name}'. Cause: #{job.error_message}"
- response.actions.add(:restart_failed_jobs)
- end
- end
- end
- end
- end
|