Skip to content

Commit

Permalink
Merge bitcoin/bitcoin#31556: validation: Send correct notification du…
Browse files Browse the repository at this point in the history
…ring snapshot completion

bc43eca test: add functional test for balance after snapshot completion (Martin Zumsande)
226d03d validation: Send correct notification during snapshot completion (Martin Zumsande)

Pull request description:

  After AssumeUtxo background sync is completed in a `ActivateBestChain()` call, the `GetRole()` function called with `BlockConnected()` returns `ChainstateRole::NORMAL` instead of `ChainstateRole::BACKGROUND` for this chainstate.
  This would make the wallet (which ignores `BlockConnected` notifications for the background chainstate) process it, change `m_last_block_processed_height` to the (ancient) snapshot height, and display an incorrect balance.

  Fix this by caching the chainstate role before calling `ActivateBestChainStep()`.
  Also contains a test for this situation that fails on master.

  Fixes #31546

ACKs for top commit:
  fjahr:
    re-ACK bc43eca
  achow101:
    ACK bc43eca
  furszy:
    Code review ACK bc43eca
  TheCharlatan:
    lgtm ACK bc43eca

Tree-SHA512: c5db677cf3fbab3a33ec127ec6c27c8812299e8368fd3c986bc34d0e515c4eb256f6104479f27829eefc098197de3af75d64ddca636b6b612900a0e21243e4f2
  • Loading branch information
achow101 committed Dec 30, 2024
2 parents fa3de03 + bc43eca commit df5c643
Show file tree
Hide file tree
Showing 2 changed files with 19 additions and 3 deletions.
6 changes: 5 additions & 1 deletion src/validation.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -3529,6 +3529,10 @@ bool Chainstate::ActivateBestChain(BlockValidationState& state, std::shared_ptr<

bool fInvalidFound = false;
std::shared_ptr<const CBlock> nullBlockPtr;
// BlockConnected signals must be sent for the original role;
// in case snapshot validation is completed during ActivateBestChainStep, the
// result of GetRole() changes from BACKGROUND to NORMAL.
const ChainstateRole chainstate_role{this->GetRole()};
if (!ActivateBestChainStep(state, pindexMostWork, pblock && pblock->GetHash() == pindexMostWork->GetBlockHash() ? pblock : nullBlockPtr, fInvalidFound, connectTrace)) {
// A system error occurred
return false;
Expand All @@ -3544,7 +3548,7 @@ bool Chainstate::ActivateBestChain(BlockValidationState& state, std::shared_ptr<
for (const PerBlockConnectTrace& trace : connectTrace.GetBlocksConnected()) {
assert(trace.pblock && trace.pindex);
if (m_chainman.m_options.signals) {
m_chainman.m_options.signals->BlockConnected(this->GetRole(), trace.pblock, trace.pindex);
m_chainman.m_options.signals->BlockConnected(chainstate_role, trace.pblock, trace.pindex);
}
}

Expand Down
16 changes: 14 additions & 2 deletions test/functional/wallet_assumeutxo.py
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@
from test_framework.util import (
assert_equal,
assert_raises_rpc_error,
ensure_for,
)
from test_framework.wallet import MiniWallet

Expand All @@ -34,17 +35,18 @@ def add_options(self, parser):

def set_test_params(self):
"""Use the pregenerated, deterministic chain up to height 199."""
self.num_nodes = 2
self.num_nodes = 3
self.rpc_timeout = 120
self.extra_args = [
[],
[],
[],
]

def setup_network(self):
"""Start with the nodes disconnected so that one can generate a snapshot
including blocks the other hasn't yet seen."""
self.add_nodes(2)
self.add_nodes(3)
self.start_nodes(extra_args=self.extra_args)

def run_test(self):
Expand All @@ -57,6 +59,7 @@ def run_test(self):
"""
n0 = self.nodes[0]
n1 = self.nodes[1]
n2 = self.nodes[2]

self.mini_wallet = MiniWallet(n0)

Expand Down Expand Up @@ -88,6 +91,7 @@ def run_test(self):

# make n1 aware of the new header, but don't give it the block.
n1.submitheader(newblock)
n2.submitheader(newblock)

# Ensure everyone is seeing the same headers.
for n in self.nodes:
Expand Down Expand Up @@ -125,6 +129,7 @@ def run_test(self):

assert_equal(n0.getblockcount(), FINAL_HEIGHT)
assert_equal(n1.getblockcount(), START_HEIGHT)
assert_equal(n2.getblockcount(), START_HEIGHT)

assert_equal(n0.getblockchaininfo()["blocks"], FINAL_HEIGHT)

Expand Down Expand Up @@ -192,6 +197,13 @@ def run_test(self):
w = n1.get_wallet_rpc("w")
assert_equal(w.getbalance(), 34)

self.log.info("Check balance of a wallet that is active during snapshot completion")
n2.restorewallet("w", "backup_w.dat")
loaded = n2.loadtxoutset(dump_output['path'])
self.connect_nodes(0, 2)
self.wait_until(lambda: len(n2.getchainstates()['chainstates']) == 1)
ensure_for(duration=1, f=lambda: (n2.getbalance() == 34))


if __name__ == '__main__':
AssumeutxoTest(__file__).main()

0 comments on commit df5c643

Please sign in to comment.