Advance

 

Storage Mappers

Welcome to the tutorial on “Storage Mappers in MultiversX.” MultiversX is a powerful decentralized platform that enables the development and deployment of smart contracts. One of the key features of MultiversX is the ability to store and manage data in a secure and efficient manner through the use of storage mappers. In this tutorial, we will dive into the world of storage mappers and how they can be used to manage and store data in MultiversX. We will cover the basics of storage mappers, how they are created, and how they can be used in your smart contracts. Whether you are a seasoned developer or just starting out, this tutorial will provide a comprehensive understanding of storage mappers in MultiversX. So, let’s get started!

The Rust framework provides various storage mappers you can use. Deciding which one to use for every situation is critical for performance. There will be a comparison section after each mapper is described.

Note: All the storage mappers support additional key arguments.

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
pub struct LinkedListNode<Type> {
value: Type,
node_id: u32,
next_id: u32,
prev_id: u32,
}
impl<Type> LinkedListNode<Type> {
pub fn get_value_cloned(&self) -> Type {
self.value.clone()
}
pub fn get_value_as_ref(&self) -> &Type {
&self.value
}
pub fn into_value(self) -> Type {
self.value
}
pub fn get_node_id(&self) -> u32 {
self.node_id
}
pub fn get_next_node_id(&self) -> u32 {
self.next_id
}
pub fn get_prev_node_id(&self) -> u32 {
self.prev_id
}
}
pub struct LinkedListNode<Type> { value: Type, node_id: u32, next_id: u32, prev_id: u32, } impl<Type> LinkedListNode<Type> { pub fn get_value_cloned(&self) -> Type { self.value.clone() } pub fn get_value_as_ref(&self) -> &Type { &self.value } pub fn into_value(self) -> Type { self.value } pub fn get_node_id(&self) -> u32 { self.node_id } pub fn get_next_node_id(&self) -> u32 { self.next_id } pub fn get_prev_node_id(&self) -> u32 { self.prev_id } }
pub struct LinkedListNode<Type> {
    value: Type,
    node_id: u32,
    next_id: u32,
    prev_id: u32,
}

impl<Type> LinkedListNode<Type> {
    pub fn get_value_cloned(&self) -> Type {
        self.value.clone()
    }

    pub fn get_value_as_ref(&self) -> &Type {
        &self.value
    }

    pub fn into_value(self) -> Type {
        self.value
    }

    pub fn get_node_id(&self) -> u32 {
        self.node_id
    }

    pub fn get_next_node_id(&self) -> u32 {
        self.next_id
    }

    pub fn get_prev_node_id(&self) -> u32 {
        self.prev_id
    }
}

pop_front/pop_back

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn pop_front(&mut self) -> Option<LinkedListNode<Type>>
fn pop_back(&mut self) -> Option<LinkedListNode<Type>>
fn pop_front(&mut self) -> Option<LinkedListNode<Type>> fn pop_back(&mut self) -> Option<LinkedListNode<Type>>
fn pop_front(&mut self) -> Option<LinkedListNode<Type>>
fn pop_back(&mut self) -> Option<LinkedListNode<Type>>

Removes and returns the first/last element from the list.

push_after/pus_after

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
pub fn push_after(node: &mut LinkedListNode<Type>, element: Type) -> Option<LinkedListNode<Type>>
pub fn push_before(node: &mut LinkedListNode<Type>, element: Type) -> Option<LinkedListNode<Type>>
pub fn push_after(node: &mut LinkedListNode<Type>, element: Type) -> Option<LinkedListNode<Type>> pub fn push_before(node: &mut LinkedListNode<Type>, element: Type) -> Option<LinkedListNode<Type>>
pub fn push_after(node: &mut LinkedListNode<Type>, element: Type) -> Option<LinkedListNode<Type>>
pub fn push_before(node: &mut LinkedListNode<Type>, element: Type) -> Option<LinkedListNode<Type>>

Inserts the given element into the list after/before the given node. Returns the newly inserted node if the insertion was successful, None otherwise.

push_after_node_id/push_before_node_id

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
pub fn push_after_node_id(node_id: usize, element: Type) -> Option<LinkedListNode<Type>>
pub fn push_before_node_id(node_id: usize, element: Type) -> Option<LinkedListNode<Type>>
pub fn push_after_node_id(node_id: usize, element: Type) -> Option<LinkedListNode<Type>> pub fn push_before_node_id(node_id: usize, element: Type) -> Option<LinkedListNode<Type>>
pub fn push_after_node_id(node_id: usize, element: Type) -> Option<LinkedListNode<Type>>
pub fn push_before_node_id(node_id: usize, element: Type) -> Option<LinkedListNode<Type>>

Same as the methods above, but uses node_id instead of a full node struct.

push_front/push_back

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn push_front(element: Type)
fn push_back(element: Type)
fn push_front(element: Type) fn push_back(element: Type)
fn push_front(element: Type)
fn push_back(element: Type)

 

General purpose mappers

SingleValueMapper

Stores a single value. Examples:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn single_value(&self) -> SingleValueMapper<Type>;
fn single_value_with_single_key_arg(&self, key_arg: Type1) -> SingleValueMapper<Type2>;
fn single_value_with_multi_key_arg(&self, key_arg1: Type1, key_arg2: Type2) -> SingleValueMapper<Type3>;
fn single_value(&self) -> SingleValueMapper<Type>; fn single_value_with_single_key_arg(&self, key_arg: Type1) -> SingleValueMapper<Type2>; fn single_value_with_multi_key_arg(&self, key_arg1: Type1, key_arg2: Type2) -> SingleValueMapper<Type3>;
fn single_value(&self) -> SingleValueMapper<Type>;
fn single_value_with_single_key_arg(&self, key_arg: Type1) -> SingleValueMapper<Type2>;
fn single_value_with_multi_key_arg(&self, key_arg1: Type1, key_arg2: Type2) -> SingleValueMapper<Type3>;

Keep in mind there is no way of iterating over all key_args, so if you need to do that, consider using another mapper.

Available methods:

get

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn get() -> Type
fn get() -> Type
fn get() -> Type

Reads the value from storage and deserializes it to the given Type. For numerical types and vector types, this will return the default value for empty storage, i.e. 0 and empty vec respectively. For custom structs, this will signal an error if trying to read from empty storage.

set

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn set(value: &Type)
fn set(value: &Type)
fn set(value: &Type)

Sets the stored value to the provided value argument. For base Rust numerical types, the reference is not needed.

