2020-05-27 15:58:57 +00:00
//!
//!
//!
//! (cd network/examples/async_recv && RUST_BACKTRACE=1 cargo run)
2020-01-13 16:53:28 +00:00
use crate ::{
2020-10-16 09:21:18 +00:00
message ::{ partial_eq_bincode , IncomingMessage , Message , OutgoingMessage } ,
2020-07-14 22:18:04 +00:00
participant ::{ A2bStreamOpen , S2bShutdownBparticipant } ,
2020-03-22 13:47:21 +00:00
scheduler ::Scheduler ,
2020-05-15 12:29:17 +00:00
types ::{ Mid , Pid , Prio , Promises , Sid } ,
2020-01-13 16:53:28 +00:00
} ;
2020-08-23 19:43:17 +00:00
use async_std ::{ io , sync ::Mutex , task } ;
2020-03-22 13:47:21 +00:00
use futures ::{
channel ::{ mpsc , oneshot } ,
sink ::SinkExt ,
stream ::StreamExt ,
2019-12-20 13:56:01 +00:00
} ;
2020-08-25 13:32:42 +00:00
#[ cfg(feature = " compression " ) ]
use lz_fear ::raw ::DecodeError ;
2020-07-14 23:34:41 +00:00
#[ cfg(feature = " metrics " ) ]
2020-04-24 10:56:04 +00:00
use prometheus ::Registry ;
2020-03-22 13:47:21 +00:00
use serde ::{ de ::DeserializeOwned , Serialize } ;
2020-02-04 15:42:04 +00:00
use std ::{
collections ::HashMap ,
2020-05-26 13:06:03 +00:00
net ::SocketAddr ,
2020-03-22 13:47:21 +00:00
sync ::{
2020-05-22 14:00:08 +00:00
atomic ::{ AtomicBool , Ordering } ,
2020-03-22 13:47:21 +00:00
Arc ,
} ,
2020-02-04 15:42:04 +00:00
} ;
2020-01-13 16:53:28 +00:00
use tracing ::* ;
2020-03-22 13:47:21 +00:00
use tracing_futures ::Instrument ;
2019-12-20 13:56:01 +00:00
2020-07-14 22:18:04 +00:00
type A2sDisconnect = Arc < Mutex < Option < mpsc ::UnboundedSender < ( Pid , S2bShutdownBparticipant ) > > > > ;
2020-07-09 07:58:21 +00:00
2020-05-10 02:07:46 +00:00
/// Represents a Tcp or Udp or Mpsc address
2020-03-22 13:47:21 +00:00
#[ derive(Clone, Debug, Hash, PartialEq, Eq) ]
2020-07-09 11:42:38 +00:00
pub enum ProtocolAddr {
2020-05-26 13:06:03 +00:00
Tcp ( SocketAddr ) ,
Udp ( SocketAddr ) ,
2020-03-10 00:07:36 +00:00
Mpsc ( u64 ) ,
2019-12-20 13:56:01 +00:00
}
2020-05-10 02:07:46 +00:00
/// `Participants` are generated by the [`Network`] and represent a connection
/// to a remote Participant. Look at the [`connect`] and [`connected`] method of
/// [`Networks`] on how to generate `Participants`
///
/// [`Networks`]: crate::api::Network
/// [`connect`]: Network::connect
/// [`connected`]: Network::connected
2019-12-20 13:56:01 +00:00
pub struct Participant {
2020-03-22 13:47:21 +00:00
local_pid : Pid ,
2020-02-10 17:25:47 +00:00
remote_pid : Pid ,
2020-08-23 19:43:17 +00:00
a2b_stream_open_s : Mutex < mpsc ::UnboundedSender < A2bStreamOpen > > ,
b2a_stream_opened_r : Mutex < mpsc ::UnboundedReceiver < Stream > > ,
2020-07-14 22:18:04 +00:00
a2s_disconnect_s : A2sDisconnect ,
2019-12-20 13:56:01 +00:00
}
2020-05-10 02:07:46 +00:00
/// `Streams` represents a channel to send `n` messages with a certain priority
/// and [`Promises`]. messages need always to be send between 2 `Streams`.
///
/// `Streams` are generated by the [`Participant`].
/// Look at the [`open`] and [`opened`] method of [`Participant`] on how to
/// generate `Streams`
///
/// Unlike [`Network`] and [`Participant`], `Streams` don't implement interior
/// mutability, as multiple threads don't need access to the same `Stream`.
///
/// [`Networks`]: crate::api::Network
/// [`open`]: Participant::open
/// [`opened`]: Participant::opened
2020-03-22 13:47:21 +00:00
#[ derive(Debug) ]
2020-02-04 15:42:04 +00:00
pub struct Stream {
2020-03-22 13:47:21 +00:00
pid : Pid ,
2020-02-04 15:42:04 +00:00
sid : Sid ,
2020-03-22 13:47:21 +00:00
mid : Mid ,
prio : Prio ,
promises : Promises ,
2020-07-16 19:39:33 +00:00
send_closed : Arc < AtomicBool > ,
2020-07-03 19:55:00 +00:00
a2b_msg_s : crossbeam_channel ::Sender < ( Prio , Sid , OutgoingMessage ) > ,
2020-10-16 09:21:18 +00:00
b2a_msg_recv_r : Option < mpsc ::UnboundedReceiver < IncomingMessage > > ,
2020-05-15 12:29:17 +00:00
a2b_close_stream_s : Option < mpsc ::UnboundedSender < Sid > > ,
2020-02-04 15:42:04 +00:00
}
2019-12-20 13:56:01 +00:00
2020-05-10 02:07:46 +00:00
/// Error type thrown by [`Networks`](Network) methods
2020-04-08 14:26:42 +00:00
#[ derive(Debug) ]
pub enum NetworkError {
NetworkClosed ,
ListenFailed ( std ::io ::Error ) ,
2020-07-04 10:17:33 +00:00
ConnectFailed ( std ::io ::Error ) ,
2020-04-08 14:26:42 +00:00
}
2020-03-22 13:47:21 +00:00
2020-05-10 02:07:46 +00:00
/// Error type thrown by [`Participants`](Participant) methods
2020-07-11 12:34:01 +00:00
#[ derive(Debug, PartialEq, Clone) ]
2020-04-08 14:26:42 +00:00
pub enum ParticipantError {
2020-07-11 12:34:01 +00:00
///Participant was closed by remote side
ParticipantDisconnected ,
///Underlying Protocol failed and wasn't able to recover, expect some Data
/// loss unfortunately, there is no method to get the exact messages
/// that failed. This is also returned when local side tries to do
/// something while remote site gracefully disconnects
ProtocolFailedUnrecoverable ,
2020-04-08 14:26:42 +00:00
}
2020-03-22 13:47:21 +00:00
2020-05-10 02:07:46 +00:00
/// Error type thrown by [`Streams`](Stream) methods
2020-08-25 13:32:42 +00:00
/// A Compression Error should only happen if a client sends malicious code.
/// A Deserialize Error probably means you are expecting Type X while you
/// actually got send type Y.
2020-06-08 09:47:39 +00:00
#[ derive(Debug) ]
2020-04-08 14:26:42 +00:00
pub enum StreamError {
StreamClosed ,
2020-08-25 13:32:42 +00:00
#[ cfg(feature = " compression " ) ]
Compression ( DecodeError ) ,
Deserialize ( bincode ::Error ) ,
2020-04-08 14:26:42 +00:00
}
2020-03-22 13:47:21 +00:00
2020-05-10 02:07:46 +00:00
/// Use the `Network` to create connections to other [`Participants`]
///
/// The `Network` is the single source that handles all connections in your
/// Application. You can pass it around multiple threads in an
/// [`Arc`](std::sync::Arc) as all commands have internal mutability.
///
2020-07-09 07:58:21 +00:00
/// The `Network` has methods to [`connect`] to other [`Participants`] actively
2020-07-09 11:42:38 +00:00
/// via their [`ProtocolAddr`], or [`listen`] passively for [`connected`]
2020-07-09 07:58:21 +00:00
/// [`Participants`].
2020-05-10 02:07:46 +00:00
///
/// # Examples
/// ```rust
2020-07-09 11:42:38 +00:00
/// use veloren_network::{Network, ProtocolAddr, Pid};
2020-05-26 13:06:03 +00:00
/// use futures::executor::block_on;
2020-05-10 02:07:46 +00:00
///
2020-05-26 13:06:03 +00:00
/// # fn main() -> std::result::Result<(), Box<dyn std::error::Error>> {
2020-05-27 15:58:57 +00:00
/// // Create a Network, listen on port `2999` to accept connections and connect to port `8080` to connect to a (pseudo) database Application
2020-07-14 23:34:41 +00:00
/// let (network, f) = Network::new(Pid::new());
2020-06-08 09:47:39 +00:00
/// std::thread::spawn(f);
2020-05-26 13:06:03 +00:00
/// block_on(async{
/// # //setup pseudo database!
2020-07-14 23:34:41 +00:00
/// # let (database, fd) = Network::new(Pid::new());
2020-06-08 09:47:39 +00:00
/// # std::thread::spawn(fd);
2020-07-09 11:42:38 +00:00
/// # database.listen(ProtocolAddr::Tcp("127.0.0.1:8080".parse().unwrap())).await?;
/// network.listen(ProtocolAddr::Tcp("127.0.0.1:2999".parse().unwrap())).await?;
/// let database = network.connect(ProtocolAddr::Tcp("127.0.0.1:8080".parse().unwrap())).await?;
2020-05-26 13:06:03 +00:00
/// # Ok(())
/// })
/// # }
2020-05-10 02:07:46 +00:00
/// ```
///
/// [`Participants`]: crate::api::Participant
/// [`connect`]: Network::connect
2020-07-09 07:58:21 +00:00
/// [`listen`]: Network::listen
/// [`connected`]: Network::connected
2020-02-21 15:10:55 +00:00
pub struct Network {
2020-03-22 13:47:21 +00:00
local_pid : Pid ,
2020-08-23 19:43:17 +00:00
participant_disconnect_sender : Mutex < HashMap < Pid , A2sDisconnect > > ,
2020-04-08 14:26:42 +00:00
listen_sender :
2020-08-23 19:43:17 +00:00
Mutex < mpsc ::UnboundedSender < ( ProtocolAddr , oneshot ::Sender < async_std ::io ::Result < ( ) > > ) > > ,
2020-04-08 14:26:42 +00:00
connect_sender :
2020-08-23 19:43:17 +00:00
Mutex < mpsc ::UnboundedSender < ( ProtocolAddr , oneshot ::Sender < io ::Result < Participant > > ) > > ,
connected_receiver : Mutex < mpsc ::UnboundedReceiver < Participant > > ,
2020-03-22 13:47:21 +00:00
shutdown_sender : Option < oneshot ::Sender < ( ) > > ,
2019-12-20 13:56:01 +00:00
}
2020-02-21 15:10:55 +00:00
impl Network {
2020-05-10 02:07:46 +00:00
/// Generates a new `Network` to handle all connections in an Application
///
/// # Arguments
/// * `participant_id` - provide it by calling [`Pid::new()`], usually you
/// don't want to reuse a Pid for 2 `Networks`
///
2020-06-08 09:47:39 +00:00
/// # Result
/// * `Self` - returns a `Network` which can be `Send` to multiple areas of
/// your code, including multiple threads. This is the base strct of this
/// crate.
/// * `FnOnce` - you need to run the returning FnOnce exactly once, probably
/// in it's own thread. this is NOT done internally, so that you are free
/// to choose the threadpool implementation of your choice. We recommend
2020-08-25 12:21:25 +00:00
/// using [`ThreadPool`] from [`uvth`] crate. This fn will run the
2020-06-08 09:47:39 +00:00
/// Scheduler to handle all `Network` internals. Additional threads will
/// be allocated on an internal async-aware threadpool
///
2020-05-10 02:07:46 +00:00
/// # Examples
/// ```rust
2020-06-08 09:47:39 +00:00
/// //Example with uvth
2020-05-10 02:07:46 +00:00
/// use uvth::ThreadPoolBuilder;
2020-07-09 11:42:38 +00:00
/// use veloren_network::{Network, Pid, ProtocolAddr};
2020-05-10 02:07:46 +00:00
///
2020-06-08 09:47:39 +00:00
/// let pool = ThreadPoolBuilder::new().build();
2020-07-14 23:34:41 +00:00
/// let (network, f) = Network::new(Pid::new());
2020-06-08 09:47:39 +00:00
/// pool.execute(f);
2020-05-10 02:07:46 +00:00
/// ```
///
2020-06-08 09:47:39 +00:00
/// ```rust
/// //Example with std::thread
2020-07-09 11:42:38 +00:00
/// use veloren_network::{Network, Pid, ProtocolAddr};
2020-06-08 09:47:39 +00:00
///
2020-07-14 23:34:41 +00:00
/// let (network, f) = Network::new(Pid::new());
2020-06-08 09:47:39 +00:00
/// std::thread::spawn(f);
/// ```
///
2020-07-14 23:34:41 +00:00
/// Usually you only create a single `Network` for an application,
2020-06-08 09:47:39 +00:00
/// except when client and server are in the same application, then you
/// will want 2. However there are no technical limitations from
/// creating more.
2020-05-10 02:07:46 +00:00
///
/// [`Pid::new()`]: crate::types::Pid::new
2020-06-08 09:47:39 +00:00
/// [`ThreadPool`]: https://docs.rs/uvth/newest/uvth/struct.ThreadPool.html
/// [`uvth`]: https://docs.rs/uvth
2020-07-14 23:34:41 +00:00
pub fn new ( participant_id : Pid ) -> ( Self , impl std ::ops ::FnOnce ( ) ) {
Self ::internal_new (
participant_id ,
#[ cfg(feature = " metrics " ) ]
None ,
)
}
/// See [`new`]
///
/// # additional Arguments
2020-08-25 12:21:25 +00:00
/// * `registry` - Provide a Registry in order to collect Prometheus metrics
2020-07-14 23:34:41 +00:00
/// by this `Network`, `None` will deactivate Tracing. Tracing is done via
/// [`prometheus`]
///
/// # Examples
/// ```rust
/// use prometheus::Registry;
/// use veloren_network::{Network, Pid, ProtocolAddr};
///
/// let registry = Registry::new();
/// let (network, f) = Network::new_with_registry(Pid::new(), ®istry);
/// std::thread::spawn(f);
/// ```
/// [`new`]: crate::api::Network::new
#[ cfg(feature = " metrics " ) ]
pub fn new_with_registry (
participant_id : Pid ,
registry : & Registry ,
) -> ( Self , impl std ::ops ::FnOnce ( ) ) {
Self ::internal_new ( participant_id , Some ( registry ) )
}
fn internal_new (
2020-06-08 09:47:39 +00:00
participant_id : Pid ,
2020-07-14 23:34:41 +00:00
#[ cfg(feature = " metrics " ) ] registry : Option < & Registry > ,
2020-06-08 09:47:39 +00:00
) -> ( Self , impl std ::ops ::FnOnce ( ) ) {
2020-03-22 13:47:21 +00:00
let p = participant_id ;
2020-07-05 22:13:53 +00:00
debug! ( ? p , " Starting Network " ) ;
2020-03-22 13:47:21 +00:00
let ( scheduler , listen_sender , connect_sender , connected_receiver , shutdown_sender ) =
2020-07-14 23:34:41 +00:00
Scheduler ::new (
participant_id ,
#[ cfg(feature = " metrics " ) ]
registry ,
) ;
2020-06-08 09:47:39 +00:00
(
Self {
local_pid : participant_id ,
2020-08-23 19:43:17 +00:00
participant_disconnect_sender : Mutex ::new ( HashMap ::new ( ) ) ,
listen_sender : Mutex ::new ( listen_sender ) ,
connect_sender : Mutex ::new ( connect_sender ) ,
connected_receiver : Mutex ::new ( connected_receiver ) ,
2020-06-08 09:47:39 +00:00
shutdown_sender : Some ( shutdown_sender ) ,
} ,
move | | {
2020-07-14 23:34:41 +00:00
trace! ( ? p , " Starting scheduler in own thread " ) ;
2020-06-08 09:47:39 +00:00
let _handle = task ::block_on (
scheduler
. run ( )
. instrument ( tracing ::info_span! ( " scheduler " , ? p ) ) ,
) ;
2020-07-14 23:34:41 +00:00
trace! ( ? p , " Stopping scheduler and his own thread " ) ;
2020-06-08 09:47:39 +00:00
} ,
)
2019-12-20 13:56:01 +00:00
}
2020-07-09 11:42:38 +00:00
/// starts listening on an [`ProtocolAddr`].
2020-05-10 02:07:46 +00:00
/// When the method returns the `Network` is ready to listen for incoming
/// connections OR has returned a [`NetworkError`] (e.g. port already used).
/// You can call [`connected`] to asynchrony wait for a [`Participant`] to
/// connect. You can call `listen` on multiple addresses, e.g. to
/// support multiple Protocols or NICs.
///
/// # Examples
/// ```rust
2020-05-26 13:06:03 +00:00
/// use futures::executor::block_on;
2020-07-09 11:42:38 +00:00
/// use veloren_network::{Network, Pid, ProtocolAddr};
2020-05-10 02:07:46 +00:00
///
2020-05-26 13:06:03 +00:00
/// # fn main() -> std::result::Result<(), Box<dyn std::error::Error>> {
2020-05-10 02:07:46 +00:00
/// // Create a Network, listen on port `2000` TCP on all NICs and `2001` UDP locally
2020-07-14 23:34:41 +00:00
/// let (network, f) = Network::new(Pid::new());
2020-06-08 09:47:39 +00:00
/// std::thread::spawn(f);
2020-05-10 02:07:46 +00:00
/// block_on(async {
/// network
2020-07-09 11:42:38 +00:00
/// .listen(ProtocolAddr::Tcp("0.0.0.0:2000".parse().unwrap()))
2020-05-10 02:07:46 +00:00
/// .await?;
/// network
2020-07-09 11:42:38 +00:00
/// .listen(ProtocolAddr::Udp("127.0.0.1:2001".parse().unwrap()))
2020-05-10 02:07:46 +00:00
/// .await?;
2020-05-26 13:06:03 +00:00
/// # Ok(())
/// })
/// # }
2020-05-10 02:07:46 +00:00
/// ```
///
/// [`connected`]: Network::connected
2020-07-09 11:42:38 +00:00
pub async fn listen ( & self , address : ProtocolAddr ) -> Result < ( ) , NetworkError > {
2020-05-15 12:29:17 +00:00
let ( s2a_result_s , s2a_result_r ) = oneshot ::channel ::< async_std ::io ::Result < ( ) > > ( ) ;
debug! ( ? address , " listening on address " ) ;
2020-04-08 14:26:42 +00:00
self . listen_sender
2020-08-23 19:43:17 +00:00
. lock ( )
2020-04-08 14:26:42 +00:00
. await
2020-05-15 12:29:17 +00:00
. send ( ( address , s2a_result_s ) )
2020-04-08 14:26:42 +00:00
. await ? ;
2020-05-15 12:29:17 +00:00
match s2a_result_r . await ? {
2020-08-25 12:21:25 +00:00
//waiting guarantees that we either listened successfully or get an error like port in
2020-04-08 14:26:42 +00:00
// use
Ok ( ( ) ) = > Ok ( ( ) ) ,
Err ( e ) = > Err ( NetworkError ::ListenFailed ( e ) ) ,
}
2019-12-20 13:56:01 +00:00
}
2020-08-25 12:21:25 +00:00
/// starts connection to an [`ProtocolAddr`].
2020-05-10 02:07:46 +00:00
/// When the method returns the Network either returns a [`Participant`]
/// ready to open [`Streams`] on OR has returned a [`NetworkError`] (e.g.
/// can't connect, or invalid Handshake) # Examples
/// ```rust
2020-05-26 13:06:03 +00:00
/// use futures::executor::block_on;
2020-07-09 11:42:38 +00:00
/// use veloren_network::{Network, Pid, ProtocolAddr};
2020-05-10 02:07:46 +00:00
///
2020-05-26 13:06:03 +00:00
/// # fn main() -> std::result::Result<(), Box<dyn std::error::Error>> {
2020-05-27 15:58:57 +00:00
/// // Create a Network, connect on port `2010` TCP and `2011` UDP like listening above
2020-07-14 23:34:41 +00:00
/// let (network, f) = Network::new(Pid::new());
2020-06-08 09:47:39 +00:00
/// std::thread::spawn(f);
2020-07-14 23:34:41 +00:00
/// # let (remote, fr) = Network::new(Pid::new());
2020-06-08 09:47:39 +00:00
/// # std::thread::spawn(fr);
2020-05-10 02:07:46 +00:00
/// block_on(async {
2020-07-09 11:42:38 +00:00
/// # remote.listen(ProtocolAddr::Tcp("0.0.0.0:2010".parse().unwrap())).await?;
/// # remote.listen(ProtocolAddr::Udp("0.0.0.0:2011".parse().unwrap())).await?;
2020-05-10 02:07:46 +00:00
/// let p1 = network
2020-07-09 11:42:38 +00:00
/// .connect(ProtocolAddr::Tcp("127.0.0.1:2010".parse().unwrap()))
2020-05-10 02:07:46 +00:00
/// .await?;
2020-05-26 13:06:03 +00:00
/// # //this doesn't work yet, so skip the test
/// # //TODO fixme!
/// # return Ok(());
2020-05-10 02:07:46 +00:00
/// let p2 = network
2020-07-09 11:42:38 +00:00
/// .connect(ProtocolAddr::Udp("127.0.0.1:2011".parse().unwrap()))
2020-05-10 02:07:46 +00:00
/// .await?;
2020-07-09 07:58:21 +00:00
/// assert_eq!(&p1, &p2);
2020-05-26 13:06:03 +00:00
/// # Ok(())
/// })
/// # }
2020-05-10 02:07:46 +00:00
/// ```
/// Usually the `Network` guarantees that a operation on a [`Participant`]
/// succeeds, e.g. by automatic retrying unless it fails completely e.g. by
2020-07-09 11:42:38 +00:00
/// disconnecting from the remote. If 2 [`ProtocolAddres`] you `connect` to
2020-05-10 02:07:46 +00:00
/// belongs to the same [`Participant`], you get the same [`Participant`] as
/// a result. This is useful e.g. by connecting to the same
/// [`Participant`] via multiple Protocols.
///
/// [`Streams`]: crate::api::Stream
2020-07-09 11:42:38 +00:00
/// [`ProtocolAddres`]: crate::api::ProtocolAddr
pub async fn connect ( & self , address : ProtocolAddr ) -> Result < Participant , NetworkError > {
2020-04-08 14:26:42 +00:00
let ( pid_sender , pid_receiver ) = oneshot ::channel ::< io ::Result < Participant > > ( ) ;
2020-07-05 22:13:53 +00:00
debug! ( ? address , " Connect to address " ) ;
2020-03-22 13:47:21 +00:00
self . connect_sender
2020-08-23 19:43:17 +00:00
. lock ( )
2020-03-22 13:47:21 +00:00
. await
. send ( ( address , pid_sender ) )
2020-04-08 14:26:42 +00:00
. await ? ;
2020-07-04 10:17:33 +00:00
let participant = match pid_receiver . await ? {
Ok ( p ) = > p ,
Err ( e ) = > return Err ( NetworkError ::ConnectFailed ( e ) ) ,
} ;
2020-04-08 14:26:42 +00:00
let pid = participant . remote_pid ;
debug! (
? pid ,
2020-07-05 22:13:53 +00:00
" Received Participant id from remote and return to user "
2020-04-08 14:26:42 +00:00
) ;
2020-07-09 07:58:21 +00:00
self . participant_disconnect_sender
2020-08-23 19:43:17 +00:00
. lock ( )
2020-03-22 13:47:21 +00:00
. await
2020-09-27 16:20:40 +00:00
. insert ( pid , Arc ::clone ( & participant . a2s_disconnect_s ) ) ;
2020-04-08 14:26:42 +00:00
Ok ( participant )
2020-03-22 13:47:21 +00:00
}
2020-03-10 00:07:36 +00:00
2020-07-09 11:42:38 +00:00
/// returns a [`Participant`] created from a [`ProtocolAddr`] you called
2020-05-10 02:07:46 +00:00
/// [`listen`] on before. This function will either return a working
/// [`Participant`] ready to open [`Streams`] on OR has returned a
/// [`NetworkError`] (e.g. Network got closed)
///
/// # Examples
/// ```rust
2020-05-26 13:06:03 +00:00
/// use futures::executor::block_on;
2020-07-09 11:42:38 +00:00
/// use veloren_network::{Network, Pid, ProtocolAddr};
2020-05-10 02:07:46 +00:00
///
2020-05-26 13:06:03 +00:00
/// # fn main() -> std::result::Result<(), Box<dyn std::error::Error>> {
2020-05-27 15:58:57 +00:00
/// // Create a Network, listen on port `2020` TCP and opens returns their Pid
2020-07-14 23:34:41 +00:00
/// let (network, f) = Network::new(Pid::new());
2020-06-08 09:47:39 +00:00
/// std::thread::spawn(f);
2020-07-14 23:34:41 +00:00
/// # let (remote, fr) = Network::new(Pid::new());
2020-06-08 09:47:39 +00:00
/// # std::thread::spawn(fr);
2020-05-10 02:07:46 +00:00
/// block_on(async {
/// network
2020-07-09 11:42:38 +00:00
/// .listen(ProtocolAddr::Tcp("0.0.0.0:2020".parse().unwrap()))
2020-05-10 02:07:46 +00:00
/// .await?;
2020-07-09 11:42:38 +00:00
/// # remote.connect(ProtocolAddr::Tcp("0.0.0.0:2020".parse().unwrap())).await?;
2020-05-26 13:06:03 +00:00
/// while let Ok(participant) = network.connected().await {
2020-05-10 02:07:46 +00:00
/// println!("Participant connected: {}", participant.remote_pid());
2020-05-26 13:06:03 +00:00
/// # //skip test here as it would be a endless loop
/// # break;
2020-05-10 02:07:46 +00:00
/// }
2020-05-26 13:06:03 +00:00
/// # Ok(())
/// })
/// # }
2020-05-10 02:07:46 +00:00
/// ```
///
/// [`Streams`]: crate::api::Stream
/// [`listen`]: crate::api::Network::listen
2020-07-09 07:58:21 +00:00
pub async fn connected ( & self ) -> Result < Participant , NetworkError > {
2020-08-23 19:43:17 +00:00
let participant = self . connected_receiver . lock ( ) . await . next ( ) . await ? ;
2020-09-27 16:20:40 +00:00
self . participant_disconnect_sender . lock ( ) . await . insert (
participant . remote_pid ,
Arc ::clone ( & participant . a2s_disconnect_s ) ,
) ;
2020-04-08 14:26:42 +00:00
Ok ( participant )
2019-12-20 13:56:01 +00:00
}
2020-04-08 14:26:42 +00:00
}
2020-02-10 17:25:47 +00:00
2020-03-22 13:47:21 +00:00
impl Participant {
pub ( crate ) fn new (
local_pid : Pid ,
remote_pid : Pid ,
2020-07-14 22:18:04 +00:00
a2b_stream_open_s : mpsc ::UnboundedSender < A2bStreamOpen > ,
2020-05-15 12:29:17 +00:00
b2a_stream_opened_r : mpsc ::UnboundedReceiver < Stream > ,
2020-07-14 22:18:04 +00:00
a2s_disconnect_s : mpsc ::UnboundedSender < ( Pid , S2bShutdownBparticipant ) > ,
2020-03-22 13:47:21 +00:00
) -> Self {
Self {
local_pid ,
remote_pid ,
2020-08-23 19:43:17 +00:00
a2b_stream_open_s : Mutex ::new ( a2b_stream_open_s ) ,
b2a_stream_opened_r : Mutex ::new ( b2a_stream_opened_r ) ,
2020-07-10 13:31:26 +00:00
a2s_disconnect_s : Arc ::new ( Mutex ::new ( Some ( a2s_disconnect_s ) ) ) ,
2020-03-04 15:52:30 +00:00
}
2020-02-10 17:25:47 +00:00
}
2020-05-10 02:07:46 +00:00
/// Opens a [`Stream`] on this `Participant` with a certain Priority and
/// [`Promises`]
///
/// # Arguments
/// * `prio` - valid between 0-63. The priority rates the throughput for
/// messages of the [`Stream`] e.g. prio 5 messages will get 1/2 the speed
/// prio0 messages have. Prio10 messages only 1/4 and Prio 15 only 1/8,
/// etc...
/// * `promises` - use a combination of you prefered [`Promises`], see the
/// link for further documentation. You can combine them, e.g.
2020-08-25 13:32:42 +00:00
/// `Promises::ORDERED | Promises::CONSISTENCY` The Stream will then
2020-08-25 12:21:25 +00:00
/// guarantee that those promises are met.
2020-05-10 02:07:46 +00:00
///
/// A [`ParticipantError`] might be thrown if the `Participant` is already
/// closed. [`Streams`] can be created without a answer from the remote
/// side, resulting in very fast creation and closing latency.
///
/// # Examples
/// ```rust
2020-05-26 13:06:03 +00:00
/// use futures::executor::block_on;
2020-08-25 13:32:42 +00:00
/// use veloren_network::{Network, Pid, Promises, ProtocolAddr};
2020-05-10 02:07:46 +00:00
///
2020-05-26 13:06:03 +00:00
/// # fn main() -> std::result::Result<(), Box<dyn std::error::Error>> {
2020-05-27 15:58:57 +00:00
/// // Create a Network, connect on port 2100 and open a stream
2020-07-14 23:34:41 +00:00
/// let (network, f) = Network::new(Pid::new());
2020-06-08 09:47:39 +00:00
/// std::thread::spawn(f);
2020-07-14 23:34:41 +00:00
/// # let (remote, fr) = Network::new(Pid::new());
2020-06-08 09:47:39 +00:00
/// # std::thread::spawn(fr);
2020-05-10 02:07:46 +00:00
/// block_on(async {
2020-07-09 11:42:38 +00:00
/// # remote.listen(ProtocolAddr::Tcp("0.0.0.0:2100".parse().unwrap())).await?;
2020-05-10 02:07:46 +00:00
/// let p1 = network
2020-07-09 11:42:38 +00:00
/// .connect(ProtocolAddr::Tcp("127.0.0.1:2100".parse().unwrap()))
2020-05-10 02:07:46 +00:00
/// .await?;
2020-08-25 13:32:42 +00:00
/// let _s1 = p1
/// .open(16, Promises::ORDERED | Promises::CONSISTENCY)
/// .await?;
2020-05-26 13:06:03 +00:00
/// # Ok(())
/// })
/// # }
2020-05-10 02:07:46 +00:00
/// ```
///
/// [`Streams`]: crate::api::Stream
2020-03-22 13:47:21 +00:00
pub async fn open ( & self , prio : u8 , promises : Promises ) -> Result < Stream , ParticipantError > {
2020-05-15 12:29:17 +00:00
let ( p2a_return_stream_s , p2a_return_stream_r ) = oneshot ::channel ( ) ;
2020-08-23 19:43:17 +00:00
if let Err ( e ) = self
. a2b_stream_open_s
. lock ( )
. await
2020-05-15 12:29:17 +00:00
. send ( ( prio , promises , p2a_return_stream_s ) )
2020-03-22 13:47:21 +00:00
. await
2020-07-16 19:39:33 +00:00
{
debug! ( ? e , " bParticipant is already closed, notifying " ) ;
return Err ( ParticipantError ::ParticipantDisconnected ) ;
}
2020-05-15 12:29:17 +00:00
match p2a_return_stream_r . await {
2020-03-22 13:47:21 +00:00
Ok ( stream ) = > {
let sid = stream . sid ;
2020-05-15 12:29:17 +00:00
debug! ( ? sid , ? self . remote_pid , " opened stream " ) ;
2020-03-22 13:47:21 +00:00
Ok ( stream )
} ,
2020-04-08 14:26:42 +00:00
Err ( _ ) = > {
2020-05-15 12:29:17 +00:00
debug! ( ? self . remote_pid , " p2a_return_stream_r failed, closing participant " ) ;
2020-07-16 19:39:33 +00:00
Err ( ParticipantError ::ParticipantDisconnected )
2020-04-08 14:26:42 +00:00
} ,
2020-03-04 15:52:30 +00:00
}
}
2020-02-21 15:10:55 +00:00
2020-05-10 02:07:46 +00:00
/// Use this method to handle [`Streams`] opened from remote site, like the
/// [`connected`] method of [`Network`]. This is the associated method
/// to [`open`]. It's guaranteed that the order of [`open`] and `opened`
/// is equal. The `nth` [`Streams`] on one side will represent the `nth` on
/// the other side. A [`ParticipantError`] might be thrown if the
/// `Participant` is already closed.
///
/// # Examples
/// ```rust
2020-08-25 13:32:42 +00:00
/// use veloren_network::{Network, Pid, ProtocolAddr, Promises};
2020-05-26 13:06:03 +00:00
/// use futures::executor::block_on;
2020-05-10 02:07:46 +00:00
///
2020-05-26 13:06:03 +00:00
/// # fn main() -> std::result::Result<(), Box<dyn std::error::Error>> {
2020-05-27 15:58:57 +00:00
/// // Create a Network, connect on port 2110 and wait for the other side to open a stream
2020-08-25 12:21:25 +00:00
/// // Note: It's quite unusual to actively connect, but then wait on a stream to be connected, usually the Application taking initiative want's to also create the first Stream.
2020-07-14 23:34:41 +00:00
/// let (network, f) = Network::new(Pid::new());
2020-06-08 09:47:39 +00:00
/// std::thread::spawn(f);
2020-07-14 23:34:41 +00:00
/// # let (remote, fr) = Network::new(Pid::new());
2020-06-08 09:47:39 +00:00
/// # std::thread::spawn(fr);
2020-05-10 02:07:46 +00:00
/// block_on(async {
2020-07-09 11:42:38 +00:00
/// # remote.listen(ProtocolAddr::Tcp("0.0.0.0:2110".parse().unwrap())).await?;
/// let p1 = network.connect(ProtocolAddr::Tcp("127.0.0.1:2110".parse().unwrap())).await?;
2020-05-26 13:06:03 +00:00
/// # let p2 = remote.connected().await?;
2020-08-25 13:32:42 +00:00
/// # p2.open(16, Promises::ORDERED | Promises::CONSISTENCY).await?;
2020-05-10 02:07:46 +00:00
/// let _s1 = p1.opened().await?;
2020-05-26 13:06:03 +00:00
/// # Ok(())
/// })
/// # }
2020-05-10 02:07:46 +00:00
/// ```
///
/// [`Streams`]: crate::api::Stream
/// [`connected`]: Network::connected
/// [`open`]: Participant::open
2020-02-21 15:10:55 +00:00
pub async fn opened ( & self ) -> Result < Stream , ParticipantError > {
2020-08-23 19:43:17 +00:00
match self . b2a_stream_opened_r . lock ( ) . await . next ( ) . await {
2020-04-08 14:26:42 +00:00
Some ( stream ) = > {
let sid = stream . sid ;
2020-07-05 22:13:53 +00:00
debug! ( ? sid , ? self . remote_pid , " Receive opened stream " ) ;
2020-04-08 14:26:42 +00:00
Ok ( stream )
} ,
None = > {
debug! ( ? self . remote_pid , " stream_opened_receiver failed, closing participant " ) ;
2020-07-16 19:39:33 +00:00
Err ( ParticipantError ::ParticipantDisconnected )
2020-04-08 14:26:42 +00:00
} ,
2020-02-21 15:10:55 +00:00
}
2020-02-10 17:25:47 +00:00
}
2020-04-08 14:26:42 +00:00
2020-07-09 07:58:21 +00:00
/// disconnecting a `Participant` in a async way.
/// Use this rather than `Participant::Drop` if you want to close multiple
/// `Participants`.
///
/// This function will wait for all [`Streams`] to properly close, including
/// all messages to be send before closing. If an error occurs with one
/// of the messages.
/// Except if the remote side already dropped the `Participant`
/// simultaneously, then messages won't be send
///
/// There is NO `disconnected` function in `Participant`, if a `Participant`
/// is no longer reachable (e.g. as the network cable was unplugged) the
/// `Participant` will fail all action, but needs to be manually
2020-08-25 12:21:25 +00:00
/// disconnected, using this function.
2020-07-09 07:58:21 +00:00
///
/// # Examples
/// ```rust
/// use futures::executor::block_on;
2020-07-09 11:42:38 +00:00
/// use veloren_network::{Network, Pid, ProtocolAddr};
2020-07-09 07:58:21 +00:00
///
/// # fn main() -> std::result::Result<(), Box<dyn std::error::Error>> {
/// // Create a Network, listen on port `2030` TCP and opens returns their Pid and close connection.
2020-07-14 23:34:41 +00:00
/// let (network, f) = Network::new(Pid::new());
2020-07-09 07:58:21 +00:00
/// std::thread::spawn(f);
2020-07-14 23:34:41 +00:00
/// # let (remote, fr) = Network::new(Pid::new());
2020-07-09 07:58:21 +00:00
/// # std::thread::spawn(fr);
/// block_on(async {
/// network
2020-07-09 11:42:38 +00:00
/// .listen(ProtocolAddr::Tcp("0.0.0.0:2030".parse().unwrap()))
2020-07-09 07:58:21 +00:00
/// .await?;
2020-07-11 12:34:01 +00:00
/// # let keep_alive = remote.connect(ProtocolAddr::Tcp("0.0.0.0:2030".parse().unwrap())).await?;
2020-07-09 07:58:21 +00:00
/// while let Ok(participant) = network.connected().await {
/// println!("Participant connected: {}", participant.remote_pid());
/// participant.disconnect().await?;
/// # //skip test here as it would be a endless loop
/// # break;
/// }
/// # Ok(())
/// })
/// # }
/// ```
///
/// [`Streams`]: crate::api::Stream
pub async fn disconnect ( self ) -> Result < ( ) , ParticipantError > {
// Remove, Close and try_unwrap error when unwrap fails!
let pid = self . remote_pid ;
debug! ( ? pid , " Closing participant from network " ) ;
2020-07-11 12:34:01 +00:00
//Streams will be closed by BParticipant
2020-07-10 13:31:26 +00:00
match self . a2s_disconnect_s . lock ( ) . await . take ( ) {
2020-07-09 07:58:21 +00:00
Some ( mut a2s_disconnect_s ) = > {
let ( finished_sender , finished_receiver ) = oneshot ::channel ( ) ;
// Participant is connecting to Scheduler here, not as usual
// Participant<->BParticipant
a2s_disconnect_s
. send ( ( pid , finished_sender ) )
. await
. expect ( " Something is wrong in internal scheduler coding " ) ;
match finished_receiver . await {
2020-07-16 19:39:33 +00:00
Ok ( res ) = > {
match res {
Ok ( ( ) ) = > trace! ( ? pid , " Participant is now closed " ) ,
Err ( ref e ) = > {
2020-08-25 12:21:25 +00:00
trace! ( ? pid , ? e , " Error occurred during shutdown of participant " )
2020-07-16 19:39:33 +00:00
} ,
} ;
res
2020-07-09 07:58:21 +00:00
} ,
Err ( e ) = > {
//this is a bug. but as i am Participant i can't destroy the network
error! (
? pid ,
? e ,
" Failed to get a message back from the scheduler, seems like the \
network is already closed "
) ;
2020-07-11 12:34:01 +00:00
Err ( ParticipantError ::ProtocolFailedUnrecoverable )
2020-07-09 07:58:21 +00:00
} ,
}
} ,
None = > {
warn! (
" seems like you are trying to disconnecting a participant after the network \
was already dropped . It was already dropped with the network ! "
) ;
2020-07-11 12:34:01 +00:00
Err ( ParticipantError ::ParticipantDisconnected )
2020-07-09 07:58:21 +00:00
} ,
}
}
2020-05-10 02:07:46 +00:00
/// Returns the remote [`Pid`]
2020-04-08 14:26:42 +00:00
pub fn remote_pid ( & self ) -> Pid { self . remote_pid }
2020-02-10 17:25:47 +00:00
}
impl Stream {
2020-06-08 09:47:39 +00:00
#[ allow(clippy::too_many_arguments) ]
2020-03-10 00:07:36 +00:00
pub ( crate ) fn new (
2020-03-22 13:47:21 +00:00
pid : Pid ,
2020-03-10 00:07:36 +00:00
sid : Sid ,
2020-03-22 13:47:21 +00:00
prio : Prio ,
promises : Promises ,
2020-07-16 19:39:33 +00:00
send_closed : Arc < AtomicBool > ,
2020-07-03 19:55:00 +00:00
a2b_msg_s : crossbeam_channel ::Sender < ( Prio , Sid , OutgoingMessage ) > ,
Fixing the DEADLOCK in handshake -> channel creation
- this bug was initially called imbris bug, as it happened on his runners and i couldn't reproduce it locally at fist :)
- When in a Handshake a seperate mpsc::Channel was created for (Cid, Frame) transport
however the protocol could already catch non handshake data any more and push in into this
mpsc::Channel.
Then this channel got dropped and a fresh one was created for the network::Channel.
These droped Frames are ofc a BUG!
I tried multiple things to solve this:
- dont create a new mpsc::Channel, but instead bind it to the Protocol itself and always use 1.
This would work theoretically, but in bParticipant side we are using 1 mpsc::Channel<(Cid, Frame)>
to handle ALL the network::channel.
If now ever Protocol would have it's own, and with that every network::Channel had it's own it would no longer work out
Bad Idea...
- using the first method but creating the mpsc::Channel inside the scheduler instead protocol neither works, as the
scheduler doesnt know the remote_pid yet
- i dont want a hack to say the protocol only listen to 2 messages and then stop no matter what
So i switched over to the simply method now:
- Do everything like before with 2 mpsc::Channels
- after the handshake. close the receiver and listen for all remaining (cid, frame) combinations
- when starting the channel, reapply them to the new sender/listener combination
- added tracing
- switched Protocol RwLock to Mutex, as it's only ever 1
- Additionally changed the layout and introduces the c2w_frame_s and w2s_cid_frame_s name schema
- Fixed a bug in scheduler which WOULD cause a DEADLOCK if handshake would fail
- fixd a but in api_send_send_main, i need to store the stream_p otherwise it's immeadiatly closed and a stream_a.send() isn't guaranteed
- add extra test to verify that a send message is received even if the Stream is already closed
- changed OutGoing to Outgoing
- fixed a bug that `metrics.tick()` was never called
- removed 2 unused nightly features and added `deny_code`
2020-06-03 07:13:00 +00:00
b2a_msg_recv_r : mpsc ::UnboundedReceiver < IncomingMessage > ,
2020-05-15 12:29:17 +00:00
a2b_close_stream_s : mpsc ::UnboundedSender < Sid > ,
2020-03-10 00:07:36 +00:00
) -> Self {
Self {
2020-03-22 13:47:21 +00:00
pid ,
2020-03-10 00:07:36 +00:00
sid ,
2020-03-22 13:47:21 +00:00
mid : 0 ,
prio ,
promises ,
2020-07-16 19:39:33 +00:00
send_closed ,
2020-05-15 12:29:17 +00:00
a2b_msg_s ,
2020-10-16 09:21:18 +00:00
b2a_msg_recv_r : Some ( b2a_msg_recv_r ) ,
2020-05-15 12:29:17 +00:00
a2b_close_stream_s : Some ( a2b_close_stream_s ) ,
2020-03-10 00:07:36 +00:00
}
}
2020-02-10 17:25:47 +00:00
2020-05-10 02:07:46 +00:00
/// use to send a arbitrary message to the remote side, by having the remote
/// side also opened a `Stream` linked to this. the message will be
/// [`Serialized`], which actually is quite slow compared to most other
/// calculations done. A faster method [`send_raw`] exists, when extra
/// speed is needed. The other side needs to use the respective [`recv`]
/// function and know the type send.
///
/// `send` is an exception to the `async` messages, as it's probably called
/// quite often so it doesn't wait for execution. Which also means, that
/// no feedback is provided. It's to assume that the Message got `send`
/// correctly. If a error occurred, the next call will return an Error.
/// If the [`Participant`] disconnected it will also be unable to be used
/// any more. A [`StreamError`] will be returned in the error case, e.g.
/// when the `Stream` got closed already.
///
2020-05-27 15:58:57 +00:00
/// Note when a `Stream` is dropped locally, it will still send all
/// messages, though the `drop` will return immediately, however, when a
/// [`Participant`] gets gracefully shut down, all remaining messages
/// will be send. If the `Stream` is dropped from remote side no further
/// messages are send, because the remote side has no way of listening
/// to them either way. If the last channel is destroyed (e.g. by losing
2020-08-25 12:21:25 +00:00
/// the internet connection or non-graceful shutdown, pending messages
2020-05-27 15:58:57 +00:00
/// are also dropped.
2020-05-10 02:07:46 +00:00
///
/// # Example
Fixing the DEADLOCK in handshake -> channel creation
- this bug was initially called imbris bug, as it happened on his runners and i couldn't reproduce it locally at fist :)
- When in a Handshake a seperate mpsc::Channel was created for (Cid, Frame) transport
however the protocol could already catch non handshake data any more and push in into this
mpsc::Channel.
Then this channel got dropped and a fresh one was created for the network::Channel.
These droped Frames are ofc a BUG!
I tried multiple things to solve this:
- dont create a new mpsc::Channel, but instead bind it to the Protocol itself and always use 1.
This would work theoretically, but in bParticipant side we are using 1 mpsc::Channel<(Cid, Frame)>
to handle ALL the network::channel.
If now ever Protocol would have it's own, and with that every network::Channel had it's own it would no longer work out
Bad Idea...
- using the first method but creating the mpsc::Channel inside the scheduler instead protocol neither works, as the
scheduler doesnt know the remote_pid yet
- i dont want a hack to say the protocol only listen to 2 messages and then stop no matter what
So i switched over to the simply method now:
- Do everything like before with 2 mpsc::Channels
- after the handshake. close the receiver and listen for all remaining (cid, frame) combinations
- when starting the channel, reapply them to the new sender/listener combination
- added tracing
- switched Protocol RwLock to Mutex, as it's only ever 1
- Additionally changed the layout and introduces the c2w_frame_s and w2s_cid_frame_s name schema
- Fixed a bug in scheduler which WOULD cause a DEADLOCK if handshake would fail
- fixd a but in api_send_send_main, i need to store the stream_p otherwise it's immeadiatly closed and a stream_a.send() isn't guaranteed
- add extra test to verify that a send message is received even if the Stream is already closed
- changed OutGoing to Outgoing
- fixed a bug that `metrics.tick()` was never called
- removed 2 unused nightly features and added `deny_code`
2020-06-03 07:13:00 +00:00
/// ```
2020-07-09 11:42:38 +00:00
/// use veloren_network::{Network, ProtocolAddr, Pid};
2020-08-25 13:32:42 +00:00
/// # use veloren_network::Promises;
2020-05-10 02:07:46 +00:00
/// use futures::executor::block_on;
///
2020-05-26 13:06:03 +00:00
/// # fn main() -> std::result::Result<(), Box<dyn std::error::Error>> {
2020-05-27 15:58:57 +00:00
/// // Create a Network, listen on Port `2200` and wait for a Stream to be opened, then answer `Hello World`
2020-07-14 23:34:41 +00:00
/// let (network, f) = Network::new(Pid::new());
2020-06-08 09:47:39 +00:00
/// std::thread::spawn(f);
2020-07-14 23:34:41 +00:00
/// # let (remote, fr) = Network::new(Pid::new());
2020-06-08 09:47:39 +00:00
/// # std::thread::spawn(fr);
2020-05-10 02:07:46 +00:00
/// block_on(async {
2020-07-09 11:42:38 +00:00
/// network.listen(ProtocolAddr::Tcp("127.0.0.1:2200".parse().unwrap())).await?;
/// # let remote_p = remote.connect(ProtocolAddr::Tcp("127.0.0.1:2200".parse().unwrap())).await?;
Fixing the DEADLOCK in handshake -> channel creation
- this bug was initially called imbris bug, as it happened on his runners and i couldn't reproduce it locally at fist :)
- When in a Handshake a seperate mpsc::Channel was created for (Cid, Frame) transport
however the protocol could already catch non handshake data any more and push in into this
mpsc::Channel.
Then this channel got dropped and a fresh one was created for the network::Channel.
These droped Frames are ofc a BUG!
I tried multiple things to solve this:
- dont create a new mpsc::Channel, but instead bind it to the Protocol itself and always use 1.
This would work theoretically, but in bParticipant side we are using 1 mpsc::Channel<(Cid, Frame)>
to handle ALL the network::channel.
If now ever Protocol would have it's own, and with that every network::Channel had it's own it would no longer work out
Bad Idea...
- using the first method but creating the mpsc::Channel inside the scheduler instead protocol neither works, as the
scheduler doesnt know the remote_pid yet
- i dont want a hack to say the protocol only listen to 2 messages and then stop no matter what
So i switched over to the simply method now:
- Do everything like before with 2 mpsc::Channels
- after the handshake. close the receiver and listen for all remaining (cid, frame) combinations
- when starting the channel, reapply them to the new sender/listener combination
- added tracing
- switched Protocol RwLock to Mutex, as it's only ever 1
- Additionally changed the layout and introduces the c2w_frame_s and w2s_cid_frame_s name schema
- Fixed a bug in scheduler which WOULD cause a DEADLOCK if handshake would fail
- fixd a but in api_send_send_main, i need to store the stream_p otherwise it's immeadiatly closed and a stream_a.send() isn't guaranteed
- add extra test to verify that a send message is received even if the Stream is already closed
- changed OutGoing to Outgoing
- fixed a bug that `metrics.tick()` was never called
- removed 2 unused nightly features and added `deny_code`
2020-06-03 07:13:00 +00:00
/// # // keep it alive
2020-08-25 13:32:42 +00:00
/// # let _stream_p = remote_p.open(16, Promises::ORDERED | Promises::CONSISTENCY).await?;
Fixing the DEADLOCK in handshake -> channel creation
- this bug was initially called imbris bug, as it happened on his runners and i couldn't reproduce it locally at fist :)
- When in a Handshake a seperate mpsc::Channel was created for (Cid, Frame) transport
however the protocol could already catch non handshake data any more and push in into this
mpsc::Channel.
Then this channel got dropped and a fresh one was created for the network::Channel.
These droped Frames are ofc a BUG!
I tried multiple things to solve this:
- dont create a new mpsc::Channel, but instead bind it to the Protocol itself and always use 1.
This would work theoretically, but in bParticipant side we are using 1 mpsc::Channel<(Cid, Frame)>
to handle ALL the network::channel.
If now ever Protocol would have it's own, and with that every network::Channel had it's own it would no longer work out
Bad Idea...
- using the first method but creating the mpsc::Channel inside the scheduler instead protocol neither works, as the
scheduler doesnt know the remote_pid yet
- i dont want a hack to say the protocol only listen to 2 messages and then stop no matter what
So i switched over to the simply method now:
- Do everything like before with 2 mpsc::Channels
- after the handshake. close the receiver and listen for all remaining (cid, frame) combinations
- when starting the channel, reapply them to the new sender/listener combination
- added tracing
- switched Protocol RwLock to Mutex, as it's only ever 1
- Additionally changed the layout and introduces the c2w_frame_s and w2s_cid_frame_s name schema
- Fixed a bug in scheduler which WOULD cause a DEADLOCK if handshake would fail
- fixd a but in api_send_send_main, i need to store the stream_p otherwise it's immeadiatly closed and a stream_a.send() isn't guaranteed
- add extra test to verify that a send message is received even if the Stream is already closed
- changed OutGoing to Outgoing
- fixed a bug that `metrics.tick()` was never called
- removed 2 unused nightly features and added `deny_code`
2020-06-03 07:13:00 +00:00
/// let participant_a = network.connected().await?;
/// let mut stream_a = participant_a.opened().await?;
2020-05-10 02:07:46 +00:00
/// //Send Message
Fixing the DEADLOCK in handshake -> channel creation
- this bug was initially called imbris bug, as it happened on his runners and i couldn't reproduce it locally at fist :)
- When in a Handshake a seperate mpsc::Channel was created for (Cid, Frame) transport
however the protocol could already catch non handshake data any more and push in into this
mpsc::Channel.
Then this channel got dropped and a fresh one was created for the network::Channel.
These droped Frames are ofc a BUG!
I tried multiple things to solve this:
- dont create a new mpsc::Channel, but instead bind it to the Protocol itself and always use 1.
This would work theoretically, but in bParticipant side we are using 1 mpsc::Channel<(Cid, Frame)>
to handle ALL the network::channel.
If now ever Protocol would have it's own, and with that every network::Channel had it's own it would no longer work out
Bad Idea...
- using the first method but creating the mpsc::Channel inside the scheduler instead protocol neither works, as the
scheduler doesnt know the remote_pid yet
- i dont want a hack to say the protocol only listen to 2 messages and then stop no matter what
So i switched over to the simply method now:
- Do everything like before with 2 mpsc::Channels
- after the handshake. close the receiver and listen for all remaining (cid, frame) combinations
- when starting the channel, reapply them to the new sender/listener combination
- added tracing
- switched Protocol RwLock to Mutex, as it's only ever 1
- Additionally changed the layout and introduces the c2w_frame_s and w2s_cid_frame_s name schema
- Fixed a bug in scheduler which WOULD cause a DEADLOCK if handshake would fail
- fixd a but in api_send_send_main, i need to store the stream_p otherwise it's immeadiatly closed and a stream_a.send() isn't guaranteed
- add extra test to verify that a send message is received even if the Stream is already closed
- changed OutGoing to Outgoing
- fixed a bug that `metrics.tick()` was never called
- removed 2 unused nightly features and added `deny_code`
2020-06-03 07:13:00 +00:00
/// stream_a.send("Hello World")?;
/// # Ok(())
2020-05-26 13:06:03 +00:00
/// })
/// # }
2020-05-10 02:07:46 +00:00
/// ```
///
/// [`send_raw`]: Stream::send_raw
/// [`recv`]: Stream::recv
/// [`Serialized`]: Serialize
2020-05-22 14:00:08 +00:00
#[ inline ]
2020-04-08 14:26:42 +00:00
pub fn send < M : Serialize > ( & mut self , msg : M ) -> Result < ( ) , StreamError > {
2020-10-16 09:21:18 +00:00
self . send_raw ( & Message ::serialize ( & msg , & self ) )
2020-04-24 10:56:04 +00:00
}
2020-08-25 13:32:42 +00:00
/// This methods give the option to skip multiple calls of [`bincode`] and
/// [`compress`], e.g. in case the same Message needs to send on
/// multiple `Streams` to multiple [`Participants`]. Other then that,
2020-10-16 09:21:18 +00:00
/// the same rules apply than for [`send`].
/// You need to create a Message via [`Message::serialize`].
2020-05-10 02:07:46 +00:00
///
/// # Example
/// ```rust
2020-10-16 09:21:18 +00:00
/// use veloren_network::{Network, ProtocolAddr, Pid, Message};
2020-08-25 13:32:42 +00:00
/// # use veloren_network::Promises;
2020-05-10 02:07:46 +00:00
/// use futures::executor::block_on;
2020-05-26 13:06:03 +00:00
/// use bincode;
/// use std::sync::Arc;
2020-05-10 02:07:46 +00:00
///
2020-05-26 13:06:03 +00:00
/// # fn main() -> std::result::Result<(), Box<dyn std::error::Error>> {
2020-07-14 23:34:41 +00:00
/// let (network, f) = Network::new(Pid::new());
2020-06-08 09:47:39 +00:00
/// std::thread::spawn(f);
2020-07-14 23:34:41 +00:00
/// # let (remote1, fr1) = Network::new(Pid::new());
2020-06-08 09:47:39 +00:00
/// # std::thread::spawn(fr1);
2020-07-14 23:34:41 +00:00
/// # let (remote2, fr2) = Network::new(Pid::new());
2020-06-08 09:47:39 +00:00
/// # std::thread::spawn(fr2);
2020-05-10 02:07:46 +00:00
/// block_on(async {
2020-07-09 11:42:38 +00:00
/// network.listen(ProtocolAddr::Tcp("127.0.0.1:2210".parse().unwrap())).await?;
/// # let remote1_p = remote1.connect(ProtocolAddr::Tcp("127.0.0.1:2210".parse().unwrap())).await?;
/// # let remote2_p = remote2.connect(ProtocolAddr::Tcp("127.0.0.1:2210".parse().unwrap())).await?;
2020-05-26 13:06:03 +00:00
/// # assert_eq!(remote1_p.remote_pid(), remote2_p.remote_pid());
2020-08-25 13:32:42 +00:00
/// # remote1_p.open(16, Promises::ORDERED | Promises::CONSISTENCY).await?;
/// # remote2_p.open(16, Promises::ORDERED | Promises::CONSISTENCY).await?;
2020-05-26 13:06:03 +00:00
/// let participant_a = network.connected().await?;
/// let participant_b = network.connected().await?;
/// let mut stream_a = participant_a.opened().await?;
/// let mut stream_b = participant_b.opened().await?;
2020-05-10 02:07:46 +00:00
///
/// //Prepare Message and decode it
2020-10-16 09:21:18 +00:00
/// let msg = Message::serialize("Hello World", &stream_a);
2020-05-10 02:07:46 +00:00
/// //Send same Message to multiple Streams
2020-10-16 09:21:18 +00:00
/// stream_a.send_raw(&msg);
/// stream_b.send_raw(&msg);
2020-05-26 13:06:03 +00:00
/// # Ok(())
/// })
/// # }
2020-05-10 02:07:46 +00:00
/// ```
///
/// [`send`]: Stream::send
/// [`Participants`]: crate::api::Participant
2020-08-25 13:32:42 +00:00
/// [`compress`]: lz_fear::raw::compress2
2020-10-16 09:21:18 +00:00
/// [`Message::serialize`]: crate::message::Message::serialize
pub fn send_raw ( & mut self , message : & Message ) -> Result < ( ) , StreamError > {
2020-07-16 19:39:33 +00:00
if self . send_closed . load ( Ordering ::Relaxed ) {
2020-04-08 14:26:42 +00:00
return Err ( StreamError ::StreamClosed ) ;
}
2020-10-16 09:21:18 +00:00
#[ cfg(debug_assertions) ]
message . verify ( & self ) ;
Fixing the DEADLOCK in handshake -> channel creation
- this bug was initially called imbris bug, as it happened on his runners and i couldn't reproduce it locally at fist :)
- When in a Handshake a seperate mpsc::Channel was created for (Cid, Frame) transport
however the protocol could already catch non handshake data any more and push in into this
mpsc::Channel.
Then this channel got dropped and a fresh one was created for the network::Channel.
These droped Frames are ofc a BUG!
I tried multiple things to solve this:
- dont create a new mpsc::Channel, but instead bind it to the Protocol itself and always use 1.
This would work theoretically, but in bParticipant side we are using 1 mpsc::Channel<(Cid, Frame)>
to handle ALL the network::channel.
If now ever Protocol would have it's own, and with that every network::Channel had it's own it would no longer work out
Bad Idea...
- using the first method but creating the mpsc::Channel inside the scheduler instead protocol neither works, as the
scheduler doesnt know the remote_pid yet
- i dont want a hack to say the protocol only listen to 2 messages and then stop no matter what
So i switched over to the simply method now:
- Do everything like before with 2 mpsc::Channels
- after the handshake. close the receiver and listen for all remaining (cid, frame) combinations
- when starting the channel, reapply them to the new sender/listener combination
- added tracing
- switched Protocol RwLock to Mutex, as it's only ever 1
- Additionally changed the layout and introduces the c2w_frame_s and w2s_cid_frame_s name schema
- Fixed a bug in scheduler which WOULD cause a DEADLOCK if handshake would fail
- fixd a but in api_send_send_main, i need to store the stream_p otherwise it's immeadiatly closed and a stream_a.send() isn't guaranteed
- add extra test to verify that a send message is received even if the Stream is already closed
- changed OutGoing to Outgoing
- fixed a bug that `metrics.tick()` was never called
- removed 2 unused nightly features and added `deny_code`
2020-06-03 07:13:00 +00:00
self . a2b_msg_s . send ( ( self . prio , self . sid , OutgoingMessage {
2020-10-16 09:21:18 +00:00
buffer : Arc ::clone ( & message . buffer ) ,
2020-05-22 14:00:08 +00:00
cursor : 0 ,
mid : self . mid ,
sid : self . sid ,
} ) ) ? ;
2020-03-22 13:47:21 +00:00
self . mid + = 1 ;
2020-02-21 15:10:55 +00:00
Ok ( ( ) )
2020-02-10 17:25:47 +00:00
}
2020-05-10 02:07:46 +00:00
/// use `recv` to wait on a Message send from the remote side by their
/// `Stream`. The Message needs to implement [`DeserializeOwned`] and
/// thus, the resulting type must already be known by the receiving side.
/// If this is not know from the Application logic, one could use a `Enum`
/// and then handle the received message via a `match` state.
///
/// A [`StreamError`] will be returned in the error case, e.g. when the
/// `Stream` got closed already.
2020-05-27 15:58:57 +00:00
///
/// # Example
Fixing the DEADLOCK in handshake -> channel creation
- this bug was initially called imbris bug, as it happened on his runners and i couldn't reproduce it locally at fist :)
- When in a Handshake a seperate mpsc::Channel was created for (Cid, Frame) transport
however the protocol could already catch non handshake data any more and push in into this
mpsc::Channel.
Then this channel got dropped and a fresh one was created for the network::Channel.
These droped Frames are ofc a BUG!
I tried multiple things to solve this:
- dont create a new mpsc::Channel, but instead bind it to the Protocol itself and always use 1.
This would work theoretically, but in bParticipant side we are using 1 mpsc::Channel<(Cid, Frame)>
to handle ALL the network::channel.
If now ever Protocol would have it's own, and with that every network::Channel had it's own it would no longer work out
Bad Idea...
- using the first method but creating the mpsc::Channel inside the scheduler instead protocol neither works, as the
scheduler doesnt know the remote_pid yet
- i dont want a hack to say the protocol only listen to 2 messages and then stop no matter what
So i switched over to the simply method now:
- Do everything like before with 2 mpsc::Channels
- after the handshake. close the receiver and listen for all remaining (cid, frame) combinations
- when starting the channel, reapply them to the new sender/listener combination
- added tracing
- switched Protocol RwLock to Mutex, as it's only ever 1
- Additionally changed the layout and introduces the c2w_frame_s and w2s_cid_frame_s name schema
- Fixed a bug in scheduler which WOULD cause a DEADLOCK if handshake would fail
- fixd a but in api_send_send_main, i need to store the stream_p otherwise it's immeadiatly closed and a stream_a.send() isn't guaranteed
- add extra test to verify that a send message is received even if the Stream is already closed
- changed OutGoing to Outgoing
- fixed a bug that `metrics.tick()` was never called
- removed 2 unused nightly features and added `deny_code`
2020-06-03 07:13:00 +00:00
/// ```
2020-07-09 11:42:38 +00:00
/// use veloren_network::{Network, ProtocolAddr, Pid};
2020-08-25 13:32:42 +00:00
/// # use veloren_network::Promises;
2020-05-27 15:58:57 +00:00
/// use futures::executor::block_on;
///
/// # fn main() -> std::result::Result<(), Box<dyn std::error::Error>> {
/// // Create a Network, listen on Port `2220` and wait for a Stream to be opened, then listen on it
2020-07-14 23:34:41 +00:00
/// let (network, f) = Network::new(Pid::new());
2020-06-08 09:47:39 +00:00
/// std::thread::spawn(f);
2020-07-14 23:34:41 +00:00
/// # let (remote, fr) = Network::new(Pid::new());
2020-06-08 09:47:39 +00:00
/// # std::thread::spawn(fr);
2020-05-27 15:58:57 +00:00
/// block_on(async {
2020-07-09 11:42:38 +00:00
/// network.listen(ProtocolAddr::Tcp("127.0.0.1:2220".parse().unwrap())).await?;
/// # let remote_p = remote.connect(ProtocolAddr::Tcp("127.0.0.1:2220".parse().unwrap())).await?;
2020-08-25 13:32:42 +00:00
/// # let mut stream_p = remote_p.open(16, Promises::ORDERED | Promises::CONSISTENCY).await?;
2020-05-27 15:58:57 +00:00
/// # stream_p.send("Hello World");
/// let participant_a = network.connected().await?;
/// let mut stream_a = participant_a.opened().await?;
2020-10-15 11:22:34 +00:00
/// //Recv Message
2020-05-27 15:58:57 +00:00
/// println!("{}", stream_a.recv::<String>().await?);
/// # Ok(())
/// })
/// # }
/// ```
2020-05-22 14:00:08 +00:00
#[ inline ]
2020-03-04 10:59:19 +00:00
pub async fn recv < M : DeserializeOwned > ( & mut self ) -> Result < M , StreamError > {
2020-10-16 09:21:18 +00:00
self . recv_raw ( ) . await ? . deserialize ( )
2020-04-24 10:56:04 +00:00
}
2020-08-25 13:32:42 +00:00
/// the equivalent like [`send_raw`] but for [`recv`], no [`bincode`] or
/// [`decompress`] is executed for performance reasons.
2020-05-10 02:07:46 +00:00
///
2020-10-16 09:21:18 +00:00
/// # Example
/// ```
/// use veloren_network::{Network, ProtocolAddr, Pid};
/// # use veloren_network::Promises;
/// use futures::executor::block_on;
///
/// # fn main() -> std::result::Result<(), Box<dyn std::error::Error>> {
/// // Create a Network, listen on Port `2230` and wait for a Stream to be opened, then listen on it
/// let (network, f) = Network::new(Pid::new());
/// std::thread::spawn(f);
/// # let (remote, fr) = Network::new(Pid::new());
/// # std::thread::spawn(fr);
/// block_on(async {
/// network.listen(ProtocolAddr::Tcp("127.0.0.1:2230".parse().unwrap())).await?;
/// # let remote_p = remote.connect(ProtocolAddr::Tcp("127.0.0.1:2230".parse().unwrap())).await?;
/// # let mut stream_p = remote_p.open(16, Promises::ORDERED | Promises::CONSISTENCY).await?;
/// # stream_p.send("Hello World");
/// let participant_a = network.connected().await?;
/// let mut stream_a = participant_a.opened().await?;
/// //Recv Message
/// let msg = stream_a.recv_raw().await?;
/// //Resend Message, without deserializing
/// stream_a.send_raw(&msg)?;
/// # Ok(())
/// })
/// # }
/// ```
///
2020-05-10 02:07:46 +00:00
/// [`send_raw`]: Stream::send_raw
/// [`recv`]: Stream::recv
2020-08-25 13:32:42 +00:00
/// [`decompress`]: lz_fear::raw::decompress_raw
2020-10-16 09:21:18 +00:00
pub async fn recv_raw ( & mut self ) -> Result < Message , StreamError > {
match & mut self . b2a_msg_recv_r {
Some ( b2a_msg_recv_r ) = > {
match b2a_msg_recv_r . next ( ) . await {
Some ( msg ) = > Ok ( Message {
buffer : Arc ::new ( msg . buffer ) ,
#[ cfg(feature = " compression " ) ]
compressed : self . promises . contains ( Promises ::COMPRESSED ) ,
} ) ,
None = > {
self . b2a_msg_recv_r = None ; //prevent panic
Err ( StreamError ::StreamClosed )
} ,
}
} ,
None = > Err ( StreamError ::StreamClosed ) ,
}
2020-02-10 17:25:47 +00:00
}
2020-10-15 11:22:34 +00:00
/// use `try_recv` to check for a Message send from the remote side by their
/// `Stream`. This function does not block and returns immediately. It's
/// intended for use in non-async context only. Other then that, the
/// same rules apply than for [`recv`].
///
/// # Example
/// ```
/// use veloren_network::{Network, ProtocolAddr, Pid};
/// # use veloren_network::Promises;
/// use futures::executor::block_on;
///
/// # fn main() -> std::result::Result<(), Box<dyn std::error::Error>> {
2020-10-16 09:21:18 +00:00
/// // Create a Network, listen on Port `2240` and wait for a Stream to be opened, then listen on it
2020-10-15 11:22:34 +00:00
/// let (network, f) = Network::new(Pid::new());
/// std::thread::spawn(f);
/// # let (remote, fr) = Network::new(Pid::new());
/// # std::thread::spawn(fr);
/// block_on(async {
2020-10-16 09:21:18 +00:00
/// network.listen(ProtocolAddr::Tcp("127.0.0.1:2240".parse().unwrap())).await?;
/// # let remote_p = remote.connect(ProtocolAddr::Tcp("127.0.0.1:2240".parse().unwrap())).await?;
2020-10-15 11:22:34 +00:00
/// # let mut stream_p = remote_p.open(16, Promises::ORDERED | Promises::CONSISTENCY).await?;
/// # stream_p.send("Hello World");
/// # std::thread::sleep(std::time::Duration::from_secs(1));
/// let participant_a = network.connected().await?;
/// let mut stream_a = participant_a.opened().await?;
/// //Try Recv Message
/// println!("{:?}", stream_a.try_recv::<String>()?);
/// # Ok(())
/// })
/// # }
/// ```
2020-10-16 09:21:18 +00:00
///
/// [`recv`]: Stream::recv
2020-10-15 11:22:34 +00:00
#[ inline ]
pub fn try_recv < M : DeserializeOwned > ( & mut self ) -> Result < Option < M > , StreamError > {
2020-10-16 09:21:18 +00:00
match & mut self . b2a_msg_recv_r {
Some ( b2a_msg_recv_r ) = > match b2a_msg_recv_r . try_next ( ) {
Err ( _ ) = > Ok ( None ) ,
Ok ( None ) = > {
self . b2a_msg_recv_r = None ; //prevent panic
Err ( StreamError ::StreamClosed )
} ,
Ok ( Some ( msg ) ) = > Ok ( Some (
Message {
buffer : Arc ::new ( msg . buffer ) ,
#[ cfg(feature = " compression " ) ]
compressed : self . promises ( ) . contains ( Promises ::COMPRESSED ) ,
}
. deserialize ( ) ? ,
) ) ,
} ,
None = > Err ( StreamError ::StreamClosed ) ,
2020-10-15 11:22:34 +00:00
}
}
2020-10-16 09:21:18 +00:00
pub fn promises ( & self ) -> Promises { self . promises }
2020-03-22 13:47:21 +00:00
}
2020-03-10 00:07:36 +00:00
2020-07-09 07:58:21 +00:00
impl core ::cmp ::PartialEq for Participant {
fn eq ( & self , other : & Self ) -> bool {
//don't check local_pid, 2 Participant from different network should match if
// they are the "same"
self . remote_pid = = other . remote_pid
}
}
2020-03-22 13:47:21 +00:00
impl Drop for Network {
fn drop ( & mut self ) {
2020-04-08 14:26:42 +00:00
let pid = self . local_pid ;
2020-07-05 22:13:53 +00:00
debug! ( ? pid , " Shutting down Network " ) ;
2020-07-09 07:58:21 +00:00
trace! (
2020-05-15 12:29:17 +00:00
? pid ,
2020-07-05 22:13:53 +00:00
" Shutting down Participants of Network, while we still have metrics "
2020-05-15 12:29:17 +00:00
) ;
2020-07-09 07:58:21 +00:00
let mut finished_receiver_list = vec! [ ] ;
2020-05-15 12:29:17 +00:00
task ::block_on ( async {
2020-07-10 13:31:26 +00:00
// we MUST avoid nested block_on, good that Network::Drop no longer triggers
// Participant::Drop directly but just the BParticipant
2020-07-09 07:58:21 +00:00
for ( remote_pid , a2s_disconnect_s ) in
2020-08-23 19:43:17 +00:00
self . participant_disconnect_sender . lock ( ) . await . drain ( )
2020-07-09 07:58:21 +00:00
{
2020-07-10 13:31:26 +00:00
match a2s_disconnect_s . lock ( ) . await . take ( ) {
2020-07-09 07:58:21 +00:00
Some ( mut a2s_disconnect_s ) = > {
trace! ( ? remote_pid , " Participants will be closed " ) ;
let ( finished_sender , finished_receiver ) = oneshot ::channel ( ) ;
finished_receiver_list . push ( ( remote_pid , finished_receiver ) ) ;
a2s_disconnect_s
. send ( ( remote_pid , finished_sender ) )
. await
. expect (
" Scheduler is closed, but nobody other should be able to close it " ,
) ;
} ,
None = > trace! ( ? remote_pid , " Participant already disconnected gracefully " ) ,
}
}
//wait after close is requested for all
for ( remote_pid , finished_receiver ) in finished_receiver_list . drain ( .. ) {
match finished_receiver . await {
Ok ( Ok ( ( ) ) ) = > trace! ( ? remote_pid , " disconnect successful " ) ,
Ok ( Err ( e ) ) = > info! ( ? remote_pid , ? e , " unclean disconnect " ) ,
Err ( e ) = > warn! (
? remote_pid ,
2020-06-08 09:47:39 +00:00
? e ,
2020-07-09 07:58:21 +00:00
" Failed to get a message back from the scheduler, seems like the network \
is already closed "
) ,
2020-05-27 15:58:57 +00:00
}
}
2020-05-15 12:29:17 +00:00
} ) ;
2020-07-09 07:58:21 +00:00
trace! ( ? pid , " Participants have shut down! " ) ;
trace! ( ? pid , " Shutting down Scheduler " ) ;
2020-04-08 14:26:42 +00:00
self . shutdown_sender
. take ( )
. unwrap ( )
. send ( ( ) )
2020-07-05 22:13:53 +00:00
. expect ( " Scheduler is closed, but nobody other should be able to close it " ) ;
2020-07-09 07:58:21 +00:00
debug! ( ? pid , " Network has shut down " ) ;
2020-03-10 00:07:36 +00:00
}
2020-03-22 13:47:21 +00:00
}
2020-03-10 00:07:36 +00:00
2020-03-22 13:47:21 +00:00
impl Drop for Participant {
fn drop ( & mut self ) {
2020-04-08 14:26:42 +00:00
// ignore closed, as we need to send it even though we disconnected the
// participant from network
let pid = self . remote_pid ;
2020-07-05 22:13:53 +00:00
debug! ( ? pid , " Shutting down Participant " ) ;
2020-07-09 07:58:21 +00:00
2020-07-10 13:31:26 +00:00
match task ::block_on ( self . a2s_disconnect_s . lock ( ) ) . take ( ) {
2020-07-09 07:58:21 +00:00
None = > trace! (
2020-05-15 12:29:17 +00:00
? pid ,
2020-08-25 12:21:25 +00:00
" Participant has been shutdown cleanly, no further waiting is required! "
2020-05-15 12:29:17 +00:00
) ,
Some ( mut a2s_disconnect_s ) = > {
2020-07-09 07:58:21 +00:00
debug! ( ? pid , " Disconnect from Scheduler " ) ;
2020-05-15 12:29:17 +00:00
task ::block_on ( async {
let ( finished_sender , finished_receiver ) = oneshot ::channel ( ) ;
a2s_disconnect_s
. send ( ( self . remote_pid , finished_sender ) )
. await
2020-07-05 22:13:53 +00:00
. expect ( " Something is wrong in internal scheduler coding " ) ;
2020-07-11 14:08:25 +00:00
if let Err ( e ) = finished_receiver
2020-07-11 12:34:01 +00:00
. await
. expect ( " Something is wrong in internal scheduler/participant coding " )
{
2020-07-11 14:08:25 +00:00
error! (
2020-05-15 12:29:17 +00:00
? pid ,
? e ,
" Error while dropping the participant, couldn't send all outgoing \
messages , dropping remaining "
2020-07-11 14:08:25 +00:00
) ;
2020-05-15 12:29:17 +00:00
} ;
} ) ;
} ,
}
2020-07-09 07:58:21 +00:00
debug! ( ? pid , " Participant dropped " ) ;
2020-03-10 00:07:36 +00:00
}
}
impl Drop for Stream {
fn drop ( & mut self ) {
2020-10-14 12:29:39 +00:00
// send if closed is unnecessary but doesn't hurt, we must not crash
2020-07-16 19:39:33 +00:00
if ! self . send_closed . load ( Ordering ::Relaxed ) {
2020-04-08 14:26:42 +00:00
let sid = self . sid ;
let pid = self . pid ;
2020-07-05 22:13:53 +00:00
debug! ( ? pid , ? sid , " Shutting down Stream " ) ;
2020-10-14 12:29:39 +00:00
task ::block_on ( self . a2b_close_stream_s . take ( ) . unwrap ( ) . send ( self . sid ) )
. expect ( " bparticipant part of a gracefully shutdown must have crashed " ) ;
2020-05-15 12:29:17 +00:00
} else {
let sid = self . sid ;
let pid = self . pid ;
2020-07-09 07:58:21 +00:00
trace! ( ? pid , ? sid , " Stream Drop not needed " ) ;
2020-03-22 13:47:21 +00:00
}
2020-03-10 00:07:36 +00:00
}
2020-02-10 17:25:47 +00:00
}
2020-04-08 14:26:42 +00:00
impl std ::fmt ::Debug for Participant {
#[ inline ]
fn fmt ( & self , f : & mut std ::fmt ::Formatter < '_ > ) -> std ::fmt ::Result {
write! (
f ,
2020-07-11 12:34:01 +00:00
" Participant {{ local_pid: {:?}, remote_pid: {:?} }} " ,
& self . local_pid , & self . remote_pid ,
2020-04-08 14:26:42 +00:00
)
}
}
2020-07-03 19:55:00 +00:00
impl < T > From < crossbeam_channel ::SendError < T > > for StreamError {
fn from ( _err : crossbeam_channel ::SendError < T > ) -> Self { StreamError ::StreamClosed }
2020-04-08 14:26:42 +00:00
}
2020-07-03 19:55:00 +00:00
impl < T > From < crossbeam_channel ::SendError < T > > for NetworkError {
fn from ( _err : crossbeam_channel ::SendError < T > ) -> Self { NetworkError ::NetworkClosed }
2020-04-08 14:26:42 +00:00
}
impl From < std ::option ::NoneError > for StreamError {
fn from ( _err : std ::option ::NoneError ) -> Self { StreamError ::StreamClosed }
}
impl From < std ::option ::NoneError > for NetworkError {
fn from ( _err : std ::option ::NoneError ) -> Self { NetworkError ::NetworkClosed }
}
impl From < mpsc ::SendError > for NetworkError {
fn from ( _err : mpsc ::SendError ) -> Self { NetworkError ::NetworkClosed }
}
impl From < oneshot ::Canceled > for NetworkError {
fn from ( _err : oneshot ::Canceled ) -> Self { NetworkError ::NetworkClosed }
}
2020-05-26 13:06:03 +00:00
2020-06-08 09:47:39 +00:00
impl From < Box < bincode ::ErrorKind > > for StreamError {
2020-08-25 13:32:42 +00:00
fn from ( err : Box < bincode ::ErrorKind > ) -> Self { StreamError ::Deserialize ( err ) }
2020-06-08 09:47:39 +00:00
}
2020-05-26 13:06:03 +00:00
impl core ::fmt ::Display for StreamError {
2020-06-08 09:47:39 +00:00
fn fmt ( & self , f : & mut core ::fmt ::Formatter < '_ > ) -> core ::fmt ::Result {
2020-05-26 13:06:03 +00:00
match self {
StreamError ::StreamClosed = > write! ( f , " stream closed " ) ,
2020-08-25 13:32:42 +00:00
#[ cfg(feature = " compression " ) ]
StreamError ::Compression ( err ) = > write! ( f , " compression error on message: {} " , err ) ,
StreamError ::Deserialize ( err ) = > write! ( f , " deserialize error on message: {} " , err ) ,
2020-05-26 13:06:03 +00:00
}
}
}
impl core ::fmt ::Display for ParticipantError {
2020-06-08 09:47:39 +00:00
fn fmt ( & self , f : & mut core ::fmt ::Formatter < '_ > ) -> core ::fmt ::Result {
2020-05-26 13:06:03 +00:00
match self {
2020-07-11 12:34:01 +00:00
ParticipantError ::ParticipantDisconnected = > write! ( f , " Participant disconnect " ) ,
ParticipantError ::ProtocolFailedUnrecoverable = > {
write! ( f , " underlying protocol failed unrecoverable " )
2020-07-09 07:58:21 +00:00
} ,
2020-05-26 13:06:03 +00:00
}
}
}
impl core ::fmt ::Display for NetworkError {
2020-06-08 09:47:39 +00:00
fn fmt ( & self , f : & mut core ::fmt ::Formatter < '_ > ) -> core ::fmt ::Result {
2020-05-26 13:06:03 +00:00
match self {
2020-07-05 22:13:53 +00:00
NetworkError ::NetworkClosed = > write! ( f , " Network closed " ) ,
NetworkError ::ListenFailed ( _ ) = > write! ( f , " Listening failed " ) ,
NetworkError ::ConnectFailed ( _ ) = > write! ( f , " Connecting failed " ) ,
2020-05-26 13:06:03 +00:00
}
}
}
2020-06-08 09:47:39 +00:00
/// implementing PartialEq as it's super convenient in tests
impl core ::cmp ::PartialEq for StreamError {
fn eq ( & self , other : & Self ) -> bool {
match self {
StreamError ::StreamClosed = > match other {
StreamError ::StreamClosed = > true ,
2020-08-25 13:32:42 +00:00
#[ cfg(feature = " compression " ) ]
StreamError ::Compression ( _ ) = > false ,
StreamError ::Deserialize ( _ ) = > false ,
} ,
#[ cfg(feature = " compression " ) ]
StreamError ::Compression ( err ) = > match other {
StreamError ::StreamClosed = > false ,
#[ cfg(feature = " compression " ) ]
StreamError ::Compression ( other_err ) = > err = = other_err ,
StreamError ::Deserialize ( _ ) = > false ,
2020-06-08 09:47:39 +00:00
} ,
2020-08-25 13:32:42 +00:00
StreamError ::Deserialize ( err ) = > match other {
2020-06-08 09:47:39 +00:00
StreamError ::StreamClosed = > false ,
2020-08-25 13:32:42 +00:00
#[ cfg(feature = " compression " ) ]
StreamError ::Compression ( _ ) = > false ,
StreamError ::Deserialize ( other_err ) = > partial_eq_bincode ( err , other_err ) ,
2020-06-08 09:47:39 +00:00
} ,
}
}
}
2020-05-26 13:06:03 +00:00
impl std ::error ::Error for StreamError { }
impl std ::error ::Error for ParticipantError { }
impl std ::error ::Error for NetworkError { }