2019-10-17 14:10:08 -04:00
|
|
|
import time
|
2019-06-02 23:50:17 -04:00
|
|
|
import asyncio
|
|
|
|
import typing
|
|
|
|
import collections
|
|
|
|
import logging
|
|
|
|
|
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
CONNECTED_EVENT = "connected"
|
|
|
|
DISCONNECTED_EVENT = "disconnected"
|
|
|
|
TRANSFERRED_EVENT = "transferred"
|
|
|
|
|
|
|
|
|
|
|
|
class ConnectionManager:
|
|
|
|
def __init__(self, loop: asyncio.AbstractEventLoop):
|
|
|
|
self.loop = loop
|
|
|
|
self.incoming_connected: typing.Set[str] = set()
|
|
|
|
self.incoming: typing.DefaultDict[str, int] = collections.defaultdict(int)
|
|
|
|
self.outgoing_connected: typing.Set[str] = set()
|
|
|
|
self.outgoing: typing.DefaultDict[str, int] = collections.defaultdict(int)
|
2019-10-17 14:10:08 -04:00
|
|
|
self._max_incoming_mbs = 0.0
|
|
|
|
self._max_outgoing_mbs = 0.0
|
2019-06-02 23:50:17 -04:00
|
|
|
self._status = {}
|
2019-06-28 09:59:31 -04:00
|
|
|
self._running = False
|
2019-06-02 23:50:17 -04:00
|
|
|
self._task: typing.Optional[asyncio.Task] = None
|
|
|
|
|
|
|
|
@property
|
|
|
|
def status(self):
|
|
|
|
return self._status
|
|
|
|
|
|
|
|
def sent_data(self, host_and_port: str, size: int):
|
2019-06-28 09:59:31 -04:00
|
|
|
if self._running:
|
|
|
|
self.outgoing[host_and_port] += size
|
2019-06-02 23:50:17 -04:00
|
|
|
|
|
|
|
def received_data(self, host_and_port: str, size: int):
|
2019-06-28 09:59:31 -04:00
|
|
|
if self._running:
|
|
|
|
self.incoming[host_and_port] += size
|
2019-06-02 23:50:17 -04:00
|
|
|
|
|
|
|
def connection_made(self, host_and_port: str):
|
2019-06-28 09:59:31 -04:00
|
|
|
if self._running:
|
|
|
|
self.outgoing_connected.add(host_and_port)
|
2019-06-02 23:50:17 -04:00
|
|
|
|
|
|
|
def connection_received(self, host_and_port: str):
|
2020-04-22 13:38:38 -04:00
|
|
|
self.incoming_connected.add(host_and_port)
|
2019-06-02 23:50:17 -04:00
|
|
|
|
|
|
|
def outgoing_connection_lost(self, host_and_port: str):
|
2019-06-28 09:59:31 -04:00
|
|
|
if self._running and host_and_port in self.outgoing_connected:
|
2019-06-02 23:50:17 -04:00
|
|
|
self.outgoing_connected.remove(host_and_port)
|
|
|
|
|
|
|
|
def incoming_connection_lost(self, host_and_port: str):
|
2019-06-28 09:59:31 -04:00
|
|
|
if self._running and host_and_port in self.incoming_connected:
|
2019-06-02 23:50:17 -04:00
|
|
|
self.incoming_connected.remove(host_and_port)
|
|
|
|
|
|
|
|
async def _update(self):
|
|
|
|
self._status = {
|
|
|
|
'incoming_bps': {},
|
|
|
|
'outgoing_bps': {},
|
|
|
|
'total_incoming_mbs': 0.0,
|
|
|
|
'total_outgoing_mbs': 0.0,
|
2019-10-04 15:46:36 +02:00
|
|
|
'total_sent': 0,
|
|
|
|
'total_received': 0,
|
2019-10-17 14:10:08 -04:00
|
|
|
'max_incoming_mbs': 0.0,
|
2020-04-22 13:38:38 -04:00
|
|
|
'max_outgoing_mbs': 0.0,
|
|
|
|
'outbound_connections': 0,
|
|
|
|
'incoming_connections': 0
|
2019-06-02 23:50:17 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
while True:
|
2019-10-17 14:10:08 -04:00
|
|
|
last = time.perf_counter()
|
|
|
|
await asyncio.sleep(0.1, loop=self.loop)
|
2019-06-02 23:50:17 -04:00
|
|
|
self._status['incoming_bps'].clear()
|
|
|
|
self._status['outgoing_bps'].clear()
|
2019-10-17 14:10:08 -04:00
|
|
|
now = time.perf_counter()
|
2019-06-02 23:50:17 -04:00
|
|
|
while self.outgoing:
|
2020-01-02 23:20:51 -05:00
|
|
|
k, sent = self.outgoing.popitem()
|
|
|
|
self._status['total_sent'] += sent
|
|
|
|
self._status['outgoing_bps'][k] = sent / (now - last)
|
2019-06-02 23:50:17 -04:00
|
|
|
while self.incoming:
|
2020-01-02 23:20:51 -05:00
|
|
|
k, received = self.incoming.popitem()
|
|
|
|
self._status['total_received'] += received
|
|
|
|
self._status['incoming_bps'][k] = received / (now - last)
|
2019-06-02 23:50:17 -04:00
|
|
|
self._status['total_outgoing_mbs'] = int(sum(list(self._status['outgoing_bps'].values())
|
2019-10-04 22:37:49 +02:00
|
|
|
)) / 1000000.0
|
2019-06-02 23:50:17 -04:00
|
|
|
self._status['total_incoming_mbs'] = int(sum(list(self._status['incoming_bps'].values())
|
2019-10-04 22:37:49 +02:00
|
|
|
)) / 1000000.0
|
2019-10-17 14:10:08 -04:00
|
|
|
self._max_incoming_mbs = max(self._max_incoming_mbs, self._status['total_incoming_mbs'])
|
|
|
|
self._max_outgoing_mbs = max(self._max_outgoing_mbs, self._status['total_outgoing_mbs'])
|
|
|
|
self._status['max_incoming_mbs'] = self._max_incoming_mbs
|
|
|
|
self._status['max_outgoing_mbs'] = self._max_outgoing_mbs
|
2020-04-22 13:38:38 -04:00
|
|
|
self._status['outbound_connections'] = len(self.outgoing_connected)
|
|
|
|
self._status['incoming_connections'] = len(self.incoming_connected)
|
2019-06-02 23:50:17 -04:00
|
|
|
|
|
|
|
def stop(self):
|
|
|
|
if self._task:
|
|
|
|
self._task.cancel()
|
|
|
|
self._task = None
|
|
|
|
self.outgoing.clear()
|
|
|
|
self.outgoing_connected.clear()
|
|
|
|
self.incoming.clear()
|
|
|
|
self.incoming_connected.clear()
|
|
|
|
self._status.clear()
|
2019-06-28 09:59:31 -04:00
|
|
|
self._running = False
|
2019-06-02 23:50:17 -04:00
|
|
|
|
|
|
|
def start(self):
|
|
|
|
self.stop()
|
2019-06-28 09:59:31 -04:00
|
|
|
self._running = True
|
2019-06-02 23:50:17 -04:00
|
|
|
self._task = self.loop.create_task(self._update())
|