network: new NetworkManager backend using sdbus (#1269)
* network: new NetworkManager backend using sdbus drop support for old methods * rescan changes * add wifi on-off
This commit is contained in:
parent
524aa0e7dc
commit
a80b881015
@ -5,35 +5,10 @@
|
|||||||
|
|
||||||
The network panel requires network-manager to function, (if you are using a fork this may not be the case)
|
The network panel requires network-manager to function, (if you are using a fork this may not be the case)
|
||||||
|
|
||||||
Check if network-manager is installed:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
dpkg -s network-manager
|
|
||||||
```
|
|
||||||
|
|
||||||
if the response is the following:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
dpkg-query: the package 'network-manager' is not installed
|
|
||||||
```
|
|
||||||
|
|
||||||
if the response is the following:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
Package: network-manager
|
|
||||||
Status: install ok installed
|
|
||||||
```
|
|
||||||
|
|
||||||
this line may appear in KlipperScreen.log:
|
|
||||||
!!! abstract "Log"
|
|
||||||
```sh
|
|
||||||
[wifi_nm.py:rescan()] [...] NetworkManager.wifi.scan request failed: not authorized
|
|
||||||
```
|
|
||||||
|
|
||||||
if version of KlipperScreen installed was previous than v0.3.9, then re-run the installer and reboot
|
if version of KlipperScreen installed was previous than v0.3.9, then re-run the installer and reboot
|
||||||
|
|
||||||
|
|
||||||
??? Alternative workaround for network-manager
|
??? "Alternative workaround for network-manager not having permissions"
|
||||||
|
|
||||||
in order to fix this polkit needs to be configured or disabled:
|
in order to fix this polkit needs to be configured or disabled:
|
||||||
|
|
||||||
|
@ -1,27 +0,0 @@
|
|||||||
## wpa_supplicant
|
|
||||||
|
|
||||||
The user is not allowed to control the interface
|
|
||||||
|
|
||||||
* Edit `/etc/wpa_supplicant/wpa_supplicant.conf` and add this line if it's not there:
|
|
||||||
|
|
||||||
```
|
|
||||||
ctrl_interface=DIR=/var/run/wpa_supplicant GROUP=netdev
|
|
||||||
```
|
|
||||||
|
|
||||||
* Run `cat /etc/group | grep netdev`
|
|
||||||
|
|
||||||
If your username is not listed under that line, you need to add it with the following command:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
usermod -a -G netdev pi
|
|
||||||
```
|
|
||||||
(if your username is not 'pi' change 'pi' to your username)
|
|
||||||
|
|
||||||
Then reboot the machine:
|
|
||||||
|
|
||||||
```sh
|
|
||||||
systemctl reboot
|
|
||||||
```
|
|
||||||
|
|
||||||
!!! tip
|
|
||||||
It's possible to just restart KlipperScreen and networking
|
|
File diff suppressed because it is too large
Load Diff
@ -168,6 +168,15 @@ class ScreenPanel:
|
|||||||
if size < unit:
|
if size < unit:
|
||||||
return f"{(1024 * size / unit):.1f} {suffix}"
|
return f"{(1024 * size / unit):.1f} {suffix}"
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def format_speed(bitrate):
|
||||||
|
bitrate = float(bitrate)
|
||||||
|
suffixes = ["Kbits/s", "Mbits/s", "Gbits/s", "Tbits/s", "Pbits/s", "Ebits/s", "Zbits/s", "Ybits/s"]
|
||||||
|
for i, suffix in enumerate(suffixes, start=1):
|
||||||
|
unit = 1000 ** i
|
||||||
|
if bitrate < unit:
|
||||||
|
return f"{(1000 * bitrate / unit):.0f} {suffix}"
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def prettify(name: str):
|
def prettify(name: str):
|
||||||
name = name.replace("_", " ")
|
name = name.replace("_", " ")
|
||||||
|
250
ks_includes/sdbus_nm.py
Normal file
250
ks_includes/sdbus_nm.py
Normal file
@ -0,0 +1,250 @@
|
|||||||
|
# This is the backend of the UI panel that communicates to sdbus-networkmanager
|
||||||
|
# TODO device selection/swtichability
|
||||||
|
# Alfredo Monclus (alfrix) 2024
|
||||||
|
|
||||||
|
import logging
|
||||||
|
|
||||||
|
from sdbus_block.networkmanager import (
|
||||||
|
NetworkManager,
|
||||||
|
NetworkDeviceGeneric,
|
||||||
|
NetworkDeviceWireless,
|
||||||
|
NetworkConnectionSettings,
|
||||||
|
NetworkManagerSettings,
|
||||||
|
AccessPoint,
|
||||||
|
NetworkManagerConnectionProperties,
|
||||||
|
IPv4Config,
|
||||||
|
ActiveConnection,
|
||||||
|
enums,
|
||||||
|
)
|
||||||
|
from sdbus import sd_bus_open_system, set_default_bus
|
||||||
|
from gi.repository import GLib
|
||||||
|
from uuid import uuid4
|
||||||
|
|
||||||
|
|
||||||
|
NM_802_11_AP_SEC_NONE = 0
|
||||||
|
NM_802_11_AP_SEC_PAIR_WEP40 = 1
|
||||||
|
NM_802_11_AP_SEC_PAIR_WEP104 = 2
|
||||||
|
NM_802_11_AP_SEC_PAIR_TKIP = 4
|
||||||
|
NM_802_11_AP_SEC_PAIR_CCMP = 8
|
||||||
|
NM_802_11_AP_SEC_GROUP_WEP40 = 16
|
||||||
|
NM_802_11_AP_SEC_GROUP_WEP104 = 32
|
||||||
|
NM_802_11_AP_SEC_GROUP_TKIP = 64
|
||||||
|
NM_802_11_AP_SEC_GROUP_CCMP = 128
|
||||||
|
NM_802_11_AP_SEC_KEY_MGMT_PSK = 256
|
||||||
|
NM_802_11_AP_SEC_KEY_MGMT_802_1X = 512
|
||||||
|
|
||||||
|
|
||||||
|
def get_encryption(flags):
|
||||||
|
encryption = ""
|
||||||
|
if (flags & NM_802_11_AP_SEC_PAIR_WEP40 or
|
||||||
|
flags & NM_802_11_AP_SEC_PAIR_WEP104 or
|
||||||
|
flags & NM_802_11_AP_SEC_GROUP_WEP40 or
|
||||||
|
flags & NM_802_11_AP_SEC_GROUP_WEP104):
|
||||||
|
encryption += "WEP "
|
||||||
|
if (flags & NM_802_11_AP_SEC_PAIR_TKIP or
|
||||||
|
flags & NM_802_11_AP_SEC_GROUP_TKIP):
|
||||||
|
encryption += "TKIP "
|
||||||
|
if (flags & NM_802_11_AP_SEC_PAIR_CCMP or
|
||||||
|
flags & NM_802_11_AP_SEC_GROUP_CCMP):
|
||||||
|
encryption += "AES "
|
||||||
|
if flags & NM_802_11_AP_SEC_KEY_MGMT_PSK:
|
||||||
|
encryption += "WPA-PSK "
|
||||||
|
if flags & NM_802_11_AP_SEC_KEY_MGMT_802_1X:
|
||||||
|
encryption += "802.1x "
|
||||||
|
return encryption
|
||||||
|
|
||||||
|
|
||||||
|
def WifiChannels(freq: str):
|
||||||
|
if freq == '2484':
|
||||||
|
return "2.4", "14"
|
||||||
|
try:
|
||||||
|
freq = float(freq)
|
||||||
|
except ValueError:
|
||||||
|
return "?", "?"
|
||||||
|
if 2412 <= freq <= 2472:
|
||||||
|
return "2.4", str(int((freq - 2407) / 5))
|
||||||
|
elif 3657.5 <= freq <= 3692.5:
|
||||||
|
return "3", str(int((freq - 3000) / 5))
|
||||||
|
elif 4915 <= freq <= 4980:
|
||||||
|
return "5", str(int((freq - 4000) / 5))
|
||||||
|
elif 5035 <= freq <= 5885:
|
||||||
|
return "5", str(int((freq - 5000) / 5))
|
||||||
|
elif 6455 <= freq <= 7115:
|
||||||
|
return "6", str(int((freq - 5950) / 5))
|
||||||
|
else:
|
||||||
|
return "?", "?"
|
||||||
|
|
||||||
|
|
||||||
|
class SdbusNm:
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
self.system_bus = sd_bus_open_system() # We need system bus
|
||||||
|
set_default_bus(self.system_bus)
|
||||||
|
self.nm = NetworkManager()
|
||||||
|
self._callbacks = {
|
||||||
|
"popup": [],
|
||||||
|
}
|
||||||
|
if self.get_wireless_interfaces():
|
||||||
|
self.wlan_device = self.get_wireless_interfaces()[0]
|
||||||
|
self.wifi = True
|
||||||
|
else:
|
||||||
|
self.wlan_device = None
|
||||||
|
self.wifi = False
|
||||||
|
|
||||||
|
def is_wifi_enabled(self):
|
||||||
|
return self.nm.wireless_enabled
|
||||||
|
|
||||||
|
def get_interfaces(self):
|
||||||
|
return [NetworkDeviceGeneric(device).interface for device in self.nm.get_devices()]
|
||||||
|
|
||||||
|
def get_wireless_interfaces(self):
|
||||||
|
devices = {path: NetworkDeviceGeneric(path) for path in self.nm.get_devices()}
|
||||||
|
return [
|
||||||
|
NetworkDeviceWireless(path)
|
||||||
|
for path, device in devices.items()
|
||||||
|
if device.device_type == enums.DeviceType.WIFI
|
||||||
|
]
|
||||||
|
|
||||||
|
def get_primary_interface(self):
|
||||||
|
if self.nm.primary_connection == '/':
|
||||||
|
# Nothing connected
|
||||||
|
if self.wlan_device:
|
||||||
|
return self.wlan_device.interface
|
||||||
|
if len(self.get_interfaces()) > 1:
|
||||||
|
# skips the loopback device
|
||||||
|
return self.get_interfaces()[1]
|
||||||
|
return None
|
||||||
|
gateway = ActiveConnection(self.nm.primary_connection).devices[0]
|
||||||
|
return NetworkDeviceGeneric(gateway).interface
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_known_networks():
|
||||||
|
known_networks = []
|
||||||
|
saved_network_paths = NetworkManagerSettings().list_connections()
|
||||||
|
for netpath in saved_network_paths:
|
||||||
|
saved_con = NetworkConnectionSettings(netpath)
|
||||||
|
con_settings = saved_con.get_settings()
|
||||||
|
# 'type': ('s', '802-11-wireless')
|
||||||
|
if con_settings['connection']['type'][1] == "802-11-wireless":
|
||||||
|
known_networks.append({
|
||||||
|
'SSID': con_settings['802-11-wireless']['ssid'][1].decode(),
|
||||||
|
'UUID': con_settings['connection']['uuid'][1]
|
||||||
|
})
|
||||||
|
return known_networks
|
||||||
|
|
||||||
|
def is_known(self, ssid):
|
||||||
|
return any(net['SSID'] == ssid for net in self.get_known_networks())
|
||||||
|
|
||||||
|
def get_ip_address(self):
|
||||||
|
active_connection_path = self.nm.primary_connection
|
||||||
|
if not active_connection_path or active_connection_path == '/':
|
||||||
|
return "?"
|
||||||
|
active_connection = ActiveConnection(active_connection_path)
|
||||||
|
ip_info = IPv4Config(active_connection.ip4_config)
|
||||||
|
|
||||||
|
return ip_info.address_data[0]['address'][1]
|
||||||
|
|
||||||
|
def get_networks(self):
|
||||||
|
networks = []
|
||||||
|
if self.wlan_device:
|
||||||
|
all_aps = [AccessPoint(result) for result in self.wlan_device.access_points]
|
||||||
|
networks.extend(
|
||||||
|
{
|
||||||
|
"SSID": ap.ssid.decode("utf-8"),
|
||||||
|
"known": self.is_known(ap.ssid.decode("utf-8")),
|
||||||
|
"security": get_encryption(ap.rsn_flags),
|
||||||
|
"frequency": WifiChannels(ap.frequency)[0],
|
||||||
|
"channel": WifiChannels(ap.frequency)[1],
|
||||||
|
"signal_level": ap.strength,
|
||||||
|
"max_bitrate": ap.max_bitrate,
|
||||||
|
"BSSID": ap.hw_address,
|
||||||
|
}
|
||||||
|
for ap in all_aps
|
||||||
|
if ap.ssid
|
||||||
|
)
|
||||||
|
return sorted(networks, key=lambda i: i['signal_level'], reverse=True)
|
||||||
|
return networks
|
||||||
|
|
||||||
|
def get_bssid_from_ssid(self, ssid):
|
||||||
|
return next(net['BSSID'] for net in self.get_networks() if ssid == net['SSID'])
|
||||||
|
|
||||||
|
def get_connected_ap(self):
|
||||||
|
if self.wlan_device.active_access_point == "/":
|
||||||
|
return None
|
||||||
|
return AccessPoint(self.wlan_device.active_access_point)
|
||||||
|
|
||||||
|
def get_connected_bssid(self):
|
||||||
|
return self.get_connected_ap().hw_address if self.get_connected_ap() is not None else None
|
||||||
|
|
||||||
|
def add_network(self, ssid, psk):
|
||||||
|
existing_network = NetworkManagerSettings().get_connections_by_id(ssid)
|
||||||
|
if existing_network:
|
||||||
|
for network in existing_network:
|
||||||
|
self.delete_connection_path(network)
|
||||||
|
|
||||||
|
properties: NetworkManagerConnectionProperties = {
|
||||||
|
"connection": {
|
||||||
|
"id": ("s", ssid),
|
||||||
|
"uuid": ("s", str(uuid4())),
|
||||||
|
"type": ("s", "802-11-wireless"),
|
||||||
|
"interface-name": ("s", self.wlan_device.interface)
|
||||||
|
},
|
||||||
|
"802-11-wireless": {
|
||||||
|
"mode": ("s", "infrastructure"),
|
||||||
|
"security": ("s", "802-11-wireless-security"),
|
||||||
|
"ssid": ("ay", ssid.encode("utf-8")),
|
||||||
|
},
|
||||||
|
"802-11-wireless-security": {
|
||||||
|
"key-mgmt": ("s", "wpa-psk"),
|
||||||
|
"auth-alg": ("s", "open"),
|
||||||
|
"psk": ("s", psk),
|
||||||
|
},
|
||||||
|
"ipv4": {"method": ("s", "auto")},
|
||||||
|
"ipv6": {"method": ("s", "auto")},
|
||||||
|
}
|
||||||
|
|
||||||
|
try:
|
||||||
|
msg = f"{ssid}\n" + _("Starting WiFi Association")
|
||||||
|
self.callback("popup", msg, 1)
|
||||||
|
NetworkManagerSettings().add_connection(properties)
|
||||||
|
return True
|
||||||
|
except Exception as e:
|
||||||
|
logging.error(f"{e}")
|
||||||
|
self.callback("popup", f"Error: {e}")
|
||||||
|
return False
|
||||||
|
|
||||||
|
def disconnect_network(self):
|
||||||
|
self.wlan_device.disconnect()
|
||||||
|
|
||||||
|
def delete_network(self, ssid):
|
||||||
|
connection = NetworkManagerSettings().get_connections_by_id(ssid)
|
||||||
|
for path in connection:
|
||||||
|
self.delete_connection_path(path)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def delete_connection_path(path):
|
||||||
|
NetworkConnectionSettings(path).delete()
|
||||||
|
|
||||||
|
def rescan(self):
|
||||||
|
return self.wlan_device.request_scan({})
|
||||||
|
|
||||||
|
def connect(self, ssid):
|
||||||
|
connection = NetworkManagerSettings().get_connections_by_id(ssid)
|
||||||
|
if not connection:
|
||||||
|
self.callback("popup", f"{ssid} {connection}")
|
||||||
|
return
|
||||||
|
msg = f"{ssid}:\n" + _("Starting WiFi Association")
|
||||||
|
self.callback("popup", msg, 1)
|
||||||
|
self.nm.activate_connection(connection[0])
|
||||||
|
|
||||||
|
def add_callback(self, name, callback):
|
||||||
|
if name in self._callbacks and callback not in self._callbacks[name]:
|
||||||
|
self._callbacks[name].append(callback)
|
||||||
|
|
||||||
|
def callback(self, cb_type, *args):
|
||||||
|
if cb_type in self._callbacks:
|
||||||
|
for cb in self._callbacks[cb_type]:
|
||||||
|
GLib.idle_add(cb, *args)
|
||||||
|
|
||||||
|
def toggle_wifi(self, enable):
|
||||||
|
self.nm.wireless_enabled = enable
|
@ -1,347 +0,0 @@
|
|||||||
import os
|
|
||||||
import logging
|
|
||||||
import re
|
|
||||||
import socket
|
|
||||||
import threading
|
|
||||||
from threading import Thread
|
|
||||||
from queue import Queue
|
|
||||||
|
|
||||||
import gi
|
|
||||||
|
|
||||||
gi.require_version("Gtk", "3.0")
|
|
||||||
from gi.repository import GLib
|
|
||||||
|
|
||||||
|
|
||||||
class WifiManager:
|
|
||||||
networks_in_supplicant = []
|
|
||||||
connected = False
|
|
||||||
_stop_loop = False
|
|
||||||
|
|
||||||
def __init__(self, interface, *args, **kwargs):
|
|
||||||
super().__init__(*args, **kwargs)
|
|
||||||
self._callbacks = {
|
|
||||||
"connected": [],
|
|
||||||
"connecting_status": [],
|
|
||||||
"scan_results": [],
|
|
||||||
"popup": [],
|
|
||||||
}
|
|
||||||
self._stop_loop = False
|
|
||||||
self.connected = False
|
|
||||||
self.connected_ssid = None
|
|
||||||
self.event = threading.Event()
|
|
||||||
self.initialized = False
|
|
||||||
self.interface = interface
|
|
||||||
self.networks = {}
|
|
||||||
self.supplicant_networks = {}
|
|
||||||
self.queue = Queue()
|
|
||||||
self.timeout = None
|
|
||||||
|
|
||||||
ks_socket_file = "/tmp/.KS_wpa_supplicant"
|
|
||||||
if os.path.exists(ks_socket_file):
|
|
||||||
os.remove(ks_socket_file)
|
|
||||||
|
|
||||||
try:
|
|
||||||
self.soc = socket.socket(socket.AF_UNIX, socket.SOCK_DGRAM)
|
|
||||||
self.soc.bind(ks_socket_file)
|
|
||||||
self.soc.connect(f"/var/run/wpa_supplicant/{interface}")
|
|
||||||
except Exception as e:
|
|
||||||
logging.critical(e, exc_info=True)
|
|
||||||
logging.error(f"Error connecting to wifi socket: {interface}")
|
|
||||||
return
|
|
||||||
|
|
||||||
self.wpa_thread = WpaSocket(self, self.queue, self.callback)
|
|
||||||
self.wpa_thread.start()
|
|
||||||
self.initialized = True
|
|
||||||
|
|
||||||
self.wpa_cli("ATTACH", False)
|
|
||||||
self.wpa_cli("SCAN", False)
|
|
||||||
GLib.idle_add(self.read_wpa_supplicant)
|
|
||||||
self.timeout = GLib.timeout_add_seconds(180, self.rescan)
|
|
||||||
|
|
||||||
def add_callback(self, name, callback):
|
|
||||||
if name in self._callbacks and callback not in self._callbacks[name]:
|
|
||||||
self._callbacks[name].append(callback)
|
|
||||||
|
|
||||||
def add_network(self, ssid, psk):
|
|
||||||
for netid in list(self.supplicant_networks):
|
|
||||||
if self.supplicant_networks[netid]['ssid'] == ssid:
|
|
||||||
# Modify network
|
|
||||||
return
|
|
||||||
|
|
||||||
# TODO: Add wpa_cli error checking
|
|
||||||
network_id = self.wpa_cli("ADD_NETWORK")
|
|
||||||
commands = [
|
|
||||||
f'ENABLE_NETWORK {network_id}',
|
|
||||||
'SET_NETWORK %s ssid "%s"' % (network_id, ssid.replace('"', '\"')),
|
|
||||||
'SET_NETWORK %s psk "%s"' % (network_id, psk.replace('"', '\"'))
|
|
||||||
]
|
|
||||||
|
|
||||||
self.wpa_cli_batch(commands)
|
|
||||||
|
|
||||||
self.read_wpa_supplicant()
|
|
||||||
netid = None
|
|
||||||
for i in list(self.supplicant_networks):
|
|
||||||
if self.supplicant_networks[i]['ssid'] == ssid:
|
|
||||||
netid = i
|
|
||||||
break
|
|
||||||
|
|
||||||
if netid is None:
|
|
||||||
logging.info("Error adding network")
|
|
||||||
return False
|
|
||||||
|
|
||||||
self.save_wpa_conf()
|
|
||||||
return True
|
|
||||||
|
|
||||||
def callback(self, cb_type, msg):
|
|
||||||
if cb_type in self._callbacks:
|
|
||||||
for cb in self._callbacks[cb_type]:
|
|
||||||
GLib.idle_add(cb, msg)
|
|
||||||
|
|
||||||
def connect(self, ssid):
|
|
||||||
netid = None
|
|
||||||
for nid, net in self.supplicant_networks.items():
|
|
||||||
if net['ssid'] == ssid:
|
|
||||||
netid = nid
|
|
||||||
break
|
|
||||||
|
|
||||||
if netid is None:
|
|
||||||
logging.info("Wifi network is not defined in wpa_supplicant")
|
|
||||||
return False
|
|
||||||
|
|
||||||
logging.info(f"Attempting to connect to wifi: {netid}")
|
|
||||||
self.callback("connecting_status", f"Attempting to connect to {ssid}")
|
|
||||||
self.wpa_cli(f"SELECT_NETWORK {netid}")
|
|
||||||
self.save_wpa_conf()
|
|
||||||
|
|
||||||
def delete_network(self, ssid):
|
|
||||||
netid = None
|
|
||||||
for i in list(self.supplicant_networks):
|
|
||||||
if self.supplicant_networks[i]['ssid'] == ssid:
|
|
||||||
netid = i
|
|
||||||
break
|
|
||||||
|
|
||||||
if netid is None:
|
|
||||||
logging.debug("Unable to find network in wpa_supplicant")
|
|
||||||
return
|
|
||||||
self.wpa_cli(f"REMOVE_NETWORK {netid}")
|
|
||||||
|
|
||||||
for netid in list(self.supplicant_networks):
|
|
||||||
if self.supplicant_networks[netid]['ssid'] == ssid:
|
|
||||||
del self.supplicant_networks[netid]
|
|
||||||
break
|
|
||||||
|
|
||||||
self.save_wpa_conf()
|
|
||||||
|
|
||||||
def get_connected_ssid(self):
|
|
||||||
return self.connected_ssid
|
|
||||||
|
|
||||||
def get_current_wifi(self):
|
|
||||||
con_ssid = os.popen("sudo iwgetid -r").read().strip()
|
|
||||||
con_bssid = os.popen("sudo iwgetid -r -a").read().strip()
|
|
||||||
# wpa_cli status output is unstable use it as backup only
|
|
||||||
status = self.wpa_cli("STATUS").split('\n')
|
|
||||||
variables = {}
|
|
||||||
for line in status:
|
|
||||||
arr = line.split('=')
|
|
||||||
variables[arr[0]] = "=".join(arr[1:])
|
|
||||||
prev_ssid = self.connected_ssid
|
|
||||||
|
|
||||||
if con_ssid != "":
|
|
||||||
self.connected = True
|
|
||||||
self.connected_ssid = con_ssid
|
|
||||||
for ssid, val in self.networks.items():
|
|
||||||
self.networks[ssid]['connected'] = ssid == con_ssid
|
|
||||||
if prev_ssid != self.connected_ssid:
|
|
||||||
for cb in self._callbacks['connected']:
|
|
||||||
args = self.connected_ssid, prev_ssid
|
|
||||||
GLib.idle_add(cb, *args)
|
|
||||||
return [con_ssid, con_bssid]
|
|
||||||
elif "ssid" in variables and "bssid" in variables:
|
|
||||||
self.connected = True
|
|
||||||
self.connected_ssid = variables['ssid']
|
|
||||||
for ssid, val in self.networks.items():
|
|
||||||
self.networks[ssid]['connected'] = ssid == variables['ssid']
|
|
||||||
if prev_ssid != self.connected_ssid:
|
|
||||||
for cb in self._callbacks['connected']:
|
|
||||||
args = self.connected_ssid, prev_ssid
|
|
||||||
GLib.idle_add(cb, *args)
|
|
||||||
return [variables['ssid'], variables['bssid']]
|
|
||||||
else:
|
|
||||||
logging.info("Resetting connected_ssid")
|
|
||||||
self.connected = False
|
|
||||||
self.connected_ssid = None
|
|
||||||
for ssid, val in self.networks.items():
|
|
||||||
self.networks[ssid]['connected'] = False
|
|
||||||
if prev_ssid != self.connected_ssid:
|
|
||||||
for cb in self._callbacks['connected']:
|
|
||||||
args = self.connected_ssid, prev_ssid
|
|
||||||
GLib.idle_add(cb, *args)
|
|
||||||
return None
|
|
||||||
|
|
||||||
def get_current_wifi_idle_add(self):
|
|
||||||
self.get_current_wifi()
|
|
||||||
return False
|
|
||||||
|
|
||||||
def get_network_info(self, ssid=None, mac=None):
|
|
||||||
if ssid is not None and ssid in self.networks:
|
|
||||||
return self.networks[ssid]
|
|
||||||
if mac is not None and ssid is None:
|
|
||||||
for net in self.networks:
|
|
||||||
if mac == net['mac']:
|
|
||||||
return net
|
|
||||||
return {}
|
|
||||||
|
|
||||||
def get_networks(self):
|
|
||||||
return list(self.networks)
|
|
||||||
|
|
||||||
def get_supplicant_networks(self):
|
|
||||||
return self.supplicant_networks
|
|
||||||
|
|
||||||
def read_wpa_supplicant(self):
|
|
||||||
results = self.wpa_cli("LIST_NETWORKS").split('\n')
|
|
||||||
results.pop(0)
|
|
||||||
self.supplicant_networks = {}
|
|
||||||
self.networks_in_supplicant = []
|
|
||||||
for net in [n.split('\t') for n in results]:
|
|
||||||
self.supplicant_networks[net[0]] = {
|
|
||||||
"ssid": net[1],
|
|
||||||
"bssid": net[2],
|
|
||||||
"flags": net[3] if len(net) == 4 else ""
|
|
||||||
}
|
|
||||||
self.networks_in_supplicant.append(self.supplicant_networks[net[0]])
|
|
||||||
|
|
||||||
def rescan(self):
|
|
||||||
self.wpa_cli("SCAN", False)
|
|
||||||
|
|
||||||
def save_wpa_conf(self):
|
|
||||||
logging.info("Saving WPA config")
|
|
||||||
self.wpa_cli("SAVE_CONFIG")
|
|
||||||
|
|
||||||
def scan_results(self):
|
|
||||||
new_networks = []
|
|
||||||
deleted_networks = list(self.networks)
|
|
||||||
|
|
||||||
results = self.wpa_cli("SCAN_RESULTS").split('\n')
|
|
||||||
results.pop(0)
|
|
||||||
|
|
||||||
aps = []
|
|
||||||
for res in results:
|
|
||||||
match = re.match("^([a-f0-9:]+)\\s+([0-9]+)\\s+([\\-0-9]+)\\s+(\\S+)\\s+(.+)?", res)
|
|
||||||
if match:
|
|
||||||
net = {
|
|
||||||
"mac": match[1],
|
|
||||||
"channel": WifiChannels.lookup(match[2])[1],
|
|
||||||
"connected": False,
|
|
||||||
"configured": False,
|
|
||||||
"frequency": match[2],
|
|
||||||
"flags": match[4],
|
|
||||||
"signal_level_dBm": match[3],
|
|
||||||
"ssid": match[5]
|
|
||||||
}
|
|
||||||
|
|
||||||
if "WPA2" in net['flags']:
|
|
||||||
net['encryption'] = "WPA2"
|
|
||||||
elif "WPA" in net['flags']:
|
|
||||||
net['encryption'] = "WPA"
|
|
||||||
elif "WEP" in net['flags']:
|
|
||||||
net['encryption'] = "WEP"
|
|
||||||
else:
|
|
||||||
net['encryption'] = "off"
|
|
||||||
|
|
||||||
aps.append(net)
|
|
||||||
|
|
||||||
cur_info = self.get_current_wifi()
|
|
||||||
self.networks = {}
|
|
||||||
for ap in aps:
|
|
||||||
self.networks[ap['ssid']] = ap
|
|
||||||
if cur_info is not None and cur_info[0] == ap['ssid'] and cur_info[1].lower() == ap['mac'].lower():
|
|
||||||
self.networks[ap['ssid']]['connected'] = True
|
|
||||||
|
|
||||||
for net in list(self.networks):
|
|
||||||
if net in deleted_networks:
|
|
||||||
deleted_networks.remove(net)
|
|
||||||
else:
|
|
||||||
new_networks.append(net)
|
|
||||||
if new_networks or deleted_networks:
|
|
||||||
for cb in self._callbacks['scan_results']:
|
|
||||||
args = new_networks, deleted_networks
|
|
||||||
GLib.idle_add(cb, *args)
|
|
||||||
|
|
||||||
def wpa_cli(self, command, wait=True):
|
|
||||||
if wait is False:
|
|
||||||
self.wpa_thread.skip_command()
|
|
||||||
self.soc.send(command.encode())
|
|
||||||
if wait is True:
|
|
||||||
return self.queue.get()
|
|
||||||
|
|
||||||
def wpa_cli_batch(self, commands):
|
|
||||||
for cmd in commands:
|
|
||||||
self.wpa_cli(cmd)
|
|
||||||
|
|
||||||
|
|
||||||
class WpaSocket(Thread):
|
|
||||||
def __init__(self, wm, queue, callback):
|
|
||||||
super().__init__()
|
|
||||||
self.queue = queue
|
|
||||||
self.callback = callback
|
|
||||||
self.soc = wm.soc
|
|
||||||
self._stop_loop = False
|
|
||||||
self.skip_commands = 0
|
|
||||||
self.wm = wm
|
|
||||||
|
|
||||||
def run(self):
|
|
||||||
logging.debug("Setting up wifi event loop")
|
|
||||||
while self._stop_loop is False:
|
|
||||||
try:
|
|
||||||
msg = self.soc.recv(4096).decode().strip()
|
|
||||||
except Exception as e:
|
|
||||||
logging.critical(e, exc_info=True)
|
|
||||||
# TODO: Socket error
|
|
||||||
continue
|
|
||||||
if msg.startswith("<"):
|
|
||||||
if "CTRL-EVENT-SCAN-RESULTS" in msg:
|
|
||||||
GLib.idle_add(self.wm.scan_results)
|
|
||||||
elif "CTRL-EVENT-DISCONNECTED" in msg:
|
|
||||||
self.callback("connecting_status", msg)
|
|
||||||
match = re.match('<3>CTRL-EVENT-DISCONNECTED bssid=(\\S+) reason=3 locally_generated=1', msg)
|
|
||||||
if match:
|
|
||||||
for net in self.wm.networks:
|
|
||||||
if self.wm.networks[net]['mac'] == match[1]:
|
|
||||||
self.wm.networks[net]['connected'] = False
|
|
||||||
break
|
|
||||||
elif "Trying to associate" in msg or "CTRL-EVENT-REGDOM-CHANGE" in msg:
|
|
||||||
self.callback("connecting_status", msg)
|
|
||||||
elif "CTRL-EVENT-CONNECTED" in msg:
|
|
||||||
GLib.idle_add(self.wm.get_current_wifi_idle_add)
|
|
||||||
self.callback("connecting_status", msg)
|
|
||||||
elif self.skip_commands > 0:
|
|
||||||
self.skip_commands = self.skip_commands - 1
|
|
||||||
else:
|
|
||||||
self.queue.put(msg)
|
|
||||||
logging.info("Wifi event loop ended")
|
|
||||||
|
|
||||||
def skip_command(self):
|
|
||||||
self.skip_commands = self.skip_commands + 1
|
|
||||||
|
|
||||||
|
|
||||||
class WifiChannels:
|
|
||||||
@staticmethod
|
|
||||||
def lookup(freq: str):
|
|
||||||
if freq == '2484':
|
|
||||||
return "2.4", "14"
|
|
||||||
try:
|
|
||||||
freq = float(freq)
|
|
||||||
except ValueError:
|
|
||||||
return None
|
|
||||||
if 2412 <= freq <= 2472:
|
|
||||||
return "2.4", str(int((freq - 2407) / 5))
|
|
||||||
elif 3657.5 <= freq <= 3692.5:
|
|
||||||
return "3", str(int((freq - 3000) / 5))
|
|
||||||
elif 4915 <= freq <= 4980:
|
|
||||||
return "5", str(int((freq - 4000) / 5))
|
|
||||||
elif 5035 <= freq <= 5885:
|
|
||||||
return "5", str(int((freq - 5000) / 5))
|
|
||||||
elif 6455 <= freq <= 7115:
|
|
||||||
return "6", str(int((freq - 5950) / 5))
|
|
||||||
else:
|
|
||||||
return "?", "?"
|
|
@ -1,272 +0,0 @@
|
|||||||
# Network in KlipperScreen is a connection in NetworkManager
|
|
||||||
# Interface in KlipperScreen is a device in NetworkManager
|
|
||||||
|
|
||||||
import logging
|
|
||||||
import uuid
|
|
||||||
import dbus
|
|
||||||
import gi
|
|
||||||
|
|
||||||
gi.require_version('Gdk', '3.0')
|
|
||||||
from gi.repository import GLib
|
|
||||||
from contextlib import suppress
|
|
||||||
from ks_includes.wifi import WifiChannels
|
|
||||||
from ks_includes import NetworkManager
|
|
||||||
from dbus.mainloop.glib import DBusGMainLoop
|
|
||||||
|
|
||||||
|
|
||||||
class WifiManager:
|
|
||||||
networks_in_supplicant = []
|
|
||||||
|
|
||||||
def __init__(self, interface_name, *args, **kwargs):
|
|
||||||
super().__init__(*args, **kwargs)
|
|
||||||
DBusGMainLoop(set_as_default=True)
|
|
||||||
self._callbacks = {
|
|
||||||
"connected": [],
|
|
||||||
"connecting_status": [],
|
|
||||||
"scan_results": [],
|
|
||||||
"popup": [],
|
|
||||||
}
|
|
||||||
self.connected = False
|
|
||||||
self.connected_ssid = None
|
|
||||||
self.interface_name = interface_name
|
|
||||||
self.known_networks = {} # List of known connections
|
|
||||||
self.visible_networks = {} # List of visible access points
|
|
||||||
self.ssid_by_path = {}
|
|
||||||
self.path_by_ssid = {}
|
|
||||||
self.hidden_ssid_index = 0
|
|
||||||
|
|
||||||
self.wifi_dev = NetworkManager.NetworkManager.GetDeviceByIpIface(interface_name)
|
|
||||||
self.wifi_dev.OnAccessPointAdded(self._ap_added)
|
|
||||||
self.wifi_dev.OnAccessPointRemoved(self._ap_removed)
|
|
||||||
self.wifi_dev.OnStateChanged(self._ap_state_changed)
|
|
||||||
|
|
||||||
for ap in self.wifi_dev.GetAccessPoints():
|
|
||||||
self._add_ap(ap)
|
|
||||||
self._update_known_connections()
|
|
||||||
self.initialized = True
|
|
||||||
|
|
||||||
def _update_known_connections(self):
|
|
||||||
self.known_networks = {}
|
|
||||||
for con in NetworkManager.Settings.ListConnections():
|
|
||||||
settings = con.GetSettings()
|
|
||||||
if "802-11-wireless" in settings:
|
|
||||||
ssid = settings["802-11-wireless"]['ssid']
|
|
||||||
self.known_networks[ssid] = con
|
|
||||||
|
|
||||||
def _ap_added(self, nm, interface, signal, access_point):
|
|
||||||
with suppress(NetworkManager.ObjectVanished):
|
|
||||||
access_point.OnPropertiesChanged(self._ap_prop_changed)
|
|
||||||
ssid = self._add_ap(access_point)
|
|
||||||
for cb in self._callbacks['scan_results']:
|
|
||||||
args = (cb, [ssid], [])
|
|
||||||
GLib.idle_add(*args)
|
|
||||||
|
|
||||||
def _ap_removed(self, dev, interface, signal, access_point):
|
|
||||||
path = access_point.object_path
|
|
||||||
if path in self.ssid_by_path:
|
|
||||||
ssid = self.ssid_by_path[path]
|
|
||||||
self._remove_ap(path)
|
|
||||||
for cb in self._callbacks['scan_results']:
|
|
||||||
args = (cb, [ssid], [])
|
|
||||||
GLib.idle_add(*args)
|
|
||||||
|
|
||||||
def _ap_state_changed(self, nm, interface, signal, old_state, new_state, reason):
|
|
||||||
msg = ""
|
|
||||||
if new_state in (NetworkManager.NM_DEVICE_STATE_UNKNOWN, NetworkManager.NM_DEVICE_STATE_REASON_UNKNOWN):
|
|
||||||
msg = "State is unknown"
|
|
||||||
elif new_state == NetworkManager.NM_DEVICE_STATE_UNMANAGED:
|
|
||||||
msg = "Error: Not managed by NetworkManager"
|
|
||||||
elif new_state == NetworkManager.NM_DEVICE_STATE_UNAVAILABLE:
|
|
||||||
msg = "Error: Not available for use:\nReasons may include the wireless switched off, missing firmware, etc."
|
|
||||||
elif new_state == NetworkManager.NM_DEVICE_STATE_DISCONNECTED:
|
|
||||||
msg = "Currently disconnected"
|
|
||||||
elif new_state == NetworkManager.NM_DEVICE_STATE_PREPARE:
|
|
||||||
msg = "Preparing the connection to the network"
|
|
||||||
elif new_state == NetworkManager.NM_DEVICE_STATE_CONFIG:
|
|
||||||
msg = "Connecting to the requested network..."
|
|
||||||
elif new_state == NetworkManager.NM_DEVICE_STATE_NEED_AUTH:
|
|
||||||
msg = "Authorizing"
|
|
||||||
elif new_state == NetworkManager.NM_DEVICE_STATE_IP_CONFIG:
|
|
||||||
msg = "Requesting IP addresses and routing information"
|
|
||||||
elif new_state == NetworkManager.NM_DEVICE_STATE_IP_CHECK:
|
|
||||||
msg = "Checking whether further action is required for the requested network connection"
|
|
||||||
elif new_state == NetworkManager.NM_DEVICE_STATE_SECONDARIES:
|
|
||||||
msg = "Waiting for a secondary connection (like a VPN)"
|
|
||||||
elif new_state == NetworkManager.NM_DEVICE_STATE_ACTIVATED:
|
|
||||||
msg = "Connected"
|
|
||||||
elif new_state == NetworkManager.NM_DEVICE_STATE_DEACTIVATING:
|
|
||||||
msg = "A disconnection from the current network connection was requested"
|
|
||||||
elif new_state == NetworkManager.NM_DEVICE_STATE_FAILED:
|
|
||||||
msg = "Failed to connect to the requested network"
|
|
||||||
self.callback("popup", msg)
|
|
||||||
elif new_state == NetworkManager.NM_DEVICE_STATE_REASON_DEPENDENCY_FAILED:
|
|
||||||
msg = "A dependency of the connection failed"
|
|
||||||
elif new_state == NetworkManager.NM_DEVICE_STATE_REASON_CARRIER:
|
|
||||||
msg = ""
|
|
||||||
else:
|
|
||||||
logging.info(f"State {new_state}")
|
|
||||||
if msg != "":
|
|
||||||
self.callback("connecting_status", msg)
|
|
||||||
|
|
||||||
if new_state == NetworkManager.NM_DEVICE_STATE_ACTIVATED:
|
|
||||||
self.connected = True
|
|
||||||
for cb in self._callbacks['connected']:
|
|
||||||
args = (cb, self.get_connected_ssid(), None)
|
|
||||||
GLib.idle_add(*args)
|
|
||||||
else:
|
|
||||||
self.connected = False
|
|
||||||
|
|
||||||
def _ap_prop_changed(self, ap, interface, signal, properties):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def _add_ap(self, ap):
|
|
||||||
ssid = ap.Ssid
|
|
||||||
if ssid == "":
|
|
||||||
ssid = _("Hidden") + f" {self.hidden_ssid_index}"
|
|
||||||
self.hidden_ssid_index += 1
|
|
||||||
self.ssid_by_path[ap.object_path] = ssid
|
|
||||||
self.path_by_ssid[ssid] = ap.object_path
|
|
||||||
self.visible_networks[ap.object_path] = ap
|
|
||||||
return ssid
|
|
||||||
|
|
||||||
def _remove_ap(self, path):
|
|
||||||
self.ssid_by_path.pop(path, None)
|
|
||||||
self.visible_networks.pop(path, None)
|
|
||||||
|
|
||||||
def add_callback(self, name, callback):
|
|
||||||
if name in self._callbacks and callback not in self._callbacks[name]:
|
|
||||||
self._callbacks[name].append(callback)
|
|
||||||
|
|
||||||
def callback(self, cb_type, msg):
|
|
||||||
if cb_type in self._callbacks:
|
|
||||||
for cb in self._callbacks[cb_type]:
|
|
||||||
GLib.idle_add(cb, msg)
|
|
||||||
|
|
||||||
def add_network(self, ssid, psk):
|
|
||||||
aps = self._visible_networks_by_ssid()
|
|
||||||
if ssid in aps:
|
|
||||||
ap = aps[ssid]
|
|
||||||
new_connection = {
|
|
||||||
'802-11-wireless': {
|
|
||||||
'mode': 'infrastructure',
|
|
||||||
'security': '802-11-wireless-security',
|
|
||||||
'ssid': ssid
|
|
||||||
},
|
|
||||||
'802-11-wireless-security': {
|
|
||||||
'auth-alg': 'open',
|
|
||||||
'key-mgmt': 'wpa-psk',
|
|
||||||
'psk': psk
|
|
||||||
},
|
|
||||||
'connection': {
|
|
||||||
'id': ssid,
|
|
||||||
'type': '802-11-wireless',
|
|
||||||
'uuid': str(uuid.uuid4())
|
|
||||||
},
|
|
||||||
'ipv4': {
|
|
||||||
'method': 'auto'
|
|
||||||
},
|
|
||||||
'ipv6': {
|
|
||||||
'method': 'auto'
|
|
||||||
}
|
|
||||||
}
|
|
||||||
try:
|
|
||||||
NetworkManager.Settings.AddConnection(new_connection)
|
|
||||||
except dbus.exceptions.DBusException as e:
|
|
||||||
msg = _("Invalid password") if "802-11-wireless-security.psk" in e else f"{e}"
|
|
||||||
self.callback("popup", msg)
|
|
||||||
logging.info(f"Error adding network {e}")
|
|
||||||
self._update_known_connections()
|
|
||||||
return True
|
|
||||||
|
|
||||||
def connect(self, ssid):
|
|
||||||
if ssid in self.known_networks:
|
|
||||||
conn = self.known_networks[ssid]
|
|
||||||
with suppress(NetworkManager.ObjectVanished):
|
|
||||||
msg = f"Connecting to: {ssid}"
|
|
||||||
logging.info(msg)
|
|
||||||
self.callback("connecting_status", msg)
|
|
||||||
NetworkManager.NetworkManager.ActivateConnection(conn, self.wifi_dev, "/")
|
|
||||||
|
|
||||||
def delete_network(self, ssid):
|
|
||||||
for ssid in self.known_networks:
|
|
||||||
con = self.known_networks[ssid]
|
|
||||||
if con.GetSettings()['connection']['id'] == ssid:
|
|
||||||
con.Delete()
|
|
||||||
self._update_known_connections()
|
|
||||||
|
|
||||||
def get_connected_ssid(self):
|
|
||||||
if self.wifi_dev.SpecificDevice().ActiveAccessPoint:
|
|
||||||
return self.wifi_dev.SpecificDevice().ActiveAccessPoint.Ssid
|
|
||||||
return None
|
|
||||||
|
|
||||||
def _get_connected_ap(self):
|
|
||||||
return self.wifi_dev.SpecificDevice().ActiveAccessPoint
|
|
||||||
|
|
||||||
def _visible_networks_by_ssid(self):
|
|
||||||
aps = self.wifi_dev.GetAccessPoints()
|
|
||||||
ret = {}
|
|
||||||
for ap in aps:
|
|
||||||
with suppress(NetworkManager.ObjectVanished):
|
|
||||||
ret[ap.Ssid] = ap
|
|
||||||
return ret
|
|
||||||
|
|
||||||
def get_network_info(self, ssid):
|
|
||||||
netinfo = {}
|
|
||||||
if ssid in self.known_networks:
|
|
||||||
con = self.known_networks[ssid]
|
|
||||||
with suppress(NetworkManager.ObjectVanished):
|
|
||||||
settings = con.GetSettings()
|
|
||||||
if settings and '802-11-wireless' in settings:
|
|
||||||
netinfo.update({
|
|
||||||
"ssid": settings['802-11-wireless']['ssid'],
|
|
||||||
"connected": self.get_connected_ssid() == ssid
|
|
||||||
})
|
|
||||||
path = self.path_by_ssid[ssid]
|
|
||||||
aps = self.visible_networks
|
|
||||||
if path in aps:
|
|
||||||
ap = aps[path]
|
|
||||||
with suppress(NetworkManager.ObjectVanished):
|
|
||||||
netinfo.update({
|
|
||||||
"mac": ap.HwAddress,
|
|
||||||
"channel": WifiChannels.lookup(ap.Frequency)[1],
|
|
||||||
"configured": ssid in self.known_networks,
|
|
||||||
"frequency": str(ap.Frequency),
|
|
||||||
"flags": ap.Flags,
|
|
||||||
"ssid": ssid,
|
|
||||||
"connected": self._get_connected_ap() == ap,
|
|
||||||
"encryption": self._get_encryption(ap.RsnFlags),
|
|
||||||
"signal_level_dBm": str(ap.Strength)
|
|
||||||
})
|
|
||||||
return netinfo
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def _get_encryption(flags):
|
|
||||||
encryption = ""
|
|
||||||
if (flags & NetworkManager.NM_802_11_AP_SEC_PAIR_WEP40 or
|
|
||||||
flags & NetworkManager.NM_802_11_AP_SEC_PAIR_WEP104 or
|
|
||||||
flags & NetworkManager.NM_802_11_AP_SEC_GROUP_WEP40 or
|
|
||||||
flags & NetworkManager.NM_802_11_AP_SEC_GROUP_WEP104):
|
|
||||||
encryption += "WEP "
|
|
||||||
if (flags & NetworkManager.NM_802_11_AP_SEC_PAIR_TKIP or
|
|
||||||
flags & NetworkManager.NM_802_11_AP_SEC_GROUP_TKIP):
|
|
||||||
encryption += "TKIP "
|
|
||||||
if (flags & NetworkManager.NM_802_11_AP_SEC_PAIR_CCMP or
|
|
||||||
flags & NetworkManager.NM_802_11_AP_SEC_GROUP_CCMP):
|
|
||||||
encryption += "AES "
|
|
||||||
if flags & NetworkManager.NM_802_11_AP_SEC_KEY_MGMT_PSK:
|
|
||||||
encryption += "WPA-PSK "
|
|
||||||
if flags & NetworkManager.NM_802_11_AP_SEC_KEY_MGMT_802_1X:
|
|
||||||
encryption += "802.1x "
|
|
||||||
return encryption.strip()
|
|
||||||
|
|
||||||
def get_networks(self):
|
|
||||||
return list(set(list(self.known_networks.keys()) + list(self.ssid_by_path.values())))
|
|
||||||
|
|
||||||
def get_supplicant_networks(self):
|
|
||||||
return {ssid: {"ssid": ssid} for ssid in self.known_networks.keys()}
|
|
||||||
|
|
||||||
def rescan(self):
|
|
||||||
try:
|
|
||||||
self.wifi_dev.RequestScan({})
|
|
||||||
except dbus.exceptions.DBusException as e:
|
|
||||||
logging.error(f"Error during rescan {e}")
|
|
@ -1,143 +1,99 @@
|
|||||||
import logging
|
import logging
|
||||||
import os
|
import os
|
||||||
import gi
|
import gi
|
||||||
import netifaces
|
|
||||||
|
|
||||||
gi.require_version("Gtk", "3.0")
|
gi.require_version("Gtk", "3.0")
|
||||||
from gi.repository import Gtk, GLib, Pango
|
from gi.repository import Gtk, GLib, Pango
|
||||||
from ks_includes.screen_panel import ScreenPanel
|
from ks_includes.screen_panel import ScreenPanel
|
||||||
|
from ks_includes.sdbus_nm import SdbusNm
|
||||||
|
|
||||||
|
|
||||||
class Panel(ScreenPanel):
|
class Panel(ScreenPanel):
|
||||||
initialized = False
|
|
||||||
|
|
||||||
def __init__(self, screen, title):
|
def __init__(self, screen, title):
|
||||||
super().__init__(screen, title)
|
super().__init__(screen, title)
|
||||||
self.show_add = False
|
self.show_add = False
|
||||||
self.networks = {}
|
|
||||||
self.interface = None
|
|
||||||
self.prev_network = None
|
|
||||||
self.update_timeout = None
|
self.update_timeout = None
|
||||||
self.network_interfaces = netifaces.interfaces()
|
self.network_list = Gtk.Box(orientation=Gtk.Orientation.VERTICAL, hexpand=True, vexpand=True)
|
||||||
self.wireless_interfaces = [iface for iface in self.network_interfaces if iface.startswith('wl')]
|
self.network_rows = {}
|
||||||
self.wifi = None
|
self.networks = {}
|
||||||
self.use_network_manager = os.system('systemctl is-active --quiet NetworkManager.service') == 0
|
self.sdbus_nm = SdbusNm()
|
||||||
if self.wireless_interfaces:
|
self.wifi_signal_icons = {
|
||||||
logging.info(f"Found wireless interfaces: {self.wireless_interfaces}")
|
'excellent': self._gtk.PixbufFromIcon('wifi_excellent'),
|
||||||
if self.use_network_manager:
|
'good': self._gtk.PixbufFromIcon('wifi_good'),
|
||||||
logging.info("Using NetworkManager")
|
'fair': self._gtk.PixbufFromIcon('wifi_fair'),
|
||||||
from ks_includes.wifi_nm import WifiManager
|
'weak': self._gtk.PixbufFromIcon('wifi_weak'),
|
||||||
else:
|
}
|
||||||
logging.info("Using wpa_cli")
|
|
||||||
from ks_includes.wifi import WifiManager
|
|
||||||
self.wifi = WifiManager(self.wireless_interfaces[0])
|
|
||||||
else:
|
|
||||||
logging.info(_("No wireless interface has been found"))
|
|
||||||
|
|
||||||
# Get IP Address
|
self.network_interfaces = self.sdbus_nm.get_interfaces()
|
||||||
gws = netifaces.gateways()
|
logging.info(f"Network interfaces: {self.network_interfaces}")
|
||||||
if "default" in gws and netifaces.AF_INET in gws["default"]:
|
|
||||||
self.interface = gws["default"][netifaces.AF_INET][1]
|
|
||||||
else:
|
|
||||||
ints = netifaces.interfaces()
|
|
||||||
if 'lo' in ints:
|
|
||||||
ints.pop(ints.index('lo'))
|
|
||||||
self.interface = ints[0] if len(ints) > 0 else 'lo'
|
|
||||||
|
|
||||||
self.labels['networks'] = {}
|
self.wireless_interfaces = [iface.interface for iface in self.sdbus_nm.get_wireless_interfaces()]
|
||||||
|
logging.info(f"Wireless interfaces: {self.wireless_interfaces}")
|
||||||
|
|
||||||
|
self.interface = self.sdbus_nm.get_primary_interface()
|
||||||
|
logging.info(f"Primary interface: {self.interface}")
|
||||||
|
|
||||||
self.labels['interface'] = Gtk.Label(hexpand=True)
|
self.labels['interface'] = Gtk.Label(hexpand=True)
|
||||||
self.labels['interface'].set_text(_("Interface") + f': {self.interface} ')
|
|
||||||
|
|
||||||
self.labels['ip'] = Gtk.Label(hexpand=True)
|
self.labels['ip'] = Gtk.Label(hexpand=True)
|
||||||
ifadd = netifaces.ifaddresses(self.interface)
|
if self.interface is not None:
|
||||||
if ifadd.get(netifaces.AF_INET):
|
self.labels['interface'].set_text(_("Interface") + f': {self.interface}')
|
||||||
self.labels['ip'].set_text(f"IP: {ifadd[netifaces.AF_INET][0]['addr']} ")
|
self.labels['ip'].set_text(f"IP: {self.sdbus_nm.get_ip_address()}")
|
||||||
|
|
||||||
reload_networks = self._gtk.Button("refresh", None, "color1", self.bts)
|
self.reload_button = self._gtk.Button("refresh", None, "color1", self.bts)
|
||||||
reload_networks.connect("clicked", self.reload_networks)
|
self.reload_button.set_no_show_all(True)
|
||||||
reload_networks.set_hexpand(False)
|
self.reload_button.show()
|
||||||
|
self.reload_button.connect("clicked", self.reload_networks)
|
||||||
|
self.reload_button.set_hexpand(False)
|
||||||
|
|
||||||
|
self.wifi_toggle = Gtk.Switch(
|
||||||
|
width_request=round(self._gtk.font_size * 2),
|
||||||
|
height_request=round(self._gtk.font_size),
|
||||||
|
active=self.sdbus_nm.is_wifi_enabled()
|
||||||
|
)
|
||||||
|
self.wifi_toggle.connect("notify::active", self.toggle_wifi)
|
||||||
|
|
||||||
sbox = Gtk.Box(hexpand=True, vexpand=False)
|
sbox = Gtk.Box(hexpand=True, vexpand=False)
|
||||||
sbox.add(self.labels['interface'])
|
sbox.add(self.labels['interface'])
|
||||||
sbox.add(self.labels['ip'])
|
sbox.add(self.labels['ip'])
|
||||||
sbox.add(reload_networks)
|
sbox.add(self.reload_button)
|
||||||
|
sbox.add(self.wifi_toggle)
|
||||||
|
|
||||||
scroll = self._gtk.ScrolledWindow()
|
scroll = self._gtk.ScrolledWindow()
|
||||||
|
self.labels['main_box'] = Gtk.Box(orientation=Gtk.Orientation.VERTICAL, vexpand=True)
|
||||||
|
|
||||||
box = Gtk.Box(orientation=Gtk.Orientation.VERTICAL, vexpand=True)
|
if self.sdbus_nm.wifi:
|
||||||
|
self.labels['main_box'].pack_start(sbox, False, False, 5)
|
||||||
self.labels['networklist'] = Gtk.Grid()
|
|
||||||
|
|
||||||
if self.wifi is not None and self.wifi.initialized:
|
|
||||||
box.pack_start(sbox, False, False, 5)
|
|
||||||
box.pack_start(scroll, True, True, 0)
|
|
||||||
|
|
||||||
GLib.idle_add(self.load_networks)
|
GLib.idle_add(self.load_networks)
|
||||||
scroll.add(self.labels['networklist'])
|
scroll.add(self.network_list)
|
||||||
|
|
||||||
self.wifi.add_callback("connected", self.connected_callback)
|
self.sdbus_nm.add_callback("popup", self.popup_callback)
|
||||||
self.wifi.add_callback("scan_results", self.scan_callback)
|
|
||||||
self.wifi.add_callback("popup", self.popup_callback)
|
|
||||||
if self.update_timeout is None:
|
|
||||||
self.update_timeout = GLib.timeout_add_seconds(5, self.update_all_networks)
|
|
||||||
else:
|
else:
|
||||||
|
self._screen.show_popup_message(_("No wireless interface has been found"), level=2)
|
||||||
self.labels['networkinfo'] = Gtk.Label()
|
self.labels['networkinfo'] = Gtk.Label()
|
||||||
self.labels['networkinfo'].get_style_context().add_class('temperature_entry')
|
scroll.add(self.labels['networkinfo'])
|
||||||
box.pack_start(self.labels['networkinfo'], False, False, 0)
|
|
||||||
self.update_single_network_info()
|
self.update_single_network_info()
|
||||||
if self.update_timeout is None:
|
|
||||||
self.update_timeout = GLib.timeout_add_seconds(5, self.update_single_network_info)
|
|
||||||
|
|
||||||
self.content.add(box)
|
self.labels['main_box'].pack_start(scroll, True, True, 0)
|
||||||
self.labels['main_box'] = box
|
self.content.add(self.labels['main_box'])
|
||||||
self.initialized = True
|
|
||||||
|
|
||||||
def load_networks(self, widget=None):
|
def popup_callback(self, msg, level=3):
|
||||||
networks = self.wifi.get_networks()
|
self._screen.show_popup_message(msg, level)
|
||||||
if not networks:
|
|
||||||
return
|
def load_networks(self):
|
||||||
for net in networks:
|
for net in self.sdbus_nm.get_networks():
|
||||||
self.add_network(net, False)
|
self.add_network(net['BSSID'])
|
||||||
self.update_all_networks()
|
GLib.timeout_add_seconds(10, self._gtk.Button_busy, self.reload_button, False)
|
||||||
if widget:
|
|
||||||
GLib.timeout_add_seconds(10, self._gtk.Button_busy, widget, False)
|
|
||||||
self.content.show_all()
|
self.content.show_all()
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def add_network(self, ssid, show=True):
|
def add_network(self, bssid):
|
||||||
|
if bssid in self.network_rows:
|
||||||
if ssid is None:
|
logging.info(f"{bssid} already in list")
|
||||||
return
|
|
||||||
ssid = ssid.strip()
|
|
||||||
if ssid in list(self.networks):
|
|
||||||
return
|
return
|
||||||
|
|
||||||
configured_networks = self.wifi.get_supplicant_networks()
|
net = next(net for net in self.sdbus_nm.get_networks() if bssid == net['BSSID'])
|
||||||
network_id = -1
|
ssid = net['SSID']
|
||||||
for net in list(configured_networks):
|
|
||||||
if configured_networks[net]['ssid'] == ssid:
|
|
||||||
network_id = net
|
|
||||||
|
|
||||||
display_name = _("Hidden") if ssid.startswith("\x00") else f"{ssid}"
|
|
||||||
netinfo = self.wifi.get_network_info(ssid)
|
|
||||||
connected_ssid = self.wifi.get_connected_ssid()
|
|
||||||
if netinfo is None:
|
|
||||||
logging.debug("Couldn't get netinfo")
|
|
||||||
netinfo = {'connected': connected_ssid == ssid}
|
|
||||||
|
|
||||||
name = Gtk.Label(hexpand=True, halign=Gtk.Align.START, wrap=True, wrap_mode=Pango.WrapMode.WORD_CHAR)
|
|
||||||
if connected_ssid == ssid:
|
|
||||||
display_name += " (" + _("Connected") + ")"
|
|
||||||
name.set_markup(f"<big><b>{display_name}</b></big>")
|
|
||||||
else:
|
|
||||||
name.set_label(display_name)
|
|
||||||
|
|
||||||
info = Gtk.Label(halign=Gtk.Align.START)
|
|
||||||
labels = Gtk.Box(orientation=Gtk.Orientation.VERTICAL, vexpand=True,
|
|
||||||
halign=Gtk.Align.START, valign=Gtk.Align.CENTER)
|
|
||||||
labels.add(name)
|
|
||||||
labels.add(info)
|
|
||||||
|
|
||||||
connect = self._gtk.Button("load", None, "color3", self.bts)
|
connect = self._gtk.Button("load", None, "color3", self.bts)
|
||||||
connect.connect("clicked", self.connect_network, ssid)
|
connect.connect("clicked", self.connect_network, ssid)
|
||||||
@ -145,54 +101,79 @@ class Panel(ScreenPanel):
|
|||||||
connect.set_halign(Gtk.Align.END)
|
connect.set_halign(Gtk.Align.END)
|
||||||
|
|
||||||
delete = self._gtk.Button("delete", None, "color3", self.bts)
|
delete = self._gtk.Button("delete", None, "color3", self.bts)
|
||||||
delete.connect("clicked", self.remove_wifi_network, ssid)
|
delete.connect("clicked", self.remove_confirm_dialog, ssid, bssid)
|
||||||
delete.set_hexpand(False)
|
delete.set_hexpand(False)
|
||||||
delete.set_halign(Gtk.Align.END)
|
delete.set_halign(Gtk.Align.END)
|
||||||
|
|
||||||
network = Gtk.Box(spacing=5, hexpand=True, vexpand=False)
|
|
||||||
network.get_style_context().add_class("frame-item")
|
|
||||||
network.add(labels)
|
|
||||||
|
|
||||||
buttons = Gtk.Box(spacing=5)
|
buttons = Gtk.Box(spacing=5)
|
||||||
if network_id != -1 or netinfo['connected']:
|
|
||||||
buttons.pack_end(connect, False, False, 0)
|
|
||||||
buttons.pack_end(delete, False, False, 0)
|
|
||||||
else:
|
|
||||||
buttons.pack_end(connect, False, False, 0)
|
|
||||||
network.add(buttons)
|
|
||||||
self.networks[ssid] = network
|
|
||||||
|
|
||||||
nets = sorted(list(self.networks), reverse=False)
|
name = Gtk.Label(hexpand=True, halign=Gtk.Align.START, wrap=True, wrap_mode=Pango.WrapMode.WORD_CHAR)
|
||||||
if connected_ssid in nets:
|
if bssid == self.sdbus_nm.get_connected_bssid():
|
||||||
nets.remove(connected_ssid)
|
ssid += ' (' + _("Connected") + ')'
|
||||||
nets.insert(0, connected_ssid)
|
name.set_markup(f"<big><b>{ssid}</b></big>")
|
||||||
if nets.index(ssid) is not None:
|
|
||||||
pos = nets.index(ssid)
|
|
||||||
else:
|
else:
|
||||||
logging.info("Error: SSID not in nets")
|
name.set_markup(f"<b>{ssid}</b>")
|
||||||
return
|
if net['known']:
|
||||||
|
buttons.add(delete)
|
||||||
|
buttons.add(connect)
|
||||||
|
|
||||||
self.labels['networks'][ssid] = {
|
info = Gtk.Label(halign=Gtk.Align.START)
|
||||||
|
labels = Gtk.Box(orientation=Gtk.Orientation.VERTICAL, vexpand=True,
|
||||||
|
halign=Gtk.Align.START, valign=Gtk.Align.CENTER)
|
||||||
|
labels.add(name)
|
||||||
|
labels.add(info)
|
||||||
|
icon = self._gtk.Image('wifi_weak')
|
||||||
|
|
||||||
|
self.network_rows[bssid] = Gtk.Box(spacing=5, hexpand=True, vexpand=False)
|
||||||
|
self.network_rows[bssid].get_style_context().add_class("frame-item")
|
||||||
|
self.network_rows[bssid].add(icon)
|
||||||
|
self.network_rows[bssid].add(labels)
|
||||||
|
self.network_rows[bssid].add(buttons)
|
||||||
|
|
||||||
|
self.networks[bssid] = {
|
||||||
"connect": connect,
|
"connect": connect,
|
||||||
"delete": delete,
|
"delete": delete,
|
||||||
|
"icon": icon,
|
||||||
"info": info,
|
"info": info,
|
||||||
"name": name,
|
"name": name,
|
||||||
"row": network
|
"row": self.network_rows[bssid],
|
||||||
}
|
}
|
||||||
|
|
||||||
self.labels['networklist'].insert_row(pos)
|
self.network_list.add(self.network_rows[bssid])
|
||||||
self.labels['networklist'].attach(self.networks[ssid], 0, pos, 1, 1)
|
|
||||||
if show:
|
def remove_confirm_dialog(self, widget, ssid, bssid):
|
||||||
self.labels['networklist'].show()
|
|
||||||
|
label = Gtk.Label(wrap=True, vexpand=True)
|
||||||
|
label.set_markup(_("Do you want to forget or disconnect %s?") % ssid)
|
||||||
|
buttons = [
|
||||||
|
{"name": _("Forget"), "response": Gtk.ResponseType.OK, "style": 'dialog-warning'},
|
||||||
|
{"name": _("Cancel"), "response": Gtk.ResponseType.CANCEL, "style": 'dialog-error'},
|
||||||
|
]
|
||||||
|
if bssid == self.sdbus_nm.get_connected_bssid():
|
||||||
|
buttons.insert(0, {"name": _("Disconnect"), "response": Gtk.ResponseType.APPLY, "style": 'dialog-info'})
|
||||||
|
self._gtk.Dialog(_("Remove network"), buttons, label, self.confirm_removal, ssid)
|
||||||
|
|
||||||
|
def confirm_removal(self, dialog, response_id, ssid):
|
||||||
|
self._gtk.remove_dialog(dialog)
|
||||||
|
if response_id == Gtk.ResponseType.CANCEL:
|
||||||
|
return
|
||||||
|
bssid = self.sdbus_nm.get_bssid_from_ssid(ssid)
|
||||||
|
self.remove_network_from_list(bssid)
|
||||||
|
if response_id == Gtk.ResponseType.OK:
|
||||||
|
logging.info(f"Deleting {ssid}")
|
||||||
|
self.sdbus_nm.delete_network(ssid)
|
||||||
|
if response_id == Gtk.ResponseType.APPLY:
|
||||||
|
logging.info(f"Disconnecting {ssid}")
|
||||||
|
self.sdbus_nm.disconnect_network()
|
||||||
|
|
||||||
def add_new_network(self, widget, ssid):
|
def add_new_network(self, widget, ssid):
|
||||||
self._screen.remove_keyboard()
|
self._screen.remove_keyboard()
|
||||||
result = self.wifi.add_network(ssid, self.labels['network_psk'].get_text())
|
result = self.sdbus_nm.add_network(ssid, self.labels['network_psk'].get_text())
|
||||||
self.close_add_network()
|
self.close_add_network()
|
||||||
if result:
|
if result:
|
||||||
self.connect_network(widget, ssid, False)
|
self.connect_network(widget, ssid, showadd=False)
|
||||||
else:
|
else:
|
||||||
self._screen.show_popup_message(f"Error adding network {ssid}")
|
self._screen.show_popup_message(_("Invalid password"))
|
||||||
|
|
||||||
def back(self):
|
def back(self):
|
||||||
if self.show_add:
|
if self.show_add:
|
||||||
@ -200,16 +181,6 @@ class Panel(ScreenPanel):
|
|||||||
return True
|
return True
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def check_missing_networks(self):
|
|
||||||
networks = self.wifi.get_networks()
|
|
||||||
for net in list(self.networks):
|
|
||||||
if net in networks:
|
|
||||||
networks.remove(net)
|
|
||||||
|
|
||||||
for net in networks:
|
|
||||||
self.add_network(net, False)
|
|
||||||
self.labels['networklist'].show_all()
|
|
||||||
|
|
||||||
def close_add_network(self):
|
def close_add_network(self):
|
||||||
if not self.show_add:
|
if not self.show_add:
|
||||||
return
|
return
|
||||||
@ -223,71 +194,27 @@ class Panel(ScreenPanel):
|
|||||||
del self.labels[i]
|
del self.labels[i]
|
||||||
self.show_add = False
|
self.show_add = False
|
||||||
|
|
||||||
def popup_callback(self, msg):
|
|
||||||
self._screen.show_popup_message(msg)
|
|
||||||
|
|
||||||
def connected_callback(self, ssid, prev_ssid):
|
|
||||||
logging.info("Now connected to a new network")
|
|
||||||
if ssid is not None:
|
|
||||||
self.remove_network(ssid)
|
|
||||||
if prev_ssid is not None:
|
|
||||||
self.remove_network(prev_ssid)
|
|
||||||
|
|
||||||
self.check_missing_networks()
|
|
||||||
|
|
||||||
def connect_network(self, widget, ssid, showadd=True):
|
def connect_network(self, widget, ssid, showadd=True):
|
||||||
isdef = any(net['ssid'] == ssid for netid, net in self.wifi.get_supplicant_networks().items())
|
self.deactivate()
|
||||||
if not isdef:
|
if showadd and not self.sdbus_nm.is_known(ssid):
|
||||||
if showadd:
|
self.show_add_network(widget, ssid)
|
||||||
self.show_add_network(widget, ssid)
|
self.activate()
|
||||||
return
|
return
|
||||||
self.prev_network = self.wifi.get_connected_ssid()
|
bssid = self.sdbus_nm.get_bssid_from_ssid(ssid)
|
||||||
|
if bssid and bssid in self.network_rows:
|
||||||
|
self.remove_network_from_list(bssid)
|
||||||
|
self.sdbus_nm.connect(ssid)
|
||||||
|
self.update_all_networks()
|
||||||
|
self.activate()
|
||||||
|
|
||||||
buttons = [
|
def remove_network_from_list(self, bssid):
|
||||||
{"name": _("Close"), "response": Gtk.ResponseType.CANCEL}
|
if bssid not in self.network_rows:
|
||||||
]
|
logging.error(f"{bssid} not in rows")
|
||||||
|
|
||||||
scroll = self._gtk.ScrolledWindow()
|
|
||||||
self.labels['connecting_info'] = Gtk.Label(
|
|
||||||
label=_("Starting WiFi Association"), halign=Gtk.Align.START, valign=Gtk.Align.START, wrap=True)
|
|
||||||
scroll.add(self.labels['connecting_info'])
|
|
||||||
self._gtk.Dialog(_("Starting WiFi Association"), buttons, scroll, self._gtk.remove_dialog)
|
|
||||||
self._screen.show_all()
|
|
||||||
|
|
||||||
if ssid in list(self.networks):
|
|
||||||
self.remove_network(ssid)
|
|
||||||
if self.prev_network in list(self.networks):
|
|
||||||
self.remove_network(self.prev_network)
|
|
||||||
|
|
||||||
self.wifi.add_callback("connecting_status", self.connecting_status_callback)
|
|
||||||
self.wifi.connect(ssid)
|
|
||||||
|
|
||||||
def connecting_status_callback(self, msg):
|
|
||||||
self.labels['connecting_info'].set_text(f"{self.labels['connecting_info'].get_text()}\n{msg}")
|
|
||||||
self.labels['connecting_info'].show_all()
|
|
||||||
|
|
||||||
def remove_network(self, ssid, show=True):
|
|
||||||
if ssid not in list(self.networks):
|
|
||||||
return
|
return
|
||||||
for i in range(len(self.labels['networklist'])):
|
self.network_list.remove(self.network_rows[bssid])
|
||||||
if self.networks[ssid] == self.labels['networklist'].get_child_at(0, i):
|
del self.network_rows[bssid]
|
||||||
self.labels['networklist'].remove_row(i)
|
del self.networks[bssid]
|
||||||
self.labels['networklist'].show()
|
return
|
||||||
del self.networks[ssid]
|
|
||||||
del self.labels['networks'][ssid]
|
|
||||||
return
|
|
||||||
|
|
||||||
def remove_wifi_network(self, widget, ssid):
|
|
||||||
self.wifi.delete_network(ssid)
|
|
||||||
self.remove_network(ssid)
|
|
||||||
self.check_missing_networks()
|
|
||||||
|
|
||||||
def scan_callback(self, new_networks, old_networks):
|
|
||||||
for net in old_networks:
|
|
||||||
self.remove_network(net, False)
|
|
||||||
for net in new_networks:
|
|
||||||
self.add_network(net, False)
|
|
||||||
self.content.show_all()
|
|
||||||
|
|
||||||
def show_add_network(self, widget, ssid):
|
def show_add_network(self, widget, ssid):
|
||||||
if self.show_add:
|
if self.show_add:
|
||||||
@ -323,100 +250,96 @@ class Panel(ScreenPanel):
|
|||||||
self.show_add = True
|
self.show_add = True
|
||||||
|
|
||||||
def update_all_networks(self):
|
def update_all_networks(self):
|
||||||
for network in list(self.networks):
|
self.interface = self.sdbus_nm.get_primary_interface()
|
||||||
self.update_network_info(network)
|
self.labels['interface'].set_text(_("Interface") + f': {self.interface}')
|
||||||
|
self.labels['ip'].set_text(f"IP: {self.sdbus_nm.get_ip_address()}")
|
||||||
|
nets = self.sdbus_nm.get_networks()
|
||||||
|
remove = [bssid for bssid in self.network_rows.keys() if bssid not in [net['BSSID'] for net in nets]]
|
||||||
|
for bssid in remove:
|
||||||
|
self.remove_network_from_list(bssid)
|
||||||
|
for net in nets:
|
||||||
|
if net['BSSID'] not in self.network_rows.keys():
|
||||||
|
self.add_network(net['BSSID'])
|
||||||
|
self.update_network_info(net)
|
||||||
|
for i, net in enumerate(nets):
|
||||||
|
for child in self.network_list.get_children():
|
||||||
|
if child == self.network_rows[net['BSSID']]:
|
||||||
|
self.network_list.reorder_child(child, i)
|
||||||
|
self.network_list.show_all()
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def update_network_info(self, ssid):
|
def update_network_info(self, net):
|
||||||
info = freq = encr = chan = lvl = ipv4 = ipv6 = ""
|
if net['BSSID'] not in self.network_rows.keys() or net['BSSID'] not in self.networks:
|
||||||
|
logging.info(f"Unknown SSID {net['SSID']}")
|
||||||
if ssid not in list(self.networks) or ssid not in self.labels['networks']:
|
|
||||||
logging.info(f"Unknown SSID {ssid}")
|
|
||||||
return
|
return
|
||||||
netinfo = self.wifi.get_network_info(ssid)
|
info = _("Password saved") + '\n' if net['known'] else ""
|
||||||
if netinfo.get('connected') or self.wifi.get_connected_ssid() == ssid:
|
chan = _("Channel") + f' {net["channel"]}'
|
||||||
ifadd = netifaces.ifaddresses(self.interface)
|
max_bitrate = _("Max:") + f"{self.format_speed(net['max_bitrate'])}"
|
||||||
if ifadd.get(netifaces.AF_INET):
|
self.networks[net['BSSID']]['icon'].set_from_pixbuf(self.get_signal_strength_icon(net["signal_level"]))
|
||||||
ipv4 = f"<b>IPv4:</b> {ifadd[netifaces.AF_INET][0]['addr']}"
|
self.networks[net['BSSID']]['info'].set_markup(
|
||||||
self.labels['ip'].set_text(f"IP: {ifadd[netifaces.AF_INET][0]['addr']} ")
|
"<small>"
|
||||||
if ifadd.get(netifaces.AF_INET6):
|
f"{info}"
|
||||||
ipv6 = f"<b>IPv6:</b> {ifadd[netifaces.AF_INET6][0]['addr'].split('%')[0]}"
|
f"{net['security']}\n"
|
||||||
|
f"{max_bitrate}\n"
|
||||||
|
f"{net['frequency']} Ghz {chan} {net['signal_level']} %\n"
|
||||||
|
f"{net['BSSID']}"
|
||||||
|
"</small>"
|
||||||
|
)
|
||||||
|
|
||||||
info = '<b>' + _("Hostname") + f':</b> {os.uname().nodename}\n{ipv4}\n{ipv6}'
|
def get_signal_strength_icon(self, signal_level):
|
||||||
else:
|
|
||||||
self.labels['networks'][ssid]['name'].set_label(_("Hidden") if ssid.startswith("\x00") else f"{ssid}")
|
|
||||||
if "psk" in netinfo:
|
|
||||||
info = _("Password saved")
|
|
||||||
if "encryption" in netinfo and netinfo['encryption'] != "off":
|
|
||||||
encr = netinfo['encryption'].upper()
|
|
||||||
if "frequency" in netinfo:
|
|
||||||
freq = "2.4 GHz" if netinfo['frequency'][:1] == "2" else "5 Ghz"
|
|
||||||
if "channel" in netinfo:
|
|
||||||
chan = _("Channel") + f' {netinfo["channel"]}'
|
|
||||||
if "signal_level_dBm" in netinfo:
|
|
||||||
unit = "%" if self.use_network_manager else _("dBm")
|
|
||||||
lvl = f"{netinfo['signal_level_dBm']} {unit}"
|
|
||||||
icon = self.signal_strength(int(netinfo["signal_level_dBm"]))
|
|
||||||
if 'icon' not in self.labels['networks'][ssid]:
|
|
||||||
self.labels['networks'][ssid]['row'].add(icon)
|
|
||||||
self.labels['networks'][ssid]['row'].reorder_child(icon, 0)
|
|
||||||
self.labels['networks'][ssid]['icon'] = icon
|
|
||||||
self.labels['networks'][ssid]['icon'] = icon
|
|
||||||
|
|
||||||
self.labels['networks'][ssid]['info'].set_markup(f"{info}\n<small>{encr} {freq} {chan} {lvl}</small>")
|
|
||||||
self.labels['networks'][ssid]['row'].show_all()
|
|
||||||
|
|
||||||
def signal_strength(self, signal_level):
|
|
||||||
# networkmanager uses percentage not dbm
|
# networkmanager uses percentage not dbm
|
||||||
# the bars of nmcli are aligned near this breakpoints
|
if signal_level > 75:
|
||||||
exc = 77 if self.use_network_manager else -50
|
return self.wifi_signal_icons['excellent']
|
||||||
good = 60 if self.use_network_manager else -60
|
elif signal_level > 60:
|
||||||
fair = 35 if self.use_network_manager else -70
|
return self.wifi_signal_icons['good']
|
||||||
if signal_level > exc:
|
elif signal_level > 30:
|
||||||
return self._gtk.Image('wifi_excellent')
|
return self.wifi_signal_icons['fair']
|
||||||
elif signal_level > good:
|
|
||||||
return self._gtk.Image('wifi_good')
|
|
||||||
elif signal_level > fair:
|
|
||||||
return self._gtk.Image('wifi_fair')
|
|
||||||
else:
|
else:
|
||||||
return self._gtk.Image('wifi_weak')
|
return self.wifi_signal_icons['weak']
|
||||||
|
|
||||||
def update_single_network_info(self):
|
def update_single_network_info(self):
|
||||||
ifadd = netifaces.ifaddresses(self.interface)
|
|
||||||
ipv6 = f"{ifadd[netifaces.AF_INET6][0]['addr'].split('%')[0]}" if ifadd.get(netifaces.AF_INET6) else ""
|
|
||||||
if netifaces.AF_INET in ifadd and ifadd[netifaces.AF_INET]:
|
|
||||||
ipv4 = f"{ifadd[netifaces.AF_INET][0]['addr']} "
|
|
||||||
self.labels['ip'].set_text(f"IP: {ifadd[netifaces.AF_INET][0]['addr']} ")
|
|
||||||
else:
|
|
||||||
ipv4 = ""
|
|
||||||
self.labels['networkinfo'].set_markup(
|
self.labels['networkinfo'].set_markup(
|
||||||
f'<b>{self.interface}</b>\n\n'
|
f'<b>{self.interface}</b>\n\n'
|
||||||
+ '<b>' + _("Hostname") + f':</b> {os.uname().nodename}\n'
|
+ '<b>' + _("Hostname") + f':</b> {os.uname().nodename}\n'
|
||||||
f'<b>IPv4:</b> {ipv4}\n'
|
f'<b>IPv4:</b> {self.sdbus_nm.get_ip_address()}\n'
|
||||||
f'<b>IPv6:</b> {ipv6}'
|
|
||||||
)
|
)
|
||||||
self.labels['networkinfo'].show_all()
|
self.labels['networkinfo'].show_all()
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def reload_networks(self, widget=None):
|
def reload_networks(self, widget=None):
|
||||||
self.networks = {}
|
self.deactivate()
|
||||||
self.labels['networklist'].remove_column(0)
|
del self.network_rows
|
||||||
if self.wifi is not None and self.wifi.initialized:
|
self.network_rows = {}
|
||||||
|
for child in self.network_list.get_children():
|
||||||
|
self.network_list.remove(child)
|
||||||
|
if self.sdbus_nm is not None and self.sdbus_nm.wifi:
|
||||||
if widget:
|
if widget:
|
||||||
self._gtk.Button_busy(widget, True)
|
self._gtk.Button_busy(widget, True)
|
||||||
self.wifi.rescan()
|
self.sdbus_nm.rescan()
|
||||||
GLib.idle_add(self.load_networks, widget)
|
self.load_networks()
|
||||||
|
self.activate()
|
||||||
|
|
||||||
def activate(self):
|
def activate(self):
|
||||||
if self.initialized:
|
if self.update_timeout is None:
|
||||||
self.reload_networks()
|
if self.sdbus_nm is not None and self.sdbus_nm.wifi:
|
||||||
if self.update_timeout is None:
|
if self.reload_button.get_sensitive():
|
||||||
if self.wifi is not None and self.wifi.initialized:
|
self._gtk.Button_busy(self.reload_button, True)
|
||||||
self.update_timeout = GLib.timeout_add_seconds(5, self.update_all_networks)
|
self.sdbus_nm.rescan()
|
||||||
else:
|
self.load_networks()
|
||||||
self.update_timeout = GLib.timeout_add_seconds(5, self.update_single_network_info)
|
self.update_timeout = GLib.timeout_add_seconds(5, self.update_all_networks)
|
||||||
|
else:
|
||||||
|
self.update_timeout = GLib.timeout_add_seconds(5, self.update_single_network_info)
|
||||||
|
|
||||||
def deactivate(self):
|
def deactivate(self):
|
||||||
if self.update_timeout is not None:
|
if self.update_timeout is not None:
|
||||||
GLib.source_remove(self.update_timeout)
|
GLib.source_remove(self.update_timeout)
|
||||||
self.update_timeout = None
|
self.update_timeout = None
|
||||||
|
|
||||||
|
def toggle_wifi(self, switch, gparams):
|
||||||
|
enable = switch.get_active()
|
||||||
|
if enable:
|
||||||
|
self.reload_button.show()
|
||||||
|
else:
|
||||||
|
self.reload_button.hide()
|
||||||
|
logging.info(f"WiFi {enable}")
|
||||||
|
self.sdbus_nm.toggle_wifi(enable)
|
||||||
|
@ -1,8 +1,7 @@
|
|||||||
jinja2==3.1.4;python_version>="3.6"
|
jinja2==3.1.4;python_version>="3.6"
|
||||||
requests==2.31.0;python_version>="3.7"
|
requests==2.31.0;python_version>="3.7"
|
||||||
netifaces==0.11.0
|
sdbus==0.11.1;python_version>="3.8"
|
||||||
six==1.16.0
|
sdbus_networkmanager==2.0.0;python_version>="3.8"
|
||||||
dbus-python==1.3.2
|
|
||||||
|
|
||||||
# libmpv-dev 0.33 is required for 1.0
|
# libmpv-dev 0.33 is required for 1.0
|
||||||
python-mpv==0.5.2;python_version<"3.10"
|
python-mpv==0.5.2;python_version<"3.10"
|
||||||
|
Loading…
x
Reference in New Issue
Block a user