How to Use Mappings in Miden Assembly

Using mappings in Miden assembly for storing key value pairs

Overview

In this example, we will explore how to use mappings in Miden Assembly. Mappings are essential data structures that store key-value pairs. We will demonstrate how to create an account that contains a mapping and then call a procedure in that account to update the mapping.

At a high level, this example involves:

  • Setting up an account with a mapping stored in one of its storage slots.
  • Writing a smart contract in Miden Assembly that includes procedures to read from and write to the mapping.
  • Creating a transaction script that calls these procedures.
  • Using Rust code to deploy the account and submit a transaction that updates the mapping.
    After the Miden Assembly snippets, we explain that the transaction script calls a procedure in the account. This procedure then updates the mapping by modifying the mapping stored in the account's storage slot.

What we'll cover

  • How to Use Mappings in Miden Assembly: See how to create a smart contract that uses a mapping.
  • How to Link Libraries in Miden Assembly: Demonstrate how to link procedures across Accounts, Notes, and Scripts.

Step-by-step process

  1. Setting up an account with a mapping
    In this step, you create an account that has a storage slot configured as a mapping. The account smart contract code (shown below) defines procedures to write to and read from this mapping.

  2. Creating a script that calls a procedure in the account:
    Next, you create a transaction script that calls the procedures defined in the account. This script sends the key-value data and then invokes the account procedure, which updates the mapping.

  3. How to read and write to a mapping in MASM:
    Finally, we demonstrate how to use MASM instructions to interact with the mapping. The smart contract uses standard procedures to set a mapping item, retrieve a value from the mapping, and get the current mapping root.


Example of smart contract that uses a mapping

use.miden::account use.std::sys # Inputs: [KEY, VALUE] # Outputs: [] export.write_to_map # The storage map is in storage slot 1 push.1 # => [index, KEY, VALUE] # Setting the key value pair in the map exec.account::set_map_item # => [OLD_MAP_ROOT, OLD_MAP_VALUE] dropw dropw dropw dropw # => [] # Incrementing the nonce by 1 push.1 exec.account::incr_nonce # => [] end # Inputs: [KEY] # Outputs: [VALUE] export.get_value_in_map # The storage map is in storage slot 1 push.1 # => [index] exec.account::get_map_item # => [VALUE] end # Inputs: [] # Outputs: [CURRENT_ROOT] export.get_current_map_root # Getting the current root from slot 1 push.1 exec.account::get_item # => [CURRENT_ROOT] exec.sys::truncate_stack # => [CURRENT_ROOT] end

Explanation of the assembly code

  • write_to_map:
    The procedure takes a key and a value as inputs. It pushes the storage index (0 for our mapping) onto the stack, then calls the set_map_item procedure from the account library to update the mapping. After updating the map, it drops any unused outputs and increments the nonce.

  • get_value_in_map:
    This procedure takes a key as input and retrieves the corresponding value from the mapping by calling get_map_item after pushing the mapping index.

  • get_current_map_root:
    This procedure retrieves the current root of the mapping (stored at index 0) by calling get_item and then truncating the stack to leave only the mapping root.

Security Note: The procedure write_to_map calls the account procedure incr_nonce. This allows any external account to be able to write to the storage map of the account. Smart contract developers should know that procedures that call the account::incr_nonce procedure allow anyone to call the procedure and modify the state of the account.

Transaction script that calls the smart contract

use.miden_by_example::mapping_example_contract use.std::sys begin push.1.2.3.4 push.0.0.0.0 # => [KEY, VALUE] call.mapping_example_contract::write_to_map # => [] push.0.0.0.0 # => [KEY] call.mapping_example_contract::get_value_in_map # => [VALUE] dropw # => [] call.mapping_example_contract::get_current_map_root # => [CURRENT_ROOT] exec.sys::truncate_stack end

Explanation of the transaction script

The transaction script does the following:

  • It pushes a key ([0.0.0.0]) and a value ([1.2.3.4]) onto the stack.
  • It calls the write_to_map procedure, which is defined in the account’s smart contract. This updates the mapping in the account.
  • It then pushes the key again and calls get_value_in_map to retrieve the value associated with the key.
  • Finally, it calls get_current_map_root to get the current state (root) of the mapping.

