PR feedbacks

- Nits
- Add `SecurityUpgradeFailure` and handle `UpgradeFailure` in Swarm.
This commit is contained in:
mhchia 2019-08-20 16:42:34 +08:00
parent 2a1367b011
commit 5768daa9bf
No known key found for this signature in database
GPG Key ID: 389EFBEA1362589A
6 changed files with 33 additions and 21 deletions

View File

@ -10,6 +10,7 @@ from libp2p.protocol_muxer.multiselect_client import MultiselectClient
from libp2p.protocol_muxer.multiselect_communicator import StreamCommunicator from libp2p.protocol_muxer.multiselect_communicator import StreamCommunicator
from libp2p.routing.interfaces import IPeerRouting from libp2p.routing.interfaces import IPeerRouting
from libp2p.stream_muxer.abc import IMuxedConn, IMuxedStream from libp2p.stream_muxer.abc import IMuxedConn, IMuxedStream
from libp2p.transport.exceptions import UpgradeFailure
from libp2p.transport.listener_interface import IListener from libp2p.transport.listener_interface import IListener
from libp2p.transport.transport_interface import ITransport from libp2p.transport.transport_interface import ITransport
from libp2p.transport.upgrader import TransportUpgrader from libp2p.transport.upgrader import TransportUpgrader
@ -197,6 +198,7 @@ class Swarm(INetwork):
# Per, https://discuss.libp2p.io/t/multistream-security/130, we first secure # Per, https://discuss.libp2p.io/t/multistream-security/130, we first secure
# the conn and then mux the conn # the conn and then mux the conn
# FIXME: This dummy `ID(b"")` for the remote peer is useless. # FIXME: This dummy `ID(b"")` for the remote peer is useless.
try:
secured_conn = await self.upgrader.upgrade_security( secured_conn = await self.upgrader.upgrade_security(
raw_conn, ID(b""), False raw_conn, ID(b""), False
) )
@ -204,6 +206,10 @@ class Swarm(INetwork):
muxed_conn = await self.upgrader.upgrade_connection( muxed_conn = await self.upgrader.upgrade_connection(
secured_conn, self.generic_protocol_handler, peer_id secured_conn, self.generic_protocol_handler, peer_id
) )
except UpgradeFailure:
# TODO: Add logging to indicate the failure
raw_conn.close()
return
# Store muxed_conn with peer id # Store muxed_conn with peer id
self.connections[peer_id] = muxed_conn self.connections[peer_id] = muxed_conn

View File

@ -80,7 +80,8 @@ class Multiselect(IMultiselectMuxer):
# Confirm that the protocols are the same # Confirm that the protocols are the same
if not validate_handshake(handshake_contents): if not validate_handshake(handshake_contents):
raise MultiselectError( raise MultiselectError(
f"multiselect protocol ID mismatch: handshake_contents={handshake_contents}" "multiselect protocol ID mismatch: "
f"received handshake_contents={handshake_contents}"
) )
# Handshake succeeded if this point is reached # Handshake succeeded if this point is reached

View File

@ -1,2 +0,0 @@
class UpgradeFailure(Exception):
pass

View File

@ -4,10 +4,10 @@ from libp2p.peer.id import ID
from libp2p.security.base_session import BaseSession from libp2p.security.base_session import BaseSession
from libp2p.security.base_transport import BaseSecureTransport from libp2p.security.base_transport import BaseSecureTransport
from libp2p.security.secure_conn_interface import ISecureConn from libp2p.security.secure_conn_interface import ISecureConn
from libp2p.transport.exceptions import SecurityUpgradeFailure
from libp2p.typing import TProtocol from libp2p.typing import TProtocol
from libp2p.utils import encode_fixedint_prefixed, read_fixedint_prefixed from libp2p.utils import encode_fixedint_prefixed, read_fixedint_prefixed
from .exceptions import UpgradeFailure
from .pb import plaintext_pb2 from .pb import plaintext_pb2
# Reference: https://github.com/libp2p/go-libp2p-core/blob/master/sec/insecure/insecure.go # Reference: https://github.com/libp2p/go-libp2p-core/blob/master/sec/insecure/insecure.go
@ -17,7 +17,6 @@ PLAINTEXT_PROTOCOL_ID = TProtocol("/plaintext/2.0.0")
class InsecureSession(BaseSession): class InsecureSession(BaseSession):
# FIXME: Update the read/write to `BaseSession`
async def run_handshake(self): async def run_handshake(self):
msg = make_exchange_message(self.local_private_key.get_public_key()) msg = make_exchange_message(self.local_private_key.get_public_key())
msg_bytes = msg.SerializeToString() msg_bytes = msg.SerializeToString()
@ -61,7 +60,7 @@ class InsecureTransport(BaseSecureTransport):
# TODO: Check if `remote_public_key is not None`. If so, check if `session.remote_peer` # TODO: Check if `remote_public_key is not None`. If so, check if `session.remote_peer`
received_peer_id = session.get_remote_peer() received_peer_id = session.get_remote_peer()
if received_peer_id != peer_id: if received_peer_id != peer_id:
raise UpgradeFailure( raise SecurityUpgradeFailure(
"remote peer sent unexpected peer ID. " "remote peer sent unexpected peer ID. "
f"expected={peer_id} received={received_peer_id}" f"expected={peer_id} received={received_peer_id}"
) )

