From a56a4b1e15b0ff355dd691fb5a44d2ecd2dc960b Mon Sep 17 00:00:00 2001 From: Rene Vergara Date: Wed, 17 Apr 2024 09:21:47 -0500 Subject: [PATCH 1/4] Add Orchard commitment tree functionality --- librustzcash-wrapper/src/lib.rs | 90 +++++++++++++++++++++++++++++++++ src/C/Zcash.chs | 21 ++++++++ src/ZcashHaskell/Orchard.hs | 37 +++++++++++++- src/ZcashHaskell/Types.hs | 10 ++++ test/Spec.hs | 23 +++++++++ 5 files changed, 180 insertions(+), 1 deletion(-) diff --git a/librustzcash-wrapper/src/lib.rs b/librustzcash-wrapper/src/lib.rs index 6f03090..4806512 100644 --- a/librustzcash-wrapper/src/lib.rs +++ b/librustzcash-wrapper/src/lib.rs @@ -111,6 +111,7 @@ use orchard::{ note::{Nullifier, TransmittedNoteCiphertext, ExtractedNoteCommitment}, note_encryption::OrchardDomain, primitives::redpallas::{VerificationKey, SpendAuth, Signature}, + tree::MerkleHashOrchard, value::ValueCommitment }; @@ -1256,3 +1257,92 @@ pub extern "C" fn rust_wrapper_read_sapling_position( } } } + +#[no_mangle] +pub extern "C" fn rust_wrapper_read_orchard_commitment_tree( + tree: *const u8, + tree_len: usize, + node: *const u8, + node_len: usize, + out: *mut u8, + out_len: &mut usize + ){ + let tree_in: Vec = marshall_from_haskell_var(tree, tree_len, RW); + let tree_reader = Cursor::new(tree_in); + let ct = read_commitment_tree::>, 32>(tree_reader); + match ct { + Ok(mut comm_tree) => { + let node_in: Vec = marshall_from_haskell_var(node, node_len, RW); + let orchard_note_comm = ExtractedNoteCommitment::from_bytes(&to_array(node_in)); + if orchard_note_comm.is_some().into() { + let n = MerkleHashOrchard::from_cmx(&orchard_note_comm.unwrap()); + comm_tree.append(n); + let mut out_bytes: Vec = Vec::new(); + let result = write_commitment_tree(&comm_tree, &mut out_bytes ); + match result { + Ok(()) => { + let h = Hhex { bytes: out_bytes}; + marshall_to_haskell_var(&h, out, out_len, RW); + }, + Err(_e) => { + let h0 = Hhex { bytes: vec![0]}; + marshall_to_haskell_var(&h0, out, out_len, RW); + } + } + } else { + let h0 = Hhex { bytes: vec![0]}; + marshall_to_haskell_var(&h0, out, out_len, RW); + } + }, + Err(_e) => { + let h0 = Hhex { bytes: vec![0]}; + marshall_to_haskell_var(&h0, out, out_len, RW); + } + } +} + +#[no_mangle] +pub extern "C" fn rust_wrapper_read_orchard_witness( + tree: *const u8, + tree_len: usize, + out: *mut u8, + out_len: &mut usize + ){ + let tree_in: Vec = marshall_from_haskell_var(tree, tree_len, RW); + let tree_reader = Cursor::new(tree_in); + let ct: CommitmentTree = read_commitment_tree(tree_reader).unwrap(); + let inc_wit = IncrementalWitness::from_tree(ct); + let mut out_bytes: Vec = Vec::new(); + let result = write_incremental_witness(&inc_wit, &mut out_bytes); + match result { + Ok(()) => { + let h = Hhex { bytes: out_bytes}; + marshall_to_haskell_var(&h, out, out_len, RW); + }, + Err(_e) => { + let h0 = Hhex { bytes: vec![0]}; + marshall_to_haskell_var(&h0, out, out_len, RW); + } + } +} + + +#[no_mangle] +pub extern "C" fn rust_wrapper_read_orchard_position( + wit: *const u8, + wit_len: usize, + ) -> u64 { + let wit_in: Vec = marshall_from_haskell_var(wit, wit_len, RW); + let wit_reader = Cursor::new(wit_in); + let iw: IncrementalWitness = read_incremental_witness(wit_reader).unwrap(); + let path = iw.path(); + match path { + Some(p) => { + let pos = p.position(); + return u64::from(pos); + }, + None => { + return 0; + } + } +} diff --git a/src/C/Zcash.chs b/src/C/Zcash.chs index 4b08f78..8eef4ad 100644 --- a/src/C/Zcash.chs +++ b/src/C/Zcash.chs @@ -232,3 +232,24 @@ import ZcashHaskell.Types -> `()' #} +{# fun unsafe rust_wrapper_read_orchard_commitment_tree as rustWrapperReadOrchardCommitmentTree + { toBorshVar* `BS.ByteString'& + , toBorshVar* `BS.ByteString'& + , getVarBuffer `Buffer HexString'& + } + -> `()' +#} + +{# fun unsafe rust_wrapper_read_orchard_witness as rustWrapperReadOrchardWitness + { toBorshVar* `BS.ByteString'& + , getVarBuffer `Buffer HexString'& + } + -> `()' +#} + +{# fun pure unsafe rust_wrapper_read_orchard_position as rustWrapperReadOrchardPosition + { toBorshVar* `BS.ByteString'& + } + -> `Word64' +#} + diff --git a/src/ZcashHaskell/Orchard.hs b/src/ZcashHaskell/Orchard.hs index 8f57ad5..7e6fc6c 100644 --- a/src/ZcashHaskell/Orchard.hs +++ b/src/ZcashHaskell/Orchard.hs @@ -23,12 +23,15 @@ import C.Zcash , rustWrapperOrchardCheck , rustWrapperOrchardNoteDecode , rustWrapperOrchardNoteDecodeSK + , rustWrapperReadOrchardCommitmentTree + , rustWrapperReadOrchardPosition + , rustWrapperReadOrchardWitness , rustWrapperUADecode , rustWrapperUfvkDecode ) import qualified Data.ByteString as BS import qualified Data.ByteString.Char8 as C -import Data.HexString (fromRawBytes, toBytes) +import Data.HexString (HexString(..), fromRawBytes, toBytes) import qualified Data.Text as T import qualified Data.Text.Encoding as E import Data.Word @@ -184,3 +187,35 @@ decryptOrchardActionSK sk scope oa = decodedAction = withPureBorshVarBuffer $ rustWrapperOrchardNoteDecodeSK (getBytes sk) oa (scope == External) + +-- | Update a Orchard commitment tree +updateOrchardCommitmentTree :: + OrchardCommitmentTree -- ^ the base tree + -> HexString -- ^ the new note commitment + -> Maybe OrchardCommitmentTree +updateOrchardCommitmentTree tree cmx = + if BS.length (hexBytes updatedTree) > 1 + then Just $ OrchardCommitmentTree updatedTree + else Nothing + where + updatedTree = + withPureBorshVarBuffer $ + rustWrapperReadOrchardCommitmentTree + (hexBytes $ orchTree tree) + (hexBytes cmx) + +-- | Get the Orchard incremental witness from a commitment tree +getOrchardWitness :: OrchardCommitmentTree -> Maybe OrchardWitness +getOrchardWitness tree = + if BS.length (hexBytes wit) > 1 + then Just $ OrchardWitness wit + else Nothing + where + wit = + withPureBorshVarBuffer $ + rustWrapperReadOrchardWitness (hexBytes $ orchTree tree) + +-- | Get the Sapling note position from a witness +getOrchardNotePosition :: OrchardWitness -> Integer +getOrchardNotePosition = + fromIntegral . rustWrapperReadOrchardPosition . hexBytes . orchWit diff --git a/src/ZcashHaskell/Types.hs b/src/ZcashHaskell/Types.hs index f36bc89..f0daaab 100644 --- a/src/ZcashHaskell/Types.hs +++ b/src/ZcashHaskell/Types.hs @@ -611,6 +611,16 @@ instance FromJSON OrchardAction where a <- obj .: "spendAuthSig" pure $ OrchardAction n r c ephKey encText outText cval a +-- | Type for a Orchard note commitment tree +newtype OrchardCommitmentTree = OrchardCommitmentTree + { orchTree :: HexString + } deriving (Eq, Prelude.Show, Read) + +-- | Type for a Sapling incremental witness +newtype OrchardWitness = OrchardWitness + { orchWit :: HexString + } deriving (Eq, Prelude.Show, Read) + -- | Type to represent a decoded note data DecodedNote = DecodedNote { a_value :: !Int64 -- ^ The amount of the transaction in _zatoshis_. diff --git a/test/Spec.hs b/test/Spec.hs index f1cc5a9..b1001ff 100644 --- a/test/Spec.hs +++ b/test/Spec.hs @@ -66,7 +66,9 @@ import ZcashHaskell.Types , CoinType(..) , DecodedNote(..) , OrchardAction(..) + , OrchardCommitmentTree(..) , OrchardSpendingKey(..) + , OrchardWitness(..) , Phrase(..) , RawData(..) , RawOutPoint(..) @@ -877,6 +879,27 @@ main = do getSaplingNotePosition <$> (getSaplingWitness =<< updateSaplingCommitmentTree tree cmu1) p `shouldBe` Just 129405 + describe "Orchard commitment trees" $ do + let tree = + OrchardCommitmentTree $ + hexString + "01d5c803729654208f33d33dc68ef539ea098abc5aec215ae67c4d8aa10a14e11d01bb83047c72eb4f71813d00dee37082169546df2d7097bf7fd187ef6a93063b281f015e710ed46b53b48b12733652e150f9dcbc7e7b571cf64f294cf903864c78882f01cac32bc901f501f714a028f7ebe44c1dd8b42661be1c96730066a6fa6ede653600000000000001746e6bc066a10e7f80a9ff8993dcb25c819edd64f2ca10ac248ef7848d41450500011e6191f91b3fceb62dc881a156e1b9d2e88e09dca25093cf9c4936c8869fb41a013bf8b923e4187754e85175748d9cce4824a6787e4258977b5bfe1ba59012c032000001f3bbdc62260c4fca5c84bf3487246d4542da48eeeec8ec40c1029b6908eef83c00000000000000000000000000000000" + let cmx = + hexString + "d5c803729654208f33d33dc68ef539ea098abc5aec215ae67c4d8aa10a14e11d" + it "Commitment tree is updated correctly" $ do + let t1 = updateOrchardCommitmentTree tree cmx + t1 `shouldNotBe` Nothing + it "Incremental witness is generated" $ do + let t1 = updateOrchardCommitmentTree tree cmx + case t1 of + Nothing -> assertFailure "Failed to append node to tree" + Just t -> getOrchardWitness t `shouldNotBe` Nothing + it "Position of note is obtained" $ do + let p = + getOrchardNotePosition <$> + (getOrchardWitness =<< updateOrchardCommitmentTree tree cmx) + p `shouldBe` Just 39432 describe "Extract Sapling Address - UA Valid" $ do let sr = getSaplingFromUA From 4efae180f5466b6efdff76c938ef17b886dc92bd Mon Sep 17 00:00:00 2001 From: Rene Vergara Date: Wed, 17 Apr 2024 11:18:56 -0500 Subject: [PATCH 2/4] Fix Rust binding for Sapling decode --- librustzcash-wrapper/src/lib.rs | 1 + 1 file changed, 1 insertion(+) diff --git a/librustzcash-wrapper/src/lib.rs b/librustzcash-wrapper/src/lib.rs index 5b71bc6..39bd98b 100644 --- a/librustzcash-wrapper/src/lib.rs +++ b/librustzcash-wrapper/src/lib.rs @@ -1348,6 +1348,7 @@ pub extern "C" fn rust_wrapper_read_orchard_position( } } +#[no_mangle] pub extern "C" fn rust_wrapper_decode_sapling_address( sapling: *const u8, sapling_len: usize, From e75175852b1247f111bdfdcc57e417decfc481fd Mon Sep 17 00:00:00 2001 From: Rene Vergara Date: Wed, 17 Apr 2024 11:21:03 -0500 Subject: [PATCH 3/4] Version bump --- CHANGELOG.md | 4 +--- zcash-haskell.cabal | 2 +- 2 files changed, 2 insertions(+), 4 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index ecf9c65..a387817 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -9,12 +9,10 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ### Added -- Added unction to encode a Sappling Address in Human Readable Format Using a SaplingReceiver +- Added function to encode a Sappling Address in Human Readable Format Using a SaplingReceiver `encodeSaplingAddress` a zcash sapling address is returned or Nothing if the function fails - Added decoding and encoding test - - ## [0.5.5.1] ### Added diff --git a/zcash-haskell.cabal b/zcash-haskell.cabal index 8d16f53..c7e5406 100644 --- a/zcash-haskell.cabal +++ b/zcash-haskell.cabal @@ -5,7 +5,7 @@ cabal-version: 3.0 -- see: https://github.com/sol/hpack name: zcash-haskell -version: 0.5.5.0 +version: 0.5.5.2 synopsis: Utilities to interact with the Zcash blockchain description: Please see the README on the repo at category: Blockchain From 7d20edd08db75894cb8d5928c7f5584cd196bcf8 Mon Sep 17 00:00:00 2001 From: Rene Vergara Date: Wed, 17 Apr 2024 20:14:03 -0500 Subject: [PATCH 4/4] Fix Orchard nullifier calculation --- CHANGELOG.md | 6 ++++++ librustzcash-wrapper/src/lib.rs | 2 +- zcash-haskell.cabal | 2 +- 3 files changed, 8 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index a387817..f7d844c 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -5,6 +5,12 @@ All notable changes to this project will be documented in this file. The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). +## [0.5.5.3] + +### Fixed + +- Orchard note nullifier calculation + ## [0.5.5.2] ### Added diff --git a/librustzcash-wrapper/src/lib.rs b/librustzcash-wrapper/src/lib.rs index 39bd98b..e42f374 100644 --- a/librustzcash-wrapper/src/lib.rs +++ b/librustzcash-wrapper/src/lib.rs @@ -908,7 +908,7 @@ pub extern "C" fn rust_wrapper_orchard_note_decrypt_sk( let result = zcash_note_encryption::try_note_decryption(&domain, &pivk, &action); match result { Some((n, r, m)) => { - let hn = Hnote {note: n.value().inner(), recipient: r.to_raw_address_bytes().to_vec(), memo: m.to_vec(), nullifier: vec![0]}; + let hn = Hnote {note: n.value().inner(), recipient: r.to_raw_address_bytes().to_vec(), memo: m.to_vec(), nullifier: n.nullifier(&fvk).to_bytes().to_vec()}; marshall_to_haskell_var(&hn, out, out_len, RW); } None => { diff --git a/zcash-haskell.cabal b/zcash-haskell.cabal index c7e5406..be2387f 100644 --- a/zcash-haskell.cabal +++ b/zcash-haskell.cabal @@ -5,7 +5,7 @@ cabal-version: 3.0 -- see: https://github.com/sol/hpack name: zcash-haskell -version: 0.5.5.2 +version: 0.5.5.3 synopsis: Utilities to interact with the Zcash blockchain description: Please see the README on the repo at category: Blockchain