is_empty

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn is_empty() -> bool
fn is_empty() -> bool
fn is_empty() -> bool

Returns true if the storage entry is empty. Usually used when storing struct types to prevent crashes on get().

set_if_empty

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn set_if_empty(value: &Type)
fn set_if_empty(value: &Type)
fn set_if_empty(value: &Type)

Sets the value only if the storage for that value is currently empty. Usually used in #init functions to not overwrite values on contract upgrade.

clear

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn clear()
fn clear()
fn clear()

Clears the entry.

update

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn update<R, F: FnOnce(&mut Type) -> R>(f: F) -> R
fn update<R, F: FnOnce(&mut Type) -> R>(f: F) -> R
fn update<R, F: FnOnce(&mut Type) -> R>(f: F) -> R

Takes a closure as an argument, applies that closure to the currently stored value saves the new value, and returns any value the given closure might return. Examples:

Incrementing a value:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn my_value(&self) -> SingleValueMapper<BigUint>;
self.my_value().update(|val| *val += 1);
fn my_value(&self) -> SingleValueMapper<BigUint>; self.my_value().update(|val| *val += 1);
fn my_value(&self) -> SingleValueMapper<BigUint>;

self.my_value().update(|val| *val += 1);
Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn my_value(&self) -> SingleValueMapper<BigUint>;
self.my_value().update(|val| *val += 1);
fn my_value(&self) -> SingleValueMapper<BigUint>; self.my_value().update(|val| *val += 1);
fn my_value(&self) -> SingleValueMapper<BigUint>;

self.my_value().update(|val| *val += 1);

Modifying a struct’s field:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
pub struct MyStruct {
pub field_1: u64,
pub field_2: u32
}
fn my_value(&self) -> SingleValueMapper<MyStruct>;
self.my_value().update(|val| val.field1 = 5);
pub struct MyStruct { pub field_1: u64, pub field_2: u32 } fn my_value(&self) -> SingleValueMapper<MyStruct>; self.my_value().update(|val| val.field1 = 5);
pub struct MyStruct {
    pub field_1: u64,
    pub field_2: u32
}

fn my_value(&self) -> SingleValueMapper<MyStruct>;

self.my_value().update(|val| val.field1 = 5);

Returning a value from the closure:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn my_value(&self) -> SingleValueMapper<BigUint>;
let new_val = self.my_value().update(|val| {
*val += 1;
*val
});
fn my_value(&self) -> SingleValueMapper<BigUint>; let new_val = self.my_value().update(|val| { *val += 1; *val });
fn my_value(&self) -> SingleValueMapper<BigUint>;

let new_val = self.my_value().update(|val| {
    *val += 1;
    *val
});

raw_byte_length

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn raw_byte_length() -> usize
fn raw_byte_length() -> usize
fn raw_byte_length() -> usize

Returns the raw byte length of the stored value. This should be rarely used.

VecMapper

Stores elements of the same type, each under their own storage key. Allows acces by index for said items. Keep in mind indexes start at 1 for VecMapper. Examples:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn my_vec(&self) -> VecMapper<Type>;
fn my_vec_with_args(&self, arg: Type1) -> VecMapper<Type2>;
fn my_vec(&self) -> VecMapper<Type>; fn my_vec_with_args(&self, arg: Type1) -> VecMapper<Type2>;
fn my_vec(&self) -> VecMapper<Type>;
fn my_vec_with_args(&self, arg: Type1) -> VecMapper<Type2>;

Available methods:

push

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn push(elem: &T)
fn push(elem: &T)
fn push(elem: &T)

Stores the element at index len and increments len afterwards.

get

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn get(index: usize) -> Type
fn get(index: usize) -> Type
fn get(index: usize) -> Type

Gets the element at the specific index. Valid indexes are 1 to len, both ends included. Attempting to read from an invalid index will signal an error.

set

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn set(index: usize, value: &Type)
fn set(index: usize, value: &Type)
fn set(index: usize, value: &Type)

Sets the element at the given index. Index must be in inclusive range 1 to len.

clear_entry

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn clear_entry(index: usize)
fn clear_entry(index: usize)
fn clear_entry(index: usize)

Clears the entry at the given index. This does not decrease the length.

is_empty

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn is_empty() -> bool
fn is_empty() -> bool
fn is_empty() -> bool

Returns true if the mapper has no elements stored.

len

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn len() -> usize
fn len() -> usize
fn len() -> usize

Returns the number of items stored in the mapper.

extend_from_slice

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn extend_from_slice(slice: &[Type])
fn extend_from_slice(slice: &[Type])
fn extend_from_slice(slice: &[Type])

Pushes all elements from the given slice at the end of the mapper. More efficient than manual for of push, as the internal length is only read and updated once.

swap_remove

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn swap_remove(index: usize)
fn swap_remove(index: usize)
fn swap_remove(index: usize)

Removes the element at index, moves the last element to index and decreases the len by 1. There is no way of removing an element and preserving the order.

clearfn clear()

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn clear()
fn clear()
fn clear()

Clears all the elements from the mapper. This function can run out of gas for big collections.

iter

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn iter() -> Iter<Type>
fn iter() -> Iter<Type>
fn iter() -> Iter<Type>

Provides an iterator over all the elements.

SetMapper

Stores a set of values, with no duplicates being allowed. It also provides methods for checking if a value already exists in the set.

Unless you need to maintain the order of the elements, consider using UnorderedSetMapper or WhitelistMapper instead, as they’re more efficient.

Examples:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn my_set(&self) -> SetMapper<Type>;
fn my_set(&self) -> SetMapper<Type>;
fn my_set(&self) -> SetMapper<Type>;

Available methods:

insert

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn insert(value: Type) -> bool
fn insert(value: Type) -> bool
fn insert(value: Type) -> bool

Insert the value into the set. Returns false if the item was already present.

remove

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn remove(value: &Type)
fn remove(value: &Type)
fn remove(value: &Type)

Removes the value from the set. Returns false if the set did not contain the value.

contains

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn contains(value: &Type) -> bool
fn contains(value: &Type) -> bool
fn contains(value: &Type) -> bool

Returns true if the mapper contains the given value.

is_empty

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn is_empty() -> bool
fn is_empty() -> bool
fn is_empty() -> bool

Returns true if the mapper has no elements stored.

len

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn len() -> usize
fn len() -> usize
fn len() -> usize

Returns the number of items stored in the mapper.

clear

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn clear()
fn clear()
fn clear()

Clears all the elements from the mapper. This function can run out of gas for big collections.

