line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
package Etsy::StatsD; |
2
|
2
|
|
|
2
|
|
35073
|
use strict; |
|
2
|
|
|
|
|
3
|
|
|
2
|
|
|
|
|
45
|
|
3
|
2
|
|
|
2
|
|
6
|
use warnings; |
|
2
|
|
|
|
|
3
|
|
|
2
|
|
|
|
|
42
|
|
4
|
2
|
|
|
2
|
|
850
|
use IO::Socket; |
|
2
|
|
|
|
|
32548
|
|
|
2
|
|
|
|
|
6
|
|
5
|
2
|
|
|
2
|
|
689
|
use Carp; |
|
2
|
|
|
|
|
2
|
|
|
2
|
|
|
|
|
1207
|
|
6
|
|
|
|
|
|
|
|
7
|
|
|
|
|
|
|
our $VERSION = 1.002000; |
8
|
|
|
|
|
|
|
|
9
|
|
|
|
|
|
|
# The CPAN verion at https://github.com/sanbeg/Etsy-Statsd should be kept in |
10
|
|
|
|
|
|
|
# sync with the version distributed with StatsD, at |
11
|
|
|
|
|
|
|
# https://github.com/etsy/statsd (in the exmaples directory), so you can get |
12
|
|
|
|
|
|
|
# it from either location. |
13
|
|
|
|
|
|
|
|
14
|
|
|
|
|
|
|
=head1 NAME |
15
|
|
|
|
|
|
|
|
16
|
|
|
|
|
|
|
Etsy::StatsD - Object-Oriented Client for Etsy's StatsD Server |
17
|
|
|
|
|
|
|
|
18
|
|
|
|
|
|
|
=head1 SYNOPSIS |
19
|
|
|
|
|
|
|
|
20
|
|
|
|
|
|
|
use Etsy::StatsD; |
21
|
|
|
|
|
|
|
|
22
|
|
|
|
|
|
|
# Increment a counter |
23
|
|
|
|
|
|
|
my $statsd = Etsy::StatsD->new(); |
24
|
|
|
|
|
|
|
$statsd->increment( 'app.method.success' ); |
25
|
|
|
|
|
|
|
|
26
|
|
|
|
|
|
|
|
27
|
|
|
|
|
|
|
# Time something |
28
|
|
|
|
|
|
|
use Time::HiRes; |
29
|
|
|
|
|
|
|
|
30
|
|
|
|
|
|
|
my $start_time = time; |
31
|
|
|
|
|
|
|
$app->do_stuff; |
32
|
|
|
|
|
|
|
my $done_time = time; |
33
|
|
|
|
|
|
|
|
34
|
|
|
|
|
|
|
# Timers are expected in milliseconds |
35
|
|
|
|
|
|
|
$statsd->timing( 'app.method', ($done_time - $start_time) * 1000 ); |
36
|
|
|
|
|
|
|
|
37
|
|
|
|
|
|
|
# Send to two StatsD Endpoints simultaneously |
38
|
|
|
|
|
|
|
my $repl_statsd = Etsy::StatsD->new(["statsd1","statsd2"]); |
39
|
|
|
|
|
|
|
|
40
|
|
|
|
|
|
|
# On two different ports: |
41
|
|
|
|
|
|
|
my $repl_statsd = Etsy::StatsD->new(["statsd1","statsd1:8126"]); |
42
|
|
|
|
|
|
|
|
43
|
|
|
|
|
|
|
# Use TCP to a collector (you must specify a port) |
44
|
|
|
|
|
|
|
my $important_stats = Etsy::StatsD->new(["bizstats1:8125:tcp"]); |
45
|
|
|
|
|
|
|
|
46
|
|
|
|
|
|
|
|
47
|
|
|
|
|
|
|
=head1 DESCRIPTION |
48
|
|
|
|
|
|
|
|
49
|
|
|
|
|
|
|
=cut |
50
|
|
|
|
|
|
|
|
51
|
|
|
|
|
|
|
=over |
52
|
|
|
|
|
|
|
|
53
|
|
|
|
|
|
|
=item new (HOST, PORT, SAMPLE_RATE) |
54
|
|
|
|
|
|
|
|
55
|
|
|
|
|
|
|
Create a new instance. |
56
|
|
|
|
|
|
|
|
57
|
|
|
|
|
|
|
=over |
58
|
|
|
|
|
|
|
|
59
|
|
|
|
|
|
|
=item HOST |
60
|
|
|
|
|
|
|
|
61
|
|
|
|
|
|
|
|
62
|
|
|
|
|
|
|
If the argument is a string, it must be a hostname or IP only. The default is |
63
|
|
|
|
|
|
|
'localhost'. The argument may also be an array reference of strings in the |
64
|
|
|
|
|
|
|
form of "", ":", or "::". If the port is |
65
|
|
|
|
|
|
|
not specified, the default port specified by the PORT argument will be used. |
66
|
|
|
|
|
|
|
If the protocol is not specified, or is not "tcp" or "udp", "udp" will be set. |
67
|
|
|
|
|
|
|
The only way to change the protocol, is to specify the host, port and protocol. |
68
|
|
|
|
|
|
|
|
69
|
|
|
|
|
|
|
=item PORT |
70
|
|
|
|
|
|
|
|
71
|
|
|
|
|
|
|
Default is 8125. Will be used as the default port for any HOST argument not explicitly defining port. |
72
|
|
|
|
|
|
|
|
73
|
|
|
|
|
|
|
=item SAMPLE_RATE |
74
|
|
|
|
|
|
|
|
75
|
|
|
|
|
|
|
Default is undefined, or no sampling performed. Specify a rate as a decimal between 0 and 1 to enable |
76
|
|
|
|
|
|
|
sampling. e.g. 0.5 for 50%. |
77
|
|
|
|
|
|
|
|
78
|
|
|
|
|
|
|
=back |
79
|
|
|
|
|
|
|
|
80
|
|
|
|
|
|
|
=cut |
81
|
|
|
|
|
|
|
|
82
|
|
|
|
|
|
|
sub new { |
83
|
5
|
|
|
5
|
1
|
2611
|
my ( $class, $host, $port, $sample_rate ) = @_; |
84
|
5
|
100
|
|
|
|
16
|
$host = 'localhost' unless defined $host; |
85
|
5
|
100
|
|
|
|
15
|
$port = 8125 unless defined $port; |
86
|
|
|
|
|
|
|
|
87
|
|
|
|
|
|
|
# Handle multiple connections and |
88
|
|
|
|
|
|
|
# allow different ports to be specified |
89
|
|
|
|
|
|
|
# in the form of "::" |
90
|
5
|
|
|
|
|
11
|
my %protos = map { $_ => 1 } qw(tcp udp); |
|
10
|
|
|
|
|
31
|
|
91
|
5
|
|
|
|
|
13
|
my @connections = (); |
92
|
|
|
|
|
|
|
|
93
|
5
|
100
|
|
|
|
16
|
if( ref $host eq 'ARRAY' ) { |
94
|
2
|
|
|
|
|
3
|
foreach my $addr ( @{ $host } ) { |
|
2
|
|
|
|
|
7
|
|
95
|
3
|
|
|
|
|
11
|
my ($addr_host,$addr_port,$addr_proto) = split /:/, $addr; |
96
|
3
|
|
66
|
|
|
14
|
$addr_port ||= $port; |
97
|
|
|
|
|
|
|
# Validate the protocol |
98
|
3
|
100
|
|
|
|
7
|
if( defined $addr_proto ) { |
99
|
2
|
|
|
|
|
4
|
$addr_proto = lc $addr_proto; # Normalize to lowercase |
100
|
|
|
|
|
|
|
# Check validity |
101
|
2
|
50
|
|
|
|
9
|
if( !exists $protos{$addr_proto} ) { |
102
|
2
|
|
|
|
|
353
|
croak sprintf("Invalid protocol '%s', valid: %s", $addr_proto, join(', ', sort keys %protos)); |
103
|
|
|
|
|
|
|
} |
104
|
|
|
|
|
|
|
} |
105
|
|
|
|
|
|
|
else { |
106
|
1
|
|
|
|
|
3
|
$addr_proto = 'udp'; |
107
|
|
|
|
|
|
|
} |
108
|
1
|
|
|
|
|
4
|
push @connections, [ $addr_host, $addr_port, $addr_proto ]; |
109
|
|
|
|
|
|
|
} |
110
|
|
|
|
|
|
|
} |
111
|
|
|
|
|
|
|
else { |
112
|
3
|
|
|
|
|
10
|
push @connections, [ $host, $port, 'udp' ]; |
113
|
|
|
|
|
|
|
} |
114
|
|
|
|
|
|
|
|
115
|
3
|
|
|
|
|
7
|
my @sockets = (); |
116
|
3
|
|
|
|
|
8
|
foreach my $conn ( @connections ) { |
117
|
3
|
50
|
|
|
|
47
|
my $sock = new IO::Socket::INET( |
118
|
|
|
|
|
|
|
PeerAddr => $conn->[0], |
119
|
|
|
|
|
|
|
PeerPort => $conn->[1], |
120
|
|
|
|
|
|
|
Proto => $conn->[2], |
121
|
|
|
|
|
|
|
) or carp "Failed to initialize socket: $!"; |
122
|
|
|
|
|
|
|
|
123
|
3
|
50
|
|
|
|
2136
|
push @sockets, $sock if defined $sock; |
124
|
|
|
|
|
|
|
} |
125
|
|
|
|
|
|
|
# Check that we have at least 1 socket to send to |
126
|
3
|
50
|
|
|
|
12
|
croak "Failed to initialize any sockets." unless @sockets; |
127
|
|
|
|
|
|
|
|
128
|
3
|
|
|
|
|
40
|
bless { sockets => \@sockets, sample_rate => $sample_rate }, $class; |
129
|
|
|
|
|
|
|
} |
130
|
|
|
|
|
|
|
|
131
|
|
|
|
|
|
|
=item timing(STAT, TIME, SAMPLE_RATE) |
132
|
|
|
|
|
|
|
|
133
|
|
|
|
|
|
|
Log timing information |
134
|
|
|
|
|
|
|
|
135
|
|
|
|
|
|
|
=cut |
136
|
|
|
|
|
|
|
|
137
|
|
|
|
|
|
|
sub timing { |
138
|
4
|
|
|
4
|
1
|
661
|
my ( $self, $stat, $time, $sample_rate ) = @_; |
139
|
4
|
|
|
|
|
18
|
$self->send( { $stat => "$time|ms" }, $sample_rate ); |
140
|
|
|
|
|
|
|
} |
141
|
|
|
|
|
|
|
|
142
|
|
|
|
|
|
|
=item increment(STATS, SAMPLE_RATE) |
143
|
|
|
|
|
|
|
|
144
|
|
|
|
|
|
|
Increment one of more stats counters. |
145
|
|
|
|
|
|
|
|
146
|
|
|
|
|
|
|
=cut |
147
|
|
|
|
|
|
|
|
148
|
|
|
|
|
|
|
sub increment { |
149
|
1
|
|
|
1
|
1
|
871
|
my ( $self, $stats, $sample_rate ) = @_; |
150
|
1
|
|
|
|
|
5
|
$self->update( $stats, 1, $sample_rate ); |
151
|
|
|
|
|
|
|
} |
152
|
|
|
|
|
|
|
|
153
|
|
|
|
|
|
|
=item decrement(STATS, SAMPLE_RATE) |
154
|
|
|
|
|
|
|
|
155
|
|
|
|
|
|
|
Decrement one of more stats counters. |
156
|
|
|
|
|
|
|
|
157
|
|
|
|
|
|
|
=cut |
158
|
|
|
|
|
|
|
|
159
|
|
|
|
|
|
|
sub decrement { |
160
|
1
|
|
|
1
|
1
|
856
|
my ( $self, $stats, $sample_rate ) = @_; |
161
|
1
|
|
|
|
|
7
|
$self->update( $stats, -1, $sample_rate ); |
162
|
|
|
|
|
|
|
} |
163
|
|
|
|
|
|
|
|
164
|
|
|
|
|
|
|
=item update(STATS, DELTA, SAMPLE_RATE) |
165
|
|
|
|
|
|
|
|
166
|
|
|
|
|
|
|
Update one of more stats counters by arbitrary amounts. |
167
|
|
|
|
|
|
|
|
168
|
|
|
|
|
|
|
=cut |
169
|
|
|
|
|
|
|
|
170
|
|
|
|
|
|
|
sub update { |
171
|
5
|
|
|
5
|
1
|
2541
|
my ( $self, $stats, $delta, $sample_rate ) = @_; |
172
|
5
|
100
|
|
|
|
18
|
$delta = 1 unless defined $delta; |
173
|
5
|
|
|
|
|
6
|
my %data; |
174
|
5
|
100
|
|
|
|
13
|
if ( ref($stats) eq 'ARRAY' ) { |
175
|
1
|
|
|
|
|
3
|
%data = map { $_ => "$delta|c" } @$stats; |
|
2
|
|
|
|
|
8
|
|
176
|
|
|
|
|
|
|
} |
177
|
|
|
|
|
|
|
else { |
178
|
4
|
|
|
|
|
14
|
%data = ( $stats => "$delta|c" ); |
179
|
|
|
|
|
|
|
} |
180
|
5
|
|
|
|
|
15
|
$self->send( \%data, $sample_rate ); |
181
|
|
|
|
|
|
|
} |
182
|
|
|
|
|
|
|
|
183
|
|
|
|
|
|
|
=item send(DATA, SAMPLE_RATE) |
184
|
|
|
|
|
|
|
|
185
|
|
|
|
|
|
|
Sending logging data; implicitly called by most of the other methods. |
186
|
|
|
|
|
|
|
|
187
|
|
|
|
|
|
|
=back |
188
|
|
|
|
|
|
|
|
189
|
|
|
|
|
|
|
=cut |
190
|
|
|
|
|
|
|
|
191
|
|
|
|
|
|
|
sub send { |
192
|
3
|
|
|
3
|
|
5
|
my ( $self, $data, $sample_rate ) = @_; |
193
|
3
|
100
|
|
|
|
6
|
$sample_rate = $self->{sample_rate} unless defined $sample_rate; |
194
|
|
|
|
|
|
|
|
195
|
3
|
|
|
|
|
4
|
my $sampled_data; |
196
|
3
|
100
|
100
|
|
|
11
|
if ( defined($sample_rate) and $sample_rate < 1 ) { |
197
|
1
|
|
|
|
|
6
|
while ( my ( $stat, $value ) = each %$data ) { |
198
|
1
|
50
|
|
|
|
27
|
$sampled_data->{$stat} = "$value|\@$sample_rate" if rand() <= $sample_rate; |
199
|
|
|
|
|
|
|
} |
200
|
|
|
|
|
|
|
} |
201
|
|
|
|
|
|
|
else { |
202
|
2
|
|
|
|
|
6
|
$sampled_data = $data; |
203
|
|
|
|
|
|
|
} |
204
|
|
|
|
|
|
|
|
205
|
3
|
100
|
|
|
|
11
|
return '0 but true' unless keys %$sampled_data; |
206
|
|
|
|
|
|
|
|
207
|
|
|
|
|
|
|
#failures in any of this can be silently ignored |
208
|
2
|
|
|
|
|
2
|
my $count = 0; |
209
|
2
|
|
|
|
|
1
|
foreach my $socket ( @{ $self->{sockets} } ) { |
|
2
|
|
|
|
|
5
|
|
210
|
|
|
|
|
|
|
# calling keys() resets the each() iterator |
211
|
2
|
|
|
|
|
3
|
keys %$sampled_data; |
212
|
2
|
|
|
|
|
6
|
while ( my ( $stat,$value ) = each %$sampled_data ) { |
213
|
2
|
|
|
|
|
5
|
_send_to_sock($socket, "$stat:$value\n", 0); |
214
|
2
|
|
|
|
|
9
|
++$count; |
215
|
|
|
|
|
|
|
} |
216
|
|
|
|
|
|
|
} |
217
|
2
|
|
|
|
|
5
|
return $count; |
218
|
|
|
|
|
|
|
} |
219
|
|
|
|
|
|
|
|
220
|
|
|
|
|
|
|
sub _send_to_sock( $$ ) { |
221
|
0
|
|
|
0
|
|
|
my ($sock,$msg) = @_; |
222
|
0
|
|
|
|
|
|
CORE::send( $sock, $msg, 0 ); |
223
|
|
|
|
|
|
|
} |
224
|
|
|
|
|
|
|
|
225
|
|
|
|
|
|
|
=head1 SEE ALSO |
226
|
|
|
|
|
|
|
|
227
|
|
|
|
|
|
|
L |
228
|
|
|
|
|
|
|
|
229
|
|
|
|
|
|
|
=head1 AUTHOR |
230
|
|
|
|
|
|
|
|
231
|
|
|
|
|
|
|
Steve Sanbeg L |
232
|
|
|
|
|
|
|
|
233
|
|
|
|
|
|
|
=cut |
234
|
|
|
|
|
|
|
|
235
|
|
|
|
|
|
|
1; |