require "json" require "base64" module Sentry class Transport PROTOCOL_VERSION = '5' USER_AGENT = "sentry-ruby/#{Sentry::VERSION}" attr_accessor :configuration def initialize(configuration) @configuration = configuration @transport_configuration = configuration.transport @dsn = configuration.dsn end def send_data(data, options = {}) raise NotImplementedError end def send_event(event) unless configuration.sending_allowed? configuration.logger.debug(LOGGER_PROGNAME) { "Event not sent: #{configuration.error_messages}" } return end encoded_data = prepare_encoded_event(event) return nil unless encoded_data send_data(encoded_data) event rescue => e failed_for_exception(e, event) nil end def generate_auth_header now = Sentry.utc_now.to_i fields = { 'sentry_version' => PROTOCOL_VERSION, 'sentry_client' => USER_AGENT, 'sentry_timestamp' => now, 'sentry_key' => @dsn.public_key } fields['sentry_secret'] = @dsn.secret_key if @dsn.secret_key 'Sentry ' + fields.map { |key, value| "#{key}=#{value}" }.join(', ') end def encode(event_hash) event_id = event_hash[:event_id] || event_hash['event_id'] event_type = event_hash[:type] || event_hash['type'] envelope = <<~ENVELOPE {"event_id":"#{event_id}","dsn":"#{configuration.dsn.to_s}","sdk":#{Sentry.sdk_meta.to_json},"sent_at":"#{Sentry.utc_now.iso8601}"} {"type":"#{event_type}","content_type":"application/json"} #{JSON.generate(event_hash)} ENVELOPE envelope end private def prepare_encoded_event(event) # Convert to hash event_hash = event.to_hash event_id = event_hash[:event_id] || event_hash['event_id'] configuration.logger.info(LOGGER_PROGNAME) { "Sending event #{event_id} to Sentry" } encode(event_hash) end def failed_for_exception(e, event) configuration.logger.warn(LOGGER_PROGNAME) { "Unable to record event with remote Sentry server (#{e.class} - #{e.message}):\n#{e.backtrace[0..10].join("\n")}" } log_not_sending(event) end def log_not_sending(event) configuration.logger.warn(LOGGER_PROGNAME) { "Failed to submit event. Unreported Event: #{Event.get_log_message(event.to_hash)}" } end end end require "sentry/transport/dummy_transport" require "sentry/transport/http_transport"