iter

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn iter() -> Iter<Type>
fn iter() -> Iter<Type>
fn iter() -> Iter<Type>

Returns an iterator over all the stored elements.

UnorderedSetMapper

Same as SetMapper, but does not guarantee the order of the items. More efficient than SetMapper, and should be used instead unless you need to maintain the order. Internally, UnorderedSetMapper uses a VecMapper to store the elements, and additionally, it stores each element’s index to provide O(1) contains.

Examples:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn my_set(&self) -> UnorderedSetMapper<Type>;
fn my_set(&self) -> UnorderedSetMapper<Type>;
fn my_set(&self) -> UnorderedSetMapper<Type>;

Available methods:

UnorderedSetMapper contains the same methods as SetMapper, the only difference being item removal. Instead of remove, we only have swap_remove available.

swap_remove

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn swap_remove(value: &Type) -> bool
fn swap_remove(value: &Type) -> bool
fn swap_remove(value: &Type) -> bool

Uses the internal VecMapper‘s swap_remove method to remove the element. Additionally, it overwrites the last element’s stored index with the removed value’s index. Returns false if the element was not present in the set.

WhitelistMapper

Stores a whitelist of items. Does not provide any means of iterating over the elements, so if you need to iterate over the elements, use UnorderedSetMapper instead. Internally, this mapper simply stores a flag in storage for each item if they’re whitelisted.

Examples:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn my_whitelist(&self) -> WhitelistMapper<Self::Api, Type>
fn my_whitelist(&self) -> WhitelistMapper<Self::Api, Type>
fn my_whitelist(&self) -> WhitelistMapper<Self::Api, Type>

Available methods:

add

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn add(value: &Type)
fn add(value: &Type)
fn add(value: &Type)

Adds the value to the whitelist.

remove

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn remove(value: &Type)
fn remove(value: &Type)
fn remove(value: &Type)

Removes the value from the whitelist.

contains

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn contains(value: &Type) -> bool
fn contains(value: &Type) -> bool
fn contains(value: &Type) -> bool

Returns true if the mapper contains the given value.

require_whitelisted

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn require_whitelisted(value: &Type)
fn require_whitelisted(value: &Type)
fn require_whitelisted(value: &Type)

Will signal an error if the item is not whitelisted. Does nothing otherwise.

LinkedListMapper

Stores a linked list, which allows fast insertion/removal of elements, as well as possibility to iterate over the whole list.

Examples:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn my_linked_list(&self) -> LinkedListMapper<Type>
fn my_linked_list(&self) -> LinkedListMapper<Type>
fn my_linked_list(&self) -> LinkedListMapper<Type>

Available methods:

is_empty

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn is_empty() -> bool
fn is_empty() -> bool
fn is_empty() -> bool

Returns true if the mapper has no elements stored.

len

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn len() -> usize
fn len() -> usize
fn len() -> usize

Returns the number of items stored in the mapper.

clear

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn clear()
fn clear()
fn clear()

Clears all the elements from the mapper. This function can run out of gas for big collections.

iter

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn iter() -> Iter<Type>
fn iter() -> Iter<Type>
fn iter() -> Iter<Type>

Returns an iterator over all the stored elements.

iter_from_node_id

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn iter_from_node_id(node_id: u32) -> Iter<Type>
fn iter_from_node_id(node_id: u32) -> Iter<Type>
fn iter_from_node_id(node_id: u32) -> Iter<Type>

Returns an iterator starting from the given node_id. Useful when splitting iteration over multiple SC calls.

front

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn front() -> Option<LinkedListNode<Type>>
fn back() -> Option<LinkedListNode<Type>>
fn front() -> Option<LinkedListNode<Type>> fn back() -> Option<LinkedListNode<Type>>
fn front() -> Option<LinkedListNode<Type>>
fn back() -> Option<LinkedListNode<Type>>

Returns the first/last element if the list is not empty, None otherwise. A LinkedListNode has the following format:Returns the first/last element if the list is not empty, None otherwise. A LinkedListNode has the following format:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
pub struct LinkedListNode<Type> {
value: Type,
node_id: u32,
next_id: u32,
prev_id: u32,
}
impl<Type> LinkedListNode<Type> {
pub fn get_value_cloned(&self) -> Type {
self.value.clone()
}
pub fn get_value_as_ref(&self) -> &Type {
&self.value
}
pub fn into_value(self) -> Type {
self.value
}
pub fn get_node_id(&self) -> u32 {
self.node_id
}
pub fn get_next_node_id(&self) -> u32 {
self.next_id
}
pub fn get_prev_node_id(&self) -> u32 {
self.prev_id
}
}
pub struct LinkedListNode<Type> { value: Type, node_id: u32, next_id: u32, prev_id: u32, } impl<Type> LinkedListNode<Type> { pub fn get_value_cloned(&self) -> Type { self.value.clone() } pub fn get_value_as_ref(&self) -> &Type { &self.value } pub fn into_value(self) -> Type { self.value } pub fn get_node_id(&self) -> u32 { self.node_id } pub fn get_next_node_id(&self) -> u32 { self.next_id } pub fn get_prev_node_id(&self) -> u32 { self.prev_id } }
pub struct LinkedListNode<Type> {
    value: Type,
    node_id: u32,
    next_id: u32,
    prev_id: u32,
}

impl<Type> LinkedListNode<Type> {
    pub fn get_value_cloned(&self) -> Type {
        self.value.clone()
    }

    pub fn get_value_as_ref(&self) -> &Type {
        &self.value
    }

    pub fn into_value(self) -> Type {
        self.value
    }

    pub fn get_node_id(&self) -> u32 {
        self.node_id
    }

    pub fn get_next_node_id(&self) -> u32 {
        self.next_id
    }

    pub fn get_prev_node_id(&self) -> u32 {
        self.prev_id
    }
}

pop_front/pop_back

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn pop_front(&mut self) -> Option<LinkedListNode<Type>>
fn pop_back(&mut self) -> Option<LinkedListNode<Type>>
fn pop_front(&mut self) -> Option<LinkedListNode<Type>> fn pop_back(&mut self) -> Option<LinkedListNode<Type>>
fn pop_front(&mut self) -> Option<LinkedListNode<Type>>
fn pop_back(&mut self) -> Option<LinkedListNode<Type>>

Removes and returns the first/last element from the list.

