py-libp2p/libp2p/kademlia/node.py

114 lines
3.1 KiB
Python
Raw Normal View History

2018-10-14 22:32:27 +08:00
from operator import itemgetter
import heapq
class Node:
def __init__(self, node_id, ip=None, port=None):
2019-01-16 01:41:41 +08:00
self.id = node_id # pylint: disable=invalid-name
self.ip = ip # pylint: disable=invalid-name
2018-10-14 22:32:27 +08:00
self.port = port
self.long_id = int(node_id.hex(), 16)
2019-01-16 01:41:41 +08:00
def same_home_as(self, node):
2018-10-14 22:32:27 +08:00
return self.ip == node.ip and self.port == node.port
2019-01-16 01:41:41 +08:00
def distance_to(self, node):
2018-10-14 22:32:27 +08:00
"""
Get the distance between this node and another.
"""
return self.long_id ^ node.long_id
def __iter__(self):
"""
Enables use of Node as a tuple - i.e., tuple(node) works.
"""
return iter([self.id, self.ip, self.port])
def __repr__(self):
return repr([self.long_id, self.ip, self.port])
def __str__(self):
return "%s:%s" % (self.ip, str(self.port))
2019-01-10 02:38:56 +08:00
class NodeHeap:
2018-10-14 22:32:27 +08:00
"""
A heap of nodes ordered by distance to a given node.
"""
def __init__(self, node, maxsize):
"""
Constructor.
@param node: The node to measure all distnaces from.
@param maxsize: The maximum size that this heap can grow to.
"""
self.node = node
self.heap = []
self.contacted = set()
self.maxsize = maxsize
2019-01-16 01:41:41 +08:00
def remove(self, peers):
2018-10-14 22:32:27 +08:00
"""
Remove a list of peer ids from this heap. Note that while this
heap retains a constant visible size (based on the iterator), it's
actual size may be quite a bit larger than what's exposed. Therefore,
removal of nodes may not change the visible size as previously added
nodes suddenly become visible.
"""
2019-01-16 01:41:41 +08:00
peers = set(peers)
if not peers:
2018-10-14 22:32:27 +08:00
return
nheap = []
for distance, node in self.heap:
2019-01-16 01:41:41 +08:00
if node.id not in peers:
2018-10-14 22:32:27 +08:00
heapq.heappush(nheap, (distance, node))
self.heap = nheap
2019-01-16 01:41:41 +08:00
def get_node(self, node_id):
2018-10-14 22:32:27 +08:00
for _, node in self.heap:
if node.id == node_id:
return node
return None
2019-01-16 01:41:41 +08:00
def have_contacted_all(self):
return len(self.get_uncontacted()) == 0
2018-10-14 22:32:27 +08:00
2019-01-16 01:41:41 +08:00
def get_ids(self):
2018-10-14 22:32:27 +08:00
return [n.id for n in self]
2019-01-16 01:41:41 +08:00
def mark_contacted(self, node):
2018-10-14 22:32:27 +08:00
self.contacted.add(node.id)
def popleft(self):
2019-01-16 01:41:41 +08:00
return heapq.heappop(self.heap)[1] if self else None
2018-10-14 22:32:27 +08:00
def push(self, nodes):
"""
Push nodes onto heap.
@param nodes: This can be a single item or a C{list}.
"""
if not isinstance(nodes, list):
nodes = [nodes]
for node in nodes:
if node not in self:
2019-01-16 01:41:41 +08:00
distance = self.node.distance_to(node)
2018-10-14 22:32:27 +08:00
heapq.heappush(self.heap, (distance, node))
def __len__(self):
return min(len(self.heap), self.maxsize)
def __iter__(self):
nodes = heapq.nsmallest(self.maxsize, self.heap)
return iter(map(itemgetter(1), nodes))
def __contains__(self, node):
2019-01-16 01:41:41 +08:00
for _, other in self.heap:
if node.id == other.id:
2018-10-14 22:32:27 +08:00
return True
return False
2019-01-16 01:41:41 +08:00
def get_uncontacted(self):
2018-10-14 22:32:27 +08:00
return [n for n in self if n.id not in self.contacted]