mirror of
https://github.com/librenms/librenms.git
synced 2024-10-07 16:52:45 +00:00
* 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>
400 lines
15 KiB
PHP
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;
|
|
}
|
|
}
|