push_after/pus_after

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
pub fn push_after(node: &mut LinkedListNode<Type>, element: Type) -> Option<LinkedListNode<Type>>
pub fn push_before(node: &mut LinkedListNode<Type>, element: Type) -> Option<LinkedListNode<Type>>
pub fn push_after(node: &mut LinkedListNode<Type>, element: Type) -> Option<LinkedListNode<Type>> pub fn push_before(node: &mut LinkedListNode<Type>, element: Type) -> Option<LinkedListNode<Type>>
pub fn push_after(node: &mut LinkedListNode<Type>, element: Type) -> Option<LinkedListNode<Type>>
pub fn push_before(node: &mut LinkedListNode<Type>, element: Type) -> Option<LinkedListNode<Type>>

Inserts the given element into the list after/before the given node. Returns the newly inserted node if the insertion was successful, None otherwise.

push_after_node_id/push_before_node_id

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
pub fn push_after_node_id(node_id: usize, element: Type) -> Option<LinkedListNode<Type>>
pub fn push_before_node_id(node_id: usize, element: Type) -> Option<LinkedListNode<Type>>
pub fn push_after_node_id(node_id: usize, element: Type) -> Option<LinkedListNode<Type>> pub fn push_before_node_id(node_id: usize, element: Type) -> Option<LinkedListNode<Type>>
pub fn push_after_node_id(node_id: usize, element: Type) -> Option<LinkedListNode<Type>>
pub fn push_before_node_id(node_id: usize, element: Type) -> Option<LinkedListNode<Type>>

Same as the methods above, but uses node_id instead of a full node struct.

push_front/push_back

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn push_front(element: Type)
fn push_back(element: Type)
fn push_front(element: Type) fn push_back(element: Type)
fn push_front(element: Type)
fn push_back(element: Type)

Pushes the given element at the front/back of the list. Can be seen as specialized versions of push_before_node_id and push_after_node_id.

set_node_value

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn set_node_value(mut node: LinkedListNode<Type>, new_value: Type)
fn set_node_value(mut node: LinkedListNode<Type>, new_value: Type)
fn set_node_value(mut node: LinkedListNode<Type>, new_value: Type)

Sets a node’s value, if the node exists in the list.

set_node_value_by_id

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn set_node_value_by_id(node_id: usize, new_value: Type)
fn set_node_value_by_id(node_id: usize, new_value: Type)
fn set_node_value_by_id(node_id: usize, new_value: Type)

Same as the method above, but uses node_id instead of a full node struct.

remove_node

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn remove_node(node: &LinkedListNode<Type>)
fn remove_node(node: &LinkedListNode<Type>)
fn remove_node(node: &LinkedListNode<Type>)

Removes the node from the list, if it exists.

remove_node_by_id

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn remove_node(node_id: usize)
fn remove_node(node_id: usize)
fn remove_node(node_id: usize)

Same as the method above, but uses node_id instead of a full node struct.

iter

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn iter() -> Iter<Type>
fn iter() -> Iter<Type>
fn iter() -> Iter<Type>

Returns an iterator over all the stored elements.

iter_from_node_id

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn iter_from_node_id(node_id: u32) -> Iter<Type>
fn iter_from_node_id(node_id: u32) -> Iter<Type>
fn iter_from_node_id(node_id: u32) -> Iter<Type>

Returns an iterator starting from the given node_id. Useful when splitting iteration over multiple SC calls.

MapMapper

Stores (key, value) pairs, while also allowing iteration over keys. This is the most expensive mapper to use, so make sure you really need to use it.

Examples:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn my_map(&self) -> MapMapper<KeyType, ValueType>
fn my_map(&self) -> MapMapper<KeyType, ValueType>
fn my_map(&self) -> MapMapper<KeyType, ValueType>

Available methods:

is_empty

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn is_empty() -> bool
fn is_empty() -> bool
fn is_empty() -> bool

Returns true if the mapper has no elements stored.

len

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn len() -> usize
fn len() -> usize
fn len() -> usize

Returns the number of items stored in the mapper.

contains_key

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn contains_key(k: &KeyType) -> bool
fn contains_key(k: &KeyType) -> bool
fn contains_key(k: &KeyType) -> bool
Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn contains_key(k: &KeyType) -> bool
fn contains_key(k: &KeyType) -> bool
fn contains_key(k: &KeyType) -> bool

Returns true if the mapper contains the given key.

get

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn get(k: &KeyType) -> Option<ValueType>
fn get(k: &KeyType) -> Option<ValueType>
fn get(k: &KeyType) -> Option<ValueType>

Returns Some(value) if the key exists. Returns None if the key does not exist in the map.

insert

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn insert(k: KeyType, v: ValueType) -> Option<V>
fn insert(k: KeyType, v: ValueType) -> Option<V>
fn insert(k: KeyType, v: ValueType) -> Option<V>

Inserts the given key, and value pair into the map, and returns Some(old_value) if the key was already present.

remove

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn remove(k: &KeyType) -> Option<ValueType>
fn remove(k: &KeyType) -> Option<ValueType>
fn remove(k: &KeyType) -> Option<ValueType>

Removes the key and the corresponding value from the map, and returns the value. If the key was not present in the map, None is returned.

keys/values/iter

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn keys() -> Keys<KeyType>
fn values() -> Values<ValueType>
fn iter() -> Iter<KeyType, ValueType>
fn keys() -> Keys<KeyType> fn values() -> Values<ValueType> fn iter() -> Iter<KeyType, ValueType>
fn keys() -> Keys<KeyType>
fn values() -> Values<ValueType>
fn iter() -> Iter<KeyType, ValueType>

Provides an iterator over all keys, values, and (key, value) pairs respectively.

Specialized mappers

FungibleTokenMapper

Stores a token identifier (like a SingleValueMapper<TokenIdentifier>) and provides methods for using this token ID directly with the most common API functions. Note that most method calls will fail if the token was not issued previously.

Examples:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn my_token_id(&self) -> FungibleTokenMapper<Self::Api>
fn my_token_id(&self) -> FungibleTokenMapper<Self::Api>
fn my_token_id(&self) -> FungibleTokenMapper<Self::Api>

Available methods:

