1 Star 0 Fork 0

wlnmp/cacti

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
克隆/下载
poller.php 31.79 KB
一键复制 编辑 原始数据 按行查看 历史
cigamit 提交于 2019-12-01 19:23 . Log the automatic enablement of boost
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993
#!/usr/bin/php -q
<?php
/*
+-------------------------------------------------------------------------+
| Copyright (C) 2004-2019 The Cacti Group |
| |
| 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 2 |
| of the License, or (at your option) 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. |
+-------------------------------------------------------------------------+
| Cacti: The Complete RRDtool-based Graphing Solution |
+-------------------------------------------------------------------------+
| This code is designed, written, and maintained by the Cacti Group. See |
| about.php and/or the AUTHORS file for specific developer information. |
+-------------------------------------------------------------------------+
| http://www.cacti.net/ |
+-------------------------------------------------------------------------+
*/
/* tick use required as of PHP 4.3.0 to accomodate signal handling */
declare(ticks = 1);
require(__DIR__ . '/include/cli_check.php');
require_once($config['base_path'] . '/lib/poller.php');
require_once($config['base_path'] . '/lib/data_query.php');
require_once($config['base_path'] . '/lib/rrd.php');
require_once($config['base_path'] . '/lib/dsstats.php');
require_once($config['base_path'] . '/lib/dsdebug.php');
require_once($config['base_path'] . '/lib/boost.php');
require_once($config['base_path'] . '/lib/reports.php');
global $poller_db_cnn_id, $remote_db_cnn_id;
if ($config['poller_id'] > 1 && $config['connection'] == 'online') {
$poller_db_cnn_id = $remote_db_cnn_id;
} else {
$poller_db_cnn_id = false;
}
function sig_handler($signo) {
global $poller_db_cnn_id;
switch ($signo) {
case SIGTERM:
case SIGINT:
cacti_log('WARNING: Cacti Master Poller process terminated by user', true, 'POLLER', POLLER_VERBOSITY_LOW);
$running_processes = db_fetch_assoc('SELECT ' . SQL_NO_CACHE . ' *
FROM poller_time
WHERE end_time=\'0000-00-00 00:00:00\'', true, $poller_db_cnn_id);
if (cacti_sizeof($running_processes)) {
foreach($running_processes as $process) {
if (function_exists('posix_kill')) {
cacti_log("WARNING: Termination poller process with pid '" . $process['pid'] . "'", true, 'POLLER', POLLER_VERBOSITY_LOW);
posix_kill($process['pid'], SIGTERM);
}
}
}
db_execute('TRUNCATE TABLE poller_time', true, $poller_db_cnn_id);
exit;
break;
default:
// ignore all other signals
}
}
// initialize some variables
$force = false;
$debug = false;
$mibs = false;
// set the poller_id
$poller_id = $config['poller_id'];
$hostname = php_uname('n');
// requires for remote poller stage out
chdir(dirname(__FILE__));
// process calling arguments
$parms = $_SERVER['argv'];
array_shift($parms);
if (cacti_sizeof($parms)) {
foreach($parms as $parameter) {
if (strpos($parameter, '=')) {
list($arg, $value) = explode('=', $parameter);
} else {
$arg = $parameter;
$value = '';
}
switch ($arg) {
case '-p':
case '--poller':
$poller_id = $value;
break;
case '-d':
case '--debug':
$debug = true;
break;
case '--force':
$force = true;
break;
case '--version':
case '-V':
case '-v':
display_version();
exit;
case '-H':
case '-h':
case '--help':
display_help();
exit;
default:
print "ERROR: Invalid Argument: ($arg)\n\n";
display_help();
exit(1);
}
}
}
$phostname = db_fetch_cell_prepared('SELECT hostname
FROM poller
WHERE id = ?',
array($poller_id), '', true, $poller_db_cnn_id);
// update the pollers hostname if it is blank, otherwise allow the user to edit it
if ($phostname == '' || $phostname == 'localhost' || $phostname == '127.0.0.1') {
db_execute_prepared('UPDATE poller
SET hostname = ?
WHERE id = ?',
array($hostname, $poller_id), true, $poller_db_cnn_id);
}
$dbhostname = db_fetch_cell_prepared('SELECT dbhost
FROM poller
WHERE id = ?',
array($poller_id), '', true, $poller_db_cnn_id);
// update the database hostname based upon the entry
if ($dbhostname == '' || $dbhostname == 'localhost' || $dbhostname == '127.0.0.1') {
if ($database_hostname != 'localhost' && $database_hostname != '127.0.0.1') {
$hostname = $database_hostname;
}
db_execute_prepared('UPDATE poller
SET dbhost = ?
WHERE id = ?',
array($hostname, $poller_id), true, $poller_db_cnn_id);
}
// if you have more than one poller, boost must be enabled
$total_pollers = db_fetch_cell('SELECT COUNT(*) FROM poller');
if ($total_pollers > 1) {
set_config_option('boost_rrd_update_system_enable', 'on');
set_config_option('boost_redirect', 'on');
}
// check to see if the poller is disabled
poller_enabled_check($poller_id);
// install signal handlers for UNIX only
if (function_exists('pcntl_signal')) {
pcntl_signal(SIGTERM, 'sig_handler');
pcntl_signal(SIGINT, 'sig_handler');
}
// record the start time
$poller_start = microtime(true);
$overhead_time = 0;
api_plugin_hook('poller_top');
// prime the poller_resource_cache for multiple pollers
update_resource_cache($poller_id);
// get number of polling items from the database
$poller_interval = read_config_option('poller_interval');
// retreive the last time the poller ran
$poller_lastrun = read_config_option('poller_lastrun_' . $poller_id);
// collect the system mibs every 4 hours
if ($poller_lastrun % 14440 < time() % 14440 || empty($poller_lastrun)) {
$mibs = true;
}
// get the poller data
$poller = db_fetch_row_prepared('SELECT *
FROM poller
WHERE id = ?',
array($poller_id), true, $poller_db_cnn_id);
// get the current cron interval from the database
$cron_interval = read_config_option('cron_interval');
if ($cron_interval != 60) {
$cron_interval = 300;
}
// see if the user wishes to use process leveling
$process_leveling = read_config_option('process_leveling');
// retreive the number of concurrent process settings
if (cacti_sizeof($poller)) {
$concurrent_processes = $poller['processes'];
} else {
$concurrent_processes = read_config_option('concurrent_processes');
}
if (!isset($concurrent_processes) || intval($concurrent_processes) < 1) {
$concurrent_processes = 1;
}
// correct for possible poller output not empty occurances
$ds_needing_fixes = db_fetch_assoc_prepared('SELECT local_data_id,
MIN(rrd_next_step) AS next_step,
COUNT(DISTINCT rrd_next_step) AS instances
FROM poller_item
WHERE poller_id = ?
AND rrd_num > 1
GROUP BY local_data_id
HAVING instances > 1',
array($poller_id));
if (cacti_sizeof($ds_needing_fixes)) {
foreach($ds_needing_fixes as $ds) {
db_execute_prepared('UPDATE poller_item
SET rrd_next_step = ?
WHERE local_data_id = ?',
array($ds['next_step'], $ds['local_data_id']));
}
}
// correct for possible poller output not empty occurances
$ds_needing_fixes = db_fetch_assoc_prepared('SELECT local_data_id,
MIN(rrd_next_step) AS next_step,
COUNT(DISTINCT rrd_next_step) AS instances
FROM poller_item
WHERE poller_id = ?
AND rrd_num > 1
GROUP BY local_data_id
HAVING instances > 1',
array($poller_id));
if (sizeof($ds_needing_fixes)) {
foreach($ds_needing_fixes as $ds) {
db_execute_prepared('UPDATE poller_item
SET rrd_next_step = ?
WHERE local_data_id = ?',
array($ds['next_step'], $ds['local_data_id']));
}
}
// assume a scheduled task of either 60 or 300 seconds
if (!empty($poller_interval)) {
$poller_runs = intval($cron_interval / $poller_interval);
$sql_where = 'WHERE rrd_next_step<=0 AND poller_id = ' . $poller_id;
define('MAX_POLLER_RUNTIME', $poller_runs * $poller_interval - 2);
} else {
$sql_where = 'WHERE poller_id = ' . $poller_id;
$poller_runs = 1;
$poller_interval = 300;
define('MAX_POLLER_RUNTIME', 298);
}
$num_polling_items = db_fetch_cell('SELECT ' . SQL_NO_CACHE . ' COUNT(*)
FROM poller_item ' . $sql_where);
if (isset($concurrent_processes) && $concurrent_processes > 1) {
$items_perhost = array_rekey(db_fetch_assoc("SELECT " . SQL_NO_CACHE . " host_id,
COUNT(*) AS data_sources
FROM poller_item
$sql_where
GROUP BY host_id
ORDER BY host_id"), 'host_id', 'data_sources');
}
if (isset($items_perhost) && cacti_sizeof($items_perhost)) {
$items_per_process = floor($num_polling_items / $concurrent_processes);
if ($items_per_process == 0) {
$process_leveling = 'off';
}
} else {
$process_leveling = 'off';
}
// some text formatting for platform specific vocabulary
if ($config['cacti_server_os'] == 'unix') {
$task_type = 'Cron';
} else {
$task_type = 'Scheduled Task';
}
if ($debug) {
$level = POLLER_VERBOSITY_NONE;
} else {
$level = POLLER_VERBOSITY_MEDIUM;
}
$poller_seconds_sincerun = 'never';
if (isset($poller_lastrun)) {
$poller_seconds_sincerun = $poller_start - $poller_lastrun;
}
cacti_log("NOTE: Poller Int: '$poller_interval', $task_type Int: '$cron_interval', Time Since Last: '" . round($poller_seconds_sincerun,2) . "', Max Runtime '" . MAX_POLLER_RUNTIME. "', Poller Runs: '$poller_runs'", true, 'POLLER', $level);
// our cron can run at either 1 or 5 minute intervals
if ($poller_interval <= 60) {
$min_period = '60';
} else {
$min_period = '300';
}
// get to see if we are polling faster than reported by the settings, if so, exit
if ((isset($poller_lastrun) && isset($poller_interval) && $poller_lastrun > 0) && (!$force)) {
// give the user some flexibility to run a little moe often
if ((($poller_start - $poller_lastrun)*1.3) < MAX_POLLER_RUNTIME) {
cacti_log("NOTE: $task_type is configured to run too often! The Poller Interval is '$poller_interval' seconds, with a minimum $task_type period of '$min_period' seconds, but only " . number_format_i18n($poller_start - $poller_lastrun, 1) . ' seconds have passed since the poller last ran.', true, 'POLLER', $level);
exit;
}
}
/* check to see whether we have the poller interval set lower than
* the poller is actually ran, if so, issue a warning
*/
if ((($poller_start - $poller_lastrun - 5) > MAX_POLLER_RUNTIME) && ($poller_lastrun > 0)) {
cacti_log("WARNING: $task_type is out of sync with the Poller Interval! The Poller Interval is '$poller_interval' seconds, with a maximum of a '$min_period' second $task_type, but " . number_format_i18n($poller_start - $poller_lastrun, 1) . ' seconds have passed since the last poll!', true, 'POLLER');
admin_email(__('Cacti System Warning'), __('WARNING: %s is out of sync with the Poller Interval for poller id %d! The Poller Interval is %d seconds, with a maximum of a %d seconds, but %d seconds have passed since the last poll!', $task_type, $poller_id, $poller_interval, $min_period, number_format_i18n($poller_start - $poller_lastrun, 1)));
}
set_config_option('poller_lastrun_' . $poller_id, (int)$poller_start);
/* let PHP only run 1 second longer than the max runtime,
* plus the poller needs lot's of memory
*/
ini_set('max_execution_time', MAX_POLLER_RUNTIME + 1);
ini_set('memory_limit', '-1');
$poller_runs_completed = 0;
$poller_items_total = 0;
while ($poller_runs_completed < $poller_runs) {
// record the start time for this loop
$loop_start = microtime(true);
if ($poller_id == '1') {
$polling_hosts = array_merge(
array(0 => array('id' => '0')),
db_fetch_assoc_prepared('SELECT ' . SQL_NO_CACHE . ' id
FROM host
WHERE poller_id = ?
AND disabled=""
AND deleted=""
ORDER BY id',
array($poller_id)));
} else {
$polling_hosts = db_fetch_assoc_prepared('SELECT ' . SQL_NO_CACHE . ' id
FROM host
WHERE poller_id = ?
AND disabled=""
AND deleted=""
ORDER BY id',
array($poller_id));
}
$script = $server = $snmp = 0;
$totals = db_fetch_assoc_prepared('SELECT action, count(*) AS totals
FROM poller_item
WHERE poller_id = ?
GROUP BY action',
array($poller_id));
if (cacti_sizeof($totals)) {
foreach($totals as $value) {
switch($value['action']) {
case '0': // SNMP
$snmp = $value['totals'];
break;
case '1': // Script
$script = $value['totals'];
break;
case '2': // Server
$server = $value['totals'];
break;
}
}
}
// update statistics
db_execute_prepared('INSERT INTO poller (id, snmp, script, server, last_status, status)
VALUES (?, ?, ?, ?, NOW(), 1)
ON DUPLICATE KEY UPDATE snmp=VALUES(snmp), script=VALUES(script),
server=VALUES(server), last_status=VALUES(last_status), status=VALUES(status)',
array($poller_id, $snmp, $script, $server), true, $poller_db_cnn_id);
// calculate overhead time
if ($overhead_time == 0) {
$overhead_time = $loop_start - $poller_start;
}
// initialize counters for script file handling
$host_count = 1;
// initialize file creation flags
$change_proc = false;
// initialize file and host count pointers
$started_processes = 0;
$first_host = 0;
$last_host = 0;
$webroot = addslashes(($config['cacti_server_os'] == 'win32') ? strtr(strtolower(substr(dirname(__FILE__), 0, 1)) . substr(dirname(__FILE__), 1),"\\", '/') : dirname(__FILE__));
// update web paths for the poller
set_config_option('path_webroot', $webroot);
// obtain some defaults from the database
$poller_type = read_config_option('poller_type');
if (cacti_sizeof($poller) && isset($poller['threads'])) {
$max_threads = $poller['threads'];
} else {
$max_threads = read_config_option('max_threads');
}
if (!isset($max_threads) || intval($max_threads) < 1) {
$max_threads = 1;
}
/* initialize poller_time and poller_output tables,
* check poller_output for issues
*/
$running_processes = db_fetch_cell_prepared('SELECT ' . SQL_NO_CACHE . ' COUNT(*)
FROM poller_time
WHERE poller_id = ?
AND end_time="0000-00-00 00:00:00"',
array($poller_id), '', true, $poller_db_cnn_id);
if ($running_processes) {
cacti_log("WARNING: There are $running_processes processes detected as overrunning a polling cycle, please investigate", true, 'POLLER');
admin_email(__('Cacti System Warning'), __('WARNING: There are %d processes detected as overrunning a polling cycle for poller id %d, please investigate.', $running_processes, $poller_id));
}
db_execute_prepared('DELETE FROM poller_time
WHERE poller_id = ?',
array($poller_id), true, $poller_db_cnn_id);
/* only report issues for the main poller or from bad local
* data ids, other pollers may insert somewhat asynchornously
*/
$issues_limit = 20;
if ($poller_id == 1) {
$issues = db_fetch_assoc_prepared('SELECT ' . SQL_NO_CACHE . ' local_data_id, rrd_name
FROM poller_output AS po
LEFT JOIN data_local AS dl
ON po.local_data_id=dl.id
LEFT JOIN host AS h
ON dl.host_id=h.id
WHERE h.poller_id = ? OR h.id IS NULL
LIMIT ' . $issues_limit,
array($poller_id));
} elseif ($config['connection'] == 'online') {
$issues = db_fetch_assoc_prepared('SELECT ' . SQL_NO_CACHE . ' local_data_id, rrd_name
FROM poller_output AS po
LEFT JOIN data_local AS dl
ON po.local_data_id=dl.id
LEFT JOIN host AS h
ON dl.host_id=h.id
WHERE (h.poller_id = ? OR h.id IS NULL)
AND time < FROM_UNIXTIME(UNIX_TIMESTAMP()-600)
LIMIT ' . $issues_limit,
array($poller_id));
} else{
$issues = array();
}
if (cacti_sizeof($issues)) {
$count = db_fetch_cell_prepared('SELECT ' . SQL_NO_CACHE . ' COUNT(*)
FROM poller_output AS po
LEFT JOIN data_local AS dl
ON po.local_data_id=dl.id
LEFT JOIN host AS h
ON dl.host_id=h.id
WHERE h.poller_id = ? OR h.id IS NULL',
array($poller_id));
if (cacti_sizeof($issues)) {
$issue_list = 'DS[';
$i = 0;
foreach($issues as $issue) {
$issue_list .= ($i > 0 ? ', ' : '') . $issue['local_data_id'];
$i++;
}
$issue_list .= ']';
}
if ($count > $issues_limit) {
$issue_list .= ", Additional Issues Remain. Only showing first $issues_limit";
}
cacti_log("WARNING: Poller Output Table not Empty. Issues: $count, $issue_list", true, 'POLLER');
admin_email(__('Cacti System Warning'), __('WARNING: Poller Output Table not empty for poller id %d. Issues: %d, %s.', $poller_id, $count, $issue_list));
db_execute_prepared('DELETE po
FROM poller_output AS po
LEFT JOIN data_local AS dl
ON po.local_data_id = dl.id
LEFT JOIN host AS h
ON dl.host_id = h.id
WHERE h.poller_id = ?
OR h.id IS NULL',
array($poller_id));
}
// mainline
if (read_config_option('poller_enabled') == 'on') {
// determine the number of hosts to process per file
$hosts_per_process = ceil(($poller_id == '1' ? cacti_sizeof($polling_hosts)-1 : cacti_sizeof($polling_hosts)) / $concurrent_processes );
$items_launched = 0;
// exit poller if spine is selected and file does not exist
if (($poller_type == '2') && (!file_exists(read_config_option('path_spine')))) {
cacti_log('ERROR: The spine path: ' . read_config_option('path_spine') . ' is invalid. Poller can not continue!', true, 'POLLER');
admin_email(__('Cacti System Warning'), __('ERROR: The spine path: %s is invalid for poller id %d. Poller can not continue!', read_config_option('path_spine'), $poller_id));
exit;
}
// determine command name
if ($poller_type == '2') {
$command_string = cacti_escapeshellcmd(read_config_option('path_spine'));
if (read_config_option('path_spine_config') != '' && file_exists(read_config_option('path_spine_config'))) {
$extra_args = ' -C ' . cacti_escapeshellarg(read_config_option('path_spine_config'));
} else {
$extra_args = '';
}
$method = 'spine';
$total_procs = $concurrent_processes * $max_threads;
chdir(dirname(read_config_option('path_spine')));
} elseif ($config['cacti_server_os'] == 'unix') {
$command_string = cacti_escapeshellcmd(read_config_option('path_php_binary'));
$extra_args = '-q ' . cacti_escapeshellarg($config['base_path'] . '/cmd.php');
$method = 'cmd.php';
$total_procs = $concurrent_processes;
} else {
$command_string = cacti_escapeshellcmd(read_config_option('path_php_binary'));
$extra_args = '-q ' . cacti_escapeshellarg(strtolower($config['base_path'] . '/cmd.php'));
$method = 'cmd.php';
$total_procs = $concurrent_processes;
}
if (read_config_option('path_stderrlog') != '' && $config['cacti_server_os'] != 'win32') {
$extra_parms = '>> ' . read_config_option('path_stderrlog') . ' 2>&1';
} else {
$extra_parms = '';
}
$extra_args = api_plugin_hook_function('poller_command_args', $extra_args);
if ($poller_id > 1) {
$extra_args .= ' --mode=' . $config['connection'];
}
/* Populate each execution file with appropriate information */
foreach ($polling_hosts as $item) {
if ($host_count == 1) {
$first_host = $item['id'];
}
if ($process_leveling != 'on') {
if ($host_count == $hosts_per_process) {
$last_host = $item['id'];
$change_proc = true;
}
} else {
if (isset($items_perhost[$item['id']])) {
$items_launched += $items_perhost[$item['id']];
}
if (($items_launched >= $items_per_process) ||
(cacti_sizeof($items_perhost) == $concurrent_processes)) {
$last_host = $item['id'];
/* if this is the dummy entry for externally updated data sources
* that are not related to any host (host id = 0), do NOT change_proc */
$change_proc = ($item['id'] == 0 ? false : true);
$items_launched = 0;
}
}
$host_count ++;
if ($change_proc) {
exec_background($command_string, "$extra_args --poller=$poller_id --first=$first_host --last=$last_host" . ($mibs ? ' --mibs':''), $extra_parms);
usleep(100000);
$host_count = 1;
$change_proc = false;
$first_host = 0;
$last_host = 0;
$started_processes++;
}
}
// launch the last process
if ($host_count > 1) {
$last_host = $item['id'];
exec_background($command_string, "$extra_args --poller=$poller_id --first=$first_host --last=$last_host" . ($mibs ? ' --mibs':''), $extra_parms);
usleep(100000);
$started_processes++;
}
if ($poller_id == 1) {
// insert the current date/time for graphs
set_config_option('date', date('Y-m-d H:i:s'));
// open a pipe to rrdtool for writing
$rrdtool_pipe = rrd_init();
}
if ($poller_type == '1') {
$max_threads = '0';
}
$rrds_processed = 0;
$poller_finishing_dispatched = false;
while (1) {
$finished_processes = db_fetch_cell_prepared('SELECT ' . SQL_NO_CACHE . " count(*)
FROM poller_time
WHERE poller_id = ?
AND end_time >'0000-00-00 00:00:00'",
array($poller_id), '', true, $poller_db_cnn_id);
if ($finished_processes >= $started_processes) {
// all scheduled pollers are finished
if ($poller_finishing_dispatched === false) {
api_plugin_hook('poller_finishing');
$poller_finishing_dispatched = true;
}
if ($poller_id == 1) {
$rrds_processed = $rrds_processed + process_poller_output($rrdtool_pipe, true);
}
log_cacti_stats($loop_start, $method, $concurrent_processes, $max_threads,
($poller_id == '1' ? cacti_sizeof($polling_hosts) - 1 : cacti_sizeof($polling_hosts)), $hosts_per_process, $num_polling_items, $rrds_processed);
break;
} else {
if (read_config_option('log_verbosity') >= POLLER_VERBOSITY_MEDIUM || $debug) {
print 'Waiting on ' . ($started_processes - $finished_processes) . ' of ' . $started_processes . " pollers.\n";
}
$mtb = microtime(true);
if ($poller_id == 1) {
$rrds_processed = $rrds_processed + process_poller_output($rrdtool_pipe);
}
// end the process if the runtime exceeds MAX_POLLER_RUNTIME
if (($poller_start + MAX_POLLER_RUNTIME) < time()) {
cacti_log('Maximum runtime of ' . MAX_POLLER_RUNTIME . ' seconds exceeded. Exiting.', true, 'POLLER');
admin_email(__('Cacti System Warning'), __('Maximum runtime of %d seconds exceeded for poller id %d. Exiting.', MAX_POLLER_RUNTIME, $poller_id));
// generate a snmp notification
snmpagent_poller_exiting();
api_plugin_hook_function('poller_exiting');
log_cacti_stats($loop_start, $method, $concurrent_processes, $max_threads,
($poller_id == '1' ? cacti_sizeof($polling_hosts) - 1 : cacti_sizeof($polling_hosts)), $hosts_per_process, $num_polling_items, $rrds_processed);
break;
} elseif (microtime(true) - $mtb < 1) {
sleep(1);
}
}
}
if ($poller_id == 1) {
rrd_close($rrdtool_pipe);
}
// process poller commands
$commands = db_fetch_cell_prepared('SELECT ' . SQL_NO_CACHE . ' COUNT(*)
FROM poller_command
WHERE poller_id = ?',
array($poller_id), '', true, $poller_db_cnn_id);
if ($commands > 0) {
$command_string = cacti_escapeshellcmd(read_config_option('path_php_binary'));
$extra_args = '-q ' . cacti_escapeshellarg($config['base_path'] . '/poller_commands.php') . ' --poller=' . $poller_id;
exec_background($command_string, $extra_args);
} else {
/* no re-index or Rechache present on this run
* in case, we have more PCOMMANDS than recaching, this has to be moved to poller_commands.php
* but then we'll have to call it each time to make sure, stats are updated */
set_config_option('stats_recache_' . $poller_id, 'Poller:' . $poller_id . ' RecacheTime:0.0 DevicesRecached:0');
}
if ($method == 'spine') {
$webroot = read_config_option('path_webroot');
if (is_dir($webroot)) {
chdir($webroot);
}
}
} else {
cacti_log('NOTE: There are no items in your poller for this polling cycle!', true, 'POLLER', $level);
}
$poller_runs_completed++;
// push records updates to the main poller
poller_push_data_to_main();
// record the start time for this loop
$loop_end = microtime(true);
$loop_time = $loop_end - $loop_start;
if ($loop_time < $poller_interval) {
// sleep the appripriate amount of time
if ($poller_runs_completed < $poller_runs) {
$plugin_start = microtime(true);
// all plugins moved to core
if ($poller_id == 1) {
snmpagent_poller_bottom();
dsstats_poller_bottom();
dsdebug_poller_bottom();
boost_poller_bottom();
api_plugin_hook('poller_bottom');
}
// record the start time for this loop
$loop_end = microtime(true);
$cur_loop_time = $loop_end - $loop_start;
if ($poller_runs_completed == 1) {
$sleep_time = $poller_interval - $cur_loop_time - $overhead_time;
} else {
$sleep_time = $poller_interval - $cur_loop_time;
}
// log some nice debug information
if ($debug) {
print 'Loop Time is: ' . round($loop_time, 2) . "\n";
print 'Sleep Time is: ' . round($sleep_time, 2) . "\n";
print 'Total Time is: ' . round($loop_end - $poller_start, 2) . "\n";
}
$plugin_end = microtime(true);
if ($sleep_time > 0) {
usleep($sleep_time * 1000000);
}
api_plugin_hook('poller_top');
}
} else {
cacti_log('WARNING: Cacti Polling Cycle Exceeded Poller Interval by ' . ($loop_end-$loop_start-$poller_interval) . ' seconds', true, 'POLLER', $level);
}
// flush the boost table if in recovery mode
poller_recovery_flush_boost($poller_id);
}
/* start post data processing */
if ($poller_id == 1) {
multiple_poller_boost_check();
poller_replicate_check();
snmpagent_poller_bottom();
boost_poller_bottom();
dsstats_poller_bottom();
dsdebug_poller_bottom();
reports_poller_bottom();
spikekill_poller_bottom();
automation_poller_bottom();
poller_maintenance();
api_plugin_hook('poller_bottom');
bad_index_check($mibs);
} else {
automation_poller_bottom();
poller_maintenance();
}
function bad_index_check($mibs) {
if ($mibs == true) {
$bad_index_devices = db_fetch_cell('SELECT GROUP_CONCAT(DISTINCT dl.host_id)
FROM data_local dl
LEFT JOIN data_template_data dtd
ON dtd.local_data_id = dl.id
WHERE dl.snmp_query_id > 0
AND dl.snmp_index = ""
AND dtd.active != ""');
if ($bad_index_devices != '') {
$bad_indexes = db_fetch_cell('SELECT COUNT(*)
FROM data_local dl
LEFT JOIN data_template_data dtd
ON dtd.local_data_id = dl.id
WHERE dl.snmp_query_id > 0
AND dl.snmp_index = ""
AND dtd.active != ""');
$devices = explode(',', $bad_index_devices);
$device_str = 'Device[' . implode('], Device[', $devices) . ']';
cacti_log('WARNING: You have ' . cacti_sizeof($devices) . ' Devices with bad SNMP Indexes. Devices: ' . $device_str . ' totalling ' . $bad_indexes . ' Data Sources. Please Either Re-Index, Delete or Disable these Data Sources.', false, 'POLLER');
}
}
}
function poller_replicate_check() {
global $config;
include_once($config['base_path'] . '/lib/poller.php');
$sync_interval = read_config_option('poller_sync_interval');
if ($sync_interval == '') {
$sync_interval = 7200;
}
$pollers = db_fetch_assoc("SELECT id
FROM poller
WHERE id > 1
AND dbhost NOT IN ('localhost', '127.0.0.1', '')
AND disabled = ''
AND sync_interval != 0
AND (last_sync = '0000-00-00 00:00:00' OR requires_sync = 'on'
OR (UNIX_TIMESTAMP()-UNIX_TIMESTAMP(last_sync) >= IFNULL(sync_interval, $sync_interval)))");
foreach($pollers as $poller) {
$command_string = cacti_escapeshellcmd(read_config_option('path_php_binary'));
$extra_args = '-q ' . cacti_escapeshellarg($config['base_path'] . '/cli/poller_replicate.php') . ' --poller=' . $poller['id'];
exec_background($command_string, $extra_args);
}
}
function poller_enabled_check($poller_id) {
global $poller_db_cnn_id;
$disabled = db_fetch_cell_prepared('SELECT disabled
FROM poller
WHERE id = ?',
array($poller_id), '', true, $poller_db_cnn_id);
if ($disabled == 'on') {
db_execute_prepared('UPDATE poller
SET last_status=NOW()
WHERE id = ?',
array($poller_id), true, $poller_db_cnn_id);
cacti_log('WARNING: Poller ' . $poller_id . ' is Disabled, graphing or other activities are running', true, 'SYSTEM');
exit(1);
}
}
function log_cacti_stats($loop_start, $method, $concurrent_processes, $max_threads, $num_hosts,
$hosts_per_process, $num_polling_items, $rrds_processed) {
global $poller_id, $poller_db_cnn_id;
// get the poller data
$poller = db_fetch_row_prepared('SELECT *
FROM poller
WHERE id = ?',
array($poller_id), true, $poller_db_cnn_id);
// take time and log performance data
$loop_end = microtime(true);
$perf_data = array(
round($loop_end-$loop_start,4),
$method,
$concurrent_processes,
$max_threads,
$num_hosts,
$hosts_per_process,
$num_polling_items,
$rrds_processed
);
$cacti_stats = vsprintf('Time:%01.4f Method:%s Processes:%s Threads:%s Hosts:%s HostsPerProcess:%s DataSources:%s RRDsProcessed:%s', $perf_data);
cacti_log('STATS: ' . $cacti_stats , true, 'SYSTEM');
// insert poller stats into the settings table
if ($poller_id > 1) {
set_config_option('stats_poller_' . $poller_id, $cacti_stats);
} else {
set_config_option('stats_poller', $cacti_stats);
}
if (array_key_exists('min_time', $poller)) {
// calculate min/max/average timings
$total_time = $loop_end-$loop_start;
$total_polls = $poller['total_polls'];
if ($total_time < $poller['min_time'] || $poller['min_time'] == '') {
$min_time = $total_time;
} else {
$min_time = $poller['min_time'];
}
if ($total_time > $poller['max_time'] || $poller['max_time'] == '') {
$max_time = $total_time;
} else {
$max_time = $poller['max_time'];
}
$avg_time = (($total_polls * $poller['avg_time']) + $total_time) / ($total_polls + 1);
// insert poller stats into the poller table
db_execute_prepared('INSERT INTO poller
(id, total_time, min_time, max_time, avg_time, total_polls, last_update, last_status, status)
VALUES (?, ?, ?, ?, ?, ?, NOW(), NOW(), 2)
ON DUPLICATE KEY UPDATE
total_time=VALUES(total_time),
min_time=VALUES(min_time),
max_time=VALUES(max_time),
avg_time=VALUES(avg_time),
total_polls=VALUES(total_polls),
last_update=VALUES(last_update),
last_status=VALUES(last_status), status=VALUES(status)',
array($poller_id, round($total_time, 4), $min_time, $max_time, $avg_time, $total_polls + 1), true, $poller_db_cnn_id);
}
// update snmpcache
snmpagent_cacti_stats_update($perf_data);
api_plugin_hook_function('cacti_stats_update', $perf_data);
}
function multiple_poller_boost_check() {
$pollers = db_fetch_cell('SELECT COUNT(*) FROM poller WHERE disabled="" AND id > 1');
if ($pollers > 0 && read_config_option('boost_rrd_update_enable') == '') {
cacti_log('NOTE: A second Cacti data collector has been added. Therfore, enabling boost automatically!', false, 'POLLER');
admin_email(__('Cacti System Notification'), __('NOTE: A second Cacti data collector has been added. Therfore, enabling boost automatically!'));
set_config_option('boost_rrd_update_enable', 'on');
set_config_option('boost_rrd_update_system_enable', 'on');
}
}
/**
* function for bulk spikekill that only runs on the main cacti server
*/
function spikekill_poller_bottom () {
global $config;
include_once($config['base_path'] . '/lib/poller.php');
$command_string = cacti_escapeshellcmd(read_config_option('path_php_binary'));
$extra_args = '-q ' . cacti_escapeshellarg($config['base_path'] . '/poller_spikekill.php');
exec_background($command_string, $extra_args);
}
/* display_version - displays version information */
function display_version() {
$version = get_cacti_version();
print "Cacti Main Poller, Version $version, " . COPYRIGHT_YEARS . "\n";
}
function display_help() {
display_version();
print "\nusage: poller.php [--poller=ID] [--force] [--debug]\n\n";
print "Cacti's main poller. This poller is the launcher of cmd.php, spine, and all other\n";
print "background processes. It is the heart of Cacti's data collection engine.\n\n";
print "Optional:\n";
print " --poller=ID Run as the poller indicated and not the default poller.\n";
print " --force Override poller overrun detection and force a poller run.\n";
print " --debug|-d Output debug information. Similar to cacti's DEBUG logging level.\n\n";
}
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
1
https://gitee.com/whsir/cacti.git
git@gitee.com:whsir/cacti.git
whsir
cacti
cacti
master

搜索帮助

D67c1975 1850385 1daf7b77 1850385