use crate::*; use core::fmt; use dht::receipt::*; use futures_util::stream::{FuturesUnordered, StreamExt}; use network_manager::*; use xx::*; #[derive(Clone, Debug, Copy, PartialEq, Eq)] pub enum ReceiptEvent { RETURNED, EXPIRED, CANCELLED, } cfg_if! { if #[cfg(target_arch = "wasm32")] { pub trait ReceiptCallback: 'static { fn call( &self, event: ReceiptEvent, nonce: ReceiptNonce, returns_so_far: u32, expected_returns: u32, ) -> SystemPinBoxFuture<()>; } impl ReceiptCallback for T where T: Fn(ReceiptEvent, ReceiptNonce, u32, u32) -> F + 'static, F: Future + 'static, { fn call( &self, event: ReceiptEvent, nonce: ReceiptNonce, returns_so_far: u32, expected_returns: u32, ) -> SystemPinBoxFuture<()> { Box::pin(self(event, nonce, returns_so_far, expected_returns)) } } } else { pub trait ReceiptCallback: Send + 'static { fn call( &self, event: ReceiptEvent, nonce: ReceiptNonce, returns_so_far: u32, expected_returns: u32, ) -> SystemPinBoxFuture<()>; } impl ReceiptCallback for T where T: Fn(ReceiptEvent, ReceiptNonce, u32, u32) -> F + Send + 'static, F: Future + Send + 'static { fn call( &self, event: ReceiptEvent, nonce: ReceiptNonce, returns_so_far: u32, expected_returns: u32, ) -> SystemPinBoxFuture<()> { Box::pin(self(event, nonce, returns_so_far, expected_returns)) } } } } type ReceiptCallbackType = Box; type ReceiptSingleShotType = SingleShotEventual; enum ReceiptRecordCallbackType { Normal(ReceiptCallbackType), SingleShot(Option), } impl fmt::Debug for ReceiptRecordCallbackType { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { write!( f, "ReceiptRecordCallbackType::{}", match self { Self::Normal(_) => "Normal".to_owned(), Self::SingleShot(_) => "SingleShot".to_owned(), } ) } } pub struct ReceiptRecord { expiration_ts: u64, nonce: ReceiptNonce, expected_returns: u32, returns_so_far: u32, receipt_callback: ReceiptRecordCallbackType, } impl fmt::Debug for ReceiptRecord { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { f.debug_struct("ReceiptRecord") .field("expiration_ts", &self.expiration_ts) .field("nonce", &self.nonce) .field("expected_returns", &self.expected_returns) .field("returns_so_far", &self.returns_so_far) .field("receipt_callback", &self.receipt_callback) .finish() } } impl ReceiptRecord { pub fn from_receipt( receipt: &Receipt, expiration_ts: u64, expected_returns: u32, receipt_callback: impl ReceiptCallback, ) -> Self { Self { expiration_ts: expiration_ts, nonce: receipt.get_nonce(), expected_returns: expected_returns, returns_so_far: 0u32, receipt_callback: ReceiptRecordCallbackType::Normal(Box::new(receipt_callback)), } } pub fn from_single_shot_receipt( receipt: &Receipt, expiration_ts: u64, eventual: ReceiptSingleShotType, ) -> Self { Self { expiration_ts: expiration_ts, nonce: receipt.get_nonce(), returns_so_far: 0u32, expected_returns: 1u32, receipt_callback: ReceiptRecordCallbackType::SingleShot(Some(eventual)), } } } /* XXX: may be useful for O(1) timestamp expiration #[derive(Clone, Debug)] struct ReceiptRecordTimestampSort { expiration_ts: u64, record: Arc>, } impl PartialEq for ReceiptRecordTimestampSort { fn eq(&self, other: &ReceiptRecordTimestampSort) -> bool { self.expiration_ts == other.expiration_ts } } impl Eq for ReceiptRecordTimestampSort {} impl Ord for ReceiptRecordTimestampSort { fn cmp(&self, other: &Self) -> std::cmp::Ordering { self.expiration_ts.cmp(&other.expiration_ts).reverse() } } impl PartialOrd for ReceiptRecordTimestampSort { fn partial_cmp(&self, other: &Self) -> Option { Some(self.cmp(&other)) } } */ /////////////////////////////////// pub struct ReceiptManagerInner { network_manager: NetworkManager, receipts_by_nonce: BTreeMap>>, next_oldest_ts: Option, timeout_task: SingleFuture<()>, } #[derive(Clone)] pub struct ReceiptManager { inner: Arc>, } impl ReceiptManager { fn new_inner(network_manager: NetworkManager) -> ReceiptManagerInner { ReceiptManagerInner { network_manager: network_manager, receipts_by_nonce: BTreeMap::new(), next_oldest_ts: None, timeout_task: SingleFuture::new(), } } pub fn new(network_manager: NetworkManager) -> Self { Self { inner: Arc::new(Mutex::new(Self::new_inner(network_manager))), } } pub fn network_manager(&self) -> NetworkManager { self.inner.lock().network_manager.clone() } pub async fn startup(&self) -> Result<(), String> { // Retrieve config /* { let config = self.core().config(); let c = config.get(); let mut inner = self.inner.lock(); inner.max_server_signal_leases = c.network.leases.max_server_signal_leases as usize; inner.max_server_relay_leases = c.network.leases.max_server_relay_leases as usize; inner.max_client_signal_leases = c.network.leases.max_client_signal_leases as usize; inner.max_client_relay_leases = c.network.leases.max_client_relay_leases as usize; } */ Ok(()) } fn perform_callback( evt: ReceiptEvent, record_mut: &mut ReceiptRecord, ) -> Option> { match &mut record_mut.receipt_callback { ReceiptRecordCallbackType::Normal(callback) => Some(callback.call( evt, record_mut.nonce, record_mut.returns_so_far, record_mut.expected_returns, )), ReceiptRecordCallbackType::SingleShot(eventual) => { // resolve this eventual with the receiptevent // don't need to wait for the instance to receive it // because this can only happen once if let Some(eventual) = eventual.take() { eventual.resolve(evt); } None } } } pub async fn timeout_task_routine(self, now: u64) { // Go through all receipts and build a list of expired nonces let mut new_next_oldest_ts: Option = None; let mut expired_records = Vec::new(); { let mut inner = self.inner.lock(); let mut expired_nonces = Vec::new(); for (k, v) in &inner.receipts_by_nonce { let receipt_inner = v.lock(); if receipt_inner.expiration_ts <= now { // Expire this receipt expired_nonces.push(k.clone()); } else if new_next_oldest_ts.is_none() || receipt_inner.expiration_ts < new_next_oldest_ts.unwrap() { // Mark the next oldest timestamp we would need to take action on as we go through everything new_next_oldest_ts = Some(receipt_inner.expiration_ts); } } if expired_nonces.len() == 0 { return; } // Now remove the expired receipts for e in expired_nonces { let expired_record = inner .receipts_by_nonce .remove(&e) .expect("key should exist"); expired_records.push(expired_record); } // Update the next oldest timestamp inner.next_oldest_ts = new_next_oldest_ts; } let mut callbacks = FuturesUnordered::new(); for expired_record in expired_records { let mut expired_record_mut = expired_record.lock(); if let Some(callback) = Self::perform_callback(ReceiptEvent::EXPIRED, &mut expired_record_mut) { callbacks.push(callback) } } // Wait on all the multi-call callbacks while callbacks.next().await.is_some() {} } pub async fn tick(&self) -> Result<(), String> { let (next_oldest_ts, timeout_task) = { let inner = self.inner.lock(); (inner.next_oldest_ts, inner.timeout_task.clone()) }; let now = intf::get_timestamp(); // If we have at least one timestamp to expire, lets do it if let Some(next_oldest_ts) = next_oldest_ts { if now >= next_oldest_ts { // Single-spawn the timeout task routine let _ = timeout_task .single_spawn(self.clone().timeout_task_routine(now)) .await; } } Ok(()) } pub async fn shutdown(&self) { let network_manager = self.network_manager(); *self.inner.lock() = Self::new_inner(network_manager); } pub fn record_receipt( &self, receipt: Receipt, expiration: u64, expected_returns: u32, callback: impl ReceiptCallback, ) { let record = Arc::new(Mutex::new(ReceiptRecord::from_receipt( &receipt, expiration, expected_returns, callback, ))); let mut inner = self.inner.lock(); inner .receipts_by_nonce .insert(receipt.get_nonce(), record.clone()); } pub fn record_single_shot_receipt( &self, receipt: Receipt, expiration: u64, eventual: ReceiptSingleShotType, ) { let record = Arc::new(Mutex::new(ReceiptRecord::from_single_shot_receipt( &receipt, expiration, eventual, ))); let mut inner = self.inner.lock(); inner .receipts_by_nonce .insert(receipt.get_nonce(), record.clone()); } fn update_next_oldest_timestamp(inner: &mut ReceiptManagerInner) { // Update the next oldest timestamp let mut new_next_oldest_ts: Option = None; for (_, v) in &inner.receipts_by_nonce { let receipt_inner = v.lock(); if new_next_oldest_ts.is_none() || receipt_inner.expiration_ts < new_next_oldest_ts.unwrap() { // Mark the next oldest timestamp we would need to take action on as we go through everything new_next_oldest_ts = Some(receipt_inner.expiration_ts); } } inner.next_oldest_ts = new_next_oldest_ts; } pub async fn cancel_receipt(&self, nonce: &ReceiptNonce) -> Result<(), String> { // Remove the record let record = { let mut inner = self.inner.lock(); let record = match inner.receipts_by_nonce.remove(nonce) { Some(r) => r, None => { return Err("receipt not recorded".to_owned()); } }; Self::update_next_oldest_timestamp(&mut *inner); record }; // Generate a cancelled callback let callback_future = { let mut record_mut = record.lock(); Self::perform_callback(ReceiptEvent::CANCELLED, &mut record_mut) }; // Issue the callback if let Some(callback_future) = callback_future { callback_future.await; } Ok(()) } pub async fn handle_receipt(&self, receipt: Receipt) -> Result<(), String> { // Increment return count let callback_future = { // Look up the receipt record from the nonce let mut inner = self.inner.lock(); let record = match inner.receipts_by_nonce.get(&receipt.get_nonce()) { Some(r) => r.clone(), None => { return Err("receipt not recorded".to_owned()); } }; // Generate the callback future let mut record_mut = record.lock(); record_mut.returns_so_far += 1; let callback_future = Self::perform_callback(ReceiptEvent::RETURNED, &mut record_mut); // Remove the record if we're done if record_mut.returns_so_far == record_mut.expected_returns { inner.receipts_by_nonce.remove(&receipt.get_nonce()); Self::update_next_oldest_timestamp(&mut *inner); } callback_future }; // Issue the callback if let Some(callback_future) = callback_future { callback_future.await; } Ok(()) } }