issue/issue_and_set_all_roles

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn issue(issue_cost: BigUint, token_display_name: ManagedBuffer, token_ticker: ManagedBuffer,initial_supply: BigUint, num_decimals: usize, opt_callback: Option<CallbackClosure<SA>>) -> !
fn issue_and_set_all_roles(issue_cost: BigUint, token_display_name: ManagedBuffer, token_ticker: ManagedBuffer,initial_supply: BigUint, num_decimals: usize, opt_callback: Option<CallbackClosure<SA>>) -> !
fn issue(issue_cost: BigUint, token_display_name: ManagedBuffer, token_ticker: ManagedBuffer,initial_supply: BigUint, num_decimals: usize, opt_callback: Option<CallbackClosure<SA>>) -> ! fn issue_and_set_all_roles(issue_cost: BigUint, token_display_name: ManagedBuffer, token_ticker: ManagedBuffer,initial_supply: BigUint, num_decimals: usize, opt_callback: Option<CallbackClosure<SA>>) -> !
fn issue(issue_cost: BigUint, token_display_name: ManagedBuffer, token_ticker: ManagedBuffer,initial_supply: BigUint, num_decimals: usize, opt_callback: Option<CallbackClosure<SA>>) -> !

fn issue_and_set_all_roles(issue_cost: BigUint, token_display_name: ManagedBuffer, token_ticker: ManagedBuffer,initial_supply: BigUint, num_decimals: usize, opt_callback: Option<CallbackClosure<SA>>) -> !

Issues a new fungible token. issue_cost is 0.05 EGLD (5000000000000000) at the time of writing this, but since this changed in the past, we’ve let it as an argument in case it changes again in the future.

This mapper allows only one issue, so trying to issue multiple types will signal an error.

opt_callback is an optional custom callback you can use for your issue call. We recommed using the default callback. To do so, you need to import multiversx-sc-modules in your Cargo.toml:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
[dependencies.multiversx-sc-modules]
version = "0.39.0"
[dependencies.multiversx-sc-modules] version = "0.39.0"
[dependencies.multiversx-sc-modules]
version = "0.39.0"

Note: current released multiversx-sc version at the time of writing this was 0.39.0, upgrade if necessary.Then you should import the DefaultCallbacksModule in your contract:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
#[multiversx_sc::contract]
pub trait MyContract: multiversx_sc_modules::default_issue_callbacks::DefaultIssueCallbacksModule {
/* ... */
}
#[multiversx_sc::contract] pub trait MyContract: multiversx_sc_modules::default_issue_callbacks::DefaultIssueCallbacksModule { /* ... */ }
#[multiversx_sc::contract]
pub trait MyContract: multiversx_sc_modules::default_issue_callbacks::DefaultIssueCallbacksModule {
    /* ... */
}

Additionally, pass None for opt_callback.

Note the “never” type -> ! as a return type for this function. This means this function will terminate the execution and launch the issue async call, so any code after this call will not be executed.

Alternatively, if you want to issue and also have all roles set for the SC, you can use the issue_and_set_all_roles method instead.

mint

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn mint(amount: BigUint) -> EsdtTokenPayment<Self::Api>
fn mint(amount: BigUint) -> EsdtTokenPayment<Self::Api>
fn mint(amount: BigUint) -> EsdtTokenPayment<Self::Api>

Mints amount tokens for the stored token ID, using the ESDTLocalMint built-in function. Returns a payment struct, containing the token ID and the given amount.

mint_and_send

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn mint_and_send(to: &ManagedAddress, amount: BigUint) -> EsdtTokenPayment<SA>
fn mint_and_send(to: &ManagedAddress, amount: BigUint) -> EsdtTokenPayment<SA>
fn mint_and_send(to: &ManagedAddress, amount: BigUint) -> EsdtTokenPayment<SA>

Same as the method above, but also sends the minted tokens to the given address.

burn

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn burn(amount: &BigUint)
fn burn(amount: &BigUint)
fn burn(amount: &BigUint)

Burns amount tokens, using the ESDTLocalBurn built-in function.

get_balance

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn get_balance() -> BigUint
fn get_balance() -> BigUint
fn get_balance() -> BigUint

Gets the current balance the SC has for the token.

NonFungibleTokenMapper

Similar to the FungibleTokenMapper, but is used for NFT, SFT, and META-ESDT tokens.

issue/issue_and_set_all_roles

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn issue(token_type: EsdtTokenType, issue_cost: BigUint, token_display_name: ManagedBuffer, token_ticker: ManagedBuffer,initial_supply: BigUint, num_decimals: usize, opt_callback: Option<CallbackClosure>) -> !
fn issue_and_set_all_roles(token_type: EsdtTokenType, issue_cost: BigUint, token_display_name: ManagedBuffer, token_ticker: ManagedBuffer,initial_supply: BigUint, num_decimals: usize, opt_callback: Option<CallbackClosure>) -> !
fn issue(token_type: EsdtTokenType, issue_cost: BigUint, token_display_name: ManagedBuffer, token_ticker: ManagedBuffer,initial_supply: BigUint, num_decimals: usize, opt_callback: Option<CallbackClosure>) -> ! fn issue_and_set_all_roles(token_type: EsdtTokenType, issue_cost: BigUint, token_display_name: ManagedBuffer, token_ticker: ManagedBuffer,initial_supply: BigUint, num_decimals: usize, opt_callback: Option<CallbackClosure>) -> !
fn issue(token_type: EsdtTokenType, issue_cost: BigUint, token_display_name: ManagedBuffer, token_ticker: ManagedBuffer,initial_supply: BigUint, num_decimals: usize, opt_callback: Option<CallbackClosure>) -> !

fn issue_and_set_all_roles(token_type: EsdtTokenType, issue_cost: BigUint, token_display_name: ManagedBuffer, token_ticker: ManagedBuffer,initial_supply: BigUint, num_decimals: usize, opt_callback: Option<CallbackClosure>) -> !

Same as the previous issue function, but also takes an EsdtTokenType enum as argument, to decide which type of token to issue. Accepted values are EsdtTokenType::NonFungible, EsdtTokenType::SemiFungible and EsdtTokenType::Meta.

nft_create/nft_create_named

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn nft_create<T: TopEncode>(amount: BigUint, attributes: &T) -> EsdtTokenPayment<Self::Api>
fn nft_create_named<T: TopEncode>(amount: BigUint, name: &ManagedBuffer, attributes: &T) -> EsdtTokenPayment<Self::Api>
fn nft_create<T: TopEncode>(amount: BigUint, attributes: &T) -> EsdtTokenPayment<Self::Api> fn nft_create_named<T: TopEncode>(amount: BigUint, name: &ManagedBuffer, attributes: &T) -> EsdtTokenPayment<Self::Api>
fn nft_create<T: TopEncode>(amount: BigUint, attributes: &T) -> EsdtTokenPayment<Self::Api>

fn nft_create_named<T: TopEncode>(amount: BigUint, name: &ManagedBuffer, attributes: &T) -> EsdtTokenPayment<Self::Api>

