statsd-instrument 3.6.1 → 3.8.0

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.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: a5e3cda783c5f0c7de2c30175ab001cef492c3ad0e0923af3d3a4840df7d6005
4
- data.tar.gz: b31685696c6e1ad8e5cd36a8115eca2363ed9a732a483534004b2163453b323a
3
+ metadata.gz: 7ce05bd8d34026227e2960ccabca96119580dc2d5737fff81a2bdfd8ea18f826
4
+ data.tar.gz: d619b08700bb735922673013d7e6314f32d76e8744dcfa07966a35595aa27cf4
5
5
  SHA512:
6
- metadata.gz: 658663a99a273170ad72436eae0142d707e35a8cc2bbf5befd89a7cd225db5673c5ec676e8f96a168411f443fa7d3e6b73f60c6f081067082b2003920cef264b
7
- data.tar.gz: bc8930e083ec158ca6a945186c2156a0890ebe74edb63080dc6ced9ca691498edc9d81ae746ec80477471c3b2f09d3abb95042002c0c8e7d72afeb9ab360526c
6
+ metadata.gz: 42317b00c680ffc079e89bad712225b8a9656faedf1f8743d9031c26070ae05af0c2ef8a10f1200d94a708023e6cd667d91fbf64a972fecb7d97d327edc31f22
7
+ data.tar.gz: 540c9a8bccc54633f40e9b2830748e764d6bf5034791bcdacb611be6a7fc615e66a9fa138e2ce58f4a6461f46bb40114fb7507f3f4357d86fccc5d9d6614d244
@@ -13,7 +13,6 @@ jobs:
13
13
  - name: Set up Ruby
14
14
  uses: ruby/setup-ruby@v1
15
15
  with:
16
- ruby-version: 3.1
17
16
  bundler-cache: true
18
17
 
19
18
  - name: Run benchmark on branch
@@ -13,7 +13,6 @@ jobs:
13
13
  - name: Set up Ruby
14
14
  uses: ruby/setup-ruby@v1
15
15
  with:
16
- ruby-version: 2.7
17
16
  bundler-cache: true
18
17
 
19
18
  - name: Run Rubocop
@@ -9,7 +9,7 @@ jobs:
9
9
  strategy:
10
10
  fail-fast: false
11
11
  matrix:
12
- ruby: ['2.6', '2.7', '3.0', '3.1', '3.2', 'ruby-head', 'jruby-9.3.7.0', 'truffleruby-22.2.0']
12
+ ruby: ['2.6', '2.7', '3.0', '3.1', '3.2', '3.3', 'ruby-head', 'jruby-9.3.7.0', 'truffleruby-22.2.0']
13
13
  # Windows on macOS builds started failing, so they are disabled for now
14
14
  # platform: [windows-2019, macOS-10.14, ubuntu-18.04]
15
15
  # exclude:
data/.rubocop.yml CHANGED
@@ -5,7 +5,6 @@ require:
5
5
  - ./lib/statsd/instrument/rubocop.rb
6
6
 
7
7
  AllCops:
8
- TargetRubyVersion: 2.7
9
8
  UseCache: true
10
9
  SuggestExtensions: false
11
10
  CacheRootDirectory: tmp/rubocop
data/.ruby-version ADDED
@@ -0,0 +1 @@
1
+ 3.3.0
data/CHANGELOG.md CHANGED
@@ -6,6 +6,17 @@ section below.
6
6
 
7
7
  ## Unreleased changes
8
8
 
9
+ ## Version 3.8.0
10
+
11
+ - UDP batching will now track statistics about its own batching performance, and
12
+ emit those statistics to the default sink when `STATSD_BATCH_STATISTICS_INTERVAL`
13
+ is set to any non-zero value. The default value is zero; additional information
14
+ on statistics tracked is available in the README.
15
+
16
+ ## Version 3.7.0
17
+
18
+ - Add public `.flush` method to sink classes.
19
+
9
20
  ## Version 3.6.1
