use ahash::HashMap;
use slab::Slab;
use std::borrow::Borrow;
use std::fmt::{self, Debug};
use std::marker::PhantomData;
use std::mem;
use std::ops::{Index, IndexMut};
use crate::store::adapter::slab::{Iter, IterMut};
use crate::store::item::{Key, Value};
use crate::store::{
Store, StoreIterable, StoreIterableMut, StoreMut, StoreMutRef,
};
pub mod items;
mod iter;
pub mod slots;
pub use items::Items;
pub use slots::{Slots, SlotsMut};
#[derive(Clone)]
pub struct Stash<K, V, S = HashMap<K, usize>> {
store: S,
items: Slab<(K, V)>,
marker: PhantomData<K>,
}
impl<K, V, S> Stash<K, V, S>
where
K: Key,
S: Store<K, usize>,
{
#[must_use]
pub fn new() -> Self
where
S: Default,
{
Self {
store: S::default(),
items: Slab::new(),
marker: PhantomData,
}
}
#[inline]
pub fn get<Q>(&self, key: &Q) -> Option<usize>
where
K: Borrow<Q>,
Q: Key,
{
self.store.get(key).copied()
}
#[inline]
pub fn key(&self, index: usize) -> Option<&K> {
self.items.get(index).map(|(key, _)| key)
}
}
impl<K, V, S> Stash<K, V, S>
where
K: Key,
S: StoreMut<K, usize>,
{
#[inline]
pub fn insert(&mut self, key: K, value: V) -> usize {
if let Some(&index) = self.store.get(&key) {
self.items[index].1 = value;
index
} else {
let index = self.items.insert((key.clone(), value));
self.store.insert(key, index);
index
}
}
#[inline]
pub fn remove(&mut self, index: usize) -> Option<(K, V)> {
if let Some((key, _)) = self.items.get(index) {
self.store.remove(key).map(|index| self.items.remove(index))
} else {
None
}
}
}
impl<K, V, S> Store<K, V> for Stash<K, V, S>
where
K: Key,
S: Store<K, usize>,
{
#[inline]
fn get<Q>(&self, key: &Q) -> Option<&V>
where
K: Borrow<Q>,
Q: Key,
{
match self.store.get(key) {
Some(&index) => self.items.get(index).map(|(_, value)| value),
None => None,
}
}
#[inline]
fn contains_key<Q>(&self, key: &Q) -> bool
where
K: Borrow<Q>,
Q: Key,
{
self.store.contains_key(key)
}
#[inline]
fn len(&self) -> usize {
self.store.len()
}
}
impl<K, V, S> StoreMut<K, V> for Stash<K, V, S>
where
K: Key,
V: Value,
S: StoreMut<K, usize>,
{
#[inline]
fn insert(&mut self, key: K, value: V) -> Option<V> {
if let Some(&index) = self.store.get(&key) {
let prior = &mut self.items[index].1;
(prior != &value).then(|| mem::replace(prior, value))
} else {
let index = self.items.insert((key.clone(), value));
self.store.insert(key, index);
None
}
}
#[inline]
fn remove<Q>(&mut self, key: &Q) -> Option<V>
where
K: Borrow<Q>,
Q: Key,
{
self.store.remove(key).map(|index| {
let (_, value) = self.items.remove(index);
value
})
}
#[inline]
fn remove_entry<Q>(&mut self, key: &Q) -> Option<(K, V)>
where
K: Borrow<Q>,
Q: Key,
{
self.store.remove(key).map(|index| self.items.remove(index))
}
#[inline]
fn clear(&mut self) {
self.store.clear();
self.items.clear();
}
}
impl<K, V, S> StoreMutRef<K, V> for Stash<K, V, S>
where
K: Key,
S: StoreMut<K, usize>,
{
#[inline]
fn get_mut<Q>(&mut self, key: &Q) -> Option<&mut V>
where
K: Borrow<Q>,
Q: Key,
{
match self.store.get(key) {
Some(&index) => self.items.get_mut(index).map(|(_, value)| value),
None => None,
}
}
#[inline]
fn get_or_insert_default(&mut self, key: &K) -> &mut V
where
V: Default,
{
if !self.store.contains_key(key) {
let n = self.items.insert((key.clone(), V::default()));
self.store.insert(key.clone(), n);
}
self.get_mut(key).expect("invariant")
}
}
impl<K, V, S> Index<usize> for Stash<K, V, S>
where
K: Key,
S: Store<K, usize>,
{
type Output = V;
#[inline]
fn index(&self, index: usize) -> &Self::Output {
&self.items[index].1
}
}
impl<K, V, S> IndexMut<usize> for Stash<K, V, S>
where
K: Key,
S: Store<K, usize>,
{
#[inline]
fn index_mut(&mut self, index: usize) -> &mut Self::Output {
&mut self.items[index].1
}
}
impl<K, V, S> FromIterator<(K, V)> for Stash<K, V, S>
where
K: Key,
S: StoreMut<K, usize> + Default,
{
#[inline]
fn from_iter<T>(iter: T) -> Self
where
T: IntoIterator<Item = (K, V)>,
{
let mut store = Stash::new();
for (key, value) in iter {
store.insert(key, value);
}
store
}
}
#[allow(clippy::into_iter_without_iter)]
impl<'a, K, V, S> IntoIterator for &'a Stash<K, V, S>
where
K: Key,
V: Value,
S: StoreIterable<K, usize>,
{
type Item = (&'a K, &'a V);
type IntoIter = Iter<'a, K, V>;
#[inline]
fn into_iter(self) -> Self::IntoIter {
StoreIterable::iter(&self.items)
}
}
#[allow(clippy::into_iter_without_iter)]
impl<'a, K, V, S> IntoIterator for &'a mut Stash<K, V, S>
where
K: Key,
V: Value,
S: StoreIterableMut<K, usize>,
{
type Item = (&'a K, &'a mut V);
type IntoIter = IterMut<'a, K, V>;
#[inline]
fn into_iter(self) -> Self::IntoIter {
StoreIterableMut::iter_mut(&mut self.items)
}
}
impl<K, V> Default for Stash<K, V>
where
K: Key,
{
#[inline]
fn default() -> Self {
Self::new()
}
}
impl<K, V, S> Debug for Stash<K, V, S>
where
K: Debug,
V: Debug,
S: Debug,
{
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
f.debug_struct("Stash")
.field("store", &self.store)
.field("items", &self.items)
.finish()
}
}