mirror of
https://github.com/Retropex/bitcoin.git
synced 2025-05-29 13:32:33 +02:00
Merge #17824: wallet: Prefer full destination groups in coin selection
a2324e4d3f
test: Improve naming and logging of avoid_reuse tests (Fabian Jahr)1abbdac677
wallet: Prefer full destination groups in coin selection (Fabian Jahr) Pull request description: Fixes #17603 (together with #17843) In the case of destination groups of >10 outputs existing in a wallet with `avoid_reuse` enabled, the grouping algorithm is adding left-over outputs as an "incomplete" group to the list of groups even when a full group has already been added. This leads to the strange behavior that if there are >10 outputs for a destination the transaction spending from that will effectively use `len(outputs) % 10` as inputs for that transaction. From the original PR and the code comment I understand the correct behavior should be the usage of 10 outputs. I opted for minimal changes in the current code although there maybe optimizations possible for cases with >20 outputs on a destination this sounds like too much of an edge case right now. ACKs for top commit: jonatack: Re-ACKa2324e4
achow101: ACKa2324e4d3f
kallewoof: ACKa2324e4d3f
meshcollider: Tested ACKa2324e4d3f
(verified the new test fails on master without this change) Tree-SHA512: 4743779c5d469fcd16df5baf166024b1d3c8eaca151df1e8281b71df62b29541cf7bfee3f8ab48d83e3b34c9256e53fd38a7b146a54c79f9caa44cce3636971a
This commit is contained in:
commit
c189bfd260
@ -2372,6 +2372,13 @@ bool CWallet::SelectCoins(const std::vector<COutput>& vAvailableCoins, const CAm
|
|||||||
++it;
|
++it;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
unsigned int limit_ancestor_count = 0;
|
||||||
|
unsigned int limit_descendant_count = 0;
|
||||||
|
chain().getPackageLimits(limit_ancestor_count, limit_descendant_count);
|
||||||
|
size_t max_ancestors = (size_t)std::max<int64_t>(1, limit_ancestor_count);
|
||||||
|
size_t max_descendants = (size_t)std::max<int64_t>(1, limit_descendant_count);
|
||||||
|
bool fRejectLongChains = gArgs.GetBoolArg("-walletrejectlongchains", DEFAULT_WALLET_REJECT_LONG_CHAINS);
|
||||||
|
|
||||||
// form groups from remaining coins; note that preset coins will not
|
// form groups from remaining coins; note that preset coins will not
|
||||||
// automatically have their associated (same address) coins included
|
// automatically have their associated (same address) coins included
|
||||||
if (coin_control.m_avoid_partial_spends && vCoins.size() > OUTPUT_GROUP_MAX_ENTRIES) {
|
if (coin_control.m_avoid_partial_spends && vCoins.size() > OUTPUT_GROUP_MAX_ENTRIES) {
|
||||||
@ -2380,14 +2387,7 @@ bool CWallet::SelectCoins(const std::vector<COutput>& vAvailableCoins, const CAm
|
|||||||
// explicitly shuffling the outputs before processing
|
// explicitly shuffling the outputs before processing
|
||||||
Shuffle(vCoins.begin(), vCoins.end(), FastRandomContext());
|
Shuffle(vCoins.begin(), vCoins.end(), FastRandomContext());
|
||||||
}
|
}
|
||||||
std::vector<OutputGroup> groups = GroupOutputs(vCoins, !coin_control.m_avoid_partial_spends);
|
std::vector<OutputGroup> groups = GroupOutputs(vCoins, !coin_control.m_avoid_partial_spends, max_ancestors);
|
||||||
|
|
||||||
unsigned int limit_ancestor_count;
|
|
||||||
unsigned int limit_descendant_count;
|
|
||||||
chain().getPackageLimits(limit_ancestor_count, limit_descendant_count);
|
|
||||||
size_t max_ancestors = (size_t)std::max<int64_t>(1, limit_ancestor_count);
|
|
||||||
size_t max_descendants = (size_t)std::max<int64_t>(1, limit_descendant_count);
|
|
||||||
bool fRejectLongChains = gArgs.GetBoolArg("-walletrejectlongchains", DEFAULT_WALLET_REJECT_LONG_CHAINS);
|
|
||||||
|
|
||||||
bool res = value_to_select <= 0 ||
|
bool res = value_to_select <= 0 ||
|
||||||
SelectCoinsMinConf(value_to_select, CoinEligibilityFilter(1, 6, 0), groups, setCoinsRet, nValueRet, coin_selection_params, bnb_used) ||
|
SelectCoinsMinConf(value_to_select, CoinEligibilityFilter(1, 6, 0), groups, setCoinsRet, nValueRet, coin_selection_params, bnb_used) ||
|
||||||
@ -4184,32 +4184,49 @@ bool CWalletTx::IsImmatureCoinBase() const
|
|||||||
return GetBlocksToMaturity() > 0;
|
return GetBlocksToMaturity() > 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::vector<OutputGroup> CWallet::GroupOutputs(const std::vector<COutput>& outputs, bool single_coin) const {
|
std::vector<OutputGroup> CWallet::GroupOutputs(const std::vector<COutput>& outputs, bool single_coin, const size_t max_ancestors) const {
|
||||||
std::vector<OutputGroup> groups;
|
std::vector<OutputGroup> groups;
|
||||||
std::map<CTxDestination, OutputGroup> gmap;
|
std::map<CTxDestination, OutputGroup> gmap;
|
||||||
CTxDestination dst;
|
std::set<CTxDestination> full_groups;
|
||||||
|
|
||||||
for (const auto& output : outputs) {
|
for (const auto& output : outputs) {
|
||||||
if (output.fSpendable) {
|
if (output.fSpendable) {
|
||||||
|
CTxDestination dst;
|
||||||
CInputCoin input_coin = output.GetInputCoin();
|
CInputCoin input_coin = output.GetInputCoin();
|
||||||
|
|
||||||
size_t ancestors, descendants;
|
size_t ancestors, descendants;
|
||||||
chain().getTransactionAncestry(output.tx->GetHash(), ancestors, descendants);
|
chain().getTransactionAncestry(output.tx->GetHash(), ancestors, descendants);
|
||||||
if (!single_coin && ExtractDestination(output.tx->tx->vout[output.i].scriptPubKey, dst)) {
|
if (!single_coin && ExtractDestination(output.tx->tx->vout[output.i].scriptPubKey, dst)) {
|
||||||
// Limit output groups to no more than 10 entries, to protect
|
auto it = gmap.find(dst);
|
||||||
// against inadvertently creating a too-large transaction
|
if (it != gmap.end()) {
|
||||||
// when using -avoidpartialspends
|
// Limit output groups to no more than OUTPUT_GROUP_MAX_ENTRIES
|
||||||
if (gmap[dst].m_outputs.size() >= OUTPUT_GROUP_MAX_ENTRIES) {
|
// number of entries, to protect against inadvertently creating
|
||||||
groups.push_back(gmap[dst]);
|
// a too-large transaction when using -avoidpartialspends to
|
||||||
gmap.erase(dst);
|
// prevent breaking consensus or surprising users with a very
|
||||||
|
// high amount of fees.
|
||||||
|
if (it->second.m_outputs.size() >= OUTPUT_GROUP_MAX_ENTRIES) {
|
||||||
|
groups.push_back(it->second);
|
||||||
|
it->second = OutputGroup{};
|
||||||
|
full_groups.insert(dst);
|
||||||
|
}
|
||||||
|
it->second.Insert(input_coin, output.nDepth, output.tx->IsFromMe(ISMINE_ALL), ancestors, descendants);
|
||||||
|
} else {
|
||||||
|
gmap[dst].Insert(input_coin, output.nDepth, output.tx->IsFromMe(ISMINE_ALL), ancestors, descendants);
|
||||||
}
|
}
|
||||||
gmap[dst].Insert(input_coin, output.nDepth, output.tx->IsFromMe(ISMINE_ALL), ancestors, descendants);
|
|
||||||
} else {
|
} else {
|
||||||
groups.emplace_back(input_coin, output.nDepth, output.tx->IsFromMe(ISMINE_ALL), ancestors, descendants);
|
groups.emplace_back(input_coin, output.nDepth, output.tx->IsFromMe(ISMINE_ALL), ancestors, descendants);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!single_coin) {
|
if (!single_coin) {
|
||||||
for (const auto& it : gmap) groups.push_back(it.second);
|
for (auto& it : gmap) {
|
||||||
|
auto& group = it.second;
|
||||||
|
if (full_groups.count(it.first) > 0) {
|
||||||
|
// Make this unattractive as we want coin selection to avoid it if possible
|
||||||
|
group.m_ancestors = max_ancestors - 1;
|
||||||
|
}
|
||||||
|
groups.push_back(group);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return groups;
|
return groups;
|
||||||
}
|
}
|
||||||
|
@ -830,7 +830,7 @@ public:
|
|||||||
bool IsSpentKey(const uint256& hash, unsigned int n) const EXCLUSIVE_LOCKS_REQUIRED(cs_wallet);
|
bool IsSpentKey(const uint256& hash, unsigned int n) const EXCLUSIVE_LOCKS_REQUIRED(cs_wallet);
|
||||||
void SetSpentKeyState(WalletBatch& batch, const uint256& hash, unsigned int n, bool used, std::set<CTxDestination>& tx_destinations) EXCLUSIVE_LOCKS_REQUIRED(cs_wallet);
|
void SetSpentKeyState(WalletBatch& batch, const uint256& hash, unsigned int n, bool used, std::set<CTxDestination>& tx_destinations) EXCLUSIVE_LOCKS_REQUIRED(cs_wallet);
|
||||||
|
|
||||||
std::vector<OutputGroup> GroupOutputs(const std::vector<COutput>& outputs, bool single_coin) const;
|
std::vector<OutputGroup> GroupOutputs(const std::vector<COutput>& outputs, bool single_coin, const size_t max_ancestors) const;
|
||||||
|
|
||||||
bool IsLockedCoin(uint256 hash, unsigned int n) const EXCLUSIVE_LOCKS_REQUIRED(cs_wallet);
|
bool IsLockedCoin(uint256 hash, unsigned int n) const EXCLUSIVE_LOCKS_REQUIRED(cs_wallet);
|
||||||
void LockCoin(const COutPoint& output) EXCLUSIVE_LOCKS_REQUIRED(cs_wallet);
|
void LockCoin(const COutPoint& output) EXCLUSIVE_LOCKS_REQUIRED(cs_wallet);
|
||||||
|
@ -85,15 +85,19 @@ class AvoidReuseTest(BitcoinTestFramework):
|
|||||||
self.sync_all()
|
self.sync_all()
|
||||||
self.test_change_remains_change(self.nodes[1])
|
self.test_change_remains_change(self.nodes[1])
|
||||||
reset_balance(self.nodes[1], self.nodes[0].getnewaddress())
|
reset_balance(self.nodes[1], self.nodes[0].getnewaddress())
|
||||||
self.test_fund_send_fund_senddirty()
|
self.test_sending_from_reused_address_without_avoid_reuse()
|
||||||
reset_balance(self.nodes[1], self.nodes[0].getnewaddress())
|
reset_balance(self.nodes[1], self.nodes[0].getnewaddress())
|
||||||
self.test_fund_send_fund_send("legacy")
|
self.test_sending_from_reused_address_fails("legacy")
|
||||||
reset_balance(self.nodes[1], self.nodes[0].getnewaddress())
|
reset_balance(self.nodes[1], self.nodes[0].getnewaddress())
|
||||||
self.test_fund_send_fund_send("p2sh-segwit")
|
self.test_sending_from_reused_address_fails("p2sh-segwit")
|
||||||
reset_balance(self.nodes[1], self.nodes[0].getnewaddress())
|
reset_balance(self.nodes[1], self.nodes[0].getnewaddress())
|
||||||
self.test_fund_send_fund_send("bech32")
|
self.test_sending_from_reused_address_fails("bech32")
|
||||||
reset_balance(self.nodes[1], self.nodes[0].getnewaddress())
|
reset_balance(self.nodes[1], self.nodes[0].getnewaddress())
|
||||||
self.test_getbalances_used()
|
self.test_getbalances_used()
|
||||||
|
reset_balance(self.nodes[1], self.nodes[0].getnewaddress())
|
||||||
|
self.test_full_destination_group_is_preferred()
|
||||||
|
reset_balance(self.nodes[1], self.nodes[0].getnewaddress())
|
||||||
|
self.test_all_destination_groups_are_used()
|
||||||
|
|
||||||
def test_persistence(self):
|
def test_persistence(self):
|
||||||
'''Test that wallet files persist the avoid_reuse flag.'''
|
'''Test that wallet files persist the avoid_reuse flag.'''
|
||||||
@ -162,13 +166,13 @@ class AvoidReuseTest(BitcoinTestFramework):
|
|||||||
for logical_tx in node.listtransactions():
|
for logical_tx in node.listtransactions():
|
||||||
assert logical_tx.get('address') != changeaddr
|
assert logical_tx.get('address') != changeaddr
|
||||||
|
|
||||||
def test_fund_send_fund_senddirty(self):
|
def test_sending_from_reused_address_without_avoid_reuse(self):
|
||||||
'''
|
'''
|
||||||
Test the same as test_fund_send_fund_send, except send the 10 BTC with
|
Test the same as test_sending_from_reused_address_fails, except send the 10 BTC with
|
||||||
the avoid_reuse flag set to false. This means the 10 BTC send should succeed,
|
the avoid_reuse flag set to false. This means the 10 BTC send should succeed,
|
||||||
where it fails in test_fund_send_fund_send.
|
where it fails in test_sending_from_reused_address_fails.
|
||||||
'''
|
'''
|
||||||
self.log.info("Test fund send fund send dirty")
|
self.log.info("Test sending from reused address with avoid_reuse=false")
|
||||||
|
|
||||||
fundaddr = self.nodes[1].getnewaddress()
|
fundaddr = self.nodes[1].getnewaddress()
|
||||||
retaddr = self.nodes[0].getnewaddress()
|
retaddr = self.nodes[0].getnewaddress()
|
||||||
@ -213,7 +217,7 @@ class AvoidReuseTest(BitcoinTestFramework):
|
|||||||
assert_approx(self.nodes[1].getbalance(), 5, 0.001)
|
assert_approx(self.nodes[1].getbalance(), 5, 0.001)
|
||||||
assert_approx(self.nodes[1].getbalance(avoid_reuse=False), 5, 0.001)
|
assert_approx(self.nodes[1].getbalance(avoid_reuse=False), 5, 0.001)
|
||||||
|
|
||||||
def test_fund_send_fund_send(self, second_addr_type):
|
def test_sending_from_reused_address_fails(self, second_addr_type):
|
||||||
'''
|
'''
|
||||||
Test the simple case where [1] generates a new address A, then
|
Test the simple case where [1] generates a new address A, then
|
||||||
[0] sends 10 BTC to A.
|
[0] sends 10 BTC to A.
|
||||||
@ -222,7 +226,7 @@ class AvoidReuseTest(BitcoinTestFramework):
|
|||||||
[1] tries to spend 10 BTC (fails; dirty).
|
[1] tries to spend 10 BTC (fails; dirty).
|
||||||
[1] tries to spend 4 BTC (succeeds; change address sufficient)
|
[1] tries to spend 4 BTC (succeeds; change address sufficient)
|
||||||
'''
|
'''
|
||||||
self.log.info("Test fund send fund send")
|
self.log.info("Test sending from reused {} address fails".format(second_addr_type))
|
||||||
|
|
||||||
fundaddr = self.nodes[1].getnewaddress(label="", address_type="legacy")
|
fundaddr = self.nodes[1].getnewaddress(label="", address_type="legacy")
|
||||||
retaddr = self.nodes[0].getnewaddress()
|
retaddr = self.nodes[0].getnewaddress()
|
||||||
@ -313,5 +317,66 @@ class AvoidReuseTest(BitcoinTestFramework):
|
|||||||
assert_unspent(self.nodes[1], total_count=2, total_sum=6, reused_count=1, reused_sum=1)
|
assert_unspent(self.nodes[1], total_count=2, total_sum=6, reused_count=1, reused_sum=1)
|
||||||
assert_balances(self.nodes[1], mine={"used": 1, "trusted": 5})
|
assert_balances(self.nodes[1], mine={"used": 1, "trusted": 5})
|
||||||
|
|
||||||
|
def test_full_destination_group_is_preferred(self):
|
||||||
|
'''
|
||||||
|
Test the case where [1] only has 11 outputs of 1 BTC in the same reused
|
||||||
|
address and tries to send a small payment of 0.5 BTC. The wallet
|
||||||
|
should use 10 outputs from the reused address as inputs and not a
|
||||||
|
single 1 BTC input, in order to join several outputs from the reused
|
||||||
|
address.
|
||||||
|
'''
|
||||||
|
self.log.info("Test that full destination groups are preferred in coin selection")
|
||||||
|
|
||||||
|
# Node under test should be empty
|
||||||
|
assert_equal(self.nodes[1].getbalance(avoid_reuse=False), 0)
|
||||||
|
|
||||||
|
new_addr = self.nodes[1].getnewaddress()
|
||||||
|
ret_addr = self.nodes[0].getnewaddress()
|
||||||
|
|
||||||
|
# Send 11 outputs of 1 BTC to the same, reused address in the wallet
|
||||||
|
for _ in range(11):
|
||||||
|
self.nodes[0].sendtoaddress(new_addr, 1)
|
||||||
|
|
||||||
|
self.nodes[0].generate(1)
|
||||||
|
self.sync_all()
|
||||||
|
|
||||||
|
# Sending a transaction that is smaller than each one of the
|
||||||
|
# available outputs
|
||||||
|
txid = self.nodes[1].sendtoaddress(address=ret_addr, amount=0.5)
|
||||||
|
inputs = self.nodes[1].getrawtransaction(txid, 1)["vin"]
|
||||||
|
|
||||||
|
# The transaction should use 10 inputs exactly
|
||||||
|
assert_equal(len(inputs), 10)
|
||||||
|
|
||||||
|
def test_all_destination_groups_are_used(self):
|
||||||
|
'''
|
||||||
|
Test the case where [1] only has 22 outputs of 1 BTC in the same reused
|
||||||
|
address and tries to send a payment of 20.5 BTC. The wallet
|
||||||
|
should use all 22 outputs from the reused address as inputs.
|
||||||
|
'''
|
||||||
|
self.log.info("Test that all destination groups are used")
|
||||||
|
|
||||||
|
# Node under test should be empty
|
||||||
|
assert_equal(self.nodes[1].getbalance(avoid_reuse=False), 0)
|
||||||
|
|
||||||
|
new_addr = self.nodes[1].getnewaddress()
|
||||||
|
ret_addr = self.nodes[0].getnewaddress()
|
||||||
|
|
||||||
|
# Send 22 outputs of 1 BTC to the same, reused address in the wallet
|
||||||
|
for _ in range(22):
|
||||||
|
self.nodes[0].sendtoaddress(new_addr, 1)
|
||||||
|
|
||||||
|
self.nodes[0].generate(1)
|
||||||
|
self.sync_all()
|
||||||
|
|
||||||
|
# Sending a transaction that needs to use the full groups
|
||||||
|
# of 10 inputs but also the incomplete group of 2 inputs.
|
||||||
|
txid = self.nodes[1].sendtoaddress(address=ret_addr, amount=20.5)
|
||||||
|
inputs = self.nodes[1].getrawtransaction(txid, 1)["vin"]
|
||||||
|
|
||||||
|
# The transaction should use 22 inputs exactly
|
||||||
|
assert_equal(len(inputs), 22)
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
AvoidReuseTest().main()
|
AvoidReuseTest().main()
|
||||||
|
Loading…
Reference in New Issue
Block a user