Creates an NFT (optionally with a display name) and returns the token ID, the created token’s nonce, and the given amount in a payment struct.

nft_create_and_send/nft_create_and_send_named

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn nft_create_and_send<T: TopEncode>(to: &ManagedAddress, amount: BigUint, attributes: &T,) -> EsdtTokenPayment<Self::Api>
fn nft_create_and_send_named<T: TopEncode>(to: &ManagedAddress, amount: BigUint, name: &ManagedBuffer, attributes: &T,) -> EsdtTokenPayment<Self::Api>
fn nft_create_and_send<T: TopEncode>(to: &ManagedAddress, amount: BigUint, attributes: &T,) -> EsdtTokenPayment<Self::Api> fn nft_create_and_send_named<T: TopEncode>(to: &ManagedAddress, amount: BigUint, name: &ManagedBuffer, attributes: &T,) -> EsdtTokenPayment<Self::Api>
fn nft_create_and_send<T: TopEncode>(to: &ManagedAddress, amount: BigUint, attributes: &T,) -> EsdtTokenPayment<Self::Api>

fn nft_create_and_send_named<T: TopEncode>(to: &ManagedAddress, amount: BigUint, name: &ManagedBuffer, attributes: &T,) -> EsdtTokenPayment<Self::Api>

Same as the methods above, but also sends the created token to the provided address.

nft_add_quantity

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn nft_add_quantity(token_nonce: u64, amount: BigUint) -> EsdtTokenPayment<Self::Api>
fn nft_add_quantity(token_nonce: u64, amount: BigUint) -> EsdtTokenPayment<Self::Api>
fn nft_add_quantity(token_nonce: u64, amount: BigUint) -> EsdtTokenPayment<Self::Api>

Adds quantity for the given token nonce. This can only be used if one of the nft_create functions was used before AND the SC holds at least 1 token for the given nonce.

nft_add_quantity_and_send

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn nft_add_quantity_and_send(to: &ManagedAddress, token_nonce: u64, amount: BigUint) -> EsdtTokenPayment<Self::Api>
fn nft_add_quantity_and_send(to: &ManagedAddress, token_nonce: u64, amount: BigUint) -> EsdtTokenPayment<Self::Api>
fn nft_add_quantity_and_send(to: &ManagedAddress, token_nonce: u64, amount: BigUint) -> EsdtTokenPayment<Self::Api>

Same as the method above, but also sends the tokens to the provided address.

nft_burn

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn nft_burn(token_nonce: u64, amount: &BigUint)
fn nft_burn(token_nonce: u64, amount: &BigUint)
fn nft_burn(token_nonce: u64, amount: &BigUint)

Burns amount tokens for the given nonce.

get_all_token_data

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn get_all_token_data(token_nonce: u64) -> EsdtTokenData<Self::Api>
fn get_all_token_data(token_nonce: u64) -> EsdtTokenData<Self::Api>
fn get_all_token_data(token_nonce: u64) -> EsdtTokenData<Self::Api>

Gets all the token data for the given nonce. The SC must own the given nonce for this function to work.

EsdtTokenData contains the following fields:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
pub struct EsdtTokenData<M: ManagedTypeApi> {
pub token_type: EsdtTokenType,
pub amount: BigUint<M>,
pub frozen: bool,
pub hash: ManagedBuffer<M>,
pub name: ManagedBuffer<M>,
pub attributes: ManagedBuffer<M>,
pub creator: ManagedAddress<M>,
pub royalties: BigUint<M>,
pub uris: ManagedVec<M, ManagedBuffer<M>>,
}
pub struct EsdtTokenData<M: ManagedTypeApi> { pub token_type: EsdtTokenType, pub amount: BigUint<M>, pub frozen: bool, pub hash: ManagedBuffer<M>, pub name: ManagedBuffer<M>, pub attributes: ManagedBuffer<M>, pub creator: ManagedAddress<M>, pub royalties: BigUint<M>, pub uris: ManagedVec<M, ManagedBuffer<M>>, }
pub struct EsdtTokenData<M: ManagedTypeApi> {
    pub token_type: EsdtTokenType,
    pub amount: BigUint<M>,
    pub frozen: bool,
    pub hash: ManagedBuffer<M>,
    pub name: ManagedBuffer<M>,
    pub attributes: ManagedBuffer<M>,
    pub creator: ManagedAddress<M>,
    pub royalties: BigUint<M>,
    pub uris: ManagedVec<M, ManagedBuffer<M>>,
}

get_balance

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn get_balance(token_nonce: u64) -> BigUint
fn get_balance(token_nonce: u64) -> BigUint
fn get_balance(token_nonce: u64) -> BigUint

Gets the SC’s balance for the given token nonce.

get_token_attributes

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn get_token_attributes<T: TopDecode>(token_nonce: u64) -> T
fn get_token_attributes<T: TopDecode>(token_nonce: u64) -> T
fn get_token_attributes<T: TopDecode>(token_nonce: u64) -> T

Gets the attributes for the given token nonce. The SC must own the given nonce for this function to work.

Common functions for FungibleTokenMapper and NonFungibleTokenMapper

Both mappers work similarly, so some functions have the same implementation for both.

is_empty

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn is_empty() -> bool
fn is_empty() -> bool
fn is_empty() -> bool

Returns true if the token ID is not set yet.

get_token_id

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn get_token_id() -> TokenIdentifier<SA>
fn get_token_id() -> TokenIdentifier<SA>
fn get_token_id() -> TokenIdentifier<SA>

Gets the stored token ID.

set_token_id

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn set_token_id(token_id: &TokenIdentifier)
fn set_token_id(token_id: &TokenIdentifier)
fn set_token_id(token_id: &TokenIdentifier)

Manually sets the token ID for this mapper. This can only be used once, and can not be overwritten afterward. This will fail if the token was issued previously, as the token ID was automatically set.

require_same_token/require_all_same_token

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn require_same_token(expected_token_id: &TokenIdentifier<SA>)
fn require_all_same_token(payments: &ManagedVec<EsdtTokenPayment<Self::Api>>)
fn require_same_token(expected_token_id: &TokenIdentifier<SA>) fn require_all_same_token(payments: &ManagedVec<EsdtTokenPayment<Self::Api>>)
fn require_same_token(expected_token_id: &TokenIdentifier<SA>)
fn require_all_same_token(payments: &ManagedVec<EsdtTokenPayment<Self::Api>>)

