2016-05-06 11:23:48 +02:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
# Copyright (c) 2014-2016 The Bitcoin Core developers
|
2015-10-08 10:22:50 +02:00
|
|
|
# Distributed under the MIT software license, see the accompanying
|
|
|
|
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
2019-01-07 10:55:35 +01:00
|
|
|
"""Test RPCs related to blockchainstate.
|
2015-10-08 10:22:50 +02:00
|
|
|
|
2019-01-07 10:55:35 +01:00
|
|
|
Test the following RPCs:
|
|
|
|
- gettxoutsetinfo
|
2017-04-21 17:12:12 +02:00
|
|
|
- getdifficulty
|
|
|
|
- getbestblockhash
|
|
|
|
- getblockhash
|
|
|
|
- getblockheader
|
2017-06-05 16:25:12 +02:00
|
|
|
- getchaintxstats
|
2017-05-14 12:43:33 +02:00
|
|
|
- getnetworkhashps
|
2019-01-07 10:55:35 +01:00
|
|
|
- verifychain
|
|
|
|
|
|
|
|
Tests correspond to code in rpc/blockchain.cpp.
|
|
|
|
"""
|
2015-10-08 10:22:50 +02:00
|
|
|
|
2015-12-02 18:12:23 +01:00
|
|
|
from decimal import Decimal
|
2019-07-11 23:22:52 +02:00
|
|
|
import http.client
|
2017-06-22 10:50:24 +02:00
|
|
|
import subprocess
|
2015-10-08 10:22:50 +02:00
|
|
|
|
2017-06-29 17:37:19 +02:00
|
|
|
from test_framework.test_framework import (BitcoinTestFramework, BITCOIND_PROC_WAIT_TIMEOUT)
|
2015-10-08 10:22:50 +02:00
|
|
|
from test_framework.util import (
|
|
|
|
assert_equal,
|
2017-06-22 10:50:24 +02:00
|
|
|
assert_raises,
|
2017-03-16 11:57:09 +01:00
|
|
|
assert_raises_jsonrpc,
|
2015-12-09 18:01:34 +01:00
|
|
|
assert_is_hex_string,
|
|
|
|
assert_is_hash_string,
|
2015-10-08 10:22:50 +02:00
|
|
|
)
|
|
|
|
|
2015-12-09 18:01:34 +01:00
|
|
|
|
2015-10-08 10:22:50 +02:00
|
|
|
class BlockchainTest(BitcoinTestFramework):
|
|
|
|
|
2016-05-20 15:16:51 +02:00
|
|
|
def __init__(self):
|
|
|
|
super().__init__()
|
|
|
|
self.setup_clean_chain = False
|
|
|
|
self.num_nodes = 1
|
2017-06-22 10:50:24 +02:00
|
|
|
self.extra_args = [['-stopatheight=207']]
|
2015-10-08 10:22:50 +02:00
|
|
|
|
|
|
|
def run_test(self):
|
2017-06-05 16:25:12 +02:00
|
|
|
self._test_getchaintxstats()
|
2015-12-09 18:01:34 +01:00
|
|
|
self._test_gettxoutsetinfo()
|
|
|
|
self._test_getblockheader()
|
2017-04-21 17:12:12 +02:00
|
|
|
self._test_getdifficulty()
|
2017-05-14 12:43:33 +02:00
|
|
|
self._test_getnetworkhashps()
|
2017-06-22 10:50:24 +02:00
|
|
|
self._test_stopatheight()
|
|
|
|
assert self.nodes[0].verifychain(4, 0)
|
2015-12-09 18:01:34 +01:00
|
|
|
|
2017-06-05 16:25:12 +02:00
|
|
|
def _test_getchaintxstats(self):
|
|
|
|
chaintxstats = self.nodes[0].getchaintxstats(1)
|
|
|
|
# 200 txs plus genesis tx
|
|
|
|
assert_equal(chaintxstats['txcount'], 201)
|
2019-07-04 17:15:52 +02:00
|
|
|
# tx rate should be 1 per ~2.6 minutes (156 seconds), or 1/156
|
2017-06-05 16:25:12 +02:00
|
|
|
# we have to round because of binary math
|
2019-07-04 17:15:52 +02:00
|
|
|
assert_equal(round(chaintxstats['txrate'] * 156, 10), Decimal(1))
|
2017-06-05 16:25:12 +02:00
|
|
|
|
2015-12-09 18:01:34 +01:00
|
|
|
def _test_gettxoutsetinfo(self):
|
2015-10-08 10:22:50 +02:00
|
|
|
node = self.nodes[0]
|
|
|
|
res = node.gettxoutsetinfo()
|
|
|
|
|
2016-05-06 11:23:48 +02:00
|
|
|
assert_equal(res['total_amount'], Decimal('98214.28571450'))
|
|
|
|
assert_equal(res['transactions'], 200)
|
|
|
|
assert_equal(res['height'], 200)
|
|
|
|
assert_equal(res['txouts'], 200)
|
2017-06-06 11:26:28 +02:00
|
|
|
assert_equal(res['bogosize'], 17000),
|
2017-06-02 00:47:58 +02:00
|
|
|
size = res['disk_size']
|
|
|
|
assert size > 6400
|
|
|
|
assert size < 64000
|
2016-05-06 11:23:48 +02:00
|
|
|
assert_equal(len(res['bestblock']), 64)
|
|
|
|
assert_equal(len(res['hash_serialized_2']), 64)
|
2015-10-08 10:22:50 +02:00
|
|
|
|
2019-03-08 09:05:10 +01:00
|
|
|
self.log.info("Test that gettxoutsetinfo() works for blockchain with just the genesis block")
|
2017-05-26 20:56:52 +02:00
|
|
|
b1hash = node.getblockhash(1)
|
|
|
|
node.invalidateblock(b1hash)
|
|
|
|
|
|
|
|
res2 = node.gettxoutsetinfo()
|
|
|
|
assert_equal(res2['transactions'], 0)
|
|
|
|
assert_equal(res2['total_amount'], Decimal('0'))
|
|
|
|
assert_equal(res2['height'], 0)
|
|
|
|
assert_equal(res2['txouts'], 0)
|
2017-06-06 11:26:28 +02:00
|
|
|
assert_equal(res2['bogosize'], 0),
|
2017-05-26 20:56:52 +02:00
|
|
|
assert_equal(res2['bestblock'], node.getblockhash(0))
|
2017-06-02 00:47:58 +02:00
|
|
|
assert_equal(len(res2['hash_serialized_2']), 64)
|
2017-05-26 20:56:52 +02:00
|
|
|
|
2019-03-08 09:05:10 +01:00
|
|
|
self.log.info("Test that gettxoutsetinfo() returns the same result after invalidate/reconsider block")
|
2017-05-26 20:56:52 +02:00
|
|
|
node.reconsiderblock(b1hash)
|
|
|
|
|
|
|
|
res3 = node.gettxoutsetinfo()
|
|
|
|
assert_equal(res['total_amount'], res3['total_amount'])
|
|
|
|
assert_equal(res['transactions'], res3['transactions'])
|
|
|
|
assert_equal(res['height'], res3['height'])
|
|
|
|
assert_equal(res['txouts'], res3['txouts'])
|
2017-06-06 11:26:28 +02:00
|
|
|
assert_equal(res['bogosize'], res3['bogosize'])
|
2017-05-26 20:56:52 +02:00
|
|
|
assert_equal(res['bestblock'], res3['bestblock'])
|
2017-06-02 00:47:58 +02:00
|
|
|
assert_equal(res['hash_serialized_2'], res3['hash_serialized_2'])
|
2017-05-26 20:56:52 +02:00
|
|
|
|
2015-12-09 18:01:34 +01:00
|
|
|
def _test_getblockheader(self):
|
|
|
|
node = self.nodes[0]
|
|
|
|
|
2017-04-21 17:12:12 +02:00
|
|
|
assert_raises_jsonrpc(-5, "Block not found",
|
|
|
|
node.getblockheader, "nonsense")
|
2015-12-09 18:01:34 +01:00
|
|
|
|
|
|
|
besthash = node.getbestblockhash()
|
|
|
|
secondbesthash = node.getblockhash(199)
|
|
|
|
header = node.getblockheader(besthash)
|
|
|
|
|
|
|
|
assert_equal(header['hash'], besthash)
|
|
|
|
assert_equal(header['height'], 200)
|
|
|
|
assert_equal(header['confirmations'], 1)
|
|
|
|
assert_equal(header['previousblockhash'], secondbesthash)
|
|
|
|
assert_is_hex_string(header['chainwork'])
|
|
|
|
assert_is_hash_string(header['hash'])
|
|
|
|
assert_is_hash_string(header['previousblockhash'])
|
|
|
|
assert_is_hash_string(header['merkleroot'])
|
|
|
|
assert_is_hash_string(header['bits'], length=None)
|
|
|
|
assert isinstance(header['time'], int)
|
|
|
|
assert isinstance(header['mediantime'], int)
|
|
|
|
assert isinstance(header['nonce'], int)
|
|
|
|
assert isinstance(header['version'], int)
|
2016-04-05 15:48:49 +02:00
|
|
|
assert isinstance(int(header['versionHex'], 16), int)
|
2016-01-18 15:17:48 +01:00
|
|
|
assert isinstance(header['difficulty'], Decimal)
|
2015-10-08 10:22:50 +02:00
|
|
|
|
2017-04-21 17:12:12 +02:00
|
|
|
def _test_getdifficulty(self):
|
|
|
|
difficulty = self.nodes[0].getdifficulty()
|
|
|
|
# 1 hash in 2 should be valid, so difficulty should be 1/2**31
|
|
|
|
# binary => decimal => binary math is why we do this check
|
|
|
|
assert abs(difficulty * 2**31 - 1) < 0.0001
|
|
|
|
|
2017-05-14 12:43:33 +02:00
|
|
|
def _test_getnetworkhashps(self):
|
|
|
|
hashes_per_second = self.nodes[0].getnetworkhashps()
|
2019-06-19 20:05:39 +02:00
|
|
|
# This should be 2 hashes every 2.6 minutes (156 seconds) or 1/78
|
|
|
|
assert abs(hashes_per_second * 78 - 1) < 0.0001
|
2017-05-14 12:43:33 +02:00
|
|
|
|
2017-06-22 10:50:24 +02:00
|
|
|
def _test_stopatheight(self):
|
|
|
|
assert_equal(self.nodes[0].getblockcount(), 200)
|
|
|
|
self.nodes[0].generate(6)
|
|
|
|
assert_equal(self.nodes[0].getblockcount(), 206)
|
|
|
|
self.log.debug('Node should not stop at this height')
|
2017-06-29 17:37:19 +02:00
|
|
|
assert_raises(subprocess.TimeoutExpired, lambda: self.bitcoind_processes[0].wait(timeout=3))
|
2019-07-11 23:22:52 +02:00
|
|
|
try:
|
|
|
|
self.nodes[0].generate(1)
|
|
|
|
except (ConnectionError, http.client.BadStatusLine):
|
|
|
|
pass # The node already shut down before response
|
2017-06-22 10:50:24 +02:00
|
|
|
self.log.debug('Node should stop at this height...')
|
2017-06-29 17:37:19 +02:00
|
|
|
self.bitcoind_processes[0].wait(timeout=BITCOIND_PROC_WAIT_TIMEOUT)
|
2017-06-22 10:50:24 +02:00
|
|
|
self.nodes[0] = self.start_node(0, self.options.tmpdir)
|
|
|
|
assert_equal(self.nodes[0].getblockcount(), 207)
|
|
|
|
|
|
|
|
|
2015-10-08 10:22:50 +02:00
|
|
|
if __name__ == '__main__':
|
|
|
|
BlockchainTest().main()
|