# Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/ class TriggerWebhookJob < ApplicationJob attr_reader :ticket, :trigger, :article, :changes, :user_id, :execution_type, :event_type retry_on TriggerWebhookJob::RequestError, attempts: 5, wait: lambda { |executions| executions * 10.seconds } discard_on(ActiveJob::DeserializationError) do |_job, e| Rails.logger.info 'Trigger, Ticket or Article may got removed before TriggerWebhookJob could be executed. Discarding job. See exception for further details.' Rails.logger.info e end def perform(trigger, ticket, article, changes:, user_id:, execution_type:, event_type:) @trigger = trigger @ticket = ticket @article = article @changes = changes @user_id = user_id @execution_type = execution_type @event_type = event_type return if abort? return if request.success? raise TriggerWebhookJob::RequestError end private def abort? if webhook_id.blank? log_wrong_trigger_config return true elsif webhook.blank? log_not_existing_webhook return true end false end def webhook_id @webhook_id ||= trigger.perform.dig('notification.webhook', 'webhook_id') end def webhook @webhook ||= begin Webhook.find_by( id: webhook_id, active: true ) end end def log_wrong_trigger_config Rails.logger.error "Can't find webhook_id for Trigger '#{trigger.name}' with ID #{trigger.id}" end def log_not_existing_webhook Rails.logger.error "Can't find Webhook for ID #{webhook_id} configured in Trigger '#{trigger.name}' with ID #{trigger.id}" end def request UserAgent.post( webhook.endpoint, payload, { json: true, jsonParseDisable: true, open_timeout: 4, read_timeout: 30, total_timeout: 60, headers: headers, signature_token: webhook.signature_token, verify_ssl: webhook.ssl_verify, user: webhook.basic_auth_username, password: webhook.basic_auth_password, log: { facility: 'webhook', }, }, ) end def headers { 'X-Zammad-Trigger' => trigger.name, 'X-Zammad-Delivery' => job_id } end def default_payload { ticket: TriggerWebhookJob::RecordPayload.generate(ticket), article: TriggerWebhookJob::RecordPayload.generate(article), } end def payload return generate_custom_payload if webhook.customized_payload || webhook.pre_defined_webhook_type.present? default_payload end def pre_defined_webhook_payload TriggerWebhookJob::CustomPayload::Track::PreDefinedWebhook.payload(webhook.pre_defined_webhook_type) end def generate_custom_payload tracks = { ticket:, article: } add_custom_tracks(tracks) payload = webhook.customized_payload ? webhook.custom_payload : pre_defined_webhook_payload return default_payload if payload.nil? hash = TriggerWebhookJob::CustomPayload.generate(payload, tracks) return hash if webhook.customized_payload pre_defined_webhook = "Webhook::PreDefined::#{webhook.pre_defined_webhook_type}".constantize.new return hash if !pre_defined_webhook.respond_to?(:post_replace) pre_defined_webhook.post_replace(hash, tracks) end def add_custom_tracks(tracks) data = { event: { type: event_type, execution: execution_type, changes:, user_id: }, webhook: webhook } TriggerWebhookJob::CustomPayload.tracks.each do |track| next if !track.respond_to?(:generate) track.generate(tracks, data) end end end