2018-12-16 15:18:17 -06:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* TopDevices.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 2018 Tony Murray
|
|
|
|
* @author Tony Murray <murraytony@gmail.com>
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace App\Http\Controllers\Widgets;
|
|
|
|
|
|
|
|
use App\Models\Device;
|
|
|
|
use App\Models\Mempool;
|
|
|
|
use App\Models\Port;
|
|
|
|
use App\Models\Processor;
|
|
|
|
use App\Models\Storage;
|
|
|
|
use Auth;
|
|
|
|
use Carbon\Carbon;
|
|
|
|
use Illuminate\Database\Eloquent\Builder;
|
|
|
|
use Illuminate\Http\Request;
|
|
|
|
use Illuminate\Support\Collection;
|
|
|
|
use Illuminate\View\View;
|
|
|
|
use LibreNMS\Util\Html;
|
|
|
|
use LibreNMS\Util\StringHelpers;
|
|
|
|
use LibreNMS\Util\Url;
|
2021-01-07 12:44:27 +01:00
|
|
|
use LibreNMS\Util\Validate;
|
2018-12-16 15:18:17 -06:00
|
|
|
|
|
|
|
class TopDevicesController extends WidgetController
|
|
|
|
{
|
|
|
|
protected $title = 'Top Devices';
|
|
|
|
protected $defaults = [
|
|
|
|
'title' => null,
|
|
|
|
'top_query' => 'traffic',
|
2020-05-02 12:09:41 -05:00
|
|
|
'sort_order' => 'desc',
|
2018-12-16 15:18:17 -06:00
|
|
|
'device_count' => 5,
|
|
|
|
'time_interval' => 15,
|
2019-08-08 02:59:14 +02:00
|
|
|
'device_group' => null,
|
2018-12-16 15:18:17 -06:00
|
|
|
];
|
|
|
|
|
|
|
|
public function title()
|
|
|
|
{
|
|
|
|
$settings = $this->getSettings();
|
2020-09-21 14:54:51 +02:00
|
|
|
|
2018-12-16 15:18:17 -06:00
|
|
|
return isset($settings['title']) ? $settings['title'] : $this->title;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param Request $request
|
|
|
|
* @return View
|
|
|
|
*/
|
|
|
|
public function getView(Request $request)
|
|
|
|
{
|
|
|
|
$settings = $this->getSettings();
|
|
|
|
$sort = $settings['sort_order'];
|
|
|
|
|
2021-01-07 12:44:27 +01:00
|
|
|
// We use raw() function below, validate input and default to sane value.
|
|
|
|
$sort = Validate::ascDesc($sort, 'ASC');
|
|
|
|
|
2018-12-16 15:18:17 -06:00
|
|
|
switch ($settings['top_query']) {
|
|
|
|
case 'traffic':
|
|
|
|
$data = $this->getTrafficData($sort);
|
|
|
|
break;
|
|
|
|
case 'uptime':
|
|
|
|
$data = $this->getUptimeData($sort);
|
|
|
|
break;
|
|
|
|
case 'ping':
|
|
|
|
$data = $this->getPingData($sort);
|
|
|
|
break;
|
|
|
|
case 'cpu':
|
|
|
|
$data = $this->getProcessorData($sort);
|
|
|
|
break;
|
|
|
|
case 'ram':
|
|
|
|
$data = $this->getMemoryData($sort);
|
|
|
|
break;
|
|
|
|
case 'poller':
|
|
|
|
$data = $this->getPollerData($sort);
|
|
|
|
break;
|
|
|
|
case 'storage':
|
|
|
|
$data = $this->getStorageData($sort);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
$data = [];
|
|
|
|
}
|
|
|
|
|
|
|
|
return view('widgets.top-devices', $data);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getSettingsView(Request $request)
|
|
|
|
{
|
2019-08-08 02:59:14 +02:00
|
|
|
return view('widgets.settings.top-devices', $this->getSettings(true));
|
2018-12-16 15:18:17 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param array|string $headers
|
|
|
|
* @param Collection $rows
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
private function formatData($headers, $rows)
|
|
|
|
{
|
|
|
|
return [
|
2020-09-21 14:54:51 +02:00
|
|
|
'headers' => (array) $headers,
|
2018-12-16 15:18:17 -06:00
|
|
|
'rows' => $rows,
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param Builder $query
|
|
|
|
* @param string $left_table
|
|
|
|
* @return Builder
|
|
|
|
*/
|
|
|
|
private function withDeviceQuery($query, $left_table)
|
|
|
|
{
|
|
|
|
$settings = $this->getSettings();
|
|
|
|
|
|
|
|
/** @var Builder $query */
|
|
|
|
return $query->with(['device' => function ($query) {
|
2020-03-23 14:45:20 +01:00
|
|
|
$query->select('device_id', 'hostname', 'sysName', 'status', 'os');
|
2018-12-16 15:18:17 -06:00
|
|
|
}])
|
|
|
|
->select("$left_table.device_id")
|
|
|
|
->leftJoin('devices', "$left_table.device_id", 'devices.device_id')
|
|
|
|
->groupBy("$left_table.device_id")
|
2019-08-08 02:59:14 +02:00
|
|
|
->where('devices.last_polled', '>', Carbon::now()->subMinutes($settings['time_interval']))
|
|
|
|
->when($settings['device_group'], function ($query) use ($settings) {
|
|
|
|
$query->inDeviceGroup($settings['device_group']);
|
|
|
|
});
|
2018-12-16 15:18:17 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param Builder $query
|
|
|
|
* @return Builder
|
|
|
|
*/
|
|
|
|
private function deviceQuery()
|
|
|
|
{
|
|
|
|
$settings = $this->getSettings();
|
|
|
|
|
2020-03-23 14:45:20 +01:00
|
|
|
return Device::hasAccess(Auth::user())->select('device_id', 'hostname', 'sysName', 'status', 'os')
|
2018-12-16 15:18:17 -06:00
|
|
|
->where('devices.last_polled', '>', Carbon::now()->subMinutes($settings['time_interval']))
|
2019-08-08 02:59:14 +02:00
|
|
|
->when($settings['device_group'], function ($query) use ($settings) {
|
|
|
|
$query->inDeviceGroup($settings['device_group']);
|
|
|
|
})
|
2018-12-16 15:18:17 -06:00
|
|
|
->limit($settings['device_count']);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param Device $device
|
|
|
|
* @param string $graph_type
|
|
|
|
* @param array $graph_params
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
private function standardRow($device, $graph_type, $graph_params = [])
|
|
|
|
{
|
|
|
|
return [
|
|
|
|
Url::deviceLink($device, $device->shortDisplayName()),
|
|
|
|
Url::deviceLink($device, Url::minigraphImage(
|
|
|
|
$device,
|
|
|
|
Carbon::now()->subDays(1)->timestamp,
|
|
|
|
Carbon::now()->timestamp,
|
|
|
|
$graph_type,
|
|
|
|
'no',
|
|
|
|
150,
|
|
|
|
21
|
|
|
|
), $graph_params, 0, 0, 0),
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
private function getTrafficData($sort)
|
|
|
|
{
|
|
|
|
$settings = $this->getSettings();
|
|
|
|
|
|
|
|
/** @var Builder $query */
|
|
|
|
$query = Port::hasAccess(Auth::user())->with(['device' => function ($query) {
|
2020-03-23 14:45:20 +01:00
|
|
|
$query->select('device_id', 'hostname', 'sysName', 'status', 'os');
|
2018-12-16 15:18:17 -06:00
|
|
|
}])
|
|
|
|
->select('device_id')
|
|
|
|
->groupBy('device_id')
|
|
|
|
->where('poll_time', '>', Carbon::now()->subMinutes($settings['time_interval'])->timestamp)
|
2019-08-08 02:59:14 +02:00
|
|
|
->when($settings['device_group'], function ($query) use ($settings) {
|
|
|
|
$query->inDeviceGroup($settings['device_group']);
|
|
|
|
}, function ($query) {
|
|
|
|
$query->has('device');
|
|
|
|
})
|
2018-12-16 15:18:17 -06:00
|
|
|
->orderByRaw('SUM(ifInOctets_rate + ifOutOctets_rate) ' . $sort)
|
|
|
|
->limit($settings['device_count']);
|
|
|
|
|
|
|
|
$results = $query->get()->map(function ($port) {
|
|
|
|
return $this->standardRow($port->device, 'device_bits');
|
|
|
|
});
|
|
|
|
|
|
|
|
return $this->formatData('Traffic', $results);
|
|
|
|
}
|
|
|
|
|
|
|
|
private function getUptimeData($sort)
|
|
|
|
{
|
2018-12-20 15:57:28 -06:00
|
|
|
$settings = $this->getSettings();
|
|
|
|
|
2018-12-16 15:18:17 -06:00
|
|
|
/** @var Builder $query */
|
2018-12-20 15:57:28 -06:00
|
|
|
$query = $this->deviceQuery()->orderBy('uptime', $sort)->limit($settings['device_count']);
|
2018-12-16 15:18:17 -06:00
|
|
|
|
|
|
|
$results = $query->get()->map(function ($device) {
|
|
|
|
return $this->standardRow($device, 'device_uptime', ['tab' => 'graphs', 'group' => 'system']);
|
|
|
|
});
|
|
|
|
|
|
|
|
return $this->formatData('Uptime', $results);
|
|
|
|
}
|
|
|
|
|
|
|
|
private function getPingData($sort)
|
|
|
|
{
|
2018-12-20 15:57:28 -06:00
|
|
|
$settings = $this->getSettings();
|
|
|
|
|
2018-12-16 15:18:17 -06:00
|
|
|
/** @var Builder $query */
|
2018-12-20 15:57:28 -06:00
|
|
|
$query = $this->deviceQuery()->orderBy('last_ping_timetaken', $sort)->limit($settings['device_count']);
|
2018-12-16 15:18:17 -06:00
|
|
|
|
|
|
|
$results = $query->get()->map(function ($device) {
|
|
|
|
return $this->standardRow($device, 'device_ping_perf', ['tab' => 'graphs', 'group' => 'poller']);
|
|
|
|
});
|
|
|
|
|
|
|
|
return $this->formatData('Response time', $results);
|
|
|
|
}
|
|
|
|
|
|
|
|
private function getProcessorData($sort)
|
|
|
|
{
|
2018-12-20 15:57:28 -06:00
|
|
|
$settings = $this->getSettings();
|
|
|
|
|
2018-12-16 15:18:17 -06:00
|
|
|
/** @var Builder $query */
|
|
|
|
$query = $this->withDeviceQuery(Processor::hasAccess(Auth::user()), (new Processor)->getTable())
|
2018-12-20 15:57:28 -06:00
|
|
|
->orderByRaw('AVG(`processor_usage`) ' . $sort)
|
|
|
|
->limit($settings['device_count']);
|
2018-12-16 15:18:17 -06:00
|
|
|
|
|
|
|
$results = $query->get()->map(function ($port) {
|
|
|
|
return $this->standardRow($port->device, 'device_processor', ['tab' => 'health', 'metric' => 'processor']);
|
|
|
|
});
|
|
|
|
|
|
|
|
return $this->formatData('CPU Load', $results);
|
|
|
|
}
|
|
|
|
|
|
|
|
private function getMemoryData($sort)
|
|
|
|
{
|
2018-12-20 15:57:28 -06:00
|
|
|
$settings = $this->getSettings();
|
|
|
|
|
2018-12-16 15:18:17 -06:00
|
|
|
/** @var Builder $query */
|
|
|
|
$query = $this->withDeviceQuery(Mempool::hasAccess(Auth::user()), (new Mempool)->getTable())
|
2018-12-20 15:57:28 -06:00
|
|
|
->orderBy('mempool_perc', $sort)
|
|
|
|
->limit($settings['device_count']);
|
2018-12-16 15:18:17 -06:00
|
|
|
|
|
|
|
$results = $query->get()->map(function ($port) {
|
|
|
|
return $this->standardRow($port->device, 'device_mempool', ['tab' => 'health', 'metric' => 'mempool']);
|
|
|
|
});
|
|
|
|
|
|
|
|
return $this->formatData('Memory usage', $results);
|
|
|
|
}
|
|
|
|
|
|
|
|
private function getPollerData($sort)
|
|
|
|
{
|
2018-12-20 15:57:28 -06:00
|
|
|
$settings = $this->getSettings();
|
|
|
|
|
2018-12-16 15:18:17 -06:00
|
|
|
/** @var Builder $query */
|
2018-12-20 15:57:28 -06:00
|
|
|
$query = $this->deviceQuery()->orderBy('last_polled_timetaken', $sort)->limit($settings['device_count']);
|
2018-12-16 15:18:17 -06:00
|
|
|
|
|
|
|
$results = $query->get()->map(function ($device) {
|
|
|
|
return $this->standardRow($device, 'device_poller_perf', ['tab' => 'graphs', 'group' => 'poller']);
|
|
|
|
});
|
|
|
|
|
|
|
|
return $this->formatData('Poller duration', $results);
|
|
|
|
}
|
|
|
|
|
|
|
|
private function getStorageData($sort)
|
|
|
|
{
|
|
|
|
$settings = $this->getSettings();
|
|
|
|
|
|
|
|
/** @var Builder $query */
|
|
|
|
$query = Storage::hasAccess(Auth::user())->with(['device' => function ($query) {
|
2020-03-23 14:45:20 +01:00
|
|
|
$query->select('device_id', 'hostname', 'sysName', 'status', 'os');
|
2018-12-16 15:18:17 -06:00
|
|
|
}])
|
|
|
|
->leftJoin('devices', 'storage.device_id', 'devices.device_id')
|
|
|
|
->select('storage.device_id', 'storage_id', 'storage_descr', 'storage_perc', 'storage_perc_warn')
|
|
|
|
->where('devices.last_polled', '>', Carbon::now()->subMinutes($settings['time_interval']))
|
2019-08-08 02:59:14 +02:00
|
|
|
->when($settings['device_group'], function ($query) use ($settings) {
|
|
|
|
$query->inDeviceGroup($settings['device_group']);
|
|
|
|
})
|
2018-12-16 15:18:17 -06:00
|
|
|
->orderBy('storage_perc', $sort)
|
|
|
|
->limit($settings['device_count']);
|
|
|
|
|
|
|
|
$results = $query->get()->map(function ($storage) {
|
|
|
|
$device = $storage->device;
|
|
|
|
|
|
|
|
$graph_array = [
|
|
|
|
'height' => 100,
|
|
|
|
'width' => 210,
|
|
|
|
'to' => Carbon::now()->timestamp,
|
|
|
|
'from' => Carbon::now()->subDay()->timestamp,
|
|
|
|
'id' => $storage->storage_id,
|
|
|
|
'type' => 'device_storage',
|
|
|
|
'legend' => 'no',
|
|
|
|
];
|
|
|
|
$overlib_content = Url::overlibContent($graph_array, $device->displayName() . ' - ' . $storage->storage_descr);
|
|
|
|
|
|
|
|
$link_array = $graph_array;
|
|
|
|
$link_array['page'] = 'graphs';
|
|
|
|
unset($link_array['height'], $link_array['width'], $link_array['legend']);
|
|
|
|
$link = Url::generate($link_array);
|
|
|
|
|
|
|
|
return [
|
|
|
|
Url::deviceLink($device, $device->shortDisplayName()),
|
|
|
|
StringHelpers::shortenText($storage->storage_descr, 50),
|
|
|
|
Url::overlibLink(
|
|
|
|
$link,
|
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
|
|
|
Html::percentageBar(150, 20, $storage->storage_perc, '', $storage->storage_perc . '%', $storage->storage_perc_warn),
|
2018-12-16 15:18:17 -06:00
|
|
|
$overlib_content
|
2020-09-21 14:54:51 +02:00
|
|
|
),
|
2018-12-16 15:18:17 -06:00
|
|
|
];
|
|
|
|
});
|
|
|
|
|
|
|
|
return $this->formatData(['Storage Device', 'Disk usage'], $results);
|
|
|
|
}
|
|
|
|
}
|