LCOV - code coverage report
Current view: top level - src/test/fuzz/util - wallet.h (source / functions) Coverage Total Hit
Test: fuzz_coverage.info Lines: 98.8 % 81 80
Test Date: 2025-01-19 04:06:55 Functions: 100.0 % 7 7
Branches: 52.5 % 160 84

             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
        

Generated by: LCOV version 2.0-1