mirror of
https://github.com/bitcoin/bitcoin.git
synced 2025-01-26 11:13:23 -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.
|
//the selection random.
|
||||||
if (nPass == 0 ? insecure_rand.randbool() : !vfIncluded[i])
|
if (nPass == 0 ? insecure_rand.randbool() : !vfIncluded[i])
|
||||||
{
|
{
|
||||||
nTotal += groups[i].m_value;
|
nTotal += groups[i].GetSelectionAmount();
|
||||||
vfIncluded[i] = true;
|
vfIncluded[i] = true;
|
||||||
if (nTotal >= nTargetValue)
|
if (nTotal >= nTargetValue)
|
||||||
{
|
{
|
||||||
|
@ -205,7 +205,7 @@ static void ApproximateBestSubset(const std::vector<OutputGroup>& groups, const
|
||||||
nBest = nTotal;
|
nBest = nTotal;
|
||||||
vfBest = vfIncluded;
|
vfBest = vfIncluded;
|
||||||
}
|
}
|
||||||
nTotal -= groups[i].m_value;
|
nTotal -= groups[i].GetSelectionAmount();
|
||||||
vfIncluded[i] = false;
|
vfIncluded[i] = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -778,6 +778,10 @@ bool CWallet::CreateTransactionInternal(
|
||||||
fee_needed = coin_selection_params.m_effective_feerate.GetFee(nBytes);
|
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
|
// Update nFeeRet in case fee_needed changed due to dropping the change output
|
||||||
if (fee_needed <= change_and_fee - change_amount) {
|
if (fee_needed <= change_and_fee - change_amount) {
|
||||||
nFeeRet = 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_subtract_fee_with_presets()
|
||||||
self.test_transaction_too_large()
|
self.test_transaction_too_large()
|
||||||
self.test_include_unsafe()
|
self.test_include_unsafe()
|
||||||
|
self.test_22670()
|
||||||
|
|
||||||
def test_change_position(self):
|
def test_change_position(self):
|
||||||
"""Ensure setting changePosition in fundraw with an exact match is handled properly."""
|
"""Ensure setting changePosition in fundraw with an exact match is handled properly."""
|
||||||
|
@ -969,6 +970,62 @@ class RawTransactionsTest(BitcoinTestFramework):
|
||||||
signedtx = wallet.signrawtransactionwithwallet(fundedtx['hex'])
|
signedtx = wallet.signrawtransactionwithwallet(fundedtx['hex'])
|
||||||
wallet.sendrawtransaction(signedtx['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__':
|
if __name__ == '__main__':
|
||||||
RawTransactionsTest().main()
|
RawTransactionsTest().main()
|
||||||
|
|
Loading…
Add table
Reference in a new issue