diff --git a/CHANGELOG.md b/CHANGELOG.md index fe654a618..22ccf51a2 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -20,7 +20,7 @@ at anytime. ### Fixed * Removed update_metadata function that could cause update problems - * + * Fix DHT contact bug * ## [0.9.2rc3] - 2017-03-29 diff --git a/lbrynet/dht/routingtable.py b/lbrynet/dht/routingtable.py index ed09fb40d..3aa493119 100644 --- a/lbrynet/dht/routingtable.py +++ b/lbrynet/dht/routingtable.py @@ -308,10 +308,9 @@ class TreeRoutingTable(RoutingTable): @return: The index of the k-bucket responsible for the specified key @rtype: int """ - valKey = long(key.encode('hex'), 16) i = 0 for bucket in self._buckets: - if bucket.keyInRange(valKey): + if bucket.keyInRange(key): return i else: i += 1 @@ -402,15 +401,15 @@ class OptimizedTreeRoutingTable(TreeRoutingTable): # Put the new contact in our replacement cache for the # corresponding k-bucket (or update it's position if # it exists already) - if not self._replacementCache.has_key(bucketIndex): + if bucketIndex not in self._replacementCache: self._replacementCache[bucketIndex] = [] if contact in self._replacementCache[bucketIndex]: self._replacementCache[bucketIndex].remove(contact) # TODO: Using k to limit the size of the contact - # replacement cache - maybe define a seperate value for + # replacement cache - maybe define a separate value for # this in constants.py? - elif len(self._replacementCache) >= constants.k: - self._replacementCache.pop(0) + elif len(self._replacementCache[bucketIndex]) >= constants.k: + self._replacementCache[bucketIndex].pop(0) self._replacementCache[bucketIndex].append(contact) def removeContact(self, contactID): @@ -428,8 +427,8 @@ class OptimizedTreeRoutingTable(TreeRoutingTable): contact.failedRPCs += 1 if contact.failedRPCs >= 5: self._buckets[bucketIndex].removeContact(contactID) - # Replace this stale contact with one from our replacemnent cache, if we have any - if self._replacementCache.has_key(bucketIndex): + # Replace this stale contact with one from our replacement cache, if we have any + if bucketIndex in self._replacementCache: if len(self._replacementCache[bucketIndex]) > 0: self._buckets[bucketIndex].addContact( self._replacementCache[bucketIndex].pop()) diff --git a/tests/unit/dht/test_routingtable.py b/tests/unit/dht/test_routingtable.py new file mode 100644 index 000000000..36d184d8c --- /dev/null +++ b/tests/unit/dht/test_routingtable.py @@ -0,0 +1,59 @@ +import unittest + +from lbrynet.dht import contact, routingtable, constants + + +class KeyErrorFixedTest(unittest.TestCase): + """ Basic tests case for boolean operators on the Contact class """ + + def setUp(self): + own_id = (2 ** constants.key_bits) - 1 + # carefully chosen own_id. here's the logic + # we want a bunch of buckets (k+1, to be exact), and we want to make sure own_id + # is not in bucket 0. so we put own_id at the end so we can keep splitting by adding to the + # end + + self.table = routingtable.OptimizedTreeRoutingTable(own_id) + + def fill_bucket(self, bucket_min): + bucket_size = constants.k + for i in range(bucket_min, bucket_min + bucket_size): + self.table.addContact(contact.Contact(long(i), '127.0.0.1', 9999, None)) + + def overflow_bucket(self, bucket_min): + bucket_size = constants.k + self.fill_bucket(bucket_min) + self.table.addContact( + contact.Contact(long(bucket_min + bucket_size + 1), '127.0.0.1', 9999, None)) + + def testKeyError(self): + + # find middle, so we know where bucket will split + bucket_middle = self.table._buckets[0].rangeMax / 2 + + # fill last bucket + self.fill_bucket(self.table._buckets[0].rangeMax - constants.k - 1) + # -1 in previous line because own_id is in last bucket + + # fill/overflow 7 more buckets + bucket_start = 0 + for i in range(0, constants.k): + self.overflow_bucket(bucket_start) + bucket_start += bucket_middle / (2 ** i) + + # replacement cache now has k-1 entries. + # adding one more contact to bucket 0 used to cause a KeyError, but it should work + self.table.addContact(contact.Contact(long(constants.k + 2), '127.0.0.1', 9999, None)) + + # import math + # print "" + # for i, bucket in enumerate(self.table._buckets): + # print "Bucket " + str(i) + " (2 ** " + str( + # math.log(bucket.rangeMin, 2) if bucket.rangeMin > 0 else 0) + " <= x < 2 ** "+str( + # math.log(bucket.rangeMax, 2)) + ")" + # for c in bucket.getContacts(): + # print " contact " + str(c.id) + # for key, bucket in self.table._replacementCache.iteritems(): + # print "Replacement Cache for Bucket " + str(key) + # for c in bucket: + # print " contact " + str(c.id)