Will signal an error if the provided token ID argument(s) differs from the stored token. Useful in #[payable] methods when you only want to this token as payment.

set_local_roles

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn set_local_roles(roles: &[EsdtLocalRole], opt_callback: Option<CallbackClosure>) -> !
fn set_local_roles(roles: &[EsdtLocalRole], opt_callback: Option<CallbackClosure>) -> !
fn set_local_roles(roles: &[EsdtLocalRole], opt_callback: Option<CallbackClosure>) -> !

Sets the provided local roles for the token. By default, no callback is used for this call, but you may provide a custom callback if you want to.

You don’t need to call this function if you use issue_and_set_all_roles for issueing.

Same as the issue function, this will terminate execution when called.

set_local_roles_for_address

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn set_local_roles_for_address(address: &ManagedAddress, roles: &[EsdtLocalRole], opt_callback: Option<CallbackClosure>) -> !
fn set_local_roles_for_address(address: &ManagedAddress, roles: &[EsdtLocalRole], opt_callback: Option<CallbackClosure>) -> !
fn set_local_roles_for_address(address: &ManagedAddress, roles: &[EsdtLocalRole], opt_callback: Option<CallbackClosure>) -> !

Similar to the previous function, but sets the roles for a specific address instead of the SC address.

UniqueIdMapper

A special mapper that holds the values from 1 to N, with the following property: if mapper[i] == i, then nothing is actually stored.

This makes it so the mapper initialization is O(1) instead of O(N). Very useful when you want to have a list of available IDs, as its name suggests.

Both the IDs and the indexes are usize.

Note: If you want an in-memory version of this, you can use the SparseArray type provided by the framework.

Examples:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn my_id_mapper(&self) -> UniqueIdMapper<Self::Api>
fn my_id_mapper(&self) -> UniqueIdMapper<Self::Api>
fn my_id_mapper(&self) -> UniqueIdMapper<Self::Api>

Available methods:

set_initial_len

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn set_initial_len(&mut self, len: usize)
fn set_initial_len(&mut self, len: usize)
fn set_initial_len(&mut self, len: usize)

Sets the initial mapper length, i.e. the N. The length may only be set once.

is_empty

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn is_empty() -> bool
fn is_empty() -> bool
fn is_empty() -> bool

Returns true if the mapper has no elements stored.

len

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn len() -> usize
fn len() -> usize
fn len() -> usize

Returns the number of items stored in the mapper.

get

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn get(index: usize) -> usize
fn get(index: usize) -> usize
fn get(index: usize) -> usize

Gets the value for the given index. If the entry is empty, then index is returned, as per the mapper’s property.

set

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn set(&mut self, index: usize, id: usize)
fn set(&mut self, index: usize, id: usize)
fn set(&mut self, index: usize, id: usize)

Sets the value at the given index. The mapper’s internal property of mapper[i] == i if empty entry is maintained.

swap_remove

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn swap_remove(index: usize) -> usize
fn swap_remove(index: usize) -> usize
fn swap_remove(index: usize) -> usize

Removes the ID at the given index and returns it. Also, the value at index is now set the value of the last entry in the map. Length is decreased by 1.

iter

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
fn iter() -> Iter<usize>
fn iter() -> Iter<usize>
fn iter() -> Iter<usize>

Provides an iterator over all the IDs.

Comparisons between the different mappers

SingleValueMapper vs old storage_set/storage_get pairs

There is no difference between SingleValueMapper and the old-school setters/getters. In fact, SingleValueMapper is basically a combination between storage_set, storage_get, storage_is_empty and storage_clear. Use of SingleValueMapper is encouraged, as it’s a lot more compact, and has no performance penalty (if, for example, you never use is_empty(), that code will be removed by the compiler).

SingleValueMapper vs VecMapper

Storing a ManagedVec<T> can be done in two ways:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
#[storage_mapper("my_vec_single)]
fn my_vec_single(&self) -> SingleValueMapper<ManagedVec<T>>
#[storage_mapper("my_vec_mapper)]
fn my_vec_mapper(&self) -> VecMapper<T>;
#[storage_mapper("my_vec_single)] fn my_vec_single(&self) -> SingleValueMapper<ManagedVec<T>> #[storage_mapper("my_vec_mapper)] fn my_vec_mapper(&self) -> VecMapper<T>;
#[storage_mapper("my_vec_single)]
fn my_vec_single(&self) -> SingleValueMapper<ManagedVec<T>>

#[storage_mapper("my_vec_mapper)]
fn my_vec_mapper(&self) -> VecMapper<T>;
Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
#[storage_mapper("my_vec_whitelist)]
fn my_vec_whitelist(&self) -> VecMapper<TokenIdentifier>
#[storage_mapper("my_set_mapper)]
fn my_set_mapper(&self) -> SetMapper<TokenIdentifier>;
#[storage_mapper("my_vec_whitelist)] fn my_vec_whitelist(&self) -> VecMapper<TokenIdentifier> #[storage_mapper("my_set_mapper)] fn my_set_mapper(&self) -> SetMapper<TokenIdentifier>;
#[storage_mapper("my_vec_whitelist)]
fn my_vec_whitelist(&self) -> VecMapper<TokenIdentifier>

#[storage_mapper("my_set_mapper)]
fn my_set_mapper(&self) -> SetMapper<TokenIdentifier>;

This might look very similar, but the implications of using VecMapper for this are very damaging to the potential gas costs. Checking for an item’s existence in VecMapper is done in O(n), with each iteration requiring a new storage read! The worst case scenario is the Token ID is not in the whitelist and the whole Vec is read.

SetMapper is vastly more efficient than this, as it provides checking for a value in O(1). However, this does not come without a cost. This is how the storage looks for a SetMapper with two elements (this snippet is taken from a scenario test):

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
"str:tokenWhitelist.info": "u32:2|u32:1|u32:2|u32:2",
"str:tokenWhitelist.node_idEGLD-123456": "2",
"str:tokenWhitelist.node_idETH-123456": "1",
"str:tokenWhitelist.node_links|u32:1": "u32:0|u32:2",
"str:tokenWhitelist.node_links|u32:2": "u32:1|u32:0",
"str:tokenWhitelist.value|u32:2": "str:EGLD-123456",
"str:tokenWhitelist.value|u32:1": "str:ETH-123456"
"str:tokenWhitelist.info": "u32:2|u32:1|u32:2|u32:2", "str:tokenWhitelist.node_idEGLD-123456": "2", "str:tokenWhitelist.node_idETH-123456": "1", "str:tokenWhitelist.node_links|u32:1": "u32:0|u32:2", "str:tokenWhitelist.node_links|u32:2": "u32:1|u32:0", "str:tokenWhitelist.value|u32:2": "str:EGLD-123456", "str:tokenWhitelist.value|u32:1": "str:ETH-123456"
"str:tokenWhitelist.info": "u32:2|u32:1|u32:2|u32:2",
"str:tokenWhitelist.node_idEGLD-123456": "2",
"str:tokenWhitelist.node_idETH-123456": "1",
"str:tokenWhitelist.node_links|u32:1": "u32:0|u32:2",
"str:tokenWhitelist.node_links|u32:2": "u32:1|u32:0",
"str:tokenWhitelist.value|u32:2": "str:EGLD-123456",
"str:tokenWhitelist.value|u32:1": "str:ETH-123456"

