use crate::api_tracing_layer::*; use crate::attachment_manager::*; use crate::crypto::Crypto; use crate::storage_manager::*; use crate::veilid_api::*; use crate::veilid_config::*; use crate::*; pub type UpdateCallback = Arc; /// Internal services startup mechanism /// Ensures that everything is started up, and shut down in the right order /// and provides an atomic state for if the system is properly operational struct ServicesContext { pub config: VeilidConfig, pub update_callback: UpdateCallback, pub protected_store: Option, pub table_store: Option, #[cfg(feature = "unstable-blockstore")] pub block_store: Option, pub crypto: Option, pub attachment_manager: Option, pub storage_manager: Option, } impl ServicesContext { pub fn new_empty(config: VeilidConfig, update_callback: UpdateCallback) -> Self { Self { config, update_callback, protected_store: None, table_store: None, #[cfg(feature = "unstable-blockstore")] block_store: None, crypto: None, attachment_manager: None, storage_manager: None, } } pub fn new_full( config: VeilidConfig, update_callback: UpdateCallback, protected_store: ProtectedStore, table_store: TableStore, #[cfg(feature = "unstable-blockstore")] block_store: BlockStore, crypto: Crypto, attachment_manager: AttachmentManager, storage_manager: StorageManager, ) -> Self { Self { config, update_callback, protected_store: Some(protected_store), table_store: Some(table_store), #[cfg(feature = "unstable-blockstore")] block_store: Some(block_store), crypto: Some(crypto), attachment_manager: Some(attachment_manager), storage_manager: Some(storage_manager), } } #[instrument(err, skip_all)] pub async fn startup(&mut self) -> EyreResult<()> { info!("Veilid API starting up"); info!("init api tracing"); ApiTracingLayer::init(self.update_callback.clone()).await; // Set up protected store trace!("init protected store"); let protected_store = ProtectedStore::new(self.config.clone()); if let Err(e) = protected_store.init().await { error!("failed to init protected store: {}", e); self.shutdown().await; return Err(e); } self.protected_store = Some(protected_store.clone()); // Set up tablestore and crypto system trace!("create table store and crypto system"); let table_store = TableStore::new(self.config.clone(), protected_store.clone()); let crypto = Crypto::new(self.config.clone(), table_store.clone()); table_store.set_crypto(crypto.clone()); // Initialize table store first, so crypto code can load caches // Tablestore can use crypto during init, just not any cached operations or things // that require flushing back to the tablestore trace!("init table store"); if let Err(e) = table_store.init().await { error!("failed to init table store: {}", e); self.shutdown().await; return Err(e); } self.table_store = Some(table_store.clone()); // Set up crypto trace!("init crypto"); if let Err(e) = crypto.init().await { error!("failed to init crypto: {}", e); self.shutdown().await; return Err(e); } self.crypto = Some(crypto.clone()); // Set up block store #[cfg(feature = "unstable-blockstore")] { trace!("init block store"); let block_store = BlockStore::new(self.config.clone()); if let Err(e) = block_store.init().await { error!("failed to init block store: {}", e); self.shutdown().await; return Err(e); } self.block_store = Some(block_store.clone()); } // Set up storage manager trace!("init storage manager"); let storage_manager = StorageManager::new( self.config.clone(), self.crypto.clone().unwrap(), self.table_store.clone().unwrap(), #[cfg(feature = "unstable-blockstore")] self.block_store.clone().unwrap(), ); if let Err(e) = storage_manager.init().await { error!("failed to init storage manager: {}", e); self.shutdown().await; return Err(e); } self.storage_manager = Some(storage_manager.clone()); // Set up attachment manager trace!("init attachment manager"); let update_callback = self.update_callback.clone(); let attachment_manager = AttachmentManager::new( self.config.clone(), storage_manager, protected_store, table_store, #[cfg(feature = "unstable-blockstore")] block_store, crypto, ); if let Err(e) = attachment_manager.init(update_callback).await { error!("failed to init attachment manager: {}", e); self.shutdown().await; return Err(e); } self.attachment_manager = Some(attachment_manager); info!("Veilid API startup complete"); Ok(()) } #[instrument(skip_all)] pub async fn shutdown(&mut self) { info!("Veilid API shutting down"); if let Some(attachment_manager) = &mut self.attachment_manager { trace!("terminate attachment manager"); attachment_manager.terminate().await; } if let Some(storage_manager) = &mut self.storage_manager { trace!("terminate storage manager"); storage_manager.terminate().await; } #[cfg(feature = "unstable-blockstore")] if let Some(block_store) = &mut self.block_store { trace!("terminate block store"); block_store.terminate().await; } if let Some(crypto) = &mut self.crypto { trace!("terminate crypto"); crypto.terminate().await; } if let Some(table_store) = &mut self.table_store { trace!("terminate table store"); table_store.terminate().await; } if let Some(protected_store) = &mut self.protected_store { trace!("terminate protected store"); protected_store.terminate().await; } info!("Veilid API shutdown complete"); // api logger terminate is idempotent ApiTracingLayer::terminate().await; // send final shutdown update (self.update_callback)(VeilidUpdate::Shutdown); } } ///////////////////////////////////////////////////////////////////////////// /// pub struct VeilidCoreContext { pub config: VeilidConfig, pub update_callback: UpdateCallback, // Services pub storage_manager: StorageManager, pub protected_store: ProtectedStore, pub table_store: TableStore, #[cfg(feature = "unstable-blockstore")] pub block_store: BlockStore, pub crypto: Crypto, pub attachment_manager: AttachmentManager, } impl VeilidCoreContext { #[instrument(err, skip_all)] async fn new_with_config_callback( update_callback: UpdateCallback, config_callback: ConfigCallback, ) -> VeilidAPIResult { // Set up config from callback trace!("setup config with callback"); let mut config = VeilidConfig::new(); config.setup(config_callback, update_callback.clone())?; Self::new_common(update_callback, config).await } #[instrument(err, skip_all)] async fn new_with_config_json( update_callback: UpdateCallback, config_json: String, ) -> VeilidAPIResult { // Set up config from callback trace!("setup config with json"); let mut config = VeilidConfig::new(); config.setup_from_json(config_json, update_callback.clone())?; Self::new_common(update_callback, config).await } #[instrument(err, skip_all)] async fn new_common( update_callback: UpdateCallback, config: VeilidConfig, ) -> VeilidAPIResult { cfg_if! { if #[cfg(target_os = "android")] { if !crate::intf::android::is_android_ready() { apibail_internal!("Android globals are not set up"); } } } let mut sc = ServicesContext::new_empty(config.clone(), update_callback); sc.startup().await.map_err(VeilidAPIError::generic)?; Ok(VeilidCoreContext { config: sc.config, update_callback: sc.update_callback, storage_manager: sc.storage_manager.unwrap(), protected_store: sc.protected_store.unwrap(), table_store: sc.table_store.unwrap(), #[cfg(feature = "unstable-blockstore")] block_store: sc.block_store.unwrap(), crypto: sc.crypto.unwrap(), attachment_manager: sc.attachment_manager.unwrap(), }) } #[instrument(skip_all)] async fn shutdown(self) { let mut sc = ServicesContext::new_full( self.config.clone(), self.update_callback.clone(), self.protected_store, self.table_store, #[cfg(feature = "unstable-blockstore")] self.block_store, self.crypto, self.attachment_manager, self.storage_manager, ); sc.shutdown().await; } } ///////////////////////////////////////////////////////////////////////////// lazy_static::lazy_static! { static ref INITIALIZED: AsyncMutex = AsyncMutex::new(false); } /// Initialize a Veilid node. /// /// Must be called only once at the start of an application /// /// * `update_callback` - called when internal state of the Veilid node changes, for example, when app-level messages are received, when private routes die and need to be reallocated, or when routing table states change /// * `config_callback` - called at startup to supply a configuration object directly to Veilid /// /// Returns a [VeilidAPI] object that can be used to operate the node #[instrument(err, skip_all)] pub async fn api_startup( update_callback: UpdateCallback, config_callback: ConfigCallback, ) -> VeilidAPIResult { // See if we have an API started up already let mut initialized_lock = INITIALIZED.lock().await; if *initialized_lock { apibail_already_initialized!(); } // Create core context let context = VeilidCoreContext::new_with_config_callback(update_callback, config_callback).await?; // Return an API object around our context let veilid_api = VeilidAPI::new(context); *initialized_lock = true; Ok(veilid_api) } /// Initialize a Veilid node, with the configuration in JSON format /// /// Must be called only once at the start of an application /// /// * `update_callback` - called when internal state of the Veilid node changes, for example, when app-level messages are received, when private routes die and need to be reallocated, or when routing table states change /// * `config_json` - called at startup to supply a JSON configuration object /// /// Returns a [VeilidAPI] object that can be used to operate the node #[instrument(err, skip_all)] pub async fn api_startup_json( update_callback: UpdateCallback, config_json: String, ) -> VeilidAPIResult { // See if we have an API started up already let mut initialized_lock = INITIALIZED.lock().await; if *initialized_lock { apibail_already_initialized!(); } // Create core context let context = VeilidCoreContext::new_with_config_json(update_callback, config_json).await?; // Return an API object around our context let veilid_api = VeilidAPI::new(context); *initialized_lock = true; Ok(veilid_api) } #[instrument(skip_all)] pub(crate) async fn api_shutdown(context: VeilidCoreContext) { let mut initialized_lock = INITIALIZED.lock().await; context.shutdown().await; *initialized_lock = false; }