multiversx_sc/storage/mappers/
unique_id_mapper.rsuse crate::{
abi::TypeAbiFrom,
codec::{
multi_encode_iter_or_handle_err, EncodeErrorHandler, TopEncodeMulti, TopEncodeMultiOutput,
},
types::ManagedAddress,
};
use super::{
set_mapper::{CurrentStorage, StorageAddress},
StorageMapper, StorageMapperFromAddress, VecMapper,
};
use crate::{
abi::{TypeAbi, TypeDescriptionContainer, TypeName},
api::{ErrorApiImpl, StorageMapperApi},
storage::StorageKey,
storage_set,
types::{ManagedType, MultiValueEncoded},
};
pub type UniqueId = usize;
const EMPTY_ENTRY: UniqueId = 0;
pub struct UniqueIdMapper<SA, A = CurrentStorage>
where
SA: StorageMapperApi,
A: StorageAddress<SA>,
{
_address: A,
base_key: StorageKey<SA>,
vec_mapper: VecMapper<SA, UniqueId, A>,
}
impl<SA> StorageMapper<SA> for UniqueIdMapper<SA, CurrentStorage>
where
SA: StorageMapperApi,
{
fn new(base_key: StorageKey<SA>) -> Self {
Self {
_address: CurrentStorage,
base_key: base_key.clone(),
vec_mapper: VecMapper::new(base_key),
}
}
}
impl<SA> StorageMapperFromAddress<SA> for UniqueIdMapper<SA, ManagedAddress<SA>>
where
SA: StorageMapperApi,
{
fn new_from_address(address: ManagedAddress<SA>, base_key: StorageKey<SA>) -> Self {
Self {
_address: address.clone(),
base_key: base_key.clone(),
vec_mapper: VecMapper::new_from_address(address, base_key),
}
}
}
impl<SA, A> UniqueIdMapper<SA, A>
where
SA: StorageMapperApi,
A: StorageAddress<SA>,
{
#[inline]
pub fn len(&self) -> usize {
self.vec_mapper.len()
}
#[inline]
pub fn is_empty(&self) -> bool {
self.vec_mapper.is_empty()
}
pub fn get(&self, index: usize) -> UniqueId {
let id: UniqueId = self.vec_mapper.get(index);
if id == EMPTY_ENTRY {
index
} else {
id
}
}
pub fn iter(&self) -> Iter<SA, A> {
Iter::new(self)
}
}
impl<SA> UniqueIdMapper<SA, CurrentStorage>
where
SA: StorageMapperApi,
{
pub fn set_initial_len(&mut self, len: usize) {
if !self.vec_mapper.is_empty() {
SA::error_api_impl().signal_error(b"len already set");
}
self.set_internal_mapper_len(len);
}
pub fn swap_remove(&mut self, index: usize) -> UniqueId {
let last_item_index = self.len();
let last_item = self.get(last_item_index);
let current_item = if index != last_item_index {
let item_at_index = self.get(index);
self.set(index, last_item);
item_at_index
} else {
last_item
};
self.vec_mapper.set(last_item_index, &EMPTY_ENTRY);
self.set_internal_mapper_len(last_item_index - 1);
current_item
}
pub fn set(&mut self, index: usize, id: UniqueId) {
if index == id {
self.vec_mapper.set(index, &EMPTY_ENTRY);
} else {
self.vec_mapper.set(index, &id);
}
}
fn set_internal_mapper_len(&mut self, new_len: usize) {
let mut len_key = self.base_key.clone();
len_key.append_bytes(&b".len"[..]);
storage_set(len_key.as_ref(), &new_len);
}
}
impl<'a, SA, A> IntoIterator for &'a UniqueIdMapper<SA, A>
where
SA: StorageMapperApi,
A: StorageAddress<SA>,
{
type Item = usize;
type IntoIter = Iter<'a, SA, A>;
fn into_iter(self) -> Self::IntoIter {
self.iter()
}
}
pub struct Iter<'a, SA, A>
where
SA: StorageMapperApi,
A: StorageAddress<SA>,
{
index: usize,
len: usize,
id_mapper: &'a UniqueIdMapper<SA, A>,
}
impl<'a, SA, A> Iter<'a, SA, A>
where
SA: StorageMapperApi,
A: StorageAddress<SA>,
{
fn new(id_mapper: &'a UniqueIdMapper<SA, A>) -> Iter<'a, SA, A> {
Iter {
index: 1,
len: id_mapper.len(),
id_mapper,
}
}
}
impl<SA, A> Iterator for Iter<'_, SA, A>
where
SA: StorageMapperApi,
A: StorageAddress<SA>,
{
type Item = usize;
#[inline]
fn next(&mut self) -> Option<Self::Item> {
let current_index = self.index;
if current_index > self.len {
return None;
}
self.index += 1;
Some(self.id_mapper.get(current_index))
}
}
impl<SA> TopEncodeMulti for UniqueIdMapper<SA, CurrentStorage>
where
SA: StorageMapperApi,
{
fn multi_encode_or_handle_err<O, H>(&self, output: &mut O, h: H) -> Result<(), H::HandledErr>
where
O: TopEncodeMultiOutput,
H: EncodeErrorHandler,
{
multi_encode_iter_or_handle_err(self.iter(), output, h)
}
}
impl<SA> TypeAbiFrom<UniqueIdMapper<SA, CurrentStorage>> for MultiValueEncoded<SA, usize> where
SA: StorageMapperApi
{
}
impl<SA> TypeAbiFrom<Self> for UniqueIdMapper<SA, CurrentStorage> where SA: StorageMapperApi {}
impl<SA> TypeAbi for UniqueIdMapper<SA, CurrentStorage>
where
SA: StorageMapperApi,
{
type Unmanaged = Self;
fn type_name() -> TypeName {
crate::abi::type_name_variadic::<usize>()
}
fn type_name_rust() -> TypeName {
crate::abi::type_name_multi_value_encoded::<usize>()
}
fn provide_type_descriptions<TDC: TypeDescriptionContainer>(accumulator: &mut TDC) {
usize::provide_type_descriptions(accumulator);
}
fn is_variadic() -> bool {
true
}
}