mirror of
https://github.com/bitcoin/bitcoin.git
synced 2025-01-10 20:03:34 -03:00
Merge bitcoin/bitcoin#22686: wallet: Use GetSelectionAmount in ApproximateBestSubset
92885c4f69
test: Test for ApproximateBestSubset edge case with too little fees (Andrew Chow)d9262324e8
wallet: Assert that enough was selected to cover the fees (Andrew Chow)2de222c401
wallet: Use GetSelectionAmount for target value calculations (Andrew Chow) Pull request description: The `m_value` used for the target calculation in `ApproximateBestSubset` is incorrect, it should be `GetSelectionAmount`. This causes a bug that is only apparent when the minimum relay fee is set to be very high. A test case is added for this, in addition to an assert in `CreateTransactionInternal` that would have also caught this issue if someone were able to hit the edge case. Fixes #22670 ACKs for top commit: instagibbs: utACK92885c4f69
Tree-SHA512: bd61fa61ffb60873e097737eebea3afe8a42296ba429de9038b3a4706763b34de9409de6cdbab21ff7f51f4787b503f840873182d9c4a1d6e12a54b017953547
This commit is contained in:
commit
820129aee9
3 changed files with 63 additions and 2 deletions
|
@ -195,7 +195,7 @@ static void ApproximateBestSubset(const std::vector<OutputGroup>& groups, const
|
|||
//the selection random.
|
||||
if (nPass == 0 ? insecure_rand.randbool() : !vfIncluded[i])
|
||||
{
|
||||
nTotal += groups[i].m_value;
|
||||
nTotal += groups[i].GetSelectionAmount();
|
||||
vfIncluded[i] = true;
|
||||
if (nTotal >= nTargetValue)
|
||||
{
|
||||
|
@ -205,7 +205,7 @@ static void ApproximateBestSubset(const std::vector<OutputGroup>& groups, const
|
|||
nBest = nTotal;
|
||||
vfBest = vfIncluded;
|
||||
}
|
||||
nTotal -= groups[i].m_value;
|
||||
nTotal -= groups[i].GetSelectionAmount();
|
||||
vfIncluded[i] = false;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -778,6 +778,10 @@ bool CWallet::CreateTransactionInternal(
|
|||
fee_needed = coin_selection_params.m_effective_feerate.GetFee(nBytes);
|
||||
}
|
||||
|
||||
// The only time that fee_needed should be less than the amount available for fees (in change_and_fee - change_amount) is when
|
||||
// we are subtracting the fee from the outputs. If this occurs at any other time, it is a bug.
|
||||
assert(coin_selection_params.m_subtract_fee_outputs || fee_needed <= change_and_fee - change_amount);
|
||||
|
||||
// Update nFeeRet in case fee_needed changed due to dropping the change output
|
||||
if (fee_needed <= change_and_fee - change_amount) {
|
||||
nFeeRet = change_and_fee - change_amount;
|
||||
|
|
|
@ -99,6 +99,7 @@ class RawTransactionsTest(BitcoinTestFramework):
|
|||
self.test_subtract_fee_with_presets()
|
||||
self.test_transaction_too_large()
|
||||
self.test_include_unsafe()
|
||||
self.test_22670()
|
||||
|
||||
def test_change_position(self):
|
||||
"""Ensure setting changePosition in fundraw with an exact match is handled properly."""
|
||||
|
@ -969,6 +970,62 @@ class RawTransactionsTest(BitcoinTestFramework):
|
|||
signedtx = wallet.signrawtransactionwithwallet(fundedtx['hex'])
|
||||
wallet.sendrawtransaction(signedtx['hex'])
|
||||
|
||||
def test_22670(self):
|
||||
# In issue #22670, it was observed that ApproximateBestSubset may
|
||||
# choose enough value to cover the target amount but not enough to cover the transaction fees.
|
||||
# This leads to a transaction whose actual transaction feerate is lower than expected.
|
||||
# However at normal feerates, the difference between the effective value and the real value
|
||||
# that this bug is not detected because the transaction fee must be at least 0.01 BTC (the minimum change value).
|
||||
# Otherwise the targeted minimum change value will be enough to cover the transaction fees that were not
|
||||
# being accounted for. So the minimum relay fee is set to 0.1 BTC/kvB in this test.
|
||||
self.log.info("Test issue 22670 ApproximateBestSubset bug")
|
||||
# Make sure the default wallet will not be loaded when restarted with a high minrelaytxfee
|
||||
self.nodes[0].unloadwallet(self.default_wallet_name, False)
|
||||
feerate = Decimal("0.1")
|
||||
self.restart_node(0, [f"-minrelaytxfee={feerate}", "-discardfee=0"]) # Set high minrelayfee, set discardfee to 0 for easier calculation
|
||||
|
||||
self.nodes[0].loadwallet(self.default_wallet_name, True)
|
||||
funds = self.nodes[0].get_wallet_rpc(self.default_wallet_name)
|
||||
self.nodes[0].createwallet(wallet_name="tester")
|
||||
tester = self.nodes[0].get_wallet_rpc("tester")
|
||||
|
||||
# Because this test is specifically for ApproximateBestSubset, the target value must be greater
|
||||
# than any single input available, and require more than 1 input. So we make 3 outputs
|
||||
for i in range(0, 3):
|
||||
funds.sendtoaddress(tester.getnewaddress(address_type="bech32"), 1)
|
||||
self.nodes[0].generate(1)
|
||||
|
||||
# Create transactions in order to calculate fees for the target bounds that can trigger this bug
|
||||
change_tx = tester.fundrawtransaction(tester.createrawtransaction([], [{funds.getnewaddress(): 1.5}]))
|
||||
tx = tester.createrawtransaction([], [{funds.getnewaddress(): 2}])
|
||||
no_change_tx = tester.fundrawtransaction(tx, {"subtractFeeFromOutputs": [0]})
|
||||
|
||||
overhead_fees = feerate * len(tx) / 2 / 1000
|
||||
cost_of_change = change_tx["fee"] - no_change_tx["fee"]
|
||||
fees = no_change_tx["fee"]
|
||||
assert_greater_than(fees, 0.01)
|
||||
|
||||
def do_fund_send(target):
|
||||
create_tx = tester.createrawtransaction([], [{funds.getnewaddress(): lower_bound}])
|
||||
funded_tx = tester.fundrawtransaction(create_tx)
|
||||
signed_tx = tester.signrawtransactionwithwallet(funded_tx["hex"])
|
||||
assert signed_tx["complete"]
|
||||
decoded_tx = tester.decoderawtransaction(signed_tx["hex"])
|
||||
assert_equal(len(decoded_tx["vin"]), 3)
|
||||
assert tester.testmempoolaccept([signed_tx["hex"]])[0]["allowed"]
|
||||
|
||||
# We want to choose more value than is available in 2 inputs when considering the fee,
|
||||
# but not enough to need 3 inputs when not considering the fee.
|
||||
# So the target value must be at least 2.00000001 - fee.
|
||||
lower_bound = Decimal("2.00000001") - fees
|
||||
# The target value must be at most 2 - cost_of_change - not_input_fees - min_change (these are all
|
||||
# included in the target before ApproximateBestSubset).
|
||||
upper_bound = Decimal("2.0") - cost_of_change - overhead_fees - Decimal("0.01")
|
||||
assert_greater_than_or_equal(upper_bound, lower_bound)
|
||||
do_fund_send(lower_bound)
|
||||
do_fund_send(upper_bound)
|
||||
|
||||
self.restart_node(0)
|
||||
|
||||
if __name__ == '__main__':
|
||||
RawTransactionsTest().main()
|
||||
|
|
Loading…
Reference in a new issue