From 47da7590b16f2bbe74b4c1cb25d52b32f97c1caf Mon Sep 17 00:00:00 2001 From: julia-zack Date: Mon, 16 Sep 2024 20:47:21 -0300 Subject: [PATCH] When signed, save the svp fund tx instead of its hash --- .../co/rsk/peg/BridgeStorageIndexKey.java | 2 +- .../co/rsk/peg/BridgeStorageProvider.java | 34 ++-- .../co/rsk/peg/BridgeStorageProviderTest.java | 163 +++++++++--------- 3 files changed, 102 insertions(+), 97 deletions(-) diff --git a/rskj-core/src/main/java/co/rsk/peg/BridgeStorageIndexKey.java b/rskj-core/src/main/java/co/rsk/peg/BridgeStorageIndexKey.java index 5089bf7b0be..977495fb5eb 100644 --- a/rskj-core/src/main/java/co/rsk/peg/BridgeStorageIndexKey.java +++ b/rskj-core/src/main/java/co/rsk/peg/BridgeStorageIndexKey.java @@ -24,7 +24,7 @@ public enum BridgeStorageIndexKey { PEGOUT_TX_SIG_HASH("pegoutTxSigHash"), SVP_FUND_TX_HASH_UNSIGNED("svpFundTxHashUnsigned"), - SVP_FUND_TX_HASH_SIGNED("svpFundTxHashSigned"), + SVP_FUND_TX_SIGNED("svpFundTxSigned"), SVP_SPEND_TX_HASH_UNSIGNED("svpSpendTxHashUnsigned"), SVP_SPEND_TX_WAITING_FOR_SIGNATURES("svpSpendTxWaitingForSignatures"), ; diff --git a/rskj-core/src/main/java/co/rsk/peg/BridgeStorageProvider.java b/rskj-core/src/main/java/co/rsk/peg/BridgeStorageProvider.java index a7aa16c80bc..7c3307b9276 100644 --- a/rskj-core/src/main/java/co/rsk/peg/BridgeStorageProvider.java +++ b/rskj-core/src/main/java/co/rsk/peg/BridgeStorageProvider.java @@ -78,8 +78,8 @@ public class BridgeStorageProvider { private Sha256Hash svpFundTxHashUnsigned; private boolean isSvpFundTxHashUnsignedSet = false; - private Sha256Hash svpFundTxHashSigned; - private boolean isSvpFundTxHashSignedSet = false; + private BtcTransaction svpFundTxSigned; + private boolean isSvpFundTxSignedSet = false; private Sha256Hash svpSpendTxHashUnsigned; private boolean isSvpSpendTxHashUnsignedSet = false; private Map.Entry svpSpendTxWaitingForSignatures; @@ -381,7 +381,7 @@ public Optional getFlyoverFederationInformation(by return Optional.empty(); } - FlyoverFederationInformation flyoverFederationInformationInStorage = this.safeGetFromRepository( + FlyoverFederationInformation flyoverFederationInformationInStorage = safeGetFromRepository( getStorageKeyForFlyoverFederationInformation(flyoverFederationRedeemScriptHash), data -> BridgeSerializationUtils.deserializeFlyoverFederationInformation(data, flyoverFederationRedeemScriptHash) ); @@ -547,23 +547,23 @@ public Optional getSvpFundTxHashUnsigned() { return Optional.ofNullable(svpFundTxHashUnsigned); } - public Optional getSvpFundTxHashSigned() { + public Optional getSvpFundTxSigned() { if (!activations.isActive(RSKIP419)) { return Optional.empty(); } - if (svpFundTxHashSigned != null) { - return Optional.of(svpFundTxHashSigned); + if (svpFundTxSigned != null) { + return Optional.of(svpFundTxSigned); } // Return empty if the svp fund tx hash unsigned was explicitly set to null - if (isSvpFundTxHashSignedSet) { + if (isSvpFundTxSignedSet) { return Optional.empty(); } - svpFundTxHashSigned = safeGetFromRepository( - SVP_FUND_TX_HASH_SIGNED.getKey(), BridgeSerializationUtils::deserializeSha256Hash); - return Optional.ofNullable(svpFundTxHashSigned); + svpFundTxSigned = safeGetFromRepository(SVP_FUND_TX_SIGNED, + data -> BridgeSerializationUtils.deserializeBtcTransaction(data, networkParameters)); + return Optional.ofNullable(svpFundTxSigned); } public Optional getSvpSpendTxHashUnsigned() { @@ -601,20 +601,20 @@ private void saveSvpFundTxHashUnsigned() { BridgeSerializationUtils::serializeSha256Hash); } - public void setSvpFundTxHashSigned(Sha256Hash hash) { - this.svpFundTxHashSigned = hash; - this.isSvpFundTxHashSignedSet = true; + public void setSvpFundTxSigned(BtcTransaction svpFundTxSigned) { + this.svpFundTxSigned = svpFundTxSigned; + this.isSvpFundTxSignedSet = true; } private void saveSvpFundTxHashSigned() { - if (!activations.isActive(RSKIP419) || !isSvpFundTxHashSignedSet) { + if (!activations.isActive(RSKIP419) || !isSvpFundTxSignedSet) { return; } safeSaveToRepository( - SVP_FUND_TX_HASH_SIGNED, - svpFundTxHashSigned, - BridgeSerializationUtils::serializeSha256Hash); + SVP_FUND_TX_SIGNED, + svpFundTxSigned, + BridgeSerializationUtils::serializeBtcTransaction); } public void setSvpSpendTxHashUnsigned(Sha256Hash hash) { diff --git a/rskj-core/src/test/java/co/rsk/peg/BridgeStorageProviderTest.java b/rskj-core/src/test/java/co/rsk/peg/BridgeStorageProviderTest.java index 5a1703f7ae0..1b1a575f98a 100644 --- a/rskj-core/src/test/java/co/rsk/peg/BridgeStorageProviderTest.java +++ b/rskj-core/src/test/java/co/rsk/peg/BridgeStorageProviderTest.java @@ -386,10 +386,10 @@ void getSvpFundTxHashUnsigned_whenHashIsCached_shouldReturnTheCachedHash() { @Nested @TestInstance(TestInstance.Lifecycle.PER_CLASS) - @Tag("save, set and get svp fund transaction hash signed tests") - class SvpFundTxHashSignedTests { - private final Sha256Hash svpFundTxHash = BitcoinTestUtils.createHash(123_456_789); - private final Sha256Hash anotherSvpFundTxHash = BitcoinTestUtils.createHash(987_654_321); + @Tag("save, set and get svp fund transaction signed tests") + class SvpFundTxSignedTests { + private final BtcTransaction svpFundTx = new BtcTransaction(mainnetBtcParams); + private final BtcTransaction anotherSvpFundTx = new BtcTransaction(mainnetBtcParams); private Repository repository; private BridgeStorageProvider bridgeStorageProvider; @@ -397,67 +397,72 @@ class SvpFundTxHashSignedTests { void setup() { repository = createRepository(); bridgeStorageProvider = createBridgeStorageProvider(repository, mainnetBtcParams, activationsAllForks); + + BtcTransaction prevTx = new BtcTransaction(mainnetBtcParams); + Address address = BitcoinTestUtils.createP2PKHAddress(mainnetBtcParams, "address"); + prevTx.addOutput(Coin.FIFTY_COINS, address); + svpFundTx.addInput(prevTx.getOutput(0)); } @Test - void saveSvpFundTxHashSigned_preLovell700_shouldNotSaveInStorage() { + void saveSvpFundTxSigned_preLovell700_shouldNotSaveInStorage() { // Arrange ActivationConfig.ForBlock arrowheadActivations = ActivationConfigsForTest.arrowhead631().forBlock(0L); bridgeStorageProvider = createBridgeStorageProvider(repository, mainnetBtcParams, arrowheadActivations); // Act - bridgeStorageProvider.setSvpFundTxHashSigned(svpFundTxHash); + bridgeStorageProvider.setSvpFundTxSigned(svpFundTx); bridgeStorageProvider.save(); // Assert - byte[] actualSvpFundTxHashSerialized = repository.getStorageBytes(bridgeAddress, SVP_FUND_TX_HASH_SIGNED.getKey()); - assertNull(actualSvpFundTxHashSerialized); + byte[] actualSvpFundTxSerialized = repository.getStorageBytes(bridgeAddress, SVP_FUND_TX_SIGNED.getKey()); + assertNull(actualSvpFundTxSerialized); } @Test - void saveSvpFundTxHashSigned_postLovell700_shouldSaveInStorage() { + void saveSvpFundTxSigned_postLovell700_shouldSaveInStorage() { // Act - bridgeStorageProvider.setSvpFundTxHashSigned(svpFundTxHash); + bridgeStorageProvider.setSvpFundTxSigned(svpFundTx); bridgeStorageProvider.save(); // Assert - byte[] svpFundTxHashSerialized = BridgeSerializationUtils.serializeSha256Hash(svpFundTxHash); - byte[] actualSvpFundTxHashSerialized = repository.getStorageBytes(bridgeAddress, SVP_FUND_TX_HASH_SIGNED.getKey()); - assertArrayEquals(svpFundTxHashSerialized, actualSvpFundTxHashSerialized); + byte[] svpFundTxSerialized = BridgeSerializationUtils.serializeBtcTransaction(svpFundTx); + byte[] actualSvpFundTxSerialized = repository.getStorageBytes(bridgeAddress, SVP_FUND_TX_SIGNED.getKey()); + assertArrayEquals(svpFundTxSerialized, actualSvpFundTxSerialized); } @Test - void saveSvpFundTxHashSigned_postLovell700AndResettingToNull_shouldSaveNullInStorage() { + void saveSvpFundTxSigned_postLovell700AndResettingToNull_shouldSaveNullInStorage() { // Initially setting a valid hash in storage - bridgeStorageProvider.setSvpFundTxHashSigned(svpFundTxHash); + bridgeStorageProvider.setSvpFundTxSigned(svpFundTx); bridgeStorageProvider.save(); // Act - bridgeStorageProvider.setSvpFundTxHashSigned(null); + bridgeStorageProvider.setSvpFundTxSigned(null); bridgeStorageProvider.save(); // Assert - byte[] actualSvpFundTxHashSerialized = repository.getStorageBytes(bridgeAddress, SVP_FUND_TX_HASH_SIGNED.getKey()); - assertNull(actualSvpFundTxHashSerialized); + byte[] actualSvpFundTxSerialized = repository.getStorageBytes(bridgeAddress, SVP_FUND_TX_SIGNED.getKey()); + assertNull(actualSvpFundTxSerialized); } @Test - void getSvpFundTxHashSigned_preLovell700_whenHashSet_shouldReturnEmpty() { + void getSvpFundTxSigned_preLovell700_whenHashSet_shouldReturnEmpty() { // Arrange ActivationConfig.ForBlock arrowheadActivations = ActivationConfigsForTest.arrowhead631().forBlock(0L); bridgeStorageProvider = createBridgeStorageProvider(repository, mainnetBtcParams, arrowheadActivations); - bridgeStorageProvider.setSvpFundTxHashSigned(svpFundTxHash); + bridgeStorageProvider.setSvpFundTxSigned(svpFundTx); // Act - Optional svpFundTxHashSigned = bridgeStorageProvider.getSvpFundTxHashSigned(); + Optional svpFundTxSigned = bridgeStorageProvider.getSvpFundTxSigned(); // Assert - assertEquals(Optional.empty(), svpFundTxHashSigned); + assertEquals(Optional.empty(), svpFundTxSigned); } @Test - void getSvpFundTxHashSigned_preLovell700_whenHashSaved_shouldReturnEmpty() { + void getSvpFundTxSigned_preLovell700_whenHashSaved_shouldReturnEmpty() { // Arrange ActivationConfig.ForBlock arrowheadActivations = ActivationConfigsForTest.arrowhead631().forBlock(0L); bridgeStorageProvider = createBridgeStorageProvider(repository, mainnetBtcParams, arrowheadActivations); @@ -465,171 +470,171 @@ void getSvpFundTxHashSigned_preLovell700_whenHashSaved_shouldReturnEmpty() { // Manually setting the value in storage to then assert that pre fork the method doesn't access the storage repository.addStorageBytes( bridgeAddress, - SVP_FUND_TX_HASH_SIGNED.getKey(), - BridgeSerializationUtils.serializeSha256Hash(svpFundTxHash) + SVP_FUND_TX_SIGNED.getKey(), + BridgeSerializationUtils.serializeBtcTransaction(svpFundTx) ); // Act - Optional svpFundTxHashSigned = bridgeStorageProvider.getSvpFundTxHashSigned(); + Optional svpFundTxSigned = bridgeStorageProvider.getSvpFundTxSigned(); // Assert - assertEquals(Optional.empty(), svpFundTxHashSigned); + assertEquals(Optional.empty(), svpFundTxSigned); } @Test - void getSvpFundTxHashSigned_whenThereIsNoSvpFundTxHashSignedSavedNorSet_shouldReturnEmpty() { - Optional svpFundTxHashSigned = bridgeStorageProvider.getSvpFundTxHashSigned(); - assertEquals(Optional.empty(), svpFundTxHashSigned); + void getSvpFundTxSigned_whenThereIsNosvpFundTxSignedSavedNorSet_shouldReturnEmpty() { + Optional svpFundTxSigned = bridgeStorageProvider.getSvpFundTxSigned(); + assertEquals(Optional.empty(), svpFundTxSigned); } @Test - void getSvpFundTxHashSigned_whenHashSet_shouldReturnTheHash() { + void getSvpFundTxSigned_whenHashSet_shouldReturnTheHash() { // Arrange - bridgeStorageProvider.setSvpFundTxHashSigned(svpFundTxHash); + bridgeStorageProvider.setSvpFundTxSigned(svpFundTx); // Act - Optional svpFundTxHashSigned = bridgeStorageProvider.getSvpFundTxHashSigned(); + Optional svpFundTxSigned = bridgeStorageProvider.getSvpFundTxSigned(); // Assert - assertTrue(svpFundTxHashSigned.isPresent()); - assertEquals(svpFundTxHash, svpFundTxHashSigned.get()); + assertTrue(svpFundTxSigned.isPresent()); + assertEquals(svpFundTx, svpFundTxSigned.get()); } @Test - void getSvpFundTxHashSigned_whenHashSetToNull_shouldReturnEmpty() { + void getSvpFundTxSigned_whenHashSetToNull_shouldReturnEmpty() { // Arrange - bridgeStorageProvider.setSvpFundTxHashSigned(null); + bridgeStorageProvider.setSvpFundTxSigned(null); // Act - Optional svpFundTxHashSigned = bridgeStorageProvider.getSvpFundTxHashSigned(); + Optional svpFundTxSigned = bridgeStorageProvider.getSvpFundTxSigned(); // Assert - assertEquals(Optional.empty(), svpFundTxHashSigned); + assertEquals(Optional.empty(), svpFundTxSigned); } @Test - void getSvpFundTxHashSigned_whenHashSavedAndHashSet_shouldReturnTheSetHash() { + void getSvpFundTxSigned_whenHashSavedAndHashSet_shouldReturnTheSetHash() { // Arrange repository.addStorageBytes( bridgeAddress, - SVP_FUND_TX_HASH_SIGNED.getKey(), - BridgeSerializationUtils.serializeSha256Hash(svpFundTxHash) + SVP_FUND_TX_SIGNED.getKey(), + BridgeSerializationUtils.serializeBtcTransaction(svpFundTx) ); - bridgeStorageProvider.setSvpFundTxHashSigned(anotherSvpFundTxHash); + bridgeStorageProvider.setSvpFundTxSigned(anotherSvpFundTx); // Act - Optional svpFundTxHashSigned = bridgeStorageProvider.getSvpFundTxHashSigned(); + Optional svpFundTxSigned = bridgeStorageProvider.getSvpFundTxSigned(); // Assert - assertTrue(svpFundTxHashSigned.isPresent()); - assertEquals(anotherSvpFundTxHash, svpFundTxHashSigned.get()); + assertTrue(svpFundTxSigned.isPresent()); + assertEquals(anotherSvpFundTx, svpFundTxSigned.get()); } @Test - void getSvpFundTxHashSigned_whenHashSavedAndHashSetToNull_shouldReturnEmpty() { + void getSvpFundTxSigned_whenHashSavedAndHashSetToNull_shouldReturnEmpty() { // Arrange repository.addStorageBytes( bridgeAddress, - SVP_FUND_TX_HASH_SIGNED.getKey(), - BridgeSerializationUtils.serializeSha256Hash(svpFundTxHash) + SVP_FUND_TX_SIGNED.getKey(), + BridgeSerializationUtils.serializeBtcTransaction(svpFundTx) ); - bridgeStorageProvider.setSvpFundTxHashSigned(null); + bridgeStorageProvider.setSvpFundTxSigned(null); // Act - Optional svpFundTxHashSigned = bridgeStorageProvider.getSvpFundTxHashSigned(); + Optional svpFundTxSigned = bridgeStorageProvider.getSvpFundTxSigned(); // Assert - assertEquals(Optional.empty(), svpFundTxHashSigned); + assertEquals(Optional.empty(), svpFundTxSigned); } @Test - void getSvpFundTxHashSigned_whenHashSaved_shouldReturnTheHash() { + void getSvpFundTxSigned_whenHashSaved_shouldReturnTheHash() { // Arrange repository.addStorageBytes( bridgeAddress, - SVP_FUND_TX_HASH_SIGNED.getKey(), - BridgeSerializationUtils.serializeSha256Hash(svpFundTxHash) + SVP_FUND_TX_SIGNED.getKey(), + BridgeSerializationUtils.serializeBtcTransaction(svpFundTx) ); // Act - Optional svpFundTxHashSigned = bridgeStorageProvider.getSvpFundTxHashSigned(); + Optional svpFundTxSigned = bridgeStorageProvider.getSvpFundTxSigned(); // Assert - assertTrue(svpFundTxHashSigned.isPresent()); - assertEquals(svpFundTxHash, svpFundTxHashSigned.get()); + assertTrue(svpFundTxSigned.isPresent()); + assertEquals(svpFundTx, svpFundTxSigned.get()); } @Test - void getSvpFundTxHashSigned_whenNullHashSaved_shouldReturnEmpty() { + void getSvpFundTxSigned_whenNullHashSaved_shouldReturnEmpty() { // Arrange repository.addStorageBytes( bridgeAddress, - SVP_FUND_TX_HASH_SIGNED.getKey(), + SVP_FUND_TX_SIGNED.getKey(), null ); // Act - Optional svpFundTxHashSigned = bridgeStorageProvider.getSvpFundTxHashSigned(); + Optional svpFundTxSigned = bridgeStorageProvider.getSvpFundTxSigned(); // Assert - assertEquals(Optional.empty(), svpFundTxHashSigned); + assertEquals(Optional.empty(), svpFundTxSigned); } @Test - void getSvpFundTxHashSigned_whenHashIsCached_shouldReturnTheCachedHash() { + void getSvpFundTxSigned_whenHashIsCached_shouldReturnTheCachedHash() { // Arrange // Manually saving a hash in storage to then cache it repository.addStorageBytes( bridgeAddress, - SVP_FUND_TX_HASH_SIGNED.getKey(), - BridgeSerializationUtils.serializeSha256Hash(svpFundTxHash) + SVP_FUND_TX_SIGNED.getKey(), + BridgeSerializationUtils.serializeBtcTransaction(svpFundTx) ); // Calling method, so it retrieves the hash from storage and caches it - bridgeStorageProvider.getSvpFundTxHashSigned(); + bridgeStorageProvider.getSvpFundTxSigned(); // Setting a different hash in storage to make sure that when calling the method again it returns the cached one, not this one repository.addStorageBytes( bridgeAddress, - SVP_FUND_TX_HASH_SIGNED.getKey(), - BridgeSerializationUtils.serializeSha256Hash(anotherSvpFundTxHash) + SVP_FUND_TX_SIGNED.getKey(), + BridgeSerializationUtils.serializeBtcTransaction(anotherSvpFundTx) ); // Act - Optional svpFundTxHashSigned = bridgeStorageProvider.getSvpFundTxHashSigned(); + Optional svpFundTxSigned = bridgeStorageProvider.getSvpFundTxSigned(); // Assert - assertTrue(svpFundTxHashSigned.isPresent()); - assertEquals(svpFundTxHash, svpFundTxHashSigned.get()); + assertTrue(svpFundTxSigned.isPresent()); + assertEquals(svpFundTx, svpFundTxSigned.get()); } @Test - void getSvpFundTxHashSigned_whenNullHashIsCached_shouldReturnNewSavedHash() { + void getSvpFundTxSigned_whenNullHashIsCached_shouldReturnNewSavedHash() { // Arrange // Manually saving a null hash in storage to then cache it repository.addStorageBytes( bridgeAddress, - SVP_FUND_TX_HASH_SIGNED.getKey(), + SVP_FUND_TX_SIGNED.getKey(), null ); // Calling method, so it retrieves the hash from storage and caches it - bridgeStorageProvider.getSvpFundTxHashSigned(); + bridgeStorageProvider.getSvpFundTxSigned(); // Setting a hash in storage repository.addStorageBytes( bridgeAddress, - SVP_FUND_TX_HASH_SIGNED.getKey(), - BridgeSerializationUtils.serializeSha256Hash(anotherSvpFundTxHash) + SVP_FUND_TX_SIGNED.getKey(), + BridgeSerializationUtils.serializeBtcTransaction(anotherSvpFundTx) ); // Act - Optional svpFundTxHashSigned = bridgeStorageProvider.getSvpFundTxHashSigned(); + Optional svpFundTxSigned = bridgeStorageProvider.getSvpFundTxSigned(); // Assert // since null hash was directly saved and not set, method returns new saved hash - assertTrue(svpFundTxHashSigned.isPresent()); - assertEquals(anotherSvpFundTxHash, svpFundTxHashSigned.get()); + assertTrue(svpFundTxSigned.isPresent()); + assertEquals(anotherSvpFundTx, svpFundTxSigned.get()); } }