Files
librenms-librenms/LibreNMS/OS/ArubaInstant.php
Tony Murray 9d35fbc6a7 Entity Physical discovery: Rewrite to modern style (#16289)
* Initial entity-physical code

* Split out Entity-MIB trait

* Cisco Cellular inventory

* Fix bad test data

* alfo80hd - we now include all entPhysical entries

* Correct aos7 test data

* Add entPhysicalClass as last resort for label in ui

* aos add previously filtered data

* Fixup arista-eos data

* Update ariast_eos data

* Arris, clean garbage in Rev fields

* Aruba Instant custom inventory ported

* ArubaOS CX add vendor type mib

* aviat-wtm test data refresh

* axos add shelf fix data fields a bit

* ciena-rls

* ciena-sds

* Skip cimc for now... no test data

* Cisco updates

* Comware data update

* Update dnos

* Clean Edgeos garbage, make code from Arris shareable

* Relaxed ifIndex match, some devices cheat and send back static strings instead of formatted OIDs

* Regex refinement and updated edgeos with new clean data

* Update edgeswitch data

* Update eltex-mes21xx data

* eltex-mes23xx

* Guess at eltex-mes24xx since there is no test data

* Update eurostor, fix firmware version

* Apply fixes from StyleCI

* fixes

* Update fortigate data

* Update fortiweb, ftd, and fusion

* Update linux LSI

* Fix hexToAscii null removal with different seperator handling

* icotera add final snmprec data to avoid snmpsim bug

* Update IOS data

* Update mrv-od

* Add junos translation

* Generic data updates n-r

* ruijie workaround snmpsim bug

* Port saf-cfm

* Recode Schleifenbauer, and fix entPhysicalIndex values

* SmartAX fixes

* sm-os and tait-infra93

* timos inventory was not right, fix it up

* ubiquoss-pon

* VRP, has custom data collection on top of normal
adapt port ifIndex lookup to handle it

* VRP exceeded the string length specified in ENTITY-MIB...

* data updates

* Final data update and code cleanup

* Apply fixes from StyleCI

* Lint fixes

* Add missing SnmpResponse->pluck() code

* Update db_schema.yaml

* Fix bad test data

* Another instant-on update

* oops

* Remove some unused code

# Conflicts:
#	includes/html/pages/device/overview.inc.php

---------

Co-authored-by: Tony Murray <murrant@users.noreply.github.com>
2024-08-21 01:12:09 -05:00

400 lines
15 KiB
PHP

<?php
/**
* ArubaInstant.php
*
* HPE Aruba Instant
*
*
* 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 <https://www.gnu.org/licenses/>.
*
* @link https://www.librenms.org
*
* @copyright 2019 Timothy Willey
* @author Timothy Willey <developer@timothywilley.net>
*/
namespace LibreNMS\OS;
use App\Models\Device;
use App\Models\EntPhysical;
use Illuminate\Support\Collection;
use LibreNMS\Device\Processor;
use LibreNMS\Device\WirelessSensor;
use LibreNMS\Interfaces\Discovery\OSDiscovery;
use LibreNMS\Interfaces\Discovery\ProcessorDiscovery;
use LibreNMS\Interfaces\Discovery\Sensors\WirelessApCountDiscovery;
use LibreNMS\Interfaces\Discovery\Sensors\WirelessClientsDiscovery;
use LibreNMS\Interfaces\Discovery\Sensors\WirelessFrequencyDiscovery;
use LibreNMS\Interfaces\Discovery\Sensors\WirelessNoiseFloorDiscovery;
use LibreNMS\Interfaces\Discovery\Sensors\WirelessPowerDiscovery;
use LibreNMS\Interfaces\Discovery\Sensors\WirelessUtilizationDiscovery;
use LibreNMS\Interfaces\Polling\Sensors\WirelessApCountPolling;
use LibreNMS\Interfaces\Polling\Sensors\WirelessClientsPolling;
use LibreNMS\Interfaces\Polling\Sensors\WirelessFrequencyPolling;
use LibreNMS\OS;
use LibreNMS\Util\Mac;
use LibreNMS\Util\Number;
class ArubaInstant extends OS implements
OSDiscovery,
ProcessorDiscovery,
WirelessApCountDiscovery,
WirelessApCountPolling,
WirelessClientsDiscovery,
WirelessClientsPolling,
WirelessFrequencyDiscovery,
WirelessFrequencyPolling,
WirelessNoiseFloorDiscovery,
WirelessPowerDiscovery,
WirelessUtilizationDiscovery
{
public function discoverOS(Device $device): void
{
parent::discoverOS($device); // yaml
$device->serial = snmp_getnext($this->getDeviceArray(), 'aiAPSerialNum', '-Oqv', 'AI-AP-MIB');
}
/**
* Discover processors.
* Returns an array of LibreNMS\Device\Processor objects that have been discovered
*
* @return array Processors
*/
public function discoverProcessors()
{
$processors = [];
$ai_mib = 'AI-AP-MIB';
$ai_ap_data = $this->getCacheTable('aiAccessPointEntry', $ai_mib);
foreach ($ai_ap_data as $ai_ap => $ai_ap_oid) {
$value = $ai_ap_oid['aiAPCPUUtilization'];
$mac = Mac::parse($ai_ap);
$combined_oid = sprintf('%s::%s.%s', $ai_mib, 'aiAPCPUUtilization', $mac->oid());
$oid = snmp_translate($combined_oid, 'ALL', 'arubaos', '-On');
$description = $ai_ap_data[$ai_ap]['aiAPSerialNum'];
$processors[] = Processor::discover('aruba-instant', $this->getDeviceId(), $oid, $mac->hex(), $description, 1, $value);
} // end foreach
return $processors;
}
/**
* Discover wireless client counts. Type is clients.
* Returns an array of LibreNMS\Device\Sensor objects that have been discovered
*
* @return array Sensors
*/
public function discoverWirelessClients()
{
$sensors = [];
$device = $this->getDeviceArray();
$ai_mib = 'AI-AP-MIB';
if (intval(explode('.', $device['version'])[0]) >= 8 && intval(explode('.', $device['version'])[1]) >= 4) {
// version is at least 8.4.0.0
$ssid_data = $this->getCacheTable('aiWlanSSIDEntry', $ai_mib);
$ap_data = array_merge_recursive(
$this->getCacheTable('aiAccessPointEntry', $ai_mib),
$this->getCacheTable('aiRadioClientNum', $ai_mib)
);
$oids = [];
$total_clients = 0;
// Clients Per SSID
foreach ($ssid_data as $index => $entry) {
$combined_oid = sprintf('%s::%s.%s', $ai_mib, 'aiSSIDClientNum', $index);
$oid = snmp_translate($combined_oid, 'ALL', 'arubaos', '-On');
$description = sprintf('SSID %s Clients', $entry['aiSSID']);
$oids[] = $oid;
$total_clients += $entry['aiSSIDClientNum'];
$sensors[] = new WirelessSensor('clients', $this->getDeviceId(), $oid, 'aruba-instant', $index, $description, $entry['aiSSIDClientNum']);
}
// Total Clients across all SSIDs
$sensors[] = new WirelessSensor('clients', $this->getDeviceId(), $oids, 'aruba-instant', 'total-clients', 'Total Clients', $total_clients);
// Clients Per Radio
foreach ($ap_data as $index => $entry) {
foreach ($entry['aiRadioClientNum'] as $radio => $value) {
$combined_oid = sprintf('%s::%s.%s.%s', $ai_mib, 'aiRadioClientNum', Mac::parse($index)->oid(), $radio);
$oid = snmp_translate($combined_oid, 'ALL', 'arubaos', '-On');
$description = sprintf('%s Radio %s', $entry['aiAPSerialNum'], $radio);
$sensor_index = sprintf('%s.%s', Mac::parse($index)->hex(), $radio);
$sensors[] = new WirelessSensor('clients', $this->getDeviceId(), $oid, 'aruba-instant', $sensor_index, $description, $value);
}
}
} else {
// version is lower than 8.4.0.0
// fetch the MAC addresses of currently connected clients, then count them to get an overall total
$client_data = $this->getCacheTable('aiClientMACAddress', $ai_mib);
$total_clients = count($client_data);
$combined_oid = sprintf('%s::%s', $ai_mib, 'aiClientMACAddress');
$oid = snmp_translate($combined_oid, 'ALL', 'arubaos', '-On');
$sensors[] = new WirelessSensor('clients', $this->getDeviceId(), $oid, 'aruba-instant', 'total-clients', 'Total Clients', $total_clients);
}
return $sensors;
}
/**
* Discover wireless AP counts. Type is ap-count.
* Returns an array of LibreNMS\Device\Sensor objects that have been discovered
*
* @return array Sensors
*/
public function discoverWirelessApCount()
{
$sensors = [];
$ai_mib = 'AI-AP-MIB';
$ap_data = $this->getCacheTable('aiAPSerialNum', $ai_mib);
$total_aps = count($ap_data);
$combined_oid = sprintf('%s::%s', $ai_mib, 'aiAPSerialNum');
$oid = snmp_translate($combined_oid, 'ALL', 'arubaos', '-On');
$sensors[] = new WirelessSensor('ap-count', $this->getDeviceId(), $oid, 'aruba-instant', 'total-aps', 'Total APs', $total_aps);
return $sensors;
}
/**
* Discover wireless frequency. This is in MHz. Type is frequency.
* Returns an array of LibreNMS\Device\Sensor objects that have been discovered
*
* @return array Sensors
*/
public function discoverWirelessFrequency()
{
// instant
return $this->discoverInstantRadio('frequency', 'aiRadioChannel');
}
/**
* Discover wireless noise floor. This is in dBm/Hz. Type is noise-floor.
* Returns an array of LibreNMS\Device\Sensor objects that have been discovered
*
* @return array
*/
public function discoverWirelessNoiseFloor()
{
// instant
return $this->discoverInstantRadio('noise-floor', 'aiRadioNoiseFloor');
}
/**
* Discover wireless tx or rx power. This is in dBm. Type is power.
* Returns an array of LibreNMS\Device\Sensor objects that have been discovered
*
* @return array
*/
public function discoverWirelessPower()
{
// instant
return $this->discoverInstantRadio('power', 'aiRadioTransmitPower', '%s Radio %s: Tx Power');
}
/**
* Discover wireless utilization. This is in %. Type is utilization.
* Returns an array of LibreNMS\Device\Sensor objects that have been discovered
*
* @return array Sensors
*/
public function discoverWirelessUtilization()
{
// instant
return $this->discoverInstantRadio('utilization', 'aiRadioUtilization64');
}
/**
* Aruba Instant Radio Discovery
*
* @return array Sensors
*/
private function discoverInstantRadio($type, $mib, $desc = '%s Radio %s')
{
$ai_mib = 'AI-AP-MIB';
$ai_sg_data = array_merge_recursive(
$this->getCacheTable('aiAPSerialNum', $ai_mib),
$this->getCacheTable('aiRadioChannel', $ai_mib),
$this->getCacheTable('aiRadioNoiseFloor', $ai_mib),
$this->getCacheTable('aiRadioTransmitPower', $ai_mib),
$this->getCacheTable('aiRadioUtilization64', $ai_mib)
);
$sensors = [];
foreach ($ai_sg_data as $ai_ap => $ai_ap_oid) {
if (isset($ai_ap_oid[$mib])) {
foreach ($ai_ap_oid[$mib] as $ai_ap_radio => $value) {
$multiplier = 1;
if ($type == 'frequency') {
$value = WirelessSensor::channelToFrequency($this->decodeChannel($value));
}
if ($type == 'noise-floor') {
$multiplier = -1;
$value = $value * $multiplier;
}
$combined_oid = sprintf('%s::%s.%s.%s', $ai_mib, $mib, Mac::parse($ai_ap)->oid(), $ai_ap_radio);
$oid = snmp_translate($combined_oid, 'ALL', 'arubaos', '-On');
$description = sprintf($desc, $ai_sg_data[$ai_ap]['aiAPSerialNum'], $ai_ap_radio);
$index = sprintf('%s.%s', Mac::parse($ai_ap)->hex(), $ai_ap_radio);
$sensors[] = new WirelessSensor($type, $this->getDeviceId(), $oid, 'aruba-instant', $index, $description, $value, $multiplier);
} // end foreach
} // end if
} // end foreach
return $sensors;
}
protected function decodeChannel($channel): int
{
// Trim off everything not a digit, like channel "116e"
$channel = Number::cast(preg_replace("/\D/", '', $channel));
return $channel & 255; // mask off the channel width information
}
/**
* Poll wireless frequency as MHz
* The returned array should be sensor_id => value pairs
*
* @param array $sensors Array of sensors needed to be polled
* @return array of polled data
*/
public function pollWirelessFrequency(array $sensors)
{
return $this->pollWirelessChannelAsFrequency($sensors, [$this, 'decodeChannel']);
}
/**
* Poll wireless clients
* The returned array should be sensor_id => value pairs
*
* @param array $sensors Array of sensors needed to be polled
* @return array of polled data
*/
public function pollWirelessClients(array $sensors)
{
$data = [];
if (! empty($sensors)) {
$device = $this->getDeviceArray();
if (intval(explode('.', $device['version'])[0]) >= 8 && intval(explode('.', $device['version'])[1]) >= 4) {
// version is at least 8.4.0.0
$oids = [];
foreach ($sensors as $sensor) {
$oids[$sensor['sensor_id']] = current($sensor['sensor_oids']);
}
$snmp_data = snmp_get_multi_oid($this->getDeviceArray(), $oids);
foreach ($oids as $id => $oid) {
$data[$id] = $snmp_data[$oid] ?? null;
}
} else {
// version is lower than 8.4.0.0
if (! empty($sensors) && count($sensors) == 1) {
$ai_mib = 'AI-AP-MIB';
$client_data = $this->getCacheTable('aiClientMACAddress', $ai_mib);
if (empty($client_data)) {
$total_clients = 0;
} else {
$total_clients = count($client_data);
}
$data[$sensors[0]['sensor_id']] = $total_clients;
}
}
}
return $data;
}
/**
* Poll AP Count
* The returned array should be sensor_id => value pairs
*
* @param array $sensors Array of sensors needed to be polled
* @return array of polled data
*/
public function pollWirelessApCount(array $sensors)
{
$data = [];
if (! empty($sensors) && count($sensors) == 1) {
$ai_mib = 'AI-AP-MIB';
$ap_data = $this->getCacheTable('aiAPSerialNum', $ai_mib);
$total_aps = 0;
if (! empty($ap_data)) {
$total_aps = count($ap_data);
}
$data[$sensors[0]['sensor_id']] = $total_aps;
}
return $data;
}
public function discoverEntityPhysical(): Collection
{
$inventory = new Collection;
$ai_ig_data = \SnmpQuery::walk('AI-AP-MIB::aiInfoGroup')->table(1);
$master_ip = $ai_ig_data[0]['AI-AP-MIB::aiMasterIPAddress'] ?? null;
if ($master_ip) {
$inventory->push(new EntPhysical([
'entPhysicalIndex' => 1,
'entPhysicalDescr' => $ai_ig_data[0]['AI-AP-MIB::aiVirtualControllerIPAddress'],
'entPhysicalClass' => 'chassis',
'entPhysicalName' => $ai_ig_data[0]['AI-AP-MIB::aiVirtualControllerName'],
'entPhysicalModelName' => 'Instant Virtual Controller Cluster',
'entPhysicalSerialNum' => $ai_ig_data[0]['AI-AP-MIB::aiVirtualControllerKey'],
'entPhysicalMfgName' => 'Aruba',
]));
}
$index = 2;
$ap_data = \SnmpQuery::hideMib()->walk('AI-AP-MIB:aiAccessPointTable')->table(1);
foreach ($ap_data as $mac => $entry) {
$type = $master_ip == $entry['aiAPIPAddress'] ? 'Master' : 'Member';
$inventory->push(new EntPhysical([
'entPhysicalIndex' => $index++,
'entPhysicalDescr' => $entry['aiAPMACAddress'],
'entPhysicalName' => sprintf('%s %s Cluster %s', $entry['aiAPName'], $entry['aiAPIPAddress'], $type),
'entPhysicalClass' => 'other',
'entPhysicalContainedIn' => 1,
'entPhysicalSerialNum' => $entry['aiAPSerialNum'],
'entPhysicalModelName' => $entry['aiAPModel'],
'entPhysicalMfgName' => 'Aruba',
'entPhysicalVendorType' => 'accessPoint',
'entPhysicalSoftwareRev' => $this->getDevice()->version,
]));
}
return $inventory;
}
}