use crate::env;
use borsh::{BorshDeserialize, BorshSerialize};
use super::ValueAndIndex;
use crate::store::key::ToKey;
use crate::store::{ERR_INCONSISTENT_STATE, IterableMap, LookupMap, Vector};
pub enum Entry<'a, K: 'a, V: 'a, H: 'a>
where
K: BorshSerialize + Ord + Clone,
V: BorshSerialize + BorshDeserialize,
H: ToKey,
{
Occupied(OccupiedEntry<'a, K, V, H>),
Vacant(VacantEntry<'a, K, V, H>),
}
impl<'a, K, V, H> Entry<'a, K, V, H>
where
K: BorshSerialize + Ord + Clone,
V: BorshSerialize + BorshDeserialize,
H: ToKey,
{
pub(super) fn new(
key: K,
keys: &'a mut Vector<K>,
values: &'a mut LookupMap<K, ValueAndIndex<V>, H>,
) -> Self {
if values.contains_key(&key) {
return Self::Occupied(OccupiedEntry { key, keys, values });
}
Self::Vacant(VacantEntry { key, keys, values })
}
}
impl<'a, K, V, H> Entry<'a, K, V, H>
where
K: BorshSerialize + Ord + Clone,
V: BorshSerialize + BorshDeserialize,
H: ToKey,
{
pub fn key(&self) -> &K {
match self {
Entry::Occupied(entry) => entry.key(),
Entry::Vacant(entry) => entry.key(),
}
}
}
impl<'a, K, V, H> Entry<'a, K, V, H>
where
K: BorshSerialize + BorshDeserialize + Ord + Clone,
V: BorshSerialize + BorshDeserialize,
H: ToKey,
{
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
where
K: BorshDeserialize,
{
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, H>
where
K: BorshSerialize + Ord + Clone,
V: BorshSerialize + BorshDeserialize,
H: ToKey,
{
key: K,
keys: &'a mut Vector<K>,
values: &'a mut LookupMap<K, ValueAndIndex<V>, H>,
}
impl<'a, K, V, H> OccupiedEntry<'a, K, V, H>
where
K: BorshSerialize + Ord + Clone,
V: BorshSerialize + BorshDeserialize,
H: ToKey,
{
pub fn key(&self) -> &K {
&self.key
}
pub fn remove_entry(self) -> (K, V)
where
K: BorshDeserialize + Ord + Clone,
V: BorshDeserialize,
{
let old_value =
self.values.remove(&self.key).unwrap_or_else(|| env::panic_str(ERR_INCONSISTENT_STATE));
let last_index = self.keys.len() - 1;
self.keys.swap_remove(old_value.key_index);
IterableMap::remove_entry_helper(self.keys, self.values, old_value.key_index, last_index);
(self.key, old_value.value)
}
pub fn get(&self) -> &V {
&self.values.get(&self.key).unwrap_or_else(|| env::panic_str(ERR_INCONSISTENT_STATE)).value
}
pub fn get_mut(&mut self) -> &mut V {
&mut self
.values
.get_mut(&self.key)
.unwrap_or_else(|| env::panic_str(ERR_INCONSISTENT_STATE))
.value
}
pub fn into_mut(self) -> &'a mut V {
&mut self
.values
.get_mut(&self.key)
.unwrap_or_else(|| env::panic_str(ERR_INCONSISTENT_STATE))
.value
}
pub fn insert(&mut self, value: V) -> V {
core::mem::replace(self.get_mut(), value)
}
pub fn remove(self) -> V
where
K: BorshDeserialize,
{
self.remove_entry().1
}
}
pub struct VacantEntry<'a, K, V, H>
where
K: BorshSerialize + Ord,
V: BorshSerialize,
H: ToKey,
{
key: K,
values: &'a mut LookupMap<K, ValueAndIndex<V>, H>,
keys: &'a mut Vector<K>,
}
impl<'a, K, V, H> VacantEntry<'a, K, V, H>
where
K: BorshSerialize + Ord,
V: BorshSerialize + BorshDeserialize,
H: ToKey,
{
pub fn key(&self) -> &K {
&self.key
}
pub fn into_key(self) -> K {
self.key
}
pub fn insert(self, value: V) -> &'a mut V
where
K: BorshDeserialize + Clone,
{
self.keys.push(self.key().to_owned());
let key_index = self.keys.len() - 1;
&mut self.values.entry(self.key).or_insert(ValueAndIndex { value, key_index }).value
}
}