2020-01-13 16:53:28 +00:00
use crate ::{
2020-04-24 10:56:04 +00:00
message ::{ self , InCommingMessage , MessageBuffer , OutGoingMessage } ,
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-04-08 14:26:42 +00:00
use async_std ::{ io , sync ::RwLock , 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-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-03-22 13:47:21 +00:00
sync ::{
2020-05-15 12:29:17 +00:00
atomic ::{ AtomicBool , AtomicU64 , 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 ;
2020-01-13 16:53:28 +00:00
use uvth ::ThreadPool ;
2019-12-20 13:56:01 +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) ]
2019-12-20 13:56:01 +00:00
pub enum Address {
Tcp ( std ::net ::SocketAddr ) ,
Udp ( std ::net ::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-05-15 12:29:17 +00:00
a2b_steam_open_s : RwLock < mpsc ::UnboundedSender < ( Prio , Promises , oneshot ::Sender < Stream > ) > > ,
b2a_stream_opened_r : RwLock < mpsc ::UnboundedReceiver < Stream > > ,
2020-03-22 13:47:21 +00:00
closed : AtomicBool ,
2020-05-15 12:29:17 +00:00
a2s_disconnect_s :
Option < mpsc ::UnboundedSender < ( Pid , oneshot ::Sender < async_std ::io ::Result < ( ) > > ) > > ,
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`.
/// [`Sync`] is not supported! In that case multiple `Streams` should be used
/// instead. However it's still possible to [`Send`] `Streams`.
///
/// [`Networks`]: crate::api::Network
/// [`open`]: Participant::open
/// [`opened`]: Participant::opened
/// [`Send`]: std::marker::Send
/// [`Sync`]: std::marker::Sync
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-05-15 12:29:17 +00:00
a2b_msg_s : std ::sync ::mpsc ::Sender < ( Prio , Pid , Sid , OutGoingMessage ) > ,
b2a_msg_recv_r : mpsc ::UnboundedReceiver < InCommingMessage > ,
2020-04-08 14:26:42 +00:00
closed : Arc < AtomicBool > ,
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-03-22 13:47:21 +00:00
2020-05-10 02:07:46 +00:00
/// Error type thrown by [`Participants`](Participant) methods
2020-04-08 14:26:42 +00:00
#[ derive(Debug, PartialEq) ]
pub enum ParticipantError {
ParticipantClosed ,
}
2020-03-22 13:47:21 +00:00
2020-05-10 02:07:46 +00:00
/// Error type thrown by [`Streams`](Stream) methods
2020-04-08 14:26:42 +00:00
#[ derive(Debug, PartialEq) ]
pub enum StreamError {
StreamClosed ,
}
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.
///
/// The `Network` has methods to [`connect`] and [`disconnect`] to other
/// [`Participants`] via their [`Address`]. All [`Participants`] will be stored
/// in the Network until explicitly disconnected, which is the only way to close
/// the sockets.
///
/// # Examples
/// ```rust
/// use veloren_network::{Network, Pid};
/// use uvth::ThreadPoolBuilder;
///
/// // Create a Network, listen on port `12345` to accept connections and connect to port `80` to connect to a (pseudo) database Application
/// let network = Network::new(Pid::new(), ThreadPoolBuilder::new().build(), None);
/// block_on(async {
/// network.listen(Address::Tcp("127.0.0.1:12345".parse().unwrap())).await?;
/// let database = network.connect(Address::Tcp("127.0.0.1:80".parse().unwrap())).await?;
/// });
/// ```
///
/// [`Participants`]: crate::api::Participant
/// [`connect`]: Network::connect
/// [`disconnect`]: Network::disconnect
2020-02-21 15:10:55 +00:00
pub struct Network {
2020-03-22 13:47:21 +00:00
local_pid : Pid ,
participants : RwLock < HashMap < Pid , Arc < Participant > > > ,
2020-04-08 14:26:42 +00:00
listen_sender :
RwLock < mpsc ::UnboundedSender < ( Address , oneshot ::Sender < async_std ::io ::Result < ( ) > > ) > > ,
connect_sender :
RwLock < mpsc ::UnboundedSender < ( Address , oneshot ::Sender < io ::Result < Participant > > ) > > ,
2020-03-22 13:47:21 +00:00
connected_receiver : RwLock < mpsc ::UnboundedReceiver < Participant > > ,
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`
/// * `thread_pool` - you need to provide a [`ThreadPool`] where exactly 1
/// thread will be created to handle all `Network` internals. Additional
/// threads will be allocated on an internal async-aware threadpool
/// * `registry` - Provide a Registy in order to collect Prometheus metrics
/// by this `Network`, `None` will deactivate Tracing. Tracing is done via
/// [`prometheus`]
///
/// # Examples
/// ```rust
/// use uvth::ThreadPoolBuilder;
/// use veloren_network::{Network, Pid};
///
/// let network = Network::new(Pid::new(), ThreadPoolBuilder::new().build(), None);
/// ```
///
/// Usually you only create a single `Network` for an application, except
/// when client and server are in the same application, then you will want
/// 2. However there are no technical limitations from creating more.
///
/// [`Pid::new()`]: crate::types::Pid::new
/// [`ThreadPool`]: uvth::ThreadPool
2020-04-24 10:56:04 +00:00
pub fn new ( participant_id : Pid , thread_pool : & ThreadPool , registry : Option < & Registry > ) -> Self {
2020-03-22 13:47:21 +00:00
let p = participant_id ;
2020-05-15 12:29:17 +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-04-24 10:56:04 +00:00
Scheduler ::new ( participant_id , registry ) ;
2020-03-22 13:47:21 +00:00
thread_pool . execute ( move | | {
2020-05-15 12:29:17 +00:00
trace! ( ? p , " starting sheduler in own thread " ) ;
2020-03-22 13:47:21 +00:00
let _handle = task ::block_on (
scheduler
. run ( )
. instrument ( tracing ::info_span! ( " scheduler " , ? p ) ) ,
) ;
2020-05-15 12:29:17 +00:00
trace! ( ? p , " stopping sheduler and his own thread " ) ;
2020-03-22 13:47:21 +00:00
} ) ;
2019-12-20 13:56:01 +00:00
Self {
2020-03-22 13:47:21 +00:00
local_pid : participant_id ,
participants : RwLock ::new ( HashMap ::new ( ) ) ,
listen_sender : RwLock ::new ( listen_sender ) ,
connect_sender : RwLock ::new ( connect_sender ) ,
connected_receiver : RwLock ::new ( connected_receiver ) ,
shutdown_sender : Some ( shutdown_sender ) ,
2019-12-20 13:56:01 +00:00
}
}
2020-05-10 02:07:46 +00:00
/// starts listening on an [`Address`].
/// 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
/// use uvth::ThreadPoolBuilder;
/// use veloren_network::{Network, Pid};
///
/// // Create a Network, listen on port `2000` TCP on all NICs and `2001` UDP locally
/// let network = Network::new(Pid::new(), ThreadPoolBuilder::new().build(), None);
/// block_on(async {
/// network
/// .listen(Address::Tcp("0.0.0.0:2000".parse().unwrap()))
/// .await?;
/// network
/// .listen(Address::Udp("127.0.0.1:2001".parse().unwrap()))
/// .await?;
/// });
/// ```
///
/// [`connected`]: Network::connected
2020-04-08 14:26:42 +00:00
pub async fn listen ( & self , address : Address ) -> 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
. write ( )
. 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-04-08 14:26:42 +00:00
//waiting guarantees that we either listened sucessfully or get an error like port in
// use
Ok ( ( ) ) = > Ok ( ( ) ) ,
Err ( e ) = > Err ( NetworkError ::ListenFailed ( e ) ) ,
}
2019-12-20 13:56:01 +00:00
}
2020-05-10 02:07:46 +00:00
/// starts connectiong to an [`Address`].
/// 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
/// use uvth::ThreadPoolBuilder;
/// use veloren_network::{Network, Pid};
///
/// // Create a Network, connect on port `2000` TCP and `2001` UDP like listening above
/// let network = Network::new(Pid::new(), ThreadPoolBuilder::new().build(), None);
/// block_on(async {
/// let p1 = network
/// .connect(Address::Tcp("127.0.0.1:2000".parse().unwrap()))
/// .await?;
/// let p2 = network
/// .connect(Address::Udp("127.0.0.1:2001".parse().unwrap()))
/// .await?;
/// assert!(p1.ptr_eq(p2));
/// });
/// ```
/// Usually the `Network` guarantees that a operation on a [`Participant`]
/// succeeds, e.g. by automatic retrying unless it fails completely e.g. by
/// disconnecting from the remote. If 2 [`Addresses`] you `connect` to
/// 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
/// [`Addresses`]: crate::api::Address
2020-03-22 13:47:21 +00:00
pub async fn connect ( & self , address : Address ) -> Result < Arc < Participant > , NetworkError > {
2020-04-08 14:26:42 +00:00
let ( pid_sender , pid_receiver ) = oneshot ::channel ::< io ::Result < Participant > > ( ) ;
2020-05-15 12:29:17 +00:00
debug! ( ? address , " connect to address " ) ;
2020-03-22 13:47:21 +00:00
self . connect_sender
. write ( )
. await
. send ( ( address , pid_sender ) )
2020-04-08 14:26:42 +00:00
. await ? ;
let participant = pid_receiver . await ? ? ;
let pid = participant . remote_pid ;
debug! (
? pid ,
" received Participant id from remote and return to user "
) ;
let participant = Arc ::new ( participant ) ;
self . participants
. write ( )
2020-03-22 13:47:21 +00:00
. await
2020-04-08 14:26:42 +00:00
. insert ( participant . remote_pid , participant . clone ( ) ) ;
Ok ( participant )
2020-03-22 13:47:21 +00:00
}
2020-03-10 00:07:36 +00:00
2020-05-10 02:07:46 +00:00
/// returns a [`Participant`] created from a [`Address`] you called
/// [`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
/// use uvth::ThreadPoolBuilder;
/// use veloren_network::{Network, Pid};
///
/// // Create a Network, listen on port `2000` TCP and opens returns their Pid
/// let network = Network::new(Pid::new(), ThreadPoolBuilder::new().build(), None);
/// block_on(async {
/// network
/// .listen(Address::Tcp("0.0.0.0:2000".parse().unwrap()))
/// .await?;
/// while let Some(participant) = network.connected().await? {
/// println!("Participant connected: {}", participant.remote_pid());
/// }
/// });
/// ```
///
/// [`Streams`]: crate::api::Stream
/// [`listen`]: crate::api::Network::listen
2020-03-22 13:47:21 +00:00
pub async fn connected ( & self ) -> Result < Arc < Participant > , NetworkError > {
2020-04-08 14:26:42 +00:00
let participant = self . connected_receiver . write ( ) . await . next ( ) . await ? ;
let participant = Arc ::new ( participant ) ;
self . participants
. write ( )
. await
. insert ( participant . remote_pid , participant . clone ( ) ) ;
Ok ( participant )
2019-12-20 13:56:01 +00:00
}
2020-05-10 02:07:46 +00:00
/// disconnecting a [`Participant`] where you move the last existing
/// [`Arc<Participant>`]. As the [`Network`] also holds [`Arc`] to the
/// [`Participant`], you need to provide the last [`Arc<Participant>`] and
/// are not allowed to keep others. If you do so the [`Participant`]
/// can't be disconnected properly. If you no longer have the respective
/// [`Participant`], try using the [`participants`] method to get it.
2020-05-15 12:29:17 +00:00
///
2020-05-10 02:07:46 +00:00
/// This function will wait for all [`Streams`] to properly close, including
2020-05-15 12:29:17 +00:00
/// all messages to be send before closing. If an error occurs with one
/// of the messavb
2020-05-10 02:07:46 +00:00
///
/// # Examples
/// ```rust
/// use uvth::ThreadPoolBuilder;
/// use veloren_network::{Network, Pid};
///
/// // Create a Network, listen on port `2000` TCP and opens returns their Pid and close connection.
/// let network = Network::new(Pid::new(), ThreadPoolBuilder::new().build(), None);
/// block_on(async {
/// network
/// .listen(Address::Tcp("0.0.0.0:2000".parse().unwrap()))
/// .await?;
/// while let Some(participant) = network.connected().await? {
/// println!("Participant connected: {}", participant.remote_pid());
/// network.disconnect(participant).await?;
/// }
/// });
/// ```
///
/// [`Arc<Participant>`]: crate::api::Participant
/// [`Streams`]: crate::api::Stream
/// [`participants`]: Network::participants
/// [`Arc`]: std::sync::Arc
2020-03-22 13:47:21 +00:00
pub async fn disconnect ( & self , participant : Arc < Participant > ) -> Result < ( ) , NetworkError > {
// Remove, Close and try_unwrap error when unwrap fails!
2020-04-08 14:26:42 +00:00
let pid = participant . remote_pid ;
debug! ( ? pid , " removing participant from network " ) ;
self . participants . write ( ) . await . remove ( & pid ) ? ;
2020-03-22 13:47:21 +00:00
participant . closed . store ( true , Ordering ::Relaxed ) ;
2020-01-13 16:53:28 +00:00
2020-05-15 12:29:17 +00:00
match Arc ::try_unwrap ( participant ) {
Err ( _ ) = > {
warn! (
" you are disconnecting and still keeping a reference to this participant, \
this is a bad idea . Participant will only be dropped when you drop your last \
reference "
) ;
} ,
Ok ( mut participant ) = > {
trace! ( " waiting now for participant to close " ) ;
let ( finished_sender , finished_receiver ) = oneshot ::channel ( ) ;
// we are deleting here asyncly before DROP is called. Because this is done
// nativly async, while drop needs an BLOCK! Drop will recognis
// that it has been delete here and don't try another double delete.
participant
. a2s_disconnect_s
. take ( )
. unwrap ( )
. send ( ( pid , finished_sender ) )
. await
. expect ( " something is wrong in internal scheduler coding " ) ;
let res = finished_receiver . await . unwrap ( ) ;
trace! ( " participant is now closed " ) ;
res ? ;
} ,
2020-03-22 13:47:21 +00:00
} ;
2020-05-15 12:29:17 +00:00
2020-03-22 13:47:21 +00:00
Ok ( ( ) )
2020-02-10 17:25:47 +00:00
}
2020-05-10 02:07:46 +00:00
/// returns a copy of all current connected [`Participants`]
///
/// [`Participants`]: crate::api::Participant
2020-04-08 14:26:42 +00:00
pub async fn participants ( & self ) -> HashMap < Pid , Arc < Participant > > {
self . participants . read ( ) . await . clone ( )
}
}
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-05-15 12:29:17 +00:00
a2b_steam_open_s : mpsc ::UnboundedSender < ( Prio , Promises , oneshot ::Sender < Stream > ) > ,
b2a_stream_opened_r : mpsc ::UnboundedReceiver < Stream > ,
a2s_disconnect_s : mpsc ::UnboundedSender < ( Pid , oneshot ::Sender < async_std ::io ::Result < ( ) > > ) > ,
2020-03-22 13:47:21 +00:00
) -> Self {
Self {
local_pid ,
remote_pid ,
2020-05-15 12:29:17 +00:00
a2b_steam_open_s : RwLock ::new ( a2b_steam_open_s ) ,
b2a_stream_opened_r : RwLock ::new ( b2a_stream_opened_r ) ,
2020-03-22 13:47:21 +00:00
closed : AtomicBool ::new ( false ) ,
2020-05-15 12:29:17 +00:00
a2s_disconnect_s : 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.
/// `PROMISES_ORDERED | PROMISES_CONSISTENCY` The Stream will then
/// guarantee that those promisses are met.
///
/// 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
/// use uvth::ThreadPoolBuilder;
/// use veloren_network::{Network, Pid, PROMISES_CONSISTENCY, PROMISES_ORDERED};
///
/// // Create a Network, connect on port 2000 and open a stream
/// let network = Network::new(Pid::new(), ThreadPoolBuilder::new().build(), None);
/// block_on(async {
/// let p1 = network
/// .connect(Address::Tcp("127.0.0.1:2000".parse().unwrap()))
/// .await?;
/// let _s1 = p1
/// .open(100, PROMISES_ORDERED | PROMISES_CONSISTENCY)
/// .await?;
/// });
/// ```
///
/// [`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-04-08 14:26:42 +00:00
//use this lock for now to make sure that only one open at a time is made,
// TODO: not sure if we can paralise that, check in future
2020-05-15 12:29:17 +00:00
let mut a2b_steam_open_s = self . a2b_steam_open_s . write ( ) . await ;
2020-04-08 14:26:42 +00:00
if self . closed . load ( Ordering ::Relaxed ) {
warn! ( ? self . remote_pid , " participant is closed but another open is tried on it " ) ;
return Err ( ParticipantError ::ParticipantClosed ) ;
}
2020-05-15 12:29:17 +00:00
let ( p2a_return_stream_s , p2a_return_stream_r ) = oneshot ::channel ( ) ;
if a2b_steam_open_s
. send ( ( prio , promises , p2a_return_stream_s ) )
2020-03-22 13:47:21 +00:00
. await
2020-04-08 14:26:42 +00:00
. is_err ( )
{
2020-05-15 12:29:17 +00:00
debug! ( ? self . remote_pid , " stream_open_sender failed, closing participant " ) ;
2020-04-08 14:26:42 +00:00
self . closed . store ( true , Ordering ::Relaxed ) ;
return Err ( ParticipantError ::ParticipantClosed ) ;
}
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-04-08 14:26:42 +00:00
self . closed . store ( true , Ordering ::Relaxed ) ;
Err ( ParticipantError ::ParticipantClosed )
} ,
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
/// use veloren_network::{Network, Pid, PROMISES_ORDERED, PROMISES_CONSISTENCY};
/// use uvth::ThreadPoolBuilder;
///
/// // Create a Network, connect on port 2000 and wait for the other side to open a stream
/// // Note: It's quite unusal to activly connect, but then wait on a stream to be connected, usually the Appication taking initiative want's to also create the first Stream.
/// let network = Network::new(Pid::new(), ThreadPoolBuilder::new().build(), None);
/// block_on(async {
/// let p1 = network.connect(Address::Tcp("127.0.0.1:2000".parse().unwrap())).await?;
/// let _s1 = p1.opened().await?;
/// });
/// ```
///
/// [`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-04-08 14:26:42 +00:00
//use this lock for now to make sure that only one open at a time is made,
// TODO: not sure if we can paralise that, check in future
2020-05-15 12:29:17 +00:00
let mut stream_opened_receiver = self . b2a_stream_opened_r . write ( ) . await ;
2020-04-08 14:26:42 +00:00
if self . closed . load ( Ordering ::Relaxed ) {
warn! ( ? self . remote_pid , " participant is closed but another open is tried on it " ) ;
return Err ( ParticipantError ::ParticipantClosed ) ;
}
match stream_opened_receiver . next ( ) . await {
Some ( stream ) = > {
let sid = stream . sid ;
debug! ( ? sid , ? self . remote_pid , " receive opened stream " ) ;
Ok ( stream )
} ,
None = > {
debug! ( ? self . remote_pid , " stream_opened_receiver failed, closing participant " ) ;
self . closed . store ( true , Ordering ::Relaxed ) ;
Err ( ParticipantError ::ParticipantClosed )
} ,
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-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-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-05-15 12:29:17 +00:00
a2b_msg_s : std ::sync ::mpsc ::Sender < ( Prio , Pid , Sid , OutGoingMessage ) > ,
b2a_msg_recv_r : mpsc ::UnboundedReceiver < InCommingMessage > ,
2020-04-08 14:26:42 +00:00
closed : Arc < AtomicBool > ,
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-05-15 12:29:17 +00:00
a2b_msg_s ,
b2a_msg_recv_r ,
2020-04-08 14:26:42 +00:00
closed ,
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.
///
/// Note when a `Stream` is dropped, 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.
///
/// # Example
/// ```rust
/// use futures::executor::block_on;
/// use veloren_network::{Network, Pid};
///
/// let network = Network::new(Pid::new(), ThreadPoolBuilder::new().build(), None);
/// block_on(async {
/// let participant_a = network.connected().await;
/// let mut stream_a = participant_a.opened().await;
/// //Send Message
/// stream_a.send("Hello World");
/// });
/// ```
///
/// [`send_raw`]: Stream::send_raw
/// [`recv`]: Stream::recv
/// [`Serialized`]: Serialize
2020-04-08 14:26:42 +00:00
pub fn send < M : Serialize > ( & mut self , msg : M ) -> Result < ( ) , StreamError > {
2020-04-24 10:56:04 +00:00
self . send_raw ( Arc ::new ( message ::serialize ( & msg ) ) )
}
2020-05-10 02:07:46 +00:00
/// This methods give the option to skip multiple calls of [`bincode`], e.g.
/// in case the same Message needs to send on multiple `Streams` to multiple
/// [`Participants`]. Other then that, the same rules apply than for
/// [`send`]
///
/// # Example
/// ```rust
/// use bincode;
/// use futures::executor::block_on;
/// use veloren_network::{Network, Pid};
///
/// let network = Network::new(Pid::new(), ThreadPoolBuilder::new().build(), None);
/// block_on(async {
/// 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_a.opened().await;
///
/// //Prepare Message and decode it
/// let msg = "Hello World";
/// let raw_msg = Arc::new(MessageBuffer {
/// data: bincode::serialize(&msg).unwrap(),
/// });
/// //Send same Message to multiple Streams
/// stream_a.send_raw(raw_msg.clone());
/// stream_b.send_raw(raw_msg.clone());
/// });
/// ```
///
/// [`send`]: Stream::send
/// [`Participants`]: crate::api::Participant
2020-04-24 10:56:04 +00:00
pub fn send_raw ( & mut self , messagebuffer : Arc < MessageBuffer > ) -> Result < ( ) , StreamError > {
2020-04-08 14:26:42 +00:00
if self . closed . load ( Ordering ::Relaxed ) {
return Err ( StreamError ::StreamClosed ) ;
}
2020-05-15 12:29:17 +00:00
//debug!(?messagebuffer, "sending a message");
self . a2b_msg_s
2020-03-22 13:47:21 +00:00
. send ( ( self . prio , self . pid , self . sid , OutGoingMessage {
2020-03-04 15:52:30 +00:00
buffer : messagebuffer ,
2020-03-04 00:37:36 +00:00
cursor : 0 ,
2020-03-22 13:47:21 +00:00
mid : self . mid ,
2020-03-04 00:37:36 +00:00
sid : self . sid ,
2020-04-08 14:26:42 +00:00
} ) ) ? ;
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-03-04 10:59:19 +00:00
pub async fn recv < M : DeserializeOwned > ( & mut self ) -> Result < M , StreamError > {
2020-04-24 10:56:04 +00:00
Ok ( message ::deserialize ( self . recv_raw ( ) . await ? ) )
}
2020-05-10 02:07:46 +00:00
/// the equivalent like [`send_raw`] but for [`recv`], no [`bincode`] is
/// executed for performance reasons.
///
/// [`send_raw`]: Stream::send_raw
/// [`recv`]: Stream::recv
2020-04-24 10:56:04 +00:00
pub async fn recv_raw ( & mut self ) -> Result < MessageBuffer , StreamError > {
2020-04-08 14:26:42 +00:00
//no need to access self.closed here, as when this stream is closed the Channel
// is closed which will trigger a None
2020-05-15 12:29:17 +00:00
let msg = self . b2a_msg_recv_r . next ( ) . await ? ;
info! ( ? msg , " delivering a message " ) ;
2020-04-24 10:56:04 +00:00
Ok ( msg . buffer )
2020-02-10 17:25:47 +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 Network {
fn drop ( & mut self ) {
2020-04-08 14:26:42 +00:00
let pid = self . local_pid ;
debug! ( ? pid , " shutting down Network " ) ;
2020-05-15 12:29:17 +00:00
debug! (
? pid ,
" shutting down Participants of Network, while we still have metrics "
) ;
task ::block_on ( async {
self . participants . write ( ) . await . clear ( ) ;
} ) ;
debug! ( ? pid , " shutting down Scheduler " ) ;
2020-04-08 14:26:42 +00:00
self . shutdown_sender
. take ( )
. unwrap ( )
. send ( ( ) )
. expect ( " scheduler is closed, but nobody other should be able to close it " ) ;
2020-05-15 12:29:17 +00:00
debug! ( ? pid , " participants have 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 ;
debug! ( ? pid , " shutting down Participant " ) ;
2020-05-15 12:29:17 +00:00
match self . a2s_disconnect_s . take ( ) {
None = > debug! (
? pid ,
" Participant has been shutdown cleanly, no further waiting is requiered! "
) ,
Some ( mut a2s_disconnect_s ) = > {
debug! (
? pid ,
" unclean shutdown detected, active waiting for client to be disconnected "
) ;
task ::block_on ( async {
let ( finished_sender , finished_receiver ) = oneshot ::channel ( ) ;
a2s_disconnect_s
. send ( ( self . remote_pid , finished_sender ) )
. await
. expect ( " something is wrong in internal scheduler coding " ) ;
match finished_receiver . await {
Ok ( Err ( e ) ) = > error! (
? pid ,
? e ,
" Error while dropping the participant, couldn't send all outgoing \
messages , dropping remaining "
) ,
Err ( e ) = > warn! (
? e ,
" //TODO i dont know why the finish doesnt work, i normally would \
expect to have sended a return message from the participant .. . \
ignoring to not caue a panic for now , please fix me "
) ,
_ = > ( ) ,
} ;
} ) ;
} ,
}
debug! ( ? pid , " network dropped " ) ;
2020-03-10 00:07:36 +00:00
}
}
impl Drop for Stream {
fn drop ( & mut self ) {
2020-04-08 14:26:42 +00:00
// a send if closed is unecessary but doesnt hurt, we must not crash here
2020-03-22 13:47:21 +00:00
if ! self . closed . load ( Ordering ::Relaxed ) {
2020-04-08 14:26:42 +00:00
let sid = self . sid ;
let pid = self . pid ;
debug! ( ? pid , ? sid , " shutting down Stream " ) ;
2020-05-15 12:29:17 +00:00
if task ::block_on ( self . a2b_close_stream_s . take ( ) . unwrap ( ) . send ( self . sid ) ) . is_err ( ) {
2020-04-08 14:26:42 +00:00
warn! (
" Other side got already dropped, probably due to timing, other side will \
handle this gracefully "
) ;
} ;
2020-05-15 12:29:17 +00:00
} else {
let sid = self . sid ;
let pid = self . pid ;
debug! ( ? pid , ? sid , " 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 {
let status = if self . closed . load ( Ordering ::Relaxed ) {
" [CLOSED] "
} else {
" [OPEN] "
} ;
write! (
f ,
" Participant {{{} local_pid: {:?}, remote_pid: {:?} }} " ,
status , & self . local_pid , & self . remote_pid ,
)
}
}
impl < T > From < std ::sync ::mpsc ::SendError < T > > for StreamError {
fn from ( _err : std ::sync ::mpsc ::SendError < T > ) -> Self { StreamError ::StreamClosed }
}
impl < T > From < std ::sync ::mpsc ::SendError < T > > for ParticipantError {
fn from ( _err : std ::sync ::mpsc ::SendError < T > ) -> Self { ParticipantError ::ParticipantClosed }
}
impl < T > From < std ::sync ::mpsc ::SendError < T > > for NetworkError {
fn from ( _err : std ::sync ::mpsc ::SendError < T > ) -> Self { NetworkError ::NetworkClosed }
}
impl From < async_std ::io ::Error > for NetworkError {
fn from ( err : async_std ::io ::Error ) -> Self { NetworkError ::ListenFailed ( err ) }
}
impl From < std ::option ::NoneError > for StreamError {
fn from ( _err : std ::option ::NoneError ) -> Self { StreamError ::StreamClosed }
}
impl From < std ::option ::NoneError > for ParticipantError {
fn from ( _err : std ::option ::NoneError ) -> Self { ParticipantError ::ParticipantClosed }
}
impl From < std ::option ::NoneError > for NetworkError {
fn from ( _err : std ::option ::NoneError ) -> Self { NetworkError ::NetworkClosed }
}
impl From < mpsc ::SendError > for ParticipantError {
fn from ( _err : mpsc ::SendError ) -> Self { ParticipantError ::ParticipantClosed }
}
impl From < mpsc ::SendError > for NetworkError {
fn from ( _err : mpsc ::SendError ) -> Self { NetworkError ::NetworkClosed }
}
impl From < oneshot ::Canceled > for ParticipantError {
fn from ( _err : oneshot ::Canceled ) -> Self { ParticipantError ::ParticipantClosed }
}
impl From < oneshot ::Canceled > for NetworkError {
fn from ( _err : oneshot ::Canceled ) -> Self { NetworkError ::NetworkClosed }
}