use hashbrown::hash_table;
use core::hash::Hash;
use core::mem;
pub enum EntryMut<'a, K, V> {
Occupied(OccupiedEntryMut<'a, K, V>),
Vacant(VacantEntryMut<'a, K, V>),
}
impl<'a, K: Eq + Hash, V> EntryMut<'a, K, V> {
pub fn and_modify(self, f: impl FnOnce(&mut V)) -> Self {
match self {
EntryMut::Occupied(mut entry) => {
f(entry.get_mut());
EntryMut::Occupied(entry)
}
EntryMut::Vacant(entry) => EntryMut::Vacant(entry),
}
}
pub fn key(&self) -> &K {
match *self {
EntryMut::Occupied(ref entry) => entry.key(),
EntryMut::Vacant(ref entry) => entry.key(),
}
}
pub fn into_key(self) -> K {
match self {
EntryMut::Occupied(entry) => entry.into_key(),
EntryMut::Vacant(entry) => entry.into_key(),
}
}
pub fn or_default(self) -> &'a mut (K, V)
where
V: Default,
{
match self {
EntryMut::Occupied(entry) => entry.into_mut(),
EntryMut::Vacant(entry) => entry.insert(V::default()),
}
}
pub fn or_insert(self, value: V) -> &'a mut (K, V) {
match self {
EntryMut::Occupied(entry) => entry.into_mut(),
EntryMut::Vacant(entry) => entry.insert(value),
}
}
pub fn or_insert_with(self, value: impl FnOnce() -> V) -> &'a mut (K, V) {
match self {
EntryMut::Occupied(entry) => entry.into_mut(),
EntryMut::Vacant(entry) => entry.insert(value()),
}
}
pub fn or_try_insert_with<E>(
self,
value: impl FnOnce() -> Result<V, E>,
) -> Result<&'a mut (K, V), E> {
match self {
EntryMut::Occupied(entry) => Ok(entry.into_mut()),
EntryMut::Vacant(entry) => Ok(entry.insert(value()?)),
}
}
pub fn insert(self, value: V) -> &'a mut (K, V) {
match self {
EntryMut::Occupied(mut entry) => {
entry.insert(value);
entry.into_mut()
}
EntryMut::Vacant(entry) => entry.insert(value),
}
}
pub fn insert_entry(self, value: V) -> OccupiedEntryMut<'a, K, V>
where
K: Clone,
{
match self {
EntryMut::Occupied(mut entry) => {
entry.insert(value);
entry
}
EntryMut::Vacant(entry) => entry.insert_entry(value),
}
}
}
pub struct VacantEntryMut<'a, K, V> {
key: K,
entry: hash_table::VacantEntry<'a, (K, V)>,
}
impl<'a, K: Eq + Hash, V> VacantEntryMut<'a, K, V> {
pub(crate) fn new(key: K, entry: hash_table::VacantEntry<'a, (K, V)>) -> Self {
Self { key, entry }
}
pub fn insert(self, value: V) -> &'a mut (K, V) {
let occupied = self.entry.insert((self.key, value));
occupied.into_mut()
}
pub fn insert_entry(self, value: V) -> OccupiedEntryMut<'a, K, V>
where
K: Clone,
{
let entry = self.entry.insert((self.key.clone(), value));
OccupiedEntryMut::new(self.key, entry)
}
pub fn into_key(self) -> K {
self.key
}
pub fn key(&self) -> &K {
&self.key
}
}
pub struct OccupiedEntryMut<'a, K, V> {
entry: hash_table::OccupiedEntry<'a, (K, V)>,
key: K,
}
impl<'a, K: Eq + Hash, V> OccupiedEntryMut<'a, K, V> {
pub(crate) fn new(key: K, entry: hash_table::OccupiedEntry<'a, (K, V)>) -> Self {
Self { key, entry }
}
pub fn get(&self) -> &V {
&self.entry.get().1
}
pub fn get_mut(&mut self) -> &mut V {
&mut self.entry.get_mut().1
}
pub fn insert(&mut self, value: V) -> V {
mem::replace(self.get_mut(), value)
}
pub fn into_mut(self) -> &'a mut (K, V) {
self.entry.into_mut()
}
pub fn into_key(self) -> K {
self.key
}
pub fn key(&self) -> &K {
&self.entry.get().0
}
pub fn remove(self) -> V {
let ((_k, v), _) = self.entry.remove();
v
}
pub fn remove_entry(self) -> (K, V) {
let ((k, v), _) = self.entry.remove();
(k, v)
}
pub fn replace_entry(self, value: V) -> (K, V) {
let (k, v) = mem::replace(self.entry.into_mut(), (self.key, value));
(k, v)
}
}
#[cfg(test)]
mod tests {
use crate::ClashMap;
use super::*;
#[test]
fn test_insert_entry_into_vacant() {
let mut map: ClashMap<u32, u32> = ClashMap::new();
let entry = map.entry_mut(1);
assert!(matches!(entry, EntryMut::Vacant(_)));
let entry = entry.insert_entry(2);
assert_eq!(*entry.get(), 2);
assert_eq!(*map.get(&1).unwrap(), 2);
}
#[test]
fn test_insert_entry_into_occupied() {
let mut map: ClashMap<u32, u32> = ClashMap::new();
map.insert(1, 1000);
let entry = map.entry_mut(1);
assert!(matches!(&entry, EntryMut::Occupied(entry) if *entry.get() == 1000));
let entry = entry.insert_entry(2);
assert_eq!(*entry.get(), 2);
assert_eq!(*map.get(&1).unwrap(), 2);
}
}