2016-03-19 20:58:06 +01:00
#!/usr/bin/env python3
2019-02-21 02:03:13 +01:00
# Copyright (c) 2014-2019 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.
2017-01-18 00:34:40 +01:00
""" Test RPCs related to blockchainstate.
2015-10-08 10:22:50 +02:00
2017-01-18 00:34:40 +01:00
Test the following RPCs :
2017-09-06 20:05:28 +02:00
- getblockchaininfo
2017-01-18 00:34:40 +01:00
- gettxoutsetinfo
2017-04-18 22:11:06 +02:00
- getdifficulty
- getbestblockhash
- getblockhash
- getblockheader
2017-06-03 06:48:32 +02:00
- getchaintxstats
2017-04-20 20:43:58 +02:00
- getnetworkhashps
2017-01-18 00:34:40 +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
2017-06-23 09:17:18 +02:00
import http . client
2017-06-19 21:40:38 +02:00
import subprocess
2015-10-08 10:22:50 +02:00
2017-08-16 17:52:24 +02:00
from test_framework . test_framework import BitcoinTestFramework
2015-10-08 10:22:50 +02:00
from test_framework . util import (
assert_equal ,
2017-09-21 01:52:20 +02:00
assert_greater_than ,
assert_greater_than_or_equal ,
2017-06-19 21:40:38 +02:00
assert_raises ,
2017-07-12 16:33:46 +02:00
assert_raises_rpc_error ,
2015-12-09 18:02:59 +01:00
assert_is_hex_string ,
assert_is_hash_string ,
2015-10-08 10:22:50 +02:00
)
2018-02-14 04:37:36 +01:00
from test_framework . blocktools import (
create_block ,
create_coinbase ,
2019-02-11 17:18:36 +01:00
TIME_GENESIS_BLOCK ,
2018-02-14 04:37:36 +01:00
)
from test_framework . messages import (
msg_block ,
)
from test_framework . mininode import (
P2PInterface ,
)
2015-10-08 10:22:50 +02:00
class BlockchainTest ( BitcoinTestFramework ) :
2017-06-10 00:21:21 +02:00
def set_test_params ( self ) :
2019-02-11 17:18:36 +01:00
self . setup_clean_chain = True
2017-05-24 00:08:04 +02:00
self . num_nodes = 1
2015-10-08 10:22:50 +02:00
2018-09-09 19:32:37 +02:00
def skip_test_if_missing_module ( self ) :
self . skip_if_no_wallet ( )
2015-10-08 10:22:50 +02:00
def run_test ( self ) :
2019-02-11 17:18:36 +01:00
self . mine_chain ( )
2018-09-10 22:58:15 +02:00
self . restart_node ( 0 , extra_args = [ ' -stopatheight=207 ' , ' -prune=1 ' ] ) # Set extra args with pruning after rescan is complete
2017-09-20 01:45:39 +02:00
self . _test_getblockchaininfo ( )
2017-06-03 06:48:32 +02:00
self . _test_getchaintxstats ( )
2015-12-09 18:02:59 +01:00
self . _test_gettxoutsetinfo ( )
self . _test_getblockheader ( )
2017-04-18 22:11:06 +02:00
self . _test_getdifficulty ( )
2017-04-20 20:43:58 +02:00
self . _test_getnetworkhashps ( )
2017-06-19 21:40:38 +02:00
self . _test_stopatheight ( )
2018-02-14 04:37:36 +01:00
self . _test_waitforblockheight ( )
2017-06-19 21:40:38 +02:00
assert self . nodes [ 0 ] . verifychain ( 4 , 0 )
2015-12-09 18:02:59 +01:00
2019-02-11 17:18:36 +01:00
def mine_chain ( self ) :
self . log . info ( ' Create some old blocks ' )
address = self . nodes [ 0 ] . get_deterministic_priv_key ( ) . address
for t in range ( TIME_GENESIS_BLOCK , TIME_GENESIS_BLOCK + 200 * 600 , 600 ) :
# ten-minute steps from genesis block time
self . nodes [ 0 ] . setmocktime ( t )
self . nodes [ 0 ] . generatetoaddress ( 1 , address )
assert_equal ( self . nodes [ 0 ] . getblockchaininfo ( ) [ ' blocks ' ] , 200 )
2017-09-20 01:45:39 +02:00
def _test_getblockchaininfo ( self ) :
self . log . info ( " Test getblockchaininfo " )
keys = [
' bestblockhash ' ,
' blocks ' ,
' chain ' ,
' chainwork ' ,
' difficulty ' ,
' headers ' ,
2017-09-06 20:05:28 +02:00
' initialblockdownload ' ,
2017-09-20 01:45:39 +02:00
' mediantime ' ,
' pruned ' ,
2017-09-21 01:52:20 +02:00
' size_on_disk ' ,
2017-09-20 01:45:39 +02:00
' softforks ' ,
' verificationprogress ' ,
2017-07-18 00:43:12 +02:00
' warnings ' ,
2017-09-20 01:45:39 +02:00
]
res = self . nodes [ 0 ] . getblockchaininfo ( )
2017-09-21 01:52:20 +02:00
# result should have these additional pruning keys if manual pruning is enabled
assert_equal ( sorted ( res . keys ( ) ) , sorted ( [ ' pruneheight ' , ' automatic_pruning ' ] + keys ) )
# size_on_disk should be > 0
assert_greater_than ( res [ ' size_on_disk ' ] , 0 )
2017-09-20 01:45:39 +02:00
# pruneheight should be greater or equal to 0
2017-09-21 01:52:20 +02:00
assert_greater_than_or_equal ( res [ ' pruneheight ' ] , 0 )
# check other pruning fields given that prune=1
assert res [ ' pruned ' ]
assert not res [ ' automatic_pruning ' ]
2017-09-20 01:45:39 +02:00
self . restart_node ( 0 , [ ' -stopatheight=207 ' ] )
res = self . nodes [ 0 ] . getblockchaininfo ( )
# should have exact keys
assert_equal ( sorted ( res . keys ( ) ) , keys )
2017-09-21 01:52:20 +02:00
self . restart_node ( 0 , [ ' -stopatheight=207 ' , ' -prune=550 ' ] )
res = self . nodes [ 0 ] . getblockchaininfo ( )
# result should have these additional pruning keys if prune=550
assert_equal ( sorted ( res . keys ( ) ) , sorted ( [ ' pruneheight ' , ' automatic_pruning ' , ' prune_target_size ' ] + keys ) )
# check related fields
assert res [ ' pruned ' ]
assert_equal ( res [ ' pruneheight ' ] , 0 )
assert res [ ' automatic_pruning ' ]
assert_equal ( res [ ' prune_target_size ' ] , 576716800 )
assert_greater_than ( res [ ' size_on_disk ' ] , 0 )
2018-02-13 15:35:22 +01:00
assert_equal ( res [ ' softforks ' ] , {
2020-02-13 16:41:28 +01:00
' bip34 ' : { ' type ' : ' buried ' , ' active ' : False , ' height ' : 1000 } ,
2018-02-13 15:35:22 +01:00
' bip66 ' : { ' type ' : ' buried ' , ' active ' : False , ' height ' : 1251 } ,
' bip65 ' : { ' type ' : ' buried ' , ' active ' : False , ' height ' : 1351 } ,
2020-02-13 16:41:28 +01:00
' csv ' : { ' type ' : ' buried ' , ' active ' : True , ' height ' : 0 } ,
' segwit ' : { ' type ' : ' buried ' , ' active ' : True , ' height ' : 150 } ,
2018-02-13 15:35:22 +01:00
' testdummy ' : {
' type ' : ' bip9 ' ,
' bip9 ' : {
' status ' : ' started ' ,
' bit ' : 28 ,
2019-09-06 13:30:15 +02:00
' start_time ' : 0 ,
2018-02-13 15:35:22 +01:00
' timeout ' : 0x7fffffffffffffff , # testdummy does not have a timeout so is set to the max int64 value
' since ' : 144 ,
' statistics ' : {
' period ' : 144 ,
' threshold ' : 108 ,
' elapsed ' : 57 ,
' count ' : 57 ,
' possible ' : True ,
} ,
} ,
' active ' : False }
} )
2017-06-03 06:48:32 +02:00
def _test_getchaintxstats ( self ) :
2018-01-31 19:47:06 +01:00
self . log . info ( " Test getchaintxstats " )
2018-01-04 02:01:05 +01:00
# Test `getchaintxstats` invalid extra parameters
assert_raises_rpc_error ( - 1 , ' getchaintxstats ' , self . nodes [ 0 ] . getchaintxstats , 0 , ' ' , 0 )
# Test `getchaintxstats` invalid `nblocks`
assert_raises_rpc_error ( - 1 , " JSON value is not an integer as expected " , self . nodes [ 0 ] . getchaintxstats , ' ' )
assert_raises_rpc_error ( - 8 , " Invalid block count: should be between 0 and the block ' s height - 1 " , self . nodes [ 0 ] . getchaintxstats , - 1 )
assert_raises_rpc_error ( - 8 , " Invalid block count: should be between 0 and the block ' s height - 1 " , self . nodes [ 0 ] . getchaintxstats , self . nodes [ 0 ] . getblockcount ( ) )
# Test `getchaintxstats` invalid `blockhash`
assert_raises_rpc_error ( - 1 , " JSON value is not a string as expected " , self . nodes [ 0 ] . getchaintxstats , blockhash = 0 )
2018-06-08 20:16:07 +02:00
assert_raises_rpc_error ( - 8 , " blockhash must be of length 64 (not 1, for ' 0 ' ) " , self . nodes [ 0 ] . getchaintxstats , blockhash = ' 0 ' )
assert_raises_rpc_error ( - 8 , " blockhash must be hexadecimal string (not ' ZZZ0000000000000000000000000000000000000000000000000000000000000 ' ) " , self . nodes [ 0 ] . getchaintxstats , blockhash = ' ZZZ0000000000000000000000000000000000000000000000000000000000000 ' )
assert_raises_rpc_error ( - 5 , " Block not found " , self . nodes [ 0 ] . getchaintxstats , blockhash = ' 0000000000000000000000000000000000000000000000000000000000000000 ' )
2018-01-04 02:01:05 +01:00
blockhash = self . nodes [ 0 ] . getblockhash ( 200 )
self . nodes [ 0 ] . invalidateblock ( blockhash )
assert_raises_rpc_error ( - 8 , " Block is not in main chain " , self . nodes [ 0 ] . getchaintxstats , blockhash = blockhash )
self . nodes [ 0 ] . reconsiderblock ( blockhash )
2018-11-13 19:42:36 +01:00
chaintxstats = self . nodes [ 0 ] . getchaintxstats ( nblocks = 1 )
2017-06-03 06:48:32 +02:00
# 200 txs plus genesis tx
assert_equal ( chaintxstats [ ' txcount ' ] , 201 )
# tx rate should be 1 per 10 minutes, or 1/600
# we have to round because of binary math
assert_equal ( round ( chaintxstats [ ' txrate ' ] * 600 , 10 ) , Decimal ( 1 ) )
2018-01-31 19:47:06 +01:00
b1_hash = self . nodes [ 0 ] . getblockhash ( 1 )
b1 = self . nodes [ 0 ] . getblock ( b1_hash )
b200_hash = self . nodes [ 0 ] . getblockhash ( 200 )
b200 = self . nodes [ 0 ] . getblock ( b200_hash )
2017-08-14 09:29:00 +02:00
time_diff = b200 [ ' mediantime ' ] - b1 [ ' mediantime ' ]
chaintxstats = self . nodes [ 0 ] . getchaintxstats ( )
assert_equal ( chaintxstats [ ' time ' ] , b200 [ ' time ' ] )
assert_equal ( chaintxstats [ ' txcount ' ] , 201 )
2018-01-31 19:47:06 +01:00
assert_equal ( chaintxstats [ ' window_final_block_hash ' ] , b200_hash )
2019-08-23 16:42:28 +02:00
assert_equal ( chaintxstats [ ' window_final_block_height ' ] , 200 )
2017-08-14 09:29:00 +02:00
assert_equal ( chaintxstats [ ' window_block_count ' ] , 199 )
assert_equal ( chaintxstats [ ' window_tx_count ' ] , 199 )
assert_equal ( chaintxstats [ ' window_interval ' ] , time_diff )
assert_equal ( round ( chaintxstats [ ' txrate ' ] * time_diff , 10 ) , Decimal ( 199 ) )
2018-01-31 19:47:06 +01:00
chaintxstats = self . nodes [ 0 ] . getchaintxstats ( blockhash = b1_hash )
2017-08-14 09:29:00 +02:00
assert_equal ( chaintxstats [ ' time ' ] , b1 [ ' time ' ] )
assert_equal ( chaintxstats [ ' txcount ' ] , 2 )
2018-01-31 19:47:06 +01:00
assert_equal ( chaintxstats [ ' window_final_block_hash ' ] , b1_hash )
2019-08-23 16:42:28 +02:00
assert_equal ( chaintxstats [ ' window_final_block_height ' ] , 1 )
2017-08-14 09:29:00 +02:00
assert_equal ( chaintxstats [ ' window_block_count ' ] , 0 )
2019-02-19 23:43:44 +01:00
assert ' window_tx_count ' not in chaintxstats
assert ' window_interval ' not in chaintxstats
assert ' txrate ' not in chaintxstats
2017-08-14 09:29:00 +02:00
2015-12-09 18:02:59 +01:00
def _test_gettxoutsetinfo ( self ) :
2015-10-08 10:22:50 +02:00
node = self . nodes [ 0 ]
res = node . gettxoutsetinfo ( )
2019-09-19 16:31:07 +02:00
assert_equal ( res [ ' total_amount ' ] , Decimal ( ' 400000200.00000000 ' ) )
2019-09-27 18:49:15 +02:00
assert_equal ( res [ ' transactions ' ] , 201 )
2016-03-19 20:58:06 +01:00
assert_equal ( res [ ' height ' ] , 200 )
2019-09-27 18:49:15 +02:00
assert_equal ( res [ ' txouts ' ] , 201 )
assert_equal ( res [ ' bogosize ' ] , 15075 ) ,
2017-05-24 00:08:04 +02:00
assert_equal ( res [ ' bestblock ' ] , node . getblockhash ( 200 ) )
2017-05-13 00:19:19 +02:00
size = res [ ' disk_size ' ]
assert size > 6400
2020-02-13 16:41:28 +01:00
assert size < 164000
2016-03-19 20:58:06 +01:00
assert_equal ( len ( res [ ' bestblock ' ] ) , 64 )
2017-04-25 20:29:18 +02:00
assert_equal ( len ( res [ ' hash_serialized_2 ' ] ) , 64 )
2015-10-08 10:22:50 +02:00
2017-05-24 00:08:04 +02:00
self . log . info ( " Test that gettxoutsetinfo() works for blockchain with just the genesis block " )
b1hash = node . getblockhash ( 1 )
node . invalidateblock ( b1hash )
res2 = node . gettxoutsetinfo ( )
2019-09-27 18:49:15 +02:00
assert_equal ( res2 [ ' transactions ' ] , 1 )
assert_equal ( res2 [ ' total_amount ' ] , Decimal ( ' 400000000.00000000 ' ) )
2017-05-24 00:08:04 +02:00
assert_equal ( res2 [ ' height ' ] , 0 )
2019-09-27 18:49:15 +02:00
assert_equal ( res2 [ ' txouts ' ] , 1 )
assert_equal ( res2 [ ' bogosize ' ] , 75 ) ,
2017-05-24 00:08:04 +02:00
assert_equal ( res2 [ ' bestblock ' ] , node . getblockhash ( 0 ) )
2017-04-25 20:29:18 +02:00
assert_equal ( len ( res2 [ ' hash_serialized_2 ' ] ) , 64 )
2017-05-24 00:08:04 +02:00
self . log . info ( " Test that gettxoutsetinfo() returns the same result after invalidate/reconsider block " )
node . reconsiderblock ( b1hash )
res3 = node . gettxoutsetinfo ( )
2018-08-09 16:16:25 +02:00
# The field 'disk_size' is non-deterministic and can thus not be
# compared between res and res3. Everything else should be the same.
del res [ ' disk_size ' ] , res3 [ ' disk_size ' ]
assert_equal ( res , res3 )
2017-05-24 00:08:04 +02:00
2015-12-09 18:02:59 +01:00
def _test_getblockheader ( self ) :
node = self . nodes [ 0 ]
2018-06-08 20:16:07 +02:00
assert_raises_rpc_error ( - 8 , " hash must be of length 64 (not 8, for ' nonsense ' ) " , node . getblockheader , " nonsense " )
assert_raises_rpc_error ( - 8 , " hash must be hexadecimal string (not ' ZZZ7bb8b1697ea987f3b223ba7819250cae33efacb068d23dc24859824a77844 ' ) " , node . getblockheader , " ZZZ7bb8b1697ea987f3b223ba7819250cae33efacb068d23dc24859824a77844 " )
assert_raises_rpc_error ( - 5 , " Block not found " , node . getblockheader , " 0cf7bb8b1697ea987f3b223ba7819250cae33efacb068d23dc24859824a77844 " )
2015-12-09 18:02:59 +01:00
besthash = node . getbestblockhash ( )
secondbesthash = node . getblockhash ( 199 )
2018-11-13 19:42:36 +01:00
header = node . getblockheader ( blockhash = besthash )
2015-12-09 18:02:59 +01:00
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 ' ] )
2018-06-12 22:39:29 +02:00
assert_equal ( header [ ' nTx ' ] , 1 )
2015-12-09 18:02:59 +01:00
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 00:21:00 +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-18 22:11:06 +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
2019-09-27 18:49:15 +02:00
assert abs ( difficulty ) < 0.0001
2017-04-18 22:11:06 +02:00
2017-04-20 20:43:58 +02:00
def _test_getnetworkhashps ( self ) :
hashes_per_second = self . nodes [ 0 ] . getnetworkhashps ( )
# This should be 2 hashes every 10 minutes or 1/300
assert abs ( hashes_per_second * 300 - 1 ) < 0.0001
2017-06-19 21:40:38 +02:00
def _test_stopatheight ( self ) :
assert_equal ( self . nodes [ 0 ] . getblockcount ( ) , 200 )
2018-09-15 11:32:12 +02:00
self . nodes [ 0 ] . generatetoaddress ( 6 , self . nodes [ 0 ] . get_deterministic_priv_key ( ) . address )
2017-06-19 21:40:38 +02:00
assert_equal ( self . nodes [ 0 ] . getblockcount ( ) , 206 )
self . log . debug ( ' Node should not stop at this height ' )
2017-06-02 20:30:36 +02:00
assert_raises ( subprocess . TimeoutExpired , lambda : self . nodes [ 0 ] . process . wait ( timeout = 3 ) )
2017-06-23 09:17:18 +02:00
try :
2018-09-15 11:32:12 +02:00
self . nodes [ 0 ] . generatetoaddress ( 1 , self . nodes [ 0 ] . get_deterministic_priv_key ( ) . address )
2017-06-23 09:17:18 +02:00
except ( ConnectionError , http . client . BadStatusLine ) :
pass # The node already shut down before response
2017-06-19 21:40:38 +02:00
self . log . debug ( ' Node should stop at this height... ' )
2017-08-16 17:52:24 +02:00
self . nodes [ 0 ] . wait_until_stopped ( )
2017-06-09 22:35:17 +02:00
self . start_node ( 0 )
2017-06-19 21:40:38 +02:00
assert_equal ( self . nodes [ 0 ] . getblockcount ( ) , 207 )
2018-02-14 04:37:36 +01:00
def _test_waitforblockheight ( self ) :
self . log . info ( " Test waitforblockheight " )
node = self . nodes [ 0 ]
node . add_p2p_connection ( P2PInterface ( ) )
current_height = node . getblock ( node . getbestblockhash ( ) ) [ ' height ' ]
# Create a fork somewhere below our current height, invalidate the tip
# of that fork, and then ensure that waitforblockheight still
# works as expected.
#
# (Previously this was broken based on setting
# `rpc/blockchain.cpp:latestblock` incorrectly.)
#
b20hash = node . getblockhash ( 20 )
b20 = node . getblock ( b20hash )
def solve_and_send_block ( prevhash , height , time ) :
b = create_block ( prevhash , create_coinbase ( height ) , time )
b . solve ( )
node . p2p . send_message ( msg_block ( b ) )
node . p2p . sync_with_ping ( )
return b
b21f = solve_and_send_block ( int ( b20hash , 16 ) , 21 , b20 [ ' time ' ] + 1 )
b22f = solve_and_send_block ( b21f . sha256 , 22 , b21f . nTime + 1 )
node . invalidateblock ( b22f . hash )
def assert_waitforheight ( height , timeout = 2 ) :
assert_equal (
2018-11-13 19:42:36 +01:00
node . waitforblockheight ( height = height , timeout = timeout ) [ ' height ' ] ,
2018-02-14 04:37:36 +01:00
current_height )
assert_waitforheight ( 0 )
assert_waitforheight ( current_height - 1 )
assert_waitforheight ( current_height )
assert_waitforheight ( current_height + 1 )
2017-06-19 21:40:38 +02:00
2015-10-08 10:22:50 +02:00
if __name__ == ' __main__ ' :
BlockchainTest ( ) . main ( )