mirror of
http://git.haproxy.org/git/haproxy.git/
synced 2024-12-11 14:05:12 +00:00
2c6962c3c0
Currently there is a ~16KB limit for a data size passed via unix socket. It is caused by a trivial bug ttat is going to fixed soon, however in most cases there is no need to dump a full stats. This patch makes possible to select a scope of dumped data by extending current "show stat" to "show stat [<iid> <type> <sid>]": - iid is a proxy id, -1 to dump all proxies - type selects type of dumpable objects: 1 for frontend, 2 for backend, 4 for server, -1 for all types. Values can be ORed, for example: 1+2=3 -> frontend+backend. 1+2+4=7 -> frontend+backend+server. - sid is a service id, -1 to dump everything from the selected proxy. To do this I implemented a new session flag (SN_STAT_BOUND), added three variables in data_ctx.stats (iid, type, sid), modified dumpstats.c and completely revorked the process_uxst_stats: now it waits for a "\n" terminated string, splits args and uses them. BTW: It should be quite easy to add new commands, for example to enable/disable servers, the only problem I can see is a not very lucky config name (*stats* socket). :| During the work I also fixed two bug: - s->flags were not initialized for proto_uxst - missing comma if throttling not enabled (caused by a stupid change in "Implement persistent id for proxies and servers") Other changes: - No more magic type valuse, use STATS_TYPE_FE/STATS_TYPE_BE/STATS_TYPE_SV - Don't memset full s->data_ctx (it was clearing s->data_ctx.stats.{iid/type/sid}, instead initialize stats.sv & stats.sv_st (stats.px and stats.px_st were already initialized) With all that changes it was extremely easy to write a short perl plugin for a perl-enabled net-snmp (also included in this patch). 29385 is my PEN (Private Enterprise Number) and I'm willing to donate the SNMPv2-SMI::enterprises.29385.106.* OIDs for HAProxy if there is nothing assigned already.
243 lines
5.3 KiB
Perl
243 lines
5.3 KiB
Perl
#
|
|
# Net-SNMP perl plugin for Haproxy
|
|
# Version 0.27
|
|
#
|
|
# Copyright 2007-2008 Krzysztof Piotr Oledzki <ole@ans.pl>
|
|
#
|
|
# 1. get a variable from stat:
|
|
# 1.3.6.1.4.1.29385.106.1.$type.$field.$iid.$sid
|
|
# type: 0->frontend, 1->backend, 2->server
|
|
#
|
|
# 2. get a variable from info
|
|
# 1.3.6.1.4.1.29385.106.2.$req.$varnr
|
|
#
|
|
# TODO:
|
|
# - implement read timeout
|
|
#
|
|
|
|
use NetSNMP::agent (':all');
|
|
use NetSNMP::ASN qw(:all);
|
|
use IO::Socket::UNIX;
|
|
|
|
use strict;
|
|
|
|
my $agent = new NetSNMP::agent('Name' => 'Haproxy');
|
|
my $sa = "/var/run/haproxy.stat";
|
|
|
|
use constant OID_HAPROXY => '1.3.6.1.4.1.29385.106';
|
|
use constant OID_HAPROXY_STATS => OID_HAPROXY . '.1';
|
|
use constant OID_HAPROXY_INFO => OID_HAPROXY . '.2';
|
|
|
|
my $oid_stat = new NetSNMP::OID(OID_HAPROXY_STATS);
|
|
my $oid_info = new NetSNMP::OID(OID_HAPROXY_INFO);
|
|
|
|
use constant STATS_PXNAME => 0;
|
|
use constant STATS_SVNAME => 1;
|
|
use constant STATS_IID => 27;
|
|
use constant STATS_SID => 28;
|
|
use constant STATS_TYPE => 32;
|
|
|
|
use constant FIELD_INDEX => 10001;
|
|
use constant FIELD_NAME => 10002;
|
|
|
|
my %info_vars = (
|
|
0 => 'Name',
|
|
1 => 'Version',
|
|
2 => 'Release_date',
|
|
3 => 'Nbproc',
|
|
4 => 'Process_num',
|
|
5 => 'Pid',
|
|
6 => 'Uptime',
|
|
7 => 'Uptime_sec',
|
|
8 => 'Memmax_MB',
|
|
9 => 'Ulimit-n',
|
|
10 => 'Maxsock',
|
|
11 => 'Maxconn',
|
|
12 => 'CurrConns',
|
|
);
|
|
|
|
sub find_next_stat_id {
|
|
my($type, $field, $proxyid, $sid) = @_;
|
|
|
|
my $obj = 1 << $type;
|
|
|
|
my $np = -1;
|
|
my $nl = -1;
|
|
|
|
my $sock = new IO::Socket::UNIX (Peer => $sa, Type => SOCK_STREAM, Timeout => 1);
|
|
next if !$sock;
|
|
|
|
print $sock "show stat -1 $obj -1\n";
|
|
|
|
while(<$sock>) {
|
|
chomp;
|
|
my @d = split(',');
|
|
|
|
last if !$d[$field] && $field != FIELD_INDEX && $field != FIELD_NAME && /^#/;
|
|
next if /^#/;
|
|
|
|
next if $d[STATS_TYPE] != $type;
|
|
|
|
next if ($d[STATS_IID] < $proxyid) || ($d[STATS_IID] == $proxyid && $d[STATS_SID] <= $sid);
|
|
|
|
if ($np == -1 || $d[STATS_IID] < $np || ($d[STATS_IID] == $np && $d[STATS_SID] < $nl)) {
|
|
$np = $d[STATS_IID];
|
|
$nl = $d[STATS_SID];
|
|
next;
|
|
}
|
|
}
|
|
|
|
close($sock);
|
|
|
|
return 0 if ($np == -1);
|
|
|
|
return "$type.$field.$np.$nl"
|
|
}
|
|
|
|
sub haproxy_stat {
|
|
my($handler, $registration_info, $request_info, $requests) = @_;
|
|
|
|
for(my $request = $requests; $request; $request = $request->next()) {
|
|
my $oid = $request->getOID();
|
|
|
|
$oid =~ s/$oid_stat//;
|
|
$oid =~ s/^\.//;
|
|
|
|
my $mode = $request_info->getMode();
|
|
|
|
my($type, $field, $proxyid, $sid, $or) = split('\.', $oid, 5);
|
|
|
|
next if $type > 2 || defined($or);
|
|
|
|
if ($mode == MODE_GETNEXT) {
|
|
|
|
$type = 0 if !$type;
|
|
$field = 0 if !$field;
|
|
$proxyid = 0 if !$proxyid;
|
|
$sid = 0 if !$sid;
|
|
|
|
my $nextid = find_next_stat_id($type, $field, $proxyid, $sid);
|
|
$nextid = find_next_stat_id($type, $field+1, 0, 0) if !$nextid;
|
|
$nextid = find_next_stat_id($type+1, 0, 0, 0) if !$nextid;
|
|
|
|
if ($nextid) {
|
|
($type, $field, $proxyid, $sid) = split('\.', $nextid);
|
|
$request->setOID(sprintf("%s.%s", OID_HAPROXY_STATS, $nextid));
|
|
$mode = MODE_GET;
|
|
}
|
|
}
|
|
|
|
if ($mode == MODE_GET) {
|
|
next if !defined($proxyid) || !defined($type) || !defined($sid) || !defined($field);
|
|
|
|
my $obj = 1 << $type;
|
|
|
|
my $sock = new IO::Socket::UNIX (Peer => $sa, Type => SOCK_STREAM, Timeout => 1);
|
|
next if !$sock;
|
|
|
|
print $sock "show stat $proxyid $obj $sid\n";
|
|
|
|
while(<$sock>) {
|
|
chomp;
|
|
my @data = split(',');
|
|
|
|
last if !defined($data[$field]) && $field != FIELD_INDEX && $field != FIELD_NAME;
|
|
|
|
if ($proxyid) {
|
|
next if $data[STATS_IID] ne $proxyid;
|
|
next if $data[STATS_SID] ne $sid;
|
|
next if $data[STATS_TYPE] ne $type;
|
|
}
|
|
|
|
if ($field == FIELD_INDEX) {
|
|
$request->setValue(ASN_OCTET_STR,
|
|
sprintf("%s.%s", $data[STATS_IID],
|
|
$data[STATS_SID]));
|
|
} elsif ($field == FIELD_NAME) {
|
|
$request->setValue(ASN_OCTET_STR,
|
|
sprintf("%s/%s", $data[STATS_PXNAME],
|
|
$data[STATS_SVNAME]));
|
|
} else {
|
|
$request->setValue(ASN_OCTET_STR, $data[$field]);
|
|
}
|
|
|
|
close($sock);
|
|
last;
|
|
}
|
|
|
|
close($sock);
|
|
next;
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
sub haproxy_info {
|
|
my($handler, $registration_info, $request_info, $requests) = @_;
|
|
|
|
for(my $request = $requests; $request; $request = $request->next()) {
|
|
my $oid = $request->getOID();
|
|
|
|
$oid =~ s/$oid_info//;
|
|
$oid =~ s/^\.//;
|
|
|
|
my $mode = $request_info->getMode();
|
|
|
|
my($req, $nr, $or) = split('\.', $oid, 3);
|
|
|
|
next if $req >= 2 || defined($or);
|
|
|
|
if ($mode == MODE_GETNEXT) {
|
|
$req = 0 if !defined($req);
|
|
$nr = -1 if !defined($nr);
|
|
|
|
if (!defined($info_vars{$nr+1})) {
|
|
$req++;
|
|
$nr = -1;
|
|
}
|
|
|
|
next if $req >= 2;
|
|
|
|
$request->setOID(sprintf("%s.%s.%s", OID_HAPROXY_INFO, $req, ++$nr));
|
|
$mode = MODE_GET;
|
|
|
|
}
|
|
|
|
if ($mode == MODE_GET) {
|
|
|
|
next if !defined($req) || !defined($nr);
|
|
|
|
if ($req == 0) {
|
|
next if !defined($info_vars{$nr});
|
|
$request->setValue(ASN_OCTET_STR, $info_vars{$nr});
|
|
next;
|
|
}
|
|
|
|
if ($req == 1) {
|
|
next if !defined($info_vars{$nr});
|
|
|
|
my $sock = new IO::Socket::UNIX (Peer => $sa, Type => SOCK_STREAM, Timeout => 1);
|
|
next if !$sock;
|
|
|
|
print $sock "show info\n";
|
|
|
|
while(<$sock>) {
|
|
chomp;
|
|
my ($key, $val) = /(.*):\s*(.*)/;
|
|
|
|
next if $info_vars{$nr} ne $key;
|
|
|
|
$request->setValue(ASN_OCTET_STR, $val);
|
|
last;
|
|
}
|
|
|
|
close($sock);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
$agent->register('Haproxy stat', OID_HAPROXY_STATS, \&haproxy_stat);
|
|
$agent->register('Haproxy info', OID_HAPROXY_INFO, \&haproxy_info);
|
|
|