Files
librenms-librenms/LibreNMS/Data/Source/SnmpResponse.php
T
Tony MurrayandGitHub a828bb00c3 lnms snmp:fetch command and new SNMP code (#13303)
* SNMP WIP

* cleanup, more types

* Include my snmp:fetch command

* Fix Facade name conflict

* Command WIP
remove mib

* Ignore exit code
cleanups

* Doc blocks and style fixes

* forgot to use parseOid

* Hopefully final fixes

* missed on (:

* small changes
deviates from existing code, hopefully doesn't re-add too many corner cases.

* add some simple tests, will make it easier to add more in the future when we find corner cases.

* test numeric

* API refinements, try to avoid setting textual net-snmp options directly

* change numeric to a toggle makes for nicer usage

* make ci happy

* Some errors happen only in stderr, pass that to SnmpResponse for parsing.
Add error message access

* More consistent naming
2021-10-01 18:58:12 -05:00

161 lines
4.5 KiB
PHP

<?php
/*
* SnmpResponse.php
*
* Responsible for parsing net-snmp output into usable PHP data structures.
*
* 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/>.
*
* @package LibreNMS
* @link http://librenms.org
* @copyright 2021 Tony Murray
* @author Tony Murray <[email protected]>
*/
namespace LibreNMS\Data\Source;
use Illuminate\Support\Arr;
use Illuminate\Support\Str;
use Log;
class SnmpResponse
{
const DELIMITER = ' = ';
/**
* @var string
*/
private $raw;
/**
* @var int
*/
private $exitCode;
/**
* @var string
*/
private $errorMessage;
/**
* @var string
*/
private $stderr;
/**
* Create a new response object filling with output from the net-snmp command.
*
* @param string $output
* @param string $errorOutput
* @param int $exitCode
*/
public function __construct(string $output, string $errorOutput = '', int $exitCode = 0)
{
$this->exitCode = $exitCode;
$this->raw = preg_replace('/Wrong Type \(should be .*\): /', '', $output);
$this->stderr = $errorOutput;
}
public function isValid(): bool
{
$this->errorMessage = '';
// not checking exitCode because I think it may lead to false negatives
$invalid = preg_match('/(Timeout: No Response from .*|Unknown user name|Authentication failure)/', $this->stderr, $errors)
|| empty($this->raw)
|| preg_match('/(No Such Instance|No Such Object|No more variables left).*/', $this->raw, $errors);
if ($invalid) {
$this->errorMessage = $errors[0] ?? 'Empty Output';
Log::debug(sprintf('SNMP query failed. Exit Code: %s Empty: %s Bad String: %s', $this->exitCode, var_export(empty($this->raw), true), $errors[0] ?? 'not found'));
return false;
}
return true;
}
/**
* Get the error message if any
*/
public function getErrorMessage(): string
{
if (empty($this->errorMessage)) {
$this->isValid(); // if no error message, double check.
}
return $this->errorMessage;
}
/**
* Get the raw output returned by net-snmp
*/
public function raw(): string
{
return $this->raw;
}
public function value(): string
{
return Arr::first($this->values(), null, '');
}
public function values(): array
{
$values = [];
$line = strtok($this->raw, PHP_EOL);
while ($line !== false) {
if (Str::contains($line, ['at this OID', 'this MIB View'])) {
// these occur when we seek past the end of data, usually the end of the response, but grab the next line and continue
$line = strtok(PHP_EOL);
continue;
}
[$oid, $value] = explode(self::DELIMITER, $line, 2);
$line = strtok(PHP_EOL); // get the next line and concatenate multi-line values
while ($line !== false && ! Str::contains($line, self::DELIMITER)) {
$value .= PHP_EOL . $line;
$line = strtok(PHP_EOL);
}
$values[$oid] = $value;
}
return $values;
}
public function table(int $group = 0, array &$array = []): array
{
foreach ($this->values() as $key => $value) {
preg_match_all('/([^[\]]+)/', $key, $parts);
$parts = $parts[1];
array_splice($parts, $group, 0, array_shift($parts)); // move the oid name to the correct depth
// merge the parts into an array, creating keys if they don't exist
$tmp = &$array;
foreach ($parts as $part) {
$key = trim($part, '"');
$tmp = &$tmp[$key];
}
$tmp = $value; // assign the value as the leaf
}
return $array;
}
/**
* @return int
*/
public function getExitCode(): int
{
return $this->exitCode;
}
}