screen: move the wifi-manager to the network panel

moving the manager should increase startup times
only load it when the user wants to manage networks

network: add ip at the top (fixes #422)
network: always reload the list of networks
wifi: fix network list getting blank
network: make the list more resilient to netinfo errors
This commit is contained in:
alfrix 2022-03-19 22:02:21 -03:00
parent 0d259d6d04
commit 82c8d62876
3 changed files with 104 additions and 68 deletions

View File

@ -12,9 +12,6 @@ import gi
gi.require_version("Gtk", "3.0") gi.require_version("Gtk", "3.0")
from gi.repository import GLib, Gdk from gi.repository import GLib, Gdk
RESCAN_INTERVAL = 180
KS_SOCKET_FILE = "/tmp/.KS_wpa_supplicant"
class WifiManager(): class WifiManager():
networks_in_supplicant = [] networks_in_supplicant = []
connected = False connected = False
@ -45,6 +42,7 @@ class WifiManager():
self.timeout = None self.timeout = None
self.scan_time = 0 self.scan_time = 0
KS_SOCKET_FILE = "/tmp/.KS_wpa_supplicant"
if os.path.exists(KS_SOCKET_FILE): if os.path.exists(KS_SOCKET_FILE):
os.remove(KS_SOCKET_FILE) os.remove(KS_SOCKET_FILE)
@ -63,7 +61,7 @@ class WifiManager():
self.wpa_cli("ATTACH", False) self.wpa_cli("ATTACH", False)
self.wpa_cli("SCAN", False) self.wpa_cli("SCAN", False)
GLib.idle_add(self.read_wpa_supplicant) GLib.idle_add(self.read_wpa_supplicant)
self.timeout = GLib.timeout_add_seconds(RESCAN_INTERVAL, self.rescan) self.timeout = GLib.timeout_add_seconds(180, self.rescan)
def add_callback(self, name, callback): def add_callback(self, name, callback):
if name in self._callbacks and callback not in self._callbacks[name]: if name in self._callbacks and callback not in self._callbacks[name]:
@ -145,14 +143,31 @@ class WifiManager():
return self.connected_ssid return self.connected_ssid
def get_current_wifi(self, interface="wlan0"): def get_current_wifi(self, interface="wlan0"):
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') status = self.wpa_cli("STATUS").split('\n')
vars = {} vars = {}
for line in status: for line in status:
arr = line.split('=') arr = line.split('=')
vars[arr[0]] = "=".join(arr[1:]) vars[arr[0]] = "=".join(arr[1:])
prev_ssid = self.connected_ssid prev_ssid = self.connected_ssid
if "ssid" in vars and "bssid" in vars:
if con_ssid != "":
self.connected = True
self.connected_ssid = con_ssid
for ssid, val in self.networks.items():
if ssid == con_ssid:
self.networks[ssid]['connected'] = True
else:
self.networks[ssid]['connected'] = False
if prev_ssid != self.connected_ssid:
for cb in self._callbacks['connected']:
Gdk.threads_add_idle(
GLib.PRIORITY_DEFAULT_IDLE,
cb, self.connected_ssid, prev_ssid)
return [con_ssid, con_bssid]
elif "ssid" in vars and "bssid" in vars:
self.connected = True self.connected = True
self.connected_ssid = vars['ssid'] self.connected_ssid = vars['ssid']
for ssid, val in self.networks.items(): for ssid, val in self.networks.items():

View File

@ -2,6 +2,8 @@ import gi
import logging import logging
import netifaces import netifaces
import os import os
import netifaces
from ks_includes.wifi import WifiManager
gi.require_version("Gtk", "3.0") gi.require_version("Gtk", "3.0")
from gi.repository import Gtk, Gdk, GLib, Pango from gi.repository import Gtk, Gdk, GLib, Pango
@ -11,19 +13,22 @@ def create_panel(*args):
return NetworkPanel(*args) return NetworkPanel(*args)
class NetworkPanel(ScreenPanel): class NetworkPanel(ScreenPanel):
networks = {}
network_list = []
def initialize(self, menu): def initialize(self, menu):
_ = self.lang.gettext _ = self.lang.gettext
self.show_add = False self.show_add = False
self.networks = {}
self.update_timeout = None
self.network_interfaces = netifaces.interfaces()
self.wireless_interfaces = [int for int in self.network_interfaces if int.startswith('w')]
self.wifi = None
if len(self.wireless_interfaces) > 0:
logging.info("Found wireless interfaces: %s" % self.wireless_interfaces)
self.wifi = WifiManager(self.wireless_interfaces[0])
grid = self._gtk.HomogeneousGrid() grid = self._gtk.HomogeneousGrid()
grid.set_hexpand(True) grid.set_hexpand(True)
# Get Hostname
stream = os.popen('hostname -A')
hostname = stream.read()
# Get IP Address # Get IP Address
gws = netifaces.gateways() gws = netifaces.gateways()
if "default" in gws and netifaces.AF_INET in gws["default"]: if "default" in gws and netifaces.AF_INET in gws["default"]:
@ -41,21 +46,21 @@ class NetworkPanel(ScreenPanel):
if netifaces.AF_INET in res and len(res[netifaces.AF_INET]) > 0: if netifaces.AF_INET in res and len(res[netifaces.AF_INET]) > 0:
ip = res[netifaces.AF_INET][0]['addr'] ip = res[netifaces.AF_INET][0]['addr']
else: else:
ip = "0.0.0.0" ip = None
self.labels['networks'] = {} self.labels['networks'] = {}
self.labels['interface'] = Gtk.Label() self.labels['interface'] = Gtk.Label()
self.labels['interface'].set_text(" %s: %s" % (_("Interface"), self.interface)) self.labels['interface'].set_text(" %s: %s " % (_("Interface"), self.interface))
self.labels['disconnect'] = self._gtk.Button(_("Disconnect"), "color2") self.labels['ip'] = Gtk.Label()
sbox = Gtk.Box() sbox = Gtk.Box()
sbox.set_hexpand(True) sbox.set_hexpand(True)
sbox.set_vexpand(False) sbox.set_vexpand(False)
sbox.add(self.labels['interface']) sbox.add(self.labels['interface'])
# sbox.add(self.labels['disconnect']) if ip is not None:
self.labels['ip'].set_text("IP: %s " % ip)
sbox.add(self.labels['ip'])
scroll = Gtk.ScrolledWindow() scroll = Gtk.ScrolledWindow()
scroll.set_property("overlay-scrolling", False) scroll.set_property("overlay-scrolling", False)
@ -67,32 +72,33 @@ class NetworkPanel(ScreenPanel):
box.set_vexpand(True) box.set_vexpand(True)
self.labels['networklist'] = Gtk.Grid() self.labels['networklist'] = Gtk.Grid()
self.files = {}
if self._screen.wifi is not None and self._screen.wifi.is_initialized(): if self.wifi is not None and self.wifi.is_initialized():
box.pack_start(sbox, False, False, 0) box.pack_start(sbox, False, False, 5)
box.pack_start(scroll, True, True, 0) 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.labels['networklist'])
self._screen.wifi.add_callback("connected", self.connected_callback) self.wifi.add_callback("connected", self.connected_callback)
self._screen.wifi.add_callback("scan_results", self.scan_callback) self.wifi.add_callback("scan_results", self.scan_callback)
self.timeout = GLib.timeout_add_seconds(5, self.update_all_networks) if self.update_timeout is None:
self.update_timeout = GLib.timeout_add_seconds(5, self.update_all_networks)
else: else:
self.labels['networkinfo'] = Gtk.Label("") self.labels['networkinfo'] = Gtk.Label("")
self.labels['networkinfo'].get_style_context().add_class('temperature_entry') self.labels['networkinfo'].get_style_context().add_class('temperature_entry')
box.pack_start(self.labels['networkinfo'], False, False, 0) box.pack_start(self.labels['networkinfo'], False, False, 0)
self.update_single_network_info() self.update_single_network_info()
self.timeout = GLib.timeout_add_seconds(5, 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.content.add(box)
self.labels['main_box'] = box self.labels['main_box'] = box
def load_networks(self): def load_networks(self):
networks = self._screen.wifi.get_networks() networks = self.wifi.get_networks()
conn_ssid = self._screen.wifi.get_connected_ssid() conn_ssid = self.wifi.get_connected_ssid()
if conn_ssid in networks: if conn_ssid in networks:
networks.remove(conn_ssid) networks.remove(conn_ssid)
self.add_network(conn_ssid, False) self.add_network(conn_ssid, False)
@ -109,17 +115,16 @@ class NetworkPanel(ScreenPanel):
if ssid is None: if ssid is None:
return return
ssid = ssid.strip() ssid = ssid.strip()
if ssid in list(self.networks): if ssid in list(self.networks):
logging.info("SSID already listed") logging.info("SSID already listed")
return return
netinfo = self._screen.wifi.get_network_info(ssid) netinfo = self.wifi.get_network_info(ssid)
if netinfo is None: if netinfo is None:
logging.debug("Couldn't get netinfo") logging.debug("Couldn't get netinfo")
return netinfo = {'connected': False}
configured_networks = self._screen.wifi.get_supplicant_networks() configured_networks = self.wifi.get_supplicant_networks()
network_id = -1 network_id = -1
for net in list(configured_networks): for net in list(configured_networks):
if configured_networks[net]['ssid'] == ssid: if configured_networks[net]['ssid'] == ssid:
@ -139,7 +144,6 @@ class NetworkPanel(ScreenPanel):
info = Gtk.Label() info = Gtk.Label()
info.set_halign(Gtk.Align.START) info.set_halign(Gtk.Align.START)
# info.set_markup(self.get_file_info_str(ssid))
labels = Gtk.Box(orientation=Gtk.Orientation.VERTICAL) labels = Gtk.Box(orientation=Gtk.Orientation.VERTICAL)
labels.add(name) labels.add(name)
labels.add(info) labels.add(info)
@ -167,7 +171,7 @@ class NetworkPanel(ScreenPanel):
buttons = Gtk.Box(orientation=Gtk.Orientation.HORIZONTAL, spacing=5) buttons = Gtk.Box(orientation=Gtk.Orientation.HORIZONTAL, spacing=5)
if network_id != -1 or netinfo['connected']: if network_id != -1 or netinfo['connected']:
buttons.pack_end(delete, False, False, 0) buttons.pack_end(delete, False, False, 0)
connected_ssid = self._screen.wifi.get_connected_ssid() connected_ssid = self.wifi.get_connected_ssid()
if ssid != connected_ssid or not netinfo['connected']: if ssid != connected_ssid or not netinfo['connected']:
buttons.pack_end(connect, False, False, 0) buttons.pack_end(connect, False, False, 0)
@ -179,6 +183,7 @@ class NetworkPanel(ScreenPanel):
nets = sorted(list(self.networks), reverse=False) nets = sorted(list(self.networks), reverse=False)
if connected_ssid == ssid: if connected_ssid == ssid:
pos = 0 pos = 0
name.set_markup("<big><b>%s</b></big> (%s)" % (ssid, _("Connected")))
elif nets.index(ssid) is not None: elif nets.index(ssid) is not None:
pos = nets.index(ssid) + 1 pos = nets.index(ssid) + 1
else: else:
@ -200,9 +205,8 @@ class NetworkPanel(ScreenPanel):
def add_new_network(self, widget, ssid, connect=False): def add_new_network(self, widget, ssid, connect=False):
self._screen.remove_keyboard() self._screen.remove_keyboard()
networks = self._screen.wifi.get_networks()
psk = self.labels['network_psk'].get_text() psk = self.labels['network_psk'].get_text()
result = self._screen.wifi.add_network(ssid, psk) result = self.wifi.add_network(ssid, psk)
self.close_add_network() self.close_add_network()
@ -219,7 +223,7 @@ class NetworkPanel(ScreenPanel):
return False return False
def check_missing_networks(self): def check_missing_networks(self):
networks = self._screen.wifi.get_networks() networks = self.wifi.get_networks()
for net in list(self.networks): for net in list(self.networks):
if net in networks: if net in networks:
networks.remove(net) networks.remove(net)
@ -256,7 +260,7 @@ class NetworkPanel(ScreenPanel):
def connect_network(self, widget, ssid, showadd=True): def connect_network(self, widget, ssid, showadd=True):
_ = self.lang.gettext _ = self.lang.gettext
snets = self._screen.wifi.get_supplicant_networks() snets = self.wifi.get_supplicant_networks()
isdef = False isdef = False
for id, net in snets.items(): for id, net in snets.items():
if net['ssid'] == ssid: if net['ssid'] == ssid:
@ -267,7 +271,7 @@ class NetworkPanel(ScreenPanel):
if showadd: if showadd:
self.show_add_network(widget, ssid) self.show_add_network(widget, ssid)
return return
self.prev_network = self._screen.wifi.get_connected_ssid() self.prev_network = self.wifi.get_connected_ssid()
buttons = [ buttons = [
{"name": _("Close"), "response": Gtk.ResponseType.CANCEL} {"name": _("Close"), "response": Gtk.ResponseType.CANCEL}
@ -283,17 +287,16 @@ class NetworkPanel(ScreenPanel):
self.labels['connecting_info'].set_halign(Gtk.Align.START) self.labels['connecting_info'].set_halign(Gtk.Align.START)
self.labels['connecting_info'].set_valign(Gtk.Align.START) self.labels['connecting_info'].set_valign(Gtk.Align.START)
scroll.add(self.labels['connecting_info']) scroll.add(self.labels['connecting_info'])
dialog = self._gtk.Dialog(self._screen, buttons, scroll, self.close_dialog) self._gtk.Dialog(self._screen, buttons, scroll, self.close_dialog)
self._screen.show_all() self._screen.show_all()
if ssid in self.networks: if ssid in self.networks:
self.remove_network(ssid) self.remove_network(ssid)
if self.prev_network in self.networks: if self.prev_network in self.networks:
self.remove_network(self.prev_network) self.remove_network(self.prev_network)
# GLib.timeout_add(500, self.add_network, self.prev_network)
self._screen.wifi.add_callback("connecting_status", self.connecting_status_callback) self.wifi.add_callback("connecting_status", self.connecting_status_callback)
self._screen.wifi.connect(ssid) self.wifi.connect(ssid)
def connecting_status_callback(self, msg): def connecting_status_callback(self, msg):
self.labels['connecting_info'].set_text(self.labels['connecting_info'].get_text() + "\n" + msg) self.labels['connecting_info'].set_text(self.labels['connecting_info'].get_text() + "\n" + msg)
@ -303,7 +306,7 @@ class NetworkPanel(ScreenPanel):
if ssid not in self.networks: if ssid not in self.networks:
return return
for i, network in enumerate(self.labels['networklist']): for i in range(len(self.labels['networklist'])):
if self.networks[ssid] == self.labels['networklist'].get_child_at(0, i): if self.networks[ssid] == self.labels['networklist'].get_child_at(0, i):
self.labels['networklist'].remove_row(i) self.labels['networklist'].remove_row(i)
self.labels['networklist'].show() self.labels['networklist'].show()
@ -312,7 +315,7 @@ class NetworkPanel(ScreenPanel):
return return
def remove_wifi_network(self, widget, ssid): def remove_wifi_network(self, widget, ssid):
self._screen.wifi.delete_network(ssid) self.wifi.delete_network(ssid)
self.remove_network(ssid) self.remove_network(ssid)
self.check_missing_networks() self.check_missing_networks()
@ -374,33 +377,40 @@ class NetworkPanel(ScreenPanel):
def update_network_info(self, ssid): def update_network_info(self, ssid):
_ = self.lang.gettext _ = self.lang.gettext
info = freq = encr = chan = lvl = ipv4 = ipv6 = ""
if ssid not in self.networks or ssid not in self.labels['networks']: if ssid not in self.networks or ssid not in self.labels['networks']:
logging.info("Unknown SSID %s", ssid)
return return
netinfo = self._screen.wifi.get_network_info(ssid) netinfo = self.wifi.get_network_info(ssid)
if netinfo is None: if netinfo is None:
return netinfo = []
if "connected" in netinfo:
connected = netinfo['connected']
connected = "" if connected or self.wifi.get_connected_ssid() == ssid:
if netinfo['connected']:
stream = os.popen('hostname -f') stream = os.popen('hostname -f')
hostname = stream.read().strip() hostname = stream.read().strip()
ifadd = netifaces.ifaddresses(self.interface) ifadd = netifaces.ifaddresses(self.interface)
ipv4 = ""
ipv6 = ""
if netifaces.AF_INET in ifadd and len(ifadd[netifaces.AF_INET]) > 0: if netifaces.AF_INET in ifadd and len(ifadd[netifaces.AF_INET]) > 0:
ipv4 = "<b>%s:</b> %s " % (_("IPv4"), ifadd[netifaces.AF_INET][0]['addr']) ipv4 = "<b>%s:</b> %s " % ("IPv4", ifadd[netifaces.AF_INET][0]['addr'])
if netifaces.AF_INET6 in ifadd and len(ifadd[netifaces.AF_INET6]) > 0: if netifaces.AF_INET6 in ifadd and len(ifadd[netifaces.AF_INET6]) > 0:
ipv6 = ipv6 = "<b>%s:</b> %s " % (_("IPv6"), ifadd[netifaces.AF_INET6][0]['addr'].split('%')[0]) ipv6 = "<b>%s:</b> %s " % ("IPv6", ifadd[netifaces.AF_INET6][0]['addr'].split('%')[0])
connected = "<b>%s</b>\n<b>%s:</b> %s\n%s%s\n" % (_("Connected"), _("Hostname"), hostname, ipv4, ipv6) info = "<b>%s:</b> %s\n%s\n%s\n" % (_("Hostname"), hostname, ipv4, ipv6)
elif "psk" in netinfo: elif "psk" in netinfo:
connected = "Password saved." info = _("Password saved")
freq = "2.4 GHz" if netinfo['frequency'][0:1] == "2" else "5 Ghz" if "encryption" in netinfo:
if netinfo['encryption'] != "off":
encr = netinfo['encryption'].upper()
if "frequency" in netinfo:
freq = "2.4 GHz" if netinfo['frequency'][0:1] == "2" else "5 Ghz"
if "channel" in netinfo:
chan = _("Channel") + " " + netinfo['channel']
if "signal_level_dBm" in netinfo:
lvl = netinfo['signal_level_dBm'] + " " + _("dBm")
self.labels['networks'][ssid]['info'].set_markup("%s%s <small>%s %s %s %s%s</small>" % ( self.labels['networks'][ssid]['info'].set_markup("%s <small>%s %s %s %s</small>" % (
connected, "" if netinfo['encryption'] == "off" else netinfo['encryption'].upper(), info, encr, freq, chan, lvl))
freq, _("Channel"), netinfo['channel'], netinfo['signal_level_dBm'], _("dBm")
))
self.labels['networks'][ssid]['info'].show_all() self.labels['networks'][ssid]['info'].show_all()
def update_single_network_info(self): def update_single_network_info(self):
@ -420,3 +430,23 @@ class NetworkPanel(ScreenPanel):
self.labels['networkinfo'].set_markup(connected) self.labels['networkinfo'].set_markup(connected)
self.labels['networkinfo'].show_all() self.labels['networkinfo'].show_all()
def activate(self):
self.networks = {}
for i in range(len(self.labels['networklist'])):
self.labels['networklist'].remove_row(i)
if self.wifi is not None and self.wifi.is_initialized():
GLib.idle_add(self.load_networks)
if self.update_timeout is None:
if self.wifi is not None and self.wifi.is_initialized():
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)
return
def deactivate(self):
if self.update_timeout is not None:
GLib.source_remove(self.update_timeout)
self.update_timeout = None
self.networks = None

View File

@ -4,7 +4,6 @@ import argparse
import gi import gi
import json import json
import netifaces
import importlib import importlib
import logging import logging
import os import os
@ -24,7 +23,6 @@ from ks_includes.KlippyRest import KlippyRest
from ks_includes.files import KlippyFiles from ks_includes.files import KlippyFiles
from ks_includes.KlippyGtk import KlippyGtk from ks_includes.KlippyGtk import KlippyGtk
from ks_includes.printer import Printer from ks_includes.printer import Printer
from ks_includes.wifi import WifiManager
from ks_includes.config import KlipperScreenConfig from ks_includes.config import KlipperScreenConfig
from panels.base_panel import BasePanel from panels.base_panel import BasePanel
@ -88,13 +86,6 @@ class KlipperScreen(Gtk.Window):
self._config = KlipperScreenConfig(configfile, self) self._config = KlipperScreenConfig(configfile, self)
self.lang = self._config.get_lang() self.lang = self._config.get_lang()
self.network_interfaces = netifaces.interfaces()
self.wireless_interfaces = [int for int in self.network_interfaces if int.startswith('w')]
self.wifi = None
if len(self.wireless_interfaces) > 0:
logging.info("Found wireless interfaces: %s" % self.wireless_interfaces)
self.wifi = WifiManager(self.wireless_interfaces[0])
logging.debug("OS Language: %s" % os.getenv('LANG')) logging.debug("OS Language: %s" % os.getenv('LANG'))
settings = Gtk.Settings.get_default() settings = Gtk.Settings.get_default()