How to Create a Custom Note

Creating notes with custom logic

Overview

In this guide, we will create a custom note on Miden that can only be consumed by someone who knows the preimage of the hash stored in the note. This approach securely embeds assets into the note and restricts spending to those who possess the correct secret number.

By following the steps below and using the Miden Assembly code and Rust example, you will learn how to:

  • Create a note with custom logic.
  • Leverage Miden’s privacy features to keep certain transaction details private.

Unlike Ethereum, where all pending transactions are publicly visible in the mempool, Miden enables you to partially or completely hide transaction details.

What we'll cover

  • Writing Miden assembly for a note
  • Consuming notes

Step-by-step process

1. Creating two accounts: Alice & Bob

First, we create two basic accounts for the two users:

  • Alice: The account that creates and funds the custom note.
  • Bob: The account that will consume the note if they know the correct secret.

2. Hashing the secret number

The security of the custom note hinges on a secret number. Here, we will:

  • Choose a secret number (for example, an array of four integers).
  • For simplicity, we're only hashing 4 elements. Therefore, we prepend an empty word—consisting of 4 zero integers—as a placeholder. This is required by the RPO hashing algorithm to ensure the input has the correct structure and length for proper processing.
  • Compute the hash of the secret. The resulting hash will serve as the note’s input, meaning that the note can only be consumed if the secret number’s hash preimage is provided during consumption.

3. Creating the custom note

Now, combine the minted asset and the secret hash to build the custom note. The note is created using the following key steps:

  1. Note Inputs:

    • The note is set up with the asset and the hash of the secret number as its input.
  2. Miden Assembly Code:

    • The Miden assembly note script ensures that the note can only be consumed if the provided secret, when hashed, matches the hash stored in the note input.

Below is the Miden Assembly code for the note:

use.miden::note use.miden::contracts::wallets::basic->wallet # => [HASH_PREIMAGE_SECRET] begin # Hashing the secret number hperm # => [F,E,D] # E is digest dropw swapw dropw # => [DIGEST] # Writing the note inputs to memory push.0 exec.note::get_inputs drop drop # => [DIGEST] # Pad stack and load note inputs from memory padw push.0 mem_loadw # => [INPUTS, DIGEST] # Assert that the note input matches the digest # Will fail if the two hashes do not match assert_eqw # => [] # Write the asset in note to memory address 0 push.0 exec.note::get_assets # => [num_assets, dest_ptr] drop # => [dest_ptr] # Load asset from memory mem_loadw # => [ASSET] # Call receive asset in wallet call.wallet::receive_asset # => [] end

How the assembly code works:

  1. Passing the Secret:
    The secret number is passed as Note Arguments into the note.

  2. Hashing the Secret:
    The hperm instruction applies a hash permutation to the secret number, resulting in a hash that takes up four stack elements.

  3. Stack Cleanup and Comparison:
    The assembly code extracts the digest, loads the note inputs from memory and checks if the computed hash matches the note’s stored hash.

  4. Asset Transfer:
    If the hash of the number passed in as Note Arguments matches the hash stored in the note inputs, the script continues, and the asset stored in the note is loaded from memory and passed to Bob’s wallet via the wallet::receive_asset function.

5. Consuming the note

With the note created, Bob can now consume it—but only if he provides the correct secret. When Bob initiates the transaction to consume the note, he must supply the same secret number used when Alice created the note. The custom note’s logic will hash the secret and compare it with its stored hash. If they match, Bob’s wallet receives the asset.


Full Rust code example

The following Rust code demonstrates how to implement the steps outlined above using the Miden client library:

