mirror of https://github.com/docusealco/docuseal
				
				
				
			
			You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							109 lines
						
					
					
						
							3.1 KiB
						
					
					
				
			
		
		
	
	
							109 lines
						
					
					
						
							3.1 KiB
						
					
					
				| # frozen_string_literal: true
 | |
| 
 | |
| module SendWebhookRequest
 | |
|   USER_AGENT = 'DocuSeal.com Webhook'
 | |
| 
 | |
|   LOCALHOSTS = %w[0.0.0.0 127.0.0.1 localhost].freeze
 | |
| 
 | |
|   MANUAL_ATTEMPT = 99_999
 | |
|   AUTOMATED_RETRY_RANGE = 1..MANUAL_ATTEMPT - 1
 | |
| 
 | |
|   HttpsError = Class.new(StandardError)
 | |
|   LocalhostError = Class.new(StandardError)
 | |
| 
 | |
|   module_function
 | |
| 
 | |
|   # rubocop:disable Metrics/AbcSize
 | |
|   def call(webhook_url, event_uuid:, event_type:, record:, data:, attempt: 0)
 | |
|     uri = begin
 | |
|       URI(webhook_url.url)
 | |
|     rescue URI::Error
 | |
|       Addressable::URI.parse(webhook_url.url).normalize
 | |
|     end
 | |
| 
 | |
|     if Docuseal.multitenant?
 | |
|       raise HttpsError, 'Only HTTPS is allowed.' if uri.scheme != 'https' &&
 | |
|                                                     !AccountConfig.exists?(key: :allow_http,
 | |
|                                                                            account_id: webhook_url.account_id)
 | |
|       raise LocalhostError, "Can't send to localhost." if uri.host.in?(LOCALHOSTS)
 | |
|     end
 | |
| 
 | |
|     webhook_event = create_webhook_event(webhook_url, event_uuid:, event_type:, record:)
 | |
| 
 | |
|     return if AUTOMATED_RETRY_RANGE.cover?(attempt.to_i) && webhook_event&.status == 'success'
 | |
| 
 | |
|     response = Faraday.post(uri) do |req|
 | |
|       req.headers['Content-Type'] = 'application/json'
 | |
|       req.headers['User-Agent'] = USER_AGENT
 | |
|       req.headers.merge!(webhook_url.secret.to_h) if webhook_url.secret.present?
 | |
| 
 | |
|       req.body = {
 | |
|         event_type: event_type,
 | |
|         timestamp: webhook_event&.created_at || Time.current,
 | |
|         data: data
 | |
|       }.to_json
 | |
| 
 | |
|       req.options.read_timeout = 8
 | |
|       req.options.open_timeout = 8
 | |
|     end
 | |
| 
 | |
|     handle_response(webhook_event, response:, attempt:)
 | |
|   rescue Faraday::SSLError, Faraday::TimeoutError, Faraday::ConnectionFailed => e
 | |
|     handle_error(webhook_event, attempt:, error_message: e.class.name.split('::').last)
 | |
|   rescue Faraday::Error => e
 | |
|     handle_error(webhook_event, attempt:, error_message: e.message&.truncate(100))
 | |
|   end
 | |
|   # rubocop:enable Metrics/AbcSize
 | |
| 
 | |
|   def create_webhook_event(webhook_url, event_uuid:, event_type:, record:)
 | |
|     return if event_uuid.blank?
 | |
| 
 | |
|     WebhookEvent.create_with(
 | |
|       event_type:,
 | |
|       record:,
 | |
|       account_id: webhook_url.account_id,
 | |
|       status: 'pending'
 | |
|     ).find_or_create_by!(webhook_url:, uuid: event_uuid)
 | |
|   end
 | |
| 
 | |
|   def handle_response(webhook_event, response:, attempt:)
 | |
|     return response unless webhook_event
 | |
| 
 | |
|     is_error = response.status.to_i >= 400
 | |
| 
 | |
|     WebhookAttempt.create!(
 | |
|       webhook_event:,
 | |
|       response_body: is_error ? response.body&.truncate(100) : nil,
 | |
|       response_status_code: response.status,
 | |
|       attempt:
 | |
|     )
 | |
| 
 | |
|     webhook_event.update!(status: is_error ? 'error' : 'success')
 | |
| 
 | |
|     response
 | |
|   rescue StandardError
 | |
|     raise if Rails.env.local?
 | |
| 
 | |
|     nil
 | |
|   end
 | |
| 
 | |
|   def handle_error(webhook_event, error_message:, attempt:)
 | |
|     return unless webhook_event
 | |
| 
 | |
|     WebhookAttempt.create!(
 | |
|       webhook_event:,
 | |
|       response_body: error_message,
 | |
|       response_status_code: 0,
 | |
|       attempt:
 | |
|     )
 | |
| 
 | |
|     webhook_event.update!(status: 'error')
 | |
| 
 | |
|     nil
 | |
|   rescue StandardError
 | |
|     raise if Rails.env.local?
 | |
| 
 | |
|     nil
 | |
|   end
 | |
| end
 |