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 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 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 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
|
# You may distribute under the terms of either the GNU General Public License
# or the Artistic License (the same terms as Perl itself)
#
# (C) Paul Evans, 2020 -- leonerd@leonerd.org.uk
package Metrics::Any::AdapterBase::Stored 0.10;
use v5.14;
use warnings;
use Carp;
=head1 NAME
C<Metrics::Any::AdapterBase::Stored> - a base class for metrics adapters which store values
=head1 DESCRIPTION
This base class assists in creating L<Metrics::Any::Adapter> classes which
store values of reported metrics directly. These can then be retrieved later
by the containing application, or the subclass code, by using the L</walk>
method.
This base class internally stores counter and gauge metrics as single scalar
values directly. In order to provide flexibility for a variety of
use-cases, it requires assistance from the implementing class on how to store
distribution and timer metrics. The implementing class should provide these
methods, returning whatever values it wishes to implement them with. These
values are stored by the base class, and handed back as part of the L</walk>
method.
The base class stores a value for each unique set of labels and values on
every metric; the subclass does not need to handle this.
=cut
sub new
{
my $class = shift;
# Metrics are keys of $self, named by handle
# Special non-metrics values can be stored by prefixing the name with "\x00"
return bless {}, $class;
}
=head1 METHODS
=cut
use constant HAVE_BATCH_MODE => 1;
sub add_batch_mode_callback
{
my $self = shift;
my ( $cb ) = @_;
push @{ $self->{"\0batch_callbacks"} }, $cb;
}
sub _make
{
my $self = shift;
my ( $type, $handle, %args ) = @_;
my $name = $args{name};
$name = join "_", @$name if ref $name eq "ARRAY";
$self->{$handle} = {
type => $type,
name => $name,
labels => $args{labels},
values => {}, # values per labelset
};
}
sub _metric
{
my $self = shift;
my ( $type, $handle ) = @_;
my $metric = $self->{$handle};
$metric->{type} eq $type or
croak "$handle is not a $type metric";
return $metric;
}
sub _labelset
{
my $self = shift;
my ( $handle, @labelvalues ) = @_;
my $metric = $self->{$handle};
my $labels = $metric->{labels} or return "";
return join "\0", map { "$labels->[$_]:$labelvalues[$_]" } 0 .. $#$labels;
}
=head2 walk
$stored->walk( $code )
$code->( $type, $name, $labels, $value )
Given a CODE reference, this method invokes it once per labelset of every
stored metric.
For each labelset, C<$type> will give the metric type (as a string, either
C<counter>, C<distribution>, C<gauge> or C<timer>), C<$name> gives the name
it was registered with, C<$labels> will be a reference to an even-sized array
containing label names and values.
For counter and gauge metrics, C<$value> will be a numerical scalar giving the
current value. For distribution and timer metrics, C<$value> will be whatever
the implementing class's corresponding C<store_distribution> or C<store_timer>
method returns for them.
=cut
sub walk
{
my $self = shift;
my ( $code ) = @_;
if( my $cbs = $self->{"\0batch_callbacks"} ) {
foreach my $cb ( @$cbs ) { $cb->() }
}
foreach my $handle ( sort keys %$self ) {
next if $handle =~ m/^\0/;
my $metric = $self->{$handle};
my $values = $metric->{values};
foreach my $labelset ( sort keys %$values ) {
my @labels = map { split m/:/, $_, 2 } split m/\0/, $labelset;
$code->( $metric->{type}, $metric->{name}, \@labels, $values->{$labelset} );
}
}
}
=head2 clear_values
$stored->clear_values
Clears all of the metric storage. Every labelset of every metric is deleted.
The metric definitions themselves remain.
=cut
sub clear_values
{
my $self = shift;
$_->{values} = {} for values %$self;
}
sub make_counter { shift->_make( counter => @_ ) }
sub inc_counter_by
{
my $self = shift;
my ( $handle, $amount, @labelvalues ) = @_;
my $metric = $self->_metric( counter => $handle );
$metric->{values}{ $self->_labelset( $handle, @labelvalues ) } += $amount;
}
sub make_distribution { shift->_make( distribution => @_ ) }
sub report_distribution
{
my $self = shift;
my ( $handle, $amount, @labelvalues ) = @_;
my $metric = $self->_metric( distribution => $handle );
my $values = $metric->{values};
my $key = $self->_labelset( $handle, @labelvalues );
$values->{$key} = $self->store_distribution( $values->{$key}, $amount );
}
sub make_gauge { shift->_make( gauge => @_ ) }
sub inc_gauge_by
{
my $self = shift;
my ( $handle, $amount, @labelvalues ) = @_;
my $metric = $self->_metric( gauge => $handle );
$metric->{values}{ $self->_labelset( $handle, @labelvalues ) } += $amount;
}
sub set_gauge_to
{
my $self = shift;
my ( $handle, $amount, @labelvalues ) = @_;
my $metric = $self->_metric( gauge => $handle );
$metric->{values}{ $self->_labelset( $handle, @labelvalues ) } = $amount;
}
sub make_timer { shift->_make( timer => @_ ) }
sub report_timer
{
my $self = shift;
my ( $handle, $duration, @labelvalues ) = @_;
my $metric = $self->_metric( timer => $handle );
my $values = $metric->{values};
my $key = $self->_labelset( $handle, @labelvalues );
$values->{$key} = $self->store_timer( $values->{$key}, $duration );
}
=head1 REQUIRED METHODS
=head2 store_distribution
=head2 store_timer
$storage = $stored->store_distribution( $storage, $amount )
$storage = $stored->store_timer( $storage, $duration )
The implementing class must provide these two methods to assist in the
management of storage for distribution and timer metrics.
When a new observation for the metric is required, the method will be invoked,
passing in the currently-stored perl value for the given metric and label
values, and the new observation. Whatever the method returns is stored by the
base class, to be passed in next time or used by the L</walk> method.
The base class stores this value directly and does not otherwise interact with
it; letting the implementing class decide what is best. For example, a simple
implementation may just store every observation individually by pushing them
into an array; so the C<$storage> would be an ARRAY reference:
sub store_distribution
{
my $self = shift;
my ( $storage, $amount ) = @_;
push @$storage, $amount;
return $storage;
}
=cut
=head1 AUTHOR
Paul Evans <leonerd@leonerd.org.uk>
=cut
0x55AA;
|