use rand::Rng; use rand_chacha::{rand_core::SeedableRng, ChaCha20Rng}; use std::{fs, path::Path, sync::Arc}; use tokio::time::{sleep, Duration}; use miden_client::{ account::{ component::{BasicFungibleFaucet, BasicWallet, RpoFalcon512}, AccountBuilder, AccountStorageMode, AccountType, }, asset::{FungibleAsset, TokenSymbol}, crypto::RpoRandomCoin, note::{ Note, NoteAssets, NoteExecutionHint, NoteExecutionMode, NoteInputs, NoteMetadata, NoteRecipient, NoteScript, NoteTag, NoteType, }, rpc::{Endpoint, TonicRpcClient}, store::{sqlite_store::SqliteStore, StoreAuthenticator}, transaction::{OutputNote, TransactionKernel, TransactionRequestBuilder}, Client, ClientError, Felt, }; use miden_crypto::{hash::rpo::Rpo256 as Hasher, rand::FeltRng}; use miden_objects::{account::AuthSecretKey, crypto::dsa::rpo_falcon512::SecretKey, Word}; // Initialize client helper pub async fn initialize_client() -> Result<Client<RpoRandomCoin>, ClientError> { let endpoint = Endpoint::new("https".into(), "rpc.testnet.miden.io".into(), Some(443)); let rpc_api = Box::new(TonicRpcClient::new(endpoint, 10_000)); let coin_seed: [u64; 4] = rand::thread_rng().gen(); let rng = RpoRandomCoin::new(coin_seed.map(Felt::new)); let store = SqliteStore::new("store.sqlite3".into()) .await .map_err(ClientError::StoreError)?; let arc_store = Arc::new(store); let authenticator = StoreAuthenticator::new_with_rng(arc_store.clone(), rng); Ok(Client::new( rpc_api, rng, arc_store, Arc::new(authenticator), true, )) } // Helper to create keys & authenticator pub fn get_new_pk_and_authenticator() -> (Word, AuthSecretKey) { let seed = [0_u8; 32]; let mut rng = ChaCha20Rng::from_seed(seed); let sec_key = SecretKey::with_rng(&mut rng); ( sec_key.public_key().into(), AuthSecretKey::RpoFalcon512(sec_key), ) } // Helper to create a basic account async fn create_basic_account( client: &mut Client<RpoRandomCoin>, ) -> Result<miden_client::account::Account, ClientError> { let mut init_seed = [0u8; 32]; client.rng().fill_bytes(&mut init_seed); let key_pair = SecretKey::with_rng(client.rng()); let anchor_block = client.get_latest_epoch_block().await.unwrap(); let builder = AccountBuilder::new(init_seed) .anchor((&anchor_block).try_into().unwrap()) .account_type(AccountType::RegularAccountUpdatableCode) .storage_mode(AccountStorageMode::Public) .with_component(RpoFalcon512::new(key_pair.public_key())) .with_component(BasicWallet); let (account, seed) = builder.build().unwrap(); client .add_account( &account, Some(seed), &AuthSecretKey::RpoFalcon512(key_pair), false, ) .await?; Ok(account) } async fn create_basic_faucet( client: &mut Client<RpoRandomCoin>, ) -> Result<miden_client::account::Account, ClientError> { let mut init_seed = [0u8; 32]; client.rng().fill_bytes(&mut init_seed); let key_pair = SecretKey::with_rng(client.rng()); let anchor_block = client.get_latest_epoch_block().await.unwrap(); let symbol = TokenSymbol::new("MID").unwrap(); let decimals = 8; let max_supply = Felt::new(1_000_000); let builder = AccountBuilder::new(init_seed) .anchor((&anchor_block).try_into().unwrap()) .account_type(AccountType::FungibleFaucet) .storage_mode(AccountStorageMode::Public) .with_component(RpoFalcon512::new(key_pair.public_key())) .with_component(BasicFungibleFaucet::new(symbol, decimals, max_supply).unwrap()); let (account, seed) = builder.build().unwrap(); client .add_account( &account, Some(seed), &AuthSecretKey::RpoFalcon512(key_pair), false, ) .await?; Ok(account) } // Helper to wait until an account has the expected number of consumable notes async fn wait_for_notes( client: &mut Client<RpoRandomCoin>, account_id: &miden_client::account::Account, expected: usize, ) -> Result<(), ClientError> { loop { client.sync_state().await?; let notes = client.get_consumable_notes(Some(account_id.id())).await?; if notes.len() >= expected { break; } println!( "{} consumable notes found for account {}. Waiting...", notes.len(), account_id.id().to_hex() ); sleep(Duration::from_secs(3)).await; } Ok(()) } #[tokio::main] async fn main() -> Result<(), ClientError> { let mut client = initialize_client().await?; println!( "Client initialized successfully. Latest block: {}", client.sync_state().await.unwrap().block_num ); // ------------------------------------------------------------------------- // STEP 1: Create accounts and deploy faucet // ------------------------------------------------------------------------- println!("\n[STEP 1] Creating new accounts"); let alice_account = create_basic_account(&mut client).await?; println!("Alice's account ID: {:?}", alice_account.id().to_hex()); let bob_account = create_basic_account(&mut client).await?; println!("Bob's account ID: {:?}", bob_account.id().to_hex()); println!("\nDeploying a new fungible faucet."); let faucet = create_basic_faucet(&mut client).await?; println!("Faucet account ID: {:?}", faucet.id().to_hex()); client.sync_state().await?; // ------------------------------------------------------------------------- // STEP 2: Mint tokens with P2ID // ------------------------------------------------------------------------- println!("\n[STEP 2] Mint tokens with P2ID"); let faucet_id = faucet.id(); let amount: u64 = 100; let mint_amount = FungibleAsset::new(faucet_id, amount).unwrap(); let tx_req = TransactionRequestBuilder::mint_fungible_asset( mint_amount.clone(), alice_account.id(), NoteType::Public, client.rng(), ) .unwrap() .build(); let tx_exec = client.new_transaction(faucet.id(), tx_req).await?; client.submit_transaction(tx_exec.clone()).await?; let p2id_note = if let OutputNote::Full(note) = tx_exec.created_notes().get_note(0) { note.clone() } else { panic!("Expected OutputNote::Full"); }; sleep(Duration::from_secs(3)).await; wait_for_notes(&mut client, &alice_account, 1).await?; let consume_req = TransactionRequestBuilder::new() .with_authenticated_input_notes([(p2id_note.id(), None)]) .build(); let tx_exec = client .new_transaction(alice_account.id(), consume_req) .await?; client.submit_transaction(tx_exec).await?; client.sync_state().await?; // ------------------------------------------------------------------------- // STEP 3: Create custom note // ------------------------------------------------------------------------- println!("\n[STEP 3] Create custom note"); let mut secret_vals = vec![Felt::new(1), Felt::new(2), Felt::new(3), Felt::new(4)]; secret_vals.splice(0..0, Word::default().iter().cloned()); let digest = Hasher::hash_elements(&secret_vals); println!("digest: {:?}", digest); let assembler = TransactionKernel::assembler().with_debug_mode(true); let code = fs::read_to_string(Path::new("../masm/notes/hash_preimage_note.masm")).unwrap(); let rng = client.rng(); let serial_num = rng.draw_word(); let note_script = NoteScript::compile(code, assembler).unwrap(); let note_inputs = NoteInputs::new(digest.to_vec()).unwrap(); let recipient = NoteRecipient::new(serial_num, note_script, note_inputs); let tag = NoteTag::for_public_use_case(0, 0, NoteExecutionMode::Local).unwrap(); let metadata = NoteMetadata::new( alice_account.id(), NoteType::Public, tag, NoteExecutionHint::always(), Felt::new(0), )?; let vault = NoteAssets::new(vec![mint_amount.clone().into()])?; let custom_note = Note::new(vault, metadata, recipient); println!("note hash: {:?}", custom_note.hash()); let note_req = TransactionRequestBuilder::new() .with_own_output_notes(vec![OutputNote::Full(custom_note.clone())]) .unwrap() .build(); let tx_result = client .new_transaction(alice_account.id(), note_req) .await .unwrap(); println!( "View transaction on MidenScan: https://testnet.midenscan.com/tx/{:?}", tx_result.executed_transaction().id() ); let _ = client.submit_transaction(tx_result).await; client.sync_state().await?; // ------------------------------------------------------------------------- // STEP 4: Consume the Custom Note // ------------------------------------------------------------------------- wait_for_notes(&mut client, &bob_account, 1).await?; println!("\n[STEP 4] Bob consumes the Custom Note with Correct Secret"); let secret = [Felt::new(1), Felt::new(2), Felt::new(3), Felt::new(4)]; let consume_custom_req = TransactionRequestBuilder::new() .with_authenticated_input_notes([(custom_note.id(), Some(secret))]) .build(); let tx_result = client .new_transaction(bob_account.id(), consume_custom_req) .await .unwrap(); println!( "Consumed Note Tx on MidenScan: https://testnet.midenscan.com/tx/{:?}", tx_result.executed_transaction().id() ); println!("account delta: {:?}", tx_result.account_delta().vault()); let _ = client.submit_transaction(tx_result).await; Ok(()) }