The script calls the write_to_map procedure in the account which writes the key value pair to the mapping.


Rust code that sets everything up

Below is the Rust code that deploys the smart contract, creates the transaction script, and submits a transaction to update the mapping in the account:

use std::{fs, path::Path, sync::Arc}; use rand::Rng; use rand_chacha::rand_core::SeedableRng; use rand_chacha::ChaCha20Rng; use miden_client::{ account::{AccountStorageMode, AccountType}, crypto::RpoRandomCoin, rpc::{Endpoint, TonicRpcClient}, store::{sqlite_store::SqliteStore, StoreAuthenticator}, transaction::{TransactionKernel, TransactionRequestBuilder}, Client, ClientError, Felt, }; use miden_objects::{ account::{AccountBuilder, AccountComponent, AuthSecretKey, StorageMap, StorageSlot}, assembly::{Assembler, DefaultSourceManager}, crypto::dsa::rpo_falcon512::SecretKey, transaction::TransactionScript, Word, }; use miden_assembly::{ ast::{Module, ModuleKind}, LibraryPath, }; pub async fn initialize_client() -> Result<Client<RpoRandomCoin>, ClientError> { // RPC endpoint and timeout let endpoint = Endpoint::new( "https".to_string(), "rpc.testnet.miden.io".to_string(), Some(443), ); let timeout_ms = 10_000; // Build RPC client let rpc_api = Box::new(TonicRpcClient::new(endpoint, timeout_ms)); // Seed RNG let mut seed_rng = rand::thread_rng(); let coin_seed: [u64; 4] = seed_rng.gen(); // Create random coin instance let rng = RpoRandomCoin::new(coin_seed.map(Felt::new)); // SQLite path let store_path = "store.sqlite3"; // Initialize SQLite store let store = SqliteStore::new(store_path.into()) .await .map_err(ClientError::StoreError)?; let arc_store = Arc::new(store); // Create authenticator referencing the store and RNG let authenticator = StoreAuthenticator::new_with_rng(arc_store.clone(), rng); // Instantiate client (toggle debug mode as needed) let client = Client::new(rpc_api, rng, arc_store, Arc::new(authenticator), true); Ok(client) } pub fn get_new_pk_and_authenticator() -> (Word, AuthSecretKey) { // Create a deterministic RNG with zeroed seed let seed = [0_u8; 32]; let mut rng = ChaCha20Rng::from_seed(seed); // Generate Falcon-512 secret key let sec_key = SecretKey::with_rng(&mut rng); // Convert public key to `Word` (4xFelt) let pub_key: Word = sec_key.public_key().into(); // Wrap secret key in `AuthSecretKey` let auth_secret_key = AuthSecretKey::RpoFalcon512(sec_key); (pub_key, auth_secret_key) } /// Creates a library from the provided source code and library path. /// /// # Arguments /// * `assembler` - The assembler instance used to build the library. /// * `library_path` - The full library path as a string (e.g., "custom_contract::mapping_example"). /// * `source_code` - The MASM source code for the module. /// /// # Returns /// A `miden_assembly::Library` that can be added to the transaction script. fn create_library( assembler: Assembler, library_path: &str, source_code: &str, ) -> Result<miden_assembly::Library, Box<dyn std::error::Error>> { let source_manager = Arc::new(DefaultSourceManager::default()); let module = Module::parser(ModuleKind::Library).parse_str( LibraryPath::new(library_path)?, source_code, &source_manager, )?; let library = assembler.clone().assemble_library([module])?; Ok(library) } #[tokio::main] async fn main() -> Result<(), ClientError> { // ------------------------------------------------------------------------- // Initialize the Miden client // ------------------------------------------------------------------------- let mut client = initialize_client().await?; println!("Client initialized successfully."); // Fetch and display the latest synchronized block number from the node. let sync_summary = client.sync_state().await.unwrap(); println!("Latest block: {}", sync_summary.block_num); // ------------------------------------------------------------------------- // STEP 1: Deploy a smart contract with a mapping // ------------------------------------------------------------------------- println!("\n[STEP 1] Deploy a smart contract with a mapping"); // Load the MASM file for the counter contract let file_path = Path::new("./masm/accounts/mapping_example_contract.masm"); let account_code = fs::read_to_string(file_path).unwrap(); // Prepare assembler (debug mode = true) let assembler: Assembler = TransactionKernel::assembler().with_debug_mode(true); // Using an empty storage value in slot 0 since this is usually resurved // for the account pub_key and metadata let empty_storage_slot = StorageSlot::empty_value(); // initialize storage map let storage_map = StorageMap::new(); let storage_slot_map = StorageSlot::Map(storage_map.clone()); // Compile the account code into `AccountComponent` with one storage slot let mapping_contract_component = AccountComponent::compile( account_code.clone(), assembler.clone(), vec![empty_storage_slot, storage_slot_map], ) .unwrap() .with_supports_all_types(); // Init seed for the counter contract let init_seed = ChaCha20Rng::from_entropy().gen(); // Anchor block of the account let anchor_block = client.get_latest_epoch_block().await.unwrap(); // Build the new `Account` with the component let (mapping_example_contract, _seed) = AccountBuilder::new(init_seed) .anchor((&anchor_block).try_into().unwrap()) .account_type(AccountType::RegularAccountImmutableCode) .storage_mode(AccountStorageMode::Public) .with_component(mapping_contract_component.clone()) .build() .unwrap(); let (_, auth_secret_key) = get_new_pk_and_authenticator(); client .add_account( &mapping_example_contract.clone(), Some(_seed), &auth_secret_key, false, ) .await .unwrap(); // ------------------------------------------------------------------------- // STEP 2: Call the Mapping Contract with a Script // ------------------------------------------------------------------------- println!("\n[STEP 2] Call Mapping Contract With Script"); let script_code = fs::read_to_string(Path::new("./masm/scripts/mapping_example_script.masm")).unwrap(); // Create the library from the account source code using the helper function. let account_component_lib = create_library( assembler.clone(), "miden_by_example::mapping_example_contract", &account_code, ) .unwrap(); // Compile the transaction script with the library. let tx_script = TransactionScript::compile( script_code, [], assembler.with_library(&account_component_lib).unwrap(), ) .unwrap(); // Build a transaction request with the custom script let tx_increment_request = TransactionRequestBuilder::new() .with_custom_script(tx_script) .unwrap() .build(); // Execute the transaction locally let tx_result = client .new_transaction(mapping_example_contract.id(), tx_increment_request) .await .unwrap(); let tx_id = tx_result.executed_transaction().id(); println!( "View transaction on MidenScan: https://testnet.midenscan.com/tx/{:?}", tx_id ); // Submit transaction to the network let _ = client.submit_transaction(tx_result).await; client.sync_state().await.unwrap(); let account = client .get_account(mapping_example_contract.id()) .await .unwrap(); let index = 1; let key = [Felt::new(0), Felt::new(0), Felt::new(0), Felt::new(0)]; println!( "Mapping state\n Index: {:?}\n Key: {:?}\n Value: {:?}", index, key, account .unwrap() .account() .storage() .get_map_item(index, key) ); Ok(()) }

What the Rust code does

  • Client Initialization:
    The client is initialized with a connection to the Miden Testnet and a SQLite store. This sets up the environment to deploy and interact with accounts.

  • Deploying the Smart Contract:
    The account containing the mapping is created by reading the MASM smart contract from a file, compiling it into an AccountComponent, and deploying it using an AccountBuilder.

  • Creating and Executing a Transaction Script:
    A separate MASM script is compiled into a TransactionScript. This script calls the smart contract's procedures to write to and then read from the mapping.

  • Displaying the Result:
    Finally, after the transaction is processed, the code reads the updated state of the mapping in the account.


Running the example

To run the full example, navigate to the rust-client directory in the miden-tutorials repository and run this command:

cd rust-client cargo run --release --bin mapping_example

This example shows how the script calls the procedure in the account, which then updates the mapping stored within the account. The mapping update is verified by reading the mapping’s key-value pair after the transaction completes.