10
21
 
11
22
  - Fix `ArgumentError` when passing an empty Hash as tags.
data/README.md CHANGED
@@ -54,6 +54,16 @@ The following environment variables are supported:
54
54
  If your network is properly configured to handle larger packets you may try
55
55
  to increase this value for better performance, but most network can't handle
56
56
  larger packets.
57
+ - `STATSD_BATCH_STATISTICS_INTERVAL`: (default: "0") If non-zero, the `BatchedUDPSink`
58
+ will track and emit statistics on this interval to the default sink for your environment.
59
+ The current tracked statistics are:
60
+
61
+ - `statsd_instrument.batched_udp_sink.batched_sends`: The number of batches sent, of any size.
62
+ - `statsd_instrument.batched_udp_sink.synchronous_sends`: The number of times the batched udp sender needed to send a statsd line synchronously, due to the buffer being full.
63
+ - `statsd_instrument.batched_udp_sink.avg_buffer_length`: The average buffer length, measured at the beginning of each batch.
64
+ - `statsd_instrument.batched_udp_sink.avg_batched_packet_size`: The average per-batch byte size of the packet sent to the underlying UDPSink.
65
+ - `statsd_instrument.batched_udp_sink.avg_batch_length`: The average number of statsd lines per batch.
66
+
57
67
 
58
68
  ## StatsD keys
59
69
 
@@ -9,6 +9,7 @@ module StatsD
9
9
  DEFAULT_BUFFER_CAPACITY = 5_000
10
10
  # https://docs.datadoghq.com/developers/dogstatsd/high_throughput/?code-lang=ruby#ensure-proper-packet-sizes
11
11
  DEFAULT_MAX_PACKET_SIZE = 1472
12
+ DEFAULT_STATISTICS_INTERVAL = 0 # in seconds, and 0 implies disabled-by-default.
12
13
 
13
14
  attr_reader :host, :port
14
15
 
@@ -28,7 +29,8 @@ module StatsD
28
29
  port,
29
30
  thread_priority: DEFAULT_THREAD_PRIORITY,
30
31
  buffer_capacity: DEFAULT_BUFFER_CAPACITY,
31
- max_packet_size: DEFAULT_MAX_PACKET_SIZE
32
+ max_packet_size: DEFAULT_MAX_PACKET_SIZE,
33
+ statistics_interval: DEFAULT_STATISTICS_INTERVAL
32
34
  )
33
35
  @host = host
34
36
  @port = port
@@ -38,6 +40,7 @@ module StatsD
38
40
  buffer_capacity,
39
41
  thread_priority,
40
42
  max_packet_size,
43
+ statistics_interval,
41
44
  )
42
45
  ObjectSpace.define_finalizer(self, self.class.finalize(@dispatcher))
43
46
  end
@@ -55,6 +58,10 @@ module StatsD
55
58
  @dispatcher.shutdown(*args)
56
59
  end
57
60
 
61
+ def flush(blocking:)
62
+ @dispatcher.flush(blocking: blocking)
63
+ end
64
+
58
65
  class Buffer < SizedQueue
59
66
  def push_nonblock(item)
60
67
  push(item, true)
@@ -73,8 +80,70 @@ module StatsD
73
80
  end
74
81
  end
75
82
 
