forked from HariSekhon/Nagios-Plugins
-
Notifications
You must be signed in to change notification settings - Fork 0
/
check_cassandra_balance.pl
executable file
·111 lines (92 loc) · 3.63 KB
/
check_cassandra_balance.pl
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
#!/usr/bin/perl -T
# nagios: -epn
#
# Author: Hari Sekhon
# Date: 2013-11-04 01:52:54 +0000 (Mon, 04 Nov 2013)
#
# https://github.com/harisekhon/nagios-plugins
#
# License: see accompanying LICENSE file
#
$DESCRIPTION = "Nagios Plugin to check the balance of ownership of tokens across nodes
Uses nodetool's status command to find token % across all nodes and alerts if the largest difference is greater than warning/critical thresholds. Returns perfdata of the max imbalance % for graphing.
Use --verbose mode to also output max & min node % token ownership and rack information
Can specify a remote host and port otherwise assumes to check via localhost
Tested on Cassandra 1.2.9, 2.0.1, 2.0.9, 2.2.5, 3.0.8, 3.5, 3.6, 3.7";
$VERSION = "0.5.3";
use strict;
use warnings;
BEGIN {
use File::Basename;
use lib dirname(__FILE__) . "/lib";
}
use HariSekhonUtils;
use HariSekhon::Cassandra::Nodetool;
set_threshold_defaults(10, 20);
my $exclude_joining_leaving = 0;
%options = (
%nodetool_options,
"exclude-joining-leaving" => [ \$exclude_joining_leaving, "Exclude Joining/Leaving nodes from the balance calculation (Downed nodes are already excluded)" ],
%thresholdoptions,
);
@usage_order = qw/nodetool host port user password exclude-joining-leaving warning critical/;
get_options();
($nodetool, $host, $port, $user, $password) = validate_nodetool_options($nodetool, $host, $port, $user, $password);
validate_thresholds(undef, undef, { "simple" => "upper", "integer" => 0, "positive" => 1, "max" => "100" });
vlog2;
set_timeout();
$status = "OK";
my $options = nodetool_options($host, $port, $user, $password);
my $cmd = "${nodetool} ${options}status";
vlog2 "fetching cluster nodes information";
my @output = cmd($cmd);
# name % rack
my @max_node = ("uninitialized_node", 0, "uninitialized_rack");
my @min_node = ("uninitialized_node", 100, "uninitialized_rack");
my $node_count = 0;
foreach(@output){
check_nodetool_errors($_);
if($_ =~ $nodetool_status_header_regex){
next;
}
# Only consider up nodes
next if(/^D[NJLM]\s+/);
next if($exclude_joining_leaving and /^U[JL]\s+/);
if(/^[^\s]+\s+([^\s]+)\s+[^\s]+(?:\s+[A-Za-z][A-Za-z])?\s+[^\s]+\s+(?:\d+\s+)?(?:(\d+(?:\.\d+)?\%|\?))\s+[^\s]+\s+([^\s]+)\s*$/){
$node_count++;
my $node = $1;
my $percentage = $2;
my $rack = $3;
vlog3 "node = $node";
vlog3 "percentage = $percentage";
vlog3 "rack = $rack\n";
if($percentage eq "?"){
quit "UNKNOWN", "nodetool returned '?' for token percentage ownership, Cassandra can't determine it's own token % we need to calculate the balance. Perhaps this node was newly started?";
}
$percentage =~ s/%$//;
if($percentage > $max_node[1]){
@max_node = ($node, $percentage, $rack);
}
if($percentage < $min_node[1]){
@min_node = ($node, $percentage, $rack);
}
} elsif(skip_nodetool_output($_)){
# ignore
} else {
die_nodetool_unrecognized_output($_);
}
}
if($node_count < 1){
quit "UNKNOWN", "no nodes found!";
} elsif($node_count == 1){
@min_node = @max_node;
}
my $max_diff_percentage = sprintf("%.2f", $max_node[1] - $min_node[1]);
plural $node_count;
$msg = "$max_diff_percentage% max imbalance between $node_count cassandra node$plural";
check_thresholds($max_diff_percentage);
$msg .= ", max node: $max_node[1]% [$max_node[0] ($max_node[2])], min node: $min_node[1]% [$min_node[0] ($min_node[2])]" if $verbose;
$msg .= " | 'max_%_imbalance'=$max_diff_percentage%";
msg_perf_thresholds();
vlog2;
quit $status, $msg;