123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146 |
- # 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
|