Class: Sentry::Client

Inherits:
Object
  • Object
show all
Includes:
LoggingHelper
Defined in:
lib/sentry/client.rb

Instance Attribute Summary collapse

Instance Method Summary collapse

Constructor Details

#initialize(configuration) ⇒ Client

Returns a new instance of Client.

Parameters:



27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
# File 'lib/sentry/client.rb', line 27

def initialize(configuration)
  @configuration = configuration
  @sdk_logger = configuration.sdk_logger

  if transport_class = configuration.transport.transport_class
    @transport = transport_class.new(configuration)
  else
    @transport =
      case configuration.dsn&.scheme
      when "http", "https"
        HTTPTransport.new(configuration)
      else
        DummyTransport.new(configuration)
      end
  end

  @spotlight_transport = SpotlightTransport.new(configuration) if configuration.spotlight

  if configuration.enable_logs
    @log_event_buffer = LogEventBuffer.new(configuration, self).start
  end
end

Instance Attribute Details

#configurationConfiguration (readonly)

The Configuration object that’s used for configuring the client and its transport.

Returns:



24
25
26
# File 'lib/sentry/client.rb', line 24

def configuration
  @configuration
end

#spotlight_transportSpotlightTransport? (readonly)

The Transport object that’ll send events for the client.

Returns:



18
19
20
# File 'lib/sentry/client.rb', line 18

def spotlight_transport
  @spotlight_transport
end

#transportTransport (readonly)

The Transport object that’ll send events for the client.

Returns:



14
15
16
# File 'lib/sentry/client.rb', line 14

def transport
  @transport
end

Instance Method Details

#buffer_log_event(event, scope) ⇒ LogEvent

Buffer a log event to be sent later with other logs in a single envelope

Parameters:

  • event (LogEvent)

    the log event to be buffered

Returns:



101
102
103
104
105
# File 'lib/sentry/client.rb', line 101

def buffer_log_event(event, scope)
  return unless event.is_a?(LogEvent)
  @log_event_buffer.add_event(scope.apply_to_event(event))
  event
end

#capture_envelope(envelope) ⇒ void

This method returns an undefined value.

Capture an envelope directly.

Parameters:

  • envelope (Envelope)

    the envelope to be captured.



110
111
112
# File 'lib/sentry/client.rb', line 110

def capture_envelope(envelope)
  Sentry.background_worker.perform { send_envelope(envelope) }
end

#capture_event(event, scope, hint = {}) ⇒ Event?

Applies the given scope’s data to the event and sends it to Sentry.

Parameters:

  • event (Event)

    the event to be sent.

  • scope (Scope)

    the scope with contextual data that’ll be applied to the event before it’s sent.

  • hint (Hash) (defaults to: {})

    the hint data that’ll be passed to ‘before_send` callback and the scope’s event processors.

Returns:



55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
# File 'lib/sentry/client.rb', line 55

def capture_event(event, scope, hint = {})
  return unless configuration.sending_allowed?

  if event.is_a?(ErrorEvent) && !configuration.sample_allowed?
    transport.record_lost_event(:sample_rate, "error")
    return
  end

  event_type = event.is_a?(Event) ? event.type : event["type"]
  data_category = Envelope::Item.data_category(event_type)

  is_transaction = event.is_a?(TransactionEvent)
  spans_before = is_transaction ? event.spans.size : 0

  event = scope.apply_to_event(event, hint)

  if event.nil?
    log_debug("Discarded event because one of the event processors returned nil")
    transport.record_lost_event(:event_processor, data_category)
    transport.record_lost_event(:event_processor, "span", num: spans_before + 1) if is_transaction
    return
  elsif is_transaction
    spans_delta = spans_before - event.spans.size
    transport.record_lost_event(:event_processor, "span", num: spans_delta) if spans_delta > 0
  end

  if async_block = configuration.async
    dispatch_async_event(async_block, event, hint)
  elsif configuration.background_worker_threads != 0 && hint.fetch(:background, true)
    unless dispatch_background_event(event, hint)
      transport.record_lost_event(:queue_overflow, data_category)
      transport.record_lost_event(:queue_overflow, "span", num: spans_before + 1) if is_transaction
    end
  else
    send_event(event, hint)
  end

  event
rescue => e
  log_error("Event capturing failed", e, debug: configuration.debug)
  nil
end

#event_from_check_in(slug, status, hint = {}, duration: nil, monitor_config: nil, check_in_id: nil) ⇒ Event

Initializes a CheckInEvent object with the given options.

Parameters:

  • slug (String)

    identifier of this monitor

  • status (Symbol)

    status of this check-in, one of Sentry::CheckInEvent::VALID_STATUSES

  • hint (Hash) (defaults to: {})

    the hint data that’ll be passed to ‘before_send` callback and the scope’s event processors.

  • duration (Integer, nil) (defaults to: nil)

    seconds elapsed since this monitor started

  • monitor_config (Cron::MonitorConfig, nil) (defaults to: nil)

    configuration for this monitor

  • check_in_id (String, nil) (defaults to: nil)

    for updating the status of an existing monitor

