mirror of
https://gitlab.com/veloren/veloren.git
synced 2024-08-30 18:12:32 +00:00
03af9937cf
- completly switch to Bytes, even in api. speed up TCP by fak 2 - improve benchmarks - speed up mpsc metrics - gracefully handle shutdown by interpreting Ok(0) as tokio::tcpstream closed now. - fix hotloop in participants by adding `Some(n)` to fix endless handing. - fix closing bug by closing streams after `recv_mgr` is shutdown even if now shutdown is triggered locally. - fix prometheus - no longer throw when a `Stream` is dropped while participant still receives a msg for it. - fix the bandwith handling, TCP network send speed is up to 1.5GiB/s while recv is 150MiB/s - add documentation - tmp require rt-multi-threaded in client for tokio, to not fail cargo check this is prob stable, i tested over 1 hour. after that some optimisations in priomgr. and impl. propper bandwith. Speed is up to 2GB/s write and 150MB/s recv on a single core sync add documentation
179 lines
6.0 KiB
Rust
179 lines
6.0 KiB
Rust
//! Network Protocol
|
|
//!
|
|
//! a I/O-Free protocol for the veloren network crate.
|
|
//! This crate defines multiple different protocols over [`UnreliableDrain`] and
|
|
//! [`UnreliableSink`] traits, which allows it to define the behavior of a
|
|
//! protocol separated from the actual io.
|
|
//!
|
|
//! For example we define the TCP protocol on top of Drains and Sinks that can
|
|
//! send chunks of bytes. You can now implement your own Drain And Sink that
|
|
//! sends the data via tokio's or std's implementation. Or you just use a
|
|
//! std::mpsc::channel for unit tests without needing a actual tcp socket.
|
|
//!
|
|
//! This crate currently defines:
|
|
//! - TCP
|
|
//! - MPSC
|
|
//!
|
|
//! a UDP implementation will quickly follow, and it's also possible to abstract
|
|
//! over QUIC.
|
|
//!
|
|
//! warning: don't mix protocol, using the TCP variant for actual UDP socket
|
|
//! will result in dropped data using UDP with a TCP socket will be a waste of
|
|
//! resources.
|
|
//!
|
|
//! A *channel* in this crate is defined as a combination of *read* and *write*
|
|
//! protocol.
|
|
//!
|
|
//! # adding a protocol
|
|
//!
|
|
//! We start by defining our DataFormat. For most this is prob [`Vec<u8>`] or
|
|
//! [`Bytes`]. MPSC can directly send a msg without serialisation.
|
|
//!
|
|
//! Create 2 structs, one for the receiving and sending end. Based on a generic
|
|
//! Drain/Sink with your required DataFormat.
|
|
//! Implement the [`SendProtocol`] and [`RecvProtocol`] traits respectively.
|
|
//!
|
|
//! Implement the Handshake: [`InitProtocol`], alternatively you can also
|
|
//! implement `ReliableDrain` and `ReliableSink`, by this, you use the default
|
|
//! Handshake.
|
|
//!
|
|
//! This crate also contains consts and definitions for the network protocol.
|
|
//!
|
|
//! For an *example* see `TcpDrain` and `TcpSink` in the [tcp.rs](tcp.rs)
|
|
//!
|
|
//! [`UnreliableDrain`]: crate::UnreliableDrain
|
|
//! [`UnreliableSink`]: crate::UnreliableSink
|
|
//! [`Vec<u8>`]: std::vec::Vec
|
|
//! [`Bytes`]: bytes::Bytes
|
|
//! [`SendProtocol`]: crate::SendProtocol
|
|
//! [`RecvProtocol`]: crate::RecvProtocol
|
|
//! [`InitProtocol`]: crate::InitProtocol
|
|
|
|
mod event;
|
|
mod frame;
|
|
mod handshake;
|
|
mod message;
|
|
mod metrics;
|
|
mod mpsc;
|
|
mod prio;
|
|
mod tcp;
|
|
mod types;
|
|
|
|
pub use event::ProtocolEvent;
|
|
pub use metrics::ProtocolMetricCache;
|
|
#[cfg(feature = "metrics")]
|
|
pub use metrics::ProtocolMetrics;
|
|
pub use mpsc::{MpscMsg, MpscRecvProtocol, MpscSendProtocol};
|
|
pub use tcp::{TcpRecvProtocol, TcpSendProtocol};
|
|
pub use types::{
|
|
Bandwidth, Cid, Mid, Pid, Prio, Promises, Sid, HIGHEST_PRIO, VELOREN_NETWORK_VERSION,
|
|
};
|
|
|
|
///use at own risk, might change any time, for internal benchmarks
|
|
pub mod _internal {
|
|
pub use crate::frame::{ITFrame, OTFrame};
|
|
}
|
|
|
|
use async_trait::async_trait;
|
|
|
|
/// Handshake: Used to connect 2 Channels.
|
|
#[async_trait]
|
|
pub trait InitProtocol {
|
|
async fn initialize(
|
|
&mut self,
|
|
initializer: bool,
|
|
local_pid: Pid,
|
|
secret: u128,
|
|
) -> Result<(Pid, Sid, u128), InitProtocolError>;
|
|
}
|
|
|
|
/// Generic Network Send Protocol.
|
|
/// Implement this for your Protocol of choice ( tcp, udp, mpsc, quic)
|
|
/// Allows the creation/deletions of `Streams` and sending messages via
|
|
/// [`ProtocolEvent`].
|
|
///
|
|
/// A `Stream` MUST be bound to a specific Channel. You MUST NOT switch the
|
|
/// channel to send a stream mid air. We will provide takeover options for
|
|
/// Channel closure in the future to allow keeping a `Stream` over a broker
|
|
/// Channel.
|
|
///
|
|
/// [`ProtocolEvent`]: crate::ProtocolEvent
|
|
#[async_trait]
|
|
pub trait SendProtocol {
|
|
/// YOU MUST inform the `SendProtocol` by any Stream Open BEFORE using it in
|
|
/// `send` and Stream Close AFTER using it in `send` via this fn.
|
|
fn notify_from_recv(&mut self, event: ProtocolEvent);
|
|
/// Send a Event via this Protocol. The `SendProtocol` MAY require `flush`
|
|
/// to be called before actual data is send to the respective `Sink`.
|
|
async fn send(&mut self, event: ProtocolEvent) -> Result<(), ProtocolError>;
|
|
/// Flush all buffered messages according to their [`Prio`] and
|
|
/// [`Bandwidth`]. provide the current bandwidth budget (per second) as
|
|
/// well as the `dt` since last call. According to the budget the
|
|
/// respective messages will be flushed.
|
|
///
|
|
/// [`Prio`]: crate::Prio
|
|
/// [`Bandwidth`]: crate::Bandwidth
|
|
async fn flush(
|
|
&mut self,
|
|
bandwidth: Bandwidth,
|
|
dt: std::time::Duration,
|
|
) -> Result<(), ProtocolError>;
|
|
}
|
|
|
|
/// Generic Network Recv Protocol. See: [`SendProtocol`]
|
|
///
|
|
/// [`SendProtocol`]: crate::SendProtocol
|
|
#[async_trait]
|
|
pub trait RecvProtocol {
|
|
/// Either recv an event or fail the Protocol, once the Recv side is closed
|
|
/// it cannot recover from the error.
|
|
async fn recv(&mut self) -> Result<ProtocolEvent, ProtocolError>;
|
|
}
|
|
|
|
/// This crate makes use of UnreliableDrains, they are expected to provide the
|
|
/// same guarantees like their IO-counterpart. E.g. ordered messages for TCP and
|
|
/// nothing for UDP. The respective Protocol needs then to handle this.
|
|
/// This trait is an abstraction above multiple Drains, e.g. [`tokio`](https://tokio.rs) [`async-std`] [`std`] or even [`async-channel`]
|
|
///
|
|
/// [`async-std`]: async-std
|
|
/// [`std`]: std
|
|
/// [`async-channel`]: async-channel
|
|
#[async_trait]
|
|
pub trait UnreliableDrain: Send {
|
|
type DataFormat;
|
|
async fn send(&mut self, data: Self::DataFormat) -> Result<(), ProtocolError>;
|
|
}
|
|
|
|
/// Sink counterpart of [`UnreliableDrain`]
|
|
///
|
|
/// [`UnreliableDrain`]: crate::UnreliableDrain
|
|
#[async_trait]
|
|
pub trait UnreliableSink: Send {
|
|
type DataFormat;
|
|
async fn recv(&mut self) -> Result<Self::DataFormat, ProtocolError>;
|
|
}
|
|
|
|
/// All possible Errors that can happen during Handshake [`InitProtocol`]
|
|
///
|
|
/// [`InitProtocol`]: crate::InitProtocol
|
|
#[derive(Debug, PartialEq)]
|
|
pub enum InitProtocolError {
|
|
Closed,
|
|
WrongMagicNumber([u8; 7]),
|
|
WrongVersion([u32; 3]),
|
|
}
|
|
|
|
/// When you return closed you must stay closed!
|
|
#[derive(Debug, PartialEq)]
|
|
pub enum ProtocolError {
|
|
Closed,
|
|
}
|
|
|
|
impl From<ProtocolError> for InitProtocolError {
|
|
fn from(err: ProtocolError) -> Self {
|
|
match err {
|
|
ProtocolError::Closed => InitProtocolError::Closed,
|
|
}
|
|
}
|
|
}
|