Files
librenms-librenms/LibreNMS/OS/Shared/Cisco.php
T
Tony Murray f2f169ae78 Modernize mempools (#12282)
* mempools to modern module
quick hacky hrstorage port

* port ucd-snmp-mib to Mempools

* Populate DB for ucd
Prep for yaml

* initial yaml attempt

* more complex conversions
fixes to YamlDiscovery, make leading $ optional and allow mib::oid format

* walk full tables and skip values
normalize percentages above 100

* handle precent only ones (specify total as 100)

* Move default polling out of YamlMempoolsDiscovery

* fixes

* Update test data hrstorage should be correct.

* perc_warn for hrstorage

* Host Resources, record buffer, cached, and shared

* Host Resources is always better, don't do both HR and UCD

* fix unix, include warning levels

* variable size

* consolidate skip_values

* define mempools schema

* number instead of integer

* more schema refactor

* one more skip_values reference

* throw error for invalid oid translation
aos6

* a*  and Cisco

* updated test data

* update almost all hrstorage data files

* b*

* c* with test data
use standard cache for hrStorage

* use cache for storage module too

* hand bsnmp properly

* bdcom

* exclude total oid from yaml so it is not polled
May add a way to ignore this behavior and poll it, but I don't know if that is needed.

* automatically handle percent only values

* ciscowlc

* only poll used or free if we have used, free, and total.

* fix skipping

* the dlinkoning
fix find value when value "name" is numeric

* support numeric oids

* dnos/ftos attempt

* I guess we can't filter on total > 0

* edgecos

* e*

* f WIP

* f*

* gwd (aka g*)

* h* + procurve

* i*

* j*

* m*

* support 0% used memory (however unlikely)

* n*

* CISCO-PROCESS-MIB memory, share cache with processors module

* ignore mempools with invalid total

* p*

* quanta

* r*
fix raisecom mibs terribly broken

* s-z

* style fixes

* Move VRP back to PHP and make it actually work

* fix zynos

* update schema

* Update Cisco processor data for description bug fixes

* fix comware processors

* comware mempools with memory size
default precision to 1

* sophos-xg updated data

* hrstorage use ram size for buffers, cache, and shared

* Show memory available instead of free in device overview

* UCD, use same rrd format, store available instead of free in the db.

* Calculate availability for HOST-RESOURCES-MIB

* Convert UCD to standard polling

* rename old rrd files

* initial graph work

* graph WIP

* Graph looking decent

* Graph looking decent for hr

* remove old ucd_graph code

* handle availability mempool
more graph cleanup

* color adjustments

* remove accidental free calculation

* Update test data and fix corner cases

* fis pfsense

* update schema

* add default value for mempool_class

* fix whitespace

* update schema

* update schema correctly

* one more time

* fortigate_1500d-sensors missing oids

* Update docs.

* fix indent

* add implements MempoolsDiscovery explicitly to OS

* remove ucd_memory graph references
remove unused device_memory graph

* remove unused functions

* set devices with mempools to rediscover to prevent/minimize gaps

* use a subquery

* add overview graph

* port health mempools table

* Update device mempool

* only show overview if multiple

* Don't override user set warn percentages in discovery

* fix missed usage

* fix style

* Safety check to not rename rrd files incorrectly if migration has not been run.

* Fix overview percent bar and represent available and used on the bar

* missed an item to convert to mempool_class

* percent on the wrong side
2020-11-23 15:35:35 -06:00

365 lines
16 KiB
PHP

<?php
/**
* Cisco.php
*
* Base Cisco OS for Cisco based devices
*
* 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/>.
*
* @link http://librenms.org
* @copyright 2018 Tony Murray
* @author Tony Murray <murraytony@gmail.com>
* @copyright 2018 Jose Augusto Cardoso
*/
namespace LibreNMS\OS\Shared;
use App\Models\Device;
use App\Models\Mempool;
use App\Models\PortsNac;
use Illuminate\Support\Arr;
use LibreNMS\Device\Processor;
use LibreNMS\Interfaces\Discovery\MempoolsDiscovery;
use LibreNMS\Interfaces\Discovery\OSDiscovery;
use LibreNMS\Interfaces\Discovery\ProcessorDiscovery;
use LibreNMS\Interfaces\Polling\NacPolling;
use LibreNMS\OS;
use LibreNMS\OS\Traits\YamlOSDiscovery;
use LibreNMS\Util\IP;
class Cisco extends OS implements OSDiscovery, ProcessorDiscovery, MempoolsDiscovery, NacPolling
{
use YamlOSDiscovery {
YamlOSDiscovery::discoverOS as discoverYamlOS;
}
public function discoverOS(Device $device): void
{
// yaml discovery overrides this
if ($this->hasYamlDiscovery('os')) {
$this->discoverYamlOS($device);
return;
}
$device->serial = $this->getMainSerial();
$hardware = null;
if (preg_match('/^Cisco IOS Software, .+? Software \([^\-]+-([^\-]+)-\w\),.+?Version ([^, ]+)/', $device->sysDescr, $regexp_result)) {
$device->features = $regexp_result[1];
$device->version = $regexp_result[2];
} elseif (preg_match('/Cisco Internetwork Operating System Software\s+IOS \(tm\) [^ ]+ Software \([^\-]+-([^\-]+)-\w\),.+?Version ([^, ]+)/', $device->sysDescr, $regexp_result)) {
$device->features = $regexp_result[1];
$device->version = $regexp_result[2];
} elseif (preg_match('/^Cisco IOS Software \[([^\]]+)\],.+Software \(([^\)]+)\), Version ([^, ]+)/', $device->sysDescr, $regexp_result)) {
$device->features = $regexp_result[1];
$device->version = $regexp_result[2] . ' ' . $regexp_result[3];
} elseif (preg_match('/^Cisco IOS Software.*?, .+? Software(\, )?([\s\w\d]+)? \([^\-]+-([\w\d]+)-\w\), Version ([^,]+)/', $device->sysDescr, $regexp_result)) {
$device->features = $regexp_result[3];
$device->version = $regexp_result[4];
$hardware = $regexp_result[2];
$tmp = preg_split('/\\r\\n|\\r|\\n/', $device->version);
if (! empty($tmp[0])) {
$device->version = $tmp[0];
}
}
$oids = [
'entPhysicalModelName.1',
'entPhysicalContainedIn.1',
'entPhysicalName.1',
'entPhysicalSoftwareRev.1',
'entPhysicalModelName.1000',
'entPhysicalModelName.1001',
'entPhysicalContainedIn.1000',
'entPhysicalContainedIn.1001',
];
$data = snmp_get_multi($this->getDeviceArray(), $oids, '-OQUs', 'ENTITY-MIB:OLD-CISCO-CHASSIS-MIB');
if (isset($data[1]['entPhysicalContainedIn']) && $data[1]['entPhysicalContainedIn'] == '0') {
if (! empty($data[1]['entPhysicalSoftwareRev'])) {
$device->version = $data[1]['entPhysicalSoftwareRev'];
}
if (! empty($data[1]['entPhysicalName'])) {
$hardware = $data[1]['entPhysicalName'];
}
if (! empty($data[1]['entPhysicalModelName'])) {
$hardware = $data[1]['entPhysicalModelName'];
}
}
if (empty($hardware) && ! empty($data[1000]['entPhysicalModelName'])) {
$hardware = $data[1000]['entPhysicalModelName'];
} elseif (empty($hardware) && ! empty($data[1000]['entPhysicalContainedIn'])) {
$hardware = $data[$data[1000]['entPhysicalContainedIn']]['entPhysicalName'];
} elseif ((preg_match('/stack/i', $hardware) || empty($hardware)) && ! empty($data[1001]['entPhysicalModelName'])) {
$hardware = $data[1001]['entPhysicalModelName'];
} elseif (empty($hardware) && ! empty($data[1001]['entPhysicalContainedIn'])) {
$hardware = $data[$data[1001]['entPhysicalContainedIn']]['entPhysicalName'];
}
$device->hardware = $hardware ?: snmp_translate($device->sysObjectID, 'SNMPv2-MIB:CISCO-PRODUCTS-MIB', 'cisco');
}
public function discoverMempools()
{
if ($this->hasYamlDiscovery('mempools')) {
return parent::discoverMempools(); // yaml
}
$mempools = collect();
$cemp = snmpwalk_cache_multi_oid($this->getDeviceArray(), 'cempMemPoolTable', [], 'CISCO-ENHANCED-MEMPOOL-MIB');
foreach (Arr::wrap($cemp) as $index => $entry) {
if (is_numeric($entry['cempMemPoolUsed']) && $entry['cempMemPoolValid'] == 'true') {
[$entPhysicalIndex] = explode('.', $index);
$entPhysicalName = $this->getCacheByIndex('entPhysicalName', 'ENTITY-MIB')[$entPhysicalIndex];
$descr = ucwords($entPhysicalName . ' - ' . $entry['cempMemPoolName']);
$descr = trim(str_replace(['Cisco ', 'Network Processing Engine'], '', $descr), ' -');
$mempools->push((new Mempool([
'mempool_index' => $index,
'entPhysicalIndex' => $entPhysicalIndex,
'mempool_type' => 'cemp',
'mempool_class' => 'system',
'mempool_precision' => 1,
'mempool_descr' => $descr,
'mempool_used_oid' => isset($entry['cempMemPoolHCUsed']) ? ".1.3.6.1.4.1.9.9.221.1.1.1.1.18.$index" : ".1.3.6.1.4.1.9.9.221.1.1.1.1.7.$index",
'mempool_free_oid' => isset($entry['cempMemPoolHCFree']) ? ".1.3.6.1.4.1.9.9.221.1.1.1.1.20.$index" : ".1.3.6.1.4.1.9.9.221.1.1.1.1.8.$index",
'mempool_perc_warn' => 90,
'mempool_largestfree' => $entry['cempMemPoolHCLargestFree'] ?? $entry['cempMemPoolLargestFree'] ?? null,
'mempool_lowestfree' => $entry['cempMemPoolHCLowestFree'] ?? $entry['cempMemPoolLowestFree'] ?? null,
]))->fillUsage($entry['cempMemPoolHCUsed'] ?? $entry['cempMemPoolUsed'], null, $entry['cempMemPoolHCFree'] ?? $entry['cempMemPoolFree']));
}
}
if ($mempools->isNotEmpty()) {
return $mempools;
}
$cmp = snmpwalk_cache_oid($this->getDeviceArray(), 'ciscoMemoryPool', [], 'CISCO-MEMORY-POOL-MIB');
foreach (Arr::wrap($cmp) as $index => $entry) {
if (is_numeric($entry['ciscoMemoryPoolUsed']) && is_numeric($index)) {
$mempools->push((new Mempool([
'mempool_index' => $index,
'mempool_type' => 'cmp',
'mempool_class' => 'system',
'mempool_precision' => 1,
'mempool_descr' => $entry['ciscoMemoryPoolName'],
'mempool_used_oid' => ".1.3.6.1.4.1.9.9.48.1.1.1.5.$index",
'mempool_free_oid' => ".1.3.6.1.4.1.9.9.48.1.1.1.6.$index",
'mempool_perc_warn' => 90,
'mempool_largestfree' => $entry['ciscoMemoryPoolLargestFree'] ?? null,
]))->fillUsage($entry['ciscoMemoryPoolUsed'], null, $entry['ciscoMemoryPoolFree']));
}
}
if ($mempools->isNotEmpty()) {
return $mempools;
}
$cpm = $this->getCacheTable('cpmCPUTotalTable', 'CISCO-PROCESS-MIB');
$count = 0;
foreach (Arr::wrap($cpm) as $index => $entry) {
$count++;
if (is_numeric($entry['cpmCPUMemoryFree']) && is_numeric($entry['cpmCPUMemoryFree'])) {
$cpu = $this->getCacheByIndex('entPhysicalName', 'ENTITY-MIB')[$entry['cpmCPUTotalPhysicalIndex'] ?? 'none'] ?? "Processor $index";
$mempools->push((new Mempool([
'mempool_index' => $index,
'mempool_type' => 'cpm',
'mempool_class' => 'system',
'mempool_precision' => 1024,
'mempool_descr' => "$cpu Memory",
'mempool_used_oid' => empty($entry['cpmCPUMemoryHCUsed']) ? ".1.3.6.1.4.1.9.9.109.1.1.1.1.12.$index" : ".1.3.6.1.4.1.9.9.109.1.1.1.1.17.$index",
'mempool_free_oid' => empty($entry['cpmCPUMemoryHCFree']) ? ".1.3.6.1.4.1.9.9.109.1.1.1.1.13.$index" : ".1.3.6.1.4.1.9.9.109.1.1.1.1.19.$index",
'mempool_perc_warn' => 90,
'mempool_lowestfree' => $entry['cpmCPUMemoryHCLowest'] ?? $entry['cpmCPUMemoryLowest'] ?? null,
]))->fillUsage(
empty($entry['cpmCPUMemoryHCUsed']) ? $entry['cpmCPUMemoryUsed'] : $entry['cpmCPUMemoryHCUsed'],
null,
empty($entry['cpmCPUMemoryHCFree']) ? $entry['cpmCPUMemoryFree'] : $entry['cpmCPUMemoryHCFree']
));
}
}
return $mempools;
}
/**
* Discover processors.
* Returns an array of LibreNMS\Device\Processor objects that have been discovered
*
* @return array Processors
*/
public function discoverProcessors()
{
$processors_data = $this->getCacheTable('cpmCPUTotalTable', 'CISCO-PROCESS-MIB');
$processors_data = snmpwalk_group($this->getDeviceArray(), 'cpmCoreTable', 'CISCO-PROCESS-MIB', 1, $processors_data);
$processors = [];
foreach ($processors_data as $index => $entry) {
if (is_numeric($entry['cpmCPUTotal5minRev'])) {
$usage_oid = '.1.3.6.1.4.1.9.9.109.1.1.1.1.8.' . $index;
$usage = $entry['cpmCPUTotal5minRev'];
} elseif (is_numeric($entry['cpmCPUTotal5min'])) {
$usage_oid = '.1.3.6.1.4.1.9.9.109.1.1.1.1.5.' . $index;
$usage = $entry['cpmCPUTotal5min'];
} else {
continue; // skip bad data
}
if (isset($entry['cpmCPUTotalPhysicalIndex'])) {
$descr = $this->getCacheByIndex('entPhysicalName', 'ENTITY-MIB')[$entry['cpmCPUTotalPhysicalIndex']];
}
if (empty($descr)) {
$descr = "Processor $index";
}
if (is_array($entry['cpmCore5min'])) {
// This CPU has data per individual core
foreach ($entry['cpmCore5min'] as $core_index => $core_usage) {
$processors[] = Processor::discover(
'cpm',
$this->getDeviceId(),
".1.3.6.1.4.1.9.9.109.1.1.2.1.5.$index.$core_index",
"$index.$core_index",
"$descr: Core $core_index",
1,
$core_usage,
null,
$entry['cpmCPUTotalPhysicalIndex']
);
}
} else {
$processors[] = Processor::discover(
'cpm',
$this->getDeviceId(),
$usage_oid,
$index,
$descr,
1,
$usage,
null,
$entry['cpmCPUTotalPhysicalIndex']
);
}
}
if (empty($processors)) {
// fallback to old pre-12.0 OID
$processors[] = Processor::discover(
'ios',
$this->getDeviceId(),
'.1.3.6.1.4.1.9.2.1.58.0', // OLD-CISCO-CPU-MIB::avgBusy5
0
);
}
// QFP processors (Forwarding Processors)
$qfp_data = snmpwalk_group($this->getDeviceArray(), 'ceqfpUtilProcessingLoad', 'CISCO-ENTITY-QFP-MIB');
foreach ($qfp_data as $entQfpPhysicalIndex => $entry) {
/*
* .2 OID suffix is for 1 min SMA ('oneMinute')
* .3 OID suffix is for 5 min SMA ('fiveMinute')
* Could be dynamically changed to appropriate value if config had pol interval value
*/
$qfp_usage_oid = '.1.3.6.1.4.1.9.9.715.1.1.6.1.14.' . $entQfpPhysicalIndex . '.3';
$qfp_usage = $entry['fiveMinute'];
if ($entQfpPhysicalIndex) {
$qfp_descr = $this->getCacheByIndex('entPhysicalName', 'ENTITY-MIB')[$entQfpPhysicalIndex];
}
$processors[] = Processor::discover(
'qfp',
$this->getDeviceId(),
$qfp_usage_oid,
$entQfpPhysicalIndex . '.3',
$qfp_descr ?? "QFP $entQfpPhysicalIndex",
1,
$qfp_usage,
null,
$entQfpPhysicalIndex
);
}
return $processors;
}
public function pollNac()
{
$nac = collect();
$portAuthSessionEntry = snmpwalk_cache_oid($this->getDeviceArray(), 'cafSessionEntry', [], 'CISCO-AUTH-FRAMEWORK-MIB');
if (! empty($portAuthSessionEntry)) {
$cafSessionMethodsInfoEntry = collect(snmpwalk_cache_oid($this->getDeviceArray(), 'cafSessionMethodsInfoEntry', [], 'CISCO-AUTH-FRAMEWORK-MIB'))->mapWithKeys(function ($item, $key) {
$key_parts = explode('.', $key);
$key = implode('.', array_slice($key_parts, 0, 2)); // remove the auth method
return [$key => ['method' => $key_parts[2], 'authc_status' => $item['cafSessionMethodState']]];
});
// cache port ifIndex -> port_id map
$ifIndex_map = $this->getDevice()->ports()->pluck('port_id', 'ifIndex');
// update the DB
foreach ($portAuthSessionEntry as $index => $portAuthSessionEntryParameters) {
[$ifIndex, $auth_id] = explode('.', str_replace("'", '', $index));
$session_info = $cafSessionMethodsInfoEntry->get($ifIndex . '.' . $auth_id);
$mac_address = strtolower(implode(array_map('zeropad', explode(':', $portAuthSessionEntryParameters['cafSessionClientMacAddress']))));
$nac->put($mac_address, new PortsNac([
'port_id' => $ifIndex_map->get($ifIndex, 0),
'mac_address' => $mac_address,
'auth_id' => $auth_id,
'domain' => $portAuthSessionEntryParameters['cafSessionDomain'],
'username' => $portAuthSessionEntryParameters['cafSessionAuthUserName'],
'ip_address' => (string) IP::fromHexString($portAuthSessionEntryParameters['cafSessionClientAddress'], true),
'host_mode' => $portAuthSessionEntryParameters['cafSessionAuthHostMode'],
'authz_status' => $portAuthSessionEntryParameters['cafSessionStatus'],
'authz_by' => $portAuthSessionEntryParameters['cafSessionAuthorizedBy'],
'timeout' => $portAuthSessionEntryParameters['cafSessionTimeout'],
'time_left' => $portAuthSessionEntryParameters['cafSessionTimeLeft'],
'vlan' => $portAuthSessionEntryParameters['cafSessionAuthVlan'],
'authc_status' => $session_info['authc_status'],
'method' => $session_info['method'],
]));
}
}
return $nac;
}
protected function getMainSerial()
{
$serial_output = snmp_get_multi($this->getDeviceArray(), ['entPhysicalSerialNum.1', 'entPhysicalSerialNum.1001'], '-OQUs', 'ENTITY-MIB:OLD-CISCO-CHASSIS-MIB');
// $serial_output = snmp_getnext($this->getDevice(), 'entPhysicalSerialNum', '-OQUs', 'ENTITY-MIB:OLD-CISCO-CHASSIS-MIB');
if (! empty($serial_output[1]['entPhysicalSerialNum'])) {
return $serial_output[1]['entPhysicalSerialNum'];
} elseif (! empty($serial_output[1000]['entPhysicalSerialNum'])) {
return $serial_output[1000]['entPhysicalSerialNum'];
} elseif (! empty($serial_output[1001]['entPhysicalSerialNum'])) {
return $serial_output[1001]['entPhysicalSerialNum'];
}
return null;
}
}