2022-04-11 18:08:37 +02:00
|
|
|
"""Test network module"""
|
|
|
|
|
|
|
|
import threading
|
|
|
|
import time
|
|
|
|
|
|
|
|
from .common import skip_python3
|
|
|
|
from .partial import TestPartialRun
|
|
|
|
|
|
|
|
skip_python3()
|
|
|
|
|
|
|
|
|
|
|
|
class TestNetwork(TestPartialRun):
|
|
|
|
"""A test case for running the network subsystem"""
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def setUpClass(cls):
|
|
|
|
super(TestNetwork, cls).setUpClass()
|
|
|
|
|
|
|
|
cls.state.maximumNumberOfHalfOpenConnections = 4
|
|
|
|
|
|
|
|
cls.config.set('bitmessagesettings', 'sendoutgoingconnections', 'True')
|
2023-12-20 02:02:42 +01:00
|
|
|
cls.config.set('bitmessagesettings', 'udp', 'True')
|
2022-04-11 18:08:37 +02:00
|
|
|
|
|
|
|
# config variable is still used inside of the network ):
|
|
|
|
import network
|
|
|
|
from network import connectionpool, stats
|
|
|
|
|
|
|
|
# beware of singleton
|
|
|
|
connectionpool.config = cls.config
|
2024-05-09 05:14:56 +02:00
|
|
|
cls.pool = connectionpool.pool
|
2022-04-11 18:08:37 +02:00
|
|
|
cls.stats = stats
|
|
|
|
|
|
|
|
network.start(cls.config, cls.state)
|
|
|
|
|
|
|
|
def test_threads(self):
|
|
|
|
"""Ensure all the network threads started"""
|
|
|
|
threads = {
|
2023-12-20 02:02:42 +01:00
|
|
|
"AddrBroadcaster", "Announcer", "Asyncore", "Downloader",
|
|
|
|
"InvBroadcaster", "Uploader"}
|
|
|
|
extra = self.config.getint('threads', 'receive')
|
2022-04-11 18:08:37 +02:00
|
|
|
for thread in threading.enumerate():
|
|
|
|
try:
|
|
|
|
threads.remove(thread.name)
|
|
|
|
except KeyError:
|
2023-12-20 02:02:42 +01:00
|
|
|
extra -= thread.name.startswith("ReceiveQueue_")
|
2022-04-11 18:08:37 +02:00
|
|
|
|
|
|
|
self.assertEqual(len(threads), 0)
|
|
|
|
self.assertEqual(extra, 0)
|
|
|
|
|
|
|
|
def test_stats(self):
|
|
|
|
"""Check that network starts connections and updates stats"""
|
|
|
|
pl = 0
|
|
|
|
for _ in range(30):
|
|
|
|
if pl == 0:
|
|
|
|
pl = len(self.pool)
|
|
|
|
if (
|
|
|
|
self.stats.receivedBytes() > 0 and self.stats.sentBytes() > 0
|
|
|
|
and pl > 0
|
|
|
|
# and len(self.stats.connectedHostsList()) > 0
|
|
|
|
):
|
|
|
|
break
|
|
|
|
time.sleep(1)
|
|
|
|
else:
|
|
|
|
self.fail('Have not started any connection in 30 sec')
|
|
|
|
|
2023-12-20 03:26:55 +01:00
|
|
|
def test_udp(self):
|
|
|
|
"""Invoke AnnounceThread.announceSelf() and check discovered peers"""
|
|
|
|
for _ in range(20):
|
|
|
|
if self.pool.udpSockets:
|
|
|
|
break
|
|
|
|
time.sleep(1)
|
|
|
|
else:
|
|
|
|
self.fail('No UDP sockets found in 20 sec')
|
|
|
|
|
|
|
|
for _ in range(10):
|
|
|
|
try:
|
|
|
|
self.state.announceThread.announceSelf()
|
|
|
|
except AttributeError:
|
|
|
|
self.fail('state.announceThread is not set properly')
|
|
|
|
time.sleep(1)
|
|
|
|
try:
|
|
|
|
peer = self.state.discoveredPeers.popitem()[0]
|
|
|
|
except KeyError:
|
|
|
|
continue
|
|
|
|
else:
|
|
|
|
self.assertEqual(peer.port, 8444)
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
self.fail('No self in discovered peers')
|
|
|
|
|
2022-04-11 18:08:37 +02:00
|
|
|
@classmethod
|
|
|
|
def tearDownClass(cls):
|
|
|
|
super(TestNetwork, cls).tearDownClass()
|
|
|
|
for thread in threading.enumerate():
|
|
|
|
if thread.name == "Asyncore":
|
|
|
|
thread.stopThread()
|