Branch data Line data Source code
1 : : // Copyright (c) 2024-present The Bitcoin Core developers
2 : : // Distributed under the MIT software license, see the accompanying
3 : : // file COPYING or http://www.opensource.org/licenses/mit-license.php.
4 : :
5 : : #ifndef BITCOIN_TEST_FUZZ_UTIL_WALLET_H
6 : : #define BITCOIN_TEST_FUZZ_UTIL_WALLET_H
7 : :
8 : : #include <test/fuzz/FuzzedDataProvider.h>
9 : : #include <test/fuzz/fuzz.h>
10 : : #include <test/fuzz/util.h>
11 : : #include <policy/policy.h>
12 : : #include <wallet/coincontrol.h>
13 : : #include <wallet/fees.h>
14 : : #include <wallet/spend.h>
15 : : #include <wallet/test/util.h>
16 : : #include <wallet/wallet.h>
17 : :
18 : : namespace wallet {
19 : :
20 : : /**
21 : : * Wraps a descriptor wallet for fuzzing.
22 : : */
23 [ + - + - : 7902 : struct FuzzedWallet {
- - ][ + -
+ - - - -
- ]
24 : : std::shared_ptr<CWallet> wallet;
25 : 4656 : FuzzedWallet(interfaces::Chain& chain, const std::string& name, const std::string& seed_insecure)
26 [ + - ]: 4656 : {
27 [ + - + - : 4656 : wallet = std::make_shared<CWallet>(&chain, name, CreateMockableWalletDatabase());
- + ]
28 : 4656 : {
29 [ + - ]: 4656 : LOCK(wallet->cs_wallet);
30 [ + - ]: 4656 : wallet->SetWalletFlag(WALLET_FLAG_DESCRIPTORS);
31 [ + - + - : 4656 : auto height{*Assert(chain.getHeight())};
+ - ]
32 [ + - + - ]: 4656 : wallet->SetLastBlockProcessed(height, chain.getBlockHash(height));
33 : 0 : }
34 [ + - ]: 4656 : wallet->m_keypool_size = 1; // Avoid timeout in TopUp()
35 [ + - - + ]: 4656 : assert(wallet->IsWalletFlagSet(WALLET_FLAG_DESCRIPTORS));
36 [ + - ]: 4656 : ImportDescriptors(seed_insecure);
37 [ - - ]: 4656 : }
38 : 4656 : void ImportDescriptors(const std::string& seed_insecure)
39 : : {
40 : 4656 : const std::vector<std::string> DESCS{
41 : : "pkh(%s/%s/*)",
42 : : "sh(wpkh(%s/%s/*))",
43 : : "tr(%s/%s/*)",
44 : : "wpkh(%s/%s/*)",
45 [ + - + + : 46560 : };
+ - + + -
- - - ]
46 : :
47 [ + + ]: 23280 : for (const std::string& desc_fmt : DESCS) {
48 [ + + ]: 55872 : for (bool internal : {true, false}) {
49 [ + - + - ]: 37248 : const auto descriptor{strprintf(tfm::RuntimeFormat{desc_fmt}, "[5aa9973a/66h/4h/2h]" + seed_insecure, int{internal})};
50 : :
51 : 37248 : FlatSigningProvider keys;
52 [ + - ]: 37248 : std::string error;
53 [ + - ]: 74496 : auto parsed_desc = std::move(Parse(descriptor, keys, error, /*require_checksum=*/false).at(0));
54 [ - + ]: 37248 : assert(parsed_desc);
55 [ - + ]: 37248 : assert(error.empty());
56 [ + - - + ]: 37248 : assert(parsed_desc->IsRange());
57 [ + - - + ]: 37248 : assert(parsed_desc->IsSingleType());
58 [ - + ]: 37248 : assert(!keys.keys.empty());
59 [ + - + - ]: 37248 : WalletDescriptor w_desc{std::move(parsed_desc), /*creation_time=*/0, /*range_start=*/0, /*range_end=*/1, /*next_index=*/0};
60 [ + - - + ]: 37248 : assert(!wallet->GetDescriptorScriptPubKeyMan(w_desc));
61 [ + - ]: 37248 : LOCK(wallet->cs_wallet);
62 [ + - + - ]: 37248 : auto spk_manager{wallet->AddWalletDescriptor(w_desc, keys, /*label=*/"", internal)};
63 [ - + ]: 37248 : assert(spk_manager);
64 [ + - + - : 37248 : wallet->AddActiveScriptPubKeyMan(spk_manager->GetID(), *Assert(w_desc.descriptor->GetOutputType()), internal);
+ - + - ]
65 : 37248 : }
66 : : }
67 : 4656 : }
68 : 98696 : CTxDestination GetDestination(FuzzedDataProvider& fuzzed_data_provider)
69 : : {
70 : 98696 : auto type{fuzzed_data_provider.PickValueInArray(OUTPUT_TYPES)};
71 [ + + ]: 98696 : if (fuzzed_data_provider.ConsumeBool()) {
72 [ + - + - : 116618 : return *Assert(wallet->GetNewDestination(type, ""));
+ - ]
73 : : } else {
74 [ + - + - ]: 80774 : return *Assert(wallet->GetNewChangeDestination(type));
75 : : }
76 : : }
77 [ + - ]: 111946 : CScript GetScriptPubKey(FuzzedDataProvider& fuzzed_data_provider) { return GetScriptForDestination(GetDestination(fuzzed_data_provider)); }
78 : 31084 : void FundTx(FuzzedDataProvider& fuzzed_data_provider, CMutableTransaction tx)
79 : : {
80 : : // The fee of "tx" is 0, so this is the total input and output amount
81 : 31084 : const CAmount total_amt{
82 : 143030 : std::accumulate(tx.vout.begin(), tx.vout.end(), CAmount{}, [](CAmount t, const CTxOut& out) { return t + out.nValue; })};
83 [ + - ]: 31084 : const uint32_t tx_size(GetVirtualTransactionSize(CTransaction{tx}));
84 : 31084 : std::set<int> subtract_fee_from_outputs;
85 [ + + ]: 31084 : if (fuzzed_data_provider.ConsumeBool()) {
86 [ + + ]: 84399 : for (size_t i{}; i < tx.vout.size(); ++i) {
87 [ + + ]: 72386 : if (fuzzed_data_provider.ConsumeBool()) {
88 [ + - ]: 52312 : subtract_fee_from_outputs.insert(i);
89 : : }
90 : : }
91 : : }
92 : 31084 : std::vector<CRecipient> recipients;
93 [ + + ]: 143030 : for (size_t idx = 0; idx < tx.vout.size(); idx++) {
94 [ + - ]: 111946 : const CTxOut& tx_out = tx.vout[idx];
95 : 111946 : CTxDestination dest;
96 [ + - ]: 111946 : ExtractDestination(tx_out.scriptPubKey, dest);
97 [ + - ]: 223892 : CRecipient recipient = {dest, tx_out.nValue, subtract_fee_from_outputs.count(idx) == 1};
98 [ + - ]: 111946 : recipients.push_back(recipient);
99 : 111946 : }
100 [ + - ]: 31084 : CCoinControl coin_control;
101 : 31084 : coin_control.m_allow_other_inputs = fuzzed_data_provider.ConsumeBool();
102 [ + - ]: 31084 : CallOneOf(
103 : 24410 : fuzzed_data_provider, [&] { coin_control.destChange = GetDestination(fuzzed_data_provider); },
104 [ - + ]: 5214 : [&] { coin_control.m_change_type.emplace(fuzzed_data_provider.PickValueInArray(OUTPUT_TYPES)); },
105 : : [&] { /* no op (leave uninitialized) */ });
106 : 31084 : coin_control.fAllowWatchOnly = fuzzed_data_provider.ConsumeBool();
107 : 31084 : coin_control.m_include_unsafe_inputs = fuzzed_data_provider.ConsumeBool();
108 : 31084 : {
109 : 31084 : auto& r{coin_control.m_signal_bip125_rbf};
110 : 31084 : CallOneOf(
111 [ - + ]: 31084 : fuzzed_data_provider, [&] { r = true; }, [&] { r = false; }, [&] { r = std::nullopt; });
112 : : }
113 : 31084 : coin_control.m_feerate = CFeeRate{
114 : : // A fee of this range should cover all cases
115 : 31084 : fuzzed_data_provider.ConsumeIntegralInRange<CAmount>(0, 2 * total_amt),
116 : : tx_size,
117 [ + - ]: 31084 : };
118 [ + + ]: 31084 : if (fuzzed_data_provider.ConsumeBool()) {
119 [ + - ]: 20176 : *coin_control.m_feerate += GetMinimumFeeRate(*wallet, coin_control, nullptr);
120 : : }
121 : 31084 : coin_control.fOverrideFeeRate = fuzzed_data_provider.ConsumeBool();
122 : : // Add solving data (m_external_provider and SelectExternal)?
123 : :
124 : 31084 : int change_position{fuzzed_data_provider.ConsumeIntegralInRange<int>(-1, tx.vout.size() - 1)};
125 : 31084 : bilingual_str error;
126 : : // Clear tx.vout since it is not meant to be used now that we are passing outputs directly.
127 : : // This sets us up for a future PR to completely remove tx from the function signature in favor of passing inputs directly
128 : 31084 : tx.vout.clear();
129 [ + - + - ]: 62168 : (void)FundTransaction(*wallet, tx, recipients, change_position, /*lockUnspents=*/false, coin_control);
130 : 31084 : }
131 : : };
132 : : }
133 : :
134 : : #endif // BITCOIN_TEST_FUZZ_UTIL_WALLET_H
|