bumblebee-status/bumblebee_status/modules/core/nic.py

150 lines
4.7 KiB
Python
Raw Normal View History

# pylint: disable=C0111,R0903
2020-03-01 14:36:12 +01:00
"""Displays the name, IP address(es) and status of each available network interface.
2020-03-01 14:36:12 +01:00
Requires the following python module:
* netifaces
2020-07-18 08:23:28 +02:00
Requires the following executable:
* iw
* (until and including 2.0.5: iwgetid)
2020-07-18 08:23:28 +02:00
2020-03-01 14:36:12 +01:00
Parameters:
* nic.exclude: Comma-separated list of interface prefixes (supporting regular expressions) to exclude (defaults to 'lo,virbr,docker,vboxnet,veth,br,.*:avahi')
2020-03-01 14:36:12 +01:00
* nic.include: Comma-separated list of interfaces to include
* nic.states: Comma-separated list of states to show (prefix with '^' to invert - i.e. ^down -> show all devices that are not in state down)
* nic.format: Format string (defaults to '{intf} {state} {ip} {ssid}')
"""
2020-03-01 14:36:12 +01:00
import re
2020-03-01 14:36:12 +01:00
import shutil
import netifaces
import subprocess
import core.module
import core.decorators
2020-03-01 14:36:12 +01:00
import util.cli
import util.format
2020-03-01 14:36:12 +01:00
class Module(core.module.Module):
@core.decorators.every(seconds=10)
def __init__(self, config, theme):
2020-03-01 14:36:12 +01:00
widgets = []
super().__init__(config, theme, widgets)
self._exclude = util.format.aslist(
self.parameter("exclude", "lo,virbr,docker,vboxnet,veth,br,.*:avahi")
)
self._include = util.format.aslist(self.parameter("include", ""))
self._states = {"include": [], "exclude": []}
for state in tuple(
filter(len, util.format.aslist(self.parameter("states", "")))
):
if state[0] == "^":
self._states["exclude"].append(state[1:])
2020-03-01 14:36:12 +01:00
else:
self._states["include"].append(state)
self._format = self.parameter("format", "{intf} {state} {ip} {ssid}")
self.iw = shutil.which("iw")
2020-03-01 14:36:12 +01:00
self._update_widgets(widgets)
def update(self):
self._update_widgets(self.widgets())
def state(self, widget):
states = []
if widget.get("state") == "down":
states.append("critical")
elif widget.get("state") != "up":
states.append("warning")
2020-03-01 14:36:12 +01:00
intf = widget.get("intf")
iftype = "wireless" if self._iswlan(intf) else "wired"
iftype = "tunnel" if self._istunnel(intf) else iftype
2020-03-01 14:36:12 +01:00
states.append("{}-{}".format(iftype, widget.get("state")))
2020-03-01 14:36:12 +01:00
return states
def _iswlan(self, intf):
# wifi, wlan, wlp, seems to work for me
if intf.startswith("w"):
return True
2020-03-01 14:36:12 +01:00
return False
def _istunnel(self, intf):
return intf.startswith("tun") or intf.startswith("wg")
2020-03-01 14:36:12 +01:00
def get_addresses(self, intf):
retval = []
try:
for ip in netifaces.ifaddresses(intf).get(netifaces.AF_INET, []):
if ip.get("addr", "") != "":
retval.append(ip.get("addr"))
2020-03-01 14:36:12 +01:00
except Exception:
return []
return retval
def _excluded(self, intf):
for e in self._exclude:
if re.match(e, intf):
return True
return False
2020-03-01 14:36:12 +01:00
def _update_widgets(self, widgets):
self.clear_widgets()
interfaces = []
for i in netifaces.interfaces():
if not self._excluded(i):
interfaces.append(i)
2020-03-01 14:36:12 +01:00
interfaces.extend([i for i in netifaces.interfaces() if i in self._include])
for intf in interfaces:
addr = []
state = "down"
2020-03-01 14:36:12 +01:00
for ip in self.get_addresses(intf):
addr.append(ip)
state = "up"
2020-03-01 14:36:12 +01:00
if len(self._states["exclude"]) > 0 and state in self._states["exclude"]:
continue
if (
len(self._states["include"]) > 0
and state not in self._states["include"]
):
continue
2020-03-01 14:36:12 +01:00
widget = self.widget(intf)
if not widget:
widget = self.add_widget(name=intf)
2020-03-01 14:36:12 +01:00
# join/split is used to get rid of multiple whitespaces (in case SSID is not available, for instance
widget.full_text(
" ".join(
self._format.format(
ip=", ".join(addr),
intf=intf,
state=state,
ssid=self.get_ssid(intf),
).split()
)
)
widget.set("intf", intf)
widget.set("state", state)
2020-03-01 14:36:12 +01:00
def get_ssid(self, intf):
if not self._iswlan(intf) or self._istunnel(intf) or not self.iw:
return ""
iw_info = util.cli.execute("{} dev {} info".format(self.iw, intf))
for line in iw_info.split("\n"):
match = re.match("^\s+ssid\s(.+)$", line)
if match:
return match.group(1)
return ""
2020-03-01 14:36:12 +01:00
# vim: tabstop=8 expandtab shiftwidth=4 softtabstop=4