2019-08-26 15:49:57 -04:00
#!/usr/bin/env python3
2022-12-24 20:49:50 -03:00
# Copyright (c) 2020-2022 The Bitcoin Core developers
2019-08-26 15:49:57 -04:00
# Distributed under the MIT software license, see the accompanying
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
""" Test coinstatsindex across nodes.
Test that the values returned by gettxoutsetinfo are consistent
between a node running the coinstatsindex and a node without
the index .
"""
2020-08-22 12:51:45 -04:00
from decimal import Decimal
from test_framework . blocktools import (
2021-05-17 10:38:19 -04:00
COINBASE_MATURITY ,
2020-08-22 12:51:45 -04:00
create_block ,
create_coinbase ,
)
from test_framework . messages import (
COIN ,
CTxOut ,
)
from test_framework . script import (
CScript ,
OP_FALSE ,
OP_RETURN ,
)
2019-08-26 15:49:57 -04:00
from test_framework . test_framework import BitcoinTestFramework
from test_framework . util import (
assert_equal ,
assert_raises_rpc_error ,
)
2022-03-17 15:54:45 -03:00
from test_framework . wallet import (
MiniWallet ,
getnewdestination ,
)
2019-08-26 15:49:57 -04:00
class CoinStatsIndexTest ( BitcoinTestFramework ) :
def set_test_params ( self ) :
self . setup_clean_chain = True
self . num_nodes = 2
self . supports_cli = False
self . extra_args = [
2022-03-17 15:54:45 -03:00
[ ] ,
2019-08-26 15:49:57 -04:00
[ " -coinstatsindex " ]
]
def run_test ( self ) :
2022-03-17 15:54:45 -03:00
self . wallet = MiniWallet ( self . nodes [ 0 ] )
2019-08-26 15:49:57 -04:00
self . _test_coin_stats_index ( )
2021-02-28 15:27:00 -03:00
self . _test_use_index_option ( )
2021-03-03 20:31:12 -03:00
self . _test_reorg_index ( )
2021-03-03 21:37:50 -03:00
self . _test_index_rejects_hash_serialized ( )
2019-08-26 15:49:57 -04:00
2020-08-22 12:51:45 -04:00
def block_sanity_check ( self , block_info ) :
block_subsidy = 50
assert_equal (
block_info [ ' prevout_spent ' ] + block_subsidy ,
block_info [ ' new_outputs_ex_coinbase ' ] + block_info [ ' coinbase ' ] + block_info [ ' unspendable ' ]
)
2019-08-26 15:49:57 -04:00
def _test_coin_stats_index ( self ) :
node = self . nodes [ 0 ]
index_node = self . nodes [ 1 ]
# Both none and muhash options allow the usage of the index
index_hash_options = [ ' none ' , ' muhash ' ]
# Generate a normal transaction and mine it
2022-03-17 15:54:45 -03:00
self . generate ( self . wallet , COINBASE_MATURITY + 1 )
self . wallet . send_self_transfer ( from_node = node )
2021-08-19 11:10:24 -04:00
self . generate ( node , 1 )
2019-08-26 15:49:57 -04:00
self . log . info ( " Test that gettxoutsetinfo() output is consistent with or without coinstatsindex option " )
res0 = node . gettxoutsetinfo ( ' none ' )
# The fields 'disk_size' and 'transactions' do not exist on the index
del res0 [ ' disk_size ' ] , res0 [ ' transactions ' ]
for hash_option in index_hash_options :
res1 = index_node . gettxoutsetinfo ( hash_option )
2020-08-22 14:21:20 -04:00
# The fields 'block_info' and 'total_unspendable_amount' only exist on the index
del res1 [ ' block_info ' ] , res1 [ ' total_unspendable_amount ' ]
2019-08-26 15:49:57 -04:00
res1 . pop ( ' muhash ' , None )
# Everything left should be the same
assert_equal ( res1 , res0 )
self . log . info ( " Test that gettxoutsetinfo() can get fetch data on specific heights with index " )
# Generate a new tip
2021-08-19 11:10:24 -04:00
self . generate ( node , 5 )
2019-08-26 15:49:57 -04:00
for hash_option in index_hash_options :
# Fetch old stats by height
res2 = index_node . gettxoutsetinfo ( hash_option , 102 )
2020-08-22 14:21:20 -04:00
del res2 [ ' block_info ' ] , res2 [ ' total_unspendable_amount ' ]
2019-08-26 15:49:57 -04:00
res2 . pop ( ' muhash ' , None )
assert_equal ( res0 , res2 )
# Fetch old stats by hash
res3 = index_node . gettxoutsetinfo ( hash_option , res0 [ ' bestblock ' ] )
2020-08-22 14:21:20 -04:00
del res3 [ ' block_info ' ] , res3 [ ' total_unspendable_amount ' ]
2019-08-26 15:49:57 -04:00
res3 . pop ( ' muhash ' , None )
assert_equal ( res0 , res3 )
# It does not work without coinstatsindex
assert_raises_rpc_error ( - 8 , " Querying specific block heights requires coinstatsindex " , node . gettxoutsetinfo , hash_option , 102 )
2020-08-22 12:51:45 -04:00
self . log . info ( " Test gettxoutsetinfo() with index and verbose flag " )
for hash_option in index_hash_options :
# Genesis block is unspendable
res4 = index_node . gettxoutsetinfo ( hash_option , 0 )
assert_equal ( res4 [ ' total_unspendable_amount ' ] , 50 )
assert_equal ( res4 [ ' block_info ' ] , {
' unspendable ' : 50 ,
' prevout_spent ' : 0 ,
' new_outputs_ex_coinbase ' : 0 ,
' coinbase ' : 0 ,
' unspendables ' : {
' genesis_block ' : 50 ,
' bip30 ' : 0 ,
' scripts ' : 0 ,
' unclaimed_rewards ' : 0
}
} )
self . block_sanity_check ( res4 [ ' block_info ' ] )
# Test an older block height that included a normal tx
res5 = index_node . gettxoutsetinfo ( hash_option , 102 )
assert_equal ( res5 [ ' total_unspendable_amount ' ] , 50 )
assert_equal ( res5 [ ' block_info ' ] , {
' unspendable ' : 0 ,
' prevout_spent ' : 50 ,
2022-03-17 15:54:45 -03:00
' new_outputs_ex_coinbase ' : Decimal ( ' 49.99968800 ' ) ,
' coinbase ' : Decimal ( ' 50.00031200 ' ) ,
2020-08-22 12:51:45 -04:00
' unspendables ' : {
' genesis_block ' : 0 ,
' bip30 ' : 0 ,
' scripts ' : 0 ,
2022-03-17 15:54:45 -03:00
' unclaimed_rewards ' : 0 ,
2020-08-22 12:51:45 -04:00
}
} )
self . block_sanity_check ( res5 [ ' block_info ' ] )
# Generate and send a normal tx with two outputs
2022-03-17 15:54:45 -03:00
tx1_txid , tx1_vout = self . wallet . send_to (
from_node = node ,
scriptPubKey = self . wallet . get_scriptPubKey ( ) ,
amount = 21 * COIN ,
)
2020-08-22 12:51:45 -04:00
# Find the right position of the 21 BTC output
2022-03-17 15:54:45 -03:00
tx1_out_21 = self . wallet . get_utxo ( txid = tx1_txid , vout = tx1_vout )
2020-08-22 12:51:45 -04:00
# Generate and send another tx with an OP_RETURN output (which is unspendable)
2022-03-17 15:54:45 -03:00
tx2 = self . wallet . create_self_transfer ( utxo_to_spend = tx1_out_21 ) [ ' tx ' ]
tx2 . vout = [ CTxOut ( int ( Decimal ( ' 20.99 ' ) * COIN ) , CScript ( [ OP_RETURN ] + [ OP_FALSE ] * 30 ) ) ]
tx2_hex = tx2 . serialize ( ) . hex ( )
2020-08-22 12:51:45 -04:00
self . nodes [ 0 ] . sendrawtransaction ( tx2_hex )
# Include both txs in a block
2021-08-19 11:10:24 -04:00
self . generate ( self . nodes [ 0 ] , 1 )
2020-08-22 12:51:45 -04:00
for hash_option in index_hash_options :
# Check all amounts were registered correctly
res6 = index_node . gettxoutsetinfo ( hash_option , 108 )
2021-09-19 15:01:45 -03:00
assert_equal ( res6 [ ' total_unspendable_amount ' ] , Decimal ( ' 70.99000000 ' ) )
2020-08-22 12:51:45 -04:00
assert_equal ( res6 [ ' block_info ' ] , {
2021-09-19 15:01:45 -03:00
' unspendable ' : Decimal ( ' 20.99000000 ' ) ,
2022-03-17 15:54:45 -03:00
' prevout_spent ' : 71 ,
' new_outputs_ex_coinbase ' : Decimal ( ' 49.99999000 ' ) ,
' coinbase ' : Decimal ( ' 50.01001000 ' ) ,
2020-08-22 12:51:45 -04:00
' unspendables ' : {
' genesis_block ' : 0 ,
' bip30 ' : 0 ,
2021-09-19 15:01:45 -03:00
' scripts ' : Decimal ( ' 20.99000000 ' ) ,
2022-03-17 15:54:45 -03:00
' unclaimed_rewards ' : 0 ,
2020-08-22 12:51:45 -04:00
}
} )
self . block_sanity_check ( res6 [ ' block_info ' ] )
# Create a coinbase that does not claim full subsidy and also
# has two outputs
cb = create_coinbase ( 109 , nValue = 35 )
cb . vout . append ( CTxOut ( 5 * COIN , CScript ( [ OP_FALSE ] ) ) )
cb . rehash ( )
# Generate a block that includes previous coinbase
tip = self . nodes [ 0 ] . getbestblockhash ( )
block_time = self . nodes [ 0 ] . getblock ( tip ) [ ' time ' ] + 1
block = create_block ( int ( tip , 16 ) , cb , block_time )
block . solve ( )
2021-06-15 18:32:18 -04:00
self . nodes [ 0 ] . submitblock ( block . serialize ( ) . hex ( ) )
2020-08-22 12:51:45 -04:00
self . sync_all ( )
for hash_option in index_hash_options :
res7 = index_node . gettxoutsetinfo ( hash_option , 109 )
2021-09-19 15:01:45 -03:00
assert_equal ( res7 [ ' total_unspendable_amount ' ] , Decimal ( ' 80.99000000 ' ) )
2020-08-22 12:51:45 -04:00
assert_equal ( res7 [ ' block_info ' ] , {
' unspendable ' : 10 ,
' prevout_spent ' : 0 ,
' new_outputs_ex_coinbase ' : 0 ,
' coinbase ' : 40 ,
' unspendables ' : {
' genesis_block ' : 0 ,
' bip30 ' : 0 ,
' scripts ' : 0 ,
' unclaimed_rewards ' : 10
}
} )
self . block_sanity_check ( res7 [ ' block_info ' ] )
2021-02-25 15:14:58 -03:00
self . log . info ( " Test that the index is robust across restarts " )
res8 = index_node . gettxoutsetinfo ( ' muhash ' )
self . restart_node ( 1 , extra_args = self . extra_args [ 1 ] )
res9 = index_node . gettxoutsetinfo ( ' muhash ' )
assert_equal ( res8 , res9 )
2020-11-10 14:02:31 -03:00
self . generate ( index_node , 1 , sync_fun = self . no_op )
2021-02-25 15:14:58 -03:00
res10 = index_node . gettxoutsetinfo ( ' muhash ' )
2022-10-04 10:18:42 -03:00
assert res8 [ ' txouts ' ] < res10 [ ' txouts ' ]
2021-02-25 15:14:58 -03:00
2022-04-06 09:28:46 -04:00
self . log . info ( " Test that the index works with -reindex " )
self . restart_node ( 1 , extra_args = [ " -coinstatsindex " , " -reindex " ] )
res11 = index_node . gettxoutsetinfo ( ' muhash ' )
assert_equal ( res11 , res10 )
self . log . info ( " Test that -reindex-chainstate is disallowed with coinstatsindex " )
2022-04-29 15:35:05 -04:00
self . stop_node ( 1 )
2022-04-06 09:28:46 -04:00
self . nodes [ 1 ] . assert_start_raises_init_error (
expected_msg = ' Error: -reindex-chainstate option is not compatible with -coinstatsindex. '
' Please temporarily disable coinstatsindex while using -reindex-chainstate, or replace -reindex-chainstate with -reindex to fully rebuild all indexes. ' ,
extra_args = [ ' -coinstatsindex ' , ' -reindex-chainstate ' ] ,
)
2022-04-29 15:35:05 -04:00
self . restart_node ( 1 , extra_args = [ " -coinstatsindex " ] )
2022-04-06 09:28:46 -04:00
2021-02-28 15:27:00 -03:00
def _test_use_index_option ( self ) :
self . log . info ( " Test use_index option for nodes running the index " )
self . connect_nodes ( 0 , 1 )
self . nodes [ 0 ] . waitforblockheight ( 110 )
res = self . nodes [ 0 ] . gettxoutsetinfo ( ' muhash ' )
option_res = self . nodes [ 1 ] . gettxoutsetinfo ( hash_type = ' muhash ' , hash_or_height = None , use_index = False )
del res [ ' disk_size ' ] , option_res [ ' disk_size ' ]
assert_equal ( res , option_res )
2021-03-03 20:31:12 -03:00
def _test_reorg_index ( self ) :
self . log . info ( " Test that index can handle reorgs " )
# Generate two block, let the index catch up, then invalidate the blocks
index_node = self . nodes [ 1 ]
2022-03-17 15:54:45 -03:00
reorg_blocks = self . generatetoaddress ( index_node , 2 , getnewdestination ( ) [ 2 ] )
2021-03-03 20:31:12 -03:00
reorg_block = reorg_blocks [ 1 ]
res_invalid = index_node . gettxoutsetinfo ( ' muhash ' )
index_node . invalidateblock ( reorg_blocks [ 0 ] )
assert_equal ( index_node . gettxoutsetinfo ( ' muhash ' ) [ ' height ' ] , 110 )
# Add two new blocks
2020-11-10 14:02:31 -03:00
block = self . generate ( index_node , 2 , sync_fun = self . no_op ) [ 1 ]
2021-03-03 20:31:12 -03:00
res = index_node . gettxoutsetinfo ( hash_type = ' muhash ' , hash_or_height = None , use_index = False )
# Test that the result of the reorged block is not returned for its old block height
res2 = index_node . gettxoutsetinfo ( hash_type = ' muhash ' , hash_or_height = 112 )
assert_equal ( res [ " bestblock " ] , block )
assert_equal ( res [ " muhash " ] , res2 [ " muhash " ] )
2022-10-04 10:18:42 -03:00
assert res [ " muhash " ] != res_invalid [ " muhash " ]
2021-03-03 20:31:12 -03:00
# Test that requesting reorged out block by hash is still returning correct results
res_invalid2 = index_node . gettxoutsetinfo ( hash_type = ' muhash ' , hash_or_height = reorg_block )
assert_equal ( res_invalid2 [ " muhash " ] , res_invalid [ " muhash " ] )
2022-10-04 10:18:42 -03:00
assert res [ " muhash " ] != res_invalid2 [ " muhash " ]
2021-03-03 20:31:12 -03:00
# Add another block, so we don't depend on reconsiderblock remembering which
# blocks were touched by invalidateblock
2021-08-19 11:10:24 -04:00
self . generate ( index_node , 1 )
2021-03-03 20:31:12 -03:00
# Ensure that removing and re-adding blocks yields consistent results
block = index_node . getblockhash ( 99 )
index_node . invalidateblock ( block )
index_node . reconsiderblock ( block )
res3 = index_node . gettxoutsetinfo ( hash_type = ' muhash ' , hash_or_height = 112 )
assert_equal ( res2 , res3 )
2021-03-03 21:37:50 -03:00
def _test_index_rejects_hash_serialized ( self ) :
self . log . info ( " Test that the rpc raises if the legacy hash is passed with the index " )
msg = " hash_serialized_2 hash type cannot be queried for a specific block "
assert_raises_rpc_error ( - 8 , msg , self . nodes [ 1 ] . gettxoutsetinfo , hash_type = ' hash_serialized_2 ' , hash_or_height = 111 )
for use_index in { True , False , None } :
assert_raises_rpc_error ( - 8 , msg , self . nodes [ 1 ] . gettxoutsetinfo , hash_type = ' hash_serialized_2 ' , hash_or_height = 111 , use_index = use_index )
2021-02-25 15:14:58 -03:00
2019-08-26 15:49:57 -04:00
if __name__ == ' __main__ ' :
CoinStatsIndexTest ( ) . main ( )