# Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/ class BackgroundServices def self.available_services BackgroundServices::Service.descendants end attr_reader :config def initialize(config) @config = Array(config) end def run Rails.logger.debug 'Starting BackgroundServices...' # Fork before starting the threads in the main process to ensure a consistent state # and minimal memory overhead (see also #5420). config .in_order_of(:start_as, %i[fork thread]) .each do |service_config| run_service service_config end Process.waitall loop do sleep 1 end rescue Interrupt nil ensure Rails.logger.debug('Stopping BackgroundServices.') end private def run_service(service_config) if !service_config.enabled? Rails.logger.debug { "Skipping disabled service #{service_config.service.service_name}." } return end service_config.service.pre_run case service_config.start_as when :fork start_as_forks(service_config.service, service_config.workers) when :thread start_as_thread(service_config.service) end end def start_as_forks(service, forks) (1..forks).map do Process.fork do Rails.logger.debug { "Starting process ##{Process.pid} for service #{service.service_name}." } service.new.run rescue Interrupt nil end end end def start_as_thread(service) Thread.new do Thread.current.abort_on_exception = true Rails.logger.debug { "Starting thread for service #{service.service_name} in the main process." } service.new.run end end end