83
+ class DispatcherStats
84
+ def initialize(interval)
85
+ # The number of times the batched udp sender needed to
86
+ # send a statsd line synchronously, due to the buffer
87
+ # being full.
88
+ @synchronous_sends = 0
89
+ # The number of times we send a batch of statsd lines,
90
+ # of any size.
91
+ @batched_sends = 0
92
+ # The average buffer length, measured at the beginning of
93
+ # each batch.
94
+ @avg_buffer_length = 0
95
+ # The average per-batch byte size of the packet sent to
96
+ # the underlying UDPSink.
97
+ @avg_batched_packet_size = 0
98
+ # The average number of statsd lines per batch.
99
+ @avg_batch_length = 0
100
+
101
+ @mutex = Mutex.new
102
+
103
+ @interval = interval
104
+ @since = Process.clock_gettime(Process::CLOCK_MONOTONIC)
105
+ end
106
+
107
+ def maybe_flush!(force: false)
108
+ return if !force && Process.clock_gettime(Process::CLOCK_MONOTONIC) - @since < @interval
109
+
110
+ synchronous_sends = 0
111
+ batched_sends = 0
112
+ avg_buffer_length = 0
113
+ avg_batched_packet_size = 0
114
+ avg_batch_length = 0
115
+ @mutex.synchronize do
116
+ synchronous_sends, @synchronous_sends = @synchronous_sends, synchronous_sends
117
+ batched_sends, @batched_sends = @batched_sends, batched_sends
118
+ avg_buffer_length, @avg_buffer_length = @avg_buffer_length, avg_buffer_length
119
+ avg_batched_packet_size, @avg_batched_packet_size = @avg_batched_packet_size, avg_batched_packet_size
120
+ avg_batch_length, @avg_batch_length = @avg_batch_length, avg_batch_length
121
+ @since = Process.clock_gettime(Process::CLOCK_MONOTONIC)
122
+ end
123
+
124
+ StatsD.increment("statsd_instrument.batched_udp_sink.synchronous_sends", synchronous_sends)
125
+ StatsD.increment("statsd_instrument.batched_udp_sink.batched_sends", batched_sends)
126
+ StatsD.gauge("statsd_instrument.batched_udp_sink.avg_buffer_length", avg_buffer_length)
127
+ StatsD.gauge("statsd_instrument.batched_udp_sink.avg_batched_packet_size", avg_batched_packet_size)
128
+ StatsD.gauge("statsd_instrument.batched_udp_sink.avg_batch_length", avg_batch_length)
129
+ end
130
+
131
+ def increment_synchronous_sends
132
+ @mutex.synchronize { @synchronous_sends += 1 }
133
+ end
134
+
135
+ def increment_batched_sends(buffer_len, packet_size, batch_len)
136
+ @mutex.synchronize do
137
+ @batched_sends += 1
138
+ @avg_buffer_length += (buffer_len - @avg_buffer_length) / @batched_sends
139
+ @avg_batched_packet_size += (packet_size - @avg_batched_packet_size) / @batched_sends
140
+ @avg_batch_length += (batch_len - @avg_batch_length) / @batched_sends
141
+ end
142
+ end
143
+ end
144
+
76
145
  class Dispatcher
77
- def initialize(host, port, buffer_capacity, thread_priority, max_packet_size)
146
+ def initialize(host, port, buffer_capacity, thread_priority, max_packet_size, statistics_interval)
78
147
  @udp_sink = UDPSink.new(host, port)
79
148
  @interrupted = false
80
149
  @thread_priority = thread_priority
@@ -83,13 +152,18 @@ module StatsD
83
152
  @buffer = Buffer.new(buffer_capacity)
84
153
  @dispatcher_thread = Thread.new { dispatch }
85
154
  @pid = Process.pid
155
+ if statistics_interval > 0
156
+ @statistics = DispatcherStats.new(statistics_interval)
157
+ end
86
158
  end
87
159
 
88
160
  def <<(datagram)
89
161
  if !thread_healthcheck || !@buffer.push_nonblock(datagram)
90
- # The buffer is full or the thread can't be respaned,
162
+ # The buffer is full or the thread can't be respawned,
91
163
  # we'll send the datagram synchronously
92
164
  @udp_sink << datagram
165
+
166
+ @statistics&.increment_synchronous_sends
93
167
  end
94
168
 
95
169
  self
@@ -104,10 +178,6 @@ module StatsD
104
178
  flush(blocking: false)
