mirror of
https://github.com/bitcoinresearchkit/brk.git
synced 2026-04-26 15:49:58 -07:00
190 lines
4.2 KiB
Rust
190 lines
4.2 KiB
Rust
use std::{
|
|
cmp::Ordering,
|
|
fmt::Debug,
|
|
path::{Path, PathBuf},
|
|
time::Duration,
|
|
};
|
|
|
|
use arc_swap::ArcSwap;
|
|
use brk_core::{Error, Height, Result, Value, Version};
|
|
|
|
use crate::{
|
|
AnyCollectableVec, AnyIterableVec, AnyVec, BoxedVecIterator, CollectableVec, Format,
|
|
GenericStoredVec, Mmap, StoredIndex, StoredType, StoredVec,
|
|
};
|
|
|
|
use super::StoredVecIterator;
|
|
|
|
#[derive(Debug, Clone)]
|
|
pub struct IndexedVec<I, T> {
|
|
height: Option<Height>,
|
|
inner: StoredVec<I, T>,
|
|
}
|
|
|
|
impl<I, T> IndexedVec<I, T>
|
|
where
|
|
I: StoredIndex,
|
|
T: StoredType,
|
|
{
|
|
pub fn forced_import(
|
|
path: &Path,
|
|
value_name: &str,
|
|
version: Version,
|
|
format: Format,
|
|
) -> Result<Self> {
|
|
let inner = StoredVec::forced_import(path, value_name, version, format)?;
|
|
|
|
Ok(Self {
|
|
height: Height::try_from(Self::path_height_(inner.path()).as_path()).ok(),
|
|
inner,
|
|
})
|
|
}
|
|
|
|
#[inline]
|
|
pub fn get_or_read(&self, index: I, mmap: &Mmap) -> Result<Option<Value<T>>> {
|
|
self.inner.get_or_read(index, mmap)
|
|
}
|
|
|
|
#[inline]
|
|
pub fn push_if_needed(&mut self, index: I, value: T) -> Result<()> {
|
|
let len = self.inner.len();
|
|
match len.cmp(&index.to_usize()?) {
|
|
Ordering::Greater => {
|
|
// dbg!(len, index, &self.pathbuf);
|
|
// panic!();
|
|
Ok(())
|
|
}
|
|
Ordering::Equal => {
|
|
self.inner.push(value);
|
|
Ok(())
|
|
}
|
|
Ordering::Less => {
|
|
dbg!(index, value, len, self.path_height());
|
|
Err(Error::IndexTooHigh)
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn truncate_if_needed(&mut self, index: I, height: Height) -> Result<()> {
|
|
if self.height.is_none_or(|self_height| self_height != height) {
|
|
height.write(&self.path_height())?;
|
|
}
|
|
self.inner.truncate_if_needed(index)?;
|
|
Ok(())
|
|
}
|
|
|
|
pub fn flush(&mut self, height: Height) -> Result<()> {
|
|
height.write(&self.path_height())?;
|
|
self.inner.flush()
|
|
}
|
|
|
|
pub fn mmap(&self) -> &ArcSwap<Mmap> {
|
|
self.inner.mmap()
|
|
}
|
|
|
|
#[inline]
|
|
pub fn hasnt(&self, index: I) -> Result<bool> {
|
|
self.inner.has(index).map(|b| !b)
|
|
}
|
|
|
|
pub fn height(&self) -> brk_core::Result<Height> {
|
|
Height::try_from(self.path_height().as_path())
|
|
}
|
|
fn path_height(&self) -> PathBuf {
|
|
Self::path_height_(self.inner.path())
|
|
}
|
|
fn path_height_(path: &Path) -> PathBuf {
|
|
path.join("height")
|
|
}
|
|
}
|
|
|
|
impl<I, T> AnyVec for IndexedVec<I, T>
|
|
where
|
|
I: StoredIndex,
|
|
T: StoredType,
|
|
{
|
|
#[inline]
|
|
fn version(&self) -> Version {
|
|
self.inner.version()
|
|
}
|
|
|
|
#[inline]
|
|
fn name(&self) -> String {
|
|
self.inner.name()
|
|
}
|
|
|
|
#[inline]
|
|
fn len(&self) -> usize {
|
|
self.inner.len()
|
|
}
|
|
|
|
#[inline]
|
|
fn modified_time(&self) -> Result<Duration> {
|
|
self.inner.modified_time()
|
|
}
|
|
|
|
#[inline]
|
|
fn index_type_to_string(&self) -> String {
|
|
I::to_string()
|
|
}
|
|
}
|
|
|
|
pub trait AnyIndexedVec: AnyVec {
|
|
fn height(&self) -> brk_core::Result<Height>;
|
|
fn flush(&mut self, height: Height) -> Result<()>;
|
|
}
|
|
|
|
impl<I, T> AnyIndexedVec for IndexedVec<I, T>
|
|
where
|
|
I: StoredIndex,
|
|
T: StoredType,
|
|
{
|
|
fn height(&self) -> brk_core::Result<Height> {
|
|
self.height()
|
|
}
|
|
|
|
fn flush(&mut self, height: Height) -> Result<()> {
|
|
self.flush(height)
|
|
}
|
|
}
|
|
|
|
impl<'a, I, T> IntoIterator for &'a IndexedVec<I, T>
|
|
where
|
|
I: StoredIndex,
|
|
T: StoredType,
|
|
{
|
|
type Item = (I, Value<'a, T>);
|
|
type IntoIter = StoredVecIterator<'a, I, T>;
|
|
|
|
fn into_iter(self) -> Self::IntoIter {
|
|
self.inner.into_iter()
|
|
}
|
|
}
|
|
|
|
impl<I, T> AnyIterableVec<I, T> for IndexedVec<I, T>
|
|
where
|
|
I: StoredIndex,
|
|
T: StoredType,
|
|
{
|
|
fn boxed_iter<'a>(&'a self) -> BoxedVecIterator<'a, I, T>
|
|
where
|
|
T: 'a,
|
|
{
|
|
Box::new(self.into_iter())
|
|
}
|
|
}
|
|
|
|
impl<I, T> AnyCollectableVec for IndexedVec<I, T>
|
|
where
|
|
I: StoredIndex,
|
|
T: StoredType,
|
|
{
|
|
fn collect_range_serde_json(
|
|
&self,
|
|
from: Option<i64>,
|
|
to: Option<i64>,
|
|
) -> Result<Vec<serde_json::Value>> {
|
|
CollectableVec::collect_range_serde_json(self, from, to)
|
|
}
|
|
}
|