summaryrefslogtreecommitdiff
path: root/lib/gitlab/error_tracking/processor/grpc_error_processor.rb
blob: 871e9c4b7c874aa1711c57cd25ca911c3c2bfdb3 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
# frozen_string_literal: true

module Gitlab
  module ErrorTracking
    module Processor
      class GrpcErrorProcessor < ::Raven::Processor
        DEBUG_ERROR_STRING_REGEX = RE2('(.*) debug_error_string:(.*)')

        def process(value)
          process_first_exception_value(value)
          process_custom_fingerprint(value)

          value
        end

        # Sentry can report multiple exceptions in an event. Sanitize
        # only the first one since that's what is used for grouping.
        def process_first_exception_value(value)
          exceptions = value.dig(:exception, :values)

          return unless exceptions.is_a?(Array)

          entry = exceptions.first

          return unless entry.is_a?(Hash)

          exception_type = entry[:type]
          raw_message = entry[:value]

          return unless exception_type&.start_with?('GRPC::')
          return unless raw_message.present?

          message, debug_str = split_debug_error_string(raw_message)

          entry[:value] = message if message
          extra = value[:extra] || {}
          extra[:grpc_debug_error_string] = debug_str if debug_str
        end

        def process_custom_fingerprint(value)
          fingerprint = value[:fingerprint]

          return value unless custom_grpc_fingerprint?(fingerprint)

          message, _ = split_debug_error_string(fingerprint[1])
          fingerprint[1] = message if message
        end

        private

        def custom_grpc_fingerprint?(fingerprint)
          fingerprint.is_a?(Array) && fingerprint.length == 2 && fingerprint[0].start_with?('GRPC::')
        end

        def split_debug_error_string(message)
          return unless message

          match = DEBUG_ERROR_STRING_REGEX.match(message)

          return unless match

          [match[1], match[2]]
        end
      end
    end
  end
end