2016-03-19 20:58:06 +01:00
|
|
|
#!/usr/bin/env python3
|
2021-07-28 13:57:16 +02:00
|
|
|
# Copyright (c) 2014-2021 The Bitcoin Core developers
|
2014-10-23 09:48:19 +08:00
|
|
|
# Distributed under the MIT software license, see the accompanying
|
2014-08-03 18:12:19 +02:00
|
|
|
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
2017-01-17 18:34:40 -05:00
|
|
|
"""Test the getchaintips RPC.
|
2014-08-03 18:12:19 +02:00
|
|
|
|
2017-01-17 18:34:40 -05:00
|
|
|
- introduce a network split
|
|
|
|
- work on chains of different lengths
|
|
|
|
- join the network together again
|
|
|
|
- verify that getchaintips now returns two chain tips.
|
|
|
|
"""
|
2014-08-03 18:12:19 +02:00
|
|
|
|
2024-08-16 11:38:55 -04:00
|
|
|
from test_framework.blocktools import (
|
|
|
|
create_block,
|
|
|
|
create_coinbase,
|
|
|
|
)
|
2015-05-02 12:53:35 +02:00
|
|
|
from test_framework.test_framework import BitcoinTestFramework
|
|
|
|
from test_framework.util import assert_equal
|
2014-08-03 18:12:19 +02:00
|
|
|
|
|
|
|
class GetChainTipsTest (BitcoinTestFramework):
|
2017-08-24 11:11:56 -04:00
|
|
|
def set_test_params(self):
|
|
|
|
self.num_nodes = 4
|
|
|
|
|
2018-09-09 13:32:37 -04:00
|
|
|
def run_test(self):
|
2024-05-29 16:48:16 -04:00
|
|
|
self.log.info("Test getchaintips behavior with two chains of different length")
|
2018-09-09 13:32:37 -04:00
|
|
|
tips = self.nodes[0].getchaintips()
|
|
|
|
assert_equal(len(tips), 1)
|
|
|
|
assert_equal(tips[0]['branchlen'], 0)
|
|
|
|
assert_equal(tips[0]['height'], 200)
|
|
|
|
assert_equal(tips[0]['status'], 'active')
|
2014-10-20 14:14:04 +02:00
|
|
|
|
2024-05-29 16:48:16 -04:00
|
|
|
self.log.info("Split the network and build two chains of different lengths.")
|
2018-09-09 13:32:37 -04:00
|
|
|
self.split_network()
|
2020-11-10 18:02:31 +01:00
|
|
|
self.generate(self.nodes[0], 10, sync_fun=lambda: self.sync_all(self.nodes[:2]))
|
|
|
|
self.generate(self.nodes[2], 20, sync_fun=lambda: self.sync_all(self.nodes[2:]))
|
2014-10-20 14:14:04 +02:00
|
|
|
|
2024-05-29 16:48:16 -04:00
|
|
|
tips = self.nodes[1].getchaintips()
|
|
|
|
assert_equal(len(tips), 1)
|
2014-10-20 14:14:04 +02:00
|
|
|
shortTip = tips[0]
|
2024-05-29 16:48:16 -04:00
|
|
|
assert_equal(shortTip['branchlen'], 0)
|
|
|
|
assert_equal(shortTip['height'], 210)
|
|
|
|
assert_equal(tips[0]['status'], 'active')
|
2014-10-20 14:14:04 +02:00
|
|
|
|
2024-05-29 16:48:16 -04:00
|
|
|
tips = self.nodes[3].getchaintips()
|
|
|
|
assert_equal(len(tips), 1)
|
2014-10-20 14:14:04 +02:00
|
|
|
longTip = tips[0]
|
2024-05-29 16:48:16 -04:00
|
|
|
assert_equal(longTip['branchlen'], 0)
|
|
|
|
assert_equal(longTip['height'], 220)
|
|
|
|
assert_equal(tips[0]['status'], 'active')
|
2014-10-20 14:14:04 +02:00
|
|
|
|
2024-05-29 16:48:16 -04:00
|
|
|
self.log.info("Join the network halves and check that we now have two tips")
|
2014-10-20 14:14:04 +02:00
|
|
|
# (at least at the nodes that previously had the short chain).
|
2024-05-29 16:48:16 -04:00
|
|
|
self.join_network()
|
2014-10-20 14:14:04 +02:00
|
|
|
|
2024-05-29 16:48:16 -04:00
|
|
|
tips = self.nodes[0].getchaintips()
|
|
|
|
assert_equal(len(tips), 2)
|
|
|
|
assert_equal(tips[0], longTip)
|
2014-10-20 14:14:04 +02:00
|
|
|
|
2024-05-29 16:48:16 -04:00
|
|
|
assert_equal(tips[1]['branchlen'], 10)
|
|
|
|
assert_equal(tips[1]['status'], 'valid-fork')
|
2014-11-27 10:46:55 +01:00
|
|
|
tips[1]['branchlen'] = 0
|
|
|
|
tips[1]['status'] = 'active'
|
2024-05-29 16:48:16 -04:00
|
|
|
assert_equal(tips[1], shortTip)
|
2014-08-03 18:12:19 +02:00
|
|
|
|
2024-08-16 11:38:55 -04:00
|
|
|
self.log.info("Test getchaintips behavior with invalid blocks")
|
|
|
|
self.disconnect_nodes(0, 1)
|
|
|
|
n0 = self.nodes[0]
|
|
|
|
tip = int(n0.getbestblockhash(), 16)
|
|
|
|
start_height = self.nodes[0].getblockcount()
|
|
|
|
# Create invalid block (too high coinbase)
|
|
|
|
block_time = n0.getblock(n0.getbestblockhash())['time'] + 1
|
|
|
|
invalid_block = create_block(tip, create_coinbase(start_height+1, nValue=100), block_time)
|
|
|
|
invalid_block.solve()
|
|
|
|
|
|
|
|
block_time += 1
|
|
|
|
block2 = create_block(invalid_block.sha256, create_coinbase(2), block_time, version=4)
|
|
|
|
block2.solve()
|
|
|
|
|
|
|
|
self.log.info("Submit headers-only chain")
|
|
|
|
n0.submitheader(invalid_block.serialize().hex())
|
|
|
|
n0.submitheader(block2.serialize().hex())
|
|
|
|
tips = n0.getchaintips()
|
|
|
|
assert_equal(len(tips), 3)
|
|
|
|
assert_equal(tips[0]['status'], 'headers-only')
|
|
|
|
|
|
|
|
self.log.info("Submit invalid block that invalidates the headers-only chain")
|
|
|
|
n0.submitblock(invalid_block.serialize().hex())
|
|
|
|
tips = n0.getchaintips()
|
|
|
|
assert_equal(len(tips), 3)
|
|
|
|
assert_equal(tips[0]['status'], 'invalid')
|
|
|
|
|
|
|
|
|
2014-08-03 18:12:19 +02:00
|
|
|
if __name__ == '__main__':
|
2024-07-16 22:05:14 +01:00
|
|
|
GetChainTipsTest(__file__).main()
|