The output of our program will look something like this:

Client initialized successfully. Latest block: 712247 [STEP 1] Creating new accounts Alice's account ID: "0x29b5fce5574c7610000aefa165f3dc" Bob's account ID: "0x771d0942da4eaa10000abaf3cd9cbe" Deploying a new fungible faucet. Faucet account ID: "0xbdcf27380ed0a720000a475c9d245c" [STEP 2] Mint tokens with P2ID one or more warnings were emitted [STEP 3] Create custom note digest: RpoDigest([14371582251229115050, 1386930022051078873, 17689831064175867466, 9632123050519021080]) note hash: RpoDigest([8719495106795865315, 8421971501316867300, 4856711660892764009, 18308680045181895277]) View transaction on MidenScan: https://testnet.midenscan.com/tx/0x5937e82b3496d2059967cf01f9ddb7560eed7808f0543b507fa3a20a0705b037 0 consumable notes found for account 0x771d0942da4eaa10000abaf3cd9cbe. Waiting... 0 consumable notes found for account 0x771d0942da4eaa10000abaf3cd9cbe. Waiting... [STEP 4] Bob consumes the Custom Note with Correct Secret one or more warnings were emitted Consumed Note Tx on MidenScan: https://testnet.midenscan.com/tx/0xf1be229416c77e7d7368de3dfffe0ac0ca8d2dceefbfcf9ba5641f1dce556d7a account delta: AccountVaultDelta { fungible: FungibleAssetDelta({V0(AccountIdV0 { prefix: 13677193715067692832, suffix: 2893212866075648 }): 100}), non_fungible: NonFungibleAssetDelta({}) }

Conclusion

You have now seen how to create a custom note on Miden that requires a secret preimage to be consumed. We covered:

  1. Creating and funding accounts (Alice and Bob)
  2. Hashing a secret number
  3. Building a note with custom logic in Miden Assembly
  4. Consuming the note by providing the correct secret

By leveraging Miden’s privacy features, you can create customized logic for secure asset transfers that depend on keeping parts of the transaction private.

Running the example

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

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