| 
									
										
										
										
											2010-02-24 13:46:12 +00:00
										 |  |  | <?php | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  | /** | 
					
						
							|  |  |  |  * rrdtool.inc.php | 
					
						
							| 
									
										
										
										
											2011-10-01 14:54:06 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-08-26 01:50:29 -05:00
										 |  |  |  * Helper for processing rrdtool requests efficiently | 
					
						
							| 
									
										
										
										
											2012-05-25 11:29:53 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  |  * 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 | 
					
						
							|  |  |  |  * (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. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program.  If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @package    LibreNMS | 
					
						
							|  |  |  |  * @link       http://librenms.org | 
					
						
							| 
									
										
										
										
											2012-05-25 11:29:53 +00:00
										 |  |  |  * @copyright  (C) 2006 - 2012 Adam Armstrong | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  |  * @copyright  2016 Tony Murray | 
					
						
							|  |  |  |  * @author     Tony Murray <murraytony@gmail.com> | 
					
						
							| 
									
										
										
										
											2011-10-01 14:54:06 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-10 14:38:25 -06:00
										 |  |  | use LibreNMS\Config; | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  | use LibreNMS\Exceptions\FileExistsException; | 
					
						
							| 
									
										
										
										
											2016-08-26 01:50:29 -05:00
										 |  |  | use LibreNMS\Proc; | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-01 14:54:06 +00:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Opens up a pipe to RRDTool using handles provided | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  |  * @param bool $dual_process start an additional process that's output should be read after every command | 
					
						
							|  |  |  |  * @return bool the process(s) have been successfully started | 
					
						
							| 
									
										
										
										
											2011-10-01 14:54:06 +00:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  | function rrdtool_initialize($dual_process = true) | 
					
						
							| 
									
										
										
										
											2015-11-16 07:42:51 +10:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-06-23 00:29:12 -05:00
										 |  |  |     global $rrd_sync_process, $rrd_async_process; | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-23 00:29:12 -05:00
										 |  |  |     $command = Config::get('rrdtool') . ' -'; | 
					
						
							| 
									
										
										
										
											2015-07-13 20:10:26 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  |     $descriptor_spec = array( | 
					
						
							|  |  |  |         0 => array('pipe', 'r'), // stdin  is a pipe that the child will read from
 | 
					
						
							|  |  |  |         1 => array('pipe', 'w'), // stdout is a pipe that the child will write to
 | 
					
						
							|  |  |  |         2 => array('pipe', 'w'), // stderr is a pipe that the child will write to
 | 
					
						
							| 
									
										
										
										
											2015-07-13 20:10:26 +02:00
										 |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-23 00:29:12 -05:00
										 |  |  |     $cwd = Config::get('rrd_dir'); | 
					
						
							| 
									
										
										
										
											2015-07-13 20:10:26 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-26 01:50:29 -05:00
										 |  |  |     if (!rrdtool_running($rrd_sync_process)) { | 
					
						
							|  |  |  |         $rrd_sync_process = new Proc($command, $descriptor_spec, $cwd); | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-07-13 20:10:26 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-26 01:50:29 -05:00
										 |  |  |     if ($dual_process && !rrdtool_running($rrd_async_process)) { | 
					
						
							|  |  |  |         $rrd_async_process = new Proc($command, $descriptor_spec, $cwd); | 
					
						
							|  |  |  |         $rrd_async_process->setSynchronous(false); | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-10-01 14:54:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-26 01:50:29 -05:00
										 |  |  |     return rrdtool_running($rrd_sync_process) && ($dual_process ? rrdtool_running($rrd_async_process) : true); | 
					
						
							| 
									
										
										
										
											2010-02-24 13:46:12 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  | /** | 
					
						
							| 
									
										
										
										
											2016-08-26 01:50:29 -05:00
										 |  |  |  * Checks if the variable is a running rrdtool process | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-08-26 01:50:29 -05:00
										 |  |  |  * @param $process | 
					
						
							|  |  |  |  * @return bool | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-08-26 01:50:29 -05:00
										 |  |  | function rrdtool_running(&$process) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return isset($process) && $process instanceof Proc && $process->isRunning(); | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-07-13 20:10:26 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-01 14:54:06 +00:00
										 |  |  | /** | 
					
						
							| 
									
										
										
										
											2016-08-26 01:50:29 -05:00
										 |  |  |  * Close all open rrdtool processes. | 
					
						
							|  |  |  |  * This should be done before exiting a script that has called rrdtool_initilize() | 
					
						
							| 
									
										
										
										
											2011-10-01 14:54:06 +00:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-08-26 01:50:29 -05:00
										 |  |  | function rrdtool_close() | 
					
						
							| 
									
										
										
										
											2015-11-16 07:42:51 +10:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-08-26 01:50:29 -05:00
										 |  |  |     global $rrd_sync_process, $rrd_async_process; | 
					
						
							| 
									
										
										
										
											2016-09-09 01:29:02 -05:00
										 |  |  |     /** @var Proc $rrd_sync_process */ | 
					
						
							|  |  |  |     /** @var Proc $rrd_async_process */ | 
					
						
							| 
									
										
										
										
											2010-07-21 18:29:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-26 01:50:29 -05:00
										 |  |  |     if (rrdtool_running($rrd_sync_process)) { | 
					
						
							| 
									
										
										
										
											2016-09-24 15:52:35 +01:00
										 |  |  |         $rrd_sync_process->close('quit'); | 
					
						
							| 
									
										
										
										
											2016-08-26 01:50:29 -05:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (rrdtool_running($rrd_async_process)) { | 
					
						
							| 
									
										
										
										
											2016-09-24 15:52:35 +01:00
										 |  |  |         $rrd_async_process->close('quit'); | 
					
						
							| 
									
										
										
										
											2016-08-26 01:50:29 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-02-24 13:46:12 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-01 14:54:06 +00:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Generates a graph file at $graph_file using $options | 
					
						
							|  |  |  |  * Opens its own rrdtool pipe. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-07-15 12:21:09 -05:00
										 |  |  |  * @param string $graph_file | 
					
						
							|  |  |  |  * @param string $options | 
					
						
							| 
									
										
										
										
											2011-10-01 14:54:06 +00:00
										 |  |  |  * @return integer | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2015-11-16 07:42:51 +10:00
										 |  |  | function rrdtool_graph($graph_file, $options) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-08-26 01:50:29 -05:00
										 |  |  |     global $debug, $rrd_sync_process; | 
					
						
							| 
									
										
										
										
											2016-09-09 01:29:02 -05:00
										 |  |  |     /** @var Proc $rrd_sync_process */ | 
					
						
							| 
									
										
										
										
											2015-07-13 20:10:26 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  |     if (rrdtool_initialize(false)) { | 
					
						
							| 
									
										
										
										
											2016-08-26 01:50:29 -05:00
										 |  |  |         $cmd = rrdtool_build_command('graph', $graph_file, $options); | 
					
						
							| 
									
										
										
										
											2011-10-01 14:54:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-26 01:50:29 -05:00
										 |  |  |         $output = implode($rrd_sync_process->sendCommand($cmd)); | 
					
						
							| 
									
										
										
										
											2015-07-13 20:10:26 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if ($debug) { | 
					
						
							| 
									
										
										
										
											2016-08-26 01:50:29 -05:00
										 |  |  |             echo "<p>$cmd</p>"; | 
					
						
							|  |  |  |             echo "<p>command returned ($output)</p>"; | 
					
						
							| 
									
										
										
										
											2015-07-13 20:10:26 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-26 01:50:29 -05:00
										 |  |  |         return $output; | 
					
						
							| 
									
										
										
										
											2016-08-09 09:31:26 -05:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2015-07-13 20:10:26 +02:00
										 |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2011-10-01 14:54:06 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-02-24 13:46:12 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-13 20:10:26 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-01 14:54:06 +00:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Generates and pipes a command to rrdtool | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-08-09 09:31:26 -05:00
										 |  |  |  * @internal | 
					
						
							| 
									
										
										
										
											2016-07-15 12:21:09 -05:00
										 |  |  |  * @param string $command create, update, updatev, graph, graphv, dump, restore, fetch, tune, first, last, lastupdate, info, resize, xport, flushcached | 
					
						
							|  |  |  |  * @param string $filename The full patth to the rrd file | 
					
						
							|  |  |  |  * @param string $options rrdtool command options | 
					
						
							| 
									
										
										
										
											2016-07-08 14:35:00 -05:00
										 |  |  |  * @return array the output of stdout and stderr in an array | 
					
						
							| 
									
										
										
										
											2016-08-26 01:50:29 -05:00
										 |  |  |  * @throws FileExistsException thrown when a create command is set to rrdtool < 1.4 and the rrd already exists | 
					
						
							|  |  |  |  * @throws Exception thrown when the rrdtool process(s) cannot be started | 
					
						
							| 
									
										
										
										
											2011-10-01 14:54:06 +00:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  | function rrdtool($command, $filename, $options) | 
					
						
							| 
									
										
										
										
											2015-11-16 07:42:51 +10:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-06-23 00:29:12 -05:00
										 |  |  |     global $debug, $vdebug, $rrd_async_process, $rrd_sync_process; | 
					
						
							| 
									
										
										
										
											2016-09-09 01:29:02 -05:00
										 |  |  |     /** @var Proc $rrd_sync_process */ | 
					
						
							|  |  |  |     /** @var Proc $rrd_async_process */ | 
					
						
							| 
									
										
										
										
											2016-08-16 01:20:49 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-11 13:44:25 -06:00
										 |  |  |     $start_time = microtime(true); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  |     try { | 
					
						
							|  |  |  |         $cmd = rrdtool_build_command($command, $filename, $options); | 
					
						
							|  |  |  |     } catch (FileExistsException $e) { | 
					
						
							|  |  |  |         c_echo('RRD[%g' . $filename . " already exists%n]\n", $debug); | 
					
						
							|  |  |  |         return array(null, null); | 
					
						
							| 
									
										
										
										
											2015-07-13 20:10:26 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-17 16:00:43 -05:00
										 |  |  |     c_echo("RRD[%g$cmd%n]\n", $debug); | 
					
						
							| 
									
										
										
										
											2016-08-15 20:40:57 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-15 12:21:09 -05:00
										 |  |  |     // do not write rrd files, but allow read-only commands
 | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  |     $ro_commands = array('graph', 'graphv', 'dump', 'fetch', 'first', 'last', 'lastupdate', 'info', 'xport'); | 
					
						
							| 
									
										
										
										
											2019-06-23 00:29:12 -05:00
										 |  |  |     if (!empty(Config::get('norrd')) && !in_array($command, $ro_commands)) { | 
					
						
							|  |  |  |         c_echo('[%rRRD Disabled%n]', !Config::get('hide_rrd_disabled')); | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  |         return array(null, null); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // send the command!
 | 
					
						
							| 
									
										
										
										
											2016-08-26 01:50:29 -05:00
										 |  |  |     if ($command == 'last' && rrdtool_initialize(false)) { | 
					
						
							|  |  |  |         // send this to our synchronous process so output is guaranteed
 | 
					
						
							|  |  |  |         $output = $rrd_sync_process->sendCommand($cmd); | 
					
						
							|  |  |  |     } elseif (rrdtool_initialize()) { | 
					
						
							|  |  |  |         // don't care about the return of other commands, so send them to the faster async process
 | 
					
						
							|  |  |  |         $output = $rrd_async_process->sendCommand($cmd); | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2016-08-26 01:50:29 -05:00
										 |  |  |         throw new Exception('rrdtool could not start'); | 
					
						
							| 
									
										
										
										
											2016-07-08 22:58:36 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-07-08 14:35:00 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-15 20:40:57 -05:00
										 |  |  |     if ($vdebug) { | 
					
						
							| 
									
										
										
										
											2016-07-25 20:12:40 -05:00
										 |  |  |         echo 'RRDtool Output: '; | 
					
						
							|  |  |  |         echo $output[0]; | 
					
						
							|  |  |  |         echo $output[1]; | 
					
						
							| 
									
										
										
										
											2015-07-13 20:10:26 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-10-01 14:54:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-11 13:44:25 -06:00
										 |  |  |     recordRrdStatistic($command, $start_time); | 
					
						
							| 
									
										
										
										
											2016-07-08 14:35:00 -05:00
										 |  |  |     return $output; | 
					
						
							| 
									
										
										
										
											2010-02-24 13:46:12 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Build a command for rrdtool | 
					
						
							|  |  |  |  * Shortens the filename as needed | 
					
						
							|  |  |  |  * Determines if --daemon and -O should be used | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @internal | 
					
						
							|  |  |  |  * @param string $command The base rrdtool command.  Usually create, update, last. | 
					
						
							|  |  |  |  * @param string $filename The full path to the rrd file | 
					
						
							|  |  |  |  * @param string $options Options for the command possibly including the rrd definition | 
					
						
							|  |  |  |  * @return string returns a full command ready to be piped to rrdtool | 
					
						
							|  |  |  |  * @throws FileExistsException if rrdtool <1.4.3 and the rrd file exists locally | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | function rrdtool_build_command($command, $filename, $options) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if ($command == 'create') { | 
					
						
							|  |  |  |         // <1.4.3 doesn't support -O, so make sure the file doesn't exist
 | 
					
						
							| 
									
										
										
										
											2018-02-10 14:38:25 -06:00
										 |  |  |         if (version_compare(Config::get('rrdtool_version', '1.4'), '1.4.3', '<')) { | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  |             if (is_file($filename)) { | 
					
						
							|  |  |  |                 throw new FileExistsException(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             $options .= ' -O'; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // no remote for create < 1.5.5 and tune < 1.5
 | 
					
						
							| 
									
										
										
										
											2018-02-10 14:38:25 -06:00
										 |  |  |     $rrdtool_version = Config::get('rrdtool_version', '1.4'); | 
					
						
							| 
									
										
										
										
											2019-06-23 00:29:12 -05:00
										 |  |  |     if (Config::get('rrdcached') && | 
					
						
							| 
									
										
										
										
											2018-02-10 14:38:25 -06:00
										 |  |  |         !($command == 'create' && version_compare($rrdtool_version, '1.5.5', '<')) && | 
					
						
							| 
									
										
										
										
											2019-06-23 00:29:12 -05:00
										 |  |  |         !($command == 'tune' && Config::get('rrdcached') && version_compare($rrdtool_version, '1.5', '<')) | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  |     ) { | 
					
						
							|  |  |  |         // only relative paths if using rrdcached
 | 
					
						
							| 
									
										
										
										
											2019-06-23 00:29:12 -05:00
										 |  |  |         $filename = str_replace([Config::get('rrd_dir') . '/', Config::get('rrd_dir')], '', $filename); | 
					
						
							|  |  |  |         $options = str_replace([Config::get('rrd_dir') . '/', Config::get('rrd_dir')], '', $options); | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-23 00:29:12 -05:00
										 |  |  |         return "$command $filename $options --daemon " . Config::get('rrdcached'); | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return "$command $filename $options"; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-07 01:33:43 -05:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Checks if the rrd file exists on the server | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  |  * This will perform a remote check if using rrdcached and rrdtool >= 1.5 | 
					
						
							| 
									
										
										
										
											2016-07-07 01:33:43 -05:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  |  * @param string $filename full path to the rrd file | 
					
						
							|  |  |  |  * @return bool whether or not the passed rrd file exists | 
					
						
							| 
									
										
										
										
											2016-07-07 01:33:43 -05:00
										 |  |  |  */ | 
					
						
							|  |  |  | function rrdtool_check_rrd_exists($filename) | 
					
						
							| 
									
										
										
										
											2015-11-16 07:42:51 +10:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-06-23 00:29:12 -05:00
										 |  |  |     if (Config::get('rrdcached') && version_compare(Config::get('rrdtool_version', '1.4'), '1.5', '>=')) { | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  |         $chk = rrdtool('last', $filename, ''); | 
					
						
							| 
									
										
										
										
											2019-06-23 00:29:12 -05:00
										 |  |  |         $filename = str_replace([Config::get('rrd_dir') . '/', Config::get('rrd_dir')], '', $filename); | 
					
						
							| 
									
										
										
										
											2016-08-24 16:42:50 -05:00
										 |  |  |         return !str_contains(implode($chk), "$filename': No such file or directory"); | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         return is_file($filename); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-02-24 13:46:12 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-25 11:29:53 +00:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Updates an rrd database at $filename using $options | 
					
						
							|  |  |  |  * Where $options is an array, each entry which is not a number is replaced with "U" | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-07-07 01:33:43 -05:00
										 |  |  |  * @internal | 
					
						
							|  |  |  |  * @param string $filename | 
					
						
							| 
									
										
										
										
											2016-08-09 09:31:26 -05:00
										 |  |  |  * @param array $data | 
					
						
							| 
									
										
										
										
											2016-07-07 01:33:43 -05:00
										 |  |  |  * @return array|string | 
					
						
							| 
									
										
										
										
											2012-05-25 11:29:53 +00:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-08-09 09:31:26 -05:00
										 |  |  | function rrdtool_update($filename, $data) | 
					
						
							| 
									
										
										
										
											2015-11-15 18:18:13 +10:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2015-07-13 20:10:26 +02:00
										 |  |  |     $values = array(); | 
					
						
							| 
									
										
										
										
											2016-08-09 09:31:26 -05:00
										 |  |  |     // Do some sanitation on the data if passed as an array.
 | 
					
						
							| 
									
										
										
										
											2015-08-18 16:26:55 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-09 09:31:26 -05:00
										 |  |  |     if (is_array($data)) { | 
					
						
							| 
									
										
										
										
											2015-07-13 20:10:26 +02:00
										 |  |  |         $values[] = 'N'; | 
					
						
							| 
									
										
										
										
											2016-08-09 09:31:26 -05:00
										 |  |  |         foreach ($data as $v) { | 
					
						
							| 
									
										
										
										
											2015-08-18 16:26:55 +00:00
										 |  |  |             if (!is_numeric($v)) { | 
					
						
							| 
									
										
										
										
											2016-08-09 09:31:26 -05:00
										 |  |  |                 $v = 'U'; | 
					
						
							| 
									
										
										
										
											2015-07-13 20:10:26 +02:00
										 |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-18 16:26:55 +00:00
										 |  |  |             $values[] = $v; | 
					
						
							| 
									
										
										
										
											2015-07-13 20:10:26 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-09 09:31:26 -05:00
										 |  |  |         $data = implode(':', $values); | 
					
						
							|  |  |  |         return rrdtool('update', $filename, $data); | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2015-08-18 16:26:55 +00:00
										 |  |  |         return 'Bad options passed to rrdtool_update'; | 
					
						
							| 
									
										
										
										
											2012-05-09 11:33:16 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-11-15 18:18:13 +10:00
										 |  |  | } // rrdtool_update
 | 
					
						
							| 
									
										
										
										
											2010-07-04 21:09:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-13 20:10:26 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-23 10:37:23 +00:00
										 |  |  | /** | 
					
						
							| 
									
										
										
										
											2016-07-15 12:21:09 -05:00
										 |  |  |  * Escapes strings for RRDtool | 
					
						
							| 
									
										
										
										
											2016-08-09 09:31:26 -05:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-07-15 12:21:09 -05:00
										 |  |  |  * @param string $string the string to escape | 
					
						
							|  |  |  |  * @param integer $length if passed, string will be padded and trimmed to exactly this length (after rrdtool unescapes it) | 
					
						
							| 
									
										
										
										
											2012-05-23 10:37:23 +00:00
										 |  |  |  * @return string | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-08-09 09:31:26 -05:00
										 |  |  | function rrdtool_escape($string, $length = null) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-07-13 20:10:26 +02:00
										 |  |  |     $result = shorten_interface_type($string); | 
					
						
							| 
									
										
										
										
											2015-11-15 11:48:39 +10:00
										 |  |  |     $result = str_replace("'", '', $result);            # remove quotes
 | 
					
						
							| 
									
										
										
										
											2020-01-07 14:06:55 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-15 12:21:09 -05:00
										 |  |  |     if (is_numeric($length)) { | 
					
						
							| 
									
										
										
										
											2020-01-07 14:06:55 -05:00
										 |  |  |         # preserve original $length for str_pad()
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # determine correct strlen() for substr_count()
 | 
					
						
							|  |  |  |         $string_length=strlen($string); | 
					
						
							|  |  |  |         $substr_count_length=$length; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ($length > $string_length) { | 
					
						
							|  |  |  |             $substr_count_length=$string_length; # If $length is greater than the haystack length, then substr_count() will produce a warning; fix warnings.
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $extra = substr_count($string, ':', 0, $substr_count_length); | 
					
						
							| 
									
										
										
										
											2016-07-15 12:21:09 -05:00
										 |  |  |         $result = substr(str_pad($result, $length), 0, ($length + $extra)); | 
					
						
							| 
									
										
										
										
											2015-07-13 20:10:26 +02:00
										 |  |  |         if ($extra > 0) { | 
					
						
							|  |  |  |             $result = substr($result, 0, (-1 * $extra)); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2015-05-11 12:14:56 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-05-23 10:18:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-15 11:48:39 +10:00
										 |  |  |     $result = str_replace(':', '\:', $result);          # escape colons
 | 
					
						
							| 
									
										
										
										
											2020-01-07 14:06:55 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-13 20:10:26 +02:00
										 |  |  |     return $result.' '; | 
					
						
							| 
									
										
										
										
											2015-11-15 18:01:09 +10:00
										 |  |  | } // rrdtool_escape
 | 
					
						
							| 
									
										
										
										
											2015-07-13 20:10:26 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-15 18:01:09 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-07 01:33:43 -05:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Generates a filename based on the hostname (or IP) and some extra items | 
					
						
							| 
									
										
										
										
											2016-08-09 09:31:26 -05:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-07-07 01:33:43 -05:00
										 |  |  |  * @param string $host Host name | 
					
						
							|  |  |  |  * @param array|string $extra Components of RRD filename - will be separated with "-", or a pre-formed rrdname | 
					
						
							|  |  |  |  * @param string $extension File extension (default is .rrd) | 
					
						
							|  |  |  |  * @return string the name of the rrd file for $host's $extra component | 
					
						
							| 
									
										
										
										
											2015-11-15 18:01:09 +10:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-07-07 01:33:43 -05:00
										 |  |  | function rrd_name($host, $extra, $extension = ".rrd") | 
					
						
							| 
									
										
										
										
											2015-11-16 07:42:51 +10:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2015-11-15 18:01:09 +10:00
										 |  |  |     $filename = safename(is_array($extra) ? implode("-", $extra) : $extra); | 
					
						
							| 
									
										
										
										
											2017-09-02 20:45:31 +02:00
										 |  |  |     return implode("/", array(get_rrd_dir($host), $filename.$extension)); | 
					
						
							| 
									
										
										
										
											2015-11-15 18:18:13 +10:00
										 |  |  | } // rrd_name
 | 
					
						
							| 
									
										
										
										
											2015-11-17 02:20:23 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-02 20:45:31 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Generates a path based on the hostname (or IP) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param string $host Host name | 
					
						
							|  |  |  |  * @return string the name of the rrd directory for $host | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | function get_rrd_dir($host) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     $host = str_replace(':', '_', trim($host, '[]')); | 
					
						
							| 
									
										
										
										
											2019-06-23 00:29:12 -05:00
										 |  |  |     return implode("/", [Config::get('rrd_dir'), $host]); | 
					
						
							| 
									
										
										
										
											2017-09-02 20:45:31 +02:00
										 |  |  | } // rrd_dir
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-12 11:05:58 -05:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Generates a filename for a proxmox cluster rrd | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param $pmxcluster | 
					
						
							|  |  |  |  * @param $vmid | 
					
						
							|  |  |  |  * @param $vmport | 
					
						
							|  |  |  |  * @return string full path to the rrd. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-08-26 01:50:29 -05:00
										 |  |  | function proxmox_rrd_name($pmxcluster, $vmid, $vmport) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-06-23 00:29:12 -05:00
										 |  |  |     $pmxcdir = join('/', [Config::get('rrd_dir'), 'proxmox', safename($pmxcluster)]); | 
					
						
							| 
									
										
										
										
											2016-08-12 11:05:58 -05:00
										 |  |  |     // this is not needed for remote rrdcached
 | 
					
						
							|  |  |  |     if (!is_dir($pmxcdir)) { | 
					
						
							|  |  |  |         mkdir($pmxcdir, 0775, true); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return join('/', array($pmxcdir, safename($vmid.'_netif_'.$vmport.'.rrd'))); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-15 12:21:09 -05:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Modify an rrd file's max value and trim the peaks as defined by rrdtool | 
					
						
							| 
									
										
										
										
											2016-08-09 09:31:26 -05:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-07-15 12:21:09 -05:00
										 |  |  |  * @param string $type only 'port' is supported at this time | 
					
						
							|  |  |  |  * @param string $filename the path to the rrd file | 
					
						
							|  |  |  |  * @param integer $max the new max value | 
					
						
							|  |  |  |  * @return bool | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-08-09 09:31:26 -05:00
										 |  |  | function rrdtool_tune($type, $filename, $max) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-12-08 15:17:53 +00:00
										 |  |  |     $fields = array(); | 
					
						
							| 
									
										
										
										
											2015-11-17 02:20:23 -08:00
										 |  |  |     if ($type === 'port') { | 
					
						
							|  |  |  |         if ($max < 10000000) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         $max = $max / 8; | 
					
						
							|  |  |  |         $fields = array( | 
					
						
							| 
									
										
										
										
											2016-08-09 09:31:26 -05:00
										 |  |  |             'INOCTETS', | 
					
						
							|  |  |  |             'OUTOCTETS', | 
					
						
							|  |  |  |             'INERRORS', | 
					
						
							|  |  |  |             'OUTERRORS', | 
					
						
							|  |  |  |             'INUCASTPKTS', | 
					
						
							|  |  |  |             'OUTUCASTPKTS', | 
					
						
							|  |  |  |             'INNUCASTPKTS', | 
					
						
							|  |  |  |             'OUTNUCASTPKTS', | 
					
						
							|  |  |  |             'INDISCARDS', | 
					
						
							|  |  |  |             'OUTDISCARDS', | 
					
						
							|  |  |  |             'INUNKNOWNPROTOS', | 
					
						
							|  |  |  |             'INBROADCASTPKTS', | 
					
						
							|  |  |  |             'OUTBROADCASTPKTS', | 
					
						
							|  |  |  |             'INMULTICASTPKTS', | 
					
						
							|  |  |  |             'OUTMULTICASTPKTS' | 
					
						
							| 
									
										
										
										
											2015-12-08 13:53:38 +00:00
										 |  |  |         ); | 
					
						
							| 
									
										
										
										
											2015-11-17 02:20:23 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-12-08 15:17:53 +00:00
										 |  |  |     if (count($fields) > 0) { | 
					
						
							| 
									
										
										
										
											2016-08-09 09:31:26 -05:00
										 |  |  |         $options = "--maximum " . implode(":$max --maximum ", $fields) . ":$max"; | 
					
						
							| 
									
										
										
										
											2015-12-08 15:17:53 +00:00
										 |  |  |         rrdtool('tune', $filename, $options); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-08-22 10:32:05 -05:00
										 |  |  |     return true; | 
					
						
							| 
									
										
										
										
											2015-11-15 18:01:09 +10:00
										 |  |  | } // rrdtool_tune
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-10 19:31:39 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-07 01:33:43 -05:00
										 |  |  | /** | 
					
						
							| 
									
										
										
										
											2016-01-10 19:31:39 +10:00
										 |  |  |  * rrdtool backend implementation of data_update | 
					
						
							| 
									
										
										
										
											2016-07-07 01:33:43 -05:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-07-15 12:21:09 -05:00
										 |  |  |  * Tags: | 
					
						
							| 
									
										
										
										
											2017-02-23 22:45:50 +00:00
										 |  |  |  *   rrd_def     RrdDefinition | 
					
						
							| 
									
										
										
										
											2016-07-15 12:21:09 -05:00
										 |  |  |  *   rrd_name    array|string: the rrd filename, will be processed with rrd_name() | 
					
						
							|  |  |  |  *   rrd_oldname array|string: old rrd filename to rename, will be processed with rrd_name() | 
					
						
							|  |  |  |  *   rrd_step             int: rrd step, defaults to 300 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param array $device device array | 
					
						
							|  |  |  |  * @param string $measurement the name of this measurement (if no rrd_name tag is given, this will be used to name the file) | 
					
						
							|  |  |  |  * @param array $tags tags to pass additional info to rrdtool | 
					
						
							|  |  |  |  * @param array $fields data values to update | 
					
						
							| 
									
										
										
										
											2015-11-15 18:01:09 +10:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-01-10 19:31:39 +10:00
										 |  |  | function rrdtool_data_update($device, $measurement, $tags, $fields) | 
					
						
							| 
									
										
										
										
											2015-11-16 07:42:51 +10:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-07-07 01:33:43 -05:00
										 |  |  |     $rrd_name = $tags['rrd_name'] ?: $measurement; | 
					
						
							| 
									
										
										
										
											2019-06-23 00:29:12 -05:00
										 |  |  |     $step = $tags['rrd_step'] ?: Config::get('rrd.step'); | 
					
						
							| 
									
										
										
										
											2016-01-12 20:59:02 +10:00
										 |  |  |     $oldname = $tags['rrd_oldname']; | 
					
						
							| 
									
										
										
										
											2016-08-26 01:50:29 -05:00
										 |  |  |     if (!empty($oldname)) { | 
					
						
							| 
									
										
										
										
											2016-01-12 20:59:02 +10:00
										 |  |  |         rrd_file_rename($device, $oldname, $rrd_name); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-01-10 19:31:39 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-12 11:05:58 -05:00
										 |  |  |     if (isset($tags['rrd_proxmox_name'])) { | 
					
						
							|  |  |  |         $pmxvars = $tags['rrd_proxmox_name']; | 
					
						
							|  |  |  |         $rrd = proxmox_rrd_name($pmxvars['pmxcluster'], $pmxvars['vmid'], $pmxvars['vmport']); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         $rrd = rrd_name($device['hostname'], $rrd_name); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 22:45:50 +00:00
										 |  |  |     if (isset($tags['rrd_def']) && !rrdtool_check_rrd_exists($rrd)) { | 
					
						
							| 
									
										
										
										
											2019-06-23 00:29:12 -05:00
										 |  |  |         $newdef = "--step $step " . $tags['rrd_def'] . Config::get('rrd_rra'); | 
					
						
							| 
									
										
										
										
											2016-08-09 09:31:26 -05:00
										 |  |  |         rrdtool('create', $rrd, $newdef); | 
					
						
							| 
									
										
										
										
											2015-11-15 18:01:09 +10:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-10 19:31:39 +10:00
										 |  |  |     rrdtool_update($rrd, $fields); | 
					
						
							|  |  |  | } // rrdtool_data_update
 | 
					
						
							| 
									
										
										
										
											2015-11-15 18:01:09 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-15 12:21:09 -05:00
										 |  |  | /** | 
					
						
							|  |  |  |  * rename an rrdfile, can only be done on the LibreNMS server hosting the rrd files | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param array $device Device object | 
					
						
							| 
									
										
										
										
											2018-02-05 07:39:13 -06:00
										 |  |  |  * @param string|array $oldname RRD name array as used with rrd_name() | 
					
						
							|  |  |  |  * @param string|array $newname RRD name array as used with rrd_name() | 
					
						
							| 
									
										
										
										
											2015-11-15 18:01:09 +10:00
										 |  |  |  * @return bool indicating rename success or failure | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | function rrd_file_rename($device, $oldname, $newname) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     $oldrrd = rrd_name($device['hostname'], $oldname); | 
					
						
							|  |  |  |     $newrrd = rrd_name($device['hostname'], $newname); | 
					
						
							| 
									
										
										
										
											2016-01-10 19:31:39 +10:00
										 |  |  |     if (is_file($oldrrd) && !is_file($newrrd)) { | 
					
						
							| 
									
										
										
										
											2016-01-15 21:28:56 +10:00
										 |  |  |         if (rename($oldrrd, $newrrd)) { | 
					
						
							| 
									
										
										
										
											2017-02-13 00:41:05 +02:00
										 |  |  |             log_event("Renamed $oldrrd to $newrrd", $device, "poller", 1); | 
					
						
							| 
									
										
										
										
											2016-01-15 21:28:56 +10:00
										 |  |  |             return true; | 
					
						
							| 
									
										
										
										
											2016-08-09 09:31:26 -05:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2017-02-13 00:41:05 +02:00
										 |  |  |             log_event("Failed to rename $oldrrd to $newrrd", $device, "poller", 5); | 
					
						
							| 
									
										
										
										
											2016-01-15 21:28:56 +10:00
										 |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-08-09 09:31:26 -05:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2016-01-15 21:28:56 +10:00
										 |  |  |         // we don't need to rename the file
 | 
					
						
							| 
									
										
										
										
											2016-01-10 19:31:39 +10:00
										 |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-11-15 18:01:09 +10:00
										 |  |  | } // rrd_file_rename
 |