From 30dafceae31c6e3022e3ae4ad6956f2abba5dcb8 Mon Sep 17 00:00:00 2001 From: julia-zack Date: Tue, 19 Nov 2024 14:08:00 -0300 Subject: [PATCH] Register spend transaction --- .../main/java/co/rsk/peg/BridgeSupport.java | 44 +++-- .../java/co/rsk/peg/BridgeSupportSvpTest.java | 187 ++++++++++++++---- .../co/rsk/peg/bitcoin/BitcoinUtilsTest.java | 2 +- 3 files changed, 173 insertions(+), 60 deletions(-) diff --git a/rskj-core/src/main/java/co/rsk/peg/BridgeSupport.java b/rskj-core/src/main/java/co/rsk/peg/BridgeSupport.java index 965e14f5c1..97287574fe 100644 --- a/rskj-core/src/main/java/co/rsk/peg/BridgeSupport.java +++ b/rskj-core/src/main/java/co/rsk/peg/BridgeSupport.java @@ -391,6 +391,14 @@ public void registerBtcTransaction( throw new RegisterBtcTransactionException("Transaction already processed"); } + if (svpIsOngoing() && isTheSvpSpendTransaction(btcTx)) { + markTxAsProcessed(btcTx); + saveNewUTXOs(btcTx); + provider.setSvpSpendTxHashUnsigned(null); + // proceed with svp success + return; + } + PegTxType pegTxType = PegUtils.getTransactionType( activations, provider, @@ -410,8 +418,8 @@ public void registerBtcTransaction( case PEGOUT_OR_MIGRATION: logger.debug("[registerBtcTransaction] This is a peg-out or migration tx {}", btcTx.getHash()); processPegoutOrMigration(btcTx); - if (svpIsOngoing()) { - updateSvpFundTransactionValuesIfPossible(btcTx); + if (svpIsOngoing() && isTheSvpFundTransaction(btcTx)) { + updateSvpFundTransactionValues(btcTx); } break; default: @@ -429,21 +437,13 @@ public void registerBtcTransaction( } } - private void updateSvpFundTransactionValuesIfPossible(BtcTransaction transaction) { - provider.getSvpFundTxHashUnsigned() - .filter(svpFundTxHashUnsigned -> isTheSvpFundTransaction(svpFundTxHashUnsigned, transaction)) - .ifPresent(isTheSvpFundTransaction -> updateSvpFundTransactionValues(transaction)); - } - - private boolean isTheSvpFundTransaction(Sha256Hash svpFundTransactionHashUnsigned, BtcTransaction transaction) { - Sha256Hash transactionHash = transaction.getHash(); - - if (!transaction.hasWitness()) { - BtcTransaction transactionCopyWithoutSignatures = new BtcTransaction(networkParameters, transaction.bitcoinSerialize()); // this is needed to not remove signatures from the actual tx - BitcoinUtils.removeSignaturesFromTransactionWithP2shMultiSigInputs(transactionCopyWithoutSignatures); - transactionHash = transactionCopyWithoutSignatures.getHash(); - } - return transactionHash.equals(svpFundTransactionHashUnsigned); + private boolean isTheSvpFundTransaction(BtcTransaction transaction) { + return provider.getSvpFundTxHashUnsigned() + .map(svpFundTransactionHashUnsigned -> { + Sha256Hash transactionHashWithoutSignatures = getMultiSigTransactionHashWithoutSignatures(networkParameters, transaction); + return transactionHashWithoutSignatures.equals(svpFundTransactionHashUnsigned); + }) + .orElse(false); } private void updateSvpFundTransactionValues(BtcTransaction transaction) { @@ -455,6 +455,15 @@ private void updateSvpFundTransactionValues(BtcTransaction transaction) { provider.setSvpFundTxHashUnsigned(null); } + private boolean isTheSvpSpendTransaction(BtcTransaction transaction) { + return provider.getSvpSpendTxHashUnsigned() + .map(svpSpendTransactionHashUnsigned -> { + Sha256Hash transactionHashWithoutSignatures = getMultiSigTransactionHashWithoutSignatures(networkParameters, transaction); + return transactionHashWithoutSignatures.equals(svpSpendTransactionHashUnsigned); + }) + .orElse(false); + } + private Script getLastRetiredFederationP2SHScript() { return federationSupport.getLastRetiredFederationP2SHScript().orElse(null); } @@ -1760,7 +1769,6 @@ private void addSvpSpendTxSignatures( provider.getSvpSpendTxWaitingForSignatures() // The svpSpendTxWFS should always be present at this point, since we already checked isTheSvpSpendTx. .ifPresent(svpSpendTxWFS -> { - Keccak256 svpSpendTxCreationRskTxHash = svpSpendTxWFS.getKey(); BtcTransaction svpSpendTx = svpSpendTxWFS.getValue(); diff --git a/rskj-core/src/test/java/co/rsk/peg/BridgeSupportSvpTest.java b/rskj-core/src/test/java/co/rsk/peg/BridgeSupportSvpTest.java index c2414dc678..f1de66bf26 100644 --- a/rskj-core/src/test/java/co/rsk/peg/BridgeSupportSvpTest.java +++ b/rskj-core/src/test/java/co/rsk/peg/BridgeSupportSvpTest.java @@ -82,6 +82,9 @@ public class BridgeSupportSvpTest { private Sha256Hash svpSpendTransactionHashUnsigned; private BtcTransaction svpSpendTransaction; + private PartialMerkleTree pmtWithTransactions; + private int btcBlockWithPmtHeight; + @BeforeEach void setUp() { long rskExecutionBlockNumber = 1000L; @@ -467,8 +470,6 @@ private void assertSvpFundTransactionValuesWereNotUpdated() { @TestInstance(TestInstance.Lifecycle.PER_CLASS) @Tag("Fund transaction registration tests") class FundTxRegistrationTests { - private PartialMerkleTree pmtWithTransactions; - private int btcBlockWithPmtHeight; @Test void registerBtcTransaction_forSvpFundTransactionChange_whenProposedFederationDoesNotExist_shouldRegisterTransactionButNotUpdateSvpFundTransactionValues() throws Exception { @@ -595,43 +596,6 @@ void registerBtcTransaction_forSvpFundTransactionChange_whenSvpPeriodIsOngoing_s assertSvpFundTransactionValuesWereUpdated(); } - private void setUpForTransactionRegistration(BtcTransaction transaction) throws BlockStoreException { - // recreate a valid chain that has the tx, so it passes the previous checks in registerBtcTransaction - BtcBlockStoreWithCache.Factory btcBlockStoreFactory = new RepositoryBtcBlockStoreWithCache.Factory(btcMainnetParams, 100, 100); - BtcBlockStoreWithCache btcBlockStoreWithCache = btcBlockStoreFactory.newInstance(repository, bridgeMainNetConstants, bridgeStorageProvider, allActivations); - - pmtWithTransactions = createValidPmtForTransactions(Collections.singletonList(transaction.getHash()), btcMainnetParams); - btcBlockWithPmtHeight = bridgeMainNetConstants.getBtcHeightWhenPegoutTxIndexActivates() + bridgeMainNetConstants.getPegoutTxIndexGracePeriodInBtcBlocks(); // we want pegout tx index to be activated - - int chainHeight = btcBlockWithPmtHeight + bridgeMainNetConstants.getBtc2RskMinimumAcceptableConfirmations(); - recreateChainFromPmt(btcBlockStoreWithCache, chainHeight, pmtWithTransactions, btcBlockWithPmtHeight, btcMainnetParams); - - bridgeStorageProvider.save(); - - bridgeSupport = bridgeSupportBuilder - .withBridgeConstants(bridgeMainNetConstants) - .withProvider(bridgeStorageProvider) - .withActivations(allActivations) - .withFederationSupport(federationSupport) - .withFeePerKbSupport(feePerKbSupport) - .withExecutionBlock(rskExecutionBlock) - .withBtcBlockStoreFactory(btcBlockStoreFactory) - .withRepository(repository) - .build(); - } - - private void assertActiveFederationUtxosSize(int expectedActiveFederationUtxosSize) { - assertEquals(expectedActiveFederationUtxosSize, federationSupport.getActiveFederationBtcUTXOs().size()); - } - - private void assertTransactionWasProcessed(Sha256Hash transactionHash) throws IOException { - Optional rskBlockHeightAtWhichBtcTxWasProcessed = bridgeStorageProvider.getHeightIfBtcTxhashIsAlreadyProcessed(transactionHash); - assertTrue(rskBlockHeightAtWhichBtcTxWasProcessed.isPresent()); - - long rskExecutionBlockNumber = rskExecutionBlock.getNumber(); - assertEquals(rskExecutionBlockNumber, rskBlockHeightAtWhichBtcTxWasProcessed.get()); - } - private void assertSvpFundTransactionValuesWereUpdated() { Optional svpFundTransactionSignedOpt = bridgeStorageProvider.getSvpFundTxSigned(); assertTrue(svpFundTransactionSignedOpt.isPresent()); @@ -666,7 +630,7 @@ void updateCollections_whenSpendTxCanBeCreated_createsExpectedSpendTxAndSavesThe bridgeStorageProvider.save(); // assert - svpSpendTransactionHashUnsigned = assertSvpSpendTxHashUnsignedWasSavedInStorage(); + svpSpendTransactionHashUnsigned = assertSvpSpendTxHashUnsignedIsSavedInStorage(); svpSpendTransaction = assertSvpSpendTxIsWaitingForSignatures(); assertSvpSpendTxHasExpectedInputsAndOutputs(); @@ -727,7 +691,7 @@ private void assertInputHasExpectedScriptSig(TransactionInput input, Script rede } } - private Sha256Hash assertSvpSpendTxHashUnsignedWasSavedInStorage() { + private Sha256Hash assertSvpSpendTxHashUnsignedIsSavedInStorage() { Optional svpSpendTransactionHashUnsignedOpt = bridgeStorageProvider.getSvpSpendTxHashUnsigned(); assertTrue(svpSpendTransactionHashUnsignedOpt.isPresent()); @@ -982,6 +946,105 @@ private void assertFederatorSignedInputs(List inputs, List rskBlockHeightAtWhichBtcTxWasProcessed = bridgeStorageProvider.getHeightIfBtcTxhashIsAlreadyProcessed(transactionHash); + assertTrue(rskBlockHeightAtWhichBtcTxWasProcessed.isPresent()); + + long rskExecutionBlockNumber = rskExecutionBlock.getNumber(); + assertEquals(rskExecutionBlockNumber, rskBlockHeightAtWhichBtcTxWasProcessed.get()); + } + + private void assertTransactionWasNotProcessed(Sha256Hash transactionHash) throws IOException { + Optional rskBlockHeightAtWhichBtcTxWasProcessed = bridgeStorageProvider.getHeightIfBtcTxhashIsAlreadyProcessed(transactionHash); + assertFalse(rskBlockHeightAtWhichBtcTxWasProcessed.isPresent()); + } + private void assertNoSvpFundTxHashUnsigned() { assertFalse(bridgeStorageProvider.getSvpFundTxHashUnsigned().isPresent()); } diff --git a/rskj-core/src/test/java/co/rsk/peg/bitcoin/BitcoinUtilsTest.java b/rskj-core/src/test/java/co/rsk/peg/bitcoin/BitcoinUtilsTest.java index 0c687c942a..779efec3ee 100644 --- a/rskj-core/src/test/java/co/rsk/peg/bitcoin/BitcoinUtilsTest.java +++ b/rskj-core/src/test/java/co/rsk/peg/bitcoin/BitcoinUtilsTest.java @@ -303,7 +303,7 @@ void test_extractRedeemScriptFromInput_migration_multiple_inputs() { } @Test - void getTransactionHashWithoutSignatures_whenTransactionDoesNotHaveInputs_shouldReturnSameTransaction() { + void getTransactionHashWithoutSignatures_whenTransactionDoesNotHaveInputs_shouldReturnExpectedTxHash() { // arrange BtcTransaction transaction = new BtcTransaction(btcMainnetParams); Sha256Hash transactionHashBeforeRemovingSignatures = transaction.getHash();