Returns:



166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
# File 'lib/sentry/client.rb', line 166

def event_from_check_in(
  slug,
  status,
  hint = {},
  duration: nil,
  monitor_config: nil,
  check_in_id: nil
)
  return unless configuration.sending_allowed?

  CheckInEvent.new(
    configuration: configuration,
    integration_meta: Sentry.integrations[hint[:integration]],
    slug: slug,
    status: status,
    duration: duration,
    monitor_config: monitor_config,
    check_in_id: check_in_id
  )
end

#event_from_exception(exception, hint = {}) ⇒ Event?

Initializes an Event object with the given exception. Returns ‘nil` if the exception’s class is excluded from reporting.

Parameters:

  • exception (Exception)

    the exception to be reported.

  • hint (Hash) (defaults to: {})

    the hint data that’ll be passed to ‘before_send` callback and the scope’s event processors.

Returns:



126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
# File 'lib/sentry/client.rb', line 126

def event_from_exception(exception, hint = {})
  return unless @configuration.sending_allowed?

  ignore_exclusions = hint.delete(:ignore_exclusions) { false }
  return if !ignore_exclusions && !@configuration.exception_class_allowed?(exception)

  integration_meta = Sentry.integrations[hint[:integration]]
  mechanism = hint.delete(:mechanism) { Mechanism.new }

  ErrorEvent.new(configuration: configuration, integration_meta: integration_meta).tap do |event|
    event.add_exception_interface(exception, mechanism: mechanism)
    event.add_threads_interface(crashed: true)
    event.level = :error
  end
end

#event_from_log(message, level:, **options) ⇒ LogEvent

Initializes a LogEvent object with the given message and options

Parameters:

  • message (String)

    the log message

  • level (Symbol)

    the log level (:trace, :debug, :info, :warn, :error, :fatal)

  • options (Hash)

    additional options

Options Hash (**options):

  • :parameters (Array)

    Array of values to replace template tokens in the message

Returns:



195
196
197
198
199
200
201
# File 'lib/sentry/client.rb', line 195

def event_from_log(message, level:, **options)
  return unless configuration.sending_allowed?

  attributes = options.reject { |k, _| k == :level || k == :severity }

  LogEvent.new(level: level, body: message, attributes: attributes)
end

#event_from_message(message, hint = {}, backtrace: nil) ⇒ Event

Initializes an Event object with the given message.

Parameters:

  • message (String)

    the message to be reported.

  • hint (Hash) (defaults to: {})

    the hint data that’ll be passed to ‘before_send` callback and the scope’s event processors.

Returns:



146
147
148
149
150
151
152
153
154
# File 'lib/sentry/client.rb', line 146

def event_from_message(message, hint = {}, backtrace: nil)
  return unless @configuration.sending_allowed?

  integration_meta = Sentry.integrations[hint[:integration]]
  event = ErrorEvent.new(configuration: configuration, integration_meta: integration_meta, message: message)
  event.add_threads_interface(backtrace: backtrace || caller)
  event.level = :error
  event
end

#event_from_transaction(transaction) ⇒ TransactionEvent

Initializes an Event object with the given Transaction object.

Parameters:

  • transaction (Transaction)

    the transaction to be recorded.

Returns:



206
207
208
# File 'lib/sentry/client.rb', line 206

def event_from_transaction(transaction)
  TransactionEvent.new(configuration: configuration, transaction: transaction)
end

#flushvoid

This method returns an undefined value.

Flush pending events to Sentry.



116
117
118
119
120
# File 'lib/sentry/client.rb', line 116

def flush
  transport.flush if configuration.sending_to_dsn_allowed?
  spotlight_transport.flush if spotlight_transport
  @log_event_buffer&.flush
end

#generate_baggage(span) ⇒ String?

Deprecated.

Use Sentry.get_baggage instead.

Generates a W3C Baggage header for distributed tracing from the given Span. Returns ‘nil` if `config.propagate_traces` is `false`.

Parameters:

  • span (Span)

    the span to generate trace from.

Returns:

  • (String, nil)


356
357
358
359
360
361
362
363
364
365
366
# File 'lib/sentry/client.rb', line 356

def generate_baggage(span)
  return unless configuration.propagate_traces

  baggage = span.to_baggage

  if baggage && !baggage.empty?
    log_debug("[Tracing] Adding #{BAGGAGE_HEADER_NAME} header to outgoing request: #{baggage}")
  end

  baggage
end

#generate_sentry_trace(span) ⇒ String?

Deprecated.

use Sentry.get_traceparent instead.

