2016-03-19 20:58:06 +01:00
#!/usr/bin/env python3
2021-07-28 13:57:16 +02:00
# Copyright (c) 2015-2021 The Bitcoin Core developers
2015-11-18 20:55:52 -05:00
# Distributed under the MIT software license, see the accompanying
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
2017-01-17 18:34:40 -05:00
""" Test the prioritisetransaction mining RPC. """
2015-11-18 20:55:52 -05:00
2022-03-21 11:43:49 +01:00
from decimal import Decimal
2018-07-07 00:10:35 +02:00
import time
2022-04-12 21:53:02 +02:00
from test_framework . messages import (
COIN ,
MAX_BLOCK_WEIGHT ,
)
2015-11-18 20:55:52 -05:00
from test_framework . test_framework import BitcoinTestFramework
2022-04-12 21:53:02 +02:00
from test_framework . util import (
assert_equal ,
assert_raises_rpc_error ,
create_lots_of_big_transactions ,
gen_return_txouts ,
)
2022-03-21 11:43:49 +01:00
from test_framework . wallet import MiniWallet
2015-11-18 20:55:52 -05:00
class PrioritiseTransactionTest ( BitcoinTestFramework ) :
2017-06-09 18:21:21 -04:00
def set_test_params ( self ) :
2021-10-19 11:13:40 +02:00
self . num_nodes = 1
2019-04-24 17:55:58 -04:00
self . extra_args = [ [
" -printpriority=1 " ,
2022-06-29 17:51:39 +02:00
" -datacarriersize=100000 " ,
2019-04-24 17:55:58 -04:00
] ] * self . num_nodes
2019-12-06 14:37:49 +00:00
self . supports_cli = False
2015-11-18 20:55:52 -05:00
2022-03-21 11:43:49 +01:00
def test_diamond ( self ) :
self . log . info ( " Test diamond-shape package with priority " )
mock_time = int ( time . time ( ) )
self . nodes [ 0 ] . setmocktime ( mock_time )
# tx_a
# / \
# / \
# tx_b tx_c
# \ /
# \ /
# tx_d
tx_o_a = self . wallet . send_self_transfer_multi (
from_node = self . nodes [ 0 ] ,
num_outputs = 2 ,
)
txid_a = tx_o_a [ " txid " ]
tx_o_b , tx_o_c = [ self . wallet . send_self_transfer (
from_node = self . nodes [ 0 ] ,
utxo_to_spend = u ,
) for u in tx_o_a [ " new_utxos " ] ]
txid_b = tx_o_b [ " txid " ]
txid_c = tx_o_c [ " txid " ]
tx_o_d = self . wallet . send_self_transfer_multi (
from_node = self . nodes [ 0 ] ,
utxos_to_spend = [
self . wallet . get_utxo ( txid = txid_b ) ,
self . wallet . get_utxo ( txid = txid_c ) ,
] ,
)
txid_d = tx_o_d [ " txid " ]
self . log . info ( " Test priority while txs are in mempool " )
raw_before = self . nodes [ 0 ] . getrawmempool ( verbose = True )
fee_delta_b = Decimal ( 9999 ) / COIN
fee_delta_c_1 = Decimal ( - 1234 ) / COIN
fee_delta_c_2 = Decimal ( 8888 ) / COIN
self . nodes [ 0 ] . prioritisetransaction ( txid = txid_b , fee_delta = int ( fee_delta_b * COIN ) )
self . nodes [ 0 ] . prioritisetransaction ( txid = txid_c , fee_delta = int ( fee_delta_c_1 * COIN ) )
self . nodes [ 0 ] . prioritisetransaction ( txid = txid_c , fee_delta = int ( fee_delta_c_2 * COIN ) )
raw_before [ txid_a ] [ " fees " ] [ " descendant " ] + = fee_delta_b + fee_delta_c_1 + fee_delta_c_2
raw_before [ txid_b ] [ " fees " ] [ " modified " ] + = fee_delta_b
raw_before [ txid_b ] [ " fees " ] [ " ancestor " ] + = fee_delta_b
raw_before [ txid_b ] [ " fees " ] [ " descendant " ] + = fee_delta_b
raw_before [ txid_c ] [ " fees " ] [ " modified " ] + = fee_delta_c_1 + fee_delta_c_2
raw_before [ txid_c ] [ " fees " ] [ " ancestor " ] + = fee_delta_c_1 + fee_delta_c_2
raw_before [ txid_c ] [ " fees " ] [ " descendant " ] + = fee_delta_c_1 + fee_delta_c_2
raw_before [ txid_d ] [ " fees " ] [ " ancestor " ] + = fee_delta_b + fee_delta_c_1 + fee_delta_c_2
raw_after = self . nodes [ 0 ] . getrawmempool ( verbose = True )
assert_equal ( raw_before [ txid_a ] , raw_after [ txid_a ] )
assert_equal ( raw_before , raw_after )
self . log . info ( " Test priority while txs are not in mempool " )
self . restart_node ( 0 , extra_args = [ " -nopersistmempool " ] )
self . nodes [ 0 ] . setmocktime ( mock_time )
assert_equal ( self . nodes [ 0 ] . getmempoolinfo ( ) [ " size " ] , 0 )
self . nodes [ 0 ] . prioritisetransaction ( txid = txid_b , fee_delta = int ( fee_delta_b * COIN ) )
self . nodes [ 0 ] . prioritisetransaction ( txid = txid_c , fee_delta = int ( fee_delta_c_1 * COIN ) )
self . nodes [ 0 ] . prioritisetransaction ( txid = txid_c , fee_delta = int ( fee_delta_c_2 * COIN ) )
for t in [ tx_o_a [ " hex " ] , tx_o_b [ " hex " ] , tx_o_c [ " hex " ] , tx_o_d [ " hex " ] ] :
self . nodes [ 0 ] . sendrawtransaction ( t )
raw_after = self . nodes [ 0 ] . getrawmempool ( verbose = True )
assert_equal ( raw_before [ txid_a ] , raw_after [ txid_a ] )
assert_equal ( raw_before , raw_after )
# Clear mempool
self . generate ( self . nodes [ 0 ] , 1 )
# Use default extra_args
self . restart_node ( 0 )
2017-04-03 09:34:04 -04:00
def run_test ( self ) :
2022-03-21 11:43:49 +01:00
self . wallet = MiniWallet ( self . nodes [ 0 ] )
2022-04-12 21:53:02 +02:00
self . wallet . rescan_utxos ( )
2022-03-21 11:43:49 +01:00
2018-01-03 01:48:20 +00:00
# Test `prioritisetransaction` required parameters
assert_raises_rpc_error ( - 1 , " prioritisetransaction " , self . nodes [ 0 ] . prioritisetransaction )
assert_raises_rpc_error ( - 1 , " prioritisetransaction " , self . nodes [ 0 ] . prioritisetransaction , ' ' )
assert_raises_rpc_error ( - 1 , " prioritisetransaction " , self . nodes [ 0 ] . prioritisetransaction , ' ' , 0 )
# Test `prioritisetransaction` invalid extra parameters
assert_raises_rpc_error ( - 1 , " prioritisetransaction " , self . nodes [ 0 ] . prioritisetransaction , ' ' , 0 , 0 , 0 )
# Test `prioritisetransaction` invalid `txid`
2018-06-08 11:16:07 -07:00
assert_raises_rpc_error ( - 8 , " txid must be of length 64 (not 3, for ' foo ' ) " , self . nodes [ 0 ] . prioritisetransaction , txid = ' foo ' , fee_delta = 0 )
assert_raises_rpc_error ( - 8 , " txid must be hexadecimal string (not ' Zd1d4e24ed99057e84c3f80fd8fbec79ed9e1acee37da269356ecea000000000 ' ) " , self . nodes [ 0 ] . prioritisetransaction , txid = ' Zd1d4e24ed99057e84c3f80fd8fbec79ed9e1acee37da269356ecea000000000 ' , fee_delta = 0 )
2018-01-03 01:48:20 +00:00
# Test `prioritisetransaction` invalid `dummy`
txid = ' 1d1d4e24ed99057e84c3f80fd8fbec79ed9e1acee37da269356ecea000000000 '
2022-07-18 11:13:22 +02:00
assert_raises_rpc_error ( - 1 , " JSON value of type string is not of expected type number " , self . nodes [ 0 ] . prioritisetransaction , txid , ' foo ' , 0 )
2018-01-03 01:48:20 +00:00
assert_raises_rpc_error ( - 8 , " Priority is no longer supported, dummy argument to prioritisetransaction must be 0. " , self . nodes [ 0 ] . prioritisetransaction , txid , 1 , 0 )
# Test `prioritisetransaction` invalid `fee_delta`
2022-07-18 11:13:22 +02:00
assert_raises_rpc_error ( - 1 , " JSON value of type string is not of expected type number " , self . nodes [ 0 ] . prioritisetransaction , txid = txid , fee_delta = ' foo ' )
2018-01-03 01:48:20 +00:00
2022-03-21 11:43:49 +01:00
self . test_diamond ( )
2016-05-14 13:01:31 +02:00
self . txouts = gen_return_txouts ( )
2015-11-18 20:55:52 -05:00
self . relayfee = self . nodes [ 0 ] . getnetworkinfo ( ) [ ' relayfee ' ]
2016-03-01 09:28:16 -05:00
utxo_count = 90
2022-04-12 21:53:02 +02:00
utxos = self . wallet . send_self_transfer_multi ( from_node = self . nodes [ 0 ] , num_outputs = utxo_count ) [ ' new_utxos ' ]
self . generate ( self . wallet , 1 )
assert_equal ( len ( self . nodes [ 0 ] . getrawmempool ( ) ) , 0 )
2015-11-18 20:55:52 -05:00
base_fee = self . relayfee * 100 # our transactions are smaller than 100kb
txids = [ ]
# Create 3 batches of transactions at 3 different fee rate levels
2016-03-01 09:28:16 -05:00
range_size = utxo_count / / 3
2016-03-19 20:58:06 +01:00
for i in range ( 3 ) :
2015-11-18 20:55:52 -05:00
txids . append ( [ ] )
2016-03-01 09:28:16 -05:00
start_range = i * range_size
end_range = start_range + range_size
2022-04-12 21:53:02 +02:00
txids [ i ] = create_lots_of_big_transactions (
self . wallet ,
self . nodes [ 0 ] ,
( i + 1 ) * base_fee ,
end_range - start_range ,
self . txouts ,
utxos [ start_range : end_range ] )
2016-03-01 09:28:16 -05:00
# Make sure that the size of each group of transactions exceeds
2021-07-01 01:43:45 +02:00
# MAX_BLOCK_WEIGHT // 4 -- otherwise the test needs to be revised to
# create more transactions.
2016-03-01 09:28:16 -05:00
mempool = self . nodes [ 0 ] . getrawmempool ( True )
sizes = [ 0 , 0 , 0 ]
2016-03-19 20:58:06 +01:00
for i in range ( 3 ) :
2016-03-01 09:28:16 -05:00
for j in txids [ i ] :
2019-02-19 17:43:44 -05:00
assert j in mempool
2018-04-17 20:17:13 +02:00
sizes [ i ] + = mempool [ j ] [ ' vsize ' ]
2021-07-01 01:43:45 +02:00
assert sizes [ i ] > MAX_BLOCK_WEIGHT / / 4 # Fail => raise utxo_count
2015-11-18 20:55:52 -05:00
# add a fee delta to something in the cheapest bucket and make sure it gets mined
2017-01-19 22:46:50 -05:00
# also check that a different entry in the cheapest bucket is NOT mined
2017-04-21 16:41:01 +00:00
self . nodes [ 0 ] . prioritisetransaction ( txid = txids [ 0 ] [ 0 ] , fee_delta = int ( 3 * base_fee * COIN ) )
2015-11-18 20:55:52 -05:00
2021-08-19 17:10:24 +02:00
self . generate ( self . nodes [ 0 ] , 1 )
2015-11-18 20:55:52 -05:00
mempool = self . nodes [ 0 ] . getrawmempool ( )
2017-03-07 18:46:17 -05:00
self . log . info ( " Assert that prioritised transaction was mined " )
2019-02-19 17:43:44 -05:00
assert txids [ 0 ] [ 0 ] not in mempool
assert txids [ 0 ] [ 1 ] in mempool
2015-11-18 20:55:52 -05:00
high_fee_tx = None
for x in txids [ 2 ] :
if x not in mempool :
high_fee_tx = x
# Something high-fee should have been mined!
2019-02-19 17:43:44 -05:00
assert high_fee_tx is not None
2015-11-18 20:55:52 -05:00
# Add a prioritisation before a tx is in the mempool (de-prioritising a
2016-03-01 09:28:16 -05:00
# high-fee transaction so that it's now low fee).
2017-04-21 16:41:01 +00:00
self . nodes [ 0 ] . prioritisetransaction ( txid = high_fee_tx , fee_delta = - int ( 2 * base_fee * COIN ) )
2015-11-18 20:55:52 -05:00
# Add everything back to mempool
self . nodes [ 0 ] . invalidateblock ( self . nodes [ 0 ] . getbestblockhash ( ) )
# Check to make sure our high fee rate tx is back in the mempool
mempool = self . nodes [ 0 ] . getrawmempool ( )
2019-02-19 17:43:44 -05:00
assert high_fee_tx in mempool
2015-11-18 20:55:52 -05:00
2016-03-01 09:28:16 -05:00
# Now verify the modified-high feerate transaction isn't mined before
# the other high fee transactions. Keep mining until our mempool has
# decreased by all the high fee size that we calculated above.
while ( self . nodes [ 0 ] . getmempoolinfo ( ) [ ' bytes ' ] > sizes [ 0 ] + sizes [ 1 ] ) :
2020-11-10 18:02:31 +01:00
self . generate ( self . nodes [ 0 ] , 1 , sync_fun = self . no_op )
2015-11-18 20:55:52 -05:00
# High fee transaction should not have been mined, but other high fee rate
# transactions should have been.
mempool = self . nodes [ 0 ] . getrawmempool ( )
2017-03-07 18:46:17 -05:00
self . log . info ( " Assert that de-prioritised transaction is still in mempool " )
2019-02-19 17:43:44 -05:00
assert high_fee_tx in mempool
2015-11-18 20:55:52 -05:00
for x in txids [ 2 ] :
if ( x != high_fee_tx ) :
2019-02-19 17:43:44 -05:00
assert x not in mempool
2015-11-18 20:55:52 -05:00
2017-01-19 22:46:50 -05:00
# Create a free transaction. Should be rejected.
2022-06-21 11:25:54 +02:00
tx_res = self . wallet . create_self_transfer ( fee_rate = 0 )
2022-04-12 21:53:02 +02:00
tx_hex = tx_res [ ' hex ' ]
tx_id = tx_res [ ' txid ' ]
2015-12-02 09:37:18 -05:00
2017-03-07 14:08:59 -05:00
# This will raise an exception due to min relay fee not being met
2018-04-26 00:30:38 +03:00
assert_raises_rpc_error ( - 26 , " min relay fee not met " , self . nodes [ 0 ] . sendrawtransaction , tx_hex )
2019-02-19 17:43:44 -05:00
assert tx_id not in self . nodes [ 0 ] . getrawmempool ( )
2015-12-02 09:37:18 -05:00
# This is a less than 1000-byte transaction, so just set the fee
2018-03-18 16:26:45 +02:00
# to be the minimum for a 1000-byte transaction and check that it is
2015-12-02 09:37:18 -05:00
# accepted.
2017-04-21 16:41:01 +00:00
self . nodes [ 0 ] . prioritisetransaction ( txid = tx_id , fee_delta = int ( self . relayfee * COIN ) )
2015-12-02 09:37:18 -05:00
2017-03-07 18:46:17 -05:00
self . log . info ( " Assert that prioritised free transaction is accepted to mempool " )
2017-01-19 22:46:50 -05:00
assert_equal ( self . nodes [ 0 ] . sendrawtransaction ( tx_hex ) , tx_id )
2019-02-19 17:43:44 -05:00
assert tx_id in self . nodes [ 0 ] . getrawmempool ( )
2015-12-02 09:37:18 -05:00
2017-04-18 12:52:08 -04:00
# Test that calling prioritisetransaction is sufficient to trigger
# getblocktemplate to (eventually) return a new block.
mock_time = int ( time . time ( ) )
self . nodes [ 0 ] . setmocktime ( mock_time )
2018-11-26 11:17:38 -05:00
template = self . nodes [ 0 ] . getblocktemplate ( { ' rules ' : [ ' segwit ' ] } )
2017-04-21 16:41:01 +00:00
self . nodes [ 0 ] . prioritisetransaction ( txid = tx_id , fee_delta = - int ( self . relayfee * COIN ) )
2017-04-18 12:52:08 -04:00
self . nodes [ 0 ] . setmocktime ( mock_time + 10 )
2018-11-26 11:17:38 -05:00
new_template = self . nodes [ 0 ] . getblocktemplate ( { ' rules ' : [ ' segwit ' ] } )
2017-04-18 12:52:08 -04:00
2019-02-19 17:43:44 -05:00
assert template != new_template
2017-04-18 12:52:08 -04:00
2015-11-18 20:55:52 -05:00
if __name__ == ' __main__ ' :
PrioritiseTransactionTest ( ) . main ( )