mirror of
https://github.com/librenms/librenms-agent.git
synced 2024-05-09 09:54:52 +00:00
546 lines
16 KiB
Perl
Executable File
546 lines
16 KiB
Perl
Executable File
#!/usr/bin/env perl
|
|
#Copyright (c) 2023, Zane C. Bowers-Hadley
|
|
#All rights reserved.
|
|
#
|
|
#Redistribution and use in source and binary forms, with or without modification,
|
|
#are permitted provided that the following conditions are met:
|
|
#
|
|
# * Redistributions of source code must retain the above copyright notice,
|
|
# this list of conditions and the following disclaimer.
|
|
# * Redistributions in binary form must reproduce the above copyright notice,
|
|
# this list of conditions and the following disclaimer in the documentation
|
|
# and/or other materials provided with the distribution.
|
|
#
|
|
#THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
#ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
#WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
#IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
|
|
#INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
|
#BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
#DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
|
#LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
|
|
#OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
|
|
#THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
=head1 NAME
|
|
|
|
dhcp - LibreNMS ISC-DHCPD stats extend
|
|
|
|
=head1 SYNOPSIS
|
|
|
|
dhcp [B<-Z>] [B<-d>] [B<-p>] [B<-l> <file>]
|
|
|
|
=head1 FLAGS
|
|
|
|
=head2 -l <lease file>
|
|
|
|
Path to the lease file.
|
|
|
|
=head2 -Z
|
|
|
|
Enable GZip+Base64 compression.
|
|
|
|
=head2 -d
|
|
|
|
Do not de-dup.
|
|
|
|
This is done via making sure the combination of UID, CLTT, IP, HW address,
|
|
client hostname, and state are unique.
|
|
|
|
=head2 -n
|
|
|
|
If no shared networks are defined, what to use for generating the network names
|
|
for reporting purposes.
|
|
|
|
- cidr :: Use the cidr for the defined subnets.
|
|
|
|
- cidr+range :: Use the cidr+range for the defined subnets.
|
|
|
|
Default is 'cidr'.
|
|
|
|
=head2 -w <file>
|
|
|
|
Write the the output to this file.
|
|
|
|
=head1 Return JSON Data Hash
|
|
|
|
- .all_networks.cur :: Current leases for all networks
|
|
- .all_networks.max :: Max possible leases for all networks
|
|
- .all_networks.percent :: Percent of total pool usage.
|
|
|
|
- .networks.[].cur :: Current leases for the network.
|
|
- .networks.[].max :: Max possible leases for the networks
|
|
- .networks.[].network :: Name of the network.
|
|
- .networks.[].subnets :: Array of subnets on the network.
|
|
- .networks.[].percent :: Percent of network usage.
|
|
- .networks.[].pools :: Pool ranges used.
|
|
|
|
- .pools.[].cur :: Current leases for the pool.
|
|
- .pools.[].max :: Max possible leases for pool.
|
|
- .pools.[].first_ip :: First IP of the pool.
|
|
- .pools.[].last_ip :: Last IP of the pool.
|
|
- .pools.[].percent :: Percent of pool usage.
|
|
- .pools.[].cidr :: CIDR for this subnet.
|
|
- .pools.[].$option :: Additional possible DHCP subnet option.
|
|
|
|
- .found_leases.[].client_hostname :: Hostname the client passed during the request.
|
|
- .found_leases.[].cltt :: The CLTT for the requist.
|
|
- .found_leases.[].ends :: Unix time of of when the lease ends.
|
|
- .found_leases.[].hw_address :: Hardware address for the client that made the request.
|
|
- .found_leases.[].ip :: IP address of the client that made the request.
|
|
- .found_leases.[].starts :: Unix time of of when the lease starts.
|
|
- .found_leases.[].state :: State of the lease.
|
|
- .found_leases.[].uid :: UID passed during the request.
|
|
- .found_leases.[].vendor_class_identifier :: Vendor class identifier passed during the request.
|
|
|
|
The following are Base64 encoded as they may include binary that breaks either SNMP or
|
|
the PHP JSON decoder.
|
|
|
|
- .found_leases.[].vendor_class_identifier
|
|
- .found_leases.[].uid :: UID passed during the request.
|
|
- .found_leases.[].vendor_class_identifier
|
|
|
|
=cut
|
|
|
|
use strict;
|
|
use warnings;
|
|
use Getopt::Std;
|
|
use JSON -convert_blessed_universally;
|
|
use MIME::Base64;
|
|
use IO::Compress::Gzip qw(gzip $GzipError);
|
|
use Net::ISC::DHCPd::Leases;
|
|
use Net::ISC::DHCPd::Config;
|
|
use File::Slurp;
|
|
|
|
my %opts;
|
|
getopts( 'l:Zdpc:n:w:', \%opts );
|
|
|
|
if ( !defined( $opts{n} ) ) {
|
|
$opts{n} = 'cidr';
|
|
} else {
|
|
if ( $opts{n} ne 'cidr' && $opts{n} ne 'cidr+range' ) {
|
|
$opts{n} = 'cidr';
|
|
}
|
|
}
|
|
|
|
if ( !defined( $opts{l} ) ) {
|
|
# if freebsd, set it to the default path as used by the version installed via ports
|
|
#
|
|
# additional elsifs should be added as they become known, but default works for most Linux distros
|
|
if ( $^O eq 'freebsd' ) {
|
|
$opts{l} = '/var/db/dhcpd/dhcpd.leases';
|
|
} else {
|
|
$opts{l} = '/var/lib/dhcpd/dhcpd.leases';
|
|
}
|
|
} ## end if ( !defined( $opts{l} ) )
|
|
|
|
if ( !defined( $opts{c} ) ) {
|
|
# if freebsd, set it to the default path as used by the version installed via ports
|
|
#
|
|
# additional elsifs should be added as they become known, but default works for most Linux distros
|
|
if ( $^O eq 'freebsd' ) {
|
|
$opts{c} = '/usr/local/etc/dhcpd.conf';
|
|
} else {
|
|
$opts{c} = '/etc/dhcp/dhcpd.conf';
|
|
}
|
|
} ## end if ( !defined( $opts{c} ) )
|
|
|
|
$Getopt::Std::STANDARD_HELP_VERSION = 1;
|
|
|
|
sub main::VERSION_MESSAGE {
|
|
print "LibreNMS ISC-DHCPD extend 0.0.2\n";
|
|
}
|
|
|
|
sub main::HELP_MESSAGE {
|
|
print '
|
|
-l <lease file> Path to the lease file.
|
|
-c <dhcpd.conf> Path to the dhcpd.conf file.
|
|
-Z Enable GZip+Base64 compression.
|
|
-d Do not de-dup.
|
|
';
|
|
|
|
exit;
|
|
} ## end sub main::HELP_MESSAGE
|
|
|
|
my $to_return = {
|
|
data => {
|
|
lease_file => $opts{l},
|
|
found_leases => [],
|
|
leases => {
|
|
abandoned => 0,
|
|
active => 0,
|
|
backup => 0,
|
|
bootp => 0,
|
|
expired => 0,
|
|
free => 0,
|
|
released => 0,
|
|
reset => 0,
|
|
total => 0,
|
|
},
|
|
networks => [],
|
|
pools => [],
|
|
all_networks => { cur => 0, max => 0, percent => 0, },
|
|
},
|
|
version => 3,
|
|
error => 0,
|
|
errorString => '',
|
|
};
|
|
|
|
if ( !-f $opts{l} && !-r $opts{l} ) {
|
|
$to_return->{error} = 2;
|
|
$to_return->{errorString} = '"' . $opts{l} . '" does not exist, is not a file, or is not readable';
|
|
print decode_json($to_return) . "\n";
|
|
exit;
|
|
}
|
|
|
|
# hash for storing found leases for later deduping
|
|
my $found_leases = {};
|
|
|
|
##
|
|
##
|
|
## read in the leases
|
|
##
|
|
##
|
|
my $leases;
|
|
eval {
|
|
my $leases_obj = Net::ISC::DHCPd::Leases->new( file => $opts{l} );
|
|
$leases_obj->parse;
|
|
$leases = $leases_obj->leases;
|
|
};
|
|
if ($@) {
|
|
$to_return->{error} = 1;
|
|
$to_return->{errorString} = 'Reading leases failed... ' . $@;
|
|
print decode_json($to_return) . "\n";
|
|
exit;
|
|
}
|
|
|
|
##
|
|
##
|
|
## process found leases
|
|
##
|
|
##
|
|
foreach my $lease ( @{$leases} ) {
|
|
if ( !defined( $lease->{uid} ) ) {
|
|
$lease->{uid} = '';
|
|
}
|
|
if ( !defined( $lease->{vendor_class_identifier} ) ) {
|
|
$lease->{vendor_class_identifier} = '';
|
|
}
|
|
if ( !defined( $lease->{cltt} ) ) {
|
|
$lease->{cltt} = '';
|
|
}
|
|
if ( !defined( $lease->{state} ) ) {
|
|
$lease->{state} = '';
|
|
}
|
|
if ( !defined( $lease->{ip_address} ) ) {
|
|
$lease->{ip_address} = '';
|
|
}
|
|
if ( !defined( $lease->{hardware_address} ) ) {
|
|
$lease->{hardware_address} = '';
|
|
}
|
|
if ( !defined( $lease->{client_hostname} ) ) {
|
|
$lease->{client_hostname} = '';
|
|
}
|
|
} ## end foreach my $lease ( @{$leases} )
|
|
|
|
##
|
|
##
|
|
## dedup or copy lease info as is
|
|
##
|
|
##
|
|
if ( !$opts{d} ) {
|
|
foreach my $lease ( @{$leases} ) {
|
|
$found_leases->{ $lease->{uid}
|
|
. $lease->{ip_address}
|
|
. $lease->{client_hostname}
|
|
. $lease->{state}
|
|
. $lease->{hardware_address} } = $lease;
|
|
}
|
|
foreach my $lease_key ( keys( %{$found_leases} ) ) {
|
|
my $uid = $found_leases->{$lease_key}{uid};
|
|
if ( $uid ne '' ) {
|
|
$uid = encode_base64($uid);
|
|
chomp($uid);
|
|
}
|
|
my $client_hostname = $found_leases->{$lease_key}{client_hostname};
|
|
if ( $client_hostname ne '' ) {
|
|
$client_hostname = encode_base64($client_hostname);
|
|
chomp($client_hostname);
|
|
}
|
|
my $vendor_class_identifier = $found_leases->{$lease_key}{vendor_class_identifier};
|
|
if ( $vendor_class_identifier ne '' ) {
|
|
$vendor_class_identifier = encode_base64($vendor_class_identifier);
|
|
chomp($vendor_class_identifier);
|
|
}
|
|
push(
|
|
@{ $to_return->{data}{found_leases} },
|
|
{
|
|
uid => $uid,
|
|
cltt => $found_leases->{$lease_key}{cltt},
|
|
state => $found_leases->{$lease_key}{state},
|
|
ip => $found_leases->{$lease_key}{ip_address},
|
|
hw_address => $found_leases->{$lease_key}{hardware_address},
|
|
starts => $found_leases->{$lease_key}{starts},
|
|
ends => $found_leases->{$lease_key}{ends},
|
|
client_hostname => $client_hostname,
|
|
vendor_class_identifier => $vendor_class_identifier,
|
|
}
|
|
);
|
|
} ## end foreach my $lease_key ( keys( %{$found_leases} ...))
|
|
} else {
|
|
foreach my $lease ( @{$leases} ) {
|
|
my $uid = $lease->{uid};
|
|
if ( $uid ne '' ) {
|
|
$uid = encode_base64($uid);
|
|
chomp($uid);
|
|
}
|
|
my $client_hostname = $lease->{client_hostname};
|
|
if ( $client_hostname ne '' ) {
|
|
$client_hostname = encode_base64($client_hostname);
|
|
chomp($client_hostname);
|
|
}
|
|
my $vendor_class_identifier = $lease->{vendor_class_identifier};
|
|
if ( $vendor_class_identifier ne '' ) {
|
|
$vendor_class_identifier = encode_base64($vendor_class_identifier);
|
|
chomp($vendor_class_identifier);
|
|
}
|
|
push(
|
|
@{ $to_return->{data}{found_leases} },
|
|
{
|
|
uid => $uid,
|
|
cltt => $lease->{cltt},
|
|
state => $lease->{state},
|
|
ip => $lease->{ip_address},
|
|
hw_address => $lease->{hardware_address},
|
|
starts => $lease->{starts},
|
|
ends => $lease->{ends},
|
|
client_hostname => $client_hostname,
|
|
vendor_class_identifier => $vendor_class_identifier,
|
|
}
|
|
);
|
|
} ## end foreach my $lease ( @{$leases} )
|
|
} ## end else [ if ( !$opts{d} ) ]
|
|
|
|
##
|
|
##
|
|
## total the lease info types
|
|
##
|
|
##
|
|
foreach my $lease ( @{ $to_return->{data}{found_leases} } ) {
|
|
$to_return->{data}{leases}{total}++;
|
|
if ( $lease->{state} eq 'free' ) {
|
|
$to_return->{data}{leases}{free}++;
|
|
} elsif ( $lease->{state} eq 'abandoned' ) {
|
|
$to_return->{data}{leases}{abandoned}++;
|
|
} elsif ( $lease->{state} eq 'active' ) {
|
|
$to_return->{data}{leases}{active}++;
|
|
} elsif ( $lease->{state} eq 'backup' ) {
|
|
$to_return->{data}{leases}{backup}++;
|
|
} elsif ( $lease->{state} eq 'bootp' ) {
|
|
$to_return->{data}{leases}{bootp}++;
|
|
} elsif ( $lease->{state} eq 'expired' ) {
|
|
$to_return->{data}{leases}{expired}++;
|
|
} elsif ( $lease->{state} eq 'released' ) {
|
|
$to_return->{data}{leases}{released}++;
|
|
} elsif ( $lease->{state} eq 'reset' ) {
|
|
$to_return->{data}{leases}{reset}++;
|
|
}
|
|
} ## end foreach my $lease ( @{ $to_return->{data}{found_leases...}})
|
|
|
|
##
|
|
##
|
|
## read in the config
|
|
##
|
|
##
|
|
my $config_obj;
|
|
eval {
|
|
$config_obj = Net::ISC::DHCPd::Config->new( file => $opts{c} );
|
|
$config_obj->parse;
|
|
};
|
|
if ($@) {
|
|
$to_return->{error} = 3;
|
|
$to_return->{errorString} = 'Reading leases failed... ' . $@;
|
|
print decode_json($to_return) . "\n";
|
|
exit;
|
|
}
|
|
|
|
##
|
|
##
|
|
## process found subnets
|
|
##
|
|
##
|
|
my $pools = {};
|
|
my @subnets = $config_obj->subnets;
|
|
foreach my $subnet (@subnets) {
|
|
my @ranges = $subnet->ranges;
|
|
my $subnet_cidr_obj = $subnet->address;
|
|
my $subnet_cidr = $subnet_cidr_obj->addr;
|
|
foreach my $range (@ranges) {
|
|
my $lower = $range->lower;
|
|
my $upper = $range->upper;
|
|
my $pool_name = $lower->addr . '-' . $upper->addr;
|
|
my $subnet_addr = $subnet_cidr_obj->addr;
|
|
my $subnet_cidr = $subnet_cidr_obj->cidr;
|
|
my $max = $upper->bigint - $lower->bigint;
|
|
$pools->{$pool_name} = {
|
|
first_ip => $lower->addr,
|
|
lower => $lower,
|
|
last_ip => $upper->addr,
|
|
upper => $upper,
|
|
subnet => $subnet_addr,
|
|
cidr => $subnet_cidr,
|
|
max => $max,
|
|
cur => 0,
|
|
percent => 0,
|
|
};
|
|
my @options = $subnet->options;
|
|
|
|
foreach my $option (@options) {
|
|
$pools->{$pool_name}{ $option->name } = $option->value;
|
|
}
|
|
} ## end foreach my $range (@ranges)
|
|
} ## end foreach my $subnet (@subnets)
|
|
|
|
##
|
|
##
|
|
## process found networks and subnets contained on in
|
|
##
|
|
##
|
|
my $networks = {};
|
|
my @found_subnets = $config_obj->sharednetworks;
|
|
my $undef_network_name_int = 0;
|
|
foreach my $network (@found_subnets) {
|
|
my $name = $network->name;
|
|
if ( !defined($name) || $name eq '' ) {
|
|
$name = 'undef' . $undef_network_name_int;
|
|
$undef_network_name_int++;
|
|
}
|
|
if ( !defined( $networks->{$name} ) ) {
|
|
$networks->{$name} = [];
|
|
}
|
|
|
|
@subnets = $network->subnets;
|
|
foreach my $subnet (@subnets) {
|
|
my @ranges = $subnet->ranges;
|
|
my $subnet_cidr_obj = $subnet->address;
|
|
my $subnet_cidr = $subnet_cidr_obj->addr;
|
|
foreach my $range (@ranges) {
|
|
my $lower = $range->lower;
|
|
my $upper = $range->upper;
|
|
my $pool_name = $lower->addr . '-' . $upper->addr;
|
|
my $max = $upper->bigint - $lower->bigint;
|
|
$pools->{$pool_name} = {
|
|
first_ip => $lower->addr,
|
|
lower => $lower,
|
|
last_ip => $upper->addr,
|
|
upper => $upper,
|
|
subnet => $subnet_cidr,
|
|
max => $max,
|
|
cur => 0,
|
|
percent => 0,
|
|
};
|
|
my @options = $subnet->options;
|
|
foreach my $option (@options) {
|
|
$pools->{$pool_name}{ $option->name } = $option->value;
|
|
}
|
|
|
|
push( @{ $networks->{$name} }, $pool_name );
|
|
} ## end foreach my $range (@ranges)
|
|
} ## end foreach my $subnet (@subnets)
|
|
} ## end foreach my $network (@found_subnets)
|
|
|
|
##
|
|
##
|
|
## puts the pools array together
|
|
##
|
|
##
|
|
foreach my $pool_key ( keys( %{$pools} ) ) {
|
|
my $lower = $pools->{$pool_key}{lower};
|
|
delete( $pools->{$pool_key}{lower} );
|
|
my $upper = $pools->{$pool_key}{upper};
|
|
delete( $pools->{$pool_key}{upper} );
|
|
|
|
# check each lease for if it is between the upper and lower IPs
|
|
# then increment current if the state is active
|
|
foreach my $lease ( @{ $to_return->{data}{found_leases} } ) {
|
|
my $lease_ip = NetAddr::IP->new( $lease->{ip} );
|
|
if ( $lower <= $lease_ip && $lease_ip <= $upper ) {
|
|
if ( $lease->{state} eq 'active' ) {
|
|
$pools->{$pool_key}{cur}++;
|
|
}
|
|
}
|
|
}
|
|
|
|
$pools->{$pool_key}{percent} = ( $pools->{$pool_key}{cur} / $pools->{$pool_key}{max}->numify() ) * 100;
|
|
$pools->{$pool_key}{max} = $pools->{$pool_key}{max}->bstr;
|
|
|
|
# add the current and max to all_networks(reall all subnets)...
|
|
$to_return->{data}{all_networks}{cur} = $to_return->{data}{all_networks}{cur} + $pools->{$pool_key}{cur};
|
|
$to_return->{data}{all_networks}{max} = $to_return->{data}{all_networks}{max} + $pools->{$pool_key}{max};
|
|
|
|
push( @{ $to_return->{data}{pools} }, $pools->{$pool_key} );
|
|
} ## end foreach my $pool_key ( keys( %{$pools} ) )
|
|
$to_return->{data}{all_networks}{percent}
|
|
= ( $to_return->{data}{all_networks}{cur} / $to_return->{data}{all_networks}{max} ) * 100;
|
|
|
|
##
|
|
##
|
|
## put the networks section together
|
|
##
|
|
##
|
|
my @network_keys = keys( %{$networks} );
|
|
if ( !defined( $network_keys[0] ) ) {
|
|
foreach my $pool_key ( keys( %{$pools} ) ) {
|
|
$networks->{ $pools->{$pool_key}{cidr} } = [$pool_key];
|
|
}
|
|
@network_keys = keys( %{$networks} );
|
|
}
|
|
foreach my $network (@network_keys) {
|
|
my $cur = 0;
|
|
my $max = 0;
|
|
foreach my $pool_name ( @{ $networks->{$network} } ) {
|
|
$cur = $cur + $pools->{$pool_name}{cur};
|
|
$max = $max + $pools->{$pool_name}{max};
|
|
}
|
|
my $percent = ( $cur / $max ) * 100;
|
|
push(
|
|
@{ $to_return->{data}{networks} },
|
|
{
|
|
cur => $cur,
|
|
max => $max,
|
|
network => $network,
|
|
percent => $percent,
|
|
pools => $networks->{$network},
|
|
}
|
|
);
|
|
} ## end foreach my $network (@network_keys)
|
|
|
|
##
|
|
##
|
|
## handle printing the output
|
|
##
|
|
##
|
|
my $json = JSON->new->allow_nonref->canonical(1);
|
|
if ( $opts{p} ) {
|
|
$json->pretty;
|
|
}
|
|
my $toReturn = $json->encode($to_return) . "\n";
|
|
if ( $opts{Z} ) {
|
|
my $toReturnCompressed;
|
|
gzip \$toReturn => \$toReturnCompressed;
|
|
my $compressed = encode_base64($toReturnCompressed);
|
|
$compressed =~ s/\n//g;
|
|
$compressed = $compressed . "\n";
|
|
if ( length($compressed) < length($toReturn) ) {
|
|
$toReturn = $compressed;
|
|
}
|
|
} ## end if ( $opts{Z} )
|
|
|
|
print $toReturn;
|
|
|
|
if ($opts{w}) {
|
|
write_file($opts{w}, $toReturn);
|
|
}
|
|
|
|
exit;
|