Generates a Sentry trace for distribted tracing from the given Span. Returns ‘nil` if `config.propagate_traces` is `false`.

Parameters:

  • span (Span)

    the span to generate trace from.

Returns:

  • (String, nil)


342
343
344
345
346
347
348
# File 'lib/sentry/client.rb', line 342

def generate_sentry_trace(span)
  return unless configuration.propagate_traces

  trace = span.to_sentry_trace
  log_debug("[Tracing] Adding #{SENTRY_TRACE_HEADER_NAME} header to outgoing request: #{trace}")
  trace
end

#send_envelope(envelope) ⇒ void

This method returns an undefined value.

Send an envelope directly to Sentry.

Parameters:

  • envelope (Envelope)

    the envelope to be sent.



323
324
325
326
327
328
329
330
331
332
333
334
# File 'lib/sentry/client.rb', line 323

def send_envelope(envelope)
  transport.send_envelope(envelope) if configuration.sending_to_dsn_allowed?
  spotlight_transport.send_envelope(envelope) if spotlight_transport
rescue => e
  log_error("Envelope sending failed", e, debug: configuration.debug)

  envelope.items.map(&:data_category).each do |data_category|
    transport.record_lost_event(:network_error, data_category)
  end

  raise
end

#send_event(event, hint = nil) ⇒ Event

Sends the event to Sentry.

Parameters:

  • event (Event)

    the event to be sent.

  • hint (Hash) (defaults to: nil)

    the hint data that’ll be passed to ‘before_send` callback.

Returns:



211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
# File 'lib/sentry/client.rb', line 211

def send_event(event, hint = nil)
  event_type = event.is_a?(Event) ? event.type : event["type"]
  data_category = Envelope::Item.data_category(event_type)
  spans_before = event.is_a?(TransactionEvent) ? event.spans.size : 0

  if event_type != TransactionEvent::TYPE && configuration.before_send
    event = configuration.before_send.call(event, hint)

    case event
    when ErrorEvent, CheckInEvent
      # do nothing
    when Hash
      log_debug(<<~MSG)
        Returning a Hash from before_send is deprecated and will be removed in the next major version.
        Please return a Sentry::ErrorEvent object instead.
      MSG
    else
      # Avoid serializing the event object in this case because we aren't sure what it is and what it contains
      log_debug(<<~MSG)
        Discarded event because before_send didn't return a Sentry::ErrorEvent object but an instance of #{event.class}
      MSG
      transport.record_lost_event(:before_send, data_category)
      return
    end
  end

  if event_type == TransactionEvent::TYPE && configuration.before_send_transaction
    event = configuration.before_send_transaction.call(event, hint)

    if event.is_a?(TransactionEvent) || event.is_a?(Hash)
      spans_after = event.is_a?(TransactionEvent) ? event.spans.size : 0
      spans_delta = spans_before - spans_after
      transport.record_lost_event(:before_send, "span", num: spans_delta) if spans_delta > 0

      if event.is_a?(Hash)
        log_debug(<<~MSG)
          Returning a Hash from before_send_transaction is deprecated and will be removed in the next major version.
          Please return a Sentry::TransactionEvent object instead.
        MSG
      end
    else
      # Avoid serializing the event object in this case because we aren't sure what it is and what it contains
      log_debug(<<~MSG)
        Discarded event because before_send_transaction didn't return a Sentry::TransactionEvent object but an instance of #{event.class}
      MSG
      transport.record_lost_event(:before_send, "transaction")
      transport.record_lost_event(:before_send, "span", num: spans_before + 1)
      return
    end
  end

  transport.send_event(event) if configuration.sending_to_dsn_allowed?
  spotlight_transport.send_event(event) if spotlight_transport

  event
rescue => e
  log_error("Event sending failed", e, debug: configuration.debug)
  transport.record_lost_event(:network_error, data_category)
  transport.record_lost_event(:network_error, "span", num: spans_before + 1) if event.is_a?(TransactionEvent)
  raise
end

#send_logs(log_events) ⇒ void

This method is part of a private API. You should avoid using this method if possible, as it may be removed or be changed in the future.

This method returns an undefined value.

Send an envelope with batched logs

Parameters:

  • log_events (Array<LogEvent>)

    the log events to be sent



277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
# File 'lib/sentry/client.rb', line 277

def send_logs(log_events)
  envelope = Envelope.new(
    event_id: Sentry::Utils.uuid,
    sent_at: Sentry.utc_now.iso8601,
    dsn: configuration.dsn,
    sdk: Sentry.sdk_meta
  )

  discarded_count = 0
  envelope_items = []

  if configuration.before_send_log
    log_events.each do |log_event|
      processed_log_event = configuration.before_send_log.call(log_event)

      if processed_log_event
        envelope_items << processed_log_event.to_hash
      else
        discarded_count += 1
      end
    end

    envelope_items
  else
    envelope_items = log_events.map(&:to_hash)
  end

  envelope.add_item(
    {
      type: "log",
      item_count: envelope_items.size,
      content_type: "application/vnd.sentry.items.log+json"
    },
    { items: envelope_items }
  )

  send_envelope(envelope)

  unless discarded_count.zero?
    transport.record_lost_event(:before_send, "log_item", num: discarded_count)
  end
end