add randomness to UTXO

This commit is contained in:
Sergio Chouhy 2025-05-16 21:55:25 -03:00
parent 4524177931
commit 1490b51f71
10 changed files with 129 additions and 65 deletions

2
Cargo.lock generated
View File

@ -5499,6 +5499,7 @@ dependencies = [
"hex", "hex",
"log", "log",
"monotree", "monotree",
"rand 0.8.5",
"serde", "serde",
"serde_json", "serde_json",
"sha2 0.10.8", "sha2 0.10.8",
@ -6207,6 +6208,7 @@ dependencies = [
"common", "common",
"env_logger", "env_logger",
"log", "log",
"rand 0.8.5",
"risc0-zkvm", "risc0-zkvm",
"serde", "serde",
"serde_json", "serde_json",

View File

@ -1,11 +1,7 @@
use std::{collections::HashMap, hash::Hash}; use std::collections::HashMap;
use anyhow::Result; use anyhow::Result;
use common::{ use common::{merkle_tree_public::TreeHashType, transaction::Tag};
merkle_tree_public::TreeHashType,
nullifier::{self, UTXONullifier},
transaction::Tag,
};
use k256::AffinePoint; use k256::AffinePoint;
use log::info; use log::info;
use serde::Serialize; use serde::Serialize;
@ -108,14 +104,12 @@ impl Account {
amount: u128, amount: u128,
privacy_flag: bool, privacy_flag: bool,
) -> Result<()> { ) -> Result<()> {
let payload_with_asset = UTXOPayload { let asset_utxo = UTXO::new(
owner: self.address, self.address,
asset: serde_json::to_vec(&asset)?, serde_json::to_vec(&asset)?,
amount, amount,
privacy_flag, privacy_flag,
}; );
let asset_utxo = UTXO::create_utxo_from_payload(payload_with_asset)?;
self.utxos.insert(asset_utxo.hash, asset_utxo); self.utxos.insert(asset_utxo.hash, asset_utxo);
@ -153,16 +147,13 @@ impl Default for Account {
mod tests { mod tests {
use super::*; use super::*;
fn generate_dummy_utxo_nullifier() -> UTXONullifier { fn generate_dummy_utxo(address: TreeHashType, amount: u128) -> UTXO {
UTXONullifier::default()
}
fn generate_dummy_utxo(address: TreeHashType, amount: u128) -> anyhow::Result<UTXO> {
let payload = UTXOPayload { let payload = UTXOPayload {
owner: address, owner: address,
asset: vec![], asset: vec![],
amount, amount,
privacy_flag: false, privacy_flag: false,
randomness: [0u8; 32],
}; };
UTXO::create_utxo_from_payload(payload) UTXO::create_utxo_from_payload(payload)
} }
@ -178,8 +169,8 @@ mod tests {
#[test] #[test]
fn test_add_new_utxo_outputs() { fn test_add_new_utxo_outputs() {
let mut account = Account::new(); let mut account = Account::new();
let utxo1 = generate_dummy_utxo(account.address, 100).unwrap(); let utxo1 = generate_dummy_utxo(account.address, 100);
let utxo2 = generate_dummy_utxo(account.address, 200).unwrap(); let utxo2 = generate_dummy_utxo(account.address, 200);
let result = account.add_new_utxo_outputs(vec![utxo1.clone(), utxo2.clone()]); let result = account.add_new_utxo_outputs(vec![utxo1.clone(), utxo2.clone()]);

View File

@ -12,6 +12,7 @@ serde.workspace = true
monotree.workspace = true monotree.workspace = true
sha2.workspace = true sha2.workspace = true
hex.workspace = true hex.workspace = true
rand.workspace = true
[dependencies.common] [dependencies.common]
path = "../common" path = "../common"

View File

@ -1,11 +1,15 @@
use std::hash::RandomState;
use anyhow::Result; use anyhow::Result;
use common::{merkle_tree_public::TreeHashType, AccountId}; use common::{merkle_tree_public::TreeHashType, AccountId};
use log::info; use log::info;
use rand::{rngs::OsRng, RngCore};
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use sha2::{digest::FixedOutput, Digest}; use sha2::{digest::FixedOutput, Digest};
///Raw asset data ///Raw asset data
pub type Asset = Vec<u8>; pub type Asset = Vec<u8>;
pub type Randomness = [u8; 32];
#[derive(Debug, PartialEq, Eq, Clone, Serialize, Deserialize)] #[derive(Debug, PartialEq, Eq, Clone, Serialize, Deserialize)]
///Container for raw utxo payload ///Container for raw utxo payload
@ -16,6 +20,7 @@ pub struct UTXO {
// TODO: change to u256 // TODO: change to u256
pub amount: u128, pub amount: u128,
pub privacy_flag: bool, pub privacy_flag: bool,
pub randomness: Randomness,
} }
#[derive(Debug, Clone, Serialize, Deserialize)] #[derive(Debug, Clone, Serialize, Deserialize)]
@ -25,25 +30,47 @@ pub struct UTXOPayload {
// TODO: change to u256 // TODO: change to u256
pub amount: u128, pub amount: u128,
pub privacy_flag: bool, pub privacy_flag: bool,
pub randomness: Randomness,
}
impl UTXOPayload {
fn to_bytes(&self) -> Vec<u8> {
let mut result = Vec::new();
result.extend_from_slice(&self.owner);
result.extend_from_slice(&self.asset);
result.extend_from_slice(&self.amount.to_be_bytes());
result.push(self.privacy_flag as u8);
result.extend_from_slice(&self.randomness);
result
}
} }
impl UTXO { impl UTXO {
pub fn create_utxo_from_payload(payload_with_asset: UTXOPayload) -> anyhow::Result<Self> { pub fn new(owner: AccountId, asset: Asset, amount: u128, privacy_flag: bool) -> Self {
let raw_payload = serde_json::to_vec(&payload_with_asset)?; let mut randomness = Randomness::default();
OsRng.fill_bytes(&mut randomness);
let payload = UTXOPayload {
owner,
asset,
amount,
privacy_flag,
randomness,
};
Self::create_utxo_from_payload(payload)
}
pub fn create_utxo_from_payload(payload_with_asset: UTXOPayload) -> Self {
let mut hasher = sha2::Sha256::new(); let mut hasher = sha2::Sha256::new();
hasher.update(&payload_with_asset.to_bytes());
hasher.update(&raw_payload);
let hash = <TreeHashType>::from(hasher.finalize_fixed()); let hash = <TreeHashType>::from(hasher.finalize_fixed());
Ok(Self { Self {
hash, hash,
owner: payload_with_asset.owner, owner: payload_with_asset.owner,
asset: payload_with_asset.asset, asset: payload_with_asset.asset,
amount: payload_with_asset.amount, amount: payload_with_asset.amount,
privacy_flag: payload_with_asset.privacy_flag, privacy_flag: payload_with_asset.privacy_flag,
}) randomness: payload_with_asset.randomness,
}
} }
pub fn interpret_asset<'de, ToInterpret: Deserialize<'de>>(&'de self) -> Result<ToInterpret> { pub fn interpret_asset<'de, ToInterpret: Deserialize<'de>>(&'de self) -> Result<ToInterpret> {
@ -56,6 +83,7 @@ impl UTXO {
asset: self.asset.clone(), asset: self.asset.clone(),
amount: self.amount, amount: self.amount,
privacy_flag: self.privacy_flag, privacy_flag: self.privacy_flag,
randomness: self.randomness,
} }
} }
@ -92,13 +120,14 @@ mod tests {
.unwrap(), .unwrap(),
amount: 10, amount: 10,
privacy_flag: false, privacy_flag: false,
randomness: Randomness::default(),
} }
} }
#[test] #[test]
fn test_create_utxo_from_payload() { fn test_create_utxo_from_payload() {
let payload = sample_payload(); let payload = sample_payload();
let utxo = UTXO::create_utxo_from_payload(payload.clone()).unwrap(); let utxo = UTXO::create_utxo_from_payload(payload.clone());
// Ensure hash is created and the UTXO fields are correctly assigned // Ensure hash is created and the UTXO fields are correctly assigned
assert_eq!(utxo.owner, payload.owner); assert_eq!(utxo.owner, payload.owner);
@ -108,7 +137,7 @@ mod tests {
#[test] #[test]
fn test_interpret_asset() { fn test_interpret_asset() {
let payload = sample_payload(); let payload = sample_payload();
let utxo = UTXO::create_utxo_from_payload(payload).unwrap(); let utxo = UTXO::create_utxo_from_payload(payload);
// Interpret asset as TestAsset // Interpret asset as TestAsset
let interpreted: TestAsset = utxo.interpret_asset().unwrap(); let interpreted: TestAsset = utxo.interpret_asset().unwrap();
@ -126,7 +155,7 @@ mod tests {
fn test_interpret_invalid_asset() { fn test_interpret_invalid_asset() {
let mut payload = sample_payload(); let mut payload = sample_payload();
payload.asset = vec![0, 1, 2, 3]; // Invalid data for deserialization payload.asset = vec![0, 1, 2, 3]; // Invalid data for deserialization
let utxo = UTXO::create_utxo_from_payload(payload).unwrap(); let utxo = UTXO::create_utxo_from_payload(payload);
// This should fail because the asset is not valid JSON for TestAsset // This should fail because the asset is not valid JSON for TestAsset
let result: Result<TestAsset> = utxo.interpret_asset(); let result: Result<TestAsset> = utxo.interpret_asset();

View File

@ -10,6 +10,7 @@ env_logger.workspace = true
log.workspace = true log.workspace = true
serde.workspace = true serde.workspace = true
thiserror.workspace = true thiserror.workspace = true
rand.workspace = true
risc0-zkvm = { git = "https://github.com/risc0/risc0.git", branch = "release-2.0" } risc0-zkvm = { git = "https://github.com/risc0/risc0.git", branch = "release-2.0" }
test-methods = { path = "test_methods" } test-methods = { path = "test_methods" }

View File

@ -1,8 +1,9 @@
use accounts::account_core::AccountAddress; use accounts::account_core::AccountAddress;
use common::ExecutionFailureKind; use common::ExecutionFailureKind;
use rand::{rngs::OsRng, RngCore};
use risc0_zkvm::{default_executor, default_prover, sha::Digest, ExecutorEnv, Receipt}; use risc0_zkvm::{default_executor, default_prover, sha::Digest, ExecutorEnv, Receipt};
use serde::Serialize; use serde::Serialize;
use utxo::utxo_core::{UTXOPayload, UTXO}; use utxo::utxo_core::{Randomness, UTXOPayload, UTXO};
pub mod gas_calculator; pub mod gas_calculator;
@ -43,6 +44,12 @@ pub fn prove_mint_utxo(
.write(&owner) .write(&owner)
.map_err(ExecutionFailureKind::write_error)?; .map_err(ExecutionFailureKind::write_error)?;
let mut randomness = Randomness::default();
OsRng.fill_bytes(&mut randomness);
builder
.write(&randomness)
.map_err(ExecutionFailureKind::write_error)?;
let env = builder let env = builder
.build() .build()
.map_err(ExecutionFailureKind::builder_error)?; .map_err(ExecutionFailureKind::builder_error)?;
@ -56,10 +63,7 @@ pub fn prove_mint_utxo(
let digest: UTXOPayload = receipt.journal.decode()?; let digest: UTXOPayload = receipt.journal.decode()?;
Ok(( Ok((UTXO::create_utxo_from_payload(digest), receipt))
UTXO::create_utxo_from_payload(digest).map_err(ExecutionFailureKind::write_error)?,
receipt,
))
} }
pub fn prove_send_utxo( pub fn prove_send_utxo(
@ -78,8 +82,18 @@ pub fn prove_send_utxo(
builder builder
.write(&utxo_payload) .write(&utxo_payload)
.map_err(ExecutionFailureKind::write_error)?; .map_err(ExecutionFailureKind::write_error)?;
let owners_parts_with_randomness = owners_parts
.into_iter()
.map(|(amount, addr)| {
let mut randomness = Randomness::default();
OsRng.fill_bytes(&mut randomness);
(amount, addr, randomness)
})
.collect::<Vec<_>>();
builder builder
.write(&owners_parts) .write(&owners_parts_with_randomness)
.map_err(ExecutionFailureKind::write_error)?; .map_err(ExecutionFailureKind::write_error)?;
let env = builder let env = builder
@ -98,9 +112,8 @@ pub fn prove_send_utxo(
Ok(( Ok((
digest digest
.into_iter() .into_iter()
.map(|(payload, addr)| (UTXO::create_utxo_from_payload(payload).map(|sel| (sel, addr)))) .map(|(payload, addr)| (UTXO::create_utxo_from_payload(payload), addr))
.collect::<anyhow::Result<Vec<(UTXO, [u8; 32])>>>() .collect(),
.map_err(ExecutionFailureKind::write_error)?,
receipt, receipt,
)) ))
} }
@ -148,14 +161,12 @@ pub fn prove_send_utxo_multiple_assets_one_receiver(
.0 .0
.into_iter() .into_iter()
.map(|payload| UTXO::create_utxo_from_payload(payload)) .map(|payload| UTXO::create_utxo_from_payload(payload))
.collect::<anyhow::Result<Vec<UTXO>>>() .collect(),
.map_err(ExecutionFailureKind::write_error)?,
digest digest
.1 .1
.into_iter() .into_iter()
.map(|payload| UTXO::create_utxo_from_payload(payload)) .map(|payload| UTXO::create_utxo_from_payload(payload))
.collect::<anyhow::Result<Vec<UTXO>>>() .collect(),
.map_err(ExecutionFailureKind::write_error)?,
receipt, receipt,
)) ))
} }
@ -171,13 +182,7 @@ pub fn prove_send_utxo_shielded(
return Err(ExecutionFailureKind::AmountMismatchError); return Err(ExecutionFailureKind::AmountMismatchError);
} }
let temp_utxo_to_spend = UTXO::create_utxo_from_payload(UTXOPayload { let temp_utxo_to_spend = UTXO::new(owner, vec![], amount, true);
owner,
asset: vec![],
amount,
privacy_flag: true,
})
.map_err(ExecutionFailureKind::write_error)?;
let utxo_payload = temp_utxo_to_spend.into_payload(); let utxo_payload = temp_utxo_to_spend.into_payload();
let mut builder = ExecutorEnv::builder(); let mut builder = ExecutorEnv::builder();
@ -185,8 +190,18 @@ pub fn prove_send_utxo_shielded(
builder builder
.write(&utxo_payload) .write(&utxo_payload)
.map_err(ExecutionFailureKind::write_error)?; .map_err(ExecutionFailureKind::write_error)?;
let owners_parts_with_randomness = owners_parts
.into_iter()
.map(|(amount, addr)| {
let mut randomness = Randomness::default();
OsRng.fill_bytes(&mut randomness);
(amount, addr, randomness)
})
.collect::<Vec<_>>();
builder builder
.write(&owners_parts) .write(&owners_parts_with_randomness)
.map_err(ExecutionFailureKind::write_error)?; .map_err(ExecutionFailureKind::write_error)?;
let env = builder let env = builder
@ -205,9 +220,8 @@ pub fn prove_send_utxo_shielded(
Ok(( Ok((
digest digest
.into_iter() .into_iter()
.map(|(payload, addr)| (UTXO::create_utxo_from_payload(payload).map(|sel| (sel, addr)))) .map(|(payload, addr)| (UTXO::create_utxo_from_payload(payload), addr))
.collect::<anyhow::Result<Vec<(UTXO, [u8; 32])>>>() .collect(),
.map_err(ExecutionFailureKind::write_error)?,
receipt, receipt,
)) ))
} }
@ -228,8 +242,18 @@ pub fn prove_send_utxo_deshielded(
builder builder
.write(&utxo_payload) .write(&utxo_payload)
.map_err(ExecutionFailureKind::write_error)?; .map_err(ExecutionFailureKind::write_error)?;
let owners_parts_with_randomness = owners_parts
.into_iter()
.map(|(amount, addr)| {
let mut randomness = Randomness::default();
OsRng.fill_bytes(&mut randomness);
(amount, addr, randomness)
})
.collect::<Vec<_>>();
builder builder
.write(&owners_parts) .write(&owners_parts_with_randomness)
.map_err(ExecutionFailureKind::write_error)?; .map_err(ExecutionFailureKind::write_error)?;
let env = builder let env = builder
@ -288,8 +312,7 @@ pub fn prove_mint_utxo_multiple_assets(
digest digest
.into_iter() .into_iter()
.map(UTXO::create_utxo_from_payload) .map(UTXO::create_utxo_from_payload)
.collect::<anyhow::Result<Vec<UTXO>>>() .collect(),
.map_err(ExecutionFailureKind::write_error)?,
receipt, receipt,
)) ))
} }
@ -308,7 +331,7 @@ pub fn execute_mint_utxo(amount_to_mint: u128, owner: AccountAddress) -> anyhow:
let digest: UTXOPayload = receipt.journal.decode()?; let digest: UTXOPayload = receipt.journal.decode()?;
UTXO::create_utxo_from_payload(digest) Ok(UTXO::create_utxo_from_payload(digest))
} }
pub fn execute_send_utxo( pub fn execute_send_utxo(
@ -320,7 +343,16 @@ pub fn execute_send_utxo(
let utxo_payload = spent_utxo.into_payload(); let utxo_payload = spent_utxo.into_payload();
builder.write(&utxo_payload)?; builder.write(&utxo_payload)?;
builder.write(&owners_parts)?; let owners_parts_with_randomness = owners_parts
.into_iter()
.map(|(amount, addr)| {
let mut randomness = Randomness::default();
OsRng.fill_bytes(&mut randomness);
(amount, addr, randomness)
})
.collect::<Vec<_>>();
builder.write(&owners_parts_with_randomness)?;
let env = builder.build()?; let env = builder.build()?;
@ -331,13 +363,12 @@ pub fn execute_send_utxo(
let digest: (UTXOPayload, Vec<(UTXOPayload, AccountAddress)>) = receipt.journal.decode()?; let digest: (UTXOPayload, Vec<(UTXOPayload, AccountAddress)>) = receipt.journal.decode()?;
Ok(( Ok((
UTXO::create_utxo_from_payload(digest.0)?, UTXO::create_utxo_from_payload(digest.0),
digest digest
.1 .1
.into_iter() .into_iter()
.map(|(payload, addr)| (UTXO::create_utxo_from_payload(payload).map(|sel| (sel, addr)))) .map(|(payload, addr)| (UTXO::create_utxo_from_payload(payload), addr))
.collect::<anyhow::Result<Vec<(UTXO, [u8; 32])>>>() .collect(),
.map_err(ExecutionFailureKind::write_error)?,
)) ))
} }

View File

@ -12,17 +12,20 @@ pub struct UTXOPayload {
// TODO: change to u256 // TODO: change to u256
pub amount: u128, pub amount: u128,
pub privacy_flag: bool, pub privacy_flag: bool,
pub randomness: [u8; 32],
} }
fn main() { fn main() {
let amount_to_mint: u128 = env::read(); let amount_to_mint: u128 = env::read();
let owner: AccountAddr = env::read(); let owner: AccountAddr = env::read();
let randomness: [u8; 32] = env::read();
let payload = UTXOPayload { let payload = UTXOPayload {
owner, owner,
asset: vec![], asset: vec![],
amount: amount_to_mint, amount: amount_to_mint,
privacy_flag: true, privacy_flag: true,
randomness,
}; };
env::commit(&(payload)); env::commit(&(payload));

View File

@ -12,12 +12,14 @@ pub struct UTXOPayload {
// TODO: change to u256 // TODO: change to u256
pub amount: u128, pub amount: u128,
pub privacy_flag: bool, pub privacy_flag: bool,
pub randomness: [u8; 32],
} }
fn main() { fn main() {
let amount_to_mint: u128 = env::read(); let amount_to_mint: u128 = env::read();
let number_of_assets: usize = env::read(); let number_of_assets: usize = env::read();
let owner: AccountAddr = env::read(); let owner: AccountAddr = env::read();
let randomness: [u8; 32] = env::read();
let mut asseted_utxos = vec![]; let mut asseted_utxos = vec![];
@ -27,6 +29,7 @@ fn main() {
asset: vec![i as u8], asset: vec![i as u8],
amount: amount_to_mint, amount: amount_to_mint,
privacy_flag: true, privacy_flag: true,
randomness
}; };
asseted_utxos.push(payload); asseted_utxos.push(payload);

View File

@ -12,18 +12,20 @@ pub struct UTXOPayload {
// TODO: change to u256 // TODO: change to u256
pub amount: u128, pub amount: u128,
pub privacy_flag: bool, pub privacy_flag: bool,
pub randomness: [u8; 32],
} }
fn main() { fn main() {
let utxo_spent: UTXOPayload = env::read(); let utxo_spent: UTXOPayload = env::read();
let owners_parts: Vec<(u128, AccountAddr)> = env::read(); let owners_parts: Vec<(u128, AccountAddr, [u8; 32])> = env::read();
let res: Vec<(UTXOPayload, AccountAddr)> = owners_parts.into_iter().map(|(amount, addr)| ( let res: Vec<(UTXOPayload, AccountAddr)> = owners_parts.into_iter().map(|(amount, addr, randomness)| (
UTXOPayload { UTXOPayload {
owner: addr.clone(), owner: addr.clone(),
asset: vec![], asset: vec![],
amount, amount,
privacy_flag: true, privacy_flag: true,
randomness,
}, },
addr addr
)).collect(); )).collect();

View File

@ -12,6 +12,7 @@ pub struct UTXOPayload {
// TODO: change to u256 // TODO: change to u256
pub amount: u128, pub amount: u128,
pub privacy_flag: bool, pub privacy_flag: bool,
pub randomness: [u8; 32],
} }
fn main() { fn main() {