2015-08-20 11:27:15 -04:00
|
|
|
import hashlib
|
|
|
|
import struct
|
|
|
|
|
2018-07-30 21:23:38 -04:00
|
|
|
from twisted.trial import unittest
|
2018-05-23 19:33:16 -04:00
|
|
|
from twisted.internet import defer
|
2018-05-23 18:28:22 -04:00
|
|
|
from lbrynet.dht.node import Node
|
|
|
|
from lbrynet.dht import constants
|
2018-11-04 14:06:29 -05:00
|
|
|
from lbrynet.p2p.utils import generate_id
|
2015-08-20 11:27:15 -04:00
|
|
|
|
2017-10-10 13:31:18 -04:00
|
|
|
|
2015-08-20 11:27:15 -04:00
|
|
|
class NodeIDTest(unittest.TestCase):
|
2018-07-30 21:23:38 -04:00
|
|
|
|
2015-08-20 11:27:15 -04:00
|
|
|
def setUp(self):
|
2018-05-23 18:28:22 -04:00
|
|
|
self.node = Node()
|
2015-08-20 11:27:15 -04:00
|
|
|
|
2018-07-30 21:23:38 -04:00
|
|
|
def test_new_node_has_auto_created_id(self):
|
|
|
|
self.assertEqual(type(self.node.node_id), bytes)
|
|
|
|
self.assertEqual(len(self.node.node_id), 48)
|
2015-08-20 11:27:15 -04:00
|
|
|
|
2018-07-30 21:23:38 -04:00
|
|
|
def test_uniqueness_and_length_of_generated_ids(self):
|
|
|
|
previous_ids = []
|
2015-08-20 11:27:15 -04:00
|
|
|
for i in range(100):
|
2018-07-30 21:23:38 -04:00
|
|
|
new_id = self.node._generateID()
|
2018-10-18 13:42:45 +03:00
|
|
|
self.assertNotIn(new_id, previous_ids, f'id at index {i} not unique')
|
2018-07-30 21:23:38 -04:00
|
|
|
self.assertEqual(len(new_id), 48, 'id at index {} wrong length: {}'.format(i, len(new_id)))
|
|
|
|
previous_ids.append(new_id)
|
2015-08-20 11:27:15 -04:00
|
|
|
|
|
|
|
|
|
|
|
class NodeDataTest(unittest.TestCase):
|
|
|
|
""" Test case for the Node class's data-related functions """
|
2018-07-30 21:23:38 -04:00
|
|
|
|
2015-08-20 11:27:15 -04:00
|
|
|
def setUp(self):
|
2017-10-10 13:31:18 -04:00
|
|
|
h = hashlib.sha384()
|
2018-07-17 21:57:02 -03:00
|
|
|
h.update(b'test')
|
2018-05-23 18:28:22 -04:00
|
|
|
self.node = Node()
|
2018-07-30 21:23:38 -04:00
|
|
|
self.contact = self.node.contact_manager.make_contact(
|
|
|
|
h.digest(), '127.0.0.1', 12345, self.node._protocol)
|
2015-08-20 11:27:15 -04:00
|
|
|
self.token = self.node.make_token(self.contact.compact_ip())
|
|
|
|
self.cases = []
|
2018-07-17 21:57:02 -03:00
|
|
|
for i in range(5):
|
|
|
|
h.update(str(i).encode())
|
2015-08-20 11:27:15 -04:00
|
|
|
self.cases.append((h.digest(), 5000+2*i))
|
|
|
|
self.cases.append((h.digest(), 5001+2*i))
|
2016-12-13 17:37:23 -06:00
|
|
|
|
2017-10-24 20:05:38 -04:00
|
|
|
@defer.inlineCallbacks
|
2018-07-30 21:23:38 -04:00
|
|
|
def test_store(self):
|
2015-08-20 11:27:15 -04:00
|
|
|
""" Tests if the node can store (and privately retrieve) some data """
|
2018-05-23 18:28:22 -04:00
|
|
|
for key, port in self.cases:
|
2018-07-30 21:23:38 -04:00
|
|
|
yield self.node.store(
|
2018-06-07 12:18:07 -04:00
|
|
|
self.contact, key, self.token, port, self.contact.id, 0
|
|
|
|
)
|
2015-08-20 11:27:15 -04:00
|
|
|
for key, value in self.cases:
|
2018-07-30 21:23:38 -04:00
|
|
|
expected_result = self.contact.compact_ip() + struct.pack('>H', value) + self.contact.id
|
2018-07-21 16:55:43 -04:00
|
|
|
self.assertTrue(self.node._dataStore.hasPeersForBlob(key),
|
2018-07-30 21:23:38 -04:00
|
|
|
"Stored key not found in node's DataStore: '%s'" % key)
|
2018-10-18 14:41:33 +03:00
|
|
|
self.assertIn(expected_result, self.node._dataStore.getPeersForBlob(key),
|
2018-07-30 21:23:38 -04:00
|
|
|
"Stored val not found in node's DataStore: key:'%s' port:'%s' %s"
|
2017-10-10 13:31:18 -04:00
|
|
|
% (key, value, self.node._dataStore.getPeersForBlob(key)))
|
|
|
|
|
2015-08-20 11:27:15 -04:00
|
|
|
|
|
|
|
class NodeContactTest(unittest.TestCase):
|
|
|
|
""" Test case for the Node class's contact management-related functions """
|
|
|
|
def setUp(self):
|
2018-05-23 18:28:22 -04:00
|
|
|
self.node = Node()
|
2017-10-10 13:31:18 -04:00
|
|
|
|
2018-05-23 18:28:22 -04:00
|
|
|
@defer.inlineCallbacks
|
2018-07-30 21:23:38 -04:00
|
|
|
def test_add_contact(self):
|
2015-08-20 11:27:15 -04:00
|
|
|
""" Tests if a contact can be added and retrieved correctly """
|
|
|
|
# Create the contact
|
2018-07-30 21:23:38 -04:00
|
|
|
contact_id = generate_id(b'node1')
|
|
|
|
contact = self.node.contact_manager.make_contact(contact_id, '127.0.0.1', 9182, self.node._protocol)
|
2015-08-20 11:27:15 -04:00
|
|
|
# Now add it...
|
2018-05-23 18:28:22 -04:00
|
|
|
yield self.node.addContact(contact)
|
2015-08-20 11:27:15 -04:00
|
|
|
# ...and request the closest nodes to it using FIND_NODE
|
2018-07-30 21:23:38 -04:00
|
|
|
closest_nodes = self.node._routingTable.findCloseNodes(contact_id, constants.k)
|
|
|
|
self.assertEqual(len(closest_nodes), 1)
|
|
|
|
self.assertIn(contact, closest_nodes)
|
2017-10-10 13:31:18 -04:00
|
|
|
|
2018-05-23 18:28:22 -04:00
|
|
|
@defer.inlineCallbacks
|
2018-07-30 21:23:38 -04:00
|
|
|
def test_add_self_as_contact(self):
|
2015-08-20 11:27:15 -04:00
|
|
|
""" Tests the node's behaviour when attempting to add itself as a contact """
|
|
|
|
# Create a contact with the same ID as the local node's ID
|
2018-05-29 16:22:30 -04:00
|
|
|
contact = self.node.contact_manager.make_contact(self.node.node_id, '127.0.0.1', 9182, None)
|
2015-08-20 11:27:15 -04:00
|
|
|
# Now try to add it
|
2018-05-23 18:28:22 -04:00
|
|
|
yield self.node.addContact(contact)
|
2015-08-20 11:27:15 -04:00
|
|
|
# ...and request the closest nodes to it using FIND_NODE
|
2018-07-30 21:23:38 -04:00
|
|
|
closest_nodes = self.node._routingTable.findCloseNodes(self.node.node_id, constants.k)
|
|
|
|
self.assertNotIn(contact, closest_nodes, 'Node added itself as a contact.')
|