From d29df460053fcb4d20997d125e853143fafe76d1 Mon Sep 17 00:00:00 2001 From: moana Date: Tue, 18 Jun 2024 16:26:43 +0200 Subject: [PATCH] core: Implement sender-en- and de-cryption methods --- core/CHANGELOG.md | 4 + core/src/encryption/elgamal.rs | 6 +- core/src/lib.rs | 4 +- core/src/note.rs | 114 +++++++++++------ core/tests/encryption.rs | 4 +- core/tests/keys.rs | 31 +++-- core/tests/note_test.rs | 222 +++++++++++++-------------------- core/tests/transaction.rs | 14 ++- 8 files changed, 204 insertions(+), 195 deletions(-) diff --git a/core/CHANGELOG.md b/core/CHANGELOG.md index 3cec32f..d96903a 100644 --- a/core/CHANGELOG.md +++ b/core/CHANGELOG.md @@ -9,6 +9,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ### Added +- Add `encrypt_sender` function to encrypt the sender with the npk [#214] +- Add `decrypt_sender` method to the `Note` [#214] - Add `elgamal::encrypt` and `elgamal::decrypt` - Add `stealth_address` and `sync_address` functions directly to note [#208] - Add a light sync method in the `ViewKey` [#199] @@ -21,6 +23,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Rename `tx_max_fee` to `max_fee` [#214] - Add `sender_enc` field to the `Note` [#214] - Add `sender_blinder` parameter for `Note` contructors [#214] +- Add `sender_pk` parameter for `Note` contructors [#214] +- Add `sender_enc` parameter for `Note::transparent_stealth` [#214] - Rename `encryption_blinder` to `value_blinder` [#214] - Rename `NOTE_ENCRYPTION_SIZE` to `NOTE_VALUE_ENC_SIZE` [#214] - Move `OUTPUT_NOTES` to crate root diff --git a/core/src/encryption/elgamal.rs b/core/src/encryption/elgamal.rs index b9fcb08..3fd80cd 100644 --- a/core/src/encryption/elgamal.rs +++ b/core/src/encryption/elgamal.rs @@ -33,9 +33,11 @@ pub fn encrypt( /// Returns a JubJubExtended plaintext. pub fn decrypt( secret_key: &JubJubScalar, - ciphertext_1: &JubJubExtended, - ciphertext_2: &JubJubExtended, + ciphertext: &(JubJubExtended, JubJubExtended), ) -> JubJubExtended { + let ciphertext_1 = ciphertext.0; + let ciphertext_2 = ciphertext.1; + // return the plaintext ciphertext_2 - ciphertext_1 * secret_key } diff --git a/core/src/lib.rs b/core/src/lib.rs index fe02181..2c065b8 100644 --- a/core/src/lib.rs +++ b/core/src/lib.rs @@ -30,7 +30,9 @@ pub use keys::hash; pub use keys::public::PublicKey; pub use keys::secret::SecretKey; pub use keys::view::ViewKey; -pub use note::{Note, NoteType, VALUE_ENC_SIZE as NOTE_VAL_ENC_SIZE}; +pub use note::{ + encrypt_sender, Note, NoteType, VALUE_ENC_SIZE as NOTE_VAL_ENC_SIZE, +}; #[cfg(feature = "alloc")] /// Transaction Skeleton used by the phoenix transaction model diff --git a/core/src/note.rs b/core/src/note.rs index 1b1f77f..7d7340b 100644 --- a/core/src/note.rs +++ b/core/src/note.rs @@ -6,20 +6,19 @@ use core::convert::{TryFrom, TryInto}; -use crate::{ - encryption::elgamal, transparent_value_commitment, value_commitment, Error, - PublicKey, SecretKey, StealthAddress, SyncAddress, ViewKey, -}; use dusk_bls12_381::BlsScalar; use dusk_bytes::{DeserializableSlice, Error as BytesError, Serializable}; use dusk_jubjub::{dhke, JubJubAffine, JubJubScalar, GENERATOR_NUMS_EXTENDED}; - -use crate::aes; - use dusk_poseidon::{Domain, Hash}; use ff::Field; +use jubjub_schnorr::{PublicKey as NotePublicKey, SecretKey as NoteSecretKey}; use rand::{CryptoRng, RngCore}; +use crate::{ + aes, elgamal, transparent_value_commitment, value_commitment, Error, + PublicKey, SecretKey, StealthAddress, SyncAddress, ViewKey, +}; + #[cfg(feature = "rkyv-impl")] use rkyv::{Archive, Deserialize, Serialize}; @@ -80,7 +79,6 @@ pub struct Note { pub(crate) sync_address: SyncAddress, pub(crate) pos: u64, pub(crate) value_enc: [u8; VALUE_ENC_SIZE], - // the elgamal encryption of the sender_pk encrypted using the output_npk pub(crate) sender_enc: [(JubJubAffine, JubJubAffine); 2], } @@ -97,16 +95,17 @@ impl Note { pub fn new( rng: &mut R, note_type: NoteType, - pk: &PublicKey, + sender_pk: &PublicKey, + receiver_pk: &PublicKey, value: u64, value_blinder: JubJubScalar, sender_blinder: [JubJubScalar; 2], ) -> Self { let r = JubJubScalar::random(&mut *rng); - let stealth_address = pk.gen_stealth_address(&r); + let stealth_address = receiver_pk.gen_stealth_address(&r); let r_sync = JubJubScalar::random(&mut *rng); - let sync_address = pk.gen_sync_address(&r_sync); + let sync_address = receiver_pk.gen_sync_address(&r_sync); let value_commitment = value_commitment(value, value_blinder); @@ -121,7 +120,7 @@ impl Note { value_enc } NoteType::Obfuscated => { - let shared_secret = dhke(&r, pk.A()); + let shared_secret = dhke(&r, receiver_pk.A()); let value_blinder = BlsScalar::from(value_blinder); let mut plaintext = value.to_bytes().to_vec(); @@ -132,22 +131,6 @@ impl Note { } }; - let sender_enc_A = elgamal::encrypt( - pk.A(), - stealth_address.note_pk.as_ref(), - &sender_blinder[0], - ); - - let sender_enc_B = elgamal::encrypt( - pk.B(), - stealth_address.note_pk.as_ref(), - &sender_blinder[1], - ); - let sender_enc_A: (JubJubAffine, JubJubAffine) = - (sender_enc_A.0.into(), sender_enc_A.1.into()); - let sender_enc_B: (JubJubAffine, JubJubAffine) = - (sender_enc_B.0.into(), sender_enc_B.1.into()); - Note { note_type, value_commitment, @@ -155,7 +138,11 @@ impl Note { sync_address, pos, value_enc, - sender_enc: [sender_enc_A, sender_enc_B], + sender_enc: encrypt_sender( + stealth_address.note_pk(), + sender_pk, + &sender_blinder, + ), } } @@ -166,14 +153,16 @@ impl Note { /// notes, so this can be trivially treated as a constant. pub fn transparent( rng: &mut R, - pk: &PublicKey, + sender_pk: &PublicKey, + receiver_pk: &PublicKey, value: u64, sender_blinder: [JubJubScalar; 2], ) -> Self { Self::new( rng, NoteType::Transparent, - pk, + sender_pk, + receiver_pk, value, TRANSPARENT_BLINDER, sender_blinder, @@ -217,7 +206,8 @@ impl Note { /// knowledge of the value commitment of this note. pub fn obfuscated( rng: &mut R, - pk: &PublicKey, + sender_pk: &PublicKey, + receiver_pk: &PublicKey, value: u64, value_blinder: JubJubScalar, sender_blinder: [JubJubScalar; 2], @@ -225,7 +215,8 @@ impl Note { Self::new( rng, NoteType::Obfuscated, - pk, + sender_pk, + receiver_pk, value, value_blinder, sender_blinder, @@ -245,7 +236,7 @@ impl Note { } } - fn decrypt_data( + fn decrypt_value( &self, vk: &ViewKey, ) -> Result<(u64, JubJubScalar), BytesError> { @@ -340,6 +331,13 @@ impl Note { &self.value_enc } + /// Returns elgamal encryption of the sender's [`PublicKey`] encrypted using + /// the [`StealthAddress::note_pk`] so only the receiver of the [`Note`] + /// can decrypt. + pub const fn sender_enc(&self) -> &[(JubJubAffine, JubJubAffine); 2] { + &self.sender_enc + } + /// Attempt to decrypt the note value provided a [`ViewKey`]. Always /// succeeds for transparent notes, might fails or return random values for /// obfuscated notes if the provided view key is wrong. @@ -351,7 +349,7 @@ impl Note { Ok(value) } (NoteType::Obfuscated, Some(vk)) => self - .decrypt_data(vk) + .decrypt_value(vk) .map(|(value, _)| value) .map_err(|_| Error::InvalidEncryption), _ => Err(Error::MissingViewKey), @@ -368,12 +366,56 @@ impl Note { match (self.note_type, vk) { (NoteType::Transparent, _) => Ok(TRANSPARENT_BLINDER), (NoteType::Obfuscated, Some(vk)) => self - .decrypt_data(vk) + .decrypt_value(vk) .map(|(_, value_blinder)| value_blinder) .map_err(|_| Error::InvalidEncryption), _ => Err(Error::MissingViewKey), } } + + /// Decrypts the [`PublicKey`] of the sender of the [`Note`], using the + /// [`NoteSecretKey`] generated by the receiver's [`SecretKey`] and the + /// [`StealthAddress`] of the [`Note`]. + /// + /// Note: Decryption with an incorrect [`NoteSecretKey`] will still yield a + /// [`PublicKey`], but it will a random one that has nothing to do with the + /// sender's [`PublicKey`]. + pub fn decrypt_sender(&self, note_sk: &NoteSecretKey) -> PublicKey { + let sender_enc_A = self.sender_enc()[0]; + let sender_enc_B = self.sender_enc()[1]; + + let decrypt_A = elgamal::decrypt( + note_sk.as_ref(), + &(sender_enc_A.0.into(), sender_enc_A.1.into()), + ); + let decrypt_B = elgamal::decrypt( + note_sk.as_ref(), + &(sender_enc_B.0.into(), sender_enc_B.1.into()), + ); + + PublicKey::new(decrypt_A, decrypt_B) + } +} + +/// Encrypt the sender [`PublicKey`] in a way that only the receiver of the note +/// can decrypt. +pub fn encrypt_sender( + note_pk: &NotePublicKey, + sender_pk: &PublicKey, + blinder: &[JubJubScalar; 2], +) -> [(JubJubAffine, JubJubAffine); 2] { + let sender_enc_A = + elgamal::encrypt(note_pk.as_ref(), sender_pk.A(), &blinder[0]); + + let sender_enc_B = + elgamal::encrypt(note_pk.as_ref(), sender_pk.B(), &blinder[1]); + + let sender_enc_A: (JubJubAffine, JubJubAffine) = + (sender_enc_A.0.into(), sender_enc_A.1.into()); + let sender_enc_B: (JubJubAffine, JubJubAffine) = + (sender_enc_B.0.into(), sender_enc_B.1.into()); + + [sender_enc_A, sender_enc_B] } const SIZE: usize = 1 diff --git a/core/tests/encryption.rs b/core/tests/encryption.rs index 52f22c4..e4b746b 100644 --- a/core/tests/encryption.rs +++ b/core/tests/encryption.rs @@ -44,10 +44,10 @@ fn test_elgamal_encrypt_and_decrypt() { let (c1, c2) = elgamal::encrypt(pk.A(), &message, &blinder); // Assert decryption - let dec_message = elgamal::decrypt(sk.a(), &c1, &c2); + let dec_message = elgamal::decrypt(sk.a(), &(c1, c2)); assert_eq!(message, dec_message); // Assert decryption using an incorrect key - let dec_message_wrong = elgamal::decrypt(sk.b(), &c1, &c2); + let dec_message_wrong = elgamal::decrypt(sk.b(), &(c1, c2)); assert_ne!(message, dec_message_wrong); } diff --git a/core/tests/keys.rs b/core/tests/keys.rs index 8c6ef99..c19308c 100644 --- a/core/tests/keys.rs +++ b/core/tests/keys.rs @@ -62,35 +62,42 @@ fn keys_consistency() { let r = JubJubScalar::random(&mut rng); - let sk = SecretKey::random(&mut rng); - let pk = PublicKey::from(&sk); - let vk = ViewKey::from(&sk); + let sender_pk = PublicKey::from(&SecretKey::random(&mut rng)); + let receiver_sk = SecretKey::random(&mut rng); + let receiver_pk = PublicKey::from(&receiver_sk); + let receiver_vk = ViewKey::from(&receiver_sk); let sender_blinder = [ JubJubScalar::random(&mut rng), JubJubScalar::random(&mut rng), ]; - let note = Note::transparent(&mut rng, &pk, NOTE_VALUE, sender_blinder); + let note = Note::transparent( + &mut rng, + &sender_pk, + &receiver_pk, + NOTE_VALUE, + sender_blinder, + ); - assert!(vk.owns(¬e)); - assert!(vk.owns_unchecked(¬e)); - assert!(sk.owns(¬e)); - assert!(sk.owns_unchecked(¬e)); + assert!(receiver_vk.owns(¬e)); + assert!(receiver_vk.owns_unchecked(¬e)); + assert!(receiver_sk.owns(¬e)); + assert!(receiver_sk.owns_unchecked(¬e)); let wrong_sk = SecretKey::random(&mut rng); let wrong_vk = ViewKey::from(&wrong_sk); - assert_ne!(sk, wrong_sk); - assert_ne!(vk, wrong_vk); + assert_ne!(receiver_sk, wrong_sk); + assert_ne!(receiver_vk, wrong_vk); assert!(!wrong_vk.owns(¬e)); assert!(!wrong_vk.owns_unchecked(¬e)); assert!(!wrong_sk.owns(¬e)); assert!(!wrong_sk.owns_unchecked(¬e)); - let sa = pk.gen_stealth_address(&r); + let sa = receiver_pk.gen_stealth_address(&r); - let note_sk = sk.gen_note_sk(&sa); + let note_sk = receiver_sk.gen_note_sk(&sa); let wrong_note_sk = wrong_sk.gen_note_sk(&sa); assert_eq!( diff --git a/core/tests/note_test.rs b/core/tests/note_test.rs index e0f05e3..e9ef95a 100644 --- a/core/tests/note_test.rs +++ b/core/tests/note_test.rs @@ -4,21 +4,25 @@ // // Copyright (c) DUSK NETWORK. All rights reserved. -use dusk_jubjub::{JubJubAffine, JubJubScalar}; +use dusk_jubjub::JubJubScalar; use ff::Field; use phoenix_core::{ - elgamal, value_commitment, Error, Note, NoteType, PublicKey, SecretKey, - ViewKey, + encrypt_sender, value_commitment, Error, Note, NoteType, PublicKey, + SecretKey, ViewKey, }; use rand::rngs::StdRng; use rand::SeedableRng; +const TRANSPARENT_BLINDER: JubJubScalar = JubJubScalar::zero(); + #[test] fn transparent_note() -> Result<(), Error> { let mut rng = StdRng::seed_from_u64(0xc0b); - let sk = SecretKey::random(&mut rng); - let pk = PublicKey::from(&sk); + let sender_pk = PublicKey::from(&SecretKey::random(&mut rng)); + let receiver_sk = SecretKey::random(&mut rng); + let receiver_pk = PublicKey::from(&receiver_sk); + let value = 25; let sender_blinder = [ @@ -26,10 +30,24 @@ fn transparent_note() -> Result<(), Error> { JubJubScalar::random(&mut rng), ]; - let note = Note::transparent(&mut rng, &pk, value, sender_blinder); + let note = Note::transparent( + &mut rng, + &sender_pk, + &receiver_pk, + value, + sender_blinder, + ); assert_eq!(note.note_type(), NoteType::Transparent); + assert_eq!( + value_commitment(value, TRANSPARENT_BLINDER), + *note.value_commitment() + ); assert_eq!(value, note.value(None)?); + assert_eq!( + sender_pk, + note.decrypt_sender(&receiver_sk.gen_note_sk(note.stealth_address())) + ); Ok(()) } @@ -38,47 +56,41 @@ fn transparent_note() -> Result<(), Error> { fn transparent_stealth_note() -> Result<(), Error> { let mut rng = StdRng::seed_from_u64(0xc0b); - let sk = SecretKey::random(&mut rng); - let pk = PublicKey::from(&sk); + let sender_pk = PublicKey::from(&SecretKey::random(&mut rng)); + let receiver_sk = SecretKey::random(&mut rng); + let receiver_pk = PublicKey::from(&receiver_sk); let r = JubJubScalar::random(&mut rng); - let stealth = pk.gen_stealth_address(&r); + let stealth = receiver_pk.gen_stealth_address(&r); let r = JubJubScalar::random(&mut rng); - let sync_address = pk.gen_sync_address(&r); + let sync_address = receiver_pk.gen_sync_address(&r); let value = 25; - let sender_blinder = [ - JubJubScalar::random(&mut rng), - JubJubScalar::random(&mut rng), - ]; - let sender_enc_a = elgamal::encrypt( - pk.A(), - stealth.note_pk().as_ref(), - &sender_blinder[0], + let sender_enc = encrypt_sender( + stealth.note_pk(), + &sender_pk, + &[ + JubJubScalar::random(&mut rng), + JubJubScalar::random(&mut rng), + ], ); - let sender_enc_b = elgamal::encrypt( - pk.B(), - stealth.note_pk().as_ref(), - &sender_blinder[0], - ); - let sender_enc_a: (JubJubAffine, JubJubAffine) = - (sender_enc_a.0.into(), sender_enc_a.1.into()); - let sender_enc_b: (JubJubAffine, JubJubAffine) = - (sender_enc_b.0.into(), sender_enc_b.1.into()); - - let note = Note::transparent_stealth( - stealth, - sync_address, - value, - [sender_enc_a, sender_enc_b], - ); + let note = + Note::transparent_stealth(stealth, sync_address, value, sender_enc); assert_eq!(note.note_type(), NoteType::Transparent); + assert_eq!( + value_commitment(value, TRANSPARENT_BLINDER), + *note.value_commitment() + ); assert_eq!(value, note.value(None)?); assert_eq!(stealth, *note.stealth_address()); + assert_eq!( + sender_pk, + note.decrypt_sender(&receiver_sk.gen_note_sk(note.stealth_address())) + ); Ok(()) } @@ -87,9 +99,11 @@ fn transparent_stealth_note() -> Result<(), Error> { fn obfuscated_note() -> Result<(), Error> { let mut rng = StdRng::seed_from_u64(0xc0b); - let sk = SecretKey::random(&mut rng); - let pk = PublicKey::from(&sk); - let vk = ViewKey::from(&sk); + let sender_pk = PublicKey::from(&SecretKey::random(&mut rng)); + let receiver_sk = SecretKey::random(&mut rng); + let receiver_pk = PublicKey::from(&receiver_sk); + let receiver_vk = ViewKey::from(&receiver_sk); + let value = 25; let value_blinder = JubJubScalar::random(&mut rng); @@ -98,11 +112,25 @@ fn obfuscated_note() -> Result<(), Error> { JubJubScalar::random(&mut rng), ]; - let note = - Note::obfuscated(&mut rng, &pk, value, value_blinder, sender_blinder); + let note = Note::obfuscated( + &mut rng, + &sender_pk, + &receiver_pk, + value, + value_blinder, + sender_blinder, + ); + assert_eq!( + value_commitment(value, value_blinder), + *note.value_commitment() + ); assert_eq!(note.note_type(), NoteType::Obfuscated); - assert_eq!(value, note.value(Some(&vk))?); + assert_eq!(value, note.value(Some(&receiver_vk))?); + assert_eq!( + sender_pk, + note.decrypt_sender(&receiver_sk.gen_note_sk(note.stealth_address())) + ); Ok(()) } @@ -111,9 +139,10 @@ fn obfuscated_note() -> Result<(), Error> { fn obfuscated_deterministic_note() -> Result<(), Error> { let mut rng = StdRng::seed_from_u64(0xc0b); - let sk = SecretKey::random(&mut rng); - let pk = PublicKey::from(&sk); - let vk = ViewKey::from(&sk); + let sender_pk = PublicKey::from(&SecretKey::random(&mut rng)); + let receiver_sk = SecretKey::random(&mut rng); + let receiver_pk = PublicKey::from(&receiver_sk); + let receiver_vk = ViewKey::from(&receiver_sk); let value = 25; let value_blinder = JubJubScalar::random(&mut rng); @@ -125,106 +154,23 @@ fn obfuscated_deterministic_note() -> Result<(), Error> { let note = Note::new( &mut rng, NoteType::Obfuscated, - &pk, + &sender_pk, + &receiver_pk, value, value_blinder, sender_blinder, ); - assert_eq!(value, note.value(Some(&vk))?); - assert_eq!(value_blinder, note.value_blinder(Some(&vk))?); + assert_eq!( + value_commitment(value, value_blinder), + *note.value_commitment() + ); + assert_eq!(value, note.value(Some(&receiver_vk))?); + assert_eq!(value_blinder, note.value_blinder(Some(&receiver_vk))?); + assert_eq!( + sender_pk, + note.decrypt_sender(&receiver_sk.gen_note_sk(note.stealth_address())) + ); Ok(()) } - -#[test] -fn value_commitment_transparent() { - let mut rng = StdRng::seed_from_u64(0xc0b); - - let sk = SecretKey::random(&mut rng); - let vk = ViewKey::from(&sk); - let pk = PublicKey::from(&sk); - let value = 25; - let sender_blinder = [ - JubJubScalar::random(&mut rng), - JubJubScalar::random(&mut rng), - ]; - - let note = Note::transparent(&mut rng, &pk, value, sender_blinder); - - let value = note - .value(Some(&vk)) - .expect("The note should be owned by the provided vk"); - - let value_blinder = note - .value_blinder(Some(&vk)) - .expect("The note should be owned by the provided vk"); - - let commitment = note.value_commitment(); - let commitment_p = value_commitment(value, value_blinder); - - assert_eq!(commitment, &commitment_p.into()); -} - -#[test] -fn value_commitment_obfuscated() { - let mut rng = StdRng::seed_from_u64(0xc0b); - - let sk = SecretKey::random(&mut rng); - let vk = ViewKey::from(&sk); - let pk = PublicKey::from(&sk); - let value = 25; - - let value_blinder = JubJubScalar::random(&mut rng); - let sender_blinder = [ - JubJubScalar::random(&mut rng), - JubJubScalar::random(&mut rng), - ]; - - let note = - Note::obfuscated(&mut rng, &pk, value, value_blinder, sender_blinder); - - let value = note - .value(Some(&vk)) - .expect("The note should be owned by the provided vk"); - - let value_blinder = note - .value_blinder(Some(&vk)) - .expect("The note should be owned by the provided vk"); - - let commitment = note.value_commitment(); - let commitment_p = value_commitment(value, value_blinder); - - assert_eq!(commitment, &commitment_p.into()); -} - -#[test] -fn note_keys_consistency() { - let mut rng = StdRng::seed_from_u64(0xc0b); - - let sk = SecretKey::random(&mut rng); - let pk = PublicKey::from(&sk); - let vk = ViewKey::from(&sk); - let value = 25; - - let wrong_sk = SecretKey::random(&mut rng); - let wrong_vk = ViewKey::from(&wrong_sk); - - assert_ne!(sk, wrong_sk); - assert_ne!(vk, wrong_vk); - - let value_blinder = JubJubScalar::random(&mut rng); - let sender_blinder = [ - JubJubScalar::random(&mut rng), - JubJubScalar::random(&mut rng), - ]; - - let note = - Note::obfuscated(&mut rng, &pk, value, value_blinder, sender_blinder); - - assert!(!wrong_vk.owns(¬e)); - assert!(vk.owns(¬e)); - - assert!(!wrong_vk.owns_unchecked(¬e)); - assert!(vk.owns_unchecked(¬e)); -} diff --git a/core/tests/transaction.rs b/core/tests/transaction.rs index b86e676..f5351f8 100644 --- a/core/tests/transaction.rs +++ b/core/tests/transaction.rs @@ -16,8 +16,8 @@ use rand::rngs::OsRng; fn transaction_parse() -> Result<(), Error> { let mut rng = OsRng; - let sk = SecretKey::random(&mut rng); - let pk = PublicKey::from(&sk); + let sender_pk = PublicKey::from(&SecretKey::random(&mut rng)); + let receiver_pk = PublicKey::from(&SecretKey::random(&mut rng)); let value = 25; let value_blinder = JubJubScalar::random(&mut rng); @@ -25,8 +25,14 @@ fn transaction_parse() -> Result<(), Error> { JubJubScalar::random(&mut rng), JubJubScalar::random(&mut rng), ]; - let note = - Note::obfuscated(&mut rng, &pk, value, value_blinder, sender_blinder); + let note = Note::obfuscated( + &mut rng, + &sender_pk, + &receiver_pk, + value, + value_blinder, + sender_blinder, + ); let root = BlsScalar::from(123); let nullifiers = vec![BlsScalar::from(456), BlsScalar::from(789)];