LCOV - code coverage report
Current view: top level - src/wallet - interfaces.cpp (source / functions) Hit Total Coverage
Test: fuzz_coverage.info Lines: 15 444 3.4 %
Date: 2023-09-26 12:08:55 Functions: 8 117 6.8 %

          Line data    Source code
       1             : // Copyright (c) 2018-2022 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             : #include <interfaces/wallet.h>
       6             : 
       7             : #include <common/args.h>
       8             : #include <consensus/amount.h>
       9             : #include <interfaces/chain.h>
      10             : #include <interfaces/handler.h>
      11             : #include <policy/fees.h>
      12             : #include <primitives/transaction.h>
      13             : #include <rpc/server.h>
      14             : #include <support/allocators/secure.h>
      15             : #include <sync.h>
      16             : #include <uint256.h>
      17           2 : #include <util/check.h>
      18           2 : #include <util/translation.h>
      19             : #include <util/ui_change_type.h>
      20             : #include <wallet/coincontrol.h>
      21             : #include <wallet/context.h>
      22             : #include <wallet/feebumper.h>
      23             : #include <wallet/fees.h>
      24             : #include <wallet/types.h>
      25             : #include <wallet/load.h>
      26             : #include <wallet/receive.h>
      27           2 : #include <wallet/rpc/wallet.h>
      28             : #include <wallet/spend.h>
      29             : #include <wallet/wallet.h>
      30             : 
      31             : #include <memory>
      32             : #include <string>
      33             : #include <utility>
      34             : #include <vector>
      35             : 
      36             : using interfaces::Chain;
      37             : using interfaces::FoundBlock;
      38             : using interfaces::Handler;
      39             : using interfaces::MakeSignalHandler;
      40             : using interfaces::Wallet;
      41             : using interfaces::WalletAddress;
      42             : using interfaces::WalletBalances;
      43             : using interfaces::WalletLoader;
      44             : using interfaces::WalletMigrationResult;
      45             : using interfaces::WalletOrderForm;
      46             : using interfaces::WalletTx;
      47             : using interfaces::WalletTxOut;
      48             : using interfaces::WalletTxStatus;
      49             : using interfaces::WalletValueMap;
      50             : 
      51             : namespace wallet {
      52             : // All members of the classes in this namespace are intentionally public, as the
      53           0 : // classes themselves are private.
      54             : namespace {
      55             : //! Construct wallet tx struct.
      56           0 : WalletTx MakeWalletTx(CWallet& wallet, const CWalletTx& wtx)
      57             : {
      58           0 :     LOCK(wallet.cs_wallet);
      59           0 :     WalletTx result;
      60           0 :     result.tx = wtx.tx;
      61           0 :     result.txin_is_mine.reserve(wtx.tx->vin.size());
      62           0 :     for (const auto& txin : wtx.tx->vin) {
      63           0 :         result.txin_is_mine.emplace_back(InputIsMine(wallet, txin));
      64             :     }
      65           0 :     result.txout_is_mine.reserve(wtx.tx->vout.size());
      66           0 :     result.txout_address.reserve(wtx.tx->vout.size());
      67           0 :     result.txout_address_is_mine.reserve(wtx.tx->vout.size());
      68           0 :     for (const auto& txout : wtx.tx->vout) {
      69           0 :         result.txout_is_mine.emplace_back(wallet.IsMine(txout));
      70           0 :         result.txout_address.emplace_back();
      71           0 :         result.txout_address_is_mine.emplace_back(ExtractDestination(txout.scriptPubKey, result.txout_address.back()) ?
      72           0 :                                                       wallet.IsMine(result.txout_address.back()) :
      73             :                                                       ISMINE_NO);
      74           2 :     }
      75           0 :     result.credit = CachedTxGetCredit(wallet, wtx, ISMINE_ALL);
      76           0 :     result.debit = CachedTxGetDebit(wallet, wtx, ISMINE_ALL);
      77           0 :     result.change = CachedTxGetChange(wallet, wtx);
      78           0 :     result.time = wtx.GetTxTime();
      79           0 :     result.value_map = wtx.mapValue;
      80           0 :     result.is_coinbase = wtx.IsCoinBase();
      81           0 :     return result;
      82           0 : }
      83             : 
      84             : //! Construct wallet tx status struct.
      85           0 : WalletTxStatus MakeWalletTxStatus(const CWallet& wallet, const CWalletTx& wtx)
      86             :     EXCLUSIVE_LOCKS_REQUIRED(wallet.cs_wallet)
      87             : {
      88           0 :     AssertLockHeld(wallet.cs_wallet);
      89             : 
      90             :     WalletTxStatus result;
      91           2 :     result.block_height =
      92           0 :         wtx.state<TxStateConfirmed>() ? wtx.state<TxStateConfirmed>()->confirmed_block_height :
      93           0 :         wtx.state<TxStateConflicted>() ? wtx.state<TxStateConflicted>()->conflicting_block_height :
      94           0 :         std::numeric_limits<int>::max();
      95           0 :     result.blocks_to_maturity = wallet.GetTxBlocksToMaturity(wtx);
      96           0 :     result.depth_in_main_chain = wallet.GetTxDepthInMainChain(wtx);
      97           0 :     result.time_received = wtx.nTimeReceived;
      98           0 :     result.lock_time = wtx.tx->nLockTime;
      99           2 :     result.is_trusted = CachedTxIsTrusted(wallet, wtx);
     100           0 :     result.is_abandoned = wtx.isAbandoned();
     101           0 :     result.is_coinbase = wtx.IsCoinBase();
     102           0 :     result.is_in_main_chain = wallet.IsTxInMainChain(wtx);
     103           0 :     return result;
     104             : }
     105             : 
     106             : //! Construct wallet TxOut struct.
     107           0 : WalletTxOut MakeWalletTxOut(const CWallet& wallet,
     108             :     const CWalletTx& wtx,
     109             :     int n,
     110             :     int depth) EXCLUSIVE_LOCKS_REQUIRED(wallet.cs_wallet)
     111             : {
     112           0 :     WalletTxOut result;
     113           0 :     result.txout = wtx.tx->vout[n];
     114           0 :     result.time = wtx.GetTxTime();
     115           0 :     result.depth_in_main_chain = depth;
     116           0 :     result.is_spent = wallet.IsSpent(COutPoint(wtx.GetHash(), n));
     117           0 :     return result;
     118           0 : }
     119             : 
     120           0 : WalletTxOut MakeWalletTxOut(const CWallet& wallet,
     121             :     const COutput& output) EXCLUSIVE_LOCKS_REQUIRED(wallet.cs_wallet)
     122             : {
     123           0 :     WalletTxOut result;
     124           0 :     result.txout = output.txout;
     125           0 :     result.time = output.time;
     126           0 :     result.depth_in_main_chain = output.depth;
     127           0 :     result.is_spent = wallet.IsSpent(output.outpoint);
     128           0 :     return result;
     129           0 : }
     130             : 
     131             : class WalletImpl : public Wallet
     132             : {
     133             : public:
     134           0 :     explicit WalletImpl(WalletContext& context, const std::shared_ptr<CWallet>& wallet) : m_context(context), m_wallet(wallet) {}
     135             : 
     136           0 :     bool encryptWallet(const SecureString& wallet_passphrase) override
     137             :     {
     138           0 :         return m_wallet->EncryptWallet(wallet_passphrase);
     139             :     }
     140           0 :     bool isCrypted() override { return m_wallet->IsCrypted(); }
     141           0 :     bool lock() override { return m_wallet->Lock(); }
     142           0 :     bool unlock(const SecureString& wallet_passphrase) override { return m_wallet->Unlock(wallet_passphrase); }
     143           0 :     bool isLocked() override { return m_wallet->IsLocked(); }
     144           0 :     bool changeWalletPassphrase(const SecureString& old_wallet_passphrase,
     145             :         const SecureString& new_wallet_passphrase) override
     146             :     {
     147           0 :         return m_wallet->ChangeWalletPassphrase(old_wallet_passphrase, new_wallet_passphrase);
     148             :     }
     149           0 :     void abortRescan() override { m_wallet->AbortRescan(); }
     150           0 :     bool backupWallet(const std::string& filename) override { return m_wallet->BackupWallet(filename); }
     151           0 :     std::string getWalletName() override { return m_wallet->GetName(); }
     152           0 :     util::Result<CTxDestination> getNewDestination(const OutputType type, const std::string& label) override
     153             :     {
     154           0 :         LOCK(m_wallet->cs_wallet);
     155           0 :         return m_wallet->GetNewDestination(type, label);
     156           0 :     }
     157           0 :     bool getPubKey(const CScript& script, const CKeyID& address, CPubKey& pub_key) override
     158             :     {
     159           0 :         std::unique_ptr<SigningProvider> provider = m_wallet->GetSolvingProvider(script);
     160           0 :         if (provider) {
     161           0 :             return provider->GetPubKey(address, pub_key);
     162             :         }
     163           2 :         return false;
     164           2 :     }
     165           2 :     SigningResult signMessage(const std::string& message, const PKHash& pkhash, std::string& str_sig) override
     166           2 :     {
     167           2 :         return m_wallet->SignMessage(message, pkhash, str_sig);
     168           2 :     }
     169           2 :     bool isSpendable(const CTxDestination& dest) override
     170           2 :     {
     171           0 :         LOCK(m_wallet->cs_wallet);
     172           0 :         return m_wallet->IsMine(dest) & ISMINE_SPENDABLE;
     173           0 :     }
     174           0 :     bool haveWatchOnly() override
     175             :     {
     176           0 :         auto spk_man = m_wallet->GetLegacyScriptPubKeyMan();
     177           0 :         if (spk_man) {
     178           0 :             return spk_man->HaveWatchOnly();
     179             :         }
     180           0 :         return false;
     181           0 :     };
     182           0 :     bool setAddressBook(const CTxDestination& dest, const std::string& name, const std::optional<AddressPurpose>& purpose) override
     183             :     {
     184           0 :         return m_wallet->SetAddressBook(dest, name, purpose);
     185             :     }
     186           0 :     bool delAddressBook(const CTxDestination& dest) override
     187             :     {
     188           0 :         return m_wallet->DelAddressBook(dest);
     189             :     }
     190           0 :     bool getAddress(const CTxDestination& dest,
     191             :         std::string* name,
     192             :         isminetype* is_mine,
     193             :         AddressPurpose* purpose) override
     194             :     {
     195           0 :         LOCK(m_wallet->cs_wallet);
     196           0 :         const auto& entry = m_wallet->FindAddressBookEntry(dest, /*allow_change=*/false);
     197           0 :         if (!entry) return false; // addr not found
     198           0 :         if (name) {
     199           0 :             *name = entry->GetLabel();
     200           0 :         }
     201           0 :         std::optional<isminetype> dest_is_mine;
     202           0 :         if (is_mine || purpose) {
     203           0 :             dest_is_mine = m_wallet->IsMine(dest);
     204           0 :         }
     205           0 :         if (is_mine) {
     206           0 :             *is_mine = *dest_is_mine;
     207           0 :         }
     208           0 :         if (purpose) {
     209             :             // In very old wallets, address purpose may not be recorded so we derive it from IsMine
     210           0 :             *purpose = entry->purpose.value_or(*dest_is_mine ? AddressPurpose::RECEIVE : AddressPurpose::SEND);
     211           0 :         }
     212           0 :         return true;
     213           0 :     }
     214           0 :     std::vector<WalletAddress> getAddresses() const override
     215             :     {
     216           0 :         LOCK(m_wallet->cs_wallet);
     217           0 :         std::vector<WalletAddress> result;
     218           0 :         m_wallet->ForEachAddrBookEntry([&](const CTxDestination& dest, const std::string& label, bool is_change, const std::optional<AddressPurpose>& purpose) EXCLUSIVE_LOCKS_REQUIRED(m_wallet->cs_wallet) {
     219           0 :             if (is_change) return;
     220           0 :             isminetype is_mine = m_wallet->IsMine(dest);
     221             :             // In very old wallets, address purpose may not be recorded so we derive it from IsMine
     222           0 :             result.emplace_back(dest, is_mine, purpose.value_or(is_mine ? AddressPurpose::RECEIVE : AddressPurpose::SEND), label);
     223           0 :         });
     224           0 :         return result;
     225           0 :     }
     226           0 :     std::vector<std::string> getAddressReceiveRequests() override {
     227           0 :         LOCK(m_wallet->cs_wallet);
     228           0 :         return m_wallet->GetAddressReceiveRequests();
     229           0 :     }
     230           0 :     bool setAddressReceiveRequest(const CTxDestination& dest, const std::string& id, const std::string& value) override {
     231             :         // Note: The setAddressReceiveRequest interface used by the GUI to store
     232             :         // receive requests is a little awkward and could be improved in the
     233             :         // future:
     234             :         //
     235             :         // - The same method is used to save requests and erase them, but
     236             :         //   having separate methods could be clearer and prevent bugs.
     237             :         //
     238             :         // - Request ids are passed as strings even though they are generated as
     239             :         //   integers.
     240             :         //
     241             :         // - Multiple requests can be stored for the same address, but it might
     242             :         //   be better to only allow one request or only keep the current one.
     243           0 :         LOCK(m_wallet->cs_wallet);
     244           0 :         WalletBatch batch{m_wallet->GetDatabase()};
     245           0 :         return value.empty() ? m_wallet->EraseAddressReceiveRequest(batch, dest, id)
     246           0 :                              : m_wallet->SetAddressReceiveRequest(batch, dest, id, value);
     247           0 :     }
     248           0 :     bool displayAddress(const CTxDestination& dest) override
     249             :     {
     250           0 :         LOCK(m_wallet->cs_wallet);
     251           0 :         return m_wallet->DisplayAddress(dest);
     252           0 :     }
     253           0 :     bool lockCoin(const COutPoint& output, const bool write_to_db) override
     254             :     {
     255           0 :         LOCK(m_wallet->cs_wallet);
     256           0 :         std::unique_ptr<WalletBatch> batch = write_to_db ? std::make_unique<WalletBatch>(m_wallet->GetDatabase()) : nullptr;
     257           0 :         return m_wallet->LockCoin(output, batch.get());
     258           0 :     }
     259           0 :     bool unlockCoin(const COutPoint& output) override
     260             :     {
     261           0 :         LOCK(m_wallet->cs_wallet);
     262           0 :         std::unique_ptr<WalletBatch> batch = std::make_unique<WalletBatch>(m_wallet->GetDatabase());
     263           0 :         return m_wallet->UnlockCoin(output, batch.get());
     264           0 :     }
     265           0 :     bool isLockedCoin(const COutPoint& output) override
     266             :     {
     267           0 :         LOCK(m_wallet->cs_wallet);
     268           0 :         return m_wallet->IsLockedCoin(output);
     269           0 :     }
     270           2 :     void listLockedCoins(std::vector<COutPoint>& outputs) override
     271             :     {
     272           0 :         LOCK(m_wallet->cs_wallet);
     273           0 :         return m_wallet->ListLockedCoins(outputs);
     274           0 :     }
     275           0 :     util::Result<CTransactionRef> createTransaction(const std::vector<CRecipient>& recipients,
     276             :         const CCoinControl& coin_control,
     277             :         bool sign,
     278             :         int& change_pos,
     279             :         CAmount& fee) override
     280             :     {
     281           0 :         LOCK(m_wallet->cs_wallet);
     282           0 :         auto res = CreateTransaction(*m_wallet, recipients, change_pos,
     283           0 :                                      coin_control, sign);
     284           0 :         if (!res) return util::Error{util::ErrorString(res)};
     285           0 :         const auto& txr = *res;
     286           0 :         fee = txr.fee;
     287           0 :         change_pos = txr.change_pos;
     288             : 
     289           0 :         return txr.tx;
     290           0 :     }
     291           0 :     void commitTransaction(CTransactionRef tx,
     292             :         WalletValueMap value_map,
     293             :         WalletOrderForm order_form) override
     294             :     {
     295           0 :         LOCK(m_wallet->cs_wallet);
     296           0 :         m_wallet->CommitTransaction(std::move(tx), std::move(value_map), std::move(order_form));
     297           0 :     }
     298           0 :     bool transactionCanBeAbandoned(const uint256& txid) override { return m_wallet->TransactionCanBeAbandoned(txid); }
     299           0 :     bool abandonTransaction(const uint256& txid) override
     300             :     {
     301           0 :         LOCK(m_wallet->cs_wallet);
     302           0 :         return m_wallet->AbandonTransaction(txid);
     303           0 :     }
     304           0 :     bool transactionCanBeBumped(const uint256& txid) override
     305             :     {
     306           0 :         return feebumper::TransactionCanBeBumped(*m_wallet.get(), txid);
     307             :     }
     308           0 :     bool createBumpTransaction(const uint256& txid,
     309             :         const CCoinControl& coin_control,
     310             :         std::vector<bilingual_str>& errors,
     311             :         CAmount& old_fee,
     312             :         CAmount& new_fee,
     313             :         CMutableTransaction& mtx) override
     314             :     {
     315           0 :         std::vector<CTxOut> outputs; // just an empty list of new recipients for now
     316           0 :         return feebumper::CreateRateBumpTransaction(*m_wallet.get(), txid, coin_control, errors, old_fee, new_fee, mtx, /* require_mine= */ true, outputs) == feebumper::Result::OK;
     317           0 :     }
     318           0 :     bool signBumpTransaction(CMutableTransaction& mtx) override { return feebumper::SignTransaction(*m_wallet.get(), mtx); }
     319           0 :     bool commitBumpTransaction(const uint256& txid,
     320             :         CMutableTransaction&& mtx,
     321             :         std::vector<bilingual_str>& errors,
     322             :         uint256& bumped_txid) override
     323             :     {
     324           0 :         return feebumper::CommitTransaction(*m_wallet.get(), txid, std::move(mtx), errors, bumped_txid) ==
     325             :                feebumper::Result::OK;
     326             :     }
     327           0 :     CTransactionRef getTx(const uint256& txid) override
     328             :     {
     329           0 :         LOCK(m_wallet->cs_wallet);
     330           0 :         auto mi = m_wallet->mapWallet.find(txid);
     331           0 :         if (mi != m_wallet->mapWallet.end()) {
     332           0 :             return mi->second.tx;
     333             :         }
     334           0 :         return {};
     335           0 :     }
     336           0 :     WalletTx getWalletTx(const uint256& txid) override
     337             :     {
     338           0 :         LOCK(m_wallet->cs_wallet);
     339           0 :         auto mi = m_wallet->mapWallet.find(txid);
     340           0 :         if (mi != m_wallet->mapWallet.end()) {
     341           0 :             return MakeWalletTx(*m_wallet, mi->second);
     342             :         }
     343           0 :         return {};
     344           0 :     }
     345           0 :     std::set<WalletTx> getWalletTxs() override
     346             :     {
     347           0 :         LOCK(m_wallet->cs_wallet);
     348           0 :         std::set<WalletTx> result;
     349           0 :         for (const auto& entry : m_wallet->mapWallet) {
     350           0 :             result.emplace(MakeWalletTx(*m_wallet, entry.second));
     351             :         }
     352           0 :         return result;
     353           0 :     }
     354           0 :     bool tryGetTxStatus(const uint256& txid,
     355             :         interfaces::WalletTxStatus& tx_status,
     356             :         int& num_blocks,
     357             :         int64_t& block_time) override
     358             :     {
     359           0 :         TRY_LOCK(m_wallet->cs_wallet, locked_wallet);
     360           0 :         if (!locked_wallet) {
     361           0 :             return false;
     362             :         }
     363           0 :         auto mi = m_wallet->mapWallet.find(txid);
     364           0 :         if (mi == m_wallet->mapWallet.end()) {
     365           0 :             return false;
     366             :         }
     367           0 :         num_blocks = m_wallet->GetLastBlockHeight();
     368           0 :         block_time = -1;
     369           0 :         CHECK_NONFATAL(m_wallet->chain().findBlock(m_wallet->GetLastBlockHash(), FoundBlock().time(block_time)));
     370           0 :         tx_status = MakeWalletTxStatus(*m_wallet, mi->second);
     371           0 :         return true;
     372           0 :     }
     373           0 :     WalletTx getWalletTxDetails(const uint256& txid,
     374             :         WalletTxStatus& tx_status,
     375             :         WalletOrderForm& order_form,
     376             :         bool& in_mempool,
     377             :         int& num_blocks) override
     378             :     {
     379           0 :         LOCK(m_wallet->cs_wallet);
     380           0 :         auto mi = m_wallet->mapWallet.find(txid);
     381           0 :         if (mi != m_wallet->mapWallet.end()) {
     382           0 :             num_blocks = m_wallet->GetLastBlockHeight();
     383           0 :             in_mempool = mi->second.InMempool();
     384           0 :             order_form = mi->second.vOrderForm;
     385           0 :             tx_status = MakeWalletTxStatus(*m_wallet, mi->second);
     386           0 :             return MakeWalletTx(*m_wallet, mi->second);
     387             :         }
     388           0 :         return {};
     389           0 :     }
     390           0 :     TransactionError fillPSBT(int sighash_type,
     391             :         bool sign,
     392             :         bool bip32derivs,
     393             :         size_t* n_signed,
     394             :         PartiallySignedTransaction& psbtx,
     395             :         bool& complete) override
     396             :     {
     397           0 :         return m_wallet->FillPSBT(psbtx, complete, sighash_type, sign, bip32derivs, n_signed);
     398             :     }
     399           0 :     WalletBalances getBalances() override
     400             :     {
     401           0 :         const auto bal = GetBalance(*m_wallet);
     402           0 :         WalletBalances result;
     403           0 :         result.balance = bal.m_mine_trusted;
     404           0 :         result.unconfirmed_balance = bal.m_mine_untrusted_pending;
     405           0 :         result.immature_balance = bal.m_mine_immature;
     406           0 :         result.have_watch_only = haveWatchOnly();
     407           0 :         if (result.have_watch_only) {
     408           0 :             result.watch_only_balance = bal.m_watchonly_trusted;
     409           0 :             result.unconfirmed_watch_only_balance = bal.m_watchonly_untrusted_pending;
     410           0 :             result.immature_watch_only_balance = bal.m_watchonly_immature;
     411           0 :         }
     412           0 :         return result;
     413             :     }
     414           0 :     bool tryGetBalances(WalletBalances& balances, uint256& block_hash) override
     415             :     {
     416           0 :         TRY_LOCK(m_wallet->cs_wallet, locked_wallet);
     417           0 :         if (!locked_wallet) {
     418           0 :             return false;
     419             :         }
     420           0 :         block_hash = m_wallet->GetLastBlockHash();
     421           0 :         balances = getBalances();
     422           0 :         return true;
     423           0 :     }
     424           0 :     CAmount getBalance() override { return GetBalance(*m_wallet).m_mine_trusted; }
     425           0 :     CAmount getAvailableBalance(const CCoinControl& coin_control) override
     426             :     {
     427           0 :         LOCK(m_wallet->cs_wallet);
     428           0 :         CAmount total_amount = 0;
     429             :         // Fetch selected coins total amount
     430           0 :         if (coin_control.HasSelected()) {
     431           0 :             FastRandomContext rng{};
     432           0 :             CoinSelectionParams params(rng);
     433             :             // Note: for now, swallow any error.
     434           0 :             if (auto res = FetchSelectedInputs(*m_wallet, coin_control, params)) {
     435           0 :                 total_amount += res->total_amount;
     436           0 :             }
     437           0 :         }
     438             : 
     439             :         // And fetch the wallet available coins
     440           0 :         if (coin_control.m_allow_other_inputs) {
     441           0 :             total_amount += AvailableCoins(*m_wallet, &coin_control).GetTotalAmount();
     442           0 :         }
     443             : 
     444           0 :         return total_amount;
     445           0 :     }
     446           0 :     isminetype txinIsMine(const CTxIn& txin) override
     447             :     {
     448           0 :         LOCK(m_wallet->cs_wallet);
     449           0 :         return InputIsMine(*m_wallet, txin);
     450           0 :     }
     451           0 :     isminetype txoutIsMine(const CTxOut& txout) override
     452             :     {
     453           0 :         LOCK(m_wallet->cs_wallet);
     454           0 :         return m_wallet->IsMine(txout);
     455           0 :     }
     456           0 :     CAmount getDebit(const CTxIn& txin, isminefilter filter) override
     457             :     {
     458           0 :         LOCK(m_wallet->cs_wallet);
     459           0 :         return m_wallet->GetDebit(txin, filter);
     460           0 :     }
     461           0 :     CAmount getCredit(const CTxOut& txout, isminefilter filter) override
     462             :     {
     463           0 :         LOCK(m_wallet->cs_wallet);
     464           0 :         return OutputGetCredit(*m_wallet, txout, filter);
     465           0 :     }
     466           0 :     CoinsList listCoins() override
     467             :     {
     468           0 :         LOCK(m_wallet->cs_wallet);
     469           0 :         CoinsList result;
     470           0 :         for (const auto& entry : ListCoins(*m_wallet)) {
     471           0 :             auto& group = result[entry.first];
     472           0 :             for (const auto& coin : entry.second) {
     473           0 :                 group.emplace_back(coin.outpoint,
     474           0 :                     MakeWalletTxOut(*m_wallet, coin));
     475             :             }
     476             :         }
     477           0 :         return result;
     478           0 :     }
     479           0 :     std::vector<WalletTxOut> getCoins(const std::vector<COutPoint>& outputs) override
     480             :     {
     481           0 :         LOCK(m_wallet->cs_wallet);
     482           0 :         std::vector<WalletTxOut> result;
     483           0 :         result.reserve(outputs.size());
     484           0 :         for (const auto& output : outputs) {
     485           0 :             result.emplace_back();
     486           0 :             auto it = m_wallet->mapWallet.find(output.hash);
     487           0 :             if (it != m_wallet->mapWallet.end()) {
     488           0 :                 int depth = m_wallet->GetTxDepthInMainChain(it->second);
     489           0 :                 if (depth >= 0) {
     490           0 :                     result.back() = MakeWalletTxOut(*m_wallet, it->second, output.n, depth);
     491           0 :                 }
     492           0 :             }
     493             :         }
     494           0 :         return result;
     495           0 :     }
     496           0 :     CAmount getRequiredFee(unsigned int tx_bytes) override { return GetRequiredFee(*m_wallet, tx_bytes); }
     497           0 :     CAmount getMinimumFee(unsigned int tx_bytes,
     498             :         const CCoinControl& coin_control,
     499             :         int* returned_target,
     500             :         FeeReason* reason) override
     501             :     {
     502           0 :         FeeCalculation fee_calc;
     503             :         CAmount result;
     504           0 :         result = GetMinimumFee(*m_wallet, tx_bytes, coin_control, &fee_calc);
     505           0 :         if (returned_target) *returned_target = fee_calc.returnedTarget;
     506           0 :         if (reason) *reason = fee_calc.reason;
     507           0 :         return result;
     508             :     }
     509           0 :     unsigned int getConfirmTarget() override { return m_wallet->m_confirm_target; }
     510           0 :     bool hdEnabled() override { return m_wallet->IsHDEnabled(); }
     511           0 :     bool canGetAddresses() override { return m_wallet->CanGetAddresses(); }
     512           0 :     bool hasExternalSigner() override { return m_wallet->IsWalletFlagSet(WALLET_FLAG_EXTERNAL_SIGNER); }
     513           0 :     bool privateKeysDisabled() override { return m_wallet->IsWalletFlagSet(WALLET_FLAG_DISABLE_PRIVATE_KEYS); }
     514           0 :     bool taprootEnabled() override {
     515           0 :         if (m_wallet->IsLegacy()) return false;
     516           0 :         auto spk_man = m_wallet->GetScriptPubKeyMan(OutputType::BECH32M, /*internal=*/false);
     517           0 :         return spk_man != nullptr;
     518           0 :     }
     519           0 :     OutputType getDefaultAddressType() override { return m_wallet->m_default_address_type; }
     520           0 :     CAmount getDefaultMaxTxFee() override { return m_wallet->m_default_max_tx_fee; }
     521           0 :     void remove() override
     522             :     {
     523           0 :         RemoveWallet(m_context, m_wallet, /*load_on_start=*/false);
     524           0 :     }
     525           0 :     bool isLegacy() override { return m_wallet->IsLegacy(); }
     526           0 :     std::unique_ptr<Handler> handleUnload(UnloadFn fn) override
     527             :     {
     528           0 :         return MakeSignalHandler(m_wallet->NotifyUnload.connect(fn));
     529           0 :     }
     530           0 :     std::unique_ptr<Handler> handleShowProgress(ShowProgressFn fn) override
     531             :     {
     532           0 :         return MakeSignalHandler(m_wallet->ShowProgress.connect(fn));
     533           0 :     }
     534           0 :     std::unique_ptr<Handler> handleStatusChanged(StatusChangedFn fn) override
     535             :     {
     536           0 :         return MakeSignalHandler(m_wallet->NotifyStatusChanged.connect([fn](CWallet*) { fn(); }));
     537           0 :     }
     538           0 :     std::unique_ptr<Handler> handleAddressBookChanged(AddressBookChangedFn fn) override
     539             :     {
     540           0 :         return MakeSignalHandler(m_wallet->NotifyAddressBookChanged.connect(
     541           0 :             [fn](const CTxDestination& address, const std::string& label, bool is_mine,
     542           0 :                  AddressPurpose purpose, ChangeType status) { fn(address, label, is_mine, purpose, status); }));
     543           0 :     }
     544           0 :     std::unique_ptr<Handler> handleTransactionChanged(TransactionChangedFn fn) override
     545             :     {
     546           0 :         return MakeSignalHandler(m_wallet->NotifyTransactionChanged.connect(
     547           0 :             [fn](const uint256& txid, ChangeType status) { fn(txid, status); }));
     548           0 :     }
     549           0 :     std::unique_ptr<Handler> handleWatchOnlyChanged(WatchOnlyChangedFn fn) override
     550             :     {
     551           0 :         return MakeSignalHandler(m_wallet->NotifyWatchonlyChanged.connect(fn));
     552           0 :     }
     553           0 :     std::unique_ptr<Handler> handleCanGetAddressesChanged(CanGetAddressesChangedFn fn) override
     554             :     {
     555           0 :         return MakeSignalHandler(m_wallet->NotifyCanGetAddressesChanged.connect(fn));
     556           0 :     }
     557           0 :     CWallet* wallet() override { return m_wallet.get(); }
     558             : 
     559             :     WalletContext& m_context;
     560             :     std::shared_ptr<CWallet> m_wallet;
     561             : };
     562             : 
     563             : class WalletLoaderImpl : public WalletLoader
     564             : {
     565             : public:
     566           0 :     WalletLoaderImpl(Chain& chain, ArgsManager& args)
     567           0 :     {
     568           0 :         m_context.chain = &chain;
     569           0 :         m_context.args = &args;
     570           0 :     }
     571           0 :     ~WalletLoaderImpl() override { UnloadWallets(m_context); }
     572             : 
     573             :     //! ChainClient methods
     574           0 :     void registerRpcs() override
     575             :     {
     576           0 :         for (const CRPCCommand& command : GetWalletRPCCommands()) {
     577           0 :             m_rpc_commands.emplace_back(command.category, command.name, [this, &command](const JSONRPCRequest& request, UniValue& result, bool last_handler) {
     578           0 :                 JSONRPCRequest wallet_request = request;
     579           0 :                 wallet_request.context = &m_context;
     580           0 :                 return command.actor(wallet_request, result, last_handler);
     581           0 :             }, command.argNames, command.unique_id);
     582           0 :             m_rpc_handlers.emplace_back(m_context.chain->handleRpc(m_rpc_commands.back()));
     583             :         }
     584           0 :     }
     585           0 :     bool verify() override { return VerifyWallets(m_context); }
     586           0 :     bool load() override { return LoadWallets(m_context); }
     587           0 :     void start(CScheduler& scheduler) override { return StartWallets(m_context, scheduler); }
     588           0 :     void flush() override { return FlushWallets(m_context); }
     589           0 :     void stop() override { return StopWallets(m_context); }
     590           0 :     void setMockTime(int64_t time) override { return SetMockTime(time); }
     591             : 
     592             :     //! WalletLoader methods
     593           0 :     util::Result<std::unique_ptr<Wallet>> createWallet(const std::string& name, const SecureString& passphrase, uint64_t wallet_creation_flags, std::vector<bilingual_str>& warnings) override
     594             :     {
     595           0 :         DatabaseOptions options;
     596             :         DatabaseStatus status;
     597           0 :         ReadDatabaseArgs(*m_context.args, options);
     598           0 :         options.require_create = true;
     599           0 :         options.create_flags = wallet_creation_flags;
     600           0 :         options.create_passphrase = passphrase;
     601           0 :         bilingual_str error;
     602           0 :         std::unique_ptr<Wallet> wallet{MakeWallet(m_context, CreateWallet(m_context, name, /*load_on_start=*/true, options, status, error, warnings))};
     603           0 :         if (wallet) {
     604           0 :             return {std::move(wallet)};
     605             :         } else {
     606           0 :             return util::Error{error};
     607             :         }
     608           0 :     }
     609           0 :     util::Result<std::unique_ptr<Wallet>> loadWallet(const std::string& name, std::vector<bilingual_str>& warnings) override
     610             :     {
     611           0 :         DatabaseOptions options;
     612             :         DatabaseStatus status;
     613           0 :         ReadDatabaseArgs(*m_context.args, options);
     614           0 :         options.require_existing = true;
     615           0 :         bilingual_str error;
     616           0 :         std::unique_ptr<Wallet> wallet{MakeWallet(m_context, LoadWallet(m_context, name, /*load_on_start=*/true, options, status, error, warnings))};
     617           0 :         if (wallet) {
     618           0 :             return {std::move(wallet)};
     619             :         } else {
     620           0 :             return util::Error{error};
     621             :         }
     622           0 :     }
     623           0 :     util::Result<std::unique_ptr<Wallet>> restoreWallet(const fs::path& backup_file, const std::string& wallet_name, std::vector<bilingual_str>& warnings) override
     624             :     {
     625             :         DatabaseStatus status;
     626           0 :         bilingual_str error;
     627           0 :         std::unique_ptr<Wallet> wallet{MakeWallet(m_context, RestoreWallet(m_context, backup_file, wallet_name, /*load_on_start=*/true, status, error, warnings))};
     628           0 :         if (wallet) {
     629           0 :             return {std::move(wallet)};
     630             :         } else {
     631           0 :             return util::Error{error};
     632             :         }
     633           0 :     }
     634           0 :     util::Result<WalletMigrationResult> migrateWallet(const std::string& name, const SecureString& passphrase) override
     635             :     {
     636           0 :         auto res = wallet::MigrateLegacyToDescriptor(name, passphrase, m_context);
     637           0 :         if (!res) return util::Error{util::ErrorString(res)};
     638           0 :         WalletMigrationResult out{
     639           0 :             .wallet = MakeWallet(m_context, res->wallet),
     640           0 :             .watchonly_wallet_name = res->watchonly_wallet ? std::make_optional(res->watchonly_wallet->GetName()) : std::nullopt,
     641           0 :             .solvables_wallet_name = res->solvables_wallet ? std::make_optional(res->solvables_wallet->GetName()) : std::nullopt,
     642           0 :             .backup_path = res->backup_path,
     643             :         };
     644           0 :         return {std::move(out)}; // std::move to work around clang bug
     645           0 :     }
     646           0 :     std::string getWalletDir() override
     647             :     {
     648           0 :         return fs::PathToString(GetWalletDir());
     649           0 :     }
     650           0 :     std::vector<std::string> listWalletDir() override
     651             :     {
     652           0 :         std::vector<std::string> paths;
     653           0 :         for (auto& path : ListDatabases(GetWalletDir())) {
     654           0 :             paths.push_back(fs::PathToString(path));
     655             :         }
     656           0 :         return paths;
     657           0 :     }
     658           0 :     std::vector<std::unique_ptr<Wallet>> getWallets() override
     659             :     {
     660           0 :         std::vector<std::unique_ptr<Wallet>> wallets;
     661           0 :         for (const auto& wallet : GetWallets(m_context)) {
     662           0 :             wallets.emplace_back(MakeWallet(m_context, wallet));
     663             :         }
     664           0 :         return wallets;
     665           0 :     }
     666           0 :     std::unique_ptr<Handler> handleLoadWallet(LoadWalletFn fn) override
     667             :     {
     668           0 :         return HandleLoadWallet(m_context, std::move(fn));
     669           0 :     }
     670           0 :     WalletContext* context() override  { return &m_context; }
     671             : 
     672             :     WalletContext m_context;
     673             :     const std::vector<std::string> m_wallet_filenames;
     674             :     std::vector<std::unique_ptr<Handler>> m_rpc_handlers;
     675             :     std::list<CRPCCommand> m_rpc_commands;
     676             : };
     677             : } // namespace
     678             : } // namespace wallet
     679             : 
     680             : namespace interfaces {
     681           0 : std::unique_ptr<Wallet> MakeWallet(wallet::WalletContext& context, const std::shared_ptr<wallet::CWallet>& wallet) { return wallet ? std::make_unique<wallet::WalletImpl>(context, wallet) : nullptr; }
     682             : 
     683           0 : std::unique_ptr<WalletLoader> MakeWalletLoader(Chain& chain, ArgsManager& args)
     684             : {
     685           0 :     return std::make_unique<wallet::WalletLoaderImpl>(chain, args);
     686             : }
     687             : } // namespace interfaces

Generated by: LCOV version 1.14