Add type hints to network folder
This commit is contained in:
parent
e1592997a8
commit
edd164c878
|
@ -1,8 +1,22 @@
|
||||||
|
import asyncio
|
||||||
|
|
||||||
from .raw_connection_interface import IRawConnection
|
from .raw_connection_interface import IRawConnection
|
||||||
|
|
||||||
class RawConnection(IRawConnection):
|
class RawConnection(IRawConnection):
|
||||||
|
|
||||||
def __init__(self, ip, port, reader, writer, initiator):
|
conn_ip: str
|
||||||
|
conn_port: str
|
||||||
|
reader: asyncio.StreamReader
|
||||||
|
writer: asyncio.StreamWriter
|
||||||
|
_next_id: int
|
||||||
|
initiator: bool
|
||||||
|
|
||||||
|
def __init__(self,
|
||||||
|
ip: str,
|
||||||
|
port: str,
|
||||||
|
reader: asyncio.StreamReader,
|
||||||
|
writer: asyncio.StreamWriter,
|
||||||
|
initiator: bool) -> None:
|
||||||
# pylint: disable=too-many-arguments
|
# pylint: disable=too-many-arguments
|
||||||
self.conn_ip = ip
|
self.conn_ip = ip
|
||||||
self.conn_port = port
|
self.conn_port = port
|
||||||
|
@ -11,12 +25,12 @@ class RawConnection(IRawConnection):
|
||||||
self._next_id = 0 if initiator else 1
|
self._next_id = 0 if initiator else 1
|
||||||
self.initiator = initiator
|
self.initiator = initiator
|
||||||
|
|
||||||
async def write(self, data):
|
async def write(self, data: bytes) -> None:
|
||||||
self.writer.write(data)
|
self.writer.write(data)
|
||||||
self.writer.write("\n".encode())
|
self.writer.write("\n".encode())
|
||||||
await self.writer.drain()
|
await self.writer.drain()
|
||||||
|
|
||||||
async def read(self):
|
async def read(self) -> bytes:
|
||||||
line = await self.reader.readline()
|
line = await self.reader.readline()
|
||||||
adjusted_line = line.decode().rstrip('\n')
|
adjusted_line = line.decode().rstrip('\n')
|
||||||
|
|
||||||
|
@ -24,10 +38,10 @@ class RawConnection(IRawConnection):
|
||||||
# encoding and decoding
|
# encoding and decoding
|
||||||
return adjusted_line.encode()
|
return adjusted_line.encode()
|
||||||
|
|
||||||
def close(self):
|
def close(self) -> None:
|
||||||
self.writer.close()
|
self.writer.close()
|
||||||
|
|
||||||
def next_stream_id(self):
|
def next_stream_id(self) -> int:
|
||||||
"""
|
"""
|
||||||
Get next available stream id
|
Get next available stream id
|
||||||
:return: next available stream id for the connection
|
:return: next available stream id for the connection
|
||||||
|
|
|
@ -1,16 +1,31 @@
|
||||||
from abc import ABC, abstractmethod
|
from abc import ABC, abstractmethod
|
||||||
|
from typing import (
|
||||||
|
Any,
|
||||||
|
Callable,
|
||||||
|
Coroutine,
|
||||||
|
Sequence,
|
||||||
|
)
|
||||||
|
|
||||||
|
from multiaddr import Multiaddr
|
||||||
|
|
||||||
|
from .notifee_interface import INotifee
|
||||||
|
from .stream.net_stream import NetStream
|
||||||
|
from libp2p.peer.id import (
|
||||||
|
ID,
|
||||||
|
)
|
||||||
|
from libp2p.stream_muxer.muxed_connection_interface import IMuxedConn
|
||||||
|
|
||||||
|
|
||||||
class INetwork(ABC):
|
class INetwork(ABC):
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def get_peer_id(self):
|
def get_peer_id(self) -> ID:
|
||||||
"""
|
"""
|
||||||
:return: the peer id
|
:return: the peer id
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def dial_peer(self, peer_id):
|
def dial_peer(self, peer_id: ID) -> Coroutine[Any, Any, IMuxedConn]:
|
||||||
"""
|
"""
|
||||||
dial_peer try to create a connection to peer_id
|
dial_peer try to create a connection to peer_id
|
||||||
|
|
||||||
|
@ -20,7 +35,7 @@ class INetwork(ABC):
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def set_stream_handler(self, protocol_id, stream_handler):
|
def set_stream_handler(self, protocol_id: str, stream_handler: Callable[[NetStream], None]) -> bool:
|
||||||
"""
|
"""
|
||||||
:param protocol_id: protocol id used on stream
|
:param protocol_id: protocol id used on stream
|
||||||
:param stream_handler: a stream handler instance
|
:param stream_handler: a stream handler instance
|
||||||
|
@ -28,7 +43,7 @@ class INetwork(ABC):
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def new_stream(self, peer_id, protocol_ids):
|
def new_stream(self, peer_id: ID, protocol_ids: Sequence[str]) -> Coroutine[Any, Any, NetStream]:
|
||||||
"""
|
"""
|
||||||
:param peer_id: peer_id of destination
|
:param peer_id: peer_id of destination
|
||||||
:param protocol_ids: available protocol ids to use for stream
|
:param protocol_ids: available protocol ids to use for stream
|
||||||
|
@ -36,14 +51,14 @@ class INetwork(ABC):
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def listen(self, *args):
|
def listen(self, *args: Multiaddr) -> Coroutine[Any, Any, bool]:
|
||||||
"""
|
"""
|
||||||
:param *args: one or many multiaddrs to start listening on
|
:param *args: one or many multiaddrs to start listening on
|
||||||
:return: True if at least one success
|
:return: True if at least one success
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def notify(self, notifee):
|
def notify(self, notifee: INotifee) -> bool:
|
||||||
"""
|
"""
|
||||||
:param notifee: object implementing Notifee interface
|
:param notifee: object implementing Notifee interface
|
||||||
:return: true if notifee registered successfully, false otherwise
|
:return: true if notifee registered successfully, false otherwise
|
||||||
|
|
|
@ -1,44 +1,51 @@
|
||||||
from abc import ABC, abstractmethod
|
from abc import ABC, abstractmethod
|
||||||
|
|
||||||
|
from multiaddr import Multiaddr
|
||||||
|
|
||||||
|
from libp2p.network.network_interface import INetwork
|
||||||
|
from libp2p.network.stream.net_stream_interface import INetStream
|
||||||
|
from libp2p.stream_muxer.muxed_connection_interface import IMuxedConn
|
||||||
|
|
||||||
|
|
||||||
class INotifee(ABC):
|
class INotifee(ABC):
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
async def opened_stream(self, network, stream):
|
async def opened_stream(self, network: INetwork, stream: INetStream) -> None:
|
||||||
"""
|
"""
|
||||||
:param network: network the stream was opened on
|
:param network: network the stream was opened on
|
||||||
:param stream: stream that was opened
|
:param stream: stream that was opened
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
async def closed_stream(self, network, stream):
|
async def closed_stream(self, network: INetwork, stream: INetStream) -> None:
|
||||||
"""
|
"""
|
||||||
:param network: network the stream was closed on
|
:param network: network the stream was closed on
|
||||||
:param stream: stream that was closed
|
:param stream: stream that was closed
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
async def connected(self, network, conn):
|
async def connected(self, network: INetwork, conn: IMuxedConn) -> None:
|
||||||
"""
|
"""
|
||||||
:param network: network the connection was opened on
|
:param network: network the connection was opened on
|
||||||
:param conn: connection that was opened
|
:param conn: connection that was opened
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
async def disconnected(self, network, conn):
|
async def disconnected(self, network: INetwork, conn: IMuxedConn) -> None:
|
||||||
"""
|
"""
|
||||||
:param network: network the connection was closed on
|
:param network: network the connection was closed on
|
||||||
:param conn: connection that was closed
|
:param conn: connection that was closed
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
async def listen(self, network, multiaddr):
|
async def listen(self, network: INetwork, multiaddr: Multiaddr) -> None:
|
||||||
"""
|
"""
|
||||||
:param network: network the listener is listening on
|
:param network: network the listener is listening on
|
||||||
:param multiaddr: multiaddress listener is listening on
|
:param multiaddr: multiaddress listener is listening on
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
async def listen_close(self, network, multiaddr):
|
async def listen_close(self, network: INetwork, multiaddr: Multiaddr) -> None:
|
||||||
"""
|
"""
|
||||||
:param network: network the connection was opened on
|
:param network: network the connection was opened on
|
||||||
:param multiaddr: multiaddress listener is no longer listening on
|
:param multiaddr: multiaddress listener is no longer listening on
|
||||||
|
|
|
@ -1,41 +1,47 @@
|
||||||
from .net_stream_interface import INetStream
|
from .net_stream_interface import INetStream
|
||||||
|
from libp2p.stream_muxer.mplex.mplex_stream import MplexStream
|
||||||
|
from libp2p.stream_muxer.mplex.mplex import Mplex
|
||||||
|
|
||||||
|
|
||||||
class NetStream(INetStream):
|
class NetStream(INetStream):
|
||||||
|
|
||||||
def __init__(self, muxed_stream):
|
muxed_stream: MplexStream
|
||||||
|
mplex_conn: Mplex
|
||||||
|
protocol_id: str
|
||||||
|
|
||||||
|
def __init__(self, muxed_stream: MplexStream) -> None:
|
||||||
self.muxed_stream = muxed_stream
|
self.muxed_stream = muxed_stream
|
||||||
self.mplex_conn = muxed_stream.mplex_conn
|
self.mplex_conn = muxed_stream.mplex_conn
|
||||||
self.protocol_id = None
|
self.protocol_id = None
|
||||||
|
|
||||||
def get_protocol(self):
|
def get_protocol(self) -> str:
|
||||||
"""
|
"""
|
||||||
:return: protocol id that stream runs on
|
:return: protocol id that stream runs on
|
||||||
"""
|
"""
|
||||||
return self.protocol_id
|
return self.protocol_id
|
||||||
|
|
||||||
def set_protocol(self, protocol_id):
|
def set_protocol(self, protocol_id: str) -> None:
|
||||||
"""
|
"""
|
||||||
:param protocol_id: protocol id that stream runs on
|
:param protocol_id: protocol id that stream runs on
|
||||||
:return: true if successful
|
:return: true if successful
|
||||||
"""
|
"""
|
||||||
self.protocol_id = protocol_id
|
self.protocol_id = protocol_id
|
||||||
|
|
||||||
async def read(self):
|
async def read(self) -> bytes:
|
||||||
"""
|
"""
|
||||||
read from stream
|
read from stream
|
||||||
:return: bytes of input until EOF
|
:return: bytes of input until EOF
|
||||||
"""
|
"""
|
||||||
return await self.muxed_stream.read()
|
return await self.muxed_stream.read()
|
||||||
|
|
||||||
async def write(self, data):
|
async def write(self, data: bytes) -> int:
|
||||||
"""
|
"""
|
||||||
write to stream
|
write to stream
|
||||||
:return: number of bytes written
|
:return: number of bytes written
|
||||||
"""
|
"""
|
||||||
return await self.muxed_stream.write(data)
|
return await self.muxed_stream.write(data)
|
||||||
|
|
||||||
async def close(self):
|
async def close(self) -> bool:
|
||||||
"""
|
"""
|
||||||
close stream
|
close stream
|
||||||
:return: true if successful
|
:return: true if successful
|
||||||
|
|
|
@ -1,37 +1,41 @@
|
||||||
from abc import ABC, abstractmethod
|
from abc import ABC, abstractmethod
|
||||||
|
from typing import (
|
||||||
|
Any,
|
||||||
|
Coroutine,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
class INetStream(ABC):
|
class INetStream(ABC):
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def get_protocol(self):
|
def get_protocol(self) -> str:
|
||||||
"""
|
"""
|
||||||
:return: protocol id that stream runs on
|
:return: protocol id that stream runs on
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def set_protocol(self, protocol_id):
|
def set_protocol(self, protocol_id: str) -> bool:
|
||||||
"""
|
"""
|
||||||
:param protocol_id: protocol id that stream runs on
|
:param protocol_id: protocol id that stream runs on
|
||||||
:return: true if successful
|
:return: true if successful
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def read(self):
|
def read(self) -> Coroutine[Any, Any, bytes]:
|
||||||
"""
|
"""
|
||||||
reads from the underlying muxed_stream
|
reads from the underlying muxed_stream
|
||||||
:return: bytes of input
|
:return: bytes of input
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def write(self, _bytes):
|
def write(self, data: bytes) -> Coroutine[Any, Any, int]:
|
||||||
"""
|
"""
|
||||||
write to the underlying muxed_stream
|
write to the underlying muxed_stream
|
||||||
:return: number of bytes written
|
:return: number of bytes written
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
def close(self):
|
def close(self) -> Coroutine[Any, Any, bool]:
|
||||||
"""
|
"""
|
||||||
close the underlying muxed stream
|
close the underlying muxed stream
|
||||||
:return: true if successful
|
:return: true if successful
|
||||||
|
|
|
@ -1,18 +1,57 @@
|
||||||
import asyncio
|
import asyncio
|
||||||
|
from typing import (
|
||||||
|
Any,
|
||||||
|
Callable,
|
||||||
|
Coroutine,
|
||||||
|
Dict,
|
||||||
|
List,
|
||||||
|
Sequence,
|
||||||
|
)
|
||||||
|
|
||||||
from libp2p.protocol_muxer.multiselect_client import MultiselectClient
|
from multiaddr import Multiaddr
|
||||||
from libp2p.protocol_muxer.multiselect import Multiselect
|
|
||||||
from libp2p.peer.id import id_b58_decode
|
|
||||||
|
|
||||||
from .network_interface import INetwork
|
from .network_interface import INetwork
|
||||||
from .notifee_interface import INotifee
|
from .notifee_interface import INotifee
|
||||||
from .stream.net_stream import NetStream
|
|
||||||
from .connection.raw_connection import RawConnection
|
from .connection.raw_connection import RawConnection
|
||||||
|
from .stream.net_stream import NetStream
|
||||||
|
from libp2p.peer.id import (
|
||||||
|
ID,
|
||||||
|
id_b58_decode,
|
||||||
|
)
|
||||||
|
from libp2p.peer.peerstore import PeerStore
|
||||||
|
from libp2p.protocol_muxer.multiselect import Multiselect
|
||||||
|
from libp2p.protocol_muxer.multiselect_client import MultiselectClient
|
||||||
|
from libp2p.routing.interfaces import IPeerRouting
|
||||||
|
from libp2p.transport.upgrader import TransportUpgrader
|
||||||
|
from libp2p.transport.transport_interface import ITransport
|
||||||
|
from libp2p.transport.listener_interface import IListener
|
||||||
|
from libp2p.stream_muxer.mplex.mplex_stream import MplexStream
|
||||||
|
from libp2p.stream_muxer.muxed_connection_interface import IMuxedConn
|
||||||
|
|
||||||
|
|
||||||
class Swarm(INetwork):
|
class Swarm(INetwork):
|
||||||
# pylint: disable=too-many-instance-attributes,cell-var-from-loop,too-many-arguments
|
# pylint: disable=too-many-instance-attributes,cell-var-from-loop,too-many-arguments
|
||||||
|
|
||||||
def __init__(self, peer_id, peerstore, upgrader, transport, router):
|
self_id: ID
|
||||||
|
peerstore: PeerStore
|
||||||
|
upgrader: TransportUpgrader
|
||||||
|
transport: ITransport
|
||||||
|
router: IPeerRouting
|
||||||
|
connections: Dict[ID, IMuxedConn]
|
||||||
|
listeners: Dict[str, IListener]
|
||||||
|
stream_handlers: Dict[NetStream, Callable[[NetStream], None]]
|
||||||
|
|
||||||
|
multiselect: Multiselect
|
||||||
|
multiselect_client: MultiselectClient
|
||||||
|
|
||||||
|
notifees: List[INotifee]
|
||||||
|
|
||||||
|
def __init__(self,
|
||||||
|
peer_id: ID,
|
||||||
|
peerstore: PeerStore,
|
||||||
|
upgrader: TransportUpgrader,
|
||||||
|
transport: ITransport,
|
||||||
|
router: IPeerRouting):
|
||||||
self.self_id = peer_id
|
self.self_id = peer_id
|
||||||
self.peerstore = peerstore
|
self.peerstore = peerstore
|
||||||
self.upgrader = upgrader
|
self.upgrader = upgrader
|
||||||
|
@ -32,10 +71,10 @@ class Swarm(INetwork):
|
||||||
# Create generic protocol handler
|
# Create generic protocol handler
|
||||||
self.generic_protocol_handler = create_generic_protocol_handler(self)
|
self.generic_protocol_handler = create_generic_protocol_handler(self)
|
||||||
|
|
||||||
def get_peer_id(self):
|
def get_peer_id(self) -> ID:
|
||||||
return self.self_id
|
return self.self_id
|
||||||
|
|
||||||
def set_stream_handler(self, protocol_id, stream_handler):
|
def set_stream_handler(self, protocol_id: str, stream_handler: Callable[[NetStream], None]) -> bool:
|
||||||
"""
|
"""
|
||||||
:param protocol_id: protocol id used on stream
|
:param protocol_id: protocol id used on stream
|
||||||
:param stream_handler: a stream handler instance
|
:param stream_handler: a stream handler instance
|
||||||
|
@ -44,7 +83,7 @@ class Swarm(INetwork):
|
||||||
self.multiselect.add_handler(protocol_id, stream_handler)
|
self.multiselect.add_handler(protocol_id, stream_handler)
|
||||||
return True
|
return True
|
||||||
|
|
||||||
async def dial_peer(self, peer_id):
|
async def dial_peer(self, peer_id: ID) -> IMuxedConn:
|
||||||
"""
|
"""
|
||||||
dial_peer try to create a connection to peer_id
|
dial_peer try to create a connection to peer_id
|
||||||
:param peer_id: peer if we want to dial
|
:param peer_id: peer if we want to dial
|
||||||
|
@ -87,7 +126,7 @@ class Swarm(INetwork):
|
||||||
|
|
||||||
return muxed_conn
|
return muxed_conn
|
||||||
|
|
||||||
async def new_stream(self, peer_id, protocol_ids):
|
async def new_stream(self, peer_id: ID, protocol_ids: Sequence[str]) -> NetStream:
|
||||||
"""
|
"""
|
||||||
:param peer_id: peer_id of destination
|
:param peer_id: peer_id of destination
|
||||||
:param protocol_id: protocol id
|
:param protocol_id: protocol id
|
||||||
|
@ -109,7 +148,7 @@ class Swarm(INetwork):
|
||||||
muxed_stream = await muxed_conn.open_stream(protocol_ids[0], multiaddr)
|
muxed_stream = await muxed_conn.open_stream(protocol_ids[0], multiaddr)
|
||||||
|
|
||||||
# Perform protocol muxing to determine protocol to use
|
# Perform protocol muxing to determine protocol to use
|
||||||
selected_protocol = await self.multiselect_client.select_one_of(protocol_ids, muxed_stream)
|
selected_protocol = await self.multiselect_client.select_one_of(list(protocol_ids), muxed_stream)
|
||||||
|
|
||||||
# Create a net stream with the selected protocol
|
# Create a net stream with the selected protocol
|
||||||
net_stream = NetStream(muxed_stream)
|
net_stream = NetStream(muxed_stream)
|
||||||
|
@ -121,7 +160,7 @@ class Swarm(INetwork):
|
||||||
|
|
||||||
return net_stream
|
return net_stream
|
||||||
|
|
||||||
async def listen(self, *args):
|
async def listen(self, *args: Multiaddr) -> bool:
|
||||||
"""
|
"""
|
||||||
:param *args: one or many multiaddrs to start listening on
|
:param *args: one or many multiaddrs to start listening on
|
||||||
:return: true if at least one success
|
:return: true if at least one success
|
||||||
|
@ -139,7 +178,7 @@ class Swarm(INetwork):
|
||||||
if str(multiaddr) in self.listeners:
|
if str(multiaddr) in self.listeners:
|
||||||
return True
|
return True
|
||||||
|
|
||||||
async def conn_handler(reader, writer):
|
async def conn_handler(reader: asyncio.StreamReader, writer: asyncio.StreamWriter) -> None:
|
||||||
# Read in first message (should be peer_id of initiator) and ack
|
# Read in first message (should be peer_id of initiator) and ack
|
||||||
peer_id = id_b58_decode((await reader.read(1024)).decode())
|
peer_id = id_b58_decode((await reader.read(1024)).decode())
|
||||||
|
|
||||||
|
@ -182,7 +221,7 @@ class Swarm(INetwork):
|
||||||
# No multiaddr succeeded
|
# No multiaddr succeeded
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def notify(self, notifee):
|
def notify(self, notifee: INotifee) -> bool:
|
||||||
"""
|
"""
|
||||||
:param notifee: object implementing Notifee interface
|
:param notifee: object implementing Notifee interface
|
||||||
:return: true if notifee registered successfully, false otherwise
|
:return: true if notifee registered successfully, false otherwise
|
||||||
|
@ -192,7 +231,7 @@ class Swarm(INetwork):
|
||||||
return True
|
return True
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def add_router(self, router):
|
def add_router(self, router: IPeerRouting) -> None:
|
||||||
self.router = router
|
self.router = router
|
||||||
|
|
||||||
# TODO: `tear_down`
|
# TODO: `tear_down`
|
||||||
|
@ -204,7 +243,10 @@ class Swarm(INetwork):
|
||||||
# TODO: `disconnect`?
|
# TODO: `disconnect`?
|
||||||
|
|
||||||
|
|
||||||
def create_generic_protocol_handler(swarm):
|
GenericProtocolHandlerFn = Callable[[MplexStream], Coroutine[Any, Any, None]]
|
||||||
|
|
||||||
|
|
||||||
|
def create_generic_protocol_handler(swarm: Swarm) -> GenericProtocolHandlerFn:
|
||||||
"""
|
"""
|
||||||
Create a generic protocol handler from the given swarm. We use swarm
|
Create a generic protocol handler from the given swarm. We use swarm
|
||||||
to extract the multiselect module so that generic_protocol_handler
|
to extract the multiselect module so that generic_protocol_handler
|
||||||
|
@ -213,7 +255,7 @@ def create_generic_protocol_handler(swarm):
|
||||||
"""
|
"""
|
||||||
multiselect = swarm.multiselect
|
multiselect = swarm.multiselect
|
||||||
|
|
||||||
async def generic_protocol_handler(muxed_stream):
|
async def generic_protocol_handler(muxed_stream: MplexStream) -> None:
|
||||||
# Perform protocol muxing to determine protocol to use
|
# Perform protocol muxing to determine protocol to use
|
||||||
protocol, handler = await multiselect.negotiate(muxed_stream)
|
protocol, handler = await multiselect.negotiate(muxed_stream)
|
||||||
|
|
||||||
|
@ -229,5 +271,6 @@ def create_generic_protocol_handler(swarm):
|
||||||
|
|
||||||
return generic_protocol_handler
|
return generic_protocol_handler
|
||||||
|
|
||||||
|
|
||||||
class SwarmException(Exception):
|
class SwarmException(Exception):
|
||||||
pass
|
pass
|
||||||
|
|
Loading…
Reference in New Issue
Block a user