Files
librenms-librenms/app/Http/Controllers/Widgets/TopDevicesController.php
T

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

335 lines
11 KiB
PHP
Raw Normal View History

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
2021-02-09 00:29:04 +01:00
* along with this program. If not, see <https://www.gnu.org/licenses/>.
2018-12-16 15:18:17 -06:00
*
2021-02-09 00:29:04 +01:00
* @link https://www.librenms.org
2021-09-10 20:09:53 +02:00
*
2018-12-16 15:18:17 -06:00
* @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 Carbon\Carbon;
use Illuminate\Database\Eloquent\Builder;
use Illuminate\Http\Request;
use Illuminate\Support\Collection;
use Illuminate\Support\Facades\Auth;
2018-12-16 15:18:17 -06:00
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',
'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;
}
/**
2021-09-08 23:35:56 +02:00
* @param Request $request
2018-12-16 15:18:17 -06:00
* @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
}
/**
2021-09-08 23:35:56 +02:00
* @param array|string $headers
* @param Collection $rows
2018-12-16 15:18:17 -06:00
* @return array
*/
private function formatData($headers, $rows)
{
return [
'headers' => (array) $headers,
'rows' => $rows,
];
}
/**
2021-09-08 23:35:56 +02:00
* @param Builder $query
* @param string $left_table
2018-12-16 15:18:17 -06:00
* @return Builder
*/
2021-03-24 15:13:43 +01:00
private function withDeviceQuery(Builder $query, $left_table)
2018-12-16 15:18:17 -06:00
{
$settings = $this->getSettings();
/** @var Builder $query */
return $query->with(['device' => function ($query) {
return $query->select('device_id', 'hostname', 'sysName', 'display', '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) {
2021-03-24 15:13:43 +01:00
/** @var Builder<\App\Models\DeviceRelatedModel> $query */
2021-07-13 16:35:43 -05:00
$inDeviceGroup = $query->inDeviceGroup($settings['device_group']); /** @var Builder $inDeviceGroup */
return $inDeviceGroup;
2019-08-08 02:59:14 +02:00
});
2018-12-16 15:18:17 -06:00
}
/**
* @return Builder
*/
private function deviceQuery()
{
$settings = $this->getSettings();
return Device::hasAccess(Auth::user())->select('device_id', 'hostname', 'sysName', 'display', '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) {
2021-07-13 16:35:43 -05:00
return $query->inDeviceGroup($settings['device_group']);
2019-08-08 02:59:14 +02:00
})
2018-12-16 15:18:17 -06:00
->limit($settings['device_count']);
}
/**
2021-09-08 23:35:56 +02:00
* @param Device $device
* @param string $graph_type
* @param array $graph_params
2018-12-16 15:18:17 -06:00
* @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();
$query = Port::hasAccess(Auth::user())->with(['device' => function ($query) {
$query->select('device_id', 'hostname', 'sysName', 'display', '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) {
2021-07-13 16:35:43 -05:00
return $query->inDeviceGroup($settings['device_group']);
2019-08-08 02:59:14 +02:00
}, function ($query) {
2021-07-13 16:35:43 -05:00
return $query->has('device');
2019-08-08 02:59:14 +02:00
})
2022-02-12 23:39:17 +01:00
->where('ifOperStatus', 'up')
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) {
2021-04-08 15:14:49 +02:00
/** @var Device $device */
2018-12-16 15:18:17 -06:00
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) {
2021-04-08 15:14:49 +02:00
/** @var Device $device */
2024-04-18 09:57:01 -05:00
return $this->standardRow($device, 'device_icmp_perf', ['tab' => 'graphs', 'group' => 'poller']);
2018-12-16 15:18:17 -06:00
});
return $this->formatData('Response time', $results);
}
private function getProcessorData($sort)
{
2018-12-20 15:57:28 -06:00
$settings = $this->getSettings();
2021-03-30 11:16:44 +02:00
/** @var Processor $query */
2018-12-16 15:18:17 -06:00
$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();
2021-03-30 11:16:44 +02:00
/** @var Mempool $query */
2018-12-16 15:18:17 -06:00
$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();
$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) {
2021-04-08 15:14:49 +02:00
/** @var Device $device */
2018-12-16 15:18:17 -06:00
return $this->standardRow($device, 'device_poller_perf', ['tab' => 'graphs', 'group' => 'poller']);
});
return $this->formatData('Poller duration', $results);
}
private function getStorageData($sort)
{
$settings = $this->getSettings();
$query = Storage::hasAccess(Auth::user())->with(['device' => function ($query) {
$query->select('device_id', 'hostname', 'sysName', 'display', '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' => 'storage_usage',
2018-12-16 15:18:17 -06:00
'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,
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
),
];
});
return $this->formatData(['Storage Device', 'Disk usage'], $results);
}
}