2024-10-08 07:40:47 +00:00
|
|
|
use crate::HashElement;
|
|
|
|
use anyhow::Result;
|
|
|
|
use std::collections::HashMap;
|
|
|
|
use std::sync::Arc;
|
|
|
|
use tracing::error;
|
|
|
|
|
|
|
|
pub struct NodeManager<E: HashElement> {
|
|
|
|
cache: HashMap<(usize, usize), E>,
|
2024-10-08 08:19:06 +00:00
|
|
|
layer_size: Vec<usize>,
|
2024-10-08 07:40:47 +00:00
|
|
|
db: Arc<dyn NodeDatabase<E>>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<E: HashElement> NodeManager<E> {
|
|
|
|
pub fn new(db: Arc<dyn NodeDatabase<E>>) -> Self {
|
|
|
|
Self {
|
|
|
|
cache: HashMap::new(),
|
2024-10-08 08:19:06 +00:00
|
|
|
layer_size: vec![],
|
2024-10-08 07:40:47 +00:00
|
|
|
db,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get_node(&self, layer: usize, pos: usize) -> Option<E> {
|
|
|
|
match self.cache.get(&(layer, pos)) {
|
|
|
|
Some(node) => Some(node.clone()),
|
|
|
|
None => self.db.get_node(layer, pos).unwrap_or_else(|e| {
|
|
|
|
error!("Failed to get node: {}", e);
|
|
|
|
None
|
|
|
|
}),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn add_node(&mut self, layer: usize, pos: usize, node: E) {
|
|
|
|
if let Err(e) = self.db.save_node(layer, pos, &node) {
|
|
|
|
error!("Failed to save node: {}", e);
|
|
|
|
}
|
|
|
|
self.cache.insert((layer, pos), node);
|
2024-10-08 08:19:06 +00:00
|
|
|
if pos + 1 > self.layer_size[layer] {
|
|
|
|
self.layer_size[layer] = pos + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn add_layer(&mut self) {
|
|
|
|
self.layer_size.push(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn layer_size(&self, layer: usize) -> usize {
|
|
|
|
self.layer_size[layer]
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn num_layers(&self) -> usize {
|
|
|
|
self.layer_size.len()
|
2024-10-08 07:40:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub trait NodeDatabase<E: HashElement>: Send + Sync {
|
|
|
|
fn get_node(&self, layer: usize, pos: usize) -> Result<Option<E>>;
|
|
|
|
fn save_node(&self, layer: usize, pos: usize, node: &E) -> Result<()>;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// A dummy database structure for in-memory merkle tree that will not read/write db.
|
|
|
|
pub struct EmptyNodeDatabase {}
|
|
|
|
impl<E: HashElement> NodeDatabase<E> for EmptyNodeDatabase {
|
|
|
|
fn get_node(&self, _layer: usize, _pos: usize) -> Result<Option<E>> {
|
|
|
|
Ok(None)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn save_node(&self, _layer: usize, _pos: usize, _node: &E) -> Result<()> {
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|