RPC: Shield and de-shield funds #110

Merged
pitmutt merged 165 commits from rav001 into milestone4 2025-01-02 18:43:42 +00:00
4 changed files with 44 additions and 6 deletions
Showing only changes of commit d2c52508d1 - Show all commits

View file

@ -24,6 +24,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
- Function `prepareTxV2` implementing `PrivacyPolicy` - Function `prepareTxV2` implementing `PrivacyPolicy`
- Functionality to shield transparent balance - Functionality to shield transparent balance
- Functionality to de-shield shielded notes - Functionality to de-shield shielded notes
- Native commitment trees
- Batch append to trees in O(log n)
### Changed ### Changed

View file

@ -1369,7 +1369,9 @@ updateCommitmentTrees pool zHost zPort zNet = do
return InvalidTree return InvalidTree
Just t1 -> do Just t1 -> do
let newTree = mkSaplingTree t1 let newTree = mkSaplingTree t1
return $ foldl' append newTree saplingComm let zippedSapComms =
zip [(getPosition (value newTree) + 1) ..] saplingComm
return $ batchAppend newTree zippedSapComms
Just (sTree, sSync) -> do Just (sTree, sSync) -> do
logDebugN $ ">Sapling tree found, synced to " <> T.pack (show sSync) logDebugN $ ">Sapling tree found, synced to " <> T.pack (show sSync)
saplingNotes <- liftIO $ getShieldedOutputs pool (sSync + 1) zNet saplingNotes <- liftIO $ getShieldedOutputs pool (sSync + 1) zNet
@ -1380,7 +1382,9 @@ updateCommitmentTrees pool zHost zPort zNet = do
, fromSqlKey (entityKey y))) , fromSqlKey (entityKey y)))
saplingNotes saplingNotes
logDebugN ">got shielded outputs" logDebugN ">got shielded outputs"
return $ foldl' append sTree saplingComm let zippedSapComms =
zip [(getPosition (value sTree) + 1) ..] saplingComm
return $ batchAppend sTree zippedSapComms
newOrchTree <- newOrchTree <-
case oTdb of case oTdb of
Nothing -> do Nothing -> do
@ -1402,7 +1406,9 @@ updateCommitmentTrees pool zHost zPort zNet = do
return InvalidTree return InvalidTree
Just t1 -> do Just t1 -> do
let newTree = mkOrchardTree t1 let newTree = mkOrchardTree t1
return $ foldl' append newTree orchardComm let zippedOrchComms =
zip [(getPosition (value newTree) + 1) ..] orchardComm
return $ batchAppend newTree zippedOrchComms
Just (oTree, oSync) -> do Just (oTree, oSync) -> do
logDebugN $ ">Orchard tree found, synced to " <> T.pack (show oSync) logDebugN $ ">Orchard tree found, synced to " <> T.pack (show oSync)
orchardNotes <- liftIO $ getOrchardActions pool (oSync + 1) zNet orchardNotes <- liftIO $ getOrchardActions pool (oSync + 1) zNet
@ -1413,7 +1419,9 @@ updateCommitmentTrees pool zHost zPort zNet = do
, fromSqlKey (entityKey y))) , fromSqlKey (entityKey y)))
orchardNotes orchardNotes
logDebugN ">got orchard actions" logDebugN ">got orchard actions"
return $ foldl' append oTree orchardComm let zippedOrchComms =
zip [(getPosition (value oTree) + 1) ..] orchardComm
return $ batchAppend oTree zippedOrchComms
case newSapTree of case newSapTree of
Branch {} -> do Branch {} -> do
logInfoN ">Saving updated Sapling tree to db" logInfoN ">Saving updated Sapling tree to db"

View file

@ -225,6 +225,33 @@ countLeaves (Leaf _) = 1
countLeaves EmptyLeaf = 0 countLeaves EmptyLeaf = 0
countLeaves InvalidTree = 0 countLeaves InvalidTree = 0
batchAppend ::
Measured a v
=> Node v => Monoid v => Tree v -> [(Int32, (a, Int64))] -> Tree v
batchAppend x [] = x
batchAppend (Branch s x y) notes
| isFull s = InvalidTree
| isFull (value x) = branch x (batchAppend y notes)
| otherwise =
branch
(batchAppend x (take leftSide notes))
(batchAppend y (drop leftSide notes))
where
leftSide = fromIntegral $ 2 ^ getLevel (value x) - countLeaves x
batchAppend (PrunedBranch k) notes
| isFull k = InvalidTree
| otherwise =
branch
(batchAppend (getEmptyRoot (getLevel k - 1)) (take leftSide notes))
(batchAppend (getEmptyRoot (getLevel k - 1)) (drop leftSide notes))
where
leftSide = fromIntegral $ 2 ^ (getLevel k - 1)
batchAppend EmptyLeaf notes
| length notes == 1 =
leaf (fst $ snd $ head notes) (fst $ head notes) (snd $ snd $ head notes)
| otherwise = InvalidTree
batchAppend _ notes = InvalidTree
data SaplingNode = SaplingNode data SaplingNode = SaplingNode
{ sn_position :: !Position { sn_position :: !Position
, sn_value :: !HexString , sn_value :: !HexString

View file

@ -663,7 +663,7 @@ main = do
it "Validate large load" $ do it "Validate large load" $ do
pool <- runNoLoggingT $ initPool "/home/rav/Zenith/zenith.db" pool <- runNoLoggingT $ initPool "/home/rav/Zenith/zenith.db"
maxBlock <- getMaxBlock pool $ ZcashNetDB TestNet maxBlock <- getMaxBlock pool $ ZcashNetDB TestNet
let startBlock = maxBlock - 2000 let startBlock = maxBlock - 310000
zebraTreesIn <- zebraTreesIn <-
getCommitmentTrees getCommitmentTrees
pool pool
@ -690,7 +690,8 @@ main = do
( getHex $ orchActionCmx $ entityVal y ( getHex $ orchActionCmx $ entityVal y
, fromSqlKey $ entityKey y)) , fromSqlKey $ entityKey y))
oAct oAct
let updatedTree = foldl' append newTree cmxs let posCmx = zip [(getPosition (value newTree) + 1) ..] cmxs
let updatedTree = batchAppend newTree posCmx
let finalAnchor = let finalAnchor =
getOrchardTreeAnchor $ getOrchardTreeAnchor $
OrchardCommitmentTree $ ztiOrchard zebraTreesOut OrchardCommitmentTree $ ztiOrchard zebraTreesOut