use crate::env::abort;
use crate::utils::CacheEntry;
pub enum Entry<'a, K: 'a, V: 'a> {
Occupied(OccupiedEntry<'a, K, V>),
Vacant(VacantEntry<'a, K, V>),
}
impl<'a, K, V> Entry<'a, K, V> {
pub fn key(&self) -> &K {
match self {
Entry::Occupied(entry) => entry.key(),
Entry::Vacant(entry) => entry.key(),
}
}
pub fn or_insert(self, default: V) -> &'a mut V {
self.or_insert_with(|| default)
}
pub fn or_insert_with<F: FnOnce() -> V>(self, default: F) -> &'a mut V {
self.or_insert_with_key(|_| default())
}
pub fn or_insert_with_key<F: FnOnce(&K) -> V>(self, default: F) -> &'a mut V {
match self {
Self::Occupied(entry) => entry.into_mut(),
Self::Vacant(entry) => {
let value = default(entry.key());
entry.insert(value)
}
}
}
pub fn or_default(self) -> &'a mut V
where
V: Default,
{
match self {
Self::Occupied(entry) => entry.into_mut(),
Self::Vacant(entry) => entry.insert(Default::default()),
}
}
pub fn and_modify<F>(mut self, f: F) -> Self
where
F: FnOnce(&mut V),
{
if let Self::Occupied(entry) = &mut self {
f(entry.get_mut());
}
self
}
}
pub struct OccupiedEntry<'a, K, V> {
pub(super) key: K,
pub(super) entry: &'a mut CacheEntry<V>,
}
impl<'a, K, V> OccupiedEntry<'a, K, V> {
pub fn key(&self) -> &K {
&self.key
}
pub fn remove_entry(self) -> (K, V) {
let value = self.entry.value_mut().take().unwrap_or_else(|| abort());
(self.key, value)
}
pub fn get(&self) -> &V {
self.entry.value().as_ref().unwrap_or_else(|| abort())
}
pub fn get_mut(&mut self) -> &mut V {
self.entry.value_mut().as_mut().unwrap_or_else(|| abort())
}
pub fn into_mut(self) -> &'a mut V {
self.entry.value_mut().as_mut().unwrap_or_else(|| abort())
}
pub fn insert(&mut self, value: V) -> V {
self.entry.replace(Some(value)).unwrap_or_else(|| abort())
}
pub fn remove(self) -> V {
self.remove_entry().1
}
}
pub struct VacantEntry<'a, K, V> {
pub(super) key: K,
pub(super) entry: &'a mut CacheEntry<V>,
}
impl<'a, K, V> VacantEntry<'a, K, V> {
pub fn key(&self) -> &K {
&self.key
}
pub fn into_key(self) -> K {
self.key
}
pub fn insert(self, value: V) -> &'a mut V {
self.entry.replace(Some(value));
self.entry.value_mut().as_mut().unwrap_or_else(|| abort())
}
}