2019-04-28 17:38:40 -04:00
|
|
|
import ast
|
2019-07-27 17:10:03 +08:00
|
|
|
from typing import (
|
|
|
|
Union,
|
|
|
|
)
|
2019-04-28 17:38:40 -04:00
|
|
|
|
2019-07-27 17:10:03 +08:00
|
|
|
from libp2p.kademlia.kad_peerinfo import (
|
|
|
|
KadPeerInfo,
|
|
|
|
create_kad_peerinfo,
|
|
|
|
)
|
|
|
|
from libp2p.kademlia.network import KademliaServer
|
|
|
|
from libp2p.peer.id import ID
|
2019-03-26 20:44:01 +01:00
|
|
|
from libp2p.routing.interfaces import IPeerRouting
|
|
|
|
|
|
|
|
|
|
|
|
class KadmeliaPeerRouter(IPeerRouting):
|
2019-04-17 20:21:59 -04:00
|
|
|
# pylint: disable=too-few-public-methods
|
2019-03-26 20:44:01 +01:00
|
|
|
|
2019-07-27 17:10:03 +08:00
|
|
|
server: KademliaServer
|
|
|
|
|
|
|
|
def __init__(self, dht_server: KademliaServer) -> None:
|
2019-03-26 20:44:01 +01:00
|
|
|
self.server = dht_server
|
|
|
|
|
2019-07-27 17:10:03 +08:00
|
|
|
async def find_peer(self, peer_id: ID) -> KadPeerInfo:
|
2019-03-26 20:44:01 +01:00
|
|
|
"""
|
2019-05-05 14:32:41 -04:00
|
|
|
Find a specific peer
|
|
|
|
:param peer_id: peer to search for
|
|
|
|
:return: KadPeerInfo of specified peer
|
2019-03-26 20:44:01 +01:00
|
|
|
"""
|
2019-04-24 22:11:54 -04:00
|
|
|
# switching peer_id to xor_id used by kademlia as node_id
|
|
|
|
xor_id = peer_id.get_xor_id()
|
2019-04-27 21:59:25 -04:00
|
|
|
value = await self.server.get(xor_id)
|
2019-04-28 17:38:40 -04:00
|
|
|
return decode_peerinfo(value)
|
2019-03-26 20:44:01 +01:00
|
|
|
|
2019-07-27 17:10:03 +08:00
|
|
|
def decode_peerinfo(encoded: Union[bytes, str]) -> KadPeerInfo:
|
2019-04-28 17:38:40 -04:00
|
|
|
if isinstance(encoded, bytes):
|
|
|
|
encoded = encoded.decode()
|
|
|
|
try:
|
|
|
|
lines = ast.literal_eval(encoded)
|
|
|
|
except SyntaxError:
|
|
|
|
return None
|
|
|
|
ip = lines[1] # pylint: disable=invalid-name
|
|
|
|
port = lines[2]
|
|
|
|
peer_id = lines[3]
|
|
|
|
peer_info = create_kad_peerinfo(peer_id, ip, port)
|
|
|
|
return peer_info
|