mirror of
https://github.com/bitcoin/bitcoin.git
synced 2025-06-30 10:42:23 +02:00
test: Fix intermittent issue in wallet_bumpfee
This commit is contained in:
@ -50,6 +50,11 @@ class BumpFeeTest(BitcoinTestFramework):
|
|||||||
def skip_test_if_missing_module(self):
|
def skip_test_if_missing_module(self):
|
||||||
self.skip_if_no_wallet()
|
self.skip_if_no_wallet()
|
||||||
|
|
||||||
|
def clear_mempool(self):
|
||||||
|
# Clear mempool between subtests. The subtests may only depend on chainstate (utxos)
|
||||||
|
self.nodes[1].generate(1)
|
||||||
|
self.sync_all()
|
||||||
|
|
||||||
def run_test(self):
|
def run_test(self):
|
||||||
# Encrypt wallet for test_locked_wallet_fails test
|
# Encrypt wallet for test_locked_wallet_fails test
|
||||||
self.nodes[1].encryptwallet(WALLET_PASSPHRASE)
|
self.nodes[1].encryptwallet(WALLET_PASSPHRASE)
|
||||||
@ -71,7 +76,7 @@ class BumpFeeTest(BitcoinTestFramework):
|
|||||||
|
|
||||||
self.log.info("Running tests")
|
self.log.info("Running tests")
|
||||||
dest_address = peer_node.getnewaddress()
|
dest_address = peer_node.getnewaddress()
|
||||||
test_invalid_parameters(rbf_node, dest_address)
|
self.test_invalid_parameters(rbf_node, dest_address)
|
||||||
test_simple_bumpfee_succeeds(self, "default", rbf_node, peer_node, dest_address)
|
test_simple_bumpfee_succeeds(self, "default", rbf_node, peer_node, dest_address)
|
||||||
test_simple_bumpfee_succeeds(self, "fee_rate", rbf_node, peer_node, dest_address)
|
test_simple_bumpfee_succeeds(self, "fee_rate", rbf_node, peer_node, dest_address)
|
||||||
test_feerate_args(self, rbf_node, peer_node, dest_address)
|
test_feerate_args(self, rbf_node, peer_node, dest_address)
|
||||||
@ -93,7 +98,7 @@ class BumpFeeTest(BitcoinTestFramework):
|
|||||||
test_small_output_with_feerate_succeeds(self, rbf_node, dest_address)
|
test_small_output_with_feerate_succeeds(self, rbf_node, dest_address)
|
||||||
test_no_more_inputs_fails(self, rbf_node, dest_address)
|
test_no_more_inputs_fails(self, rbf_node, dest_address)
|
||||||
|
|
||||||
def test_invalid_parameters(node, dest_address):
|
def test_invalid_parameters(self, node, dest_address):
|
||||||
txid = spend_one_input(node, dest_address)
|
txid = spend_one_input(node, dest_address)
|
||||||
# invalid estimate mode
|
# invalid estimate mode
|
||||||
assert_raises_rpc_error(-8, "Invalid estimate_mode parameter", node.bumpfee, txid, {
|
assert_raises_rpc_error(-8, "Invalid estimate_mode parameter", node.bumpfee, txid, {
|
||||||
@ -115,6 +120,8 @@ def test_invalid_parameters(node, dest_address):
|
|||||||
"confTarget": 123,
|
"confTarget": 123,
|
||||||
"conf_target": 456,
|
"conf_target": 456,
|
||||||
})
|
})
|
||||||
|
self.clear_mempool()
|
||||||
|
|
||||||
|
|
||||||
def test_simple_bumpfee_succeeds(self, mode, rbf_node, peer_node, dest_address):
|
def test_simple_bumpfee_succeeds(self, mode, rbf_node, peer_node, dest_address):
|
||||||
self.log.info('Test simple bumpfee: {}'.format(mode))
|
self.log.info('Test simple bumpfee: {}'.format(mode))
|
||||||
@ -148,6 +155,7 @@ def test_simple_bumpfee_succeeds(self, mode, rbf_node, peer_node, dest_address):
|
|||||||
bumpedwtx = rbf_node.gettransaction(bumped_tx["txid"])
|
bumpedwtx = rbf_node.gettransaction(bumped_tx["txid"])
|
||||||
assert_equal(oldwtx["replaced_by_txid"], bumped_tx["txid"])
|
assert_equal(oldwtx["replaced_by_txid"], bumped_tx["txid"])
|
||||||
assert_equal(bumpedwtx["replaces_txid"], rbfid)
|
assert_equal(bumpedwtx["replaces_txid"], rbfid)
|
||||||
|
self.clear_mempool()
|
||||||
|
|
||||||
|
|
||||||
def test_feerate_args(self, rbf_node, peer_node, dest_address):
|
def test_feerate_args(self, rbf_node, peer_node, dest_address):
|
||||||
@ -167,6 +175,7 @@ def test_feerate_args(self, rbf_node, peer_node, dest_address):
|
|||||||
assert_raises_rpc_error(-3, "Amount out of range", rbf_node.bumpfee, rbfid, {"fee_rate": -1})
|
assert_raises_rpc_error(-3, "Amount out of range", rbf_node.bumpfee, rbfid, {"fee_rate": -1})
|
||||||
|
|
||||||
assert_raises_rpc_error(-4, "is too high (cannot be higher than", rbf_node.bumpfee, rbfid, {"fee_rate": TOO_HIGH})
|
assert_raises_rpc_error(-4, "is too high (cannot be higher than", rbf_node.bumpfee, rbfid, {"fee_rate": TOO_HIGH})
|
||||||
|
self.clear_mempool()
|
||||||
|
|
||||||
|
|
||||||
def test_segwit_bumpfee_succeeds(self, rbf_node, dest_address):
|
def test_segwit_bumpfee_succeeds(self, rbf_node, dest_address):
|
||||||
@ -198,12 +207,14 @@ def test_segwit_bumpfee_succeeds(self, rbf_node, dest_address):
|
|||||||
bumped_tx = rbf_node.bumpfee(rbfid)
|
bumped_tx = rbf_node.bumpfee(rbfid)
|
||||||
assert bumped_tx["txid"] in rbf_node.getrawmempool()
|
assert bumped_tx["txid"] in rbf_node.getrawmempool()
|
||||||
assert rbfid not in rbf_node.getrawmempool()
|
assert rbfid not in rbf_node.getrawmempool()
|
||||||
|
self.clear_mempool()
|
||||||
|
|
||||||
|
|
||||||
def test_nonrbf_bumpfee_fails(self, peer_node, dest_address):
|
def test_nonrbf_bumpfee_fails(self, peer_node, dest_address):
|
||||||
self.log.info('Test that we cannot replace a non RBF transaction')
|
self.log.info('Test that we cannot replace a non RBF transaction')
|
||||||
not_rbfid = peer_node.sendtoaddress(dest_address, Decimal("0.00090000"))
|
not_rbfid = peer_node.sendtoaddress(dest_address, Decimal("0.00090000"))
|
||||||
assert_raises_rpc_error(-4, "not BIP 125 replaceable", peer_node.bumpfee, not_rbfid)
|
assert_raises_rpc_error(-4, "not BIP 125 replaceable", peer_node.bumpfee, not_rbfid)
|
||||||
|
self.clear_mempool()
|
||||||
|
|
||||||
|
|
||||||
def test_notmine_bumpfee_fails(self, rbf_node, peer_node, dest_address):
|
def test_notmine_bumpfee_fails(self, rbf_node, peer_node, dest_address):
|
||||||
@ -226,6 +237,7 @@ def test_notmine_bumpfee_fails(self, rbf_node, peer_node, dest_address):
|
|||||||
rbfid = rbf_node.sendrawtransaction(signedtx["hex"])
|
rbfid = rbf_node.sendrawtransaction(signedtx["hex"])
|
||||||
assert_raises_rpc_error(-4, "Transaction contains inputs that don't belong to this wallet",
|
assert_raises_rpc_error(-4, "Transaction contains inputs that don't belong to this wallet",
|
||||||
rbf_node.bumpfee, rbfid)
|
rbf_node.bumpfee, rbfid)
|
||||||
|
self.clear_mempool()
|
||||||
|
|
||||||
|
|
||||||
def test_bumpfee_with_descendant_fails(self, rbf_node, rbf_node_address, dest_address):
|
def test_bumpfee_with_descendant_fails(self, rbf_node, rbf_node_address, dest_address):
|
||||||
@ -236,6 +248,7 @@ def test_bumpfee_with_descendant_fails(self, rbf_node, rbf_node_address, dest_ad
|
|||||||
tx = rbf_node.signrawtransactionwithwallet(tx)
|
tx = rbf_node.signrawtransactionwithwallet(tx)
|
||||||
rbf_node.sendrawtransaction(tx["hex"])
|
rbf_node.sendrawtransaction(tx["hex"])
|
||||||
assert_raises_rpc_error(-8, "Transaction has descendants in the wallet", rbf_node.bumpfee, parent_id)
|
assert_raises_rpc_error(-8, "Transaction has descendants in the wallet", rbf_node.bumpfee, parent_id)
|
||||||
|
self.clear_mempool()
|
||||||
|
|
||||||
|
|
||||||
def test_small_output_with_feerate_succeeds(self, rbf_node, dest_address):
|
def test_small_output_with_feerate_succeeds(self, rbf_node, dest_address):
|
||||||
@ -277,6 +290,7 @@ def test_small_output_with_feerate_succeeds(self, rbf_node, dest_address):
|
|||||||
|
|
||||||
rbf_node.generatetoaddress(1, rbf_node.getnewaddress())
|
rbf_node.generatetoaddress(1, rbf_node.getnewaddress())
|
||||||
assert_equal(rbf_node.gettransaction(rbfid)["confirmations"], 1)
|
assert_equal(rbf_node.gettransaction(rbfid)["confirmations"], 1)
|
||||||
|
self.clear_mempool()
|
||||||
|
|
||||||
|
|
||||||
def test_dust_to_fee(self, rbf_node, dest_address):
|
def test_dust_to_fee(self, rbf_node, dest_address):
|
||||||
@ -299,6 +313,7 @@ def test_dust_to_fee(self, rbf_node, dest_address):
|
|||||||
assert_equal(len(fulltx["vout"]), 2)
|
assert_equal(len(fulltx["vout"]), 2)
|
||||||
assert_equal(len(full_bumped_tx["vout"]), 1) # change output is eliminated
|
assert_equal(len(full_bumped_tx["vout"]), 1) # change output is eliminated
|
||||||
assert_equal(full_bumped_tx["vout"][0]['value'], Decimal("0.00050000"))
|
assert_equal(full_bumped_tx["vout"][0]['value'], Decimal("0.00050000"))
|
||||||
|
self.clear_mempool()
|
||||||
|
|
||||||
|
|
||||||
def test_settxfee(self, rbf_node, dest_address):
|
def test_settxfee(self, rbf_node, dest_address):
|
||||||
@ -321,6 +336,8 @@ def test_settxfee(self, rbf_node, dest_address):
|
|||||||
assert_raises_rpc_error(-8, "txfee cannot be more than wallet max tx fee", rbf_node.settxfee, Decimal('0.00003'))
|
assert_raises_rpc_error(-8, "txfee cannot be more than wallet max tx fee", rbf_node.settxfee, Decimal('0.00003'))
|
||||||
self.restart_node(1, self.extra_args[1])
|
self.restart_node(1, self.extra_args[1])
|
||||||
rbf_node.walletpassphrase(WALLET_PASSPHRASE, WALLET_PASSPHRASE_TIMEOUT)
|
rbf_node.walletpassphrase(WALLET_PASSPHRASE, WALLET_PASSPHRASE_TIMEOUT)
|
||||||
|
self.connect_nodes(1, 0)
|
||||||
|
self.clear_mempool()
|
||||||
|
|
||||||
|
|
||||||
def test_maxtxfee_fails(self, rbf_node, dest_address):
|
def test_maxtxfee_fails(self, rbf_node, dest_address):
|
||||||
@ -334,6 +351,8 @@ def test_maxtxfee_fails(self, rbf_node, dest_address):
|
|||||||
assert_raises_rpc_error(-4, "Unable to create transaction. Fee exceeds maximum configured by -maxtxfee", rbf_node.bumpfee, rbfid)
|
assert_raises_rpc_error(-4, "Unable to create transaction. Fee exceeds maximum configured by -maxtxfee", rbf_node.bumpfee, rbfid)
|
||||||
self.restart_node(1, self.extra_args[1])
|
self.restart_node(1, self.extra_args[1])
|
||||||
rbf_node.walletpassphrase(WALLET_PASSPHRASE, WALLET_PASSPHRASE_TIMEOUT)
|
rbf_node.walletpassphrase(WALLET_PASSPHRASE, WALLET_PASSPHRASE_TIMEOUT)
|
||||||
|
self.connect_nodes(1, 0)
|
||||||
|
self.clear_mempool()
|
||||||
|
|
||||||
|
|
||||||
def test_watchonly_psbt(self, peer_node, rbf_node, dest_address):
|
def test_watchonly_psbt(self, peer_node, rbf_node, dest_address):
|
||||||
@ -416,6 +435,7 @@ def test_watchonly_psbt(self, peer_node, rbf_node, dest_address):
|
|||||||
|
|
||||||
rbf_node.unloadwallet("watcher")
|
rbf_node.unloadwallet("watcher")
|
||||||
rbf_node.unloadwallet("signer")
|
rbf_node.unloadwallet("signer")
|
||||||
|
self.clear_mempool()
|
||||||
|
|
||||||
|
|
||||||
def test_rebumping(self, rbf_node, dest_address):
|
def test_rebumping(self, rbf_node, dest_address):
|
||||||
@ -424,6 +444,7 @@ def test_rebumping(self, rbf_node, dest_address):
|
|||||||
bumped = rbf_node.bumpfee(rbfid, {"fee_rate": ECONOMICAL})
|
bumped = rbf_node.bumpfee(rbfid, {"fee_rate": ECONOMICAL})
|
||||||
assert_raises_rpc_error(-4, "already bumped", rbf_node.bumpfee, rbfid, {"fee_rate": NORMAL})
|
assert_raises_rpc_error(-4, "already bumped", rbf_node.bumpfee, rbfid, {"fee_rate": NORMAL})
|
||||||
rbf_node.bumpfee(bumped["txid"], {"fee_rate": NORMAL})
|
rbf_node.bumpfee(bumped["txid"], {"fee_rate": NORMAL})
|
||||||
|
self.clear_mempool()
|
||||||
|
|
||||||
|
|
||||||
def test_rebumping_not_replaceable(self, rbf_node, dest_address):
|
def test_rebumping_not_replaceable(self, rbf_node, dest_address):
|
||||||
@ -432,6 +453,7 @@ def test_rebumping_not_replaceable(self, rbf_node, dest_address):
|
|||||||
bumped = rbf_node.bumpfee(rbfid, {"fee_rate": ECONOMICAL, "replaceable": False})
|
bumped = rbf_node.bumpfee(rbfid, {"fee_rate": ECONOMICAL, "replaceable": False})
|
||||||
assert_raises_rpc_error(-4, "Transaction is not BIP 125 replaceable", rbf_node.bumpfee, bumped["txid"],
|
assert_raises_rpc_error(-4, "Transaction is not BIP 125 replaceable", rbf_node.bumpfee, bumped["txid"],
|
||||||
{"fee_rate": NORMAL})
|
{"fee_rate": NORMAL})
|
||||||
|
self.clear_mempool()
|
||||||
|
|
||||||
|
|
||||||
def test_unconfirmed_not_spendable(self, rbf_node, rbf_node_address):
|
def test_unconfirmed_not_spendable(self, rbf_node, rbf_node_address):
|
||||||
@ -471,6 +493,7 @@ def test_unconfirmed_not_spendable(self, rbf_node, rbf_node_address):
|
|||||||
assert_equal(
|
assert_equal(
|
||||||
sum(1 for t in rbf_node.listunspent(minconf=0, include_unsafe=False)
|
sum(1 for t in rbf_node.listunspent(minconf=0, include_unsafe=False)
|
||||||
if t["txid"] == rbfid and t["address"] == rbf_node_address and t["spendable"]), 1)
|
if t["txid"] == rbfid and t["address"] == rbf_node_address and t["spendable"]), 1)
|
||||||
|
self.clear_mempool()
|
||||||
|
|
||||||
|
|
||||||
def test_bumpfee_metadata(self, rbf_node, dest_address):
|
def test_bumpfee_metadata(self, rbf_node, dest_address):
|
||||||
@ -482,6 +505,7 @@ def test_bumpfee_metadata(self, rbf_node, dest_address):
|
|||||||
bumped_wtx = rbf_node.gettransaction(bumped_tx["txid"])
|
bumped_wtx = rbf_node.gettransaction(bumped_tx["txid"])
|
||||||
assert_equal(bumped_wtx["comment"], "comment value")
|
assert_equal(bumped_wtx["comment"], "comment value")
|
||||||
assert_equal(bumped_wtx["to"], "to value")
|
assert_equal(bumped_wtx["to"], "to value")
|
||||||
|
self.clear_mempool()
|
||||||
|
|
||||||
|
|
||||||
def test_locked_wallet_fails(self, rbf_node, dest_address):
|
def test_locked_wallet_fails(self, rbf_node, dest_address):
|
||||||
@ -491,6 +515,7 @@ def test_locked_wallet_fails(self, rbf_node, dest_address):
|
|||||||
assert_raises_rpc_error(-13, "Please enter the wallet passphrase with walletpassphrase first.",
|
assert_raises_rpc_error(-13, "Please enter the wallet passphrase with walletpassphrase first.",
|
||||||
rbf_node.bumpfee, rbfid)
|
rbf_node.bumpfee, rbfid)
|
||||||
rbf_node.walletpassphrase(WALLET_PASSPHRASE, WALLET_PASSPHRASE_TIMEOUT)
|
rbf_node.walletpassphrase(WALLET_PASSPHRASE, WALLET_PASSPHRASE_TIMEOUT)
|
||||||
|
self.clear_mempool()
|
||||||
|
|
||||||
|
|
||||||
def test_change_script_match(self, rbf_node, dest_address):
|
def test_change_script_match(self, rbf_node, dest_address):
|
||||||
@ -511,6 +536,7 @@ def test_change_script_match(self, rbf_node, dest_address):
|
|||||||
assert_equal(change_addresses, get_change_address(bumped_total_tx['txid']))
|
assert_equal(change_addresses, get_change_address(bumped_total_tx['txid']))
|
||||||
bumped_rate_tx = rbf_node.bumpfee(bumped_total_tx["txid"])
|
bumped_rate_tx = rbf_node.bumpfee(bumped_total_tx["txid"])
|
||||||
assert_equal(change_addresses, get_change_address(bumped_rate_tx['txid']))
|
assert_equal(change_addresses, get_change_address(bumped_rate_tx['txid']))
|
||||||
|
self.clear_mempool()
|
||||||
|
|
||||||
|
|
||||||
def spend_one_input(node, dest_address, change_size=Decimal("0.00049000")):
|
def spend_one_input(node, dest_address, change_size=Decimal("0.00049000")):
|
||||||
@ -549,6 +575,7 @@ def test_no_more_inputs_fails(self, rbf_node, dest_address):
|
|||||||
# spend all funds, no change output
|
# spend all funds, no change output
|
||||||
rbfid = rbf_node.sendtoaddress(rbf_node.getnewaddress(), rbf_node.getbalance(), "", "", True)
|
rbfid = rbf_node.sendtoaddress(rbf_node.getnewaddress(), rbf_node.getbalance(), "", "", True)
|
||||||
assert_raises_rpc_error(-4, "Unable to create transaction. Insufficient funds", rbf_node.bumpfee, rbfid)
|
assert_raises_rpc_error(-4, "Unable to create transaction. Insufficient funds", rbf_node.bumpfee, rbfid)
|
||||||
|
self.clear_mempool()
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
Reference in New Issue
Block a user