2017-03-09 17:14:55 -05:00
#!/usr/bin/env python3
2022-12-24 23:49:50 +00:00
# Copyright (c) 2014-2022 The Bitcoin Core developers
2017-03-09 17:14:55 -05:00
# Distributed under the MIT software license, see the accompanying
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
""" Test mempool persistence.
By default , bitcoind will dump mempool on shutdown and
then reload it on startup . This can be overridden with
2017-05-05 09:17:39 -04:00
the - persistmempool = 0 command line option .
2017-03-09 17:14:55 -05:00
Test is as follows :
2017-05-05 09:17:39 -04:00
- start node0 , node1 and node2 . node1 has - persistmempool = 0
2017-03-09 17:14:55 -05:00
- create 5 transactions on node2 to its own address . Note that these
are not sent to node0 or node1 addresses because we don ' t want
them to be saved in the wallet .
- check that node0 and node1 have 5 transactions in their mempools
- shutdown all nodes .
- startup node0 . Verify that it still has 5 transactions
in its mempool . Shutdown node0 . This tests that by default the
mempool is persistent .
- startup node1 . Verify that its mempool is empty . Shutdown node1 .
2017-05-05 09:17:39 -04:00
This tests that with - persistmempool = 0 , the mempool is not
2017-03-09 17:14:55 -05:00
dumped to disk when the node is shut down .
2017-05-05 09:17:39 -04:00
- Restart node0 with - persistmempool = 0. Verify that its mempool is
empty . Shutdown node0 . This tests that with - persistmempool = 0 ,
2017-03-09 17:14:55 -05:00
the mempool is not loaded from disk on start up .
2017-05-05 09:17:39 -04:00
- Restart node0 with - persistmempool . Verify that it has 5
transactions in its mempool . This tests that - persistmempool = 0
2017-03-09 17:14:55 -05:00
does not overwrite a previously valid mempool stored on disk .
2017-08-21 13:23:18 +02:00
- Remove node0 mempool . dat and verify savemempool RPC recreates it
2018-03-18 16:26:45 +02:00
and verify that node1 can load it and has 5 transactions in its
2017-08-21 13:23:18 +02:00
mempool .
- Verify that savemempool throws when the RPC is called if
node1 can ' t write to disk.
2017-03-09 17:14:55 -05:00
"""
2018-07-07 00:10:35 +02:00
from decimal import Decimal
2017-08-21 13:23:18 +02:00
import os
2019-09-18 13:36:29 -04:00
import time
2017-03-09 17:14:55 -05:00
2020-07-19 14:47:05 +07:00
from test_framework . p2p import P2PTxInvStore
2020-08-17 10:45:44 +01:00
from test_framework . test_framework import BitcoinTestFramework
2019-09-18 13:36:29 -04:00
from test_framework . util import (
assert_equal ,
assert_greater_than_or_equal ,
assert_raises_rpc_error ,
)
2023-04-13 13:13:18 +02:00
from test_framework . wallet import MiniWallet , COIN
2017-03-09 17:14:55 -05:00
2018-12-11 20:51:13 -05:00
2017-03-09 17:14:55 -05:00
class MempoolPersistTest ( BitcoinTestFramework ) :
2022-11-09 12:53:13 +01:00
def add_options ( self , parser ) :
self . add_wallet_options ( parser , legacy = False )
2017-06-09 18:21:21 -04:00
def set_test_params ( self ) :
2017-03-09 17:14:55 -05:00
self . num_nodes = 3
2017-05-05 09:17:39 -04:00
self . extra_args = [ [ ] , [ " -persistmempool=0 " ] , [ ] ]
2017-03-09 17:14:55 -05:00
def run_test ( self ) :
2021-09-20 15:25:49 +02:00
self . mini_wallet = MiniWallet ( self . nodes [ 2 ] )
if self . is_sqlite_compiled ( ) :
self . nodes [ 2 ] . createwallet (
wallet_name = " watch " ,
descriptors = True ,
disable_private_keys = True ,
load_on_startup = False ,
)
wallet_watch = self . nodes [ 2 ] . get_wallet_rpc ( " watch " )
assert_equal ( [ { ' success ' : True } ] , wallet_watch . importdescriptors ( [ { ' desc ' : self . mini_wallet . get_descriptor ( ) , ' timestamp ' : 0 } ] ) )
2017-03-09 17:14:55 -05:00
self . log . debug ( " Send 5 transactions from node2 (to its own address) " )
2019-09-18 13:36:29 -04:00
tx_creation_time_lower = int ( time . time ( ) )
2020-08-03 01:10:56 +02:00
for _ in range ( 5 ) :
2021-09-20 15:25:49 +02:00
last_txid = self . mini_wallet . send_self_transfer ( from_node = self . nodes [ 2 ] ) [ " txid " ]
if self . is_sqlite_compiled ( ) :
self . nodes [ 2 ] . syncwithvalidationinterfacequeue ( ) # Flush mempool to wallet
node2_balance = wallet_watch . getbalance ( )
2017-03-09 17:14:55 -05:00
self . sync_all ( )
2019-09-18 13:36:29 -04:00
tx_creation_time_higher = int ( time . time ( ) )
2017-03-09 17:14:55 -05:00
self . log . debug ( " Verify that node0 and node1 have 5 transactions in their mempools " )
assert_equal ( len ( self . nodes [ 0 ] . getrawmempool ( ) ) , 5 )
assert_equal ( len ( self . nodes [ 1 ] . getrawmempool ( ) ) , 5 )
2021-01-15 16:39:37 +01:00
total_fee_old = self . nodes [ 0 ] . getmempoolinfo ( ) [ ' total_fee ' ]
2018-12-11 20:51:13 -05:00
self . log . debug ( " Prioritize a transaction on node0 " )
fees = self . nodes [ 0 ] . getmempoolentry ( txid = last_txid ) [ ' fees ' ]
assert_equal ( fees [ ' base ' ] , fees [ ' modified ' ] )
self . nodes [ 0 ] . prioritisetransaction ( txid = last_txid , fee_delta = 1000 )
fees = self . nodes [ 0 ] . getmempoolentry ( txid = last_txid ) [ ' fees ' ]
assert_equal ( fees [ ' base ' ] + Decimal ( ' 0.00001000 ' ) , fees [ ' modified ' ] )
2021-01-15 16:39:37 +01:00
self . log . info ( ' Check the total base fee is unchanged after prioritisetransaction ' )
assert_equal ( total_fee_old , self . nodes [ 0 ] . getmempoolinfo ( ) [ ' total_fee ' ] )
assert_equal ( total_fee_old , sum ( v [ ' fees ' ] [ ' base ' ] for k , v in self . nodes [ 0 ] . getrawmempool ( verbose = True ) . items ( ) ) )
2021-09-20 15:48:09 +02:00
last_entry = self . nodes [ 0 ] . getmempoolentry ( txid = last_txid )
tx_creation_time = last_entry [ ' time ' ]
2019-09-18 13:36:29 -04:00
assert_greater_than_or_equal ( tx_creation_time , tx_creation_time_lower )
assert_greater_than_or_equal ( tx_creation_time_higher , tx_creation_time )
2020-03-17 10:39:25 -07:00
# disconnect nodes & make a txn that remains in the unbroadcast set.
2020-09-17 00:46:07 -07:00
self . disconnect_nodes ( 0 , 1 )
2021-09-20 15:29:20 +02:00
assert_equal ( len ( self . nodes [ 0 ] . getpeerinfo ( ) ) , 0 )
assert_equal ( len ( self . nodes [ 0 ] . p2ps ) , 0 )
2021-09-20 15:25:49 +02:00
self . mini_wallet . send_self_transfer ( from_node = self . nodes [ 0 ] )
2020-03-17 10:39:25 -07:00
2022-07-11 16:12:18 +01:00
# Test persistence of prioritisation for transactions not in the mempool.
# Create a tx and prioritise but don't submit until after the restart.
tx_prioritised_not_submitted = self . mini_wallet . create_self_transfer ( )
self . nodes [ 0 ] . prioritisetransaction ( txid = tx_prioritised_not_submitted [ ' txid ' ] , fee_delta = 9999 )
2017-12-06 12:08:40 -05:00
self . log . debug ( " Stop-start the nodes. Verify that node0 has the transactions in its mempool and node1 does not. Verify that node2 calculates its balance correctly after loading wallet transactions. " )
2017-03-23 23:56:31 -04:00
self . stop_nodes ( )
2018-03-06 18:43:50 -05:00
# Give this node a head-start, so we can be "extra-sure" that it didn't load anything later
# Also don't store the mempool, to keep the datadir clean
self . start_node ( 1 , extra_args = [ " -persistmempool=0 " ] )
2017-06-09 16:35:17 -04:00
self . start_node ( 0 )
2017-12-06 12:08:40 -05:00
self . start_node ( 2 )
2020-05-04 20:06:38 -04:00
assert self . nodes [ 0 ] . getmempoolinfo ( ) [ " loaded " ] # start_node is blocking on the mempool being loaded
assert self . nodes [ 2 ] . getmempoolinfo ( ) [ " loaded " ]
2020-03-17 10:39:25 -07:00
assert_equal ( len ( self . nodes [ 0 ] . getrawmempool ( ) ) , 6 )
2019-02-01 14:09:36 -08:00
assert_equal ( len ( self . nodes [ 2 ] . getrawmempool ( ) ) , 5 )
2018-01-18 10:16:34 -05:00
# The others have loaded their mempool. If node_1 loaded anything, we'd probably notice by now:
2017-03-09 17:14:55 -05:00
assert_equal ( len ( self . nodes [ 1 ] . getrawmempool ( ) ) , 0 )
2018-12-11 20:51:13 -05:00
self . log . debug ( ' Verify prioritization is loaded correctly ' )
fees = self . nodes [ 0 ] . getmempoolentry ( txid = last_txid ) [ ' fees ' ]
assert_equal ( fees [ ' base ' ] + Decimal ( ' 0.00001000 ' ) , fees [ ' modified ' ] )
2021-09-20 15:48:09 +02:00
self . log . debug ( ' Verify all fields are loaded correctly ' )
assert_equal ( last_entry , self . nodes [ 0 ] . getmempoolentry ( txid = last_txid ) )
2022-07-11 16:12:18 +01:00
self . nodes [ 0 ] . sendrawtransaction ( tx_prioritised_not_submitted [ ' hex ' ] )
entry_prioritised_before_restart = self . nodes [ 0 ] . getmempoolentry ( txid = tx_prioritised_not_submitted [ ' txid ' ] )
assert_equal ( entry_prioritised_before_restart [ ' fees ' ] [ ' base ' ] + Decimal ( ' 0.00009999 ' ) , entry_prioritised_before_restart [ ' fees ' ] [ ' modified ' ] )
2019-09-18 13:36:29 -04:00
2017-12-06 12:08:40 -05:00
# Verify accounting of mempool transactions after restart is correct
2021-09-20 15:25:49 +02:00
if self . is_sqlite_compiled ( ) :
self . nodes [ 2 ] . loadwallet ( " watch " )
wallet_watch = self . nodes [ 2 ] . get_wallet_rpc ( " watch " )
self . nodes [ 2 ] . syncwithvalidationinterfacequeue ( ) # Flush mempool to wallet
assert_equal ( node2_balance , wallet_watch . getbalance ( ) )
2017-12-06 12:08:40 -05:00
scripted-diff: Use wallets_path and chain_path where possible
Instead of passing the datadir and chain name to os.path.join, just use
the existing properties, which are the same.
-BEGIN VERIFY SCRIPT-
sed -i --regexp-extended 's|\.datadir, self\.chain, .wallets.|.wallets_path|g' $(git grep -l '\.datadir, self\.chain,')
sed -i --regexp-extended 's|\.datadir, self\.chain,|.chain_path,|g' $(git grep -l '\.datadir, self\.chain,')
-END VERIFY SCRIPT-
2023-06-20 11:22:03 +02:00
mempooldat0 = os . path . join ( self . nodes [ 0 ] . chain_path , ' mempool.dat ' )
mempooldat1 = os . path . join ( self . nodes [ 1 ] . chain_path , ' mempool.dat ' )
2022-07-07 19:30:30 -04:00
self . log . debug ( " Force -persistmempool=0 node1 to savemempool to disk via RPC " )
assert not os . path . exists ( mempooldat1 )
result1 = self . nodes [ 1 ] . savemempool ( )
assert os . path . isfile ( mempooldat1 )
assert_equal ( result1 [ ' filename ' ] , mempooldat1 )
os . remove ( mempooldat1 )
2017-05-05 09:17:39 -04:00
self . log . debug ( " Stop-start node0 with -persistmempool=0. Verify that it doesn ' t load its mempool.dat file. " )
2017-03-23 23:56:31 -04:00
self . stop_nodes ( )
2021-09-20 15:25:49 +02:00
self . start_node ( 0 , extra_args = [ " -persistmempool=0 " ] )
2020-05-04 20:06:38 -04:00
assert self . nodes [ 0 ] . getmempoolinfo ( ) [ " loaded " ]
2017-03-09 17:14:55 -05:00
assert_equal ( len ( self . nodes [ 0 ] . getrawmempool ( ) ) , 0 )
2023-04-13 13:13:18 +02:00
self . log . debug ( " Import mempool at runtime to node0. " )
assert_equal ( { } , self . nodes [ 0 ] . importmempool ( mempooldat0 ) )
assert_equal ( len ( self . nodes [ 0 ] . getrawmempool ( ) ) , 7 )
fees = self . nodes [ 0 ] . getmempoolentry ( txid = last_txid ) [ " fees " ]
assert_equal ( fees [ " base " ] , fees [ " modified " ] )
assert_equal ( { } , self . nodes [ 0 ] . importmempool ( mempooldat0 , { " apply_fee_delta_priority " : True , " apply_unbroadcast_set " : True } ) )
assert_equal ( 2 , self . nodes [ 0 ] . getmempoolinfo ( ) [ " unbroadcastcount " ] )
fees = self . nodes [ 0 ] . getmempoolentry ( txid = last_txid ) [ " fees " ]
assert_equal ( fees [ " base " ] + Decimal ( " 0.00001000 " ) , fees [ " modified " ] )
2017-03-09 17:14:55 -05:00
self . log . debug ( " Stop-start node0. Verify that it has the transactions in its mempool. " )
2017-03-23 23:56:31 -04:00
self . stop_nodes ( )
2017-06-09 16:35:17 -04:00
self . start_node ( 0 )
2020-05-04 20:06:38 -04:00
assert self . nodes [ 0 ] . getmempoolinfo ( ) [ " loaded " ]
2022-07-11 16:12:18 +01:00
assert_equal ( len ( self . nodes [ 0 ] . getrawmempool ( ) ) , 7 )
2017-03-09 17:14:55 -05:00
2017-08-21 13:23:18 +02:00
self . log . debug ( " Remove the mempool.dat file. Verify that savemempool to disk via RPC re-creates it " )
os . remove ( mempooldat0 )
2021-10-30 15:20:35 -03:00
result0 = self . nodes [ 0 ] . savemempool ( )
2017-08-21 13:23:18 +02:00
assert os . path . isfile ( mempooldat0 )
2021-10-30 15:20:35 -03:00
assert_equal ( result0 [ ' filename ' ] , mempooldat0 )
2017-08-21 13:23:18 +02:00
2022-07-11 16:12:18 +01:00
self . log . debug ( " Stop nodes, make node1 use mempool.dat from node0. Verify it has 7 transactions " )
2017-08-21 13:23:18 +02:00
os . rename ( mempooldat0 , mempooldat1 )
self . stop_nodes ( )
2021-09-22 11:32:25 +02:00
self . start_node ( 1 , extra_args = [ " -persistmempool " ] )
2020-05-04 20:06:38 -04:00
assert self . nodes [ 1 ] . getmempoolinfo ( ) [ " loaded " ]
2022-07-11 16:12:18 +01:00
assert_equal ( len ( self . nodes [ 1 ] . getrawmempool ( ) ) , 7 )
2017-08-21 13:23:18 +02:00
self . log . debug ( " Prevent bitcoind from writing mempool.dat to disk. Verify that `savemempool` fails " )
2018-04-08 12:46:12 -04:00
# to test the exception we are creating a tmp folder called mempool.dat.new
2017-08-21 13:23:18 +02:00
# which is an implementation detail that could change and break this test
mempooldotnew1 = mempooldat1 + ' .new '
2018-04-08 12:46:12 -04:00
os . mkdir ( mempooldotnew1 )
2017-07-12 10:33:46 -04:00
assert_raises_rpc_error ( - 1 , " Unable to dump mempool to disk " , self . nodes [ 1 ] . savemempool )
2018-04-08 12:46:12 -04:00
os . rmdir ( mempooldotnew1 )
2023-04-13 13:13:18 +02:00
self . test_importmempool_union ( )
2020-03-17 10:39:25 -07:00
self . test_persist_unbroadcast ( )
def test_persist_unbroadcast ( self ) :
node0 = self . nodes [ 0 ]
self . start_node ( 0 )
2023-02-28 09:58:06 -03:00
self . start_node ( 2 )
2020-03-17 10:39:25 -07:00
# clear out mempool
2020-11-10 18:02:31 +01:00
self . generate ( node0 , 1 , sync_fun = self . no_op )
2020-03-17 10:39:25 -07:00
2020-05-01 14:03:51 -07:00
# ensure node0 doesn't have any connections
# make a transaction that will remain in the unbroadcast set
2021-09-20 15:29:20 +02:00
assert_equal ( len ( node0 . getpeerinfo ( ) ) , 0 )
assert_equal ( len ( node0 . p2ps ) , 0 )
2021-09-20 15:25:49 +02:00
self . mini_wallet . send_self_transfer ( from_node = node0 )
2020-03-17 10:39:25 -07:00
# shutdown, then startup with wallet disabled
2021-09-20 15:25:49 +02:00
self . restart_node ( 0 , extra_args = [ " -disablewallet " ] )
2020-03-17 10:39:25 -07:00
# check that txn gets broadcast due to unbroadcast logic
conn = node0 . add_p2p_connection ( P2PTxInvStore ( ) )
2021-09-20 15:29:20 +02:00
node0 . mockscheduler ( 16 * 60 ) # 15 min + 1 for buffer
2020-08-17 17:50:47 +02:00
self . wait_until ( lambda : len ( conn . get_invs ( ) ) == 1 )
2017-08-21 13:23:18 +02:00
2023-04-13 13:13:18 +02:00
def test_importmempool_union ( self ) :
self . log . debug ( " Submit different transactions to node0 and node1 ' s mempools " )
self . start_node ( 0 )
self . start_node ( 2 )
tx_node0 = self . mini_wallet . send_self_transfer ( from_node = self . nodes [ 0 ] )
tx_node1 = self . mini_wallet . send_self_transfer ( from_node = self . nodes [ 1 ] )
tx_node01 = self . mini_wallet . create_self_transfer ( )
tx_node01_secret = self . mini_wallet . create_self_transfer ( )
self . nodes [ 0 ] . prioritisetransaction ( tx_node01 [ " txid " ] , 0 , COIN )
self . nodes [ 0 ] . prioritisetransaction ( tx_node01_secret [ " txid " ] , 0 , 2 * COIN )
self . nodes [ 1 ] . prioritisetransaction ( tx_node01_secret [ " txid " ] , 0 , 3 * COIN )
self . nodes [ 0 ] . sendrawtransaction ( tx_node01 [ " hex " ] )
self . nodes [ 1 ] . sendrawtransaction ( tx_node01 [ " hex " ] )
assert tx_node0 [ " txid " ] in self . nodes [ 0 ] . getrawmempool ( )
assert not tx_node0 [ " txid " ] in self . nodes [ 1 ] . getrawmempool ( )
assert not tx_node1 [ " txid " ] in self . nodes [ 0 ] . getrawmempool ( )
assert tx_node1 [ " txid " ] in self . nodes [ 1 ] . getrawmempool ( )
assert tx_node01 [ " txid " ] in self . nodes [ 0 ] . getrawmempool ( )
assert tx_node01 [ " txid " ] in self . nodes [ 1 ] . getrawmempool ( )
assert not tx_node01_secret [ " txid " ] in self . nodes [ 0 ] . getrawmempool ( )
assert not tx_node01_secret [ " txid " ] in self . nodes [ 1 ] . getrawmempool ( )
self . log . debug ( " Check that importmempool can add txns without replacing the entire mempool " )
mempooldat0 = str ( self . nodes [ 0 ] . chain_path / " mempool.dat " )
result0 = self . nodes [ 0 ] . savemempool ( )
assert_equal ( mempooldat0 , result0 [ " filename " ] )
assert_equal ( { } , self . nodes [ 1 ] . importmempool ( mempooldat0 , { " apply_fee_delta_priority " : True } ) )
# All transactions should be in node1's mempool now.
assert tx_node0 [ " txid " ] in self . nodes [ 1 ] . getrawmempool ( )
assert tx_node1 [ " txid " ] in self . nodes [ 1 ] . getrawmempool ( )
assert not tx_node1 [ " txid " ] in self . nodes [ 0 ] . getrawmempool ( )
# For transactions that already existed, priority should be changed
entry_node01 = self . nodes [ 1 ] . getmempoolentry ( tx_node01 [ " txid " ] )
assert_equal ( entry_node01 [ " fees " ] [ " base " ] + 1 , entry_node01 [ " fees " ] [ " modified " ] )
# Deltas for not-yet-submitted transactions should be applied as well (prioritisation is stackable).
self . nodes [ 1 ] . sendrawtransaction ( tx_node01_secret [ " hex " ] )
entry_node01_secret = self . nodes [ 1 ] . getmempoolentry ( tx_node01_secret [ " txid " ] )
assert_equal ( entry_node01_secret [ " fees " ] [ " base " ] + 5 , entry_node01_secret [ " fees " ] [ " modified " ] )
self . stop_nodes ( )
2021-09-20 15:29:20 +02:00
if __name__ == " __main__ " :
2017-03-09 17:14:55 -05:00
MempoolPersistTest ( ) . main ( )