. * * @package LibreNMS * @link http://librenms.org * @copyright 2017 Tony Murray * @author Tony Murray */ namespace LibreNMS\Device; use LibreNMS\Interfaces\Discovery\DiscoveryItem; use LibreNMS\OS; class YamlDiscovery { /** * @param OS $os * @param DiscoveryItem|string $class * @param $yaml_data * @return array */ public static function discover(OS $os, $class, $yaml_data) { $pre_cache = $os->preCache(); $items = array(); // convert to class name for static call below if (is_object($class)) { $class = get_class($class); } d_echo("YAML Discovery Data: "); d_echo($yaml_data); foreach ($yaml_data as $first_key => $first_yaml) { if ($first_key == 'pre-cache') { continue; } $group_options = isset($first_yaml['options']) ? $first_yaml['options'] : array(); // find the data array, we could already be at for simple modules if (isset($data['data'])) { $first_yaml = $first_yaml['data']; } elseif ($first_key !== 'data') { continue; } foreach ($first_yaml as $data) { $raw_data = (array)$pre_cache[$data['oid']]; d_echo("Data {$data['oid']}: "); d_echo($raw_data); $count = 0; foreach ($raw_data as $index => $snmp_data) { $count++; $current_data = array(); if (!isset($data['value'])) { $data['value'] = $data['oid']; } foreach ($data as $name => $value) { if ($name == '$oid' || $name == 'skip_values') { $current_data[$name] = $value; } elseif (str_contains($value, '{{')) { // replace embedded values $current_data[$name] = static::replaceValues($name, $index, $count, $data, $pre_cache); } else { // replace references to data $current_data[$name] = static::getValueFromData($name, $index, $data, $pre_cache, $value); } } if (static::canSkipItem($current_data['value'], $index, $current_data, $group_options, $snmp_data)) { continue; } $item = $class::fromYaml($os, $index, $current_data); if ($item->isValid()) { $items[] = $item; } } } } return $items; } public static function replaceValues($name, $index, $count, $data, $pre_cache) { $value = static::getValueFromData($name, $index, $data, $pre_cache); if (is_null($value)) { // built in replacements $search = [ '{{ $index }}', '{{ $count }}', ]; $replace = [ $index, $count, ]; // prepare the $subindexX match variable replacement foreach (explode('.', $index) as $pos => $subindex) { $search[] = '{{ $subindex' . $pos . ' }}'; $replace[] = $subindex; } $value = str_replace($search, $replace, $data[$name]); // search discovery data for values $value = preg_replace_callback('/{{ \$([a-zA-Z0-9.]+) }}/', function ($matches) use ($index, $data, $pre_cache) { $replace = static::getValueFromData($matches[1], $index, $data, $pre_cache, null); if (is_null($replace)) { d_echo('Warning: No variable available to replace ' . $matches[1] . ".\n"); return ''; // remove the unavailable variable } return $replace; }, $value); } return $value; } /** * Helper function for dynamic discovery to search for data from pre_cached snmp data * * @param string $name The name of the field from the discovery data or just an oid * @param string $index The index of the current sensor * @param array $discovery_data The discovery data for the current sensor * @param array $pre_cache all pre-cached snmp data * @param mixed $default The default value to return if data is not found * @return mixed */ public static function getValueFromData($name, $index, $discovery_data, $pre_cache, $default = null) { if (isset($discovery_data[$name])) { $name = $discovery_data[$name]; } if (isset($pre_cache[$discovery_data['oid']][$index][$name])) { return $pre_cache[$discovery_data['oid']][$index][$name]; } if (isset($pre_cache[$name])) { if (is_array($pre_cache[$name])) { if (isset($pre_cache[$name][$index][$name])) { return $pre_cache[$name][$index][$name]; } elseif (isset($pre_cache[$index][$name])) { return $pre_cache[$index][$name]; } elseif (isset($pre_cache[$name][$index])) { return $pre_cache[$name][$index]; } elseif (count($pre_cache[$name]) === 1) { return current($pre_cache[$name]); } } else { return $pre_cache[$name]; } } return $default; } public static function preCache(OS $os) { // Pre-cache data for later use $pre_cache = array(); $device = $os->getDevice(); $pre_cache_file = 'includes/discovery/sensors/pre-cache/' . $device['os'] . '.inc.php'; if (is_file($pre_cache_file)) { echo "Pre-cache {$device['os']}: "; include $pre_cache_file; echo PHP_EOL; d_echo($pre_cache); } // TODO change to exclude os with pre-cache php file, but just exclude them by hand for now (like avtech) if ($device['os'] == 'avtech') { return $pre_cache; } if (!empty($device['dynamic_discovery']['modules'])) { echo "Caching data: "; foreach ($device['dynamic_discovery']['modules'] as $module => $discovery_data) { echo "$module "; foreach ($discovery_data as $key => $data_array) { // find the data array, we could already be at for simple modules if (isset($data_array['data'])) { $data_array = $data_array['data']; } elseif ($key !== 'data') { continue; } foreach ($data_array as $data) { foreach ((array)$data['oid'] as $oid) { if (!array_key_exists($oid, $pre_cache)) { if (isset($data['snmp_flags'])) { $snmp_flag = array_wrap($data['snmp_flags']); } else { $snmp_flag = ['-OteQUs']; } $snmp_flag[] = '-Ih'; $mib = $device['dynamic_discovery']['mib']; $pre_cache[$oid] = snmpwalk_cache_oid($device, $oid, $pre_cache[$oid], $mib, null, $snmp_flag); } } } } } echo PHP_EOL; } return $pre_cache; } /** * Check to see if we should skip this discovery item * * @param mixed $value * @param array $yaml_item_data The data key from this item * @param array $group_options The options key from this group of items * @param array $item_snmp_data The pre-cache data array * @return bool */ public static function canSkipItem($value, $index, $yaml_item_data, $group_options, $pre_cache = array()) { $skip_values = array_replace((array)$group_options['skip_values'], (array)$yaml_item_data['skip_values']); foreach ($skip_values as $skip_value) { if (is_array($skip_value) && $pre_cache) { // Dynamic skipping of data $op = isset($skip_value['op']) ? $skip_value['op'] : '!='; $tmp_value = static::getValueFromData($skip_value['oid'], $index, $yaml_item_data, $pre_cache); if (str_contains($skip_value['oid'], '.')) { list($skip_value['oid'], $targeted_index) = explode('.', $skip_value['oid'], 2); $tmp_value = static::getValueFromData($skip_value['oid'], $targeted_index, $yaml_item_data, $pre_cache); } if (compare_var($tmp_value, $skip_value['value'], $op)) { return true; } } if ($value == $skip_value) { return true; } } $skip_value_lt = array_replace((array)$group_options['skip_value_lt'], (array)$yaml_item_data['skip_value_lt']); foreach ($skip_value_lt as $skip_value) { if ($value < $skip_value) { return true; } } $skip_value_gt = array_replace((array)$group_options['skip_value_gt'], (array)$yaml_item_data['skip_value_gt']); foreach ($skip_value_gt as $skip_value) { if ($value > $skip_value) { return true; } } return false; } }