105
179
  end
106
180
 
107
- private
108
-
109
- NEWLINE = "\n".b.freeze
110
-
111
181
  def flush(blocking:)
112
182
  packet = "".b
113
183
  next_datagram = nil
@@ -119,6 +189,8 @@ module StatsD
119
189
  next_datagram ||= @buffer.pop_nonblock
120
190
  break if next_datagram.nil? # no datagram in buffer
121
191
  end
192
+ buffer_len = @buffer.length + 1
193
+ batch_len = 1
122
194
 
123
195
  packet << next_datagram
124
196
  next_datagram = nil
@@ -126,17 +198,26 @@ module StatsD
126
198
  while (next_datagram = @buffer.pop_nonblock)
127
199
  if @max_packet_size - packet.bytesize - 1 > next_datagram.bytesize
128
200
  packet << NEWLINE << next_datagram
201
+ batch_len += 1
129
202
  else
130
203
  break
131
204
  end
132
205
  end
133
206
  end
134
207
 
208
+ packet_size = packet.bytesize
135
209
  @udp_sink << packet
136
210
  packet.clear
211
+
212
+ @statistics&.increment_batched_sends(buffer_len, packet_size, batch_len)
213
+ @statistics&.maybe_flush!
137
214
  end
138
215
  end
139
216
 
217
+ private
218
+
219
+ NEWLINE = "\n".b.freeze
220
+
140
221
  def thread_healthcheck
141
222
  # TODO: We have a race condition on JRuby / Truffle here. It could cause multiple
142
223
  # dispatcher threads to be spawned, which would cause problems.
@@ -26,6 +26,10 @@ module StatsD
26
26
  def clear
27
27
  @datagrams.clear
28
28
  end
29
+
30
+ def flush(blocking:)
31
+ @parent.flush(blocking: blocking)
32
+ end
29
33
  end
30
34
  end
31
35
  end
@@ -16,7 +16,7 @@ module StatsD
16
16
  :d
17
17
  end
18
18
 
19
- # Constricts an event datagram.
19
+ # Constructs an event datagram.
20
20
  #
21
21
  # @param [String] title Event title.
22
22
  # @param [String] text Event description. Newlines are allowed.
@@ -57,7 +57,7 @@ module StatsD
57
57
  datagram
58
58
  end
59
59
 
60
- # Constricts a service check datagram.
60
+ # Constructs a service check datagram.
61
61
  #
62
62
  # @param [String] name Name of the service
63
63
  # @param [Symbol] status Either `:ok`, `:warning`, `:critical` or `:unknown`
@@ -98,6 +98,13 @@ module StatsD
98
98
  Float(env.fetch("STATSD_MAX_PACKET_SIZE", StatsD::Instrument::BatchedUDPSink::DEFAULT_MAX_PACKET_SIZE))
99
99
  end
100
100
 
101
+ def statsd_batch_statistics_interval
102
+ Integer(env.fetch(
103
+ "STATSD_BATCH_STATISTICS_INTERVAL",
104
+ StatsD::Instrument::BatchedUDPSink::DEFAULT_STATISTICS_INTERVAL,
105
+ ))
106
+ end
107
+
101
108
  def client
102
109
  StatsD::Instrument::Client.from_env(self)
103
110
  end
@@ -110,6 +117,7 @@ module StatsD
110
117
  statsd_addr,
111
118
  buffer_capacity: statsd_buffer_capacity,
112
119
  max_packet_size: statsd_max_packet_size,
120
+ statistics_interval: statsd_batch_statistics_interval,
113
121
  )
114
122
  else
115
123
  StatsD::Instrument::UDPSink.for_addr(statsd_addr)
@@ -23,6 +23,10 @@ module StatsD
23
23
  logger.add(severity, "[StatsD] #{datagram.chomp}")
24
24
  self
25
25
  end
