274 lines
9.8 KiB
Perl
274 lines
9.8 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 blockchain::hyperledger::exporter::mode::channels;
|
||
|
|
||
|
use base qw(centreon::plugins::templates::counter);
|
||
|
|
||
|
use strict;
|
||
|
use warnings;
|
||
|
use centreon::common::monitoring::openmetrics::scrape;
|
||
|
use Digest::MD5 qw(md5_hex);
|
||
|
|
||
|
sub set_counters {
|
||
|
my ($self, %options) = @_;
|
||
|
|
||
|
$self->{maps_counters_type} = [
|
||
|
{
|
||
|
name => 'channel', type => 3, cb_prefix_output => 'prefix_channel_output', cb_long_output => 'channel_long_output', indent_long_output => ' ', message_multiple => 'All channels are ok',
|
||
|
group => [
|
||
|
{ name => 'channel_global', type => 0, message_separator => ' - ', skipped_code => { -10 => 1 } },
|
||
|
{ name => 'channel_gscd', type => 0, cb_prefix_output => 'prefix_gscd_output', skipped_code => { -10 => 1 } },
|
||
|
{ name => 'channel_gpvd', type => 0, cb_prefix_output => 'prefix_gpvd_output', skipped_code => { -10 => 1 } },
|
||
|
]
|
||
|
}
|
||
|
];
|
||
|
|
||
|
foreach ((
|
||
|
['gscd', 'gossip.state.commit', 'gossip_state_commit_duration'],
|
||
|
['gpvd', 'gossip.privdata.validation', 'gossip_privdata_validation_duration']
|
||
|
)) {
|
||
|
$self->{maps_counters}->{'channel_' . $_->[0]} = [
|
||
|
{ label => $_->[0] . '-total', nlabel => 'channel.' . $_->[1] . '.total.count', set => {
|
||
|
key_values => [ { name => $_->[2] . '_count', diff => 1 } ],
|
||
|
output_template => '%s (total)',
|
||
|
perfdatas => [
|
||
|
{ value => $_->[2] . '_count_absolute', template => '%s', min => 0,
|
||
|
label_extra_instance => 1 },
|
||
|
],
|
||
|
}
|
||
|
}
|
||
|
];
|
||
|
|
||
|
foreach my $label (('0.005', '0.01', '0.025', '0.05', '0.1', '0.25', '0.5', '1', '2.5', '5', '10', '+Inf')) {
|
||
|
my $perf_label = $label;
|
||
|
$perf_label =~ s/\+Inf/infinite/;
|
||
|
push @{$self->{maps_counters}->{'channel_' . $_->[0]}},
|
||
|
{
|
||
|
label => $_->[0] . '-time-le-' . $perf_label, nlabel => 'channel.' . $_->[1] . '.time.le.' . $perf_label . '.count', set => {
|
||
|
key_values => [ { name => $_->[2] . '_bucket_' . $label, diff => 1 } ],
|
||
|
output_template => '%s (<= ' . $perf_label . ' sec)',
|
||
|
perfdatas => [
|
||
|
{ value => $_->[2] . '_bucket_' . $label . '_absolute', template => '%s', min => 0,
|
||
|
label_extra_instance => 1 },
|
||
|
],
|
||
|
}
|
||
|
};
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$self->{maps_counters}->{channel_global} = [
|
||
|
{ label => 'ledger-transaction', nlabel => 'channel.ledger.transaction.count', set => {
|
||
|
key_values => [ { name => 'ledger_transaction_count', diff => 1 } ],
|
||
|
output_template => 'number of transactions processed: %s',
|
||
|
perfdatas => [
|
||
|
{ value => 'ledger_transaction_count_absolute', template => '%s', min => 0,
|
||
|
label_extra_instance => 1 },
|
||
|
],
|
||
|
}
|
||
|
},
|
||
|
{ label => 'gossip-membership-total-peers-known', nlabel => 'channel.gossip.membership.total.peers.known.count', set => {
|
||
|
key_values => [ { name => 'gossip_membership_total_peers_known' } ],
|
||
|
output_template => 'total known peers: %s',
|
||
|
perfdatas => [
|
||
|
{ value => 'gossip_membership_total_peers_known_absolute', template => '%s', min => 0,
|
||
|
label_extra_instance => 1 },
|
||
|
],
|
||
|
}
|
||
|
},
|
||
|
{ label => 'gossip-state-height', nlabel => 'channel.gossip.state.height.count', set => {
|
||
|
key_values => [ { name => 'gossip_state_height' } ],
|
||
|
output_template => 'current ledger height: %s',
|
||
|
perfdatas => [
|
||
|
{ value => 'gossip_state_height_absolute', template => '%s', min => 0,
|
||
|
label_extra_instance => 1 },
|
||
|
],
|
||
|
}
|
||
|
},
|
||
|
{ label => 'ledger-blockchain-height', nlabel => 'channel.ledger.blockchain.height.count', set => {
|
||
|
key_values => [ { name => 'ledger_blockchain_height' } ],
|
||
|
output_template => 'height of the chain in blocks: %s',
|
||
|
perfdatas => [
|
||
|
{ value => 'ledger_blockchain_height_absolute', template => '%s', min => 0,
|
||
|
label_extra_instance => 1 },
|
||
|
],
|
||
|
}
|
||
|
},
|
||
|
];
|
||
|
}
|
||
|
|
||
|
sub channel_long_output {
|
||
|
my ($self, %options) = @_;
|
||
|
|
||
|
return "checking channel '" . $options{instance_value}->{display} . "'";
|
||
|
}
|
||
|
|
||
|
sub prefix_channel_output {
|
||
|
my ($self, %options) = @_;
|
||
|
|
||
|
return "channel '" . $options{instance_value}->{display} . "' ";
|
||
|
}
|
||
|
|
||
|
sub prefix_gscd_output {
|
||
|
my ($self, %options) = @_;
|
||
|
|
||
|
return 'time it takes to commit a block: ';
|
||
|
}
|
||
|
|
||
|
sub prefix_gpvd_output {
|
||
|
my ($self, %options) = @_;
|
||
|
|
||
|
return 'time it takes to validate a block: ';
|
||
|
}
|
||
|
|
||
|
sub new {
|
||
|
my ($class, %options) = @_;
|
||
|
my $self = $class->SUPER::new(package => __PACKAGE__, %options, statefile => 1, force_new_perfdata => 1);
|
||
|
bless $self, $class;
|
||
|
|
||
|
$options{options}->add_options(arguments => {
|
||
|
'filter-channel:s' => { name => 'filter_channel' },
|
||
|
});
|
||
|
|
||
|
return $self;
|
||
|
}
|
||
|
|
||
|
sub change_macros {
|
||
|
my ($self, %options) = @_;
|
||
|
|
||
|
$options{template} =~ s/%\{(.*?)\}/$options{dimensions}->{$1}/g;
|
||
|
if (defined($options{escape})) {
|
||
|
$options{template} =~ s/([\Q$options{escape}\E])/\\$1/g;
|
||
|
}
|
||
|
return $options{template};
|
||
|
}
|
||
|
|
||
|
sub search_metric {
|
||
|
my ($self, %options) = @_;
|
||
|
|
||
|
return if (!defined($options{metrics}->{$options{label}}));
|
||
|
|
||
|
foreach (@{$options{metrics}->{$options{label}}->{data}}) {
|
||
|
next if (!defined($_->{dimensions}->{$options{dimension}}));
|
||
|
my $dimension = $_->{dimensions}->{$options{dimension}};
|
||
|
next if (defined($self->{option_results}->{filter_channel}) && $self->{option_results}->{filter_channel} ne '' &&
|
||
|
$dimension !~ /$self->{option_results}->{filter_channel}/);
|
||
|
|
||
|
if (!defined($self->{channel}->{$dimension})) {
|
||
|
$self->{channel}->{$dimension} = { display => $dimension };
|
||
|
}
|
||
|
$self->{channel}->{$dimension}->{$options{store}} = {} if (!defined($self->{channel}->{$dimension}->{$options{store}}));
|
||
|
my $key = $self->change_macros(template => $options{key}, dimensions => $_->{dimensions});
|
||
|
$self->{channel}->{$dimension}->{$options{store}}->{$key} = $_->{value};
|
||
|
}
|
||
|
}
|
||
|
|
||
|
sub manage_selection {
|
||
|
my ($self, %options) = @_;
|
||
|
|
||
|
my $metrics = centreon::common::monitoring::openmetrics::scrape::parse(%options, strip_chars => "[\"']");
|
||
|
$self->{channel} = {};
|
||
|
$self->search_metric(
|
||
|
metrics => $metrics,
|
||
|
label => 'gossip_state_commit_duration_bucket',
|
||
|
dimension => 'channel',
|
||
|
key => 'gossip_state_commit_duration_bucket_%{le}',
|
||
|
store => 'channel_gscd'
|
||
|
);
|
||
|
$self->search_metric(
|
||
|
metrics => $metrics,
|
||
|
label => 'gossip_state_commit_duration_count',
|
||
|
dimension => 'channel',
|
||
|
key => 'gossip_state_commit_duration_count',
|
||
|
store => 'channel_gscd'
|
||
|
);
|
||
|
|
||
|
$self->search_metric(
|
||
|
metrics => $metrics,
|
||
|
label => 'gossip_privdata_validation_duration_bucket',
|
||
|
dimension => 'channel',
|
||
|
key => 'gossip_privdata_validation_duration_bucket_%{le}',
|
||
|
store => 'channel_gpvd'
|
||
|
);
|
||
|
$self->search_metric(
|
||
|
metrics => $metrics,
|
||
|
label => 'gossip_privdata_validation_duration_count',
|
||
|
dimension => 'channel',
|
||
|
key => 'gossip_privdata_validation_duration_count',
|
||
|
store => 'channel_gpvd'
|
||
|
);
|
||
|
|
||
|
$self->search_metric(
|
||
|
metrics => $metrics,
|
||
|
label => 'ledger_transaction_count',
|
||
|
dimension => 'channel',
|
||
|
key => 'ledger_transaction_count',
|
||
|
store => 'channel_global'
|
||
|
);
|
||
|
$self->search_metric(
|
||
|
metrics => $metrics,
|
||
|
label => 'gossip_membership_total_peers_known',
|
||
|
dimension => 'channel',
|
||
|
key => 'gossip_membership_total_peers_known',
|
||
|
store => 'channel_global'
|
||
|
);
|
||
|
$self->search_metric(
|
||
|
metrics => $metrics,
|
||
|
label => 'gossip_state_height',
|
||
|
dimension => 'channel',
|
||
|
key => 'gossip_state_height',
|
||
|
store => 'channel_global'
|
||
|
);
|
||
|
$self->search_metric(
|
||
|
metrics => $metrics,
|
||
|
label => 'ledger_blockchain_height',
|
||
|
dimension => 'channel',
|
||
|
key => 'ledger_blockchain_height',
|
||
|
store => 'channel_global'
|
||
|
);
|
||
|
|
||
|
$self->{cache_name} = 'hyperledger_' . $options{custom}->get_uuid() . '_' . $self->{mode} . '_' .
|
||
|
(defined($self->{option_results}->{filter_counters}) ? md5_hex($self->{option_results}->{filter_counters}) : md5_hex('all')) . '_' .
|
||
|
(defined($self->{option_results}->{filter_channel}) ? md5_hex($self->{option_results}->{filter_channel}) : md5_hex('all'));
|
||
|
|
||
|
}
|
||
|
|
||
|
1;
|
||
|
|
||
|
__END__
|
||
|
|
||
|
=head1 MODE
|
||
|
|
||
|
Check blockchain system.
|
||
|
|
||
|
=over 8
|
||
|
|
||
|
=item B<--filter-name>
|
||
|
|
||
|
Filter channel channel (can be a regexp).
|
||
|
|
||
|
=item B<--warning-*> B<--critical-*>
|
||
|
|
||
|
Thresholds. Use --list-counters to get available thresholds options.
|
||
|
|
||
|
=back
|
||
|
|
||
|
=cut
|