use crate::intf::*; use crate::xx::*; use crate::*; #[derive(Clone, Debug)] pub struct ConnectionTableEntry { pub conn: NetworkConnection, pub established_time: u64, pub last_message_sent_time: Option, pub last_message_recv_time: Option, pub stopper: Eventual, } impl PartialEq for ConnectionTableEntry { fn eq(&self, other: &ConnectionTableEntry) -> bool { if self.conn != other.conn { return false; } if self.established_time != other.established_time { return false; } if self.last_message_sent_time != other.last_message_sent_time { return false; } if self.last_message_recv_time != other.last_message_recv_time { return false; } true } } #[derive(Debug)] pub struct ConnectionTableInner { conn_by_addr: BTreeMap, } #[derive(Clone)] pub struct ConnectionTable { inner: Arc>, } impl core::fmt::Debug for ConnectionTable { fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { f.debug_struct("ConnectionTable") .field("inner", &*self.inner.lock()) .finish() } } impl Default for ConnectionTable { fn default() -> Self { Self::new() } } impl ConnectionTable { pub fn new() -> Self { Self { inner: Arc::new(Mutex::new(ConnectionTableInner { conn_by_addr: BTreeMap::new(), })), } } pub fn add_connection( &self, descriptor: ConnectionDescriptor, conn: NetworkConnection, ) -> Result { trace!("descriptor: {:?}", descriptor); assert_ne!( descriptor.protocol_type(), ProtocolType::UDP, "Only connection oriented protocols go in the table!" ); let mut inner = self.inner.lock(); if inner.conn_by_addr.contains_key(&descriptor) { return Err(format!( "Connection already added to table: {:?}", descriptor )); } let timestamp = get_timestamp(); let entry = ConnectionTableEntry { conn, established_time: timestamp, last_message_sent_time: None, last_message_recv_time: None, stopper: Eventual::new(), }; let res = inner.conn_by_addr.insert(descriptor, entry.clone()); assert!(res.is_none()); Ok(entry) } pub fn get_connection( &self, descriptor: &ConnectionDescriptor, ) -> Option { let inner = self.inner.lock(); inner.conn_by_addr.get(descriptor).cloned() } pub fn connection_count(&self) -> usize { let inner = self.inner.lock(); inner.conn_by_addr.len() } pub fn remove_connection( &self, descriptor: &ConnectionDescriptor, ) -> Result { trace!("descriptor: {:?}", descriptor); let mut inner = self.inner.lock(); let res = inner.conn_by_addr.remove(descriptor); match res { Some(v) => Ok(v), None => Err(format!("Connection not in table: {:?}", descriptor)), } } }