2016-05-06 11:23:48 +02:00
|
|
|
#!/usr/bin/env python3
|
2020-12-31 18:50:11 +01:00
|
|
|
# Copyright (c) 2014-2020 The Bitcoin Core developers
|
2015-04-09 17:08:39 +02:00
|
|
|
# Distributed under the MIT software license, see the accompanying
|
|
|
|
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
2020-06-11 10:39:04 +02:00
|
|
|
"""Test running dashd with -reindex and -reindex-chainstate options.
|
2019-01-07 10:55:35 +01:00
|
|
|
|
|
|
|
- Start a single node and generate 3 blocks.
|
|
|
|
- Stop the node and restart it with -reindex. Verify that the node has reindexed up to block 3.
|
|
|
|
- Stop the node and restart it with -reindex-chainstate. Verify that the node has reindexed up to block 3.
|
2024-10-13 18:46:11 +02:00
|
|
|
- Verify that out-of-order blocks are correctly processed, see LoadExternalBlockFile()
|
2019-01-07 10:55:35 +01:00
|
|
|
"""
|
2015-04-09 17:08:39 +02:00
|
|
|
|
2024-10-13 18:46:11 +02:00
|
|
|
import os
|
2015-05-02 12:53:35 +02:00
|
|
|
from test_framework.test_framework import BitcoinTestFramework
|
2024-10-13 18:46:11 +02:00
|
|
|
from test_framework.p2p import MAGIC_BYTES
|
2022-09-07 20:48:44 +02:00
|
|
|
from test_framework.util import assert_equal
|
2015-04-09 17:08:39 +02:00
|
|
|
|
|
|
|
|
2022-09-07 20:48:44 +02:00
|
|
|
class ReindexTest(BitcoinTestFramework):
|
2017-09-01 18:47:13 +02:00
|
|
|
def set_test_params(self):
|
2016-05-20 15:16:51 +02:00
|
|
|
self.setup_clean_chain = True
|
|
|
|
self.num_nodes = 1
|
2015-04-09 17:08:39 +02:00
|
|
|
|
2024-10-08 12:46:37 +02:00
|
|
|
def reindex(self, justchainstate=False, txindex=0):
|
2024-10-01 21:25:52 +02:00
|
|
|
self.generatetoaddress(self.nodes[0], 3, self.nodes[0].get_deterministic_priv_key().address)
|
2017-07-10 16:41:14 +02:00
|
|
|
blockcount = self.nodes[0].getblockcount()
|
2017-06-02 12:08:48 +02:00
|
|
|
self.stop_nodes()
|
2024-10-08 12:46:37 +02:00
|
|
|
extra_args = [["-reindex-chainstate", "-txindex=0"]] if justchainstate else [["-reindex", f"-txindex={txindex}"]]
|
2017-09-01 18:47:13 +02:00
|
|
|
self.start_nodes(extra_args)
|
2022-09-07 20:48:44 +02:00
|
|
|
assert_equal(self.nodes[0].getblockcount(), blockcount) # start_node is blocking on reindex
|
2017-03-09 21:16:20 +01:00
|
|
|
self.log.info("Success")
|
2017-07-10 16:41:14 +02:00
|
|
|
|
2024-10-13 18:46:11 +02:00
|
|
|
# Check that blocks can be processed out of order
|
|
|
|
def out_of_order(self):
|
|
|
|
# The previous test created 24 blocks
|
|
|
|
assert_equal(self.nodes[0].getblockcount(), 24)
|
|
|
|
self.stop_nodes()
|
|
|
|
|
|
|
|
# In this test environment, blocks will always be in order (since
|
|
|
|
# we're generating them rather than getting them from peers), so to
|
|
|
|
# test out-of-order handling, swap blocks 1 and 2 on disk.
|
|
|
|
blk0 = os.path.join(self.nodes[0].datadir, self.nodes[0].chain, 'blocks', 'blk00000.dat')
|
|
|
|
with open(blk0, 'r+b') as bf:
|
|
|
|
# Read at least the first few blocks (including genesis)
|
|
|
|
b = bf.read(2000)
|
|
|
|
|
|
|
|
# Find the offsets of blocks 2, 3, and 4 (the first 3 blocks beyond genesis)
|
|
|
|
# by searching for the regtest marker bytes (see pchMessageStart).
|
|
|
|
def find_block(b, start):
|
|
|
|
return b.find(MAGIC_BYTES["regtest"], start)+4
|
|
|
|
|
|
|
|
genesis_start = find_block(b, 0)
|
|
|
|
assert_equal(genesis_start, 4)
|
|
|
|
b2_start = find_block(b, genesis_start)
|
|
|
|
b3_start = find_block(b, b2_start)
|
|
|
|
b4_start = find_block(b, b3_start)
|
|
|
|
|
|
|
|
# Blocks 2 and 3 should be the same size.
|
|
|
|
assert_equal(b3_start-b2_start, b4_start-b3_start)
|
|
|
|
|
|
|
|
# Swap the second and third blocks (don't disturb the genesis block).
|
|
|
|
bf.seek(b2_start)
|
|
|
|
bf.write(b[b3_start:b4_start])
|
|
|
|
bf.write(b[b2_start:b3_start])
|
|
|
|
|
|
|
|
# The reindexing code should detect and accommodate out of order blocks.
|
|
|
|
with self.nodes[0].assert_debug_log([
|
|
|
|
'LoadExternalBlockFile: Out of order block',
|
|
|
|
'LoadExternalBlockFile: Processing out of order child',
|
|
|
|
]):
|
|
|
|
extra_args = [["-reindex"]]
|
|
|
|
self.start_nodes(extra_args)
|
|
|
|
|
|
|
|
# All blocks should be accepted and processed.
|
|
|
|
assert_equal(self.nodes[0].getblockcount(), 24)
|
|
|
|
|
2017-07-10 16:41:14 +02:00
|
|
|
def run_test(self):
|
2024-10-08 12:46:37 +02:00
|
|
|
for txindex in [0, 1]:
|
|
|
|
self.reindex(False, txindex)
|
|
|
|
self.reindex(True, txindex)
|
|
|
|
self.reindex(False, txindex)
|
|
|
|
self.reindex(True, txindex)
|
2015-04-09 17:08:39 +02:00
|
|
|
|
2024-10-13 18:46:11 +02:00
|
|
|
self.out_of_order()
|
|
|
|
|
|
|
|
|
2015-04-09 17:08:39 +02:00
|
|
|
if __name__ == '__main__':
|
|
|
|
ReindexTest().main()
|