Observium_CE/poller.php

278 lines
9.7 KiB
PHP
Executable File

#!/usr/bin/env php
<?php
/**
* Observium
*
* This file is part of Observium.
*
* @package observium
* @subpackage cli
* @copyright (C) 2006-2013 Adam Armstrong, (C) 2013-2021 Observium Limited
*
*/
chdir(dirname($argv[0]));
// Get options before definitions!
$options = getopt("h:i:m:n:p:dqrMV");
include("includes/sql-config.inc.php");
include("includes/polling/functions.inc.php");
$cli = TRUE;
$poller_start = utime();
// get_versions();
if (isset($options['V'])) {
// Print current version and exit
print_message(OBSERVIUM_PRODUCT." ".OBSERVIUM_VERSION);
if (is_array($options['V'])) { print_versions(); }
exit;
}
if (isset($options['M'])) {
// List available modules and exit
print_message(OBSERVIUM_PRODUCT." ".OBSERVIUM_VERSION);
print_message('Enabled poller modules:');
$m_disabled = array();
foreach ($config['poller_modules'] as $module => $ok) {
if ($ok) {
print_message(' '.$module);
} else {
$m_disabled[] = $module;
}
}
if (count($m_disabled)) {
print_message('Disabled poller modules:');
print_message(' '.implode("\n ", $m_disabled));
}
exit;
}
if (!isset($options['q'])) {
print_cli_banner();
$latest['version'] = get_obs_attrib('latest_ver');
$latest['revision'] = get_obs_attrib('latest_rev');
$latest['date'] = get_obs_attrib('latest_rev_date');
if ($latest['revision'] > OBSERVIUM_REV) {
print_message("%GThere is a newer revision of Observium available!%n", 'color');
print_message("%GVersion %r" . $latest['version']."%G (" . format_unixtime(datetime_to_unixtime($latest['date']), 'jS F Y').") is %r". ($latest['revision']-OBSERVIUM_REV) ."%G revisions ahead.%n\n", 'color');
}
// print_message("%g".OBSERVIUM_PRODUCT." ".OBSERVIUM_VERSION."\n%WPoller%n\n", 'color');
if (OBS_DEBUG) { print_versions(); }
}
$doing_single = FALSE; // Poll single device? mostly from poller wrapper
if ($options['h'] === "odd") {
$options['n'] = "1";
$options['i'] = "2";
} elseif ($options['h'] === "even") {
$options['n'] = "0";
$options['i'] = "2";
} elseif ($options['h'] === "all") {
$where = " ";
$doing = "all";
} elseif ($options['h']) {
$params = array();
if (is_numeric($options['h'])) {
$where = "AND `device_id` = ?";
$doing = $options['h'];
$params[] = $options['h'];
$doing_single = TRUE; // Poll single device! (from wrapper)
} else {
$where = "AND `hostname` LIKE ?";
$doing = $options['h'];
$params[] = str_replace('*', '%', $options['h']);
}
}
if (isset($options['i']) && $options['i'] && isset($options['n'])) {
$where = true; // FIXME
$query = 'SELECT `device_id` FROM (SELECT @rownum :=0) r,
(
SELECT @rownum := @rownum +1 AS rownum, `device_id`
FROM `devices`
WHERE `disabled` = 0 AND `poller_id` = '.$config['poller_id'].'
ORDER BY `device_id` ASC
) temp
WHERE MOD(temp.rownum, '.$options['i'].') = ?;';
$doing = $options['n'] ."/".$options['i'];
$params[] = $options['n'];
//print_vars($query);
//print_vars($params);
}
if (!$where)
{
print_message("%n
USAGE:
$scriptname [-drqV] [-i instances] [-n number] [-m module] [-h device]
EXAMPLE:
-h <device id> | <device hostname wildcard> Poll single device
-h odd Poll odd numbered devices (same as -i 2 -n 0)
-h even Poll even numbered devices (same as -i 2 -n 1)
-h all Poll all devices
-h new Poll all devices that have not had a discovery run before
-i <instances> -n <id/number> Poll as instance <id/number> of <instances>
Instance numbers start at 0. 0-3 for -i 4
Example:
-i 4 -n 0
-i 4 -n 1
-i 4 -n 2
-i 4 -n 3
OPTIONS:
-h Device hostname, id or key odd/even/all/new.
-i Poll instances count.
-n Instance id (number), must start from 0 and to be less than instances count.
-q Quiet output.
-M Show globally enabled/disabled modules and exit.
-V Show version and exit.
DEBUGGING OPTIONS:
-r Do not create or update RRDs
-d Enable debugging output.
-dd More verbose debugging output.
-m Specify module(s) (separated by commas) to be run.
%rInvalid arguments!%n", 'color', FALSE);
exit;
}
if (isset($options['r'])) {
$config['norrd'] = TRUE;
}
$cache['maint'] = cache_alert_maintenance();
rrdtool_pipe_open($rrd_process, $rrd_pipes);
print_cli_heading("%WStarting polling run at ".date("Y-m-d H:i:s"), 0);
$polled_devices = 0;
if (!isset($query)) {
$query = "SELECT `device_id` FROM `devices` WHERE `disabled` = 0 $where AND `poller_id` = ? ORDER BY `device_id` ASC";
$params[] = $config['poller_id'];
}
foreach (dbFetch($query, $params) as $device) {
$device = dbFetchRow("SELECT * FROM `devices` WHERE `device_id` = ?", array($device['device_id']));
poll_device($device, $options);
$polled_devices++;
}
$poller_end = utime(); $poller_run = $poller_end - $poller_start; $poller_time = substr($poller_run, 0, 5);
if ($polled_devices) {
if (is_numeric($doing)) { $doing = $device['hostname']; } // Single device ID convert to hostname for log
} else {
print_warning("WARNING: 0 devices polled. Did you specify a device that does not exist?");
}
$string = OBS_SCRIPT_NAME . ": $doing - $polled_devices devices polled in $poller_time secs";
print_debug($string);
print_cli_heading("%WCompleted polling run at ".date("Y-m-d H:i:s"), 0);
// Total MySQL usage
$mysql_time = 0;
$mysql_count = 0;
$mysql_times = [];
foreach($db_stats as $cmd => $count) {
if (isset($db_stats[$cmd.'_sec'])) {
$mysql_times[] = ucfirst(str_replace("fetch", "", $cmd))."[".$count."/".round($db_stats[$cmd.'_sec'],3)."s]";
$mysql_time += $db_stats[$cmd.'_sec'];
$mysql_count += $db_stats[$cmd];
}
}
$db_stats['total'] = $mysql_count;
$db_stats['total_sec'] = $mysql_time;
print_debug_vars($db_stats);
// Store MySQL/Memory stats per device polling (only for single device poll)
if ($doing_single && !isset($options['m'])) {
rrdtool_update_ng($device, 'perf-pollerdb', $db_stats); // MySQL usage stats
rrdtool_update_ng($device, 'perf-pollermemory', array('usage' => memory_get_usage(TRUE), // Memory usage stats
'peak' => memory_get_peak_usage(TRUE)));
print_debug_vars($GLOBALS['snmp_stats']);
$poller_snmp_stats = array('total' => 0, 'total_sec' => 0);
$poller_snmp_errors = array('total' => 0, 'total_sec' => 0);
foreach ($GLOBALS['snmp_stats'] as $snmp_cmd => $entry) {
if ($snmp_cmd === 'errors') { continue; }
$poller_snmp_stats[$snmp_cmd] = $entry['count']; // Count
$poller_snmp_stats[$snmp_cmd.'_sec'] = $entry['time']; // Runtime
$poller_snmp_stats['total'] += $entry['count'];
$poller_snmp_stats['total_sec'] += $entry['time'];
}
foreach ($GLOBALS['snmp_stats']['errors'] as $snmp_cmd => $entry) {
$poller_snmp_errors[$snmp_cmd] = $entry['count']; // Count
$poller_snmp_errors[$snmp_cmd.'_sec'] = $entry['time']; // Runtime
$poller_snmp_errors['total'] += $entry['count'];
$poller_snmp_errors['total_sec'] += $entry['time'];
}
rrdtool_update_ng($device, 'perf-pollersnmp', $poller_snmp_stats); // SNMP walk stats
rrdtool_update_ng($device, 'perf-pollersnmp_errors', $poller_snmp_errors); // SNMP error stats
// FIXME. RRDTool usage
}
if (!isset($options['q'])) {
if ($config['snmp']['hide_auth']) {
print_debug("NOTE, \$config['snmp']['hide_auth'] is set to TRUE, snmp community and snmp v3 auth hidden from debug output.");
}
print_cli_data('Devices Polled', $polled_devices, 0);
print_cli_data('Poller Time', $poller_time ." secs", 0);
print_cli_data('Memory usage', formatStorage(memory_get_usage(TRUE), 2, 4).' (peak: '.formatStorage(memory_get_peak_usage(TRUE), 2, 4).')', 0);
print_cli_data('MySQL Usage', implode(" ", $mysql_times) . ' ('.round($mysql_time, 3).'s '.round($mysql_time/$poller_time*100, 3).'%)', 0);
$rrd_time = 0;
$rrd_times = [];
foreach($GLOBALS['rrdtool'] as $cmd => $data) {
$rrd_times[] = $cmd."[".$data['count']."/".round($data['time'],3)."s]";
$rrd_time += $data['time'];
}
print_cli_data('RRDTool Usage', implode(" ", $rrd_times) . ' ('.round($rrd_time, 3).'s '.round($rrd_time/$poller_time*100, 3).'%)', 0);
$snmp_time = 0;
$snmp_times = [];
foreach($GLOBALS['snmp_stats'] as $cmd => $data) {
$snmp_times[] = $cmd."[".$data['count']."/".round($data['time'],3)."s]";
$snmp_time += $data['time'];
}
print_cli_data('SNMP Usage', implode(" ", $snmp_times) . ' ('.round($snmp_time, 3).'s '.round($snmp_time/$poller_time*100, 3).'%)', 0);
if ($GLOBALS['influxdb_stats']) {
$s = $GLOBALS['influxdb_stats'];
$t = $s[ 'time' ];
print_cli_data('InfluxDB Usage', $s[ 'count' ] . ' data points ('.round($t, 3).'s '.round($t/$poller_time*100, 3).'%)', 0);
}
}
logfile($string);
rrdtool_pipe_close($rrd_process, $rrd_pipes);
unset($config); // Remove this for testing
#print_vars(get_defined_vars());
echo("\n");
print_debug_vars($snmp_stats);
// EOF