From adf31c8adda33fb48f40e36b79e27e85a7d9d9fa Mon Sep 17 00:00:00 2001 From: Max Inden Date: Mon, 9 Nov 2020 15:03:51 +0100 Subject: [PATCH] misc/multistream-select: Fail negotiation for all Poll::Ready(None) --- misc/multistream-select/src/dialer_select.rs | 16 +++++++------ .../multistream-select/src/listener_select.rs | 23 +++++++++++-------- 2 files changed, 22 insertions(+), 17 deletions(-) diff --git a/misc/multistream-select/src/dialer_select.rs b/misc/multistream-select/src/dialer_select.rs index 18509e942b7..38111d99c25 100644 --- a/misc/multistream-select/src/dialer_select.rs +++ b/misc/multistream-select/src/dialer_select.rs @@ -24,7 +24,7 @@ use crate::{Negotiated, NegotiationError}; use crate::protocol::{Protocol, ProtocolError, MessageIO, Message, Version}; use futures::{future::Either, prelude::*}; -use std::{convert::TryFrom as _, io, iter, mem, pin::Pin, task::{Context, Poll}}; +use std::{convert::TryFrom as _, iter, mem, pin::Pin, task::{Context, Poll}}; /// Returns a `Future` that negotiates a protocol on the given I/O stream /// for a peer acting as the _dialer_ (or _initiator_). @@ -235,9 +235,10 @@ where *this.state = SeqState::AwaitProtocol { io, protocol }; return Poll::Pending } - Poll::Ready(None) => - return Poll::Ready(Err(NegotiationError::from( - io::Error::from(io::ErrorKind::UnexpectedEof)))), + // Treat EOF error as [`NegotiationError::Failed`], not as + // [`NegotiationError::ProtocolError`], allowing dropping or closing an I/O + // stream as a permissible way to "gracefully" fail a negotiation. + Poll::Ready(None) => return Poll::Ready(Err(NegotiationError::Failed)), }; match msg { @@ -358,9 +359,10 @@ where *this.state = ParState::RecvProtocols { io }; return Poll::Pending } - Poll::Ready(None) => - return Poll::Ready(Err(NegotiationError::from( - io::Error::from(io::ErrorKind::UnexpectedEof)))), + // Treat EOF error as [`NegotiationError::Failed`], not as + // [`NegotiationError::ProtocolError`], allowing dropping or closing an I/O + // stream as a permissible way to "gracefully" fail a negotiation. + Poll::Ready(None) => return Poll::Ready(Err(NegotiationError::Failed)), }; match &msg { diff --git a/misc/multistream-select/src/listener_select.rs b/misc/multistream-select/src/listener_select.rs index 2d8989ad3b0..04f23228369 100644 --- a/misc/multistream-select/src/listener_select.rs +++ b/misc/multistream-select/src/listener_select.rs @@ -26,7 +26,7 @@ use crate::protocol::{Protocol, ProtocolError, MessageIO, Message, Version}; use futures::prelude::*; use smallvec::SmallVec; -use std::{convert::TryFrom as _, io, iter::FromIterator, mem, pin::Pin, task::{Context, Poll}}; +use std::{convert::TryFrom as _, iter::FromIterator, mem, pin::Pin, task::{Context, Poll}}; /// Returns a `Future` that negotiates a protocol on the given I/O stream /// for a peer acting as the _listener_ (or _responder_). @@ -118,10 +118,10 @@ where return Poll::Ready(Err(ProtocolError::InvalidMessage.into())) }, Poll::Ready(Some(Err(err))) => return Poll::Ready(Err(From::from(err))), - Poll::Ready(None) => - return Poll::Ready(Err(NegotiationError::from( - ProtocolError::IoError( - io::ErrorKind::UnexpectedEof.into())))), + // Treat EOF error as [`NegotiationError::Failed`], not as + // [`NegotiationError::ProtocolError`], allowing dropping or closing an I/O + // stream as a permissible way to "gracefully" fail a negotiation. + Poll::Ready(None) => return Poll::Ready(Err(NegotiationError::Failed)), Poll::Pending => { *this.state = State::RecvHeader { io }; return Poll::Pending @@ -153,11 +153,14 @@ where let msg = match Pin::new(&mut io).poll_next(cx) { Poll::Ready(Some(Ok(msg))) => msg, // Treat EOF error as [`NegotiationError::Failed`], not as - // [`NegotiationError::ProtocolError`]. This is e.g. important when a - // listener rejects a protocol with [`Message::NotAvailable`] and the dialer - // does not have alternative protocols to propose. Then the dialer will stop - // the negotiation and drop the corresponding stream. As a listener this EOF - // should be interpreted as a failed negotiation. + // [`NegotiationError::ProtocolError`], allowing dropping or closing an I/O + // stream as a permissible way to "gracefully" fail a negotiation. + // + // This is e.g. important when a listener rejects a protocol with + // [`Message::NotAvailable`] and the dialer does not have alternative + // protocols to propose. Then the dialer will stop the negotiation and drop + // the corresponding stream. As a listener this EOF should be interpreted as + // a failed negotiation. Poll::Ready(None) => return Poll::Ready(Err(NegotiationError::Failed)), Poll::Pending => { *this.state = State::RecvMessage { io };