feat(network): implement server side KeepAlive (#376)

This commit is contained in:
Harper 2024-01-08 10:50:37 +00:00 committed by GitHub
parent ca27aa91ac
commit a6a6ffcfed
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
5 changed files with 165 additions and 23 deletions

View file

@ -125,11 +125,12 @@ impl PeerClient {
/// Server of N2N Ouroboros
pub struct PeerServer {
plexer: RunningPlexer,
handshake: handshake::N2NServer,
chainsync: chainsync::N2NServer,
blockfetch: blockfetch::Server,
txsubmission: txsubmission::Server,
pub plexer: RunningPlexer,
pub handshake: handshake::N2NServer,
pub chainsync: chainsync::N2NServer,
pub blockfetch: blockfetch::Server,
pub txsubmission: txsubmission::Server,
pub keepalive: keepalive::Server,
accepted_address: Option<SocketAddr>,
accepted_version: Option<u64>,
}
@ -142,11 +143,13 @@ impl PeerServer {
let cs_channel = plexer.subscribe_server(PROTOCOL_N2N_CHAIN_SYNC);
let bf_channel = plexer.subscribe_server(PROTOCOL_N2N_BLOCK_FETCH);
let txsub_channel = plexer.subscribe_server(PROTOCOL_N2N_TX_SUBMISSION);
let keepalive_channel = plexer.subscribe_server(PROTOCOL_N2N_KEEP_ALIVE);
let hs = handshake::N2NServer::new(hs_channel);
let cs = chainsync::N2NServer::new(cs_channel);
let bf = blockfetch::Server::new(bf_channel);
let txsub = txsubmission::Server::new(txsub_channel);
let keepalive = keepalive::Server::new(keepalive_channel);
let plexer = plexer.spawn();
@ -156,6 +159,7 @@ impl PeerServer {
chainsync: cs,
blockfetch: bf,
txsubmission: txsub,
keepalive,
accepted_address: None,
accepted_version: None,
}
@ -200,6 +204,10 @@ impl PeerServer {
&mut self.txsubmission
}
pub fn keepalive(&mut self) -> &mut keepalive::Server {
&mut self.keepalive
}
pub async fn abort(self) {
self.plexer.abort().await
}

View file

@ -7,7 +7,7 @@ use super::protocol::*;
use crate::multiplexer;
#[derive(Error, Debug)]
pub enum Error {
pub enum ClientError {
#[error("attempted to receive message while agency is ours")]
AgencyIsOurs,
@ -58,54 +58,57 @@ impl Client {
}
}
fn assert_agency_is_ours(&self) -> Result<(), Error> {
fn assert_agency_is_ours(&self) -> Result<(), ClientError> {
if !self.has_agency() {
Err(Error::AgencyIsTheirs)
Err(ClientError::AgencyIsTheirs)
} else {
Ok(())
}
}
fn assert_agency_is_theirs(&self) -> Result<(), Error> {
fn assert_agency_is_theirs(&self) -> Result<(), ClientError> {
if self.has_agency() {
Err(Error::AgencyIsOurs)
Err(ClientError::AgencyIsOurs)
} else {
Ok(())
}
}
fn assert_outbound_state(&self, msg: &Message) -> Result<(), Error> {
fn assert_outbound_state(&self, msg: &Message) -> Result<(), ClientError> {
match (&self.0, msg) {
(State::Client, Message::KeepAlive(..)) => Ok(()),
(State::Client, Message::Done) => Ok(()),
_ => Err(Error::InvalidOutbound),
_ => Err(ClientError::InvalidOutbound),
}
}
fn assert_inbound_state(&self, msg: &Message) -> Result<(), Error> {
fn assert_inbound_state(&self, msg: &Message) -> Result<(), ClientError> {
match (&self.0, msg) {
(State::Server, Message::ResponseKeepAlive(..)) => Ok(()),
_ => Err(Error::InvalidInbound),
_ => Err(ClientError::InvalidInbound),
}
}
pub async fn send_message(&mut self, msg: &Message) -> Result<(), Error> {
pub async fn send_message(&mut self, msg: &Message) -> Result<(), ClientError> {
self.assert_agency_is_ours()?;
self.assert_outbound_state(msg)?;
self.1.send_msg_chunks(msg).await.map_err(Error::Plexer)?;
self.1
.send_msg_chunks(msg)
.await
.map_err(ClientError::Plexer)?;
Ok(())
}
pub async fn recv_message(&mut self) -> Result<Message, Error> {
pub async fn recv_message(&mut self) -> Result<Message, ClientError> {
self.assert_agency_is_theirs()?;
let msg = self.1.recv_full_msg().await.map_err(Error::Plexer)?;
let msg = self.1.recv_full_msg().await.map_err(ClientError::Plexer)?;
self.assert_inbound_state(&msg)?;
Ok(msg)
}
pub async fn send_keepalive(&mut self) -> Result<(), Error> {
pub async fn send_keepalive(&mut self) -> Result<(), ClientError> {
// generate random cookie value
let cookie = rand::thread_rng().gen::<KeepAliveCookie>();
let msg = Message::KeepAlive(cookie);
@ -119,7 +122,7 @@ impl Client {
Ok(())
}
async fn recv_while_sending_keepalive(&mut self) -> Result<(), Error> {
async fn recv_while_sending_keepalive(&mut self) -> Result<(), ClientError> {
match self.recv_message().await? {
Message::ResponseKeepAlive(cookie) => {
debug!("received keepalive response with cookie {}", cookie);
@ -127,10 +130,10 @@ impl Client {
self.0 = State::Client;
Ok(())
} else {
Err(Error::KeepAliveCookieMismatch)
Err(ClientError::KeepAliveCookieMismatch)
}
}
_ => Err(Error::InvalidInbound),
_ => Err(ClientError::InvalidInbound),
}
}
}

View file

@ -1,6 +1,8 @@
mod client;
mod codec;
mod protocol;
mod server;
pub use client::*;
pub use protocol::*;
pub use server::*;

View file

@ -0,0 +1,129 @@
use std::fmt::Debug;
use thiserror::*;
use tracing::debug;
use super::protocol::*;
use crate::multiplexer;
#[derive(Error, Debug)]
pub enum ServerError {
#[error("attempted to receive message while agency is ours")]
AgencyIsOurs,
#[error("attempted to send message while agency is theirs")]
AgencyIsTheirs,
#[error("inbound message is not valid for current state")]
InvalidInbound,
#[error("outbound message is not valid for current state")]
InvalidOutbound,
#[error("error while sending or receiving data through the channel")]
Plexer(multiplexer::Error),
}
pub struct Server(State, multiplexer::ChannelBuffer);
impl Server {
pub fn new(channel: multiplexer::AgentChannel) -> Self {
Self(State::Client, multiplexer::ChannelBuffer::new(channel))
}
pub fn state(&self) -> &State {
&self.0
}
pub fn is_done(&self) -> bool {
self.0 == State::Done
}
fn has_agency(&self) -> bool {
match &self.0 {
State::Client => false,
State::Server => true,
State::Done => false,
}
}
fn assert_agency_is_ours(&self) -> Result<(), ServerError> {
if !self.has_agency() {
Err(ServerError::AgencyIsTheirs)
} else {
Ok(())
}
}
fn assert_agency_is_theirs(&self) -> Result<(), ServerError> {
if self.has_agency() {
Err(ServerError::AgencyIsOurs)
} else {
Ok(())
}
}
fn assert_outbound_state(&self, msg: &Message) -> Result<(), ServerError> {
match (&self.0, msg) {
(State::Server, Message::ResponseKeepAlive(..)) => Ok(()),
_ => Err(ServerError::InvalidOutbound),
}
}
fn assert_inbound_state(&self, msg: &Message) -> Result<(), ServerError> {
match (&self.0, msg) {
(State::Client, Message::KeepAlive(..)) => Ok(()),
(State::Client, Message::Done) => Ok(()),
_ => Err(ServerError::InvalidInbound),
}
}
pub async fn send_message(&mut self, msg: &Message) -> Result<(), ServerError> {
self.assert_agency_is_ours()?;
self.assert_outbound_state(msg)?;
self.1
.send_msg_chunks(msg)
.await
.map_err(ServerError::Plexer)?;
Ok(())
}
pub async fn recv_message(&mut self) -> Result<Message, ServerError> {
self.assert_agency_is_theirs()?;
let msg = self.1.recv_full_msg().await.map_err(ServerError::Plexer)?;
self.assert_inbound_state(&msg)?;
Ok(msg)
}
pub async fn send_keepalive_response(
&mut self,
cookie: KeepAliveCookie,
) -> Result<(), ServerError> {
let msg = Message::ResponseKeepAlive(cookie);
self.send_message(&msg).await?;
self.0 = State::Client;
debug!("sent keepalive response message with cookie {}", cookie);
Ok(())
}
pub async fn keepalive_receive_and_respond(&mut self) -> Result<Option<()>, ServerError> {
match self.recv_message().await? {
Message::KeepAlive(cookie) => {
debug!("received keepalive message with cookie {}", cookie);
self.0 = State::Server;
Some(self.send_keepalive_response(cookie).await).transpose()
}
Message::Done => {
debug!("client sent done message in keepalive protocol");
self.0 = State::Done;
Ok(None)
}
_ => Err(ServerError::InvalidInbound),
}
}
}