mirror of
https://github.com/bitcoin/bitcoin.git
synced 2025-01-22 01:13:03 -03:00
3405f3eed5
The test checks that parent txs are broadcast before child txs. The previous behavior is that the rebroadcasting would simply iterate mapWallet. As mapWallet is a std::unsorted_map, the child can sometimes come before the parent and thus be rebroadcast in the wrong order and fail the test.
121 lines
5.6 KiB
Python
Executable file
121 lines
5.6 KiB
Python
Executable file
#!/usr/bin/env python3
|
|
# Copyright (c) 2017-2021 The Bitcoin Core developers
|
|
# Distributed under the MIT software license, see the accompanying
|
|
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
"""Test that the wallet resends transactions periodically."""
|
|
import time
|
|
|
|
from test_framework.blocktools import (
|
|
create_block,
|
|
create_coinbase,
|
|
)
|
|
from test_framework.messages import DEFAULT_MEMPOOL_EXPIRY_HOURS
|
|
from test_framework.p2p import P2PTxInvStore
|
|
from test_framework.test_framework import BitcoinTestFramework
|
|
from test_framework.util import (
|
|
assert_equal,
|
|
assert_raises_rpc_error,
|
|
)
|
|
|
|
class ResendWalletTransactionsTest(BitcoinTestFramework):
|
|
def set_test_params(self):
|
|
self.num_nodes = 1
|
|
|
|
def skip_test_if_missing_module(self):
|
|
self.skip_if_no_wallet()
|
|
|
|
def run_test(self):
|
|
node = self.nodes[0] # alias
|
|
|
|
peer_first = node.add_p2p_connection(P2PTxInvStore())
|
|
|
|
self.log.info("Create a new transaction and wait until it's broadcast")
|
|
parent_utxo, indep_utxo = node.listunspent()[:2]
|
|
addr = node.getnewaddress()
|
|
txid = node.send(outputs=[{addr: 1}], options={"inputs": [parent_utxo]})["txid"]
|
|
|
|
# Can take a few seconds due to transaction trickling
|
|
peer_first.wait_for_broadcast([txid])
|
|
|
|
# Add a second peer since txs aren't rebroadcast to the same peer (see m_tx_inventory_known_filter)
|
|
peer_second = node.add_p2p_connection(P2PTxInvStore())
|
|
|
|
self.log.info("Create a block")
|
|
# Create and submit a block without the transaction.
|
|
# Transactions are only rebroadcast if there has been a block at least five minutes
|
|
# after the last time we tried to broadcast. Use mocktime and give an extra minute to be sure.
|
|
block_time = int(time.time()) + 6 * 60
|
|
node.setmocktime(block_time)
|
|
block = create_block(int(node.getbestblockhash(), 16), create_coinbase(node.getblockcount() + 1), block_time)
|
|
block.solve()
|
|
node.submitblock(block.serialize().hex())
|
|
|
|
# Set correct m_best_block_time, which is used in ResubmitWalletTransactions
|
|
node.syncwithvalidationinterfacequeue()
|
|
now = int(time.time())
|
|
|
|
# Transaction should not be rebroadcast within first 12 hours
|
|
# Leave 2 mins for buffer
|
|
twelve_hrs = 12 * 60 * 60
|
|
two_min = 2 * 60
|
|
node.setmocktime(now + twelve_hrs - two_min)
|
|
node.mockscheduler(60) # Tell scheduler to call MaybeResendWalletTxs now
|
|
assert_equal(int(txid, 16) in peer_second.get_invs(), False)
|
|
|
|
self.log.info("Bump time & check that transaction is rebroadcast")
|
|
# Transaction should be rebroadcast approximately 24 hours in the future,
|
|
# but can range from 12-36. So bump 36 hours to be sure.
|
|
with node.assert_debug_log(['resubmit 1 unconfirmed transactions']):
|
|
node.setmocktime(now + 36 * 60 * 60)
|
|
# Tell scheduler to call MaybeResendWalletTxs now.
|
|
node.mockscheduler(60)
|
|
# Give some time for trickle to occur
|
|
node.setmocktime(now + 36 * 60 * 60 + 600)
|
|
peer_second.wait_for_broadcast([txid])
|
|
|
|
self.log.info("Chain of unconfirmed not-in-mempool txs are rebroadcast")
|
|
# This tests that the node broadcasts the parent transaction before the child transaction.
|
|
# To test that scenario, we need a method to reliably get a child transaction placed
|
|
# in mapWallet positioned before the parent. We cannot predict the position in mapWallet,
|
|
# but we can observe it using listreceivedbyaddress and other related RPCs.
|
|
#
|
|
# So we will create the child transaction, use listreceivedbyaddress to see what the
|
|
# ordering of mapWallet is, if the child is not before the parent, we will create a new
|
|
# child (via bumpfee) and remove the old child (via removeprunedfunds) until we get the
|
|
# ordering of child before parent.
|
|
child_txid = node.send(outputs=[{addr: 0.5}], options={"inputs": [{"txid":txid, "vout":0}]})["txid"]
|
|
while True:
|
|
txids = node.listreceivedbyaddress(minconf=0, address_filter=addr)[0]["txids"]
|
|
if txids == [child_txid, txid]:
|
|
break
|
|
bumped = node.bumpfee(child_txid)
|
|
node.removeprunedfunds(child_txid)
|
|
child_txid = bumped["txid"]
|
|
entry_time = node.getmempoolentry(child_txid)["time"]
|
|
|
|
block_time = entry_time + 6 * 60
|
|
node.setmocktime(block_time)
|
|
block = create_block(int(node.getbestblockhash(), 16), create_coinbase(node.getblockcount() + 1), block_time)
|
|
block.solve()
|
|
node.submitblock(block.serialize().hex())
|
|
node.syncwithvalidationinterfacequeue()
|
|
|
|
# Evict these txs from the mempool
|
|
evict_time = block_time + 60 * 60 * DEFAULT_MEMPOOL_EXPIRY_HOURS + 5
|
|
node.setmocktime(evict_time)
|
|
indep_send = node.send(outputs=[{node.getnewaddress(): 1}], options={"inputs": [indep_utxo]})
|
|
node.syncwithvalidationinterfacequeue()
|
|
node.getmempoolentry(indep_send["txid"])
|
|
assert_raises_rpc_error(-5, "Transaction not in mempool", node.getmempoolentry, txid)
|
|
assert_raises_rpc_error(-5, "Transaction not in mempool", node.getmempoolentry, child_txid)
|
|
|
|
# Rebroadcast and check that parent and child are both in the mempool
|
|
with node.assert_debug_log(['resubmit 2 unconfirmed transactions']):
|
|
node.setmocktime(evict_time + 36 * 60 * 60) # 36 hrs is the upper limit of the resend timer
|
|
node.mockscheduler(60)
|
|
node.getmempoolentry(txid)
|
|
node.getmempoolentry(child_txid)
|
|
|
|
|
|
if __name__ == '__main__':
|
|
ResendWalletTransactionsTest().main()
|