A SetMapper uses 3 * N + 1 storage entries, where N is the number of elements. Checking for an element is very easy, as the only thing the mapper has to do is check the node_id entry for the provided token ID.

Even so, for this particular case, SetMapper is way better than VecMapper.

VecMapper vs LinkedListMapper

LinkedListMapper can be seen as a specialization for the VecMapper. It allows insertion/removal only at either end of the list, known as pushing/popping. It’s also storage-efficient, as it only requires 2 * N + 1 storage entries. The storage for such a mapper looks like this:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
"str:list_mapper.node_links|u32:1": "u32:0|u32:2",
"str:list_mapper.node_links|u32:2": "u32:1|u32:0",
"str:list_mapper.value|u32:1": "123",
"str:list_mapper.value|u32:2": "111",
"str:list_mapper.info": "u32:2|u32:1|u32:2|u32:2"
"str:list_mapper.node_links|u32:1": "u32:0|u32:2", "str:list_mapper.node_links|u32:2": "u32:1|u32:0", "str:list_mapper.value|u32:1": "123", "str:list_mapper.value|u32:2": "111", "str:list_mapper.info": "u32:2|u32:1|u32:2|u32:2"
"str:list_mapper.node_links|u32:1": "u32:0|u32:2",
"str:list_mapper.node_links|u32:2": "u32:1|u32:0",
"str:list_mapper.value|u32:1": "123",
"str:list_mapper.value|u32:2": "111",
"str:list_mapper.info": "u32:2|u32:1|u32:2|u32:2"

This is one of the lesser-used mappers, as its purpose is very specific, but it’s very useful if you ever need to store a queue.

SingleValueMapper vs MapMapper

Believe it or not, most of the time, MapMapper is not even needed, and can simply be replaced by a SingleValueMapper. For example, let’s say you want to store an ID for every Address. It might be tempting to use MapMapper, which would look like this:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
#[storage_mapper("address_id_mapper")]
fn address_id_mapper(&self) -> MapMapper<ManagedAddress, u64>;
#[storage_mapper("address_id_mapper")] fn address_id_mapper(&self) -> MapMapper<ManagedAddress, u64>;
#[storage_mapper("address_id_mapper")]
fn address_id_mapper(&self) -> MapMapper<ManagedAddress, u64>;

This can be replaced with the following SingleValueMapper:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
#[storage_mapper("address_id_mapper")]
fn address_id_mapper(&self, address: &ManagedAddress) -> SingleValueMapper<u64>;
#[storage_mapper("address_id_mapper")] fn address_id_mapper(&self, address: &ManagedAddress) -> SingleValueMapper<u64>;
#[storage_mapper("address_id_mapper")]
fn address_id_mapper(&self, address: &ManagedAddress) -> SingleValueMapper<u64>;

Both of them provide (almost) the same functionality. The difference is that the SingleValueMapper does not provide a way to iterate over all the keys, i.e. Addresses in this case, but it’s also 4-5 times more efficient.

Unless you need to iterate over all the entries, MapMapper should be avoided, as this is the most expensive mapper. It uses 4 * N + 1 storage entries. The storage for a MapMapper looks like this:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
"str:map_mapper.node_links|u32:1": "u32:0|u32:2",
"str:map_mapper.node_links|u32:2": "u32:1|u32:0",
"str:map_mapper.value|u32:1": "123",
"str:map_mapper.value|u32:2": "111",
"str:map_mapper.node_id|u32:123": "1",
"str:map_mapper.node_id|u32:111": "2",
"str:map_mapper.mapped|u32:123": "456",
"str:map_mapper.mapped|u32:111": "222",
"str:map_mapper.info": "u32:2|u32:1|u32:2|u32:2"
"str:map_mapper.node_links|u32:1": "u32:0|u32:2", "str:map_mapper.node_links|u32:2": "u32:1|u32:0", "str:map_mapper.value|u32:1": "123", "str:map_mapper.value|u32:2": "111", "str:map_mapper.node_id|u32:123": "1", "str:map_mapper.node_id|u32:111": "2", "str:map_mapper.mapped|u32:123": "456", "str:map_mapper.mapped|u32:111": "222", "str:map_mapper.info": "u32:2|u32:1|u32:2|u32:2"
"str:map_mapper.node_links|u32:1": "u32:0|u32:2",
"str:map_mapper.node_links|u32:2": "u32:1|u32:0",
"str:map_mapper.value|u32:1": "123",
"str:map_mapper.value|u32:2": "111",
"str:map_mapper.node_id|u32:123": "1",
"str:map_mapper.node_id|u32:111": "2",
"str:map_mapper.mapped|u32:123": "456",
"str:map_mapper.mapped|u32:111": "222",
"str:map_mapper.info": "u32:2|u32:1|u32:2|u32:2"

Keep in mind that all the mappers can have as many additional arguments for the main key. For example, you can have a VecMapper for every user pair, like this:

Plain text
Copy to clipboard
Open code in new window
EnlighterJS 3 Syntax Highlighter
#[storage_mapper("list_per_user_pair")]
fn list_per_user_pair(&self, first_addr: &ManagedAddress, second_addr: &ManagedAddress) -> VecMapper<T>;
#[storage_mapper("list_per_user_pair")] fn list_per_user_pair(&self, first_addr: &ManagedAddress, second_addr: &ManagedAddress) -> VecMapper<T>;
#[storage_mapper("list_per_user_pair")]
fn list_per_user_pair(&self, first_addr: &ManagedAddress, second_addr: &ManagedAddress) -> VecMapper<T>;

Using the correct mapper for your situation can greatly decrease gas costs and complexity, so always remember to carefully evaluate your use case.

With this, you complete this workshop successfully!!