View File

@ -6,6 +6,7 @@ from libp2p.peer.id import ID
from libp2p.security.base_transport import BaseSecureTransport from libp2p.security.base_transport import BaseSecureTransport
from libp2p.security.insecure.transport import InsecureSession from libp2p.security.insecure.transport import InsecureSession
from libp2p.security.secure_conn_interface import ISecureConn from libp2p.security.secure_conn_interface import ISecureConn
from libp2p.transport.exceptions import SecurityUpgradeFailure
class SimpleSecurityTransport(BaseSecureTransport): class SimpleSecurityTransport(BaseSecureTransport):
@ -25,14 +26,14 @@ class SimpleSecurityTransport(BaseSecureTransport):
incoming = (await conn.read()).decode() incoming = (await conn.read()).decode()
if incoming != self.key_phrase: if incoming != self.key_phrase:
raise Exception( raise SecurityUpgradeFailure(
"Key phrase differed between nodes. Expected " + self.key_phrase "Key phrase differed between nodes. Expected " + self.key_phrase
) )
session = InsecureSession(self, conn, ID(b"")) session = InsecureSession(self, conn, ID(b""))
# TODO: Calls handshake to make them know the peer id each other, otherwise tests fail. # NOTE: Here we calls `run_handshake` for both sides to exchange their public keys and
# However, it seems pretty weird that `SimpleSecurityTransport` sends peer id through # peer ids, otherwise tests fail. However, it seems pretty weird that
# `Insecure`. # `SimpleSecurityTransport` sends peer id through `Insecure`.
await session.run_handshake() await session.run_handshake()
# NOTE: this is abusing the abstraction we have here # NOTE: this is abusing the abstraction we have here
# but this code may be deprecated soon and this exists # but this code may be deprecated soon and this exists
@ -55,14 +56,14 @@ class SimpleSecurityTransport(BaseSecureTransport):
await asyncio.sleep(0) await asyncio.sleep(0)
if incoming != self.key_phrase: if incoming != self.key_phrase:
raise Exception( raise SecurityUpgradeFailure(
"Key phrase differed between nodes. Expected " + self.key_phrase "Key phrase differed between nodes. Expected " + self.key_phrase
) )
session = InsecureSession(self, conn, peer_id) session = InsecureSession(self, conn, peer_id)
# TODO: Calls handshake to make them know the peer id each other, otherwise tests fail. # NOTE: Here we calls `run_handshake` for both sides to exchange their public keys and
# However, it seems pretty weird that `SimpleSecurityTransport` sends peer id through # peer ids, otherwise tests fail. However, it seems pretty weird that
# `Insecure`. # `SimpleSecurityTransport` sends peer id through `Insecure`.
await session.run_handshake() await session.run_handshake()
# NOTE: this is abusing the abstraction we have here # NOTE: this is abusing the abstraction we have here
# but this code may be deprecated soon and this exists # but this code may be deprecated soon and this exists

View File

@ -0,0 +1,7 @@
# TODO: Add `BaseLibp2pError` and `UpgradeFailure` can inherit from it?
class UpgradeFailure(Exception):
pass
class SecurityUpgradeFailure(UpgradeFailure):
pass