miden_node_store/server/
mod.rsuse std::{net::ToSocketAddrs, sync::Arc};
use miden_node_proto::generated::store::api_server;
use miden_node_utils::errors::ApiError;
use tokio::net::TcpListener;
use tokio_stream::wrappers::TcpListenerStream;
use tracing::info;
use crate::{blocks::BlockStore, config::StoreConfig, db::Db, state::State, COMPONENT};
mod api;
pub struct Store {
api_service: api_server::ApiServer<api::StoreApi>,
listener: TcpListener,
}
impl Store {
pub async fn init(config: StoreConfig) -> Result<Self, ApiError> {
info!(target: COMPONENT, %config, "Loading database");
let block_store = Arc::new(BlockStore::new(config.blockstore_dir.clone()).await?);
let db = Db::setup(config.clone(), Arc::clone(&block_store))
.await
.map_err(|err| ApiError::ApiInitialisationFailed(err.to_string()))?;
let state = Arc::new(
State::load(db, block_store)
.await
.map_err(|err| ApiError::DatabaseConnectionFailed(err.to_string()))?,
);
let api_service = api_server::ApiServer::new(api::StoreApi { state });
let addr = config
.endpoint
.to_socket_addrs()
.map_err(ApiError::EndpointToSocketFailed)?
.next()
.ok_or_else(|| ApiError::AddressResolutionFailed(config.endpoint.to_string()))?;
let listener = TcpListener::bind(addr).await?;
info!(target: COMPONENT, "Database loaded");
Ok(Self { api_service, listener })
}
pub async fn serve(self) -> Result<(), ApiError> {
tonic::transport::Server::builder()
.add_service(self.api_service)
.serve_with_incoming(TcpListenerStream::new(self.listener))
.await
.map_err(ApiError::ApiServeFailed)
}
}