2016-08-08 12:48:16 +00:00
|
|
|
<?php
|
|
|
|
|
2020-02-13 23:08:46 -03:30
|
|
|
require_once 'config.php';
|
|
|
|
|
2016-08-08 12:48:16 +00:00
|
|
|
/**
|
2018-10-19 23:36:40 +02:00
|
|
|
* Show dashboard page.
|
2016-08-08 12:48:16 +00:00
|
|
|
*/
|
2020-02-02 08:37:15 +00:00
|
|
|
function DisplayDashboard(&$extraFooterScripts)
|
2019-04-10 08:37:35 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
$status = new StatusMessages();
|
|
|
|
// Need this check interface name for proper shell execution.
|
|
|
|
if (!preg_match('/^([a-zA-Z0-9]+)$/', RASPI_WIFI_CLIENT_INTERFACE)) {
|
|
|
|
$status->addMessage(_('Interface name invalid.'), 'danger');
|
|
|
|
$status->showMessages();
|
|
|
|
return;
|
2018-10-23 11:57:55 +02:00
|
|
|
}
|
2019-04-10 08:37:35 +00:00
|
|
|
|
|
|
|
if (!function_exists('exec')) {
|
|
|
|
$status->addMessage(_('Required exec function is disabled. Check if exec is not added to php disable_functions.'), 'danger');
|
|
|
|
$status->showMessages();
|
|
|
|
return;
|
2018-10-23 11:57:55 +02:00
|
|
|
}
|
2018-10-19 23:36:40 +02:00
|
|
|
|
2019-04-10 08:37:35 +00:00
|
|
|
exec('ip a show '.RASPI_WIFI_CLIENT_INTERFACE, $stdoutIp);
|
|
|
|
$stdoutIpAllLinesGlued = implode(" ", $stdoutIp);
|
|
|
|
$stdoutIpWRepeatedSpaces = preg_replace('/\s\s+/', ' ', $stdoutIpAllLinesGlued);
|
2018-10-19 23:36:40 +02:00
|
|
|
|
2019-04-10 08:37:35 +00:00
|
|
|
preg_match('/link\/ether ([0-9a-f:]+)/i', $stdoutIpWRepeatedSpaces, $matchesMacAddr) || $matchesMacAddr[1] = _('No MAC Address Found');
|
|
|
|
$macAddr = $matchesMacAddr[1];
|
2018-10-19 23:36:40 +02:00
|
|
|
|
2019-04-10 08:37:35 +00:00
|
|
|
$ipv4Addrs = '';
|
|
|
|
$ipv4Netmasks = '';
|
2019-07-30 23:15:08 +02:00
|
|
|
if (!preg_match_all('/inet (\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3})\/([0-3][0-9])/i', $stdoutIpWRepeatedSpaces, $matchesIpv4AddrAndSubnet, PREG_SET_ORDER)) {
|
2019-04-10 08:37:35 +00:00
|
|
|
$ipv4Addrs = _('No IPv4 Address Found');
|
|
|
|
} else {
|
2019-07-30 23:15:08 +02:00
|
|
|
foreach ($matchesIpv4AddrAndSubnet as $inet) {
|
2019-09-07 16:42:31 +01:00
|
|
|
$address = $inet[1];
|
|
|
|
$suffix = (int) $inet[2];
|
|
|
|
$netmask = long2ip(-1 << (32 - $suffix));
|
2019-07-30 23:15:08 +02:00
|
|
|
|
2019-09-07 16:42:31 +01:00
|
|
|
$ipv4Addrs .= " $address";
|
|
|
|
$ipv4Netmasks .= " $netmask";
|
2019-04-10 08:37:35 +00:00
|
|
|
}
|
2019-07-30 23:15:08 +02:00
|
|
|
$ipv4Addrs = trim($ipv4Addrs);
|
|
|
|
$ipv4Netmasks = trim($ipv4Netmasks);
|
2019-04-10 08:37:35 +00:00
|
|
|
}
|
2019-11-03 09:45:55 +00:00
|
|
|
$ipv4Netmasks = empty($ipv4Netmasks) ? "-" : $ipv4Netmasks;
|
2018-10-19 23:36:40 +02:00
|
|
|
|
2019-04-10 08:37:35 +00:00
|
|
|
$ipv6Addrs = '';
|
|
|
|
if (!preg_match_all('/inet6 ([a-f0-9:]+)/i', $stdoutIpWRepeatedSpaces, $matchesIpv6Addr)) {
|
|
|
|
$ipv6Addrs = _('No IPv6 Address Found');
|
|
|
|
} else {
|
2019-07-30 22:24:01 +02:00
|
|
|
if (isset($matchesIpv6Addr[1])) {
|
|
|
|
$ipv6Addrs = implode(' ', $matchesIpv6Addr[1]);
|
2019-04-10 08:37:35 +00:00
|
|
|
}
|
|
|
|
}
|
2018-10-19 23:36:40 +02:00
|
|
|
|
2019-04-10 08:37:35 +00:00
|
|
|
preg_match('/state (UP|DOWN)/i', $stdoutIpWRepeatedSpaces, $matchesState) || $matchesState[1] = 'unknown';
|
|
|
|
$interfaceState = $matchesState[1];
|
2018-10-19 23:36:40 +02:00
|
|
|
|
2019-04-10 08:37:35 +00:00
|
|
|
// Because of table layout used in the ip output we get the interface statistics directly from
|
|
|
|
// the system. One advantage of this is that it could work when interface is disable.
|
|
|
|
exec('cat /sys/class/net/'.RASPI_WIFI_CLIENT_INTERFACE.'/statistics/rx_packets ', $stdoutCatRxPackets);
|
|
|
|
$strRxPackets = _('No data');
|
|
|
|
if (ctype_digit($stdoutCatRxPackets[0])) {
|
|
|
|
$strRxPackets = $stdoutCatRxPackets[0];
|
|
|
|
}
|
2018-10-19 23:36:40 +02:00
|
|
|
|
2019-04-10 08:37:35 +00:00
|
|
|
exec('cat /sys/class/net/'.RASPI_WIFI_CLIENT_INTERFACE.'/statistics/tx_packets ', $stdoutCatTxPackets);
|
|
|
|
$strTxPackets = _('No data');
|
|
|
|
if (ctype_digit($stdoutCatTxPackets[0])) {
|
|
|
|
$strTxPackets = $stdoutCatTxPackets[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
exec('cat /sys/class/net/'.RASPI_WIFI_CLIENT_INTERFACE.'/statistics/rx_bytes ', $stdoutCatRxBytes);
|
|
|
|
$strRxBytes = _('No data');
|
|
|
|
if (ctype_digit($stdoutCatRxBytes[0])) {
|
|
|
|
$strRxBytes = $stdoutCatRxBytes[0];
|
|
|
|
$strRxBytes .= getHumanReadableDatasize($strRxBytes);
|
|
|
|
}
|
|
|
|
|
|
|
|
exec('cat /sys/class/net/'.RASPI_WIFI_CLIENT_INTERFACE.'/statistics/tx_bytes ', $stdoutCatTxBytes);
|
|
|
|
$strTxBytes = _('No data');
|
|
|
|
if (ctype_digit($stdoutCatTxBytes[0])) {
|
|
|
|
$strTxBytes = $stdoutCatTxBytes[0];
|
|
|
|
$strTxBytes .= getHumanReadableDatasize($strTxBytes);
|
2018-10-19 23:36:40 +02:00
|
|
|
}
|
|
|
|
|
2019-04-10 08:37:35 +00:00
|
|
|
define('SSIDMAXLEN', 32);
|
|
|
|
// Warning iw comes with: "Do NOT screenscrape this tool, we don't consider its output stable."
|
|
|
|
exec('iw dev '.RASPI_WIFI_CLIENT_INTERFACE.' link ', $stdoutIw);
|
|
|
|
$stdoutIwAllLinesGlued = implode(' ', $stdoutIw);
|
|
|
|
$stdoutIwWRepSpaces = preg_replace('/\s\s+/', ' ', $stdoutIwAllLinesGlued);
|
2016-08-08 12:48:16 +00:00
|
|
|
|
2019-04-10 08:37:35 +00:00
|
|
|
preg_match('/Connected to (([0-9A-Fa-f]{2}:){5}([0-9A-Fa-f]{2}))/', $stdoutIwWRepSpaces, $matchesBSSID) || $matchesBSSID[1] = '';
|
|
|
|
$connectedBSSID = $matchesBSSID[1];
|
2019-11-03 09:45:55 +00:00
|
|
|
$connectedBSSID = empty($connectedBSSID) ? "-" : $connectedBSSID;
|
2018-10-19 23:36:40 +02:00
|
|
|
|
2019-04-10 08:37:35 +00:00
|
|
|
$wlanHasLink = false;
|
2018-10-20 14:38:27 +02:00
|
|
|
if ($interfaceState === 'UP') {
|
2019-04-10 08:37:35 +00:00
|
|
|
$wlanHasLink = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!preg_match('/SSID: ([^ ]{1,'.SSIDMAXLEN.'})/', $stdoutIwWRepSpaces, $matchesSSID)) {
|
|
|
|
$wlanHasLink = false;
|
2020-02-03 09:28:06 +00:00
|
|
|
$matchesSSID[1] = 'None';
|
2019-04-10 08:37:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
$connectedSSID = $matchesSSID[1];
|
|
|
|
|
|
|
|
preg_match('/freq: (\d+)/i', $stdoutIwWRepSpaces, $matchesFrequency) || $matchesFrequency[1] = '';
|
|
|
|
$frequency = $matchesFrequency[1].' MHz';
|
|
|
|
|
|
|
|
preg_match('/signal: (-?[0-9]+ dBm)/i', $stdoutIwWRepSpaces, $matchesSignal) || $matchesSignal[1] = '';
|
|
|
|
$signalLevel = $matchesSignal[1];
|
2019-11-03 09:45:55 +00:00
|
|
|
$signalLevel = empty($signalLevel) ? "-" : $signalLevel;
|
2019-04-10 08:37:35 +00:00
|
|
|
|
|
|
|
preg_match('/tx bitrate: ([0-9\.]+ [KMGT]?Bit\/s)/', $stdoutIwWRepSpaces, $matchesBitrate) || $matchesBitrate[1] = '';
|
|
|
|
$bitrate = $matchesBitrate[1];
|
2019-11-03 09:45:55 +00:00
|
|
|
$bitrate = empty($bitrate) ? "-" : $bitrate;
|
2019-04-10 08:37:35 +00:00
|
|
|
|
|
|
|
// txpower is now displayed on iw dev(..) info command, not on link command.
|
|
|
|
exec('iw dev '.RASPI_WIFI_CLIENT_INTERFACE.' info ', $stdoutIwInfo);
|
|
|
|
$stdoutIwInfoAllLinesGlued = implode(' ', $stdoutIwInfo);
|
|
|
|
$stdoutIpInfoWRepSpaces = preg_replace('/\s\s+/', ' ', $stdoutIwInfoAllLinesGlued);
|
|
|
|
|
|
|
|
preg_match('/txpower ([0-9\.]+ dBm)/i', $stdoutIpInfoWRepSpaces, $matchesTxPower) || $matchesTxPower[1] = '';
|
|
|
|
$txPower = $matchesTxPower[1];
|
|
|
|
|
|
|
|
// iw does not have the "Link Quality". This is a is an aggregate value,
|
|
|
|
// and depends on the driver and hardware.
|
|
|
|
// Display link quality as signal quality for now.
|
|
|
|
$strLinkQuality = 0;
|
|
|
|
if ($signalLevel > -100 && $wlanHasLink) {
|
|
|
|
if ($signalLevel >= 0) {
|
|
|
|
$strLinkQuality = 100;
|
|
|
|
} else {
|
|
|
|
$strLinkQuality = 100 + $signalLevel;
|
|
|
|
}
|
2016-08-08 21:37:44 +01:00
|
|
|
}
|
2019-04-10 08:37:35 +00:00
|
|
|
|
|
|
|
$wlan0up = false;
|
|
|
|
$classMsgDevicestatus = 'warning';
|
|
|
|
if ($interfaceState === 'UP') {
|
|
|
|
$wlan0up = true;
|
|
|
|
$classMsgDevicestatus = 'success';
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-02-13 23:08:46 -03:30
|
|
|
if (!RASPI_MONITOR_ENABLED) {
|
|
|
|
if (isset($_POST['ifdown_wlan0'])) {
|
|
|
|
// Pressed stop button
|
|
|
|
if ($interfaceState === 'UP') {
|
|
|
|
$status->addMessage(sprintf(_('Interface is going %s.'), _('down')), 'warning');
|
|
|
|
exec('sudo ip link set '.RASPI_WIFI_CLIENT_INTERFACE.' down');
|
|
|
|
$wlan0up = false;
|
|
|
|
$status->addMessage(sprintf(_('Interface is now %s.'), _('down')), 'success');
|
|
|
|
} elseif ($interfaceState === 'unknown') {
|
|
|
|
$status->addMessage(_('Interface state unknown.'), 'danger');
|
|
|
|
} else {
|
|
|
|
$status->addMessage(sprintf(_('Interface already %s.'), _('down')), 'warning');
|
|
|
|
}
|
|
|
|
} elseif (isset($_POST['ifup_wlan0'])) {
|
|
|
|
// Pressed start button
|
|
|
|
if ($interfaceState === 'DOWN') {
|
|
|
|
$status->addMessage(sprintf(_('Interface is going %s.'), _('up')), 'warning');
|
|
|
|
exec('sudo ip link set ' . RASPI_WIFI_CLIENT_INTERFACE . ' up');
|
|
|
|
exec('sudo ip -s a f label ' . RASPI_WIFI_CLIENT_INTERFACE);
|
|
|
|
$wlan0up = true;
|
|
|
|
$status->addMessage(sprintf(_('Interface is now %s.'), _('up')), 'success');
|
|
|
|
} elseif ($interfaceState === 'unknown') {
|
|
|
|
$status->addMessage(_('Interface state unknown.'), 'danger');
|
|
|
|
} else {
|
|
|
|
$status->addMessage(sprintf(_('Interface already %s.'), _('up')), 'warning');
|
|
|
|
}
|
2019-04-10 08:37:35 +00:00
|
|
|
} else {
|
2020-02-13 23:08:46 -03:30
|
|
|
$status->addMessage(sprintf(_('Interface is %s.'), strtolower($interfaceState)), $classMsgDevicestatus);
|
2019-04-10 08:37:35 +00:00
|
|
|
}
|
2016-08-08 21:37:44 +01:00
|
|
|
}
|
2016-08-08 12:48:16 +00:00
|
|
|
|
2019-08-19 00:04:53 +01:00
|
|
|
echo renderTemplate("dashboard", compact(
|
|
|
|
"status",
|
2019-09-07 16:42:31 +01:00
|
|
|
"ipv4Addrs",
|
|
|
|
"ipv4Netmasks",
|
|
|
|
"ipv6Addrs",
|
|
|
|
"macAddr",
|
|
|
|
"strRxPackets",
|
|
|
|
"strRxBytes",
|
|
|
|
"strTxPackets",
|
|
|
|
"strTxBytes",
|
|
|
|
"connectedSSID",
|
|
|
|
"connectedBSSID",
|
|
|
|
"bitrate",
|
|
|
|
"signalLevel",
|
|
|
|
"txPower",
|
|
|
|
"frequency",
|
|
|
|
"strLinkQuality",
|
2019-08-19 00:04:53 +01:00
|
|
|
"wlan0up"
|
|
|
|
));
|
2020-02-02 08:37:15 +00:00
|
|
|
$extraFooterScripts[] = array('src'=>'app/js/dashboardchart.js', 'defer'=>false);
|
2016-08-08 12:48:16 +00:00
|
|
|
}
|
|
|
|
|
2018-10-19 23:36:40 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a human readable data size string from a number of bytes.
|
|
|
|
*
|
|
|
|
* @param long $numbytes The number of bytes.
|
|
|
|
* @param int $precision The number of numbers to round to after the dot/comma.
|
|
|
|
* @return string Data size in units: PB, TB, GB, MB or KB otherwise an empty string.
|
|
|
|
*/
|
|
|
|
function getHumanReadableDatasize($numbytes, $precision = 2)
|
|
|
|
{
|
2019-04-10 08:37:35 +00:00
|
|
|
$humanDatasize = '';
|
|
|
|
$kib = 1024;
|
|
|
|
$mib = $kib * 1024;
|
|
|
|
$gib = $mib * 1024;
|
|
|
|
$tib = $gib * 1024;
|
|
|
|
$pib = $tib * 1024;
|
|
|
|
if ($numbytes >= $pib) {
|
|
|
|
$humanDatasize = ' ('.round($numbytes / $pib, $precision).' PB)';
|
|
|
|
} elseif ($numbytes >= $tib) {
|
|
|
|
$humanDatasize = ' ('.round($numbytes / $tib, $precision).' TB)';
|
|
|
|
} elseif ($numbytes >= $gib) {
|
|
|
|
$humanDatasize = ' ('.round($numbytes / $gib, $precision).' GB)';
|
|
|
|
} elseif ($numbytes >= $mib) {
|
|
|
|
$humanDatasize = ' ('.round($numbytes / $mib, $precision).' MB)';
|
|
|
|
} elseif ($numbytes >= $kib) {
|
|
|
|
$humanDatasize = ' ('.round($numbytes / $kib, $precision).' KB)';
|
|
|
|
}
|
|
|
|
|
|
|
|
return $humanDatasize;
|
2018-10-19 23:36:40 +02:00
|
|
|
}
|