mirror of
https://github.com/fluencelabs/rust-libp2p
synced 2025-06-27 16:51:34 +00:00
Add libp2p-request-response protocol. (#1596)
* Add the libp2p-request-response protocol. This crate provides a generic implementation for request/response protocols, whereby each request is sent on a new substream. * Fix OneShotHandler usage in floodsub. * Custom ProtocolsHandler and multiple protocols. 1. Implement a custom ProtocolsHandler instead of using the OneShotHandler for better control and error handling. In particular, all request/response sending/receiving is kept in the substreams upgrades and thus the background task of a connection. 2. Support multiple protocols (usually protocol versions) with a single `RequestResponse` instance, with configurable inbound/outbound support. * Small doc clarification. * Remove unnecessary Sync bounds. * Remove redundant Clone constraint. * Update protocols/request-response/Cargo.toml Co-authored-by: Toralf Wittner <tw@dtex.org> * Update dev-dependencies. * Update Cargo.tomls. * Add changelog. * Remove Sync bound from RequestResponseCodec::Protocol. Apparently the compiler just needs some help with the scope of borrows, which is unfortunate. * Try async-trait. * Allow checking whether a ResponseChannel is still open. Also expand the commentary on `send_response` to indicate that responses may be discard if they come in too late. * Add `RequestResponse::is_pending`. As an analogue of `ResponseChannel::is_open` for outbound requests. * Revert now unnecessary changes to the OneShotHandler. Since `libp2p-request-response` is no longer using it. * Update CHANGELOG for libp2p-swarm. Co-authored-by: Toralf Wittner <tw@dtex.org>
This commit is contained in:
326
protocols/request-response/src/handler.rs
Normal file
326
protocols/request-response/src/handler.rs
Normal file
@ -0,0 +1,326 @@
|
||||
// Copyright 2020 Parity Technologies (UK) Ltd.
|
||||
//
|
||||
// Permission is hereby granted, free of charge, to any person obtaining a
|
||||
// copy of this software and associated documentation files (the "Software"),
|
||||
// to deal in the Software without restriction, including without limitation
|
||||
// the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
||||
// and/or sell copies of the Software, and to permit persons to whom the
|
||||
// Software is furnished to do so, subject to the following conditions:
|
||||
//
|
||||
// The above copyright notice and this permission notice shall be included in
|
||||
// all copies or substantial portions of the Software.
|
||||
//
|
||||
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
||||
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
||||
// DEALINGS IN THE SOFTWARE.
|
||||
|
||||
mod protocol;
|
||||
|
||||
use crate::{EMPTY_QUEUE_SHRINK_THRESHOLD, RequestId};
|
||||
use crate::codec::RequestResponseCodec;
|
||||
|
||||
pub use protocol::{RequestProtocol, ResponseProtocol, ProtocolSupport};
|
||||
|
||||
use futures::{
|
||||
channel::oneshot,
|
||||
future::BoxFuture,
|
||||
prelude::*,
|
||||
stream::FuturesUnordered
|
||||
};
|
||||
use libp2p_core::{
|
||||
upgrade::{UpgradeError, NegotiationError},
|
||||
};
|
||||
use libp2p_swarm::{
|
||||
SubstreamProtocol,
|
||||
protocols_handler::{
|
||||
KeepAlive,
|
||||
ProtocolsHandler,
|
||||
ProtocolsHandlerEvent,
|
||||
ProtocolsHandlerUpgrErr,
|
||||
}
|
||||
};
|
||||
use smallvec::SmallVec;
|
||||
use std::{
|
||||
collections::VecDeque,
|
||||
io,
|
||||
time::Duration,
|
||||
task::{Context, Poll}
|
||||
};
|
||||
use wasm_timer::Instant;
|
||||
|
||||
/// A connection handler of a `RequestResponse` protocol.
|
||||
#[doc(hidden)]
|
||||
pub struct RequestResponseHandler<TCodec>
|
||||
where
|
||||
TCodec: RequestResponseCodec,
|
||||
{
|
||||
/// The supported inbound protocols.
|
||||
inbound_protocols: SmallVec<[TCodec::Protocol; 2]>,
|
||||
/// The request/response message codec.
|
||||
codec: TCodec,
|
||||
/// The keep-alive timeout of idle connections. A connection is considered
|
||||
/// idle if there are no outbound substreams.
|
||||
keep_alive_timeout: Duration,
|
||||
/// The timeout for inbound and outbound substreams (i.e. request
|
||||
/// and response processing).
|
||||
substream_timeout: Duration,
|
||||
/// The current connection keep-alive.
|
||||
keep_alive: KeepAlive,
|
||||
/// A pending fatal error that results in the connection being closed.
|
||||
pending_error: Option<ProtocolsHandlerUpgrErr<io::Error>>,
|
||||
/// Queue of events to emit in `poll()`.
|
||||
pending_events: VecDeque<RequestResponseHandlerEvent<TCodec>>,
|
||||
/// Outbound upgrades waiting to be emitted as an `OutboundSubstreamRequest`.
|
||||
outbound: VecDeque<RequestProtocol<TCodec>>,
|
||||
/// Inbound upgrades waiting for the incoming request.
|
||||
inbound: FuturesUnordered<BoxFuture<'static,
|
||||
Result<
|
||||
(TCodec::Request, oneshot::Sender<TCodec::Response>),
|
||||
oneshot::Canceled
|
||||
>>>,
|
||||
}
|
||||
|
||||
impl<TCodec> RequestResponseHandler<TCodec>
|
||||
where
|
||||
TCodec: RequestResponseCodec,
|
||||
{
|
||||
pub(super) fn new(
|
||||
inbound_protocols: SmallVec<[TCodec::Protocol; 2]>,
|
||||
codec: TCodec,
|
||||
keep_alive_timeout: Duration,
|
||||
substream_timeout: Duration,
|
||||
) -> Self {
|
||||
Self {
|
||||
inbound_protocols,
|
||||
codec,
|
||||
keep_alive: KeepAlive::Yes,
|
||||
keep_alive_timeout,
|
||||
substream_timeout,
|
||||
outbound: VecDeque::new(),
|
||||
inbound: FuturesUnordered::new(),
|
||||
pending_events: VecDeque::new(),
|
||||
pending_error: None,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// The events emitted by the [`RequestResponseHandler`].
|
||||
#[doc(hidden)]
|
||||
pub enum RequestResponseHandlerEvent<TCodec>
|
||||
where
|
||||
TCodec: RequestResponseCodec
|
||||
{
|
||||
/// An inbound request.
|
||||
Request {
|
||||
request: TCodec::Request,
|
||||
sender: oneshot::Sender<TCodec::Response>
|
||||
},
|
||||
/// An inbound response.
|
||||
Response {
|
||||
request_id: RequestId,
|
||||
response: TCodec::Response
|
||||
},
|
||||
/// An outbound upgrade (i.e. request) timed out.
|
||||
OutboundTimeout(RequestId),
|
||||
/// An outbound request failed to negotiate a mutually supported protocol.
|
||||
OutboundUnsupportedProtocols(RequestId),
|
||||
/// An inbound request timed out.
|
||||
InboundTimeout,
|
||||
/// An inbound request failed to negotiate a mutually supported protocol.
|
||||
InboundUnsupportedProtocols,
|
||||
}
|
||||
|
||||
impl<TCodec> ProtocolsHandler for RequestResponseHandler<TCodec>
|
||||
where
|
||||
TCodec: RequestResponseCodec + Send + Clone + 'static,
|
||||
{
|
||||
type InEvent = RequestProtocol<TCodec>;
|
||||
type OutEvent = RequestResponseHandlerEvent<TCodec>;
|
||||
type Error = ProtocolsHandlerUpgrErr<io::Error>;
|
||||
type InboundProtocol = ResponseProtocol<TCodec>;
|
||||
type OutboundProtocol = RequestProtocol<TCodec>;
|
||||
type OutboundOpenInfo = RequestId;
|
||||
|
||||
fn listen_protocol(&self) -> SubstreamProtocol<Self::InboundProtocol> {
|
||||
// A channel for notifying the handler when the inbound
|
||||
// upgrade received the request.
|
||||
let (rq_send, rq_recv) = oneshot::channel();
|
||||
|
||||
// A channel for notifying the inbound upgrade when the
|
||||
// response is sent.
|
||||
let (rs_send, rs_recv) = oneshot::channel();
|
||||
|
||||
// By keeping all I/O inside the `ResponseProtocol` and thus the
|
||||
// inbound substream upgrade via above channels, we ensure that it
|
||||
// is all subject to the configured timeout without extra bookkeeping
|
||||
// for inbound substreams as well as their timeouts and also make the
|
||||
// implementation of inbound and outbound upgrades symmetric in
|
||||
// this sense.
|
||||
let proto = ResponseProtocol {
|
||||
protocols: self.inbound_protocols.clone(),
|
||||
codec: self.codec.clone(),
|
||||
request_sender: rq_send,
|
||||
response_receiver: rs_recv,
|
||||
};
|
||||
|
||||
// The handler waits for the request to come in. It then emits
|
||||
// `RequestResponseHandlerEvent::Request` together with a
|
||||
// `ResponseChannel`.
|
||||
self.inbound.push(rq_recv.map_ok(move |rq| (rq, rs_send)).boxed());
|
||||
|
||||
SubstreamProtocol::new(proto).with_timeout(self.substream_timeout)
|
||||
}
|
||||
|
||||
fn inject_fully_negotiated_inbound(
|
||||
&mut self,
|
||||
(): (),
|
||||
) {
|
||||
// Nothing to do, as the response has already been sent
|
||||
// as part of the upgrade.
|
||||
}
|
||||
|
||||
fn inject_fully_negotiated_outbound(
|
||||
&mut self,
|
||||
response: TCodec::Response,
|
||||
request_id: RequestId,
|
||||
) {
|
||||
self.pending_events.push_back(
|
||||
RequestResponseHandlerEvent::Response {
|
||||
request_id, response
|
||||
});
|
||||
}
|
||||
|
||||
fn inject_event(&mut self, request: Self::InEvent) {
|
||||
self.keep_alive = KeepAlive::Yes;
|
||||
self.outbound.push_back(request);
|
||||
}
|
||||
|
||||
fn inject_dial_upgrade_error(
|
||||
&mut self,
|
||||
info: RequestId,
|
||||
error: ProtocolsHandlerUpgrErr<io::Error>,
|
||||
) {
|
||||
match error {
|
||||
ProtocolsHandlerUpgrErr::Timeout => {
|
||||
self.pending_events.push_back(
|
||||
RequestResponseHandlerEvent::OutboundTimeout(info));
|
||||
}
|
||||
ProtocolsHandlerUpgrErr::Upgrade(UpgradeError::Select(NegotiationError::Failed)) => {
|
||||
// The remote merely doesn't support the protocol(s) we requested.
|
||||
// This is no reason to close the connection, which may
|
||||
// successfully communicate with other protocols already.
|
||||
// An event is reported to permit user code to react to the fact that
|
||||
// the remote peer does not support the requested protocol(s).
|
||||
self.pending_events.push_back(
|
||||
RequestResponseHandlerEvent::OutboundUnsupportedProtocols(info));
|
||||
}
|
||||
_ => {
|
||||
// Anything else is considered a fatal error or misbehaviour of
|
||||
// the remote peer and results in closing the connection.
|
||||
self.pending_error = Some(error);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fn inject_listen_upgrade_error(
|
||||
&mut self,
|
||||
error: ProtocolsHandlerUpgrErr<io::Error>
|
||||
) {
|
||||
match error {
|
||||
ProtocolsHandlerUpgrErr::Timeout => {
|
||||
self.pending_events.push_back(
|
||||
RequestResponseHandlerEvent::InboundTimeout);
|
||||
}
|
||||
ProtocolsHandlerUpgrErr::Upgrade(UpgradeError::Select(NegotiationError::Failed)) => {
|
||||
// The local peer merely doesn't support the protocol(s) requested.
|
||||
// This is no reason to close the connection, which may
|
||||
// successfully communicate with other protocols already.
|
||||
// An event is reported to permit user code to react to the fact that
|
||||
// the local peer does not support the requested protocol(s).
|
||||
self.pending_events.push_back(
|
||||
RequestResponseHandlerEvent::InboundUnsupportedProtocols);
|
||||
}
|
||||
_ => {
|
||||
// Anything else is considered a fatal error or misbehaviour of
|
||||
// the remote peer and results in closing the connection.
|
||||
self.pending_error = Some(error);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fn connection_keep_alive(&self) -> KeepAlive {
|
||||
self.keep_alive
|
||||
}
|
||||
|
||||
fn poll(
|
||||
&mut self,
|
||||
cx: &mut Context,
|
||||
) -> Poll<
|
||||
ProtocolsHandlerEvent<RequestProtocol<TCodec>, RequestId, Self::OutEvent, Self::Error>,
|
||||
> {
|
||||
// Check for a pending (fatal) error.
|
||||
if let Some(err) = self.pending_error.take() {
|
||||
// The handler will not be polled again by the `Swarm`.
|
||||
return Poll::Ready(ProtocolsHandlerEvent::Close(err))
|
||||
}
|
||||
|
||||
// Drain pending events.
|
||||
if let Some(event) = self.pending_events.pop_front() {
|
||||
return Poll::Ready(ProtocolsHandlerEvent::Custom(event))
|
||||
} else if self.pending_events.capacity() > EMPTY_QUEUE_SHRINK_THRESHOLD {
|
||||
self.pending_events.shrink_to_fit();
|
||||
}
|
||||
|
||||
// Check for inbound requests.
|
||||
while let Poll::Ready(Some(result)) = self.inbound.poll_next_unpin(cx) {
|
||||
match result {
|
||||
Ok((rq, rs_sender)) => {
|
||||
// We received an inbound request.
|
||||
self.keep_alive = KeepAlive::Yes;
|
||||
return Poll::Ready(ProtocolsHandlerEvent::Custom(
|
||||
RequestResponseHandlerEvent::Request {
|
||||
request: rq, sender: rs_sender
|
||||
}))
|
||||
}
|
||||
Err(oneshot::Canceled) => {
|
||||
// The inbound upgrade has errored or timed out reading
|
||||
// or waiting for the request. The handler is informed
|
||||
// via `inject_listen_upgrade_error`.
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Emit outbound requests.
|
||||
if let Some(request) = self.outbound.pop_front() {
|
||||
let info = request.request_id;
|
||||
return Poll::Ready(
|
||||
ProtocolsHandlerEvent::OutboundSubstreamRequest {
|
||||
protocol: SubstreamProtocol::new(request)
|
||||
.with_timeout(self.substream_timeout),
|
||||
info,
|
||||
},
|
||||
)
|
||||
}
|
||||
|
||||
debug_assert!(self.outbound.is_empty());
|
||||
|
||||
if self.outbound.capacity() > EMPTY_QUEUE_SHRINK_THRESHOLD {
|
||||
self.outbound.shrink_to_fit();
|
||||
}
|
||||
|
||||
if self.inbound.is_empty() {
|
||||
// No new inbound or outbound requests. However, we may just have
|
||||
// started the latest inbound or outbound upgrade(s), so make sure
|
||||
// the keep-alive timeout is preceded by the substream timeout.
|
||||
let until = Instant::now() + self.substream_timeout + self.keep_alive_timeout;
|
||||
self.keep_alive = KeepAlive::Until(until);
|
||||
}
|
||||
|
||||
Poll::Pending
|
||||
}
|
||||
}
|
||||
|
Reference in New Issue
Block a user