123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- class Job < ApplicationModel
- include ChecksClientNotification
- include ChecksConditionValidation
- include ChecksHtmlSanitized
- include HasTimeplan
- include HasSearchIndexBackend
- include CanSelector
- include CanSearch
- include Job::Assets
- include Job::SearchIndex
- OBJECTS_BATCH_SIZE = 2_000
- store :condition
- store :perform
- validates :name, presence: true, uniqueness: { case_sensitive: false }
- validates :object, presence: true, inclusion: { in: %w[Ticket User Organization] }
- validates :perform, 'validations/verify_perform_rules': true
- before_save :updated_matching, :update_next_run_at
- validates :note, length: { maximum: 250 }
- sanitized_html :note
- =begin
- verify each job if needed to run (e. g. if true and times are matching) and execute it
- Job.run
- =end
- def self.run
- start_at = Time.zone.now
- Job.where(active: true).each do |job|
- next if !job.executable?
- job.run(false, start_at)
- end
- true
- end
- =begin
- execute a single job if needed (e. g. if true and times are matching)
- job = Job.find(123)
- job.run
- force to run job (ignore times are matching)
- job.run(true)
- =end
- def run(force = false, start_at = Time.zone.now)
- logger.debug { "Execute job #{inspect}" }
- object_ids = start_job(start_at, force)
- return if object_ids.nil?
- object_ids.each_slice(10) do |slice|
- run_slice(slice)
- end
- finish_job
- end
- def executable?(start_at = Time.zone.now)
- return false if !active
- # only execute jobs older than 1 min to give admin time to make last-minute changes
- return false if updated_at > 1.minute.ago
- # check if job got stuck
- return false if running == true && last_run_at && 1.day.ago < last_run_at
- # check if jobs need to be executed
- # ignore if job was running within last 10 min.
- return false if last_run_at && last_run_at > start_at - 10.minutes
- true
- end
- def matching_count
- object_count, _objects = object.constantize.selectors(condition, limit: 1, execution_time: true)
- object_count || 0
- end
- private
- def next_run_at_calculate(time = Time.zone.now)
- return nil if !active
- if last_run_at && (time - last_run_at).positive?
- time += 10.minutes
- end
- timeplan_calculation.next_at(time)
- end
- def updated_matching
- self.matching = matching_count
- end
- def update_next_run_at
- self.next_run_at = next_run_at_calculate
- end
- def finish_job
- Transaction.execute(reset_user_id: true) do
- mark_as_finished
- end
- end
- def mark_as_finished
- self.running = false
- self.last_run_at = Time.zone.now
- save!
- end
- def start_job(start_at, force)
- Transaction.execute(reset_user_id: true) do
- next if !start_job_executable?(start_at, force)
- next if !start_job_in_timeplan?(start_at, force)
- object_count, objects = object.constantize.selectors(condition, limit: OBJECTS_BATCH_SIZE, execution_time: true)
- logger.debug { "Job #{name} with #{object_count} object(s)" }
- mark_as_started(objects&.count || 0)
- objects&.pluck(:id) || []
- end
- end
- def start_job_executable?(start_at, force)
- return true if executable?(start_at) || force
- if next_run_at && next_run_at <= Time.zone.now
- save!
- end
- false
- end
- def start_job_in_timeplan?(start_at, force)
- return true if in_timeplan?(start_at) || force
- save! # trigger updating matching tickets count and next_run_at time even if not in timeplan
- false
- end
- def mark_as_started(batch_count)
- self.processed = batch_count
- self.running = true
- self.last_run_at = Time.zone.now
- save!
- end
- def run_slice(slice)
- Transaction.execute(disable_notification: disable_notification, reset_user_id: true) do
- _, objects = object.constantize.selectors(condition, limit: OBJECTS_BATCH_SIZE, execution_time: true)
- return if objects.nil?
- objects
- .where(id: slice)
- .each do |object|
- object.perform_changes(self, 'job')
- end
- end
- end
- end
|