centreon-plugins/apps/varnish/local/mode/stats.pm

334 lines
18 KiB
Perl

#
# Copyright 2020 Centreon (http://www.centreon.com/)
#
# Centreon is a full-fledged industry-strength solution that meets
# the needs in IT infrastructure and application monitoring for
# service performance.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
package apps::varnish::local::mode::stats;
use base qw(centreon::plugins::templates::counter);
use strict;
use warnings;
use Digest::MD5 qw(md5_hex);
use JSON::XS;
sub configure_varnish_stats {
my ($self, %options) = @_;
$self->{varnish_stats} = [
{ entry => 'client_conn', nlabel => 'connections.client.accepted.persecond', display_ok => 1, diff => 1, per_second => 1 },
{ entry => 'client_drop', nlabel => 'connections.client.dropped.persecond', display_ok => 1, diff => 1, per_second => 1 },
{ entry => 'client_req', nlabel => 'connections.client.request.received.persecond', display_ok => 1, diff => 1, per_second => 1 },
{ entry => 'client_drop_late', nlabel => 'connections.client.dropped.late.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'client_req_400', nlabel => 'connections.client.request400.received.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'client_req_411', nlabel => 'connections.client.request411.received.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'client_req_413', nlabel => 'connections.client.request413.received.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'client_req_417', nlabel => 'connections.client.request417.received.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'backend_conn', nlabel => 'backends.connections.success.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'backend_unhealthy', nlabel => 'backends.connections.unhealthy.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'backend_busy', nlabel => 'backends.connections.busy.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'backend_fail', nlabel => 'backends.connections.fail.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'backend_reuse', nlabel => 'backends.connections.reuse.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'backend_recycle', nlabel => 'backends.connections.recycle.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'backend_retry', nlabel => 'backends.connections.retry.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'backend_req', nlabel => 'backends.requests.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'cache_hit', nlabel => 'cache.hit.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'cache_hitpass', nlabel => 'cache.hitpass.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'cache_miss', nlabel => 'cache.miss.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'n_sess_mem', nlabel => 'structure.session.memory.count', display_ok => 0 },
{ entry => 'n_sess', nlabel => 'structure.session.count', display_ok => 0 },
{ entry => 'n_sess', nlabel => 'structure.session.count', display_ok => 0 },
{ entry => 'n_object', nlabel => 'structure.object.count', display_ok => 0 },
{ entry => 'n_vampireobject', nlabel => 'object.unresurrected.count', display_ok => 0 },
{ entry => 'n_objectcore', nlabel => 'structure.objectcore.count', display_ok => 0 },
{ entry => 'n_objecthead', nlabel => 'structure.objecthead.count', display_ok => 0 },
{ entry => 'n_waitinglist', nlabel => 'structure.waitinglist.count', display_ok => 0 },
{ entry => 'n_vbc', nlabel => 'structure.vbc.count', display_ok => 0 },
{ entry => 'n_backend', nlabel => 'backend.count', display_ok => 0 },
{ entry => 'n_expired', nlabel => 'object.expired.count', display_ok => 0 },
{ entry => 'n_lru_nuked', nlabel => 'object.lru.nuked.count', display_ok => 0 },
{ entry => 'n_lru_moved', nlabel => 'object.lru.moved.count', display_ok => 0 },
{ entry => 'n_objsendfile', nlabel => 'object.sent.file.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'n_objwrite', nlabel => 'object.sent.write.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'n_objoverflow', nlabel => 'object.overflow.workspace.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'shm_records', nlabel => 'shm.records.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'shm_writes', nlabel => 'shm.writes.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'shm_flushes', nlabel => 'shm.flushes.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'shm_cont', nlabel => 'shm.contentions.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'shm_cycles', nlabel => 'shm.cycles.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'sms_nreq', nlabel => 'sms.allocator.requests.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'sms_nobj', nlabel => 'sms.outstanding.allocations.count', display_ok => 0 },
{ entry => 'sms_nbytes', nlabel => 'sms.outstanding.bytes', display_ok => 0, custom_output => $self->can('custom_output_scale_bytes') },
{ entry => 'sms_balloc', nlabel => 'sms.outstanding.allocated.bytes', display_ok => 0, custom_output => $self->can('custom_output_scale_bytes') },
{ entry => 'sms_bfree', nlabel => 'sms.outstanding.freed.bytes', display_ok => 0, custom_output => $self->can('custom_output_scale_bytes') },
{ entry => 'fetch_head', nlabel => 'fetch.head.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'fetch_length', nlabel => 'fetch.length.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'fetch_chunked', nlabel => 'fetch.chunked.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'fetch_eof', nlabel => 'fetch.eof.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'fetch_bad', nlabel => 'fetch.badheaders.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'fetch_close', nlabel => 'fetch.close.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'fetch_oldhttp', nlabel => 'fetch.oldhttp.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'fetch_zero', nlabel => 'fetch.zero.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'fetch_failed', nlabel => 'fetch.failed.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'fetch_1xx', nlabel => 'fetch.1xx.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'fetch_204', nlabel => 'fetch.204.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'fetch_304', nlabel => 'fetch.304.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'n_ban', nlabel => 'ban.total.active.count', display_ok => 0 },
{ entry => 'n_ban_add', nlabel => 'ban.new.added.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'n_ban_retire', nlabel => 'ban.old.deleted.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'n_ban_obj_test', nlabel => 'ban.object.tested.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'n_ban_re_test', nlabel => 'ban.object.tested.regexp.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'n_ban_dups', nlabel => 'ban.duplicate.removed.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'dir_dns_lookups', nlabel => 'dns.director.lookups.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'dir_dns_failed', nlabel => 'dns.director.lookups.failed.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'dir_dns_hit', nlabel => 'dns.director.lookups.cachehit.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'dir_dns_cache_full', nlabel => 'dns.director.cache.full.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'esi_errors', nlabel => 'esi.parse.errors.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'esi_warnings', nlabel => 'esi.parse.warnings.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'hcb_nolock', nlabel => 'hck.lookups.nolock.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'hcb_lock', nlabel => 'hck.lookups.lock.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'hcb_insert', nlabel => 'hck.inserts.persecond', display_ok => 0, diff => 1, per_second => 1 },
{ entry => 'n_vcl', nlabel => 'vlc.total.count', display_ok => 0, diff => 1 },
{ entry => 'n_vcl_avail', nlabel => 'vlc.available.count', display_ok => 0, diff => 1 },
{ entry => 'n_vcl_discard', nlabel => 'vlc.discarded.count', display_ok => 0, diff => 1 },
{ entry => 'sess_conn', nlabel => 'sessions.accepted.count', display_ok => 0, diff => 1 },
{ entry => 'sess_drop', nlabel => 'sessions.dropped.count', display_ok => 0, diff => 1 },
{ entry => 'sess_fail', nlabel => 'sessions.failed.count', display_ok => 0, diff => 1 },
{ entry => 'sess_pipe_overflow', nlabel => 'sessions.pipe.overflow.count', display_ok => 0, diff => 1 },
{ entry => 'sess_queued', nlabel => 'sessions.queued.count', display_ok => 0, diff => 1 },
{ entry => 'sess_readahead', nlabel => 'sessions.readahead.count', display_ok => 0, diff => 1 },
{ entry => 'sess_closed', nlabel => 'sessions.closed.count', display_ok => 0, diff => 1 },
{ entry => 'sess_herd', nlabel => 'sessions.herd.count', display_ok => 0, diff => 1 },
{ entry => 'sess_linger', nlabel => 'sessions.linger.count', display_ok => 0, diff => 1 },
{ entry => 'sess_closed', nlabel => 'sessions.closed.count', display_ok => 0, diff => 1 },
{ entry => 'sess_pipeline', nlabel => 'sessions.pipeline.count', display_ok => 0, diff => 1 },
{ entry => 'threads', nlabel => 'threads.total.count', display_ok => 0 },
{ entry => 'threads_created', nlabel => 'threads.created.count', display_ok => 0, diff => 1 },
{ entry => 'threads_limited', nlabel => 'threads.limited.count', display_ok => 0, diff => 1 },
{ entry => 'threads_destroyed', nlabel => 'threads.destroyed.count', display_ok => 0, diff => 1 },
{ entry => 'threads_failed', nlabel => 'threads.failed.count', display_ok => 0, diff => 1 },
{ entry => 'thread_queue_len', nlabel => 'threads.queue.length.count', display_ok => 0 },
{ entry => 's_sess', nlabel => 'total.sessions.seen.count', display_ok => 0, diff => 1 },
{ entry => 's_req', nlabel => 'total.requests.count', display_ok => 0, diff => 1 },
{ entry => 's_fetch', nlabel => 'total.backends.fetch.count', display_ok => 0, diff => 1 },
{ entry => 'n_wrk', nlabel => 'workers.threads.count', display_ok => 0 },
{ entry => 'n_wrk_create', nlabel => 'workers.threads.created.count', display_ok => 0, diff => 1 },
{ entry => 'n_wrk_failed', nlabel => 'workers.threads.failed.count', display_ok => 0, diff => 1 },
{ entry => 'n_wrk_max', nlabel => 'workers.threads.limited.count', display_ok => 0, diff => 1 },
{ entry => 'n_wrk_lqueue', nlabel => 'workers.requests.queue.length.count', display_ok => 0, diff => 1 },
{ entry => 'n_wrk_queued', nlabel => 'workers.requests.queued.count', display_ok => 0, diff => 1 },
{ entry => 'n_wrk_drop', nlabel => 'workers.requests.dropped.count', display_ok => 0, diff => 1 },
];
}
sub custom_output_scale_bytes {
my ($self, %options) = @_;
my $label = $self->{label};
$label =~ s/-/_/g;
my ($value, $unit) = $self->{perfdata}->change_bytes(value => $self->{result_values}->{$label . '_absolute'});
my $msg = sprintf('%s: %.2f %s', $self->{result_values}->{$label . '_description_absolute'}, $value, $unit);
return $msg;
}
sub custom_output_second {
my ($self, %options) = @_;
my $label = $self->{label};
$label =~ s/-/_/g;
my $msg = sprintf('%s: %.2f/s', $self->{result_values}->{$label . '_description_absolute'}, $self->{result_values}->{$label . '_per_second'});
return $msg;
}
sub custom_output {
my ($self, %options) = @_;
my $label = $self->{label};
$label =~ s/-/_/g;
my $msg = sprintf('%s: %s', $self->{result_values}->{$label . '_description_absolute'}, $self->{result_values}->{$label . '_absolute'});
return $msg;
}
sub set_counters {
my ($self, %options) = @_;
$self->configure_varnish_stats();
$self->{maps_counters_type} = [
{ name => 'global', type => 0, skipped_code => { -10 => 1 } },
];
$self->{maps_counters}->{global} = [];
foreach (@{$self->{varnish_stats}}) {
my $label = $_->{entry};
$label =~ s/_/-/g;
push @{$self->{maps_counters}->{global}},
{ label => $label, nlabel => $_->{nlabel}, display_ok => $_->{display_ok}, set => {
key_values => [ { name => $_->{entry}, diff => $_->{diff} }, { name => $_->{entry}. '_description' } ],
closure_custom_output => defined($_->{custom_output}) ? $_->{custom_output} :
(defined($_->{per_second}) ? $self->can('custom_output_second') : $self->can('custom_output')),
per_second => $_->{per_second},
perfdatas => [
{ label => $_->{entry}, value => $_->{entry} . (defined($_->{per_second}) ? '_per_second' : '_absolute'),
template => defined($_->{per_second}) ? '%.2f' : '%s',
min => 0 },
],
}
}
;
}
}
sub new {
my ($class, %options) = @_;
my $self = $class->SUPER::new(package => __PACKAGE__, %options, statefile => 1);
bless $self, $class;
$options{options}->add_options(arguments => {
'hostname:s' => { name => 'hostname' },
'remote' => { name => 'remote' },
'ssh-option:s@' => { name => 'ssh_option' },
'ssh-path:s' => { name => 'ssh_path' },
'ssh-command:s' => { name => 'ssh_command', default => 'ssh' },
'timeout:s' => { name => 'timeout', default => 30 },
'sudo' => { name => 'sudo' },
'command:s' => { name => 'command', default => 'varnishstat' },
'command-path:s' => { name => 'command_path', default => '/usr/bin' },
'command-options:s' => { name => 'command_options', default => ' -1 -j 2>&1' },
});
return $self;
}
sub check_varnish_old {
my ($self, %options) = @_;
return if (!defined($options{json}->{uptime}));
# "cache_hit": {"value": 56320, "flag": "a", "description": "Cache hits"},
# "cache_hitpass": {"value": 0, "flag": "a", "description": "Cache hits for pass"},
foreach (@{$self->{varnish_stats}}) {
next if (!defined($options{json}->{$_->{entry}}));
$self->{global}->{$_->{entry}} = $options{json}->{$_->{entry}}->{value};
$self->{global}->{$_->{entry} . '_description'} = $options{json}->{$_->{entry}}->{description};
}
}
sub check_varnish_new {
my ($self, %options) = @_;
return if (!defined($options{json}->{'MAIN.uptime'}));
# "MAIN.cache_hit": {"type": "MAIN", "value": 18437, "flag": "a", "description": "Cache hits"},
# "MAIN.cache_hitpass": {"type": "MAIN", "value": 3488, "flag": "a", "description": "Cache hits for pass"},
# "MAIN.cache_miss": {"type": "MAIN", "value": 5782, "flag": "a", "description": "Cache misses"},
foreach (@{$self->{varnish_stats}}) {
next if (!defined($options{json}->{'MAIN.' . $_->{entry}}));
$self->{global}->{$_->{entry}} = $options{json}->{'MAIN.' . $_->{entry}}->{value};
$self->{global}->{$_->{entry} . '_description'} = $options{json}->{'MAIN.' . $_->{entry}}->{description};
}
}
sub manage_selection {
my ($self, %options) = @_;
my ($stdout) = centreon::plugins::misc::execute(
output => $self->{output},
options => $self->{option_results},
sudo => $self->{option_results}->{sudo},
command => $self->{option_results}->{command},
command_path => $self->{option_results}->{command_path},
command_options => $self->{option_results}->{command_options}
);
$self->{global} = {};
my $content;
eval {
$content = JSON::XS->new->utf8->decode($stdout);
};
if ($@) {
$self->{output}->add_option_msg(short_msg => "Cannot decode json response: $@");
$self->{output}->option_exit();
}
$self->check_varnish_old(json => $content);
$self->check_varnish_new(json => $content);
$self->{cache_name} = "cache_varnish_" . $self->{mode} . '_' .
(defined($self->{option_results}->{hostname}) ? md5_hex($self->{option_results}->{hostname}) : md5_hex('all')) . '_' .
(defined($self->{option_results}->{filter_counters}) ? md5_hex($self->{option_results}->{filter_counters}) : md5_hex('all'));
};
1;
__END__
=head1 MODE
Check statistics with varnishstat command
=over 8
=item B<--remote>
If you dont run this script locally, if you wanna use it remote, you can run it remotely with 'ssh'.
=item B<--hostname>
Hostname to query (need --remote).
=item B<--ssh-option>
Specify multiple options like the user (example: --ssh-option='-l=centreon-engine' --ssh-option='-p=52').
=item B<--command>
Varnishstat Binary Filename (Default: varnishstat)
=item B<--command-path>
Directory Path to Varnishstat Binary File (Default: /usr/bin)
=item B<--command-options>
Parameter for Binary File (Default: ' -1 -j 2>&1')
=item B<--warning-[countername]> B<--critical-[countername]>
Thresholds. Use option --list-counters to see available counters
=back
=cut