2019-08-19 15:27:17 +00:00
|
|
|
|
<?php
|
|
|
|
|
|
2020-02-15 17:57:46 +00:00
|
|
|
|
require_once 'functions.php';
|
2019-08-19 15:27:17 +00:00
|
|
|
|
|
2023-03-24 11:01:47 +00:00
|
|
|
|
const MIN_RSSI = -100;
|
|
|
|
|
const MAX_RSSI = -55;
|
|
|
|
|
|
2019-08-19 15:27:17 +00:00
|
|
|
|
function knownWifiStations(&$networks)
|
|
|
|
|
{
|
|
|
|
|
// Find currently configured networks
|
|
|
|
|
exec(' sudo cat ' . RASPI_WPA_SUPPLICANT_CONFIG, $known_return);
|
2023-12-02 19:53:16 +00:00
|
|
|
|
//$index = 0;
|
2019-08-19 15:27:17 +00:00
|
|
|
|
foreach ($known_return as $line) {
|
|
|
|
|
if (preg_match('/network\s*=/', $line)) {
|
2023-12-02 19:53:16 +00:00
|
|
|
|
$network = array('visible' => false, 'configured' => true, 'connected' => false, 'index' => null);
|
2021-09-02 08:58:34 +00:00
|
|
|
|
++$index;
|
2021-02-02 12:26:14 +00:00
|
|
|
|
} elseif (isset($network) && $network !== null) {
|
2019-08-19 15:27:17 +00:00
|
|
|
|
if (preg_match('/^\s*}\s*$/', $line)) {
|
|
|
|
|
$networks[$ssid] = $network;
|
|
|
|
|
$network = null;
|
|
|
|
|
$ssid = null;
|
2023-02-09 17:27:51 +00:00
|
|
|
|
} elseif ($lineArr = preg_split('/\s*=\s*/', trim($line), 2)) {
|
2019-08-19 15:27:17 +00:00
|
|
|
|
switch (strtolower($lineArr[0])) {
|
2020-02-15 17:57:46 +00:00
|
|
|
|
case 'ssid':
|
|
|
|
|
$ssid = trim($lineArr[1], '"');
|
2021-08-18 18:37:45 +00:00
|
|
|
|
$ssid = str_replace('P"','',$ssid);
|
2021-05-30 22:46:10 +00:00
|
|
|
|
$network['ssid'] = $ssid;
|
2023-12-02 19:53:16 +00:00
|
|
|
|
$index = getNetworkIdBySSID($ssid);
|
|
|
|
|
$network['index'] = $index;
|
2020-02-15 17:57:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case 'psk':
|
2023-10-08 09:46:12 +00:00
|
|
|
|
$network['passkey'] = trim($lineArr[1]);
|
|
|
|
|
$network['protocol'] = 'WPA';
|
|
|
|
|
break;
|
2020-02-15 17:57:46 +00:00
|
|
|
|
case '#psk':
|
|
|
|
|
$network['protocol'] = 'WPA';
|
|
|
|
|
case 'wep_key0': // Untested
|
|
|
|
|
$network['passphrase'] = trim($lineArr[1], '"');
|
|
|
|
|
break;
|
|
|
|
|
case 'key_mgmt':
|
|
|
|
|
if (! array_key_exists('passphrase', $network) && $lineArr[1] === 'NONE') {
|
|
|
|
|
$network['protocol'] = 'Open';
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 'priority':
|
|
|
|
|
$network['priority'] = trim($lineArr[1], '"');
|
|
|
|
|
break;
|
2019-08-19 15:27:17 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function nearbyWifiStations(&$networks, $cached = true)
|
|
|
|
|
{
|
|
|
|
|
$cacheTime = filemtime(RASPI_WPA_SUPPLICANT_CONFIG);
|
|
|
|
|
$cacheKey = "nearby_wifi_stations_$cacheTime";
|
|
|
|
|
|
|
|
|
|
if ($cached == false) {
|
|
|
|
|
deleteCache($cacheKey);
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-15 17:57:46 +00:00
|
|
|
|
$scan_results = cache(
|
|
|
|
|
$cacheKey, function () {
|
2020-12-15 09:02:33 +00:00
|
|
|
|
exec('sudo wpa_cli -i ' .$_SESSION['wifi_client_interface']. ' scan');
|
2020-02-15 17:57:46 +00:00
|
|
|
|
sleep(3);
|
2022-12-29 05:27:32 +00:00
|
|
|
|
$stdout = shell_exec('sudo wpa_cli -i ' .$_SESSION['wifi_client_interface']. ' scan_results');
|
|
|
|
|
return preg_split("/\n/", $stdout);
|
2020-02-15 17:57:46 +00:00
|
|
|
|
}
|
|
|
|
|
);
|
2020-05-24 07:24:42 +00:00
|
|
|
|
// get the name of the AP. Should be excluded from nearby networks
|
|
|
|
|
exec('cat '.RASPI_HOSTAPD_CONFIG.' | sed -rn "s/ssid=(.*)\s*$/\1/p" ', $ap_ssid);
|
2020-05-24 07:14:07 +00:00
|
|
|
|
$ap_ssid = $ap_ssid[0];
|
2020-05-24 07:24:42 +00:00
|
|
|
|
|
2021-09-02 08:58:34 +00:00
|
|
|
|
$index = 0;
|
|
|
|
|
if ( !empty($networks) ) {
|
|
|
|
|
$lastnet = end($networks);
|
|
|
|
|
if ( isset($lastnet['index']) ) $index = $lastnet['index'] + 1;
|
|
|
|
|
}
|
2020-05-24 07:14:07 +00:00
|
|
|
|
|
2023-09-06 10:25:39 +00:00
|
|
|
|
if (is_array($scan_results)) {
|
|
|
|
|
array_shift($scan_results);
|
|
|
|
|
foreach ($scan_results as $network) {
|
|
|
|
|
$arrNetwork = preg_split("/[\t]+/", $network); // split result into array
|
|
|
|
|
$ssid = $arrNetwork[4];
|
2021-05-30 20:56:10 +00:00
|
|
|
|
|
2023-09-06 10:25:39 +00:00
|
|
|
|
// exclude raspap ssid
|
|
|
|
|
if (empty($ssid) || $ssid == $ap_ssid) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2021-05-30 20:56:10 +00:00
|
|
|
|
|
2023-09-06 10:25:39 +00:00
|
|
|
|
// filter SSID string: unprintable 7bit ASCII control codes, delete or quotes -> ignore network
|
|
|
|
|
if (preg_match('[\x00-\x1f\x7f\'\`\´\"]', $ssid)) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2019-08-19 15:27:17 +00:00
|
|
|
|
|
2023-09-06 10:25:39 +00:00
|
|
|
|
// If network is saved
|
|
|
|
|
if (array_key_exists($ssid, $networks)) {
|
|
|
|
|
$networks[$ssid]['visible'] = true;
|
|
|
|
|
$networks[$ssid]['channel'] = ConvertToChannel($arrNetwork[1]);
|
|
|
|
|
// TODO What if the security has changed?
|
|
|
|
|
} else {
|
|
|
|
|
$networks[$ssid] = array(
|
|
|
|
|
'ssid' => $ssid,
|
|
|
|
|
'configured' => false,
|
|
|
|
|
'protocol' => ConvertToSecurity($arrNetwork[3]),
|
|
|
|
|
'channel' => ConvertToChannel($arrNetwork[1]),
|
|
|
|
|
'passphrase' => '',
|
|
|
|
|
'visible' => true,
|
|
|
|
|
'connected' => false,
|
|
|
|
|
'index' => $index
|
|
|
|
|
);
|
|
|
|
|
++$index;
|
|
|
|
|
}
|
2019-08-19 15:27:17 +00:00
|
|
|
|
|
2023-09-06 10:25:39 +00:00
|
|
|
|
// Save RSSI, if the current value is larger than the already stored
|
|
|
|
|
if (array_key_exists(4, $arrNetwork) && array_key_exists($arrNetwork[4], $networks)) {
|
|
|
|
|
if (! array_key_exists('RSSI', $networks[$arrNetwork[4]]) || $networks[$ssid]['RSSI'] < $arrNetwork[2]) {
|
|
|
|
|
$networks[$ssid]['RSSI'] = $arrNetwork[2];
|
|
|
|
|
}
|
2020-05-24 07:24:42 +00:00
|
|
|
|
}
|
2019-08-19 15:27:17 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function connectedWifiStations(&$networks)
|
|
|
|
|
{
|
2020-06-09 18:25:19 +00:00
|
|
|
|
exec('iwconfig ' .$_SESSION['wifi_client_interface'], $iwconfig_return);
|
2019-08-19 15:27:17 +00:00
|
|
|
|
foreach ($iwconfig_return as $line) {
|
|
|
|
|
if (preg_match('/ESSID:\"([^"]+)\"/i', $line, $iwconfig_ssid)) {
|
2021-08-31 14:18:00 +00:00
|
|
|
|
$networks[hexSequence2lower($iwconfig_ssid[1])]['connected'] = true;
|
2019-08-19 15:27:17 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-06-07 16:13:43 +00:00
|
|
|
|
|
2020-05-24 07:24:42 +00:00
|
|
|
|
function sortNetworksByRSSI(&$networks)
|
|
|
|
|
{
|
|
|
|
|
$valRSSI = array();
|
|
|
|
|
foreach ($networks as $SSID => $net) {
|
|
|
|
|
if (!array_key_exists('RSSI', $net)) {
|
|
|
|
|
$net['RSSI'] = -1000;
|
2020-05-24 07:14:07 +00:00
|
|
|
|
}
|
2020-05-24 07:24:42 +00:00
|
|
|
|
$valRSSI[$SSID] = $net['RSSI'];
|
|
|
|
|
}
|
|
|
|
|
$nets = $networks;
|
|
|
|
|
arsort($valRSSI);
|
|
|
|
|
$networks = array();
|
|
|
|
|
foreach ($valRSSI as $SSID => $RSSI) {
|
2020-05-24 07:30:32 +00:00
|
|
|
|
$networks[$SSID] = $nets[$SSID];
|
2020-05-24 07:24:42 +00:00
|
|
|
|
$networks[$SSID]['RSSI'] = $RSSI;
|
|
|
|
|
}
|
2020-05-24 07:14:07 +00:00
|
|
|
|
}
|
2020-06-07 16:20:09 +00:00
|
|
|
|
|
2020-06-09 14:44:21 +00:00
|
|
|
|
/*
|
|
|
|
|
* Determines the configured wireless AP interface
|
|
|
|
|
*
|
|
|
|
|
* If not saved in /etc/raspap/hostapd.ini, check for a second
|
|
|
|
|
* wireless interface with iw dev. Fallback to the constant
|
|
|
|
|
* value defined in config.php
|
|
|
|
|
*/
|
2020-06-07 16:13:43 +00:00
|
|
|
|
function getWifiInterface()
|
|
|
|
|
{
|
|
|
|
|
$arrHostapdConf = parse_ini_file(RASPI_CONFIG.'/hostapd.ini');
|
2020-06-09 18:53:11 +00:00
|
|
|
|
$iface = $_SESSION['ap_interface'] = isset($arrHostapdConf['WifiInterface']) ? $arrHostapdConf['WifiInterface'] : RASPI_WIFI_AP_INTERFACE;
|
|
|
|
|
// check for 2nd wifi interface -> wifi client on different interface
|
|
|
|
|
exec("iw dev | awk '$1==\"Interface\" && $2!=\"$iface\" {print $2}'",$iface2);
|
2020-12-15 10:45:27 +00:00
|
|
|
|
$client_iface = $_SESSION['wifi_client_interface'] = (empty($iface2) ? $iface : trim($iface2[0]));
|
2020-12-15 08:05:07 +00:00
|
|
|
|
|
|
|
|
|
// specifically for rpi0W in AP-STA mode, the above check ends up with the interfaces
|
|
|
|
|
// crossed over (wifi_client_interface vs 'ap_interface'), because the second interface (uap0) is
|
|
|
|
|
// created by raspap and used as the access point.
|
2020-12-15 10:45:27 +00:00
|
|
|
|
if ($client_iface == "uap0" && ($arrHostapdConf['WifiAPEnable'] ?? 0)){
|
2020-12-15 08:05:07 +00:00
|
|
|
|
$_SESSION['wifi_client_interface'] = $iface;
|
|
|
|
|
$_SESSION['ap_interface'] = $client_iface;
|
|
|
|
|
}
|
2020-06-07 16:13:43 +00:00
|
|
|
|
}
|
|
|
|
|
|
2021-04-29 21:58:46 +00:00
|
|
|
|
/*
|
2021-04-30 16:24:45 +00:00
|
|
|
|
* Reinitializes wpa_supplicant for the wireless client interface
|
|
|
|
|
* The 'force' parameter deletes the socket in /var/run/wpa_supplicant/
|
2021-04-29 21:58:46 +00:00
|
|
|
|
*
|
|
|
|
|
* @param boolean $force
|
|
|
|
|
*/
|
|
|
|
|
function reinitializeWPA($force)
|
|
|
|
|
{
|
2023-12-04 08:13:13 +00:00
|
|
|
|
$iface = escapeshellarg($_SESSION['wifi_client_interface']);
|
2021-04-29 21:58:46 +00:00
|
|
|
|
if ($force == true) {
|
2023-12-04 08:13:13 +00:00
|
|
|
|
$cmd = "sudo /bin/rm /var/run/wpa_supplicant/$iface";
|
2023-12-02 14:32:13 +00:00
|
|
|
|
$result = shell_exec($cmd);
|
2021-04-29 21:58:46 +00:00
|
|
|
|
}
|
2024-01-16 17:29:16 +00:00
|
|
|
|
$cmd = "sudo wpa_supplicant -B -Dnl80211 -c/etc/wpa_supplicant/wpa_supplicant.conf -i$iface";
|
2021-04-29 21:58:46 +00:00
|
|
|
|
$result = shell_exec($cmd);
|
2023-12-04 08:13:13 +00:00
|
|
|
|
sleep(1);
|
2021-04-29 21:58:46 +00:00
|
|
|
|
return $result;
|
|
|
|
|
}
|
2021-04-30 07:37:41 +00:00
|
|
|
|
|
2021-08-18 19:57:30 +00:00
|
|
|
|
/*
|
|
|
|
|
* Replace escaped bytes (hex) by binary - assume UTF8 encoding
|
|
|
|
|
*
|
|
|
|
|
* @param string $ssid
|
|
|
|
|
*/
|
2023-03-24 11:01:47 +00:00
|
|
|
|
function ssid2utf8($ssid)
|
|
|
|
|
{
|
2021-08-18 19:57:30 +00:00
|
|
|
|
return evalHexSequence($ssid);
|
|
|
|
|
}
|
|
|
|
|
|
2023-03-24 11:01:47 +00:00
|
|
|
|
/*
|
|
|
|
|
* Returns a signal strength indicator based on RSSI value
|
|
|
|
|
*
|
|
|
|
|
* @param string $rssi
|
|
|
|
|
*/
|
|
|
|
|
function getSignalBars($rssi)
|
|
|
|
|
{
|
|
|
|
|
// assign css class based on RSSI value
|
|
|
|
|
if ($rssi >= MAX_RSSI) {
|
|
|
|
|
$class = 'strong';
|
|
|
|
|
} elseif ($rssi >= -56) {
|
|
|
|
|
$class = 'medium';
|
|
|
|
|
} elseif ($rssi >= -67) {
|
|
|
|
|
$class = 'weak';
|
|
|
|
|
} elseif ($rssi >= -89) {
|
|
|
|
|
$class = '';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// calculate percent strength
|
|
|
|
|
if ($rssi >= -50) {
|
|
|
|
|
$pct = 100;
|
|
|
|
|
} elseif ($rssi <= MIN_RSSI) {
|
|
|
|
|
$pct = 0;
|
|
|
|
|
} else {
|
|
|
|
|
$pct = 2*($rssi + 100);
|
|
|
|
|
}
|
2023-03-25 07:52:56 +00:00
|
|
|
|
$elem = '<div data-toggle="tooltip" title="' . _("Signal strength"). ': ' .$pct. '%" class="signal-icon ' .$class. '">'.PHP_EOL;
|
2023-03-24 11:01:47 +00:00
|
|
|
|
for ($n = 0; $n < 3; $n++ ) {
|
|
|
|
|
$elem .= '<div class="signal-bar"></div>'.PHP_EOL;
|
|
|
|
|
}
|
|
|
|
|
$elem .= '</div>'.PHP_EOL;
|
|
|
|
|
return $elem;
|
|
|
|
|
}
|
2023-03-25 07:52:56 +00:00
|
|
|
|
|
2023-12-02 17:32:15 +00:00
|
|
|
|
/*
|
|
|
|
|
* Parses output of wpa_cli list_networks, compares with known networks
|
|
|
|
|
* from wpa_supplicant, and adds with wpa_cli if not found
|
|
|
|
|
*
|
|
|
|
|
* @param array $networks
|
|
|
|
|
*/
|
|
|
|
|
function setKnownStationsWPA($networks)
|
|
|
|
|
{
|
|
|
|
|
$iface = escapeshellarg($_SESSION['wifi_client_interface']);
|
|
|
|
|
$output = shell_exec("sudo wpa_cli -i $iface list_networks");
|
|
|
|
|
$lines = explode("\n", $output);
|
2023-12-02 19:53:16 +00:00
|
|
|
|
array_shift($lines);
|
2023-12-02 17:32:15 +00:00
|
|
|
|
$wpaCliNetworks = [];
|
|
|
|
|
|
|
|
|
|
foreach ($lines as $line) {
|
|
|
|
|
$data = explode("\t", trim($line));
|
|
|
|
|
if (!empty($data) && count($data) >= 2) {
|
|
|
|
|
$id = $data[0];
|
|
|
|
|
$ssid = $data[1];
|
|
|
|
|
$item = [
|
|
|
|
|
'id' => $id,
|
|
|
|
|
'ssid' => $ssid
|
|
|
|
|
];
|
|
|
|
|
$wpaCliNetworks[] = $item;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
foreach ($networks as $network) {
|
|
|
|
|
$ssid = $network['ssid'];
|
|
|
|
|
if (!networkExists($ssid, $wpaCliNetworks)) {
|
|
|
|
|
$ssid = escapeshellarg('"'.$network['ssid'].'"');
|
|
|
|
|
$psk = escapeshellarg('"'.$network['passphrase'].'"');
|
2024-07-24 08:04:19 +00:00
|
|
|
|
$protocol = $network['protocol'];
|
2023-12-02 17:32:15 +00:00
|
|
|
|
$netid = trim(shell_exec("sudo wpa_cli -i $iface add_network"));
|
|
|
|
|
if (isset($netid) && !isset($known[$netid])) {
|
|
|
|
|
$commands = [
|
|
|
|
|
"sudo wpa_cli -i $iface set_network $netid ssid $ssid",
|
|
|
|
|
"sudo wpa_cli -i $iface set_network $netid psk $psk",
|
|
|
|
|
"sudo wpa_cli -i $iface enable_network $netid"
|
|
|
|
|
];
|
2024-07-24 08:04:19 +00:00
|
|
|
|
if ($protocol === 'Open') {
|
|
|
|
|
$commands[1] = "sudo wpa_cli -i $iface set_network $netid key_mgmt NONE";
|
|
|
|
|
}
|
2023-12-02 17:32:15 +00:00
|
|
|
|
foreach ($commands as $cmd) {
|
|
|
|
|
exec($cmd);
|
|
|
|
|
usleep(1000);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-02 19:53:16 +00:00
|
|
|
|
/*
|
|
|
|
|
* Parses wpa_cli list_networks output and returns the id
|
|
|
|
|
* of a corresponding network SSID
|
|
|
|
|
*
|
|
|
|
|
* @param string $ssid
|
|
|
|
|
* @return integer id
|
|
|
|
|
*/
|
|
|
|
|
function getNetworkIdBySSID($ssid) {
|
|
|
|
|
$iface = escapeshellarg($_SESSION['wifi_client_interface']);
|
|
|
|
|
$cmd = "sudo wpa_cli -i $iface list_networks";
|
|
|
|
|
$output = [];
|
|
|
|
|
exec($cmd, $output);
|
|
|
|
|
array_shift($output);
|
|
|
|
|
foreach ($output as $line) {
|
|
|
|
|
$columns = preg_split('/\t/', $line);
|
|
|
|
|
if (count($columns) >= 4 && trim($columns[1]) === trim($ssid)) {
|
|
|
|
|
return $columns[0]; // return network ID
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-02 17:32:15 +00:00
|
|
|
|
function networkExists($ssid, $collection)
|
|
|
|
|
{
|
|
|
|
|
foreach ($collection as $network) {
|
|
|
|
|
if ($network['ssid'] === $ssid) {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|