2021-09-16 22:08:35 +02:00
|
|
|
#!/usr/bin/perl
|
|
|
|
# -*- perl -*-
|
|
|
|
|
2021-10-09 08:18:15 +02:00
|
|
|
# CAVE: Es muss ein Hostanme per AP gesetzt sein!
|
|
|
|
|
2021-09-16 22:08:35 +02:00
|
|
|
=encoding utf8
|
|
|
|
|
|
|
|
=head1 NAME
|
|
|
|
|
|
|
|
unifi_api - Munin plugin to display device and network information from the
|
|
|
|
Ubiquiti unifi API
|
|
|
|
|
|
|
|
=head1 APPLICABLE SYSTEMS
|
|
|
|
|
|
|
|
Unifi controllors with direct API access
|
|
|
|
|
|
|
|
Controller version 5+ required (tested with 5.8.x)
|
|
|
|
|
|
|
|
WebRTC is not supported at this time
|
|
|
|
|
|
|
|
=head1 CONFIGURATION
|
|
|
|
|
|
|
|
This script uses the multigraph functionality to generate many graphs. As such, there
|
|
|
|
are a significant amount of available configuration options
|
|
|
|
|
|
|
|
=head2 API Details
|
|
|
|
|
|
|
|
You will need to supply your API login details:
|
|
|
|
|
|
|
|
[unifi_api]
|
|
|
|
# User name to login to unifi controller API. Default is "ubnt". Ideally, this should
|
|
|
|
# point to a read-only account.
|
|
|
|
env.user Controller_Username
|
|
|
|
|
|
|
|
# Password to login to unifi controller API. Default is "ubnt"
|
|
|
|
env.pass Controller_Password
|
|
|
|
|
|
|
|
# URL of the API, with port if needed. No trailing slash.
|
|
|
|
# Default is https://localhost:8443
|
|
|
|
env.api_url https://unifi.fqdn.com:8443
|
|
|
|
|
|
|
|
# Verify SSL certificate name against host.
|
|
|
|
# Note: if using a default cloudkey certificate, this will fail unless you manually add it
|
|
|
|
# to the local keystore.
|
|
|
|
# Default is "yes"
|
|
|
|
env.ssl_verify_host no
|
|
|
|
|
|
|
|
# Verify Peer's SSL vertiicate.
|
|
|
|
# Note: if using a default cloudkey certificate, this will fail
|
|
|
|
# Default is "yes"
|
|
|
|
env.ssl_verify_peer no
|
|
|
|
|
|
|
|
# The human readable name of the unifi site - used for graph titles
|
|
|
|
env.name Site Name
|
|
|
|
|
|
|
|
# "Site" string - the internal unifi API site identifier.
|
|
|
|
# default is "default" - found when you connect to the web interface
|
|
|
|
# it's the term in the URL - /manage/site/site_string/dashboard
|
|
|
|
env.site site_string
|
|
|
|
|
|
|
|
|
|
|
|
=head2 Graph Categories / Host Management
|
|
|
|
|
|
|
|
Sometimes, you need more control over where the unifi graphs appear.
|
|
|
|
|
|
|
|
env.force_category 0
|
|
|
|
# By default, Use standard munin well know categories -
|
|
|
|
# system: cpu, mem, load, & uptime
|
|
|
|
# network: clients, transfer statistics.
|
|
|
|
#
|
|
|
|
|
|
|
|
To use this feature, set "force_category" to a text string (i.e. "unifi").
|
|
|
|
|
|
|
|
This is very helpful if your graphs are going to appear inside another host - for instance
|
|
|
|
if your munin graphs for that host are monitoring the host the controller is running on, and
|
|
|
|
the unifi API instance.
|
|
|
|
|
|
|
|
Sometimes however, you want to monitor either an offsite API, or a cloudkey which, at least by
|
|
|
|
default, does not run munin-node. In that case, you can actually create a "virtual" munin host to
|
|
|
|
display only these graphs (or any combination you like). This is documented in the main munin docs,
|
|
|
|
but in a nutshell:
|
|
|
|
|
|
|
|
In your munin-node plugin configuration: (Something like: /etc/munin/plugin-conf.d/munin-node)
|
|
|
|
|
|
|
|
[unifi_api]
|
|
|
|
host_name hostname.whatever.youlike
|
|
|
|
env.force_category unifi
|
|
|
|
|
|
|
|
And, in your munin master configuration: (Something like: /etc/munin/munin.conf)
|
|
|
|
|
|
|
|
[hostname.whatever.youlike]
|
|
|
|
address ip.of.munin.node
|
|
|
|
|
|
|
|
Make sure you do *not* set "use_node_name" on this new host. It may be necessary to define "host_name"
|
|
|
|
in your munin-node configuration as well, if you have not already (Likely, on a multi-homed host, this
|
|
|
|
has been done to keep munin-node from advertising itself as localhost)
|
|
|
|
|
|
|
|
More information:
|
|
|
|
|
|
|
|
* L<host_name|http://guide.munin-monitoring.org/en/latest/plugin/use.html>
|
|
|
|
|
|
|
|
|
|
|
|
=head2 Toggling of graphs / Individual options
|
|
|
|
|
|
|
|
You can turn off individual graphs. A few graphs have extra configuration
|
|
|
|
options.
|
|
|
|
|
|
|
|
By default, everything is enabled. Set to "no" to disable
|
|
|
|
|
|
|
|
[unifi_api]
|
|
|
|
# Show device CPU utilization
|
|
|
|
env.enable_device_cpu yes
|
|
|
|
|
|
|
|
# Show device memory usage
|
|
|
|
env.enable_device_mem yes
|
|
|
|
|
|
|
|
# Show device load average (switches and APs only)
|
|
|
|
env.enable_device_load yes
|
|
|
|
|
|
|
|
# Show device uptime
|
|
|
|
env.enable_device_uptime yes
|
|
|
|
|
|
|
|
# Show number of clients connected to each device
|
|
|
|
env.enable_clients_device yes
|
|
|
|
# Show detailed graphs for each device (per device graphs)
|
|
|
|
env.enable_detail_clients_device yes
|
|
|
|
|
|
|
|
# Show number of clients connected to each network type
|
|
|
|
env.enable_clients_type yes
|
|
|
|
# Show detailed graphs for each client type (per type graphs)
|
|
|
|
env.enable_detail_clients_type yes
|
|
|
|
# Show unauthorized / authorized client list
|
|
|
|
# if you are not using the guest portal, this is useless
|
|
|
|
env.show_authorized_clients_type yes
|
|
|
|
|
|
|
|
# Show transfer statistics on switch ports
|
|
|
|
env.enable_xfer_port yes
|
|
|
|
# Show detailed graphs per switch port
|
|
|
|
env.enable_detail_xfer_port yes
|
|
|
|
# Hide ports that have no link (When set to no, unplugged ports will transfer 0, not be undefined)
|
|
|
|
env.hide_empty_xfer_port yes
|
|
|
|
|
|
|
|
# Show transfer statistics per device
|
|
|
|
env.enable_xfer_device yes
|
|
|
|
# Show detailed graphs for each device
|
|
|
|
env.enable_detail_xfer_device yes
|
|
|
|
|
|
|
|
# Show transfer statistics per named network
|
|
|
|
env.enable_xfer_network yes
|
|
|
|
# Show detailed graphs for each named network
|
|
|
|
env.enable_detail_xfer_network yes
|
|
|
|
|
|
|
|
# Show transfer statistics per radio
|
|
|
|
env.enable_xfer_radio yes
|
|
|
|
# Show detailed graphs for each radio
|
|
|
|
env.enable_detail_xfer_radio yes
|
|
|
|
|
|
|
|
|
|
|
|
=head1 CAPABILITIES
|
|
|
|
|
|
|
|
This plugin supports L<DIRTYCONFIG|http://guide.munin-monitoring.org/en/latest/plugin/protocol-dirtyconfig.html>
|
|
|
|
|
|
|
|
=head1 DEPENDENCIES
|
|
|
|
|
|
|
|
This plugin requires munin-multiugraph.
|
|
|
|
|
|
|
|
=over
|
|
|
|
|
|
|
|
=item WWW::Curl::Easy
|
|
|
|
|
|
|
|
Perl extension interface for libcurl
|
|
|
|
|
|
|
|
=item JSON
|
|
|
|
|
|
|
|
JSON (JavaScript Object Notation) encoder/decoder
|
|
|
|
|
|
|
|
=back
|
|
|
|
|
|
|
|
=head1 PERFORMANCE CONCERNS
|
|
|
|
|
|
|
|
The main performance concern on this is the huge number of graphs that may be
|
|
|
|
generated. Using the cron version of munin-graph may hurt a lot.
|
|
|
|
|
|
|
|
A bit of a case study:
|
|
|
|
|
|
|
|
| My Site | UBNT Demo
|
|
|
|
---------------------------------------
|
|
|
|
Devices | 8 | 126
|
|
|
|
AP's | 4 | 118
|
|
|
|
24xSwitch | 1 | 5
|
|
|
|
8xSwitch | 2 | 2
|
|
|
|
Output Bytes | 64,262 | 431,434
|
|
|
|
Output Lines | 1,761 | 14,586
|
|
|
|
Output Graphs | 77 | 530
|
|
|
|
|
|
|
|
So, just note that the growth in the amount of graphed date can be extreme.
|
|
|
|
|
|
|
|
|
|
|
|
=head1 LICENSE
|
|
|
|
|
|
|
|
Copyright (C) 2018 J.T.Sage (jtsage@gmail.com)
|
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see L<http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
=head1 MAGIC MARKERS
|
|
|
|
|
|
|
|
#%# family=manual
|
|
|
|
#%# capabilities=
|
|
|
|
|
|
|
|
=cut
|
|
|
|
|
|
|
|
use warnings;
|
|
|
|
use strict;
|
|
|
|
use utf8;
|
|
|
|
use Munin::Plugin;
|
|
|
|
|
|
|
|
# Check dependencies
|
|
|
|
my @errorCode;
|
|
|
|
my $me = (split '/', $0)[-1];
|
|
|
|
|
|
|
|
if (! eval {require JSON; JSON->import(); 1; } ) {
|
|
|
|
push @errorCode, "JSON module not found";
|
|
|
|
}
|
|
|
|
if (! eval {require WWW::Curl::Easy; 1;} ) {
|
|
|
|
push @errorCode, "WWW::Curl::Easy module not found";
|
|
|
|
}
|
|
|
|
|
|
|
|
# Fail on not found dependencies
|
|
|
|
if ( @errorCode != 0 ) {
|
|
|
|
die "FATAL:$me: Perl dependencies not installed (", join(", " => @errorCode), ")\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
# Multigraph cabability is required for this plugin
|
|
|
|
need_multigraph();
|
|
|
|
|
|
|
|
# Somewhat (in)sane defaults for host, pass, etc
|
|
|
|
my %APIconfig = (
|
|
|
|
'user' => env_default_text('user' , 'ubnt'),
|
|
|
|
'pass' => env_default_text('pass' , 'ubnt'),
|
|
|
|
'api_url' => env_default_text('api_url' , 'https://localhost:8443'),
|
|
|
|
'site' => env_default_text('site' , 'default'),
|
|
|
|
'ssl_verify_host' => env_default_text('ssl_verify_host', 'yes'),
|
|
|
|
'ssl_verify_peer' => env_default_text('ssl_verify_peer', 'yes'),
|
|
|
|
'name' => env_default_text('name' , 'Unnamed Site'),
|
|
|
|
);
|
|
|
|
|
|
|
|
# The big table of plugin options - see POD documentation for what these do.
|
|
|
|
my %PluginConfig = (
|
|
|
|
'enable_device_cpu' => env_default_bool_true('enable_device_cpu'),
|
|
|
|
'enable_device_mem' => env_default_bool_true('enable_device_mem'),
|
|
|
|
'enable_device_load' => env_default_bool_true('enable_device_load'),
|
|
|
|
'enable_device_uptime' => env_default_bool_true('enable_device_uptime'),
|
|
|
|
'enable_clients_device' => env_default_bool_true('enable_clients_device'),
|
|
|
|
'enable_clients_type' => env_default_bool_true('enable_clients_network'),
|
|
|
|
'enable_xfer_port' => env_default_bool_true('enable_xfer_port'),
|
|
|
|
'enable_xfer_device' => env_default_bool_true('enable_xfer_device'),
|
|
|
|
'enable_xfer_network' => env_default_bool_true('enable_xfer_network'),
|
|
|
|
'enable_xfer_radio' => env_default_bool_true('enable_xfer_radio'),
|
|
|
|
'enable_detail_xfer_port' => env_default_bool_true('enable_detail_xfer_port'),
|
|
|
|
'enable_detail_xfer_device' => env_default_bool_true('enable_detail_xfer_device'),
|
|
|
|
'enable_detail_xfer_network' => env_default_bool_true('enable_detail_xfer_network'),
|
|
|
|
'enable_detail_xfer_radio' => env_default_bool_true('enable_detail_xfer_radio'),
|
|
|
|
'enable_detail_clients_device' => env_default_bool_true('enable_detail_clients_device'),
|
|
|
|
'enable_detail_clients_type' => env_default_bool_true('enable_detail_clients_network'),
|
|
|
|
'hide_empty_xfer_port' => env_default_bool_true('hide_empty_xfer_port'),
|
|
|
|
'show_authorized_clients_type' => env_default_bool_true('show_authorized_clients_type'),
|
|
|
|
'force_category' => env_default_text('force_category', 0),
|
|
|
|
);
|
|
|
|
|
|
|
|
# Set up needed API endpoints
|
|
|
|
my %APIPoint = (
|
|
|
|
'login' => $APIconfig{"api_url"} . "/api/login",
|
|
|
|
'device' => $APIconfig{"api_url"} . "/api/s/" . $APIconfig{"site"} . "/stat/device",
|
|
|
|
'wlan' => $APIconfig{"api_url"} . "/api/s/" . $APIconfig{"site"} . "/rest/wlanconf",
|
|
|
|
'sta' => $APIconfig{"api_url"} . "/api/s/" . $APIconfig{"site"} . "/stat/sta",
|
|
|
|
);
|
|
|
|
|
|
|
|
my %APIResponse;
|
|
|
|
my %APIJsonResponse;
|
|
|
|
my %Data;
|
|
|
|
my $retcode;
|
|
|
|
|
|
|
|
# Init curl and JSON
|
|
|
|
my $curl = WWW::Curl::Easy->new() or die "FATAL:$me: WWW::Curl::Easy init failed!\n";
|
|
|
|
my $jsonOBJ = JSON->new() or die "FATAL:$me: JSON init failed!\n";
|
|
|
|
|
|
|
|
|
|
|
|
## Fetch the data from the API
|
|
|
|
|
|
|
|
# The rest is a way to use local files from the local disk. Undocumented and not really supported.
|
|
|
|
|
|
|
|
if ( !env_default_bool_true('USE_API') ) {
|
|
|
|
if (! eval {require File::Slurp; File::Slurp->import(); 1; } ) {
|
|
|
|
die "Local debug unavailable, File::Slurp CPAN module required\n";
|
|
|
|
}
|
|
|
|
foreach ( "./demo-test-files/device", "./demo-test-files/sta", "./demo-test-files/wlanconf" ) {
|
|
|
|
if ( ! -f $_ ) { die "File not found: $_\n"; }
|
|
|
|
}
|
|
|
|
$APIJsonResponse{'device'} = $jsonOBJ->allow_nonref->utf8->relaxed->decode(read_file('./demo-test-files/device'));
|
|
|
|
$APIJsonResponse{'sta'} = $jsonOBJ->allow_nonref->utf8->relaxed->decode(read_file('./demo-test-files/sta'));
|
|
|
|
$APIJsonResponse{'wlan'} = $jsonOBJ->allow_nonref->utf8->relaxed->decode(read_file('./demo-test-files/wlanconf'));
|
|
|
|
} else {
|
|
|
|
fetch_data();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
## Process the data
|
|
|
|
|
|
|
|
make_data();
|
|
|
|
|
|
|
|
if ( defined($ARGV[0]) && $ARGV[0] eq "config" ) {
|
|
|
|
# Do the config step for each set of graphs
|
|
|
|
do_config_mem();
|
|
|
|
do_config_cpu();
|
|
|
|
do_config_load();
|
|
|
|
do_config_uptime();
|
|
|
|
do_config_xfer_by_device();
|
|
|
|
do_config_xfer_by_uplink();
|
|
|
|
do_config_xfer_by_port();
|
|
|
|
do_config_xfer_by_network();
|
|
|
|
do_config_xfer_by_radio();
|
|
|
|
do_config_clients_by_device();
|
|
|
|
do_config_clients_by_type();
|
|
|
|
|
|
|
|
# If dirtyconfig is not supported, or turned off, exit here. Otherwise, continue to fetch section
|
|
|
|
if ( !defined($ENV{'MUNIN_CAP_DIRTYCONFIG'}) || !$ENV{'MUNIN_CAP_DIRTYCONFIG'} ) { exit 0; }
|
|
|
|
}
|
|
|
|
|
|
|
|
# Do the fetch step for each set of graphs
|
|
|
|
do_values_cpu();
|
|
|
|
do_values_mem();
|
|
|
|
do_values_load();
|
|
|
|
do_values_uptime();
|
|
|
|
do_values_xfer_by_device();
|
|
|
|
do_values_xfer_by_uplink();
|
|
|
|
do_values_xfer_by_port();
|
|
|
|
do_values_xfer_by_network();
|
|
|
|
do_values_xfer_by_radio();
|
|
|
|
do_values_clients_by_device();
|
|
|
|
do_values_clients_by_type();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#######################
|
|
|
|
# SUBROUTINES CONFIG #
|
|
|
|
#######################
|
|
|
|
|
|
|
|
sub do_config_clients_by_type {
|
|
|
|
# Provide client count by type - CONFIG
|
|
|
|
if ( !$PluginConfig{'enable_clients_type'} ) { return 0; }
|
|
|
|
|
|
|
|
graph_prologue(
|
|
|
|
'unifi_clients_per_network',
|
|
|
|
'Clients Connected / Network',
|
|
|
|
'-l 0 --base 1000',
|
|
|
|
'clients',
|
|
|
|
'network',
|
|
|
|
'Clients connected per type - manually summing these numbers may be meaningful, as clients are often of multiple types'
|
|
|
|
);
|
|
|
|
|
|
|
|
foreach ( @{$Data{'typesOrder'}} ) {
|
|
|
|
print $_ , ".label " , $Data{'types'}{$_}[0] , "\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( ! $PluginConfig{'enable_detail_clients_type'} ) { return 1; }
|
|
|
|
|
|
|
|
foreach ( @{$Data{'typesOrder'}} ) {
|
|
|
|
if ( $Data{'types'}{$_}[1] == 1 ) {
|
|
|
|
graph_prologue(
|
|
|
|
'unifi_clients_per_network.' . $_,
|
|
|
|
'Clients Connected : ' . $Data{'types'}{$_}[0],
|
|
|
|
'-l 0 --base 1000',
|
|
|
|
'clients',
|
|
|
|
'network',
|
|
|
|
'Clients connected via that are of type: ' . $Data{'types'}{$_}[0]
|
|
|
|
);
|
|
|
|
print "users.label Users\n";
|
|
|
|
print "guests.label Guests\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub do_config_clients_by_device {
|
|
|
|
# Provide client count by device - CONFIG
|
|
|
|
if ( !$PluginConfig{'enable_clients_device'} ) { return 0; }
|
|
|
|
|
|
|
|
graph_prologue(
|
|
|
|
'unifi_clients_per_device',
|
|
|
|
'Clients Connected / Device',
|
|
|
|
'-l 0 --base 1000',
|
|
|
|
'clients',
|
|
|
|
'network',
|
|
|
|
'Clients connected to each unifi device'
|
|
|
|
);
|
|
|
|
|
|
|
|
foreach ( sort keys %{$Data{'device'}} ) {
|
|
|
|
print $_ , ".label " , $Data{'device'}{$_}->{'label'} , "\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( ! $PluginConfig{'enable_detail_clients_device'} ) { return 1; }
|
|
|
|
|
|
|
|
foreach ( sort keys %{$Data{'device'}} ) {
|
|
|
|
graph_prologue(
|
|
|
|
'unifi_clients_per_device.' . $_,
|
|
|
|
'Clients / Device : ' . $Data{'device'}{$_}->{'label'},
|
|
|
|
'-l 0 --base 1000',
|
|
|
|
'clients',
|
|
|
|
'network',
|
|
|
|
'Clients connected to the ' . $Data{'device'}{$_}->{'label'} . " unifi device"
|
|
|
|
);
|
|
|
|
print "users.label Users\n";
|
|
|
|
print "guests.label Guests\n";
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub do_config_xfer_by_radio {
|
|
|
|
# Provide transfer for radios - CONFIG
|
|
|
|
if ( !$PluginConfig{'enable_xfer_radio'} ) { return 0; }
|
|
|
|
|
|
|
|
graph_prologue(
|
|
|
|
'unifi_xfer_per_radio',
|
|
|
|
'Transfer / radio',
|
|
|
|
'--base 1000',
|
|
|
|
'Packets/${graph_period}',
|
|
|
|
'network',
|
|
|
|
'Number of packets transferred per individual radio band'
|
|
|
|
);
|
|
|
|
|
|
|
|
foreach my $thisDevice ( sort keys %{$Data{'device'}} ) {
|
|
|
|
if ( $Data{'device'}{$thisDevice}->{'type'} ne "uap" ) { next; }
|
|
|
|
|
|
|
|
foreach ( @{$Data{'device'}{$thisDevice}{'radio'}} ) {
|
|
|
|
print $thisDevice , "_" , $_->{"name"} , "_pack.label " , $_->{"label"} , "\n";
|
|
|
|
print $thisDevice , "_" , $_->{"name"} , "_pack.type DERIVE\n";
|
|
|
|
print $thisDevice , "_" , $_->{"name"} , "_pack.min 0\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( ! $PluginConfig{'enable_detail_xfer_radio'} ) { return 1; }
|
|
|
|
|
|
|
|
foreach my $thisDevice ( sort keys %{$Data{'device'}} ) {
|
|
|
|
if ( $Data{'device'}{$thisDevice}->{'type'} ne "uap" ) { next; }
|
|
|
|
|
|
|
|
graph_prologue(
|
|
|
|
'unifi_xfer_per_radio.' . $thisDevice,
|
|
|
|
'Transfer / radio : ' . $Data{'device'}{$thisDevice}->{'name'},
|
|
|
|
'--base 1000',
|
|
|
|
'Packets/${graph_period}',
|
|
|
|
'network',
|
|
|
|
'Transfered Packets, Dropped / Retried Packets, and Error Packets for the WLAN device: ' . $Data{'device'}{$thisDevice}->{'name'}
|
|
|
|
);
|
|
|
|
|
|
|
|
foreach ( @{$Data{'device'}{$thisDevice}{'radio'}} ) {
|
|
|
|
print $_->{"name"} , "_pkt.label " , $_->{"type"} , " Packets\n";
|
|
|
|
print $_->{"name"} , "_pkt.type DERIVE\n";
|
|
|
|
print $_->{"name"} , "_pkt.min 0\n";
|
|
|
|
print $_->{"name"} , "_dret.label " , $_->{"type"} , " Dropped / Retries\n";
|
|
|
|
print $_->{"name"} , "_dret.type DERIVE\n";
|
|
|
|
print $_->{"name"} , "_dret.min 0\n";
|
|
|
|
print $_->{"name"} , "_err.label " , $_->{"type"} , " Errors\n";
|
|
|
|
print $_->{"name"} , "_err.type DERIVE\n";
|
|
|
|
print $_->{"name"} , "_err.min 0\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub do_config_xfer_by_network {
|
|
|
|
# Provide transfer for named networks - CONFIG
|
|
|
|
if ( !$PluginConfig{'enable_xfer_network'} ) { return 0; }
|
|
|
|
|
|
|
|
graph_prologue(
|
|
|
|
'unifi_xfer_per_network',
|
|
|
|
'Transfer / named network',
|
|
|
|
'--base 1000',
|
|
|
|
'Bytes/${graph_period} rcvd (-) / trans (+)',
|
|
|
|
'network',
|
|
|
|
'Bytes sent and received per each named network'
|
|
|
|
);
|
|
|
|
|
|
|
|
foreach my $thisNet ( sort keys %{$Data{'networks'}} ) {
|
|
|
|
foreach ( "_rxbytes", "_txbytes" ) {
|
|
|
|
print $thisNet , $_ , ".label " , $Data{'networks'}{$thisNet}->{"label"} . "\n";
|
|
|
|
print $thisNet , $_ , ".type DERIVE\n";
|
|
|
|
print $thisNet , $_ , ".min 0\n";
|
|
|
|
}
|
|
|
|
print $thisNet , "_rxbytes.graph no\n";
|
|
|
|
print $thisNet , "_txbytes.negative " , $thisNet , "_rxbytes\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( ! $PluginConfig{'enable_detail_xfer_network'} ) { return 1; }
|
|
|
|
|
|
|
|
foreach my $thisNet ( sort keys %{$Data{'networks'}} ) {
|
|
|
|
graph_prologue(
|
|
|
|
'unifi_xfer_per_network.' . $thisNet,
|
|
|
|
'Transfer / named network : ' . $Data{'networks'}{$thisNet}->{'label'},
|
|
|
|
'--base 1000',
|
|
|
|
'Bytes/${graph_period} rcvd (-) / trans (+)',
|
|
|
|
'network',
|
|
|
|
'Bytes sent and received for the network named: ' . $Data{'networks'}{$thisNet}->{'label'}
|
|
|
|
);
|
|
|
|
foreach ( "rxbyte", "txbyte" ) {
|
|
|
|
print $_ , ".label Bytes\n";
|
|
|
|
print $_ , ".type DERIVE\n";
|
|
|
|
print $_ , ".min 0\n";
|
|
|
|
}
|
|
|
|
print "rxbyte.graph no\n";
|
|
|
|
print "txbyte.negative rxbyte\n";
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub do_config_xfer_by_port {
|
|
|
|
# Provide transfer for switch ports - CONFIG
|
|
|
|
if ( !$PluginConfig{'enable_xfer_port'} ) { return 0; }
|
|
|
|
|
|
|
|
foreach my $thisDevice ( sort keys %{$Data{'device'}} ) {
|
|
|
|
if ( $Data{'device'}{$thisDevice}->{'type'} ne "usw" ) { next; }
|
|
|
|
graph_prologue(
|
|
|
|
'unifi_xfer_per_port_' . $thisDevice,
|
|
|
|
'Transfer / port : ' . $Data{'device'}{$thisDevice}->{'label'},
|
|
|
|
'--base 1000',
|
|
|
|
'Bytes/${graph_period} rcvd (-) / trans (+)',
|
|
|
|
'network',
|
|
|
|
'Bytes sent and received per port on the switch named: ' . $Data{'device'}{$thisDevice}->{'label'}
|
|
|
|
);
|
|
|
|
foreach my $thisPort ( @{$Data{'device'}{$thisDevice}{'ports'}} ) {
|
|
|
|
foreach ( "_rxbytes", "_txbytes" ) {
|
|
|
|
print $thisDevice , "_" , $thisPort->{"name"} , $_ , ".label " , $thisPort->{"label"} . "\n";
|
|
|
|
print $thisDevice , "_" , $thisPort->{"name"} , $_ , ".type DERIVE\n";
|
|
|
|
print $thisDevice , "_" , $thisPort->{"name"} , $_ , ".min 0\n";
|
|
|
|
}
|
|
|
|
print $thisDevice , "_" , $thisPort->{"name"} , "_rxbytes.graph no\n";
|
|
|
|
print $thisDevice , "_" , $thisPort->{"name"} , "_txbytes.negative " , $thisDevice , "_" , $thisPort->{"name"} , "_rxbytes\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( ! $PluginConfig{'enable_detail_xfer_port'} ) { return 1; }
|
|
|
|
|
|
|
|
# Extended graphs
|
|
|
|
foreach my $thisDevice ( sort keys %{$Data{'device'}} ) {
|
|
|
|
if ( $Data{'device'}{$thisDevice}->{'type'} ne "usw" ) { next; }
|
|
|
|
foreach my $thisPort ( @{$Data{'device'}{$thisDevice}{'ports'}} ) {
|
|
|
|
graph_prologue(
|
|
|
|
'unifi_xfer_per_port_' . $thisDevice . "." . $thisPort->{'name'},
|
|
|
|
'Transfer / port : ' . $Data{'device'}{$thisDevice}->{'label'} . " : " . $thisPort->{'label'},
|
|
|
|
'--base 1000',
|
|
|
|
'Bytes/${graph_period} rcvd (-) / trans (+)',
|
|
|
|
'network',
|
|
|
|
'Bytes sent and received on port "' . $thisPort->{'label'} . '" of the switch "' . $Data{'device'}{$thisDevice}->{'label'} . '"'
|
|
|
|
);
|
|
|
|
foreach ( "rxbyte", "txbyte" ) {
|
|
|
|
print $_ . ".label Bytes\n";
|
|
|
|
print $_ . ".type DERIVE\n";
|
|
|
|
print $_ . ".min 0\n";
|
|
|
|
}
|
|
|
|
print "rxbyte.graph no\n";
|
|
|
|
print "txbyte.negative rxbyte\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub do_config_xfer_by_uplink {
|
|
|
|
# Provide transfer for unifi uplink - CONFIG
|
|
|
|
if ( !$PluginConfig{'enable_xfer_device'} ) { return 0; }
|
|
|
|
|
|
|
|
graph_prologue(
|
|
|
|
'unifi_xfer_by_uplink',
|
|
|
|
'Transfer on uplink : ' . $Data{'uplink'}{'devName'},
|
|
|
|
'--base 1000',
|
|
|
|
'Bytes/${graph_period} rcvd (-) / trans (+)',
|
|
|
|
'network',
|
|
|
|
'Bytes sent and received on the WAN port of the USG, and the speedtest result of the same port'
|
|
|
|
);
|
|
|
|
|
|
|
|
foreach ( "rx", "tx" ) {
|
|
|
|
print $_ , "_speed.label Speedtest\n";
|
|
|
|
print $_ , "_bytes.label Transferred\n";
|
|
|
|
print $_ , "_speed.type GAUGE\n";
|
|
|
|
print $_ , "_bytes.type DERIVE\n";
|
|
|
|
print $_ , "_speed.min 0\n";
|
|
|
|
print $_ , "_bytes.min 0\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
print "rx_speed.graph no\n";
|
|
|
|
print "rx_bytes.graph no\n";
|
|
|
|
print "tx_speed.negative rx_speed\n";
|
|
|
|
print "tx_bytes.negative rx_bytes\n";
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub do_config_xfer_by_device {
|
|
|
|
# Provide transfer for each unifi device - CONFIG
|
|
|
|
if ( !$PluginConfig{'enable_xfer_device'} ) { return 0; }
|
|
|
|
|
|
|
|
graph_prologue(
|
|
|
|
'unifi_xfer_per_device',
|
|
|
|
'Transfer / device',
|
|
|
|
'--base 1000',
|
|
|
|
'Bytes/${graph_period} rcvd (-) / trans (+)',
|
|
|
|
'network',
|
|
|
|
'Bytes sent and received per unifi device'
|
|
|
|
);
|
|
|
|
|
|
|
|
foreach my $thisDevice ( sort keys %{$Data{'device'}} ) {
|
|
|
|
foreach ( "_rxbytes", "_txbytes" ) {
|
|
|
|
print $thisDevice , $_ , ".label " , $Data{'device'}{$thisDevice}->{'label'} , "\n";
|
|
|
|
print $thisDevice , $_ , ".type DERIVE\n";
|
|
|
|
print $thisDevice , $_ , ".min 0\n";
|
|
|
|
}
|
|
|
|
print $thisDevice , "_rxbytes.graph no\n";
|
|
|
|
print $thisDevice , "_txbytes.negative " , $thisDevice , "_rxbytes\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( $PluginConfig{'enable_detail_xfer_device'} ) {
|
|
|
|
foreach my $thisDevice ( sort keys %{$Data{'device'}} ) {
|
|
|
|
graph_prologue(
|
|
|
|
'unifi_xfer_per_device.' . $thisDevice,
|
|
|
|
'Transfer / device : ' . $Data{'device'}{$thisDevice}->{'label'},
|
|
|
|
'--base 1000',
|
|
|
|
'Bytes/${graph_period} rcvd (-) / trans (+)',
|
|
|
|
'network',
|
|
|
|
'Bytes sent and received on the unifi device named: ' . $Data{'device'}{$thisDevice}->{'label'}
|
|
|
|
);
|
|
|
|
foreach ( "rxbyte", "txbyte" ) {
|
|
|
|
print $_ , ".label Bytes\n";
|
|
|
|
print $_ , ".type DERIVE\n";
|
|
|
|
print $_ , ".min 0\n";
|
|
|
|
}
|
|
|
|
print "rxbyte.graph no\n";
|
|
|
|
print "txbyte.negative rxbyte\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub do_config_uptime {
|
|
|
|
# Provide device uptime for each unifi device - CONFIG
|
|
|
|
if ( !$PluginConfig{'enable_device_uptime'} ) { return 0; }
|
|
|
|
graph_prologue(
|
|
|
|
'unifi_device_uptime',
|
|
|
|
'Uptime',
|
|
|
|
'--base 1000 -r --lower-limit 0',
|
|
|
|
'days',
|
|
|
|
'system',
|
|
|
|
'Uptime in days for each unifi device'
|
|
|
|
);
|
|
|
|
|
|
|
|
foreach ( sort keys %{$Data{'device'}} ) {
|
|
|
|
print $_ , ".label " , $Data{'device'}{$_}->{"name"} , "\n";
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub do_config_cpu {
|
|
|
|
# Provide device CPU usage for each unifi device - CONFIG
|
|
|
|
if ( !$PluginConfig{'enable_device_cpu'} ) { return 0; }
|
|
|
|
graph_prologue(
|
|
|
|
'unifi_device_cpu',
|
|
|
|
'CPU Usage',
|
|
|
|
'--base 1000 -r --lower-limit 0 --upper-limit 100',
|
|
|
|
'%',
|
|
|
|
'system',
|
|
|
|
'CPU usage as a percentage for each unifi device'
|
|
|
|
);
|
|
|
|
|
|
|
|
foreach ( sort keys %{$Data{'device'}} ) {
|
|
|
|
print $_ , ".label " , $Data{'device'}{$_}->{"name"} , "\n";
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub do_config_load {
|
|
|
|
# Provide device load average for each unifi device - CONFIG
|
|
|
|
if ( !$PluginConfig{'enable_device_load'} ) { return 0; }
|
|
|
|
graph_prologue(
|
|
|
|
'unifi_device_load',
|
|
|
|
'Load Average',
|
|
|
|
'-l 0 --base 1000',
|
|
|
|
'load',
|
|
|
|
'system',
|
|
|
|
'Load average for each unifi Access Point or Switch'
|
|
|
|
);
|
|
|
|
|
|
|
|
foreach ( sort keys %{$Data{'device'}} ) {
|
|
|
|
if ( $Data{'device'}{$_}->{'type'} eq 'ugw' ) { next; }
|
|
|
|
print $_ , ".label " , $Data{'device'}{$_}->{"name"} , "\n";
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub do_config_mem {
|
|
|
|
# Provide device memory usage for each unifi device - CONFIG
|
|
|
|
if ( !$PluginConfig{'enable_device_mem'} ) { return 0; }
|
|
|
|
graph_prologue(
|
|
|
|
'unifi_device_mem',
|
|
|
|
'Memory Usage',
|
|
|
|
'--base 1000 -r --lower-limit 0 --upper-limit 100',
|
|
|
|
'%',
|
|
|
|
'system',
|
|
|
|
'Memory usage as a percentage for each unifi device'
|
|
|
|
);
|
|
|
|
|
|
|
|
foreach ( sort keys %{$Data{'device'}} ) {
|
|
|
|
print $_ , ".label " , $Data{'device'}{$_}->{"name"} , "\n";
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#########################
|
|
|
|
# SUBROUTINES VALUES #
|
|
|
|
#########################
|
|
|
|
|
|
|
|
sub do_values_clients_by_type {
|
|
|
|
# Provide client count by type - VALUES
|
|
|
|
if ( !$PluginConfig{'enable_clients_type'} ) { return 0; }
|
|
|
|
|
|
|
|
print "multigraph unifi_clients_per_network\n";
|
|
|
|
|
|
|
|
foreach ( @{$Data{'typesOrder'}} ) {
|
|
|
|
print $_ , ".value " , ( $Data{'types'}{$_}[2] + $Data{'types'}{$_}[3] ) , "\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( ! $PluginConfig{'enable_detail_clients_type'} ) { return 1; }
|
|
|
|
|
|
|
|
foreach ( @{$Data{'typesOrder'}} ) {
|
|
|
|
if ( $Data{'types'}{$_}[1] == 1 ) {
|
|
|
|
print "multigraph unifi_clients_per_network.$_\n";
|
|
|
|
print "users.value " , $Data{'types'}{$_}[2] , "\n";
|
|
|
|
print "guests.value " , $Data{'types'}{$_}[3] , "\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub do_values_clients_by_device {
|
|
|
|
# Provide client count by device - VALUES
|
|
|
|
if ( !$PluginConfig{'enable_clients_device'} ) { return 0; }
|
|
|
|
|
|
|
|
print "multigraph unifi_clients_per_device\n";
|
|
|
|
|
|
|
|
foreach ( sort keys %{$Data{'device'}} ) {
|
|
|
|
print $_ , ".value " , $Data{'device'}{$_}->{'clients'} , "\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( ! $PluginConfig{'enable_detail_clients_device'} ) { return 1; }
|
|
|
|
|
|
|
|
foreach ( sort keys %{$Data{'device'}} ) {
|
|
|
|
print "multigraph unifi_clients_per_device.$_\n";
|
|
|
|
print "users.value " , $Data{'device'}{$_}->{'users'} , "\n";
|
|
|
|
print "guests.value " , $Data{'device'}{$_}->{'guests'} , "\n";
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub do_values_xfer_by_radio {
|
|
|
|
# Provide transfer for radios - VALUES
|
|
|
|
if ( !$PluginConfig{'enable_xfer_radio'} ) { return 0; }
|
|
|
|
|
|
|
|
print "multigraph unifi_xfer_per_radio\n";
|
|
|
|
|
|
|
|
foreach my $thisDevice ( sort keys %{$Data{'device'}} ) {
|
|
|
|
if ( $Data{'device'}{$thisDevice}->{'type'} ne "uap" ) { next; }
|
|
|
|
|
|
|
|
foreach ( @{$Data{'device'}{$thisDevice}{'radio'}} ) {
|
|
|
|
print $thisDevice , "_" , $_->{"name"} , "_pack.value " , ($_->{"pckt"} // 0), "\n";;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( ! $PluginConfig{'enable_detail_xfer_radio'} ) { return 1; }
|
|
|
|
|
|
|
|
foreach my $thisDevice ( sort keys %{$Data{'device'}} ) {
|
|
|
|
if ( $Data{'device'}{$thisDevice}->{'type'} ne "uap" ) { next; }
|
|
|
|
|
|
|
|
print "multigraph unifi_xfer_per_radio.$thisDevice\n";
|
|
|
|
|
|
|
|
foreach ( @{$Data{'device'}{$thisDevice}{'radio'}} ) {
|
|
|
|
print $_->{"name"} , "_pkt.value " , ($_->{"pckt"} // 0) , "\n";
|
|
|
|
print $_->{"name"} , "_dret.value " , ($_->{"dret"} // 0) , "\n";
|
|
|
|
print $_->{"name"} , "_err.value " , ($_->{"err"} // 0) , "\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub do_values_xfer_by_network {
|
|
|
|
# Provide transfer for named networks - CONFIG
|
|
|
|
if ( !$PluginConfig{'enable_xfer_network'} ) { return 0; }
|
|
|
|
|
|
|
|
print "multigraph unifi_xfer_per_network\n";
|
|
|
|
|
|
|
|
foreach my $thisNet ( sort keys %{$Data{'networks'}} ) {
|
|
|
|
print $thisNet , "_rxbytes.value " , ($Data{'networks'}{$thisNet}->{"rx"} // 0) , "\n";
|
|
|
|
print $thisNet , "_txbytes.value " , ($Data{'networks'}{$thisNet}->{"tx"} // 0) , "\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( ! $PluginConfig{'enable_detail_xfer_network'} ) { return 1; }
|
|
|
|
|
|
|
|
foreach my $thisNet ( sort keys %{$Data{'networks'}} ) {
|
|
|
|
print "multigraph unifi_xfer_per_network.$thisNet\n";
|
|
|
|
print "rxbyte.value " , ($Data{'networks'}{$thisNet}->{"rx"} // 0) , "\n";
|
|
|
|
print "txbyte.value " , ($Data{'networks'}{$thisNet}->{"tx"} // 0) , "\n";
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub do_values_xfer_by_port {
|
|
|
|
# Provide transfer for switch ports - VALUES
|
|
|
|
if ( !$PluginConfig{'enable_xfer_port'} ) { return 0; }
|
|
|
|
|
|
|
|
foreach my $thisDevice ( sort keys %{$Data{'device'}} ) {
|
|
|
|
if ( $Data{'device'}{$thisDevice}->{'type'} ne "usw" ) { next; }
|
|
|
|
print "multigraph unifi_xfer_per_port_$thisDevice\n";
|
|
|
|
|
|
|
|
foreach ( @{$Data{'device'}{$thisDevice}{'ports'}} ) {
|
|
|
|
print $thisDevice , "_" , $_->{"name"} , "_rxbytes.value " , $_->{"rx"} , "\n";
|
|
|
|
print $thisDevice , "_" , $_->{"name"} , "_txbytes.value " , $_->{"tx"} , "\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( ! $PluginConfig{'enable_detail_xfer_port'} ) { return 1; }
|
|
|
|
|
|
|
|
# Extended graphs
|
|
|
|
foreach my $thisDevice ( sort keys %{$Data{'device'}} ) {
|
|
|
|
if ( $Data{'device'}{$thisDevice}->{'type'} ne "usw" ) { next; }
|
|
|
|
foreach ( @{$Data{'device'}{$thisDevice}{'ports'}} ) {
|
|
|
|
print 'multigraph unifi_xfer_per_port_' . $thisDevice . "." . $_->{'name'} . "\n";
|
|
|
|
print "rxbyte.value " , $_->{"rx"} , "\n";
|
|
|
|
print "txbyte.value " , $_->{"tx"} , "\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub do_values_xfer_by_uplink {
|
|
|
|
# Provide transfer for unifi uplink - CONFIG
|
|
|
|
if ( !$PluginConfig{'enable_xfer_device'} ) { return 0; }
|
|
|
|
|
|
|
|
print "multigraph unifi_xfer_by_uplink\n";
|
|
|
|
print "rx_speed.value " . $Data{'uplink'}{"rx_speed"} . "\n";
|
|
|
|
print "tx_speed.value " . $Data{'uplink'}{"tx_speed"} . "\n";
|
|
|
|
print "rx_bytes.value " . $Data{'uplink'}{"rx_bytes"} . "\n";
|
|
|
|
print "tx_bytes.value " . $Data{'uplink'}{"tx_bytes"} . "\n";
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub do_values_xfer_by_device {
|
|
|
|
# Provide transfer for each unifi device - CONFIG
|
|
|
|
if ( !$PluginConfig{'enable_xfer_device'} ) { return 0; }
|
|
|
|
|
|
|
|
print "multigraph unifi_xfer_per_device\n";
|
|
|
|
foreach ( sort keys %{$Data{'device'}} ) {
|
|
|
|
print $_ . "_rxbytes.value " . $Data{'device'}{$_}->{"rx"} , "\n";
|
|
|
|
print $_ . "_txbytes.value " . $Data{'device'}{$_}->{"tx"} , "\n";
|
|
|
|
}
|
|
|
|
if ( $PluginConfig{'enable_detail_xfer_device'} ) {
|
|
|
|
foreach ( sort keys %{$Data{'device'}} ) {
|
|
|
|
print "multigraph unifi_xfer_per_device." , $_ , "\n";
|
|
|
|
print "rxbyte.value " , $Data{'device'}{$_}->{"rx"} , "\n";
|
|
|
|
print "txbyte.value " , $Data{'device'}{$_}->{"tx"} , "\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub do_values_cpu {
|
|
|
|
# Provide device CPU usage for each unifi device - VALUES
|
|
|
|
if ( !$PluginConfig{'enable_device_cpu'} ) { return 0; }
|
|
|
|
|
|
|
|
print "multigraph unifi_device_cpu\n";
|
|
|
|
foreach ( sort keys %{$Data{'device'}} ) {
|
|
|
|
print $_ , ".value " , ( $Data{'device'}{$_}->{"cpu"} ) , "\n";
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub do_values_mem {
|
|
|
|
# Provide device memory usage for each unifi device - VALUES
|
|
|
|
if ( !$PluginConfig{'enable_device_mem'} ) { return 0; }
|
|
|
|
|
|
|
|
print "multigraph unifi_device_mem\n";
|
|
|
|
foreach ( sort keys %{$Data{'device'}} ) {
|
|
|
|
print $_ , ".value " , ( $Data{'device'}{$_}->{"mem"} ) , "\n";
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub do_values_load {
|
|
|
|
# Provide device load average for each unifi device - VALUES
|
|
|
|
if ( !$PluginConfig{'enable_device_load'} ) { return 0; }
|
|
|
|
|
|
|
|
print "multigraph unifi_device_load\n";
|
|
|
|
foreach ( sort keys %{$Data{'device'}} ) {
|
|
|
|
if ( $Data{'device'}{$_}->{'type'} eq 'ugw' ) { next; }
|
|
|
|
print $_ , ".value " , ( $Data{'device'}{$_}->{"load"} ) , "\n";
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub do_values_uptime {
|
|
|
|
# Provide device uptime for each unifi device - VALUES
|
|
|
|
if ( !$PluginConfig{'enable_device_uptime'} ) { return 0; }
|
|
|
|
|
|
|
|
print "multigraph unifi_device_uptime\n";
|
|
|
|
foreach ( sort keys %{$Data{'device'}} ) {
|
|
|
|
print $_ , ".value " , ( $Data{'device'}{$_}->{"uptime"} / 86400 ) , "\n";
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#########################
|
|
|
|
# SUBROUTINES GENERAL #
|
|
|
|
#########################
|
|
|
|
|
|
|
|
sub graph_prologue {
|
|
|
|
# Generate graph prologues - slightly less copy-pasta, and less chance for things to go wrong
|
|
|
|
my ( $id, $title, $args, $vlabel, $category, $info ) = (@_);
|
|
|
|
|
|
|
|
print "multigraph $id\n";
|
|
|
|
print 'graph_title ' , $title , ' : ' , $APIconfig{"name"} , "\n";
|
|
|
|
print "graph_args $args\n";
|
|
|
|
print "graph_vlabel $vlabel\n";
|
|
|
|
if ( $PluginConfig{'force_category'} ) {
|
|
|
|
print "graph_category ", $PluginConfig{'force_category'}, "\n";
|
|
|
|
} else {
|
|
|
|
print "graph_category $category\n";
|
|
|
|
}
|
|
|
|
if ( $info ) {
|
|
|
|
print 'graph_info For the unifi site named "' , $APIconfig{"name"} , "\", $info\n";
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
# Collate all collected data into something we can use.
|
|
|
|
sub make_data {
|
|
|
|
foreach my $thisDevice ( @{$APIJsonResponse{'device'}->{'data'}} ) {
|
|
|
|
# Grab everything we care to know about each device.
|
|
|
|
$Data{'device'}{ make_safe($thisDevice->{'name'}, $thisDevice->{'serial'}) } = {
|
|
|
|
'label' => $thisDevice->{'name'},
|
|
|
|
'users' => ($thisDevice->{'user-num_sta'} || 0),
|
|
|
|
'guests' => ($thisDevice->{'guest-num_sta'} || 0),
|
|
|
|
'clients' => ($thisDevice->{'user-num_sta'} + $thisDevice->{'guest-num_sta'} || 0),
|
|
|
|
'tx' => $thisDevice->{'rx_bytes'},
|
|
|
|
'rx' => $thisDevice->{'tx_bytes'},
|
|
|
|
'name' => $thisDevice->{'name'},
|
|
|
|
'uptime' => $thisDevice->{'uptime'},
|
|
|
|
'cpu' => $thisDevice->{'system-stats'}->{'cpu'},
|
|
|
|
'mem' => $thisDevice->{'system-stats'}->{'mem'},
|
|
|
|
'load' => ( $thisDevice->{'type'} eq 'ugw' ? 'U' : $thisDevice->{'sys_stats'}->{'loadavg_1'} ),
|
|
|
|
'type' => $thisDevice->{'type'}
|
|
|
|
};
|
|
|
|
|
|
|
|
if ( $thisDevice->{'type'} eq 'ugw' ) { # Handle firewall specially, record uplink and networks
|
|
|
|
foreach my $thisNet ( @{$thisDevice->{'network_table'}} ) {
|
|
|
|
$Data{'networks'}{ make_safe($thisNet->{'name'}, $thisNet->{'_id'} ) } = {
|
|
|
|
'label' => $thisNet->{'name'},
|
|
|
|
'tx' => $thisNet->{'tx_bytes'},
|
|
|
|
'rx' => $thisNet->{'rx_bytes'}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$Data{'uplink'}{'devName'} = $thisDevice->{'name'};
|
|
|
|
$Data{'uplink'}{'rx_speed'} = $thisDevice->{'speedtest-status'}->{'xput_download'} * 1000000;
|
|
|
|
$Data{'uplink'}{'tx_speed'} = $thisDevice->{'speedtest-status'}->{'xput_upload'} * 1000000;
|
|
|
|
|
|
|
|
foreach ( @{$thisDevice->{"port_table"}} ) {
|
|
|
|
if ( $_->{name} eq "wan" ) {
|
|
|
|
$Data{'uplink'}{'rx_bytes'} = $_->{'rx_bytes'};
|
|
|
|
$Data{'uplink'}{'tx_bytes'} = $_->{'tx_bytes'};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( $thisDevice->{'type'} eq 'usw' ) { # Handle swiches specially - record port stats
|
|
|
|
my @port_list;
|
|
|
|
|
|
|
|
foreach my $port ( @{$thisDevice->{'port_table'}} ) {
|
|
|
|
if ( !$PluginConfig{'hide_empty_xfer_port'} || $port->{'up'} ) {
|
|
|
|
push @port_list , {
|
|
|
|
'name' => 'port_' . zPad($port->{'port_idx'}),
|
|
|
|
'label' => zPad($port->{'port_idx'}) . '-' . $port->{'name'},
|
|
|
|
'rx' => $port->{'rx_bytes'},
|
|
|
|
'tx' => $port->{'tx_bytes'}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$Data{'device'}{ make_safe($thisDevice->{'name'}, $thisDevice->{'serial'}) }{'ports'} = \@port_list;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( $thisDevice->{'type'} eq 'uap' ) { # Handle APS specially - record radio stats
|
|
|
|
my @theseRadios;
|
|
|
|
|
|
|
|
foreach my $thisRadio ( @{$thisDevice->{'radio_table_stats'}} ) {
|
|
|
|
my $name = make_safe( $thisRadio->{'name'}, "" );
|
|
|
|
my $label = ( $thisRadio->{'channel'} < 12 ) ? '2.4Ghz' : '5Ghz';
|
|
|
|
|
|
|
|
$_ = $thisDevice->{'stat'}->{'ap'};
|
|
|
|
|
|
|
|
push @theseRadios, {
|
|
|
|
'name' => $name,
|
|
|
|
'label' => $label . '-' . $thisDevice->{'name'},
|
|
|
|
'pckt' => ($_->{$name . '-rx_packets'} // 0) + ($_->{$name . '-tx_packets'} // 0),
|
|
|
|
'dret' => ($_->{$name . '-rx_dropped'} // 0) + ($_->{$name . '-tx_retries'} // 0) + ($_->{$name . '-tx_dropped'} // 0),
|
|
|
|
'err' => ($_->{$name . '-rx_errors'} // 0) + ($_->{$name . '-tx_errors'} // 0),
|
|
|
|
'type' => $label
|
|
|
|
};
|
|
|
|
}
|
|
|
|
$Data{'device'}{ make_safe($thisDevice->{'name'}, $thisDevice->{'serial'}) }{'radio'} = \@theseRadios;
|
|
|
|
}
|
|
|
|
} # END PROCESSING OF DEVICE DATA
|
|
|
|
|
|
|
|
|
|
|
|
# PROCESS NETWORK TYPE DATA
|
|
|
|
|
|
|
|
# -> UNLESS, type graph is disabled.
|
|
|
|
#
|
|
|
|
# WHY: if the client list is large (huge. 10,000+), this is CPU intensive
|
|
|
|
if ( !$PluginConfig{'enable_clients_type'} ) { return 1; }
|
|
|
|
|
|
|
|
$Data{'types'} = {
|
|
|
|
"wired" => ["Wired Connection", 1, 0, 0],
|
|
|
|
"wifi" => ["Wireless Connection", 1, 0, 0],
|
|
|
|
"tuser" => ["Total Users", 0, 0, 0],
|
|
|
|
"tguest" => ["Total Guests", 0, 0, 0],
|
|
|
|
"authed" => ["Authorized Guests", 0, 0, 0],
|
|
|
|
"unauth" => ["Unauthorized Guests", 0, 0, 0],
|
|
|
|
};
|
|
|
|
|
|
|
|
$Data{'typesOrder'} = ( $PluginConfig{'show_authorized_clients_type'} ) ?
|
|
|
|
[ "wired", "wifi", "tuser", "tguest", "authed", "unauth"] :
|
|
|
|
[ "wired", "wifi", "tuser", "tguest" ];
|
|
|
|
|
|
|
|
|
|
|
|
my @wlans;
|
|
|
|
|
|
|
|
foreach my $thisNet ( @{$APIJsonResponse{'wlan'}->{'data'}} ) {
|
|
|
|
$Data{'types'}{ make_safe($thisNet->{'name'}, "") } = [ $thisNet->{'name'}, 1, 0, 0 ];
|
|
|
|
push @wlans, make_safe($thisNet->{'name'}, "");
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ( sort @wlans ) {
|
|
|
|
push @{$Data{'typesOrder'}}, $_;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach my $client ( @{$APIJsonResponse{'sta'}->{'data'}} ) {
|
|
|
|
if ( $client->{"is_wired"} ) {
|
|
|
|
if ( $client->{"is_guest"} ) {
|
|
|
|
$Data{'types'}->{'wired'}[3]++;
|
|
|
|
$Data{'types'}->{'guest'}[3]++;
|
|
|
|
} else {
|
|
|
|
$Data{'types'}->{'wired'}[2]++;
|
|
|
|
$Data{'types'}->{'user'}[2]++;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if ( $client->{"is_guest"} ) {
|
|
|
|
$Data{'types'}->{make_safe($client->{"essid"}, "")}[3]++;
|
|
|
|
$Data{'types'}->{'wifi'}[3]++;
|
|
|
|
$Data{'types'}->{'guest'}[3]++;
|
|
|
|
if ( $client->{"authorized"} ) {
|
|
|
|
$Data{'types'}->{'authed'}[3]++;
|
|
|
|
} else {
|
|
|
|
$Data{'types'}->{'unauth'}[3]++;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$Data{'types'}->{make_safe($client->{"essid"}, "")}[2]++;
|
|
|
|
$Data{'types'}->{'wifi'}[2]++;
|
|
|
|
$Data{'types'}->{'user'}[2]++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
sub fetch_data {
|
|
|
|
# Set up curl, and login to API
|
|
|
|
$curl->setopt($curl->CURLOPT_POST,1);
|
|
|
|
$curl->setopt($curl->CURLOPT_COOKIEFILE,""); # Session only cookie
|
|
|
|
$curl->setopt($curl->CURLOPT_SSL_VERIFYPEER, (( $APIconfig{"ssl_verify_peer"} =~ m/no/i ) ? 0 : 1) );
|
|
|
|
$curl->setopt($curl->CURLOPT_SSL_VERIFYHOST, (( $APIconfig{"ssl_verify_host"} =~ m/no/i ) ? 0 : 2) );
|
|
|
|
$curl->setopt($curl->CURL_SSLVERSION_TLSv1, 1);
|
|
|
|
$curl->setopt($curl->CURLOPT_URL, $APIPoint{'login'});
|
|
|
|
$curl->setopt($curl->CURLOPT_POSTFIELDS, q[{"username":"] . $APIconfig{"user"} . q[", "password":"] . $APIconfig{"pass"} . q["}] );
|
|
|
|
$curl->setopt($curl->CURLOPT_WRITEDATA, \$APIResponse{'login'});
|
|
|
|
$retcode = $curl->perform;
|
|
|
|
|
|
|
|
if ( $retcode != 0 ) {
|
|
|
|
die "FATAL:$me: Unable to connect to API: " . $curl->strerror($retcode) . " " . $curl->errbuf . "\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
$APIJsonResponse{'login'} = $jsonOBJ->allow_nonref->utf8->relaxed->decode($APIResponse{'login'});
|
|
|
|
|
|
|
|
if ( $APIJsonResponse{'login'}->{'meta'}->{'rc'} ne 'ok' ) {
|
|
|
|
die "FATAL:$me: Unable to login to API - it said: " , $APIJsonResponse{'login'}->{'meta'}->{'msg'} , "\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
# Change method to GET
|
|
|
|
$curl->setopt($curl->CURLOPT_HTTPGET,1);
|
|
|
|
|
|
|
|
|
|
|
|
# Get some API data.
|
|
|
|
|
|
|
|
# Device data
|
|
|
|
$curl->setopt($curl->CURLOPT_WRITEDATA, \$APIResponse{'device'});
|
|
|
|
$curl->setopt($curl->CURLOPT_URL, $APIPoint{'device'});
|
|
|
|
$retcode = $curl->perform;
|
|
|
|
|
|
|
|
if ( $retcode != 0 ) {
|
|
|
|
die "FATAL:$me: Unable to connect to API: " . $curl->strerror($retcode) . " " . $curl->errbuf . "\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
$APIJsonResponse{'device'} = $jsonOBJ->allow_nonref->utf8->relaxed->decode($APIResponse{'device'});
|
|
|
|
|
|
|
|
if ( $APIJsonResponse{'device'}->{'meta'}->{'rc'} ne 'ok' ) {
|
|
|
|
die "FATAL:$me: Unable get device data from API - it said: " , $APIJsonResponse{'device'}->{'meta'}->{'msg'} , "\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
# STA (client) data
|
|
|
|
$curl->setopt($curl->CURLOPT_WRITEDATA, \$APIResponse{'sta'});
|
|
|
|
$curl->setopt($curl->CURLOPT_URL, $APIPoint{'sta'});
|
|
|
|
$retcode = $curl->perform;
|
|
|
|
|
|
|
|
if ( $retcode != 0 ) {
|
|
|
|
die "FATAL:$me: Unable to connect to API: " . $curl->strerror($retcode) . " " . $curl->errbuf . "\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
$APIJsonResponse{'sta'} = $jsonOBJ->allow_nonref->utf8->relaxed->decode($APIResponse{'sta'});
|
|
|
|
|
|
|
|
if ( $APIJsonResponse{'sta'}->{'meta'}->{'rc'} ne 'ok' ) {
|
|
|
|
die "FATAL:$me: Unable get sta data from API - it said: " , $APIJsonResponse{'sta'}->{'meta'}->{'msg'} , "\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
# WLAN data
|
|
|
|
$curl->setopt($curl->CURLOPT_WRITEDATA, \$APIResponse{'wlan'});
|
|
|
|
$curl->setopt($curl->CURLOPT_URL, $APIPoint{'wlan'});
|
|
|
|
$retcode = $curl->perform;
|
|
|
|
|
|
|
|
if ( $retcode != 0 ) {
|
|
|
|
die "FATAL:$me: Unable to connect to API: " . $curl->strerror($retcode) . " " . $curl->errbuf . "\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
$APIJsonResponse{'wlan'} = $jsonOBJ->allow_nonref->utf8->relaxed->decode($APIResponse{'wlan'});
|
|
|
|
|
|
|
|
if ( $APIJsonResponse{'wlan'}->{'meta'}->{'rc'} ne 'ok' ) {
|
|
|
|
die "FATAL:$me: Unable get wlan data from API - it said: " , $APIJsonResponse{'wlan'}->{'meta'}->{'msg'} , "\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# Make field names safe, and lowercase.
|
|
|
|
#
|
|
|
|
# Typically, $extraName should be the MAC address of the unique ID identifier as the unifi
|
|
|
|
# controller software does not enforce that device names or network names are unique.
|
|
|
|
sub make_safe {
|
|
|
|
my ( $name, $extraName ) = ( @_ );
|
|
|
|
if ( $extraName ne "" ) {
|
|
|
|
return clean_fieldname(lc($name) . "_" . $extraName);
|
|
|
|
} else {
|
|
|
|
return lc(clean_fieldname($name));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# Get a default from an environmental variable - return text
|
|
|
|
#
|
|
|
|
# env_default(<variable name>, <default>)
|
|
|
|
sub env_default_text {
|
|
|
|
my ( $env_var, $default ) = (@_);
|
|
|
|
return ( ( defined $ENV{$env_var} ) ? $ENV{$env_var} : $default ),
|
|
|
|
}
|
|
|
|
|
|
|
|
# Get a default from an environmental variable - boolean true
|
|
|
|
#
|
|
|
|
# env_default_bool_true (<variable name>, <default>)
|
|
|
|
sub env_default_bool_true {
|
|
|
|
my $env_var = $_[0];
|
|
|
|
return ( ( defined $ENV{$env_var} && $ENV{$env_var} =~ m/no/i ) ? 0 : 1 );
|
|
|
|
}
|
|
|
|
|
|
|
|
# Quick 2 digit zero pad
|
|
|
|
sub zPad { return sprintf("%02d", $_[0]); }
|