use std::path::Path;
use std::time::Duration;
use miden_node_utils::spawn::spawn_blocking_in_span;
use miden_node_utils::tracing::OpenTelemetrySpanExt;
use tracing::info_span;
use crate::COMPONENT;
use crate::state::State;
impl State {
pub fn spawn_disk_monitor(&self) -> tokio::task::JoinHandle<()> {
let data_directory = self.data_directory.clone();
tokio::spawn(async move {
let mut interval = tokio::time::interval(Duration::from_mins(5));
loop {
interval.tick().await;
let dir = data_directory.clone();
let span = info_span!(target: COMPONENT, "measure_disk_space_usage");
let result =
spawn_blocking_in_span(move || measure_disk_usage_bytes(&dir), span.clone())
.await;
match result {
Ok(usage) => {
span.set_attribute("db.sqlite.size", usage.sqlite_db);
span.set_attribute("db.sqlite.wal.size", usage.sqlite_wal);
span.set_attribute("db.block_store.size", usage.block_store);
#[cfg(feature = "rocksdb")]
{
span.set_attribute("db.account_tree.size", usage.account_tree);
span.set_attribute("db.nullifier_tree.size", usage.nullifier_tree);
span.set_attribute(
"db.account_state_forest.size",
usage.account_state_forest,
);
}
},
Err(err) => span.set_error(&err),
}
}
})
}
}
struct DiskUsage {
sqlite_db: u64,
sqlite_wal: u64,
block_store: u64,
#[cfg(feature = "rocksdb")]
account_tree: u64,
#[cfg(feature = "rocksdb")]
nullifier_tree: u64,
#[cfg(feature = "rocksdb")]
account_state_forest: u64,
}
fn measure_disk_usage_bytes(data_dir: &Path) -> DiskUsage {
DiskUsage {
sqlite_db: path_size_bytes(&data_dir.join("miden-store.sqlite3")),
sqlite_wal: path_size_bytes(&data_dir.join("miden-store.sqlite3-wal")),
block_store: dir_size_bytes(&data_dir.join("blocks")),
#[cfg(feature = "rocksdb")]
account_tree: dir_size_bytes(&data_dir.join("accounttree")),
#[cfg(feature = "rocksdb")]
nullifier_tree: dir_size_bytes(&data_dir.join("nullifiertree")),
#[cfg(feature = "rocksdb")]
account_state_forest: dir_size_bytes(&data_dir.join("accountstateforest")),
}
}
fn path_size_bytes(path: &Path) -> u64 {
fs_err::metadata(path).map(|m| m.len()).unwrap_or(0)
}
fn dir_size_bytes(path: &Path) -> u64 {
let mut to_process = vec![path.to_path_buf()];
let mut total = 0u64;
while let Some(dir) = to_process.pop() {
let Ok(entries) = fs_err::read_dir(&dir) else {
continue;
};
for entry in entries.flatten() {
if let Ok(meta) = entry.metadata() {
if meta.is_dir() {
to_process.push(entry.path());
} else {
total += meta.len();
}
}
}
}
total
}