use crate::internals::hashtrie::cursor::CursorReadOps;
use crate::internals::hashtrie::cursor::{CursorRead, CursorWrite, SuperBlock};
use crate::internals::hashtrie::iter::*;
use std::borrow::Borrow;
use crate::internals::lincowcell::LinCowCellCapable;
use std::fmt::Debug;
use std::hash::Hash;
use std::iter::FromIterator;
pub struct HashTrie<K, V>
where
K: Hash + Eq + Clone + Debug + Sync + Send + 'static,
V: Clone + Sync + Send + 'static,
{
inner: LinCowCell<SuperBlock<K, V>, CursorRead<K, V>, CursorWrite<K, V>>,
}
unsafe impl<K: Hash + Eq + Clone + Debug + Sync + Send + 'static, V: Clone + Sync + Send + 'static>
Send for HashTrie<K, V>
{
}
unsafe impl<K: Hash + Eq + Clone + Debug + Sync + Send + 'static, V: Clone + Sync + Send + 'static>
Sync for HashTrie<K, V>
{
}
pub struct HashTrieReadTxn<'a, K, V>
where
K: Hash + Eq + Clone + Debug + Sync + Send + 'static,
V: Clone + Sync + Send + 'static,
{
inner: LinCowCellReadTxn<'a, SuperBlock<K, V>, CursorRead<K, V>, CursorWrite<K, V>>,
}
pub struct HashTrieWriteTxn<'a, K, V>
where
K: Hash + Eq + Clone + Debug + Sync + Send + 'static,
V: Clone + Sync + Send + 'static,
{
inner: LinCowCellWriteTxn<'a, SuperBlock<K, V>, CursorRead<K, V>, CursorWrite<K, V>>,
}
enum SnapshotType<'a, K, V>
where
K: Hash + Eq + Clone + Debug + Sync + Send + 'static,
V: Clone + Sync + Send + 'static,
{
R(&'a CursorRead<K, V>),
W(&'a CursorWrite<K, V>),
}
pub struct HashTrieReadSnapshot<'a, K, V>
where
K: Hash + Eq + Clone + Debug + Sync + Send + 'static,
V: Clone + Sync + Send + 'static,
{
inner: SnapshotType<'a, K, V>,
}
impl<K: Hash + Eq + Clone + Debug + Sync + Send + 'static, V: Clone + Sync + Send + 'static> Default
for HashTrie<K, V>
{
fn default() -> Self {
Self::new()
}
}
impl<K: Hash + Eq + Clone + Debug + Sync + Send + 'static, V: Clone + Sync + Send + 'static>
FromIterator<(K, V)> for HashTrie<K, V>
{
fn from_iter<I: IntoIterator<Item = (K, V)>>(iter: I) -> Self {
let mut new_sblock = unsafe { SuperBlock::new() };
let prev = new_sblock.create_reader();
let mut cursor = new_sblock.create_writer();
cursor.extend(iter);
let _ = new_sblock.pre_commit(cursor, &prev);
HashTrie {
inner: LinCowCell::new(new_sblock),
}
}
}
impl<K: Hash + Eq + Clone + Debug + Sync + Send + 'static, V: Clone + Sync + Send + 'static>
Extend<(K, V)> for HashTrieWriteTxn<'_, K, V>
{
fn extend<I: IntoIterator<Item = (K, V)>>(&mut self, iter: I) {
self.inner.as_mut().extend(iter);
}
}
impl<K: Hash + Eq + Clone + Debug + Sync + Send + 'static, V: Clone + Sync + Send + 'static>
HashTrieWriteTxn<'_, K, V>
{
pub(crate) fn get_prehashed<Q>(&self, k: &Q, k_hash: u64) -> Option<&V>
where
K: Borrow<Q>,
Q: Hash + Eq + ?Sized,
{
self.inner.as_ref().search(k_hash, k)
}
pub fn get<Q>(&self, k: &Q) -> Option<&V>
where
K: Borrow<Q>,
Q: Hash + Eq + ?Sized,
{
let k_hash = self.inner.as_ref().hash_key(k);
self.get_prehashed(k, k_hash)
}
pub fn contains_key<Q>(&self, k: &Q) -> bool
where
K: Borrow<Q>,
Q: Hash + Eq + ?Sized,
{
self.get(k).is_some()
}
pub fn len(&self) -> usize {
self.inner.as_ref().len()
}
pub fn is_empty(&self) -> bool {
self.inner.as_ref().len() == 0
}
pub fn iter(&self) -> Iter<'_, K, V> {
self.inner.as_ref().kv_iter()
}
pub fn values(&self) -> ValueIter<'_, K, V> {
self.inner.as_ref().v_iter()
}
pub fn keys(&self) -> KeyIter<'_, K, V> {
self.inner.as_ref().k_iter()
}
pub fn clear(&mut self) {
self.inner.as_mut().clear()
}
pub fn insert(&mut self, k: K, v: V) -> Option<V> {
let k_hash = self.inner.as_ref().hash_key(&k);
self.inner.as_mut().insert(k_hash, k, v)
}
pub fn remove(&mut self, k: &K) -> Option<V> {
let k_hash = self.inner.as_ref().hash_key(k);
self.inner.as_mut().remove(k_hash, k)
}
pub fn get_mut(&mut self, k: &K) -> Option<&mut V> {
let k_hash = self.inner.as_ref().hash_key(k);
self.inner.as_mut().get_mut_ref(k_hash, k)
}
pub fn to_snapshot(&self) -> HashTrieReadSnapshot<'_, K, V> {
HashTrieReadSnapshot {
inner: SnapshotType::W(self.inner.as_ref()),
}
}
}
impl<K: Hash + Eq + Clone + Debug + Sync + Send + 'static, V: Clone + Sync + Send + 'static>
HashTrieReadTxn<'_, K, V>
{
pub(crate) fn get_prehashed<Q>(&self, k: &Q, k_hash: u64) -> Option<&V>
where
K: Borrow<Q>,
Q: Hash + Eq + ?Sized,
{
self.inner.search(k_hash, k)
}
pub fn get<Q>(&self, k: &Q) -> Option<&V>
where
K: Borrow<Q>,
Q: Hash + Eq + ?Sized,
{
let k_hash = self.inner.as_ref().hash_key(k);
self.get_prehashed(k, k_hash)
}
pub fn contains_key<Q>(&self, k: &Q) -> bool
where
K: Borrow<Q>,
Q: Hash + Eq + ?Sized,
{
self.get(k).is_some()
}
pub fn len(&self) -> usize {
self.inner.as_ref().len()
}
pub fn is_empty(&self) -> bool {
self.inner.as_ref().len() == 0
}
pub fn iter(&self) -> Iter<'_, K, V> {
self.inner.as_ref().kv_iter()
}
pub fn values(&self) -> ValueIter<'_, K, V> {
self.inner.as_ref().v_iter()
}
pub fn keys(&self) -> KeyIter<'_, K, V> {
self.inner.as_ref().k_iter()
}
pub fn to_snapshot(&self) -> HashTrieReadSnapshot<'_, K, V> {
HashTrieReadSnapshot {
inner: SnapshotType::R(self.inner.as_ref()),
}
}
}
impl<K: Hash + Eq + Clone + Debug + Sync + Send + 'static, V: Clone + Sync + Send + 'static>
HashTrieReadSnapshot<'_, K, V>
{
pub fn get<Q>(&self, k: &Q) -> Option<&V>
where
K: Borrow<Q>,
Q: Hash + Eq + ?Sized,
{
match self.inner {
SnapshotType::R(inner) => {
let k_hash = inner.hash_key(k);
inner.search(k_hash, k)
}
SnapshotType::W(inner) => {
let k_hash = inner.hash_key(k);
inner.search(k_hash, k)
}
}
}
pub fn contains_key<Q>(&self, k: &Q) -> bool
where
K: Borrow<Q>,
Q: Hash + Eq + ?Sized,
{
self.get(k).is_some()
}
pub fn len(&self) -> usize {
match self.inner {
SnapshotType::R(inner) => inner.len(),
SnapshotType::W(inner) => inner.len(),
}
}
pub fn is_empty(&self) -> bool {
self.len() == 0
}
pub fn iter(&self) -> Iter<'_, K, V> {
match self.inner {
SnapshotType::R(inner) => inner.kv_iter(),
SnapshotType::W(inner) => inner.kv_iter(),
}
}
pub fn values(&self) -> ValueIter<'_, K, V> {
match self.inner {
SnapshotType::R(inner) => inner.v_iter(),
SnapshotType::W(inner) => inner.v_iter(),
}
}
pub fn keys(&self) -> KeyIter<'_, K, V> {
match self.inner {
SnapshotType::R(inner) => inner.k_iter(),
SnapshotType::W(inner) => inner.k_iter(),
}
}
}