26
+
27
+ def flush(blocking:)
28
+ # noop
29
+ end
26
30
  end
27
31
  end
28
32
  end
@@ -12,6 +12,10 @@ module StatsD
12
12
  def <<(_datagram)
13
13
  self # noop
14
14
  end
15
+
16
+ def flush(blocking:)
17
+ # noop
18
+ end
15
19
  end
16
20
  end
17
21
  end
@@ -53,6 +53,10 @@ module StatsD
53
53
  self
54
54
  end
55
55
 
56
+ def flush(blocking:)
57
+ # noop
58
+ end
59
+
56
60
  private
57
61
 
58
62
  def invalidate_socket
@@ -2,6 +2,6 @@
2
2
 
3
3
  module StatsD
4
4
  module Instrument
5
- VERSION = "3.6.1"
5
+ VERSION = "3.8.0"
6
6
  end
7
7
  end
@@ -20,5 +20,7 @@ Gem::Specification.new do |spec|
20
20
  spec.test_files = spec.files.grep(%r{^(test|spec|features)/})
21
21
  spec.require_paths = ["lib"]
22
22
 
23
+ spec.required_ruby_version = ">= 2.6.0"
24
+
23
25
  spec.metadata['allowed_push_host'] = "https://rubygems.org"
24
26
  end
@@ -0,0 +1,69 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "test_helper"
4
+
5
+ class DispatcherStatsTest < Minitest::Test
6
+ include StatsD::Instrument::Assertions
7
+
8
+ def test_maybe_flush
9
+ stats = StatsD::Instrument::BatchedUDPSink::DispatcherStats.new(0)
10
+
11
+ stats.increment_synchronous_sends
12
+ stats.increment_batched_sends(1, 1, 1)
13
+
14
+ expectations = [
15
+ StatsD::Instrument::Expectation.increment("statsd_instrument.batched_udp_sink.synchronous_sends", 1),
16
+ StatsD::Instrument::Expectation.increment("statsd_instrument.batched_udp_sink.batched_sends", 1),
17
+ StatsD::Instrument::Expectation.gauge("statsd_instrument.batched_udp_sink.avg_buffer_length", 1),
18
+ StatsD::Instrument::Expectation.gauge("statsd_instrument.batched_udp_sink.avg_batched_packet_size", 1),
19
+ StatsD::Instrument::Expectation.gauge("statsd_instrument.batched_udp_sink.avg_batch_length", 1),
20
+ ]
21
+ assert_statsd_expectations(expectations) { stats.maybe_flush! }
22
+ assert_equal(0, stats.instance_variable_get(:@synchronous_sends))
23
+ assert_equal(0, stats.instance_variable_get(:@batched_sends))
24
+ assert_equal(0, stats.instance_variable_get(:@avg_buffer_length))
25
+ assert_equal(0, stats.instance_variable_get(:@avg_batched_packet_size))
26
+ assert_equal(0, stats.instance_variable_get(:@avg_batch_length))
27
+
28
+ stats = StatsD::Instrument::BatchedUDPSink::DispatcherStats.new(1)
29
+ stats.increment_batched_sends(1, 1, 1)
30
+ assert_no_statsd_calls { stats.maybe_flush! }
31
+ end
32
+
33
+ def test_calculations_are_correct
34
+ stats = StatsD::Instrument::BatchedUDPSink::DispatcherStats.new(0)
35
+
36
+ 5.times { stats.increment_synchronous_sends }
37
+ assert_equal(5, stats.instance_variable_get(:@synchronous_sends))
38
+
39
+ batches = [
40
+ { buffer_len: 100, packet_size: 1472, batch_len: 10 },
41
+ { buffer_len: 90, packet_size: 1300, batch_len: 20 },
42
+ { buffer_len: 110, packet_size: 1470, batch_len: 8 },
43
+ { buffer_len: 500, packet_size: 1000, batch_len: 1 },
44
+ { buffer_len: 100, packet_size: 30, batch_len: 99 },
45
+ ]
46
+ batches.each do |batch|
47
+ stats.increment_batched_sends(batch[:buffer_len], batch[:packet_size], batch[:batch_len])
48
+ end
49
+ assert_equal(batches.length, stats.instance_variable_get(:@batched_sends))
50
+ assert_equal(
51
+ batches.map { |b|
52
+ b[:buffer_len]
53
+ }.sum / batches.length,
54
+ stats.instance_variable_get(:@avg_buffer_length),
55
+ )
56
+ assert_equal(
57
+ batches.map { |b|
58
+ b[:packet_size]
59
+ }.sum / batches.length,
60
+ stats.instance_variable_get(:@avg_batched_packet_size),
61
+ )
62
+ assert_equal(
63
+ batches.map { |b|
64
+ b[:batch_len]
65
+ }.sum / batches.length,
66
+ stats.instance_variable_get(:@avg_batch_length),
67
+ )
68
+ end
69
+ end
@@ -57,6 +57,9 @@ class EnvironmentTest < Minitest::Test
57
57
  end
