2020-04-11 09:02:11 +02:00
|
|
|
# pylint: disable=C0111,R0903
|
|
|
|
|
|
|
|
"""Displays network IO for interfaces.
|
|
|
|
|
|
|
|
Parameters:
|
2020-04-11 09:02:47 +02:00
|
|
|
* traffic.exclude: Comma-separated list of interface prefixes to exclude (defaults to 'lo,virbr,docker,vboxnet,veth')
|
|
|
|
* traffic.states: Comma-separated list of states to show (prefix with '^' to invert - i.e. ^down -> show all devices that are not in state down)
|
2020-04-11 09:02:11 +02:00
|
|
|
* traffic.showname: If set to False, hide network interface name (defaults to True)
|
|
|
|
* traffic.format: Format string for download/upload speeds.
|
2020-04-11 09:02:47 +02:00
|
|
|
Defaults to '{:.2f}'
|
2020-04-11 09:02:11 +02:00
|
|
|
* traffic.graphlen: Graph lenth in seconds. Positive even integer. Each
|
|
|
|
char shows 2 seconds. If set, enables up/down traffic
|
|
|
|
graphs
|
|
|
|
"""
|
|
|
|
|
|
|
|
import re
|
|
|
|
import time
|
|
|
|
import psutil
|
|
|
|
import netifaces
|
|
|
|
|
|
|
|
import bumblebee.util
|
|
|
|
import bumblebee.input
|
|
|
|
import bumblebee.output
|
|
|
|
import bumblebee.engine
|
|
|
|
|
|
|
|
class Module(bumblebee.engine.Module):
|
|
|
|
def __init__(self, engine, config):
|
|
|
|
widgets = []
|
|
|
|
super(Module, self).__init__(engine, config, widgets)
|
2020-04-11 09:02:47 +02:00
|
|
|
self._exclude = tuple(filter(len, self.parameter('exclude', 'lo,virbr,docker,vboxnet,veth').split(',')))
|
|
|
|
self._status = ''
|
2020-04-11 09:02:11 +02:00
|
|
|
|
2020-04-11 09:02:47 +02:00
|
|
|
self._showname = bumblebee.util.asbool(self.parameter('showname', True))
|
|
|
|
self._format = self.parameter('format', '{:.2f}')
|
2020-04-11 09:02:11 +02:00
|
|
|
self._prev = {}
|
|
|
|
self._states = {}
|
|
|
|
self._lastcheck = 0
|
2020-04-11 09:02:47 +02:00
|
|
|
self._states['include'] = []
|
|
|
|
self._states['exclude'] = []
|
|
|
|
for state in tuple(filter(len, self.parameter('states', '').split(','))):
|
|
|
|
if state[0] == '^':
|
|
|
|
self._states['exclude'].append(state[1:])
|
2020-04-11 09:02:11 +02:00
|
|
|
else:
|
2020-04-11 09:02:47 +02:00
|
|
|
self._states['include'].append(state)
|
|
|
|
self._graphlen = int(self.parameter('graphlen', 0))
|
2020-04-11 09:02:11 +02:00
|
|
|
if self._graphlen > 0:
|
|
|
|
self._graphdata = {}
|
|
|
|
self._first_run = True
|
|
|
|
self._update_widgets(widgets)
|
|
|
|
|
|
|
|
def state(self, widget):
|
2020-04-11 09:02:47 +02:00
|
|
|
if 'traffic.rx' in widget.name:
|
|
|
|
return 'rx'
|
|
|
|
if 'traffic.tx' in widget.name:
|
|
|
|
return 'tx'
|
2020-04-11 09:02:11 +02:00
|
|
|
return self._status
|
|
|
|
|
|
|
|
def update(self, widgets):
|
|
|
|
self._update_widgets(widgets)
|
|
|
|
|
|
|
|
def create_widget(self, widgets, name, txt=None, attributes={}):
|
|
|
|
widget = bumblebee.output.Widget(name=name)
|
|
|
|
widget.full_text(txt)
|
|
|
|
widgets.append(widget)
|
|
|
|
|
|
|
|
for key in attributes:
|
|
|
|
widget.set(key, attributes[key])
|
|
|
|
|
|
|
|
return widget
|
|
|
|
|
|
|
|
def get_addresses(self, intf):
|
|
|
|
retval = []
|
|
|
|
try:
|
|
|
|
for ip in netifaces.ifaddresses(intf).get(netifaces.AF_INET, []):
|
2020-04-11 09:02:47 +02:00
|
|
|
if ip.get('addr', '') != '':
|
|
|
|
retval.append(ip.get('addr'))
|
2020-04-11 09:02:11 +02:00
|
|
|
except Exception:
|
|
|
|
return []
|
|
|
|
return retval
|
|
|
|
|
|
|
|
def get_minwidth_str(self):
|
|
|
|
"""
|
|
|
|
computes theme.minwidth string
|
|
|
|
based on traffic.format and traffic.graphlen parameters
|
|
|
|
"""
|
2020-04-11 09:02:47 +02:00
|
|
|
minwidth_str = ''
|
2020-04-11 09:02:11 +02:00
|
|
|
if self._graphlen > 0:
|
|
|
|
graph_len = int(self._graphlen / 2)
|
2020-04-11 09:02:47 +02:00
|
|
|
graph_prefix = '0' * graph_len
|
2020-04-11 09:02:11 +02:00
|
|
|
minwidth_str += graph_prefix
|
2020-04-11 09:02:47 +02:00
|
|
|
minwidth_str += '1000'
|
2020-04-11 09:02:11 +02:00
|
|
|
try:
|
2020-04-11 09:02:47 +02:00
|
|
|
length = int(re.match('{:\.(\d+)f}', self._format).group(1))
|
2020-04-11 09:02:11 +02:00
|
|
|
if length > 0:
|
2020-04-11 09:02:47 +02:00
|
|
|
minwidth_str += '.' + '0' * length
|
2020-04-11 09:02:11 +02:00
|
|
|
except AttributeError:
|
|
|
|
# return default value
|
2020-04-11 09:02:47 +02:00
|
|
|
return '1000.00KiB/s'
|
2020-04-11 09:02:11 +02:00
|
|
|
finally:
|
2020-04-11 09:02:47 +02:00
|
|
|
minwidth_str += 'KiB/s'
|
2020-04-11 09:02:11 +02:00
|
|
|
return minwidth_str
|
|
|
|
|
|
|
|
def _update_widgets(self, widgets):
|
|
|
|
interfaces = [i for i in netifaces.interfaces() if not i.startswith(self._exclude)]
|
|
|
|
|
|
|
|
del widgets[:]
|
|
|
|
|
|
|
|
counters = psutil.net_io_counters(pernic=True)
|
|
|
|
now = time.time()
|
|
|
|
timediff = now - (self._lastcheck if self._lastcheck else now)
|
|
|
|
if timediff <= 0: timediff = 1
|
|
|
|
self._lastcheck = now
|
|
|
|
for interface in interfaces:
|
|
|
|
if self._graphlen > 0:
|
|
|
|
if interface not in self._graphdata:
|
|
|
|
self._graphdata[interface] = {
|
2020-04-11 09:02:47 +02:00
|
|
|
'rx': [0] * self._graphlen,
|
|
|
|
'tx': [0] * self._graphlen}
|
|
|
|
if not interface: interface = 'lo'
|
|
|
|
state = 'down'
|
2020-04-11 09:02:11 +02:00
|
|
|
if len(self.get_addresses(interface)) > 0:
|
2020-04-11 09:02:47 +02:00
|
|
|
state = 'up'
|
|
|
|
elif bumblebee.util.asbool(self.parameter('hide_down', True)):
|
2020-04-11 09:02:11 +02:00
|
|
|
continue
|
|
|
|
|
2020-04-11 09:02:47 +02: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-04-11 09:02:11 +02:00
|
|
|
|
|
|
|
data = {
|
2020-04-11 09:02:47 +02:00
|
|
|
'rx': counters[interface].bytes_recv,
|
|
|
|
'tx': counters[interface].bytes_sent,
|
2020-04-11 09:02:11 +02:00
|
|
|
}
|
|
|
|
|
2020-04-11 09:02:47 +02:00
|
|
|
name = 'traffic-{}'.format(interface)
|
2020-04-11 09:02:11 +02:00
|
|
|
|
|
|
|
if self._showname:
|
|
|
|
self.create_widget(widgets, name, interface)
|
|
|
|
|
2020-04-11 09:02:47 +02:00
|
|
|
for direction in ['rx', 'tx']:
|
|
|
|
name = 'traffic.{}-{}'.format(direction, interface)
|
|
|
|
widget = self.create_widget(widgets, name, attributes={'theme.minwidth': self.get_minwidth_str()})
|
2020-04-11 09:02:11 +02:00
|
|
|
prev = self._prev.get(name, 0)
|
|
|
|
bspeed = (int(data[direction]) - int(prev))/timediff
|
|
|
|
speed = bumblebee.util.bytefmt(bspeed, self._format)
|
|
|
|
txtspeed = '{0}/s'.format(speed)
|
|
|
|
if self._graphlen > 0:
|
|
|
|
# skip first value returned by psutil, because it is
|
|
|
|
# giant and ruins the grapth ratio until it gets pushed
|
|
|
|
# out of saved list
|
|
|
|
if self._first_run is True:
|
|
|
|
self._first_run = False
|
|
|
|
else:
|
|
|
|
self._graphdata[interface][direction] = self._graphdata[interface][direction][1:]
|
|
|
|
self._graphdata[interface][direction].append(bspeed)
|
2020-04-11 09:02:47 +02:00
|
|
|
txtspeed = '{}{}'.format(bumblebee.output.bgraph(self._graphdata[interface][direction]), txtspeed)
|
2020-04-11 09:02:11 +02:00
|
|
|
widget.full_text(txtspeed)
|
|
|
|
self._prev[name] = data[direction]
|
|
|
|
|
|
|
|
# vim: tabstop=8 expandtab shiftwidth=4 softtabstop=4
|