Files
librenms-librenms/LibreNMS/Modules/Isis.php
Tony Murray c79b187d72 Poller rewrite (Try 2) (#13525)
* core WIP

* try to finish up

* trim space too
and a couple of cleanups

* update test data

* put escapes back

* another net-snmp difference

* correct copy paste error

* WIP

* Use new code YAY

* a tiny bit more

* Kind of working

* Handle manual modules correctly

* convert core to modern module

* Only save metrics if modules is not overridden

* correct module exists check

* database error handling

* debug handling

* restore bad changes

* Introduce Actions
 RunAlertRulesAction
 UpdateDeviceGroupsAction

* tweaks to output

* Fix some issues in outside code

* Style fixes

* fixes to module status checks

* typehints!

* Use logger only and DI

* OS module not named correctly

* Work on quiet output a bit more

* generically don't change output when disabling debug if the driver is already stack

* Fix missing $device variable for legacy os polling
Fix missing dbFacile functions when no legacy modules polled in RunAlertRulesAction

* restore legacy os module shim

* use the new poller code for tests

* PollingDevice event

* Fix some issues and enable/disable error reporting around legacy modules

* typehints

* fully update baseline

* Use Process for version commands so we don't leak debug output.

* don't detect rrdtool version in ci every time

* style fixes

* Warning fixes

* more fixes

* re-update baseline

* remove diff noise

* fix up alerts

* Catch exceptions in device ip lookup

* Revert accidental snmp.inc.php poller target change
(should have been ?: not ??)
2021-11-17 19:23:55 -06:00

179 lines
6.5 KiB
PHP

<?php
/**
* Isis.php
*
* -Description-
*
* 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 2021 Otto Reinikainen
* @author Otto Reinikainen <otto@ottorei.fi>
*/
namespace LibreNMS\Modules;
use App\Models\IsisAdjacency;
use App\Observers\ModuleModelObserver;
use Illuminate\Support\Arr;
use Illuminate\Support\Collection;
use LibreNMS\DB\SyncsModels;
use LibreNMS\Interfaces\Discovery\IsIsDiscovery;
use LibreNMS\Interfaces\Module;
use LibreNMS\Interfaces\Polling\IsIsPolling;
use LibreNMS\OS;
use LibreNMS\Util\IP;
class Isis implements Module
{
use SyncsModels;
protected $isis_codes = [
'l1IntermediateSystem' => 'L1',
'l2IntermediateSystem' => 'L2',
'l1L2IntermediateSystem' => 'L1L2',
'unknown' => 'unknown',
];
/**
* Discover this module. Heavier processes can be run here
* Run infrequently (default 4 times a day)
*
* @param \LibreNMS\OS $os
*/
public function discover(OS $os)
{
$adjacencies = $os instanceof IsIsDiscovery
? $os->discoverIsIs()
: $this->discoverIsIsMib($os);
ModuleModelObserver::observe('\App\Models\IsisAdjacency');
$this->syncModels($os->getDevice(), 'isisAdjacencies', $adjacencies);
}
/**
* Poll data for this module and update the DB / RRD.
* Try to keep this efficient and only run if discovery has indicated there is a reason to run.
* Run frequently (default every 5 minutes)
*
* @param \LibreNMS\OS $os
*/
public function poll(OS $os)
{
$adjacencies = $os->getDevice()->isisAdjacencies;
if (empty($adjacencies)) {
return; // no data to poll
}
$updated = $os instanceof IsIsPolling
? $os->pollIsIs($adjacencies)
: $this->pollIsIsMib($adjacencies, $os);
$updated->each->save();
}
/**
* Remove all DB data for this module.
* This will be run when the module is disabled.
*
* @param Os $os
*/
public function cleanup(OS $os)
{
$os->getDevice()->isisAdjacencies()->delete();
// clean up legacy components from old code
$os->getDevice()->components()->where('type', 'ISIS')->delete();
}
public function discoverIsIsMib(OS $os): Collection
{
// Check if the device has any ISIS enabled interfaces
$circuits = snmpwalk_cache_oid($os->getDeviceArray(), 'ISIS-MIB::isisCirc', []);
$adjacencies = new Collection;
if (! empty($circuits)) {
$adjacencies_data = snmpwalk_cache_twopart_oid($os->getDeviceArray(), 'ISIS-MIB::isisISAdj', [], null, null, '-OQUstx');
$ifIndex_port_id_map = $os->getDevice()->ports()->pluck('port_id', 'ifIndex');
// No ISIS enabled interfaces -> delete the component
foreach ($circuits as $circuit_id => $circuit_data) {
if (! isset($circuit_data['isisCircIfIndex'])) {
continue;
}
if ($circuit_data['isisCircPassiveCircuit'] == 'true') {
continue; // Do not poll passive interfaces
}
$adjacency_data = Arr::last($adjacencies_data[$circuit_id] ?? [[]]);
$attributes = [
'device_id' => $os->getDeviceId(),
'ifIndex' => $circuit_data['isisCircIfIndex'],
'port_id' => $ifIndex_port_id_map[$circuit_data['isisCircIfIndex']] ?? null,
'isisCircAdminState' => $circuit_data['isisCircAdminState'] ?? 'down',
'isisISAdjState' => $adjacency_data['isisISAdjState'] ?? 'down',
];
if (! empty($adjacency_data)) {
$attributes = array_merge($attributes, [
'isisISAdjNeighSysType' => Arr::get($this->isis_codes, $adjacency_data['isisISAdjNeighSysType'] ?? 'unknown', 'unknown'),
'isisISAdjNeighSysID' => str_replace(' ', '.', trim($adjacency_data['isisISAdjNeighSysID'] ?? '')),
'isisISAdjNeighPriority' => $adjacency_data['isisISAdjNeighPriority'] ?? '',
'isisISAdjLastUpTime' => $this->parseAdjacencyTime($adjacency_data),
'isisISAdjAreaAddress' => str_replace(' ', '.', trim($adjacency_data['isisISAdjAreaAddress'] ?? '')),
'isisISAdjIPAddrType' => $adjacency_data['isisISAdjIPAddrType'] ?? '',
'isisISAdjIPAddrAddress' => (string) IP::fromHexstring($adjacency_data['isisISAdjIPAddrAddress'] ?? null, true),
]);
}
$adjacencies->push(new IsisAdjacency($attributes));
}
}
return $adjacencies;
}
public function pollIsIsMib(Collection $adjacencies, OS $os): Collection
{
$data = snmpwalk_cache_twopart_oid($os->getDeviceArray(), 'isisISAdjState', [], 'ISIS-MIB');
if (count($data) !== $adjacencies->where('isisISAdjState', 'up')->count()) {
echo 'New Adjacencies, running discovery';
// don't enable, might be a bad heuristic
return $this->fillNew($adjacencies, $this->discoverIsIsMib($os));
}
$data = snmpwalk_cache_twopart_oid($os->getDeviceArray(), 'isisISAdjLastUpTime', $data, 'ISIS-MIB', null, '-OQUst');
$adjacencies->each(function (IsisAdjacency $adjacency) use (&$data) {
$adjacency_data = Arr::last($data[$adjacency->ifIndex]);
$adjacency->isisISAdjState = $adjacency_data['isisISAdjState'] ?? $adjacency->isisISAdjState;
$adjacency->isisISAdjLastUpTime = $this->parseAdjacencyTime($adjacency_data);
$adjacency->save();
unset($data[$adjacency->ifIndex]);
});
return $adjacencies;
}
protected function parseAdjacencyTime($data): int
{
return (int) max($data['isisISAdjLastUpTime'] ?? 1, 1) / 100;
}
}