58
58
 
59
59
  def test_client_from_env_uses_regular_udp_sink_when_flush_interval_is_0
60
+ StatsD::Instrument::Environment.any_instance.expects(:warn).with(
61
+ "STATSD_FLUSH_INTERVAL=0.0 is deprecated, please set STATSD_BUFFER_CAPACITY=0 instead.",
62
+ ).once
60
63
  env = StatsD::Instrument::Environment.new(
61
64
  "STATSD_USE_NEW_CLIENT" => "1",
62
65
  "STATSD_ENV" => "staging",
@@ -122,79 +122,112 @@ module UDPSinkTests
122
122
  end
123
123
  datagrams
124
124
  end
125
+ end
125
126
 
126
- class UDPSinkTest < Minitest::Test
127
- include UDPSinkTests
127
+ class UDPSinkTest < Minitest::Test
128
+ include UDPSinkTests
128
129
 
129
- def setup
130
- @receiver = UDPSocket.new
131
- @receiver.bind("localhost", 0)
132
- @host = @receiver.addr[2]
133
- @port = @receiver.addr[1]
134
- @sink_class = StatsD::Instrument::UDPSink
135
- end
130
+ def setup
131
+ @receiver = UDPSocket.new
132
+ @receiver.bind("localhost", 0)
133
+ @host = @receiver.addr[2]
134
+ @port = @receiver.addr[1]
135
+ @sink_class = StatsD::Instrument::UDPSink
136
+ end
136
137
 
137
- def teardown
138
- @receiver.close
139
- end
138
+ def teardown
139
+ @receiver.close
140
+ end
140
141
 
141
- def test_socket_error_should_invalidate_socket
142
- previous_logger = StatsD.logger
143
- begin
144
- logs = StringIO.new
145
- StatsD.logger = Logger.new(logs)
146
- StatsD.logger.formatter = SimpleFormatter.new
147
- UDPSocket.stubs(:new).returns(socket = mock("socket"))
148
-
149
- seq = sequence("connect_fail_connect_succeed")
150
- socket.expects(:connect).with("localhost", 8125).in_sequence(seq)
151
- socket.expects(:send).raises(Errno::EDESTADDRREQ).in_sequence(seq)
152
- socket.expects(:close).in_sequence(seq)
153
- socket.expects(:connect).with("localhost", 8125).in_sequence(seq)
154
- socket.expects(:send).twice.returns(1).in_sequence(seq)
155
-
156
- udp_sink = build_sink("localhost", 8125)
157
- udp_sink << "foo:1|c"
158
- udp_sink << "bar:1|c"
159
-
160
- assert_equal(
161
- "[#{@sink_class}] Resetting connection because of " \
162
- "Errno::EDESTADDRREQ: Destination address required\n",
163
- logs.string,
164
- )
165
- ensure
166
- StatsD.logger = previous_logger
167
- end
142
+ def test_socket_error_should_invalidate_socket
143
+ previous_logger = StatsD.logger
144
+ begin
145
+ logs = StringIO.new
146
+ StatsD.logger = Logger.new(logs)
147
+ StatsD.logger.formatter = SimpleFormatter.new
148
+ UDPSocket.stubs(:new).returns(socket = mock("socket"))
149
+
150
+ seq = sequence("connect_fail_connect_succeed")
151
+ socket.expects(:connect).with("localhost", 8125).in_sequence(seq)
152
+ socket.expects(:send).raises(Errno::EDESTADDRREQ).in_sequence(seq)
153
+ socket.expects(:close).in_sequence(seq)
154
+ socket.expects(:connect).with("localhost", 8125).in_sequence(seq)
155
+ socket.expects(:send).twice.returns(1).in_sequence(seq)
156
+ socket.expects(:close).in_sequence(seq)
157
+
158
+ udp_sink = build_sink("localhost", 8125)
159
+ udp_sink << "foo:1|c"
160
+ udp_sink << "bar:1|c"
161
+
162
+ assert_equal(
163
+ "[#{@sink_class}] Resetting connection because of " \
164
+ "Errno::EDESTADDRREQ: Destination address required\n",
165
+ logs.string,
166
+ )
167
+ ensure
168
+ StatsD.logger = previous_logger
169
+ # Make sure our fake socket is closed so that it doesn't interfere with other tests
170
+ udp_sink&.send(:invalidate_socket)
168
171
  end
169
172
  end
173
+ end
170
174
 
171
- module BatchedUDPSinkTests
172
- include UDPSinkTests
175
+ class BatchedUDPSinkTest < Minitest::Test
176
+ include UDPSinkTests
173
177
 
174
- def setup
175
- @receiver = UDPSocket.new
176
- @receiver.bind("localhost", 0)
177
- @host = @receiver.addr[2]
178
- @port = @receiver.addr[1]
179
- @sink_class = StatsD::Instrument::BatchedUDPSink
180
- @sinks = []
181
- end
178
+ def setup
179
+ @receiver = UDPSocket.new
180
+ @receiver.bind("localhost", 0)
181
+ @host = @receiver.addr[2]
182
+ @port = @receiver.addr[1]
183
+ @sink_class = StatsD::Instrument::BatchedUDPSink
184
+ @sinks = []
185
+ end
182
186
 
183
- def teardown
184
- @receiver.close
185
- @sinks.each(&:shutdown)
186
- end
187
+ def teardown
188
+ @receiver.close
189
+ @sinks.each(&:shutdown)
190
+ end
187
191
 
188
- private
192
+ def test_flush
193
+ buffer_size = 50
194
+
195
+ sink = build_sink(@host, @port, buffer_capacity: buffer_size)
196
+ dispatcher = sink.instance_variable_get(:@dispatcher)
197
+ buffer = dispatcher.instance_variable_get(:@buffer)
198
+ # Send a few datagrams to fill the buffer
199
+ (buffer_size * 2).times { |i| sink << "foo:#{i}|c" }
200
+ assert(!buffer.empty?)
201
+ sink.flush(blocking: false)
202
+ assert(buffer.empty?)
203
+ end
189
204
 
190
- def build_sink(host = @host, port = @port)
191
- sink = @sink_class.new(host, port, buffer_capacity: 50)
192
- @sinks << sink
193
- sink
205
+ def test_statistics
206
+ datagrams = StatsD.singleton_client.capture do
207
+ buffer_size = 2
208
+ sink = build_sink(@host, @port, buffer_capacity: buffer_size, statistics_interval: 0.1)
209
+ 2.times { |i| sink << "foo:#{i}|c" }
210
+ sink.flush(blocking: false)
211
+ sink.instance_variable_get(:@dispatcher).instance_variable_get(:@statistics).maybe_flush!(force: true)
194
212
  end
213
+
214
+ assert(datagrams.any? { |d| d.name.start_with?("statsd_instrument.batched_udp_sink.avg_batch_length") })
215
+ assert(datagrams.any? { |d| d.name.start_with?("statsd_instrument.batched_udp_sink.avg_batched_packet_size") })
216
+ assert(datagrams.any? { |d| d.name.start_with?("statsd_instrument.batched_udp_sink.avg_buffer_length") })
217
+ assert(datagrams.any? { |d| d.name.start_with?("statsd_instrument.batched_udp_sink.batched_sends") })
218
+ assert(datagrams.any? { |d| d.name.start_with?("statsd_instrument.batched_udp_sink.synchronous_sends") })
195
219
  end
196
220
 
197
- class BatchedUDPSinkTest < Minitest::Test
198
- include BatchedUDPSinkTests
221
+ private
222
+
223
+ def build_sink(host = @host, port = @port, buffer_capacity: 50, statistics_interval: 0)
224
+ sink = @sink_class.new(
225
+ host,
226
+ port,
227
+ buffer_capacity: buffer_capacity,
228
+ statistics_interval: statistics_interval,
229
+ )
230
+ @sinks << sink
231
+ sink
199
232
  end
200
233
  end
metadata CHANGED
@@ -1,7 +1,7 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: statsd-instrument
3
3
  version: !ruby/object:Gem::Version
4
- version: 3.6.1
4
+ version: 3.8.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Jesse Storimer
@@ -10,7 +10,7 @@ authors:
10
10
  autorequire:
11
11
  bindir: bin
12
12
  cert_chain: []
13
- date: 2023-11-06 00:00:00.000000000 Z
13
+ date: 2024-06-19 00:00:00.000000000 Z
14
14
  dependencies: []
15
15
  description: A StatsD client for Ruby apps. Provides metaprogramming methods to inject
16
16
  StatsD instrumentation into your code.
@@ -27,6 +27,7 @@ files:
27
27
  - ".github/workflows/tests.yml"
28
28
  - ".gitignore"
29
29
  - ".rubocop.yml"
30
+ - ".ruby-version"
30
31
  - ".yardopts"
31
32
  - CHANGELOG.md
32
33
  - CONTRIBUTING.md
@@ -81,6 +82,7 @@ files:
81
82
  - test/client_test.rb
82
83
  - test/datagram_builder_test.rb
83
84
  - test/datagram_test.rb
85
+ - test/dispatcher_stats_test.rb
84
86
  - test/dogstatsd_datagram_builder_test.rb
85
87
  - test/environment_test.rb
86
88
  - test/helpers/rubocop_helper.rb
@@ -115,14 +117,14 @@ required_ruby_version: !ruby/object:Gem::Requirement
115
117
  requirements:
116
118
  - - ">="
117
119
  - !ruby/object:Gem::Version
118
- version: '0'
120
+ version: 2.6.0
119
121
  required_rubygems_version: !ruby/object:Gem::Requirement
120
122
  requirements:
121
123
  - - ">="
122
124
  - !ruby/object:Gem::Version
123
125
  version: '0'
124
126
  requirements: []
125
- rubygems_version: 3.4.21
127
+ rubygems_version: 3.5.11
126
128
  signing_key:
127
129
  specification_version: 4
128
130
  summary: A StatsD client for Ruby apps
@@ -136,6 +138,7 @@ test_files:
136
138
  - test/client_test.rb
137
139
  - test/datagram_builder_test.rb
138
140
  - test/datagram_test.rb
141
+ - test/dispatcher_stats_test.rb
139
142
  - test/dogstatsd_datagram_builder_test.rb
140
143
  - test/environment_test.rb
141
144
  - test/helpers/rubocop_helper.rb