360 lines
12 KiB
Rust
360 lines
12 KiB
Rust
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<dyn Fn(VeilidUpdate) + Send + Sync>;
|
|
|
|
/// 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<ProtectedStore>,
|
|
pub table_store: Option<TableStore>,
|
|
#[cfg(feature = "unstable-blockstore")]
|
|
pub block_store: Option<BlockStore>,
|
|
pub crypto: Option<Crypto>,
|
|
pub attachment_manager: Option<AttachmentManager>,
|
|
pub storage_manager: Option<StorageManager>,
|
|
}
|
|
|
|
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<VeilidCoreContext> {
|
|
// 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<VeilidCoreContext> {
|
|
// 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<VeilidCoreContext> {
|
|
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<bool> = 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<VeilidAPI> {
|
|
// 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<VeilidAPI> {
|
|
// 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;
|
|
}
|