dogstatsd-ruby 3.3.0 → 5.3.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,161 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Datadog
4
+ class Statsd
5
+ # Sender is using a companion thread to flush and pack messages
6
+ # in a `MessageBuffer`.
7
+ # The communication with this thread is done using a `Queue`.
8
+ # If the thread is dead, it is starting a new one to avoid having a blocked
9
+ # Sender with no companion thread to communicate with (most of the time, having
10
+ # a dead companion thread means that a fork just happened and that we are
11
+ # running in the child process).
12
+ class Sender
13
+ CLOSEABLE_QUEUES = Queue.instance_methods.include?(:close)
14
+
15
+ def initialize(message_buffer, logger: nil)
16
+ @message_buffer = message_buffer
17
+ @logger = logger
18
+ @mx = Mutex.new
19
+ end
20
+
21
+ def flush(sync: false)
22
+ # keep a copy around in case another thread is calling #stop while this method is running
23
+ current_message_queue = message_queue
24
+
25
+ # don't try to flush if there is no message_queue instantiated or
26
+ # no companion thread running
27
+ if !current_message_queue
28
+ @logger.debug { "Statsd: can't flush: no message queue ready" } if @logger
29
+ return
30
+ end
31
+ if !sender_thread.alive?
32
+ @logger.debug { "Statsd: can't flush: no sender_thread alive" } if @logger
33
+ return
34
+ end
35
+
36
+ current_message_queue.push(:flush)
37
+ rendez_vous if sync
38
+ end
39
+
40
+ def rendez_vous
41
+ # could happen if #start hasn't be called
42
+ return unless message_queue
43
+
44
+ # Initialize and get the thread's sync queue
45
+ queue = (Thread.current[:statsd_sync_queue] ||= Queue.new)
46
+ # tell sender-thread to notify us in the current
47
+ # thread's queue
48
+ message_queue.push(queue)
49
+ # wait for the sender thread to send a message
50
+ # once the flush is done
51
+ queue.pop
52
+ end
53
+
54
+ def add(message)
55
+ raise ArgumentError, 'Start sender first' unless message_queue
56
+
57
+ # if the thread does not exist, we assume we are running in a forked process,
58
+ # empty the message queue and message buffers (these messages belong to
59
+ # the parent process) and spawn a new companion thread.
60
+ if !sender_thread.alive?
61
+ @mx.synchronize {
62
+ # a call from another thread has already re-created
63
+ # the companion thread before this one acquired the lock
64
+ break if sender_thread.alive?
65
+ @logger.debug { "Statsd: companion thread is dead, re-creating one" } if @logger
66
+
67
+ message_queue.close if CLOSEABLE_QUEUES
68
+ @message_queue = nil
69
+ message_buffer.reset
70
+ start
71
+ }
72
+ end
73
+
74
+ message_queue << message
75
+ end
76
+
77
+ def start
78
+ raise ArgumentError, 'Sender already started' if message_queue
79
+
80
+ # initialize a new message queue for the background thread
81
+ @message_queue = Queue.new
82
+ # start background thread
83
+ @sender_thread = Thread.new(&method(:send_loop))
84
+ end
85
+
86
+ if CLOSEABLE_QUEUES
87
+ # when calling stop, make sure that no other threads is trying
88
+ # to close the sender nor trying to continue to `#add` more message
89
+ # into the sender.
90
+ def stop(join_worker: true)
91
+ message_queue = @message_queue
92
+ message_queue.close if message_queue
93
+
94
+ sender_thread = @sender_thread
95
+ sender_thread.join if sender_thread && join_worker
96
+ end
97
+ else
98
+ # when calling stop, make sure that no other threads is trying
99
+ # to close the sender nor trying to continue to `#add` more message
100
+ # into the sender.
101
+ def stop(join_worker: true)
102
+ message_queue = @message_queue
103
+ message_queue << :close if message_queue
104
+
105
+ sender_thread = @sender_thread
106
+ sender_thread.join if sender_thread && join_worker
107
+ end
108
+ end
109
+
110
+ private
111
+
112
+ attr_reader :message_buffer
113
+ attr_reader :message_queue
114
+ attr_reader :sender_thread
115
+
116
+ if CLOSEABLE_QUEUES
117
+ def send_loop
118
+ until (message = message_queue.pop).nil? && message_queue.closed?
119
+ # skip if message is nil, e.g. when message_queue
120
+ # is empty and closed
121
+ next unless message
122
+
123
+ case message
124
+ when :flush
125
+ message_buffer.flush
126
+ when Queue
127
+ message.push(:go_on)
128
+ else
129
+ message_buffer.add(message)
130
+ end
131
+ end
132
+
133
+ @message_queue = nil
134
+ @sender_thread = nil
135
+ end
136
+ else
137
+ def send_loop
138
+ loop do
139
+ message = message_queue.pop
140
+
141
+ next unless message
142
+
143
+ case message
144
+ when :close
145
+ break
146
+ when :flush
147
+ message_buffer.flush
148
+ when Queue
149
+ message.push(:go_on)
150
+ else
151
+ message_buffer.add(message)
152
+ end
153
+ end
154
+
155
+ @message_queue = nil
156
+ @sender_thread = nil
157
+ end
158
+ end
159
+ end
160
+ end
161
+ end
@@ -0,0 +1,71 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Datadog
4
+ class Statsd
5
+ module Serialization
6
+ class EventSerializer
7
+ EVENT_BASIC_OPTIONS = {
8
+ date_happened: 'd:',
9
+ hostname: 'h:',
10
+ aggregation_key: 'k:',
11
+ priority: 'p:',
12
+ source_type_name: 's:',
13
+ alert_type: 't:',
14
+ }.freeze
15
+
16
+ def initialize(global_tags: [])
17
+ @tag_serializer = TagSerializer.new(global_tags)
18
+ end
19
+
20
+ def format(title, text, options = EMPTY_OPTIONS)
21
+ title = escape(title)
22
+ text = escape(text)
23
+
24
+ String.new.tap do |event|
25
+ event << '_e{'
26
+ event << title.bytesize.to_s
27
+ event << ','
28
+ event << text.bytesize.to_s
29
+ event << '}:'
30
+ event << title
31
+ event << '|'
32
+ event << text
33
+
34
+ # we are serializing the generic service check options
35
+ # before serializing specialized options that need edge-cases
36
+ EVENT_BASIC_OPTIONS.each do |option_key, shortcut|
37
+ if value = options[option_key]
38
+ event << '|'
39
+ event << shortcut
40
+ event << value.to_s.delete('|')
41
+ end
42
+ end
43
+
44
+ # also returns the global tags from serializer
45
+ if tags = tag_serializer.format(options[:tags])
46
+ event << '|#'
47
+ event << tags
48
+ end
49
+
50
+ if event.bytesize > MAX_EVENT_SIZE
51
+ if options[:truncate_if_too_long]
52
+ event.slice!(MAX_EVENT_SIZE..event.length)
53
+ else
54
+ raise "Event #{title} payload is too big (more that 8KB), event discarded"
55
+ end
56
+ end
57
+ end
58
+ end
59
+
60
+ protected
61
+ attr_reader :tag_serializer
62
+
63
+ def escape(text)
64
+ text.delete('|').tap do |t|
65
+ t.gsub!("\n", '\n')
66
+ end
67
+ end
68
+ end
69
+ end
70
+ end
71
+ end
@@ -0,0 +1,41 @@
1
+ # frozen_string_literal: true
2
+
3
+ # require 'forwardable'
4
+
5
+ module Datadog
6
+ class Statsd
7
+ module Serialization
8
+ class Serializer
9
+ def initialize(prefix: nil, global_tags: [])
10
+ @stat_serializer = StatSerializer.new(prefix, global_tags: global_tags)
11
+ @service_check_serializer = ServiceCheckSerializer.new(global_tags: global_tags)
12
+ @event_serializer = EventSerializer.new(global_tags: global_tags)
13
+ end
14
+
15
+ # using *args would make new allocations
16
+ def to_stat(name, delta, type, tags: [], sample_rate: 1)
17
+ stat_serializer.format(name, delta, type, tags: tags, sample_rate: sample_rate)
18
+ end
19
+
20
+ # using *args would make new allocations
21
+ def to_service_check(name, status, options = EMPTY_OPTIONS)
22
+ service_check_serializer.format(name, status, options)
23
+ end
24
+
25
+ # using *args would make new allocations
26
+ def to_event(title, text, options = EMPTY_OPTIONS)
27
+ event_serializer.format(title, text, options)
28
+ end
29
+
30
+ def global_tags
31
+ stat_serializer.global_tags
32
+ end
33
+
34
+ protected
35
+ attr_reader :stat_serializer
36
+ attr_reader :service_check_serializer
37
+ attr_reader :event_serializer
38
+ end
39
+ end
40
+ end
41
+ end
@@ -0,0 +1,60 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Datadog
4
+ class Statsd
5
+ module Serialization
6
+ class ServiceCheckSerializer
7
+ SERVICE_CHECK_BASIC_OPTIONS = {
8
+ timestamp: 'd:',
9
+ hostname: 'h:',
10
+ }.freeze
11
+
12
+ def initialize(global_tags: [])
13
+ @tag_serializer = TagSerializer.new(global_tags)
14
+ end
15
+
16
+ def format(name, status, options = EMPTY_OPTIONS)
17
+ String.new.tap do |service_check|
18
+ # line basics
19
+ service_check << "_sc"
20
+ service_check << "|"
21
+ service_check << name.to_s
22
+ service_check << "|"
23
+ service_check << status.to_s
24
+
25
+ # we are serializing the generic service check options
26
+ # before serializing specialized options that need edge-cases
27
+ SERVICE_CHECK_BASIC_OPTIONS.each do |option_key, shortcut|
28
+ if value = options[option_key]
29
+ service_check << '|'
30
+ service_check << shortcut
31
+ service_check << value.to_s.delete('|')
32
+ end
33
+ end
34
+
35
+ if message = options[:message]
36
+ service_check << '|m:'
37
+ service_check << escape_message(message)
38
+ end
39
+
40
+ # also returns the global tags from serializer
41
+ if tags = tag_serializer.format(options[:tags])
42
+ service_check << '|#'
43
+ service_check << tags
44
+ end
45
+ end
46
+ end
47
+
48
+ protected
49
+ attr_reader :tag_serializer
50
+
51
+ def escape_message(message)
52
+ message.delete('|').tap do |m|
53
+ m.gsub!("\n", '\n')
54
+ m.gsub!('m:', 'm\:')
55
+ end
56
+ end
57
+ end
58
+ end
59
+ end
60
+ end
@@ -0,0 +1,55 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Datadog
4
+ class Statsd
5
+ module Serialization
6
+ class StatSerializer
7
+ def initialize(prefix, global_tags: [])
8
+ @prefix = prefix
9
+ @prefix_str = prefix.to_s
10
+ @tag_serializer = TagSerializer.new(global_tags)
11
+ end
12
+
13
+ def format(name, delta, type, tags: [], sample_rate: 1)
14
+ name = formated_name(name)
15
+
16
+ if sample_rate != 1
17
+ if tags_list = tag_serializer.format(tags)
18
+ "#{@prefix_str}#{name}:#{delta}|#{type}|@#{sample_rate}|##{tags_list}"
19
+ else
20
+ "#{@prefix_str}#{name}:#{delta}|#{type}|@#{sample_rate}"
21
+ end
22
+ else
23
+ if tags_list = tag_serializer.format(tags)
24
+ "#{@prefix_str}#{name}:#{delta}|#{type}|##{tags_list}"
25
+ else
26
+ "#{@prefix_str}#{name}:#{delta}|#{type}"
27
+ end
28
+ end
29
+ end
30
+
31
+ def global_tags
32
+ tag_serializer.global_tags
33
+ end
34
+
35
+ private
36
+
37
+ attr_reader :prefix
38
+ attr_reader :tag_serializer
39
+
40
+ def formated_name(name)
41
+ if name.is_a?(String)
42
+ # DEV: gsub is faster than dup.gsub!
43
+ formated = name.gsub('::', '.')
44
+ else
45
+ formated = name.to_s
46
+ formated.gsub!('::', '.')
47
+ end
48
+
49
+ formated.tr!(':|@', '_')
50
+ formated
51
+ end
52
+ end
53
+ end
54
+ end
55
+ end
@@ -0,0 +1,96 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Datadog
4
+ class Statsd
5
+ module Serialization
6
+ class TagSerializer
7
+ def initialize(global_tags = [], env = ENV)
8
+ # Convert to hash
9
+ global_tags = to_tags_hash(global_tags)
10
+
11
+ # Merge with default tags
12
+ global_tags = default_tags(env).merge(global_tags)
13
+
14
+ # Convert to tag list and set
15
+ @global_tags = to_tags_list(global_tags)
16
+ if @global_tags.any?
17
+ @global_tags_formatted = @global_tags.join(',')
18
+ else
19
+ @global_tags_formatted = nil
20
+ end
21
+ end
22
+
23
+ def format(message_tags)
24
+ if !message_tags || message_tags.empty?
25
+ return @global_tags_formatted
26
+ end
27
+
28
+ tags = if @global_tags_formatted
29
+ [@global_tags_formatted, to_tags_list(message_tags)]
30
+ else
31
+ to_tags_list(message_tags)
32
+ end
33
+
34
+ tags.join(',')
35
+ end
36
+
37
+ attr_reader :global_tags
38
+
39
+ private
40
+
41
+ def to_tags_hash(tags)
42
+ case tags
43
+ when Hash
44
+ tags.dup
45
+ when Array
46
+ Hash[
47
+ tags.map do |string|
48
+ tokens = string.split(':')
49
+ tokens << nil if tokens.length == 1
50
+ tokens.length == 2 ? tokens : nil
51
+ end.compact
52
+ ]
53
+ else
54
+ {}
55
+ end
56
+ end
57
+
58
+ def to_tags_list(tags)
59
+ case tags
60
+ when Hash
61
+ tags.map do |name, value|
62
+ if value
63
+ escape_tag_content("#{name}:#{value}")
64
+ else
65
+ escape_tag_content(name)
66
+ end
67
+ end
68
+ when Array
69
+ tags.map { |tag| escape_tag_content(tag) }
70
+ else
71
+ []
72
+ end
73
+ end
74
+
75
+ def escape_tag_content(tag)
76
+ tag.to_s.delete('|,')
77
+ end
78
+
79
+ def dd_tags(env = ENV)
80
+ return {} unless dd_tags = env['DD_TAGS']
81
+
82
+ to_tags_hash(dd_tags.split(','))
83
+ end
84
+
85
+ def default_tags(env = ENV)
86
+ dd_tags(env).tap do |tags|
87
+ tags['dd.internal.entity_id'] = env['DD_ENTITY_ID'] if env.key?('DD_ENTITY_ID')
88
+ tags['env'] = env['DD_ENV'] if env.key?('DD_ENV')
89
+ tags['service'] = env['DD_SERVICE'] if env.key?('DD_SERVICE')
90
+ tags['version'] = env['DD_VERSION'] if env.key?('DD_VERSION')
91
+ end
92
+ end
93
+ end
94
+ end
95
+ end
96
+ end
@@ -0,0 +1,15 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Datadog
4
+ class Statsd
5
+ module Serialization
6
+ end
7
+ end
8
+ end
9
+
10
+ require_relative 'serialization/tag_serializer'
11
+ require_relative 'serialization/service_check_serializer'
12
+ require_relative 'serialization/event_serializer'
13
+ require_relative 'serialization/stat_serializer'
14
+
15
+ require_relative 'serialization/serializer'
@@ -0,0 +1,62 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Datadog
4
+ class Statsd
5
+ # The SingleThreadSender is a sender synchronously buffering messages
6
+ # in a `MessageBuffer`.
7
+ # It is using current Process.PID to check it is the result of a recent fork
8
+ # and it is reseting the MessageBuffer if that's the case.
9
+ class SingleThreadSender
10
+ def initialize(message_buffer, logger: nil)
11
+ @message_buffer = message_buffer
12
+ @logger = logger
13
+ @mx = Mutex.new
14
+ # store the pid for which this sender has been created
15
+ update_fork_pid
16
+ end
17
+
18
+ def add(message)
19
+ @mx.synchronize {
20
+ # we have just forked, meaning we have messages in the buffer that we should
21
+ # not send, they belong to the parent process, let's clear the buffer.
22
+ if forked?
23
+ @message_buffer.reset
24
+ update_fork_pid
25
+ end
26
+ @message_buffer.add(message)
27
+ }
28
+ end
29
+
30
+ def flush(*)
31
+ @mx.synchronize {
32
+ @message_buffer.flush()
33
+ }
34
+ end
35
+
36
+ # Compatibility with `Sender`
37
+ def start()
38
+ end
39
+
40
+ # Compatibility with `Sender`
41
+ def stop()
42
+ end
43
+
44
+ # Compatibility with `Sender`
45
+ def rendez_vous()
46
+ end
47
+
48
+ private
49
+
50
+ # below are "fork management" methods to be able to clean the MessageBuffer
51
+ # if it detects that it is running in a unknown PID.
52
+
53
+ def forked?
54
+ Process.pid != @fork_pid
55
+ end
56
+
57
+ def update_fork_pid
58
+ @fork_pid = Process.pid
59
+ end
60
+ end
61
+ end
62
+ end
@@ -0,0 +1,96 @@
1
+ # frozen_string_literal: true
2
+ require 'time'
3
+
4
+ module Datadog
5
+ class Statsd
6
+ class Telemetry
7
+ attr_reader :metrics
8
+ attr_reader :events
9
+ attr_reader :service_checks
10
+ attr_reader :bytes_sent
11
+ attr_reader :bytes_dropped
12
+ attr_reader :packets_sent
13
+ attr_reader :packets_dropped
14
+
15
+ # Rough estimation of maximum telemetry message size without tags
16
+ MAX_TELEMETRY_MESSAGE_SIZE_WT_TAGS = 50 # bytes
17
+
18
+ def initialize(flush_interval, global_tags: [], transport_type: :udp)
19
+ @flush_interval = flush_interval
20
+ @global_tags = global_tags
21
+ @transport_type = transport_type
22
+ reset
23
+
24
+ # TODO: Karim: I don't know why but telemetry tags are serialized
25
+ # before global tags so by refactoring this, I am keeping the same behavior
26
+ @serialized_tags = Serialization::TagSerializer.new(
27
+ client: 'ruby',
28
+ client_version: VERSION,
29
+ client_transport: transport_type,
30
+ ).format(global_tags)
31
+ end
32
+
33
+ def would_fit_in?(max_buffer_payload_size)
34
+ MAX_TELEMETRY_MESSAGE_SIZE_WT_TAGS + serialized_tags.size < max_buffer_payload_size
35
+ end
36
+
37
+ def reset
38
+ @metrics = 0
39
+ @events = 0
40
+ @service_checks = 0
41
+ @bytes_sent = 0
42
+ @bytes_dropped = 0
43
+ @packets_sent = 0
44
+ @packets_dropped = 0
45
+ @next_flush_time = now_in_s + @flush_interval
46
+ end
47
+
48
+ def sent(metrics: 0, events: 0, service_checks: 0, bytes: 0, packets: 0)
49
+ @metrics += metrics
50
+ @events += events
51
+ @service_checks += service_checks
52
+
53
+ @bytes_sent += bytes
54
+ @packets_sent += packets
55
+ end
56
+
57
+ def dropped(bytes: 0, packets: 0)
58
+ @bytes_dropped += bytes
59
+ @packets_dropped += packets
60
+ end
61
+
62
+ def should_flush?
63
+ @next_flush_time < now_in_s
64
+ end
65
+
66
+ def flush
67
+ [
68
+ sprintf(pattern, 'metrics', @metrics),
69
+ sprintf(pattern, 'events', @events),
70
+ sprintf(pattern, 'service_checks', @service_checks),
71
+ sprintf(pattern, 'bytes_sent', @bytes_sent),
72
+ sprintf(pattern, 'bytes_dropped', @bytes_dropped),
73
+ sprintf(pattern, 'packets_sent', @packets_sent),
74
+ sprintf(pattern, 'packets_dropped', @packets_dropped),
75
+ ]
76
+ end
77
+
78
+ private
79
+ attr_reader :serialized_tags
80
+
81
+ def pattern
82
+ @pattern ||= "datadog.dogstatsd.client.%s:%d|#{COUNTER_TYPE}|##{serialized_tags}"
83
+ end
84
+
85
+ if Kernel.const_defined?('Process') && Process.respond_to?(:clock_gettime)
86
+ def now_in_s
87
+ Process.clock_gettime(Process::CLOCK_MONOTONIC, :second)
88
+ end
89
+ else
90
+ def now_in_s
91
+ Time.now.to_i
92
+ end
93
+ end
94
+ end
95
+ end
96
+ end