LCOV - code coverage report
Current view: top level - src/wallet/rpc - wallet.cpp (source / functions) Coverage Total Hit
Test: fuzz_coverage.info Lines: 0.0 % 542 0
Test Date: 2025-06-01 05:27:21 Functions: 0.0 % 25 0
Branches: 0.0 % 2704 0

             Branch data     Line data    Source code
       1                 :             : // Copyright (c) 2010 Satoshi Nakamoto
       2                 :             : // Copyright (c) 2009-present The Bitcoin Core developers
       3                 :             : // Distributed under the MIT software license, see the accompanying
       4                 :             : // file COPYING or http://www.opensource.org/licenses/mit-license.php.
       5                 :             : 
       6                 :             : #include <bitcoin-build-config.h> // IWYU pragma: keep
       7                 :             : 
       8                 :             : #include <core_io.h>
       9                 :             : #include <key_io.h>
      10                 :             : #include <rpc/server.h>
      11                 :             : #include <rpc/util.h>
      12                 :             : #include <univalue.h>
      13                 :             : #include <util/translation.h>
      14                 :             : #include <wallet/context.h>
      15                 :             : #include <wallet/receive.h>
      16                 :             : #include <wallet/rpc/util.h>
      17                 :             : #include <wallet/rpc/wallet.h>
      18                 :             : #include <wallet/wallet.h>
      19                 :             : #include <wallet/walletutil.h>
      20                 :             : 
      21                 :             : #include <optional>
      22                 :             : 
      23                 :             : 
      24                 :             : namespace wallet {
      25                 :             : 
      26                 :             : static const std::map<uint64_t, std::string> WALLET_FLAG_CAVEATS{
      27                 :             :     {WALLET_FLAG_AVOID_REUSE,
      28                 :             :      "You need to rescan the blockchain in order to correctly mark used "
      29                 :             :      "destinations in the past. Until this is done, some destinations may "
      30                 :             :      "be considered unused, even if the opposite is the case."},
      31                 :             : };
      32                 :             : 
      33                 :           0 : static RPCHelpMan getwalletinfo()
      34                 :             : {
      35                 :           0 :     return RPCHelpMan{"getwalletinfo",
      36                 :             :                 "Returns an object containing various wallet state info.\n",
      37                 :             :                 {},
      38                 :           0 :                 RPCResult{
      39                 :             :                     RPCResult::Type::OBJ, "", "",
      40                 :             :                     {
      41                 :             :                         {
      42                 :             :                         {RPCResult::Type::STR, "walletname", "the wallet name"},
      43                 :             :                         {RPCResult::Type::NUM, "walletversion", "the wallet version"},
      44                 :             :                         {RPCResult::Type::STR, "format", "the database format (only sqlite)"},
      45                 :             :                         {RPCResult::Type::STR_AMOUNT, "balance", "DEPRECATED. Identical to getbalances().mine.trusted"},
      46                 :             :                         {RPCResult::Type::STR_AMOUNT, "unconfirmed_balance", "DEPRECATED. Identical to getbalances().mine.untrusted_pending"},
      47                 :             :                         {RPCResult::Type::STR_AMOUNT, "immature_balance", "DEPRECATED. Identical to getbalances().mine.immature"},
      48                 :             :                         {RPCResult::Type::NUM, "txcount", "the total number of transactions in the wallet"},
      49                 :             :                         {RPCResult::Type::NUM, "keypoolsize", "how many new keys are pre-generated (only counts external keys)"},
      50                 :             :                         {RPCResult::Type::NUM, "keypoolsize_hd_internal", /*optional=*/true, "how many new keys are pre-generated for internal use (used for change outputs, only appears if the wallet is using this feature, otherwise external keys are used)"},
      51         [ #  # ]:           0 :                         {RPCResult::Type::NUM_TIME, "unlocked_until", /*optional=*/true, "the " + UNIX_EPOCH_TIME + " until which the wallet is unlocked for transfers, or 0 if the wallet is locked (only present for passphrase-encrypted wallets)"},
      52         [ #  # ]:           0 :                         {RPCResult::Type::STR_AMOUNT, "paytxfee", "the transaction fee configuration, set in " + CURRENCY_UNIT + "/kvB"},
      53                 :             :                         {RPCResult::Type::BOOL, "private_keys_enabled", "false if privatekeys are disabled for this wallet (enforced watch-only wallet)"},
      54                 :             :                         {RPCResult::Type::BOOL, "avoid_reuse", "whether this wallet tracks clean/dirty coins in terms of reuse"},
      55                 :             :                         {RPCResult::Type::OBJ, "scanning", "current scanning details, or false if no scan is in progress",
      56                 :             :                         {
      57                 :             :                             {RPCResult::Type::NUM, "duration", "elapsed seconds since scan start"},
      58                 :             :                             {RPCResult::Type::NUM, "progress", "scanning progress percentage [0.0, 1.0]"},
      59                 :             :                         }, /*skip_type_check=*/true},
      60                 :             :                         {RPCResult::Type::BOOL, "descriptors", "whether this wallet uses descriptors for output script management"},
      61                 :             :                         {RPCResult::Type::BOOL, "external_signer", "whether this wallet is configured to use an external signer such as a hardware wallet"},
      62                 :             :                         {RPCResult::Type::BOOL, "blank", "Whether this wallet intentionally does not contain any keys, scripts, or descriptors"},
      63                 :             :                         {RPCResult::Type::NUM_TIME, "birthtime", /*optional=*/true, "The start time for blocks scanning. It could be modified by (re)importing any descriptor with an earlier timestamp."},
      64                 :             :                         RESULT_LAST_PROCESSED_BLOCK,
      65                 :             :                     }},
      66   [ #  #  #  #  :           0 :                 },
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
                #  #  # ]
      67                 :           0 :                 RPCExamples{
      68   [ #  #  #  #  :           0 :                     HelpExampleCli("getwalletinfo", "")
                   #  # ]
      69   [ #  #  #  #  :           0 :             + HelpExampleRpc("getwalletinfo", "")
             #  #  #  # ]
      70         [ #  # ]:           0 :                 },
      71                 :           0 :         [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
      72                 :             : {
      73         [ #  # ]:           0 :     const std::shared_ptr<const CWallet> pwallet = GetWalletForJSONRPCRequest(request);
      74         [ #  # ]:           0 :     if (!pwallet) return UniValue::VNULL;
      75                 :             : 
      76                 :             :     // Make sure the results are valid at least up to the most recent block
      77                 :             :     // the user could have gotten from another RPC command prior to now
      78         [ #  # ]:           0 :     pwallet->BlockUntilSyncedToCurrentChain();
      79                 :             : 
      80         [ #  # ]:           0 :     LOCK(pwallet->cs_wallet);
      81                 :             : 
      82                 :           0 :     UniValue obj(UniValue::VOBJ);
      83                 :             : 
      84         [ #  # ]:           0 :     size_t kpExternalSize = pwallet->KeypoolCountExternalKeys();
      85         [ #  # ]:           0 :     const auto bal = GetBalance(*pwallet);
      86   [ #  #  #  #  :           0 :     obj.pushKV("walletname", pwallet->GetName());
                   #  # ]
      87   [ #  #  #  #  :           0 :     obj.pushKV("walletversion", pwallet->GetVersion());
             #  #  #  # ]
      88   [ #  #  #  #  :           0 :     obj.pushKV("format", pwallet->GetDatabase().Format());
                   #  # ]
      89   [ #  #  #  #  :           0 :     obj.pushKV("balance", ValueFromAmount(bal.m_mine_trusted));
                   #  # ]
      90   [ #  #  #  #  :           0 :     obj.pushKV("unconfirmed_balance", ValueFromAmount(bal.m_mine_untrusted_pending));
                   #  # ]
      91   [ #  #  #  #  :           0 :     obj.pushKV("immature_balance", ValueFromAmount(bal.m_mine_immature));
                   #  # ]
      92   [ #  #  #  #  :           0 :     obj.pushKV("txcount",       (int)pwallet->mapWallet.size());
                   #  # ]
      93   [ #  #  #  #  :           0 :     obj.pushKV("keypoolsize", (int64_t)kpExternalSize);
                   #  # ]
      94                 :             : 
      95   [ #  #  #  # ]:           0 :     if (pwallet->CanSupportFeature(FEATURE_HD_SPLIT)) {
      96   [ #  #  #  #  :           0 :         obj.pushKV("keypoolsize_hd_internal",   (int64_t)(pwallet->GetKeyPoolSize() - kpExternalSize));
             #  #  #  # ]
      97                 :             :     }
      98   [ #  #  #  # ]:           0 :     if (pwallet->IsCrypted()) {
      99   [ #  #  #  #  :           0 :         obj.pushKV("unlocked_until", pwallet->nRelockTime);
                   #  # ]
     100                 :             :     }
     101   [ #  #  #  #  :           0 :     obj.pushKV("paytxfee", ValueFromAmount(pwallet->m_pay_tx_fee.GetFeePerK()));
                   #  # ]
     102   [ #  #  #  #  :           0 :     obj.pushKV("private_keys_enabled", !pwallet->IsWalletFlagSet(WALLET_FLAG_DISABLE_PRIVATE_KEYS));
             #  #  #  # ]
     103   [ #  #  #  #  :           0 :     obj.pushKV("avoid_reuse", pwallet->IsWalletFlagSet(WALLET_FLAG_AVOID_REUSE));
             #  #  #  # ]
     104         [ #  # ]:           0 :     if (pwallet->IsScanning()) {
     105                 :           0 :         UniValue scanning(UniValue::VOBJ);
     106   [ #  #  #  #  :           0 :         scanning.pushKV("duration", Ticks<std::chrono::seconds>(pwallet->ScanningDuration()));
                   #  # ]
     107   [ #  #  #  #  :           0 :         scanning.pushKV("progress", pwallet->ScanningProgress());
                   #  # ]
     108   [ #  #  #  # ]:           0 :         obj.pushKV("scanning", std::move(scanning));
     109                 :           0 :     } else {
     110   [ #  #  #  #  :           0 :         obj.pushKV("scanning", false);
                   #  # ]
     111                 :             :     }
     112   [ #  #  #  #  :           0 :     obj.pushKV("descriptors", pwallet->IsWalletFlagSet(WALLET_FLAG_DESCRIPTORS));
             #  #  #  # ]
     113   [ #  #  #  #  :           0 :     obj.pushKV("external_signer", pwallet->IsWalletFlagSet(WALLET_FLAG_EXTERNAL_SIGNER));
             #  #  #  # ]
     114   [ #  #  #  #  :           0 :     obj.pushKV("blank", pwallet->IsWalletFlagSet(WALLET_FLAG_BLANK_WALLET));
             #  #  #  # ]
     115         [ #  # ]:           0 :     if (int64_t birthtime = pwallet->GetBirthTime(); birthtime != UNKNOWN_TIME) {
     116   [ #  #  #  #  :           0 :         obj.pushKV("birthtime", birthtime);
                   #  # ]
     117                 :             :     }
     118                 :             : 
     119         [ #  # ]:           0 :     AppendLastProcessedBlock(obj, *pwallet);
     120                 :           0 :     return obj;
     121         [ #  # ]:           0 : },
     122   [ #  #  #  #  :           0 :     };
             #  #  #  # ]
     123   [ #  #  #  #  :           0 : }
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
             #  #  #  # ]
     124                 :             : 
     125                 :           0 : static RPCHelpMan listwalletdir()
     126                 :             : {
     127                 :           0 :     return RPCHelpMan{"listwalletdir",
     128                 :             :                 "Returns a list of wallets in the wallet directory.\n",
     129                 :             :                 {},
     130                 :           0 :                 RPCResult{
     131                 :             :                     RPCResult::Type::OBJ, "", "",
     132                 :             :                     {
     133                 :             :                         {RPCResult::Type::ARR, "wallets", "",
     134                 :             :                         {
     135                 :             :                             {RPCResult::Type::OBJ, "", "",
     136                 :             :                             {
     137                 :             :                                 {RPCResult::Type::STR, "name", "The wallet name"},
     138                 :             :                                 {RPCResult::Type::ARR, "warnings", /*optional=*/true, "Warning messages, if any, related to loading the wallet.",
     139                 :             :                                 {
     140                 :             :                                     {RPCResult::Type::STR, "", ""},
     141                 :             :                                 }},
     142                 :             :                             }},
     143                 :             :                         }},
     144                 :             :                     }
     145   [ #  #  #  #  :           0 :                 },
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
             #  #  #  #  
                      # ]
     146                 :           0 :                 RPCExamples{
     147   [ #  #  #  #  :           0 :                     HelpExampleCli("listwalletdir", "")
                   #  # ]
     148   [ #  #  #  #  :           0 :             + HelpExampleRpc("listwalletdir", "")
             #  #  #  # ]
     149         [ #  # ]:           0 :                 },
     150                 :           0 :         [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
     151                 :             : {
     152                 :           0 :     UniValue wallets(UniValue::VARR);
     153   [ #  #  #  #  :           0 :     for (const auto& [path, db_type] : ListDatabases(GetWalletDir())) {
                   #  # ]
     154                 :           0 :         UniValue wallet(UniValue::VOBJ);
     155   [ #  #  #  #  :           0 :         wallet.pushKV("name", path.utf8string());
             #  #  #  # ]
     156                 :           0 :                 UniValue warnings(UniValue::VARR);
     157         [ #  # ]:           0 :         if (db_type == "bdb") {
     158   [ #  #  #  # ]:           0 :             warnings.push_back("This wallet is a legacy wallet and will need to be migrated with migratewallet before it can be loaded");
     159                 :             :         }
     160   [ #  #  #  #  :           0 :         wallet.pushKV("warnings", warnings);
                   #  # ]
     161         [ #  # ]:           0 :         wallets.push_back(std::move(wallet));
     162                 :           0 :     }
     163                 :             : 
     164                 :           0 :     UniValue result(UniValue::VOBJ);
     165   [ #  #  #  # ]:           0 :     result.pushKV("wallets", std::move(wallets));
     166                 :           0 :     return result;
     167                 :           0 : },
     168   [ #  #  #  #  :           0 :     };
             #  #  #  # ]
     169   [ #  #  #  #  :           0 : }
          #  #  #  #  #  
             #  #  #  #  
                      # ]
     170                 :             : 
     171                 :           0 : static RPCHelpMan listwallets()
     172                 :             : {
     173                 :           0 :     return RPCHelpMan{"listwallets",
     174                 :             :                 "Returns a list of currently loaded wallets.\n"
     175                 :             :                 "For full information on the wallet, use \"getwalletinfo\"\n",
     176                 :             :                 {},
     177                 :           0 :                 RPCResult{
     178                 :             :                     RPCResult::Type::ARR, "", "",
     179                 :             :                     {
     180                 :             :                         {RPCResult::Type::STR, "walletname", "the wallet name"},
     181                 :             :                     }
     182   [ #  #  #  #  :           0 :                 },
          #  #  #  #  #  
          #  #  #  #  #  
             #  #  #  # ]
     183                 :           0 :                 RPCExamples{
     184   [ #  #  #  #  :           0 :                     HelpExampleCli("listwallets", "")
                   #  # ]
     185   [ #  #  #  #  :           0 :             + HelpExampleRpc("listwallets", "")
             #  #  #  # ]
     186         [ #  # ]:           0 :                 },
     187                 :           0 :         [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
     188                 :             : {
     189                 :           0 :     UniValue obj(UniValue::VARR);
     190                 :             : 
     191         [ #  # ]:           0 :     WalletContext& context = EnsureWalletContext(request.context);
     192   [ #  #  #  # ]:           0 :     for (const std::shared_ptr<CWallet>& wallet : GetWallets(context)) {
     193         [ #  # ]:           0 :         LOCK(wallet->cs_wallet);
     194   [ #  #  #  #  :           0 :         obj.push_back(wallet->GetName());
                   #  # ]
     195                 :           0 :     }
     196                 :             : 
     197                 :           0 :     return obj;
     198                 :           0 : },
     199   [ #  #  #  #  :           0 :     };
             #  #  #  # ]
     200   [ #  #  #  # ]:           0 : }
     201                 :             : 
     202                 :           0 : static RPCHelpMan loadwallet()
     203                 :             : {
     204                 :           0 :     return RPCHelpMan{
     205                 :             :         "loadwallet",
     206                 :             :         "Loads a wallet from a wallet file or directory."
     207                 :             :                 "\nNote that all wallet command-line options used when starting bitcoind will be"
     208                 :             :                 "\napplied to the new wallet.\n",
     209                 :             :                 {
     210         [ #  # ]:           0 :                     {"filename", RPCArg::Type::STR, RPCArg::Optional::NO, "The path to the directory of the wallet to be loaded, either absolute or relative to the \"wallets\" directory. The \"wallets\" directory is set by the -walletdir option and defaults to the \"wallets\" folder within the data directory."},
     211         [ #  # ]:           0 :                     {"load_on_startup", RPCArg::Type::BOOL, RPCArg::Optional::OMITTED, "Save wallet name to persistent settings and load on startup. True to add wallet to startup list, false to remove, null to leave unchanged."},
     212                 :             :                 },
     213                 :           0 :                 RPCResult{
     214                 :             :                     RPCResult::Type::OBJ, "", "",
     215                 :             :                     {
     216                 :             :                         {RPCResult::Type::STR, "name", "The wallet name if loaded successfully."},
     217                 :             :                         {RPCResult::Type::ARR, "warnings", /*optional=*/true, "Warning messages, if any, related to loading the wallet.",
     218                 :             :                         {
     219                 :             :                             {RPCResult::Type::STR, "", ""},
     220                 :             :                         }},
     221                 :             :                     }
     222   [ #  #  #  #  :           0 :                 },
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
             #  #  #  #  
                      # ]
     223                 :           0 :                 RPCExamples{
     224                 :             :                     "\nLoad wallet from the wallet dir:\n"
     225   [ #  #  #  #  :           0 :                     + HelpExampleCli("loadwallet", "\"walletname\"")
             #  #  #  # ]
     226   [ #  #  #  #  :           0 :                     + HelpExampleRpc("loadwallet", "\"walletname\"")
             #  #  #  # ]
     227                 :           0 :                     + "\nLoad wallet using absolute path (Unix):\n"
     228   [ #  #  #  #  :           0 :                     + HelpExampleCli("loadwallet", "\"/path/to/walletname/\"")
             #  #  #  # ]
     229   [ #  #  #  #  :           0 :                     + HelpExampleRpc("loadwallet", "\"/path/to/walletname/\"")
             #  #  #  # ]
     230                 :           0 :                     + "\nLoad wallet using absolute path (Windows):\n"
     231   [ #  #  #  #  :           0 :                     + HelpExampleCli("loadwallet", "\"DriveLetter:\\path\\to\\walletname\\\"")
             #  #  #  # ]
     232   [ #  #  #  #  :           0 :                     + HelpExampleRpc("loadwallet", "\"DriveLetter:\\path\\to\\walletname\\\"")
             #  #  #  # ]
     233         [ #  # ]:           0 :                 },
     234                 :           0 :         [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
     235                 :             : {
     236                 :           0 :     WalletContext& context = EnsureWalletContext(request.context);
     237                 :           0 :     const std::string name(request.params[0].get_str());
     238                 :             : 
     239         [ #  # ]:           0 :     DatabaseOptions options;
     240                 :           0 :     DatabaseStatus status;
     241         [ #  # ]:           0 :     ReadDatabaseArgs(*context.args, options);
     242                 :           0 :     options.require_existing = true;
     243         [ #  # ]:           0 :     bilingual_str error;
     244                 :           0 :     std::vector<bilingual_str> warnings;
     245   [ #  #  #  #  :           0 :     std::optional<bool> load_on_start = request.params[1].isNull() ? std::nullopt : std::optional<bool>(request.params[1].get_bool());
             #  #  #  # ]
     246                 :             : 
     247                 :           0 :     {
     248         [ #  # ]:           0 :         LOCK(context.wallets_mutex);
     249   [ #  #  #  #  :           0 :         if (std::any_of(context.wallets.begin(), context.wallets.end(), [&name](const auto& wallet) { return wallet->GetName() == name; })) {
          #  #  #  #  #  
          #  #  #  #  #  
                   #  # ]
     250   [ #  #  #  # ]:           0 :             throw JSONRPCError(RPC_WALLET_ALREADY_LOADED, "Wallet \"" + name + "\" is already loaded.");
     251                 :             :         }
     252                 :           0 :     }
     253                 :             : 
     254         [ #  # ]:           0 :     std::shared_ptr<CWallet> const wallet = LoadWallet(context, name, load_on_start, options, status, error, warnings);
     255                 :             : 
     256   [ #  #  #  # ]:           0 :     HandleWalletError(wallet, status, error);
     257                 :             : 
     258                 :           0 :     UniValue obj(UniValue::VOBJ);
     259   [ #  #  #  #  :           0 :     obj.pushKV("name", wallet->GetName());
                   #  # ]
     260         [ #  # ]:           0 :     PushWarnings(warnings, obj);
     261                 :             : 
     262         [ #  # ]:           0 :     return obj;
     263                 :           0 : },
     264   [ #  #  #  #  :           0 :     };
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
                      # ]
     265   [ #  #  #  #  :           0 : }
          #  #  #  #  #  
          #  #  #  #  #  
             #  #  #  # ]
     266                 :             : 
     267                 :           0 : static RPCHelpMan setwalletflag()
     268                 :             : {
     269                 :           0 :             std::string flags;
     270         [ #  # ]:           0 :             for (auto& it : WALLET_FLAG_MAP)
     271         [ #  # ]:           0 :                 if (it.second & MUTABLE_WALLET_FLAGS)
     272   [ #  #  #  # ]:           0 :                     flags += (flags == "" ? "" : ", ") + it.first;
     273                 :             : 
     274         [ #  # ]:           0 :     return RPCHelpMan{
     275                 :             :         "setwalletflag",
     276                 :             :         "Change the state of the given wallet flag for a wallet.\n",
     277                 :             :                 {
     278         [ #  # ]:           0 :                     {"flag", RPCArg::Type::STR, RPCArg::Optional::NO, "The name of the flag to change. Current available flags: " + flags},
     279         [ #  # ]:           0 :                     {"value", RPCArg::Type::BOOL, RPCArg::Default{true}, "The new state."},
     280                 :             :                 },
     281                 :           0 :                 RPCResult{
     282                 :             :                     RPCResult::Type::OBJ, "", "",
     283                 :             :                     {
     284                 :             :                         {RPCResult::Type::STR, "flag_name", "The name of the flag that was modified"},
     285                 :             :                         {RPCResult::Type::BOOL, "flag_state", "The new state of the flag"},
     286                 :             :                         {RPCResult::Type::STR, "warnings", /*optional=*/true, "Any warnings associated with the change"},
     287                 :             :                     }
     288   [ #  #  #  #  :           0 :                 },
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
                      # ]
     289                 :           0 :                 RPCExamples{
     290   [ #  #  #  #  :           0 :                     HelpExampleCli("setwalletflag", "avoid_reuse")
                   #  # ]
     291   [ #  #  #  #  :           0 :                   + HelpExampleRpc("setwalletflag", "\"avoid_reuse\"")
             #  #  #  # ]
     292         [ #  # ]:           0 :                 },
     293                 :           0 :         [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
     294                 :             : {
     295                 :           0 :     std::shared_ptr<CWallet> const pwallet = GetWalletForJSONRPCRequest(request);
     296         [ #  # ]:           0 :     if (!pwallet) return UniValue::VNULL;
     297                 :             : 
     298   [ #  #  #  #  :           0 :     std::string flag_str = request.params[0].get_str();
                   #  # ]
     299   [ #  #  #  #  :           0 :     bool value = request.params[1].isNull() || request.params[1].get_bool();
          #  #  #  #  #  
                      # ]
     300                 :             : 
     301         [ #  # ]:           0 :     if (!WALLET_FLAG_MAP.count(flag_str)) {
     302   [ #  #  #  # ]:           0 :         throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Unknown wallet flag: %s", flag_str));
     303                 :             :     }
     304                 :             : 
     305         [ #  # ]:           0 :     auto flag = WALLET_FLAG_MAP.at(flag_str);
     306                 :             : 
     307         [ #  # ]:           0 :     if (!(flag & MUTABLE_WALLET_FLAGS)) {
     308   [ #  #  #  # ]:           0 :         throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Wallet flag is immutable: %s", flag_str));
     309                 :             :     }
     310                 :             : 
     311                 :           0 :     UniValue res(UniValue::VOBJ);
     312                 :             : 
     313   [ #  #  #  # ]:           0 :     if (pwallet->IsWalletFlagSet(flag) == value) {
     314   [ #  #  #  #  :           0 :         throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Wallet flag is already set to %s: %s", value ? "true" : "false", flag_str));
                   #  # ]
     315                 :             :     }
     316                 :             : 
     317   [ #  #  #  #  :           0 :     res.pushKV("flag_name", flag_str);
                   #  # ]
     318   [ #  #  #  #  :           0 :     res.pushKV("flag_state", value);
                   #  # ]
     319                 :             : 
     320         [ #  # ]:           0 :     if (value) {
     321         [ #  # ]:           0 :         pwallet->SetWalletFlag(flag);
     322                 :             :     } else {
     323         [ #  # ]:           0 :         pwallet->UnsetWalletFlag(flag);
     324                 :             :     }
     325                 :             : 
     326   [ #  #  #  #  :           0 :     if (flag && value && WALLET_FLAG_CAVEATS.count(flag)) {
                   #  # ]
     327   [ #  #  #  #  :           0 :         res.pushKV("warnings", WALLET_FLAG_CAVEATS.at(flag));
             #  #  #  # ]
     328                 :             :     }
     329                 :             : 
     330                 :           0 :     return res;
     331                 :           0 : },
     332   [ #  #  #  #  :           0 :     };
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
                #  #  # ]
     333   [ #  #  #  #  :           0 : }
          #  #  #  #  #  
          #  #  #  #  #  
             #  #  #  # ]
     334                 :             : 
     335                 :           0 : static RPCHelpMan createwallet()
     336                 :             : {
     337                 :           0 :     return RPCHelpMan{
     338                 :             :         "createwallet",
     339                 :             :         "Creates and loads a new wallet.\n",
     340                 :             :         {
     341         [ #  # ]:           0 :             {"wallet_name", RPCArg::Type::STR, RPCArg::Optional::NO, "The name for the new wallet. If this is a path, the wallet will be created at the path location."},
     342         [ #  # ]:           0 :             {"disable_private_keys", RPCArg::Type::BOOL, RPCArg::Default{false}, "Disable the possibility of private keys (only watchonlys are possible in this mode)."},
     343         [ #  # ]:           0 :             {"blank", RPCArg::Type::BOOL, RPCArg::Default{false}, "Create a blank wallet. A blank wallet has no keys."},
     344         [ #  # ]:           0 :             {"passphrase", RPCArg::Type::STR, RPCArg::Optional::OMITTED, "Encrypt the wallet with this passphrase."},
     345         [ #  # ]:           0 :             {"avoid_reuse", RPCArg::Type::BOOL, RPCArg::Default{false}, "Keep track of coin reuse, and treat dirty and clean coins differently with privacy considerations in mind."},
     346         [ #  # ]:           0 :             {"descriptors", RPCArg::Type::BOOL, RPCArg::Default{true}, "If set, must be \"true\""},
     347         [ #  # ]:           0 :             {"load_on_startup", RPCArg::Type::BOOL, RPCArg::Optional::OMITTED, "Save wallet name to persistent settings and load on startup. True to add wallet to startup list, false to remove, null to leave unchanged."},
     348         [ #  # ]:           0 :             {"external_signer", RPCArg::Type::BOOL, RPCArg::Default{false}, "Use an external signer such as a hardware wallet. Requires -signer to be configured. Wallet creation will fail if keys cannot be fetched. Requires disable_private_keys and descriptors set to true."},
     349                 :             :         },
     350                 :           0 :         RPCResult{
     351                 :             :             RPCResult::Type::OBJ, "", "",
     352                 :             :             {
     353                 :             :                 {RPCResult::Type::STR, "name", "The wallet name if created successfully. If the wallet was created using a full path, the wallet_name will be the full path."},
     354                 :             :                 {RPCResult::Type::ARR, "warnings", /*optional=*/true, "Warning messages, if any, related to creating and loading the wallet.",
     355                 :             :                 {
     356                 :             :                     {RPCResult::Type::STR, "", ""},
     357                 :             :                 }},
     358                 :             :             }
     359   [ #  #  #  #  :           0 :         },
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
             #  #  #  #  
                      # ]
     360                 :           0 :         RPCExamples{
     361   [ #  #  #  #  :           0 :             HelpExampleCli("createwallet", "\"testwallet\"")
                   #  # ]
     362   [ #  #  #  #  :           0 :             + HelpExampleRpc("createwallet", "\"testwallet\"")
             #  #  #  # ]
     363   [ #  #  #  #  :           0 :             + HelpExampleCliNamed("createwallet", {{"wallet_name", "descriptors"}, {"avoid_reuse", true}, {"load_on_startup", true}})
          #  #  #  #  #  
                #  #  # ]
     364   [ #  #  #  #  :           0 :             + HelpExampleRpcNamed("createwallet", {{"wallet_name", "descriptors"}, {"avoid_reuse", true}, {"load_on_startup", true}})
          #  #  #  #  #  
                #  #  # ]
     365         [ #  # ]:           0 :         },
     366                 :           0 :         [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
     367                 :             : {
     368                 :           0 :     WalletContext& context = EnsureWalletContext(request.context);
     369                 :           0 :     uint64_t flags = 0;
     370   [ #  #  #  # ]:           0 :     if (!request.params[1].isNull() && request.params[1].get_bool()) {
     371                 :             :         flags |= WALLET_FLAG_DISABLE_PRIVATE_KEYS;
     372                 :             :     }
     373                 :             : 
     374   [ #  #  #  # ]:           0 :     if (!request.params[2].isNull() && request.params[2].get_bool()) {
     375                 :           0 :         flags |= WALLET_FLAG_BLANK_WALLET;
     376                 :             :     }
     377         [ #  # ]:           0 :     SecureString passphrase;
     378         [ #  # ]:           0 :     passphrase.reserve(100);
     379                 :           0 :     std::vector<bilingual_str> warnings;
     380   [ #  #  #  # ]:           0 :     if (!request.params[3].isNull()) {
     381   [ #  #  #  #  :           0 :         passphrase = std::string_view{request.params[3].get_str()};
                   #  # ]
     382         [ #  # ]:           0 :         if (passphrase.empty()) {
     383                 :             :             // Empty string means unencrypted
     384   [ #  #  #  #  :           0 :             warnings.emplace_back(Untranslated("Empty string given as passphrase, wallet will not be encrypted."));
                   #  # ]
     385                 :             :         }
     386                 :             :     }
     387                 :             : 
     388   [ #  #  #  #  :           0 :     if (!request.params[4].isNull() && request.params[4].get_bool()) {
          #  #  #  #  #  
                      # ]
     389                 :           0 :         flags |= WALLET_FLAG_AVOID_REUSE;
     390                 :             :     }
     391                 :           0 :     flags |= WALLET_FLAG_DESCRIPTORS;
     392   [ #  #  #  # ]:           0 :     if (!self.Arg<bool>("descriptors")) {
     393   [ #  #  #  # ]:           0 :         throw JSONRPCError(RPC_WALLET_ERROR, "descriptors argument must be set to \"true\"; it is no longer possible to create a legacy wallet.");
     394                 :             :     }
     395   [ #  #  #  #  :           0 :     if (!request.params[7].isNull() && request.params[7].get_bool()) {
          #  #  #  #  #  
                      # ]
     396                 :             : #ifdef ENABLE_EXTERNAL_SIGNER
     397                 :             :         flags |= WALLET_FLAG_EXTERNAL_SIGNER;
     398                 :             : #else
     399   [ #  #  #  # ]:           0 :         throw JSONRPCError(RPC_WALLET_ERROR, "Compiled without external signing support (required for external signing)");
     400                 :             : #endif
     401                 :             :     }
     402                 :             : 
     403         [ #  # ]:           0 :     DatabaseOptions options;
     404                 :           0 :     DatabaseStatus status;
     405         [ #  # ]:           0 :     ReadDatabaseArgs(*context.args, options);
     406                 :           0 :     options.require_create = true;
     407                 :           0 :     options.create_flags = flags;
     408         [ #  # ]:           0 :     options.create_passphrase = passphrase;
     409         [ #  # ]:           0 :     bilingual_str error;
     410   [ #  #  #  #  :           0 :     std::optional<bool> load_on_start = request.params[6].isNull() ? std::nullopt : std::optional<bool>(request.params[6].get_bool());
             #  #  #  # ]
     411   [ #  #  #  #  :           0 :     const std::shared_ptr<CWallet> wallet = CreateWallet(context, request.params[0].get_str(), load_on_start, options, status, error, warnings);
                   #  # ]
     412         [ #  # ]:           0 :     if (!wallet) {
     413         [ #  # ]:           0 :         RPCErrorCode code = status == DatabaseStatus::FAILED_ENCRYPT ? RPC_WALLET_ENCRYPTION_FAILED : RPC_WALLET_ERROR;
     414         [ #  # ]:           0 :         throw JSONRPCError(code, error.original);
     415                 :             :     }
     416                 :             : 
     417                 :           0 :     UniValue obj(UniValue::VOBJ);
     418   [ #  #  #  #  :           0 :     obj.pushKV("name", wallet->GetName());
                   #  # ]
     419         [ #  # ]:           0 :     PushWarnings(warnings, obj);
     420                 :             : 
     421         [ #  # ]:           0 :     return obj;
     422                 :           0 : },
     423   [ #  #  #  #  :           0 :     };
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
             #  #  #  #  
                      # ]
     424   [ #  #  #  #  :           0 : }
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
                   #  # ]
     425                 :             : 
     426                 :           0 : static RPCHelpMan unloadwallet()
     427                 :             : {
     428                 :           0 :     return RPCHelpMan{"unloadwallet",
     429                 :             :                 "Unloads the wallet referenced by the request endpoint, otherwise unloads the wallet specified in the argument.\n"
     430                 :             :                 "Specifying the wallet name on a wallet endpoint is invalid.",
     431                 :             :                 {
     432         [ #  # ]:           0 :                     {"wallet_name", RPCArg::Type::STR, RPCArg::DefaultHint{"the wallet name from the RPC endpoint"}, "The name of the wallet to unload. If provided both here and in the RPC endpoint, the two must be identical."},
     433         [ #  # ]:           0 :                     {"load_on_startup", RPCArg::Type::BOOL, RPCArg::Optional::OMITTED, "Save wallet name to persistent settings and load on startup. True to add wallet to startup list, false to remove, null to leave unchanged."},
     434                 :             :                 },
     435                 :           0 :                 RPCResult{RPCResult::Type::OBJ, "", "", {
     436                 :             :                     {RPCResult::Type::ARR, "warnings", /*optional=*/true, "Warning messages, if any, related to unloading the wallet.",
     437                 :             :                     {
     438                 :             :                         {RPCResult::Type::STR, "", ""},
     439                 :             :                     }},
     440   [ #  #  #  #  :           0 :                 }},
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
             #  #  #  # ]
     441                 :           0 :                 RPCExamples{
     442   [ #  #  #  #  :           0 :                     HelpExampleCli("unloadwallet", "wallet_name")
                   #  # ]
     443   [ #  #  #  #  :           0 :             + HelpExampleRpc("unloadwallet", "wallet_name")
             #  #  #  # ]
     444         [ #  # ]:           0 :                 },
     445                 :           0 :         [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
     446                 :             : {
     447         [ #  # ]:           0 :     std::string wallet_name;
     448   [ #  #  #  # ]:           0 :     if (GetWalletNameFromJSONRPCRequest(request, wallet_name)) {
     449   [ #  #  #  #  :           0 :         if (!(request.params[0].isNull() || request.params[0].get_str() == wallet_name)) {
          #  #  #  #  #  
                      # ]
     450   [ #  #  #  # ]:           0 :             throw JSONRPCError(RPC_INVALID_PARAMETER, "RPC endpoint wallet and wallet_name parameter specify different wallets");
     451                 :             :         }
     452                 :             :     } else {
     453   [ #  #  #  #  :           0 :         wallet_name = request.params[0].get_str();
                   #  # ]
     454                 :             :     }
     455                 :             : 
     456         [ #  # ]:           0 :     WalletContext& context = EnsureWalletContext(request.context);
     457         [ #  # ]:           0 :     std::shared_ptr<CWallet> wallet = GetWallet(context, wallet_name);
     458         [ #  # ]:           0 :     if (!wallet) {
     459   [ #  #  #  # ]:           0 :         throw JSONRPCError(RPC_WALLET_NOT_FOUND, "Requested wallet does not exist or is not loaded");
     460                 :             :     }
     461                 :             : 
     462                 :           0 :     std::vector<bilingual_str> warnings;
     463                 :           0 :     {
     464                 :           0 :         WalletRescanReserver reserver(*wallet);
     465         [ #  # ]:           0 :         if (!reserver.reserve()) {
     466   [ #  #  #  # ]:           0 :             throw JSONRPCError(RPC_WALLET_ERROR, "Wallet is currently rescanning. Abort existing rescan or wait.");
     467                 :             :         }
     468                 :             : 
     469                 :             :         // Release the "main" shared pointer and prevent further notifications.
     470                 :             :         // Note that any attempt to load the same wallet would fail until the wallet
     471                 :             :         // is destroyed (see CheckUniqueFileid).
     472         [ #  # ]:           0 :         std::optional<bool> load_on_start{self.MaybeArg<bool>("load_on_startup")};
     473   [ #  #  #  # ]:           0 :         if (!RemoveWallet(context, wallet, load_on_start, warnings)) {
     474   [ #  #  #  # ]:           0 :             throw JSONRPCError(RPC_MISC_ERROR, "Requested wallet already unloaded");
     475                 :             :         }
     476                 :           0 :     }
     477                 :             : 
     478         [ #  # ]:           0 :     WaitForDeleteWallet(std::move(wallet));
     479                 :             : 
     480                 :           0 :     UniValue result(UniValue::VOBJ);
     481         [ #  # ]:           0 :     PushWarnings(warnings, result);
     482                 :             : 
     483                 :           0 :     return result;
     484         [ #  # ]:           0 : },
     485   [ #  #  #  #  :           0 :     };
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
                      # ]
     486   [ #  #  #  #  :           0 : }
          #  #  #  #  #  
             #  #  #  #  
                      # ]
     487                 :             : 
     488                 :           0 : static RPCHelpMan upgradewallet()
     489                 :             : {
     490                 :           0 :     return RPCHelpMan{
     491                 :             :         "upgradewallet",
     492                 :             :         "Upgrade the wallet. Upgrades to the latest version if no version number is specified.\n"
     493                 :             :         "New keys may be generated and a new wallet backup will need to be made.",
     494                 :             :         {
     495         [ #  # ]:           0 :             {"version", RPCArg::Type::NUM, RPCArg::Default{int{FEATURE_LATEST}}, "The version number to upgrade to. Default is the latest wallet version."}
     496                 :             :         },
     497                 :           0 :         RPCResult{
     498                 :             :             RPCResult::Type::OBJ, "", "",
     499                 :             :             {
     500                 :             :                 {RPCResult::Type::STR, "wallet_name", "Name of wallet this operation was performed on"},
     501                 :             :                 {RPCResult::Type::NUM, "previous_version", "Version of wallet before this operation"},
     502                 :             :                 {RPCResult::Type::NUM, "current_version", "Version of wallet after this operation"},
     503                 :             :                 {RPCResult::Type::STR, "result", /*optional=*/true, "Description of result, if no error"},
     504                 :             :                 {RPCResult::Type::STR, "error", /*optional=*/true, "Error message (if there is one)"}
     505                 :             :             },
     506   [ #  #  #  #  :           0 :         },
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
                #  #  # ]
     507                 :           0 :         RPCExamples{
     508   [ #  #  #  #  :           0 :             HelpExampleCli("upgradewallet", "169900")
                   #  # ]
     509   [ #  #  #  #  :           0 :             + HelpExampleRpc("upgradewallet", "169900")
             #  #  #  # ]
     510         [ #  # ]:           0 :         },
     511                 :           0 :         [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
     512                 :             : {
     513                 :           0 :     std::shared_ptr<CWallet> const pwallet = GetWalletForJSONRPCRequest(request);
     514         [ #  # ]:           0 :     if (!pwallet) return UniValue::VNULL;
     515                 :             : 
     516         [ #  # ]:           0 :     EnsureWalletIsUnlocked(*pwallet);
     517                 :             : 
     518                 :           0 :     int version = 0;
     519   [ #  #  #  # ]:           0 :     if (!request.params[0].isNull()) {
     520   [ #  #  #  # ]:           0 :         version = request.params[0].getInt<int>();
     521                 :             :     }
     522         [ #  # ]:           0 :     bilingual_str error;
     523         [ #  # ]:           0 :     const int previous_version{pwallet->GetVersion()};
     524         [ #  # ]:           0 :     const bool wallet_upgraded{pwallet->UpgradeWallet(version, error)};
     525         [ #  # ]:           0 :     const int current_version{pwallet->GetVersion()};
     526         [ #  # ]:           0 :     std::string result;
     527                 :             : 
     528         [ #  # ]:           0 :     if (wallet_upgraded) {
     529         [ #  # ]:           0 :         if (previous_version == current_version) {
     530         [ #  # ]:           0 :             result = "Already at latest version. Wallet version unchanged.";
     531                 :             :         } else {
     532         [ #  # ]:           0 :             result = strprintf("Wallet upgraded successfully from version %i to version %i.", previous_version, current_version);
     533                 :             :         }
     534                 :             :     }
     535                 :             : 
     536                 :           0 :     UniValue obj(UniValue::VOBJ);
     537   [ #  #  #  #  :           0 :     obj.pushKV("wallet_name", pwallet->GetName());
                   #  # ]
     538   [ #  #  #  #  :           0 :     obj.pushKV("previous_version", previous_version);
                   #  # ]
     539   [ #  #  #  #  :           0 :     obj.pushKV("current_version", current_version);
                   #  # ]
     540         [ #  # ]:           0 :     if (!result.empty()) {
     541   [ #  #  #  #  :           0 :         obj.pushKV("result", result);
                   #  # ]
     542                 :             :     } else {
     543         [ #  # ]:           0 :         CHECK_NONFATAL(!error.empty());
     544   [ #  #  #  #  :           0 :         obj.pushKV("error", error.original);
                   #  # ]
     545                 :             :     }
     546                 :           0 :     return obj;
     547                 :           0 : },
     548   [ #  #  #  #  :           0 :     };
          #  #  #  #  #  
          #  #  #  #  #  
                   #  # ]
     549   [ #  #  #  #  :           0 : }
          #  #  #  #  #  
          #  #  #  #  #  
             #  #  #  # ]
     550                 :             : 
     551                 :           0 : RPCHelpMan simulaterawtransaction()
     552                 :             : {
     553                 :           0 :     return RPCHelpMan{
     554                 :             :         "simulaterawtransaction",
     555                 :             :         "Calculate the balance change resulting in the signing and broadcasting of the given transaction(s).\n",
     556                 :             :         {
     557         [ #  # ]:           0 :             {"rawtxs", RPCArg::Type::ARR, RPCArg::Optional::OMITTED, "An array of hex strings of raw transactions.\n",
     558                 :             :                 {
     559         [ #  # ]:           0 :                     {"rawtx", RPCArg::Type::STR_HEX, RPCArg::Optional::OMITTED, ""},
     560                 :             :                 },
     561                 :             :             },
     562         [ #  # ]:           0 :             {"options", RPCArg::Type::OBJ_NAMED_PARAMS, RPCArg::Optional::OMITTED, "",
     563                 :             :                 {
     564         [ #  # ]:           0 :                     {"include_watchonly", RPCArg::Type::BOOL, RPCArg::DefaultHint{"true for watch-only wallets, otherwise false"}, "Whether to include watch-only addresses (see RPC importaddress)"},
     565                 :             :                 },
     566                 :             :             },
     567                 :             :         },
     568                 :           0 :         RPCResult{
     569                 :             :             RPCResult::Type::OBJ, "", "",
     570                 :             :             {
     571                 :             :                 {RPCResult::Type::STR_AMOUNT, "balance_change", "The wallet balance change (negative means decrease)."},
     572                 :             :             }
     573   [ #  #  #  #  :           0 :         },
          #  #  #  #  #  
          #  #  #  #  #  
             #  #  #  # ]
     574                 :           0 :         RPCExamples{
     575   [ #  #  #  #  :           0 :             HelpExampleCli("simulaterawtransaction", "[\"myhex\"]")
                   #  # ]
     576   [ #  #  #  #  :           0 :             + HelpExampleRpc("simulaterawtransaction", "[\"myhex\"]")
             #  #  #  # ]
     577         [ #  # ]:           0 :         },
     578                 :           0 :     [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
     579                 :             : {
     580         [ #  # ]:           0 :     const std::shared_ptr<const CWallet> rpc_wallet = GetWalletForJSONRPCRequest(request);
     581         [ #  # ]:           0 :     if (!rpc_wallet) return UniValue::VNULL;
     582         [ #  # ]:           0 :     const CWallet& wallet = *rpc_wallet;
     583                 :             : 
     584         [ #  # ]:           0 :     LOCK(wallet.cs_wallet);
     585                 :             : 
     586                 :           0 :     UniValue include_watchonly(UniValue::VNULL);
     587   [ #  #  #  # ]:           0 :     if (request.params[1].isObject()) {
     588   [ #  #  #  # ]:           0 :         UniValue options = request.params[1];
     589   [ #  #  #  #  :           0 :         RPCTypeCheckObj(options,
                   #  # ]
     590                 :             :             {
     591         [ #  # ]:           0 :                 {"include_watchonly", UniValueType(UniValue::VBOOL)},
     592                 :             :             },
     593                 :             :             true, true);
     594                 :             : 
     595   [ #  #  #  #  :           0 :         include_watchonly = options["include_watchonly"];
                   #  # ]
     596                 :           0 :     }
     597                 :             : 
     598                 :           0 :     isminefilter filter = ISMINE_SPENDABLE;
     599   [ #  #  #  # ]:           0 :     if (ParseIncludeWatchonly(include_watchonly, wallet)) {
     600                 :           0 :         filter |= ISMINE_WATCH_ONLY;
     601                 :             :     }
     602                 :             : 
     603   [ #  #  #  # ]:           0 :     const auto& txs = request.params[0].get_array();
     604                 :           0 :     CAmount changes{0};
     605                 :           0 :     std::map<COutPoint, CAmount> new_utxos; // UTXO:s that were made available in transaction array
     606                 :           0 :     std::set<COutPoint> spent;
     607                 :             : 
     608         [ #  # ]:           0 :     for (size_t i = 0; i < txs.size(); ++i) {
     609         [ #  # ]:           0 :         CMutableTransaction mtx;
     610   [ #  #  #  #  :           0 :         if (!DecodeHexTx(mtx, txs[i].get_str(), /* try_no_witness */ true, /* try_witness */ true)) {
             #  #  #  # ]
     611   [ #  #  #  # ]:           0 :             throw JSONRPCError(RPC_DESERIALIZATION_ERROR, "Transaction hex string decoding failure.");
     612                 :             :         }
     613                 :             : 
     614                 :             :         // Fetch previous transactions (inputs)
     615                 :           0 :         std::map<COutPoint, Coin> coins;
     616         [ #  # ]:           0 :         for (const CTxIn& txin : mtx.vin) {
     617         [ #  # ]:           0 :             coins[txin.prevout]; // Create empty map entry keyed by prevout.
     618                 :             :         }
     619         [ #  # ]:           0 :         wallet.chain().findCoins(coins);
     620                 :             : 
     621                 :             :         // Fetch debit; we are *spending* these; if the transaction is signed and
     622                 :             :         // broadcast, we will lose everything in these
     623         [ #  # ]:           0 :         for (const auto& txin : mtx.vin) {
     624                 :           0 :             const auto& outpoint = txin.prevout;
     625         [ #  # ]:           0 :             if (spent.count(outpoint)) {
     626   [ #  #  #  # ]:           0 :                 throw JSONRPCError(RPC_INVALID_PARAMETER, "Transaction(s) are spending the same output more than once");
     627                 :             :             }
     628         [ #  # ]:           0 :             if (new_utxos.count(outpoint)) {
     629         [ #  # ]:           0 :                 changes -= new_utxos.at(outpoint);
     630                 :           0 :                 new_utxos.erase(outpoint);
     631                 :             :             } else {
     632   [ #  #  #  # ]:           0 :                 if (coins.at(outpoint).IsSpent()) {
     633   [ #  #  #  # ]:           0 :                     throw JSONRPCError(RPC_INVALID_PARAMETER, "One or more transaction inputs are missing or have been spent already");
     634                 :             :                 }
     635         [ #  # ]:           0 :                 changes -= wallet.GetDebit(txin, filter);
     636                 :             :             }
     637         [ #  # ]:           0 :             spent.insert(outpoint);
     638                 :             :         }
     639                 :             : 
     640                 :             :         // Iterate over outputs; we are *receiving* these, if the wallet considers
     641                 :             :         // them "mine"; if the transaction is signed and broadcast, we will receive
     642                 :             :         // everything in these
     643                 :             :         // Also populate new_utxos in case these are spent in later transactions
     644                 :             : 
     645         [ #  # ]:           0 :         const auto& hash = mtx.GetHash();
     646         [ #  # ]:           0 :         for (size_t i = 0; i < mtx.vout.size(); ++i) {
     647         [ #  # ]:           0 :             const auto& txout = mtx.vout[i];
     648         [ #  # ]:           0 :             bool is_mine = 0 < (wallet.IsMine(txout) & filter);
     649   [ #  #  #  # ]:           0 :             changes += new_utxos[COutPoint(hash, i)] = is_mine ? txout.nValue : 0;
     650                 :             :         }
     651                 :           0 :     }
     652                 :             : 
     653                 :           0 :     UniValue result(UniValue::VOBJ);
     654   [ #  #  #  #  :           0 :     result.pushKV("balance_change", ValueFromAmount(changes));
                   #  # ]
     655                 :             : 
     656                 :           0 :     return result;
     657   [ #  #  #  #  :           0 : }
                   #  # ]
     658   [ #  #  #  #  :           0 :     };
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
                      # ]
     659   [ #  #  #  #  :           0 : }
          #  #  #  #  #  
          #  #  #  #  #  
                   #  # ]
     660                 :             : 
     661                 :           0 : static RPCHelpMan migratewallet()
     662                 :             : {
     663                 :           0 :     return RPCHelpMan{
     664                 :             :         "migratewallet",
     665                 :             :         "Migrate the wallet to a descriptor wallet.\n"
     666                 :             :         "A new wallet backup will need to be made.\n"
     667                 :             :         "\nThe migration process will create a backup of the wallet before migrating. This backup\n"
     668                 :             :         "file will be named <wallet name>-<timestamp>.legacy.bak and can be found in the directory\n"
     669                 :             :         "for this wallet. In the event of an incorrect migration, the backup can be restored using restorewallet."
     670                 :             :         "\nEncrypted wallets must have the passphrase provided as an argument to this call.\n"
     671                 :             :         "\nThis RPC may take a long time to complete. Increasing the RPC client timeout is recommended.",
     672                 :             :         {
     673         [ #  # ]:           0 :             {"wallet_name", RPCArg::Type::STR, RPCArg::DefaultHint{"the wallet name from the RPC endpoint"}, "The name of the wallet to migrate. If provided both here and in the RPC endpoint, the two must be identical."},
     674         [ #  # ]:           0 :             {"passphrase", RPCArg::Type::STR, RPCArg::Optional::OMITTED, "The wallet passphrase"},
     675                 :             :         },
     676                 :           0 :         RPCResult{
     677                 :             :             RPCResult::Type::OBJ, "", "",
     678                 :             :             {
     679                 :             :                 {RPCResult::Type::STR, "wallet_name", "The name of the primary migrated wallet"},
     680                 :             :                 {RPCResult::Type::STR, "watchonly_name", /*optional=*/true, "The name of the migrated wallet containing the watchonly scripts"},
     681                 :             :                 {RPCResult::Type::STR, "solvables_name", /*optional=*/true, "The name of the migrated wallet containing solvable but not watched scripts"},
     682                 :             :                 {RPCResult::Type::STR, "backup_path", "The location of the backup of the original wallet"},
     683                 :             :             }
     684   [ #  #  #  #  :           0 :         },
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
                   #  # ]
     685                 :           0 :         RPCExamples{
     686   [ #  #  #  #  :           0 :             HelpExampleCli("migratewallet", "")
                   #  # ]
     687   [ #  #  #  #  :           0 :             + HelpExampleRpc("migratewallet", "")
             #  #  #  # ]
     688         [ #  # ]:           0 :         },
     689                 :           0 :         [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
     690                 :             :         {
     691         [ #  # ]:           0 :             std::string wallet_name;
     692   [ #  #  #  # ]:           0 :             if (GetWalletNameFromJSONRPCRequest(request, wallet_name)) {
     693   [ #  #  #  #  :           0 :                 if (!(request.params[0].isNull() || request.params[0].get_str() == wallet_name)) {
          #  #  #  #  #  
                      # ]
     694   [ #  #  #  # ]:           0 :                     throw JSONRPCError(RPC_INVALID_PARAMETER, "RPC endpoint wallet and wallet_name parameter specify different wallets");
     695                 :             :                 }
     696                 :             :             } else {
     697   [ #  #  #  # ]:           0 :                 if (request.params[0].isNull()) {
     698   [ #  #  #  # ]:           0 :                     throw JSONRPCError(RPC_INVALID_PARAMETER, "Either RPC endpoint wallet or wallet_name parameter must be provided");
     699                 :             :                 }
     700   [ #  #  #  #  :           0 :                 wallet_name = request.params[0].get_str();
                   #  # ]
     701                 :             :             }
     702                 :             : 
     703         [ #  # ]:           0 :             SecureString wallet_pass;
     704         [ #  # ]:           0 :             wallet_pass.reserve(100);
     705   [ #  #  #  # ]:           0 :             if (!request.params[1].isNull()) {
     706   [ #  #  #  #  :           0 :                 wallet_pass = std::string_view{request.params[1].get_str()};
                   #  # ]
     707                 :             :             }
     708                 :             : 
     709         [ #  # ]:           0 :             WalletContext& context = EnsureWalletContext(request.context);
     710         [ #  # ]:           0 :             util::Result<MigrationResult> res = MigrateLegacyToDescriptor(wallet_name, wallet_pass, context);
     711         [ #  # ]:           0 :             if (!res) {
     712   [ #  #  #  # ]:           0 :                 throw JSONRPCError(RPC_WALLET_ERROR, util::ErrorString(res).original);
     713                 :             :             }
     714                 :             : 
     715                 :           0 :             UniValue r{UniValue::VOBJ};
     716   [ #  #  #  #  :           0 :             r.pushKV("wallet_name", res->wallet_name);
                   #  # ]
     717         [ #  # ]:           0 :             if (res->watchonly_wallet) {
     718   [ #  #  #  #  :           0 :                 r.pushKV("watchonly_name", res->watchonly_wallet->GetName());
                   #  # ]
     719                 :             :             }
     720         [ #  # ]:           0 :             if (res->solvables_wallet) {
     721   [ #  #  #  #  :           0 :                 r.pushKV("solvables_name", res->solvables_wallet->GetName());
                   #  # ]
     722                 :             :             }
     723   [ #  #  #  #  :           0 :             r.pushKV("backup_path", res->backup_path.utf8string());
             #  #  #  # ]
     724                 :             : 
     725                 :           0 :             return r;
     726                 :           0 :         },
     727   [ #  #  #  #  :           0 :     };
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
                      # ]
     728   [ #  #  #  #  :           0 : }
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
                      # ]
     729                 :             : 
     730                 :           0 : RPCHelpMan gethdkeys()
     731                 :             : {
     732                 :           0 :     return RPCHelpMan{
     733                 :             :         "gethdkeys",
     734                 :             :         "List all BIP 32 HD keys in the wallet and which descriptors use them.\n",
     735                 :             :         {
     736         [ #  # ]:           0 :             {"options", RPCArg::Type::OBJ_NAMED_PARAMS, RPCArg::Optional::OMITTED, "", {
     737         [ #  # ]:           0 :                 {"active_only", RPCArg::Type::BOOL, RPCArg::Default{false}, "Show the keys for only active descriptors"},
     738         [ #  # ]:           0 :                 {"private", RPCArg::Type::BOOL, RPCArg::Default{false}, "Show private keys"}
     739                 :             :             }},
     740                 :             :         },
     741                 :           0 :         RPCResult{RPCResult::Type::ARR, "", "", {
     742                 :             :             {
     743                 :             :                 {RPCResult::Type::OBJ, "", "", {
     744                 :             :                     {RPCResult::Type::STR, "xpub", "The extended public key"},
     745                 :             :                     {RPCResult::Type::BOOL, "has_private", "Whether the wallet has the private key for this xpub"},
     746                 :             :                     {RPCResult::Type::STR, "xprv", /*optional=*/true, "The extended private key if \"private\" is true"},
     747                 :             :                     {RPCResult::Type::ARR, "descriptors", "Array of descriptor objects that use this HD key",
     748                 :             :                     {
     749                 :             :                         {RPCResult::Type::OBJ, "", "", {
     750                 :             :                             {RPCResult::Type::STR, "desc", "Descriptor string representation"},
     751                 :             :                             {RPCResult::Type::BOOL, "active", "Whether this descriptor is currently used to generate new addresses"},
     752                 :             :                         }},
     753                 :             :                     }},
     754                 :             :                 }},
     755                 :             :             }
     756   [ #  #  #  #  :           0 :         }},
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
                #  #  # ]
     757                 :           0 :         RPCExamples{
     758   [ #  #  #  #  :           0 :             HelpExampleCli("gethdkeys", "") + HelpExampleRpc("gethdkeys", "")
          #  #  #  #  #  
             #  #  #  #  
                      # ]
     759   [ #  #  #  #  :           0 :             + HelpExampleCliNamed("gethdkeys", {{"active_only", "true"}, {"private", "true"}}) + HelpExampleRpcNamed("gethdkeys", {{"active_only", "true"}, {"private", "true"}})
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
             #  #  #  #  
                      # ]
     760         [ #  # ]:           0 :         },
     761                 :           0 :         [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
     762                 :             :         {
     763         [ #  # ]:           0 :             const std::shared_ptr<const CWallet> wallet = GetWalletForJSONRPCRequest(request);
     764         [ #  # ]:           0 :             if (!wallet) return UniValue::VNULL;
     765                 :             : 
     766   [ #  #  #  # ]:           0 :             if (!wallet->IsWalletFlagSet(WALLET_FLAG_DESCRIPTORS)) {
     767   [ #  #  #  # ]:           0 :                 throw JSONRPCError(RPC_WALLET_ERROR, "gethdkeys is not available for non-descriptor wallets");
     768                 :             :             }
     769                 :             : 
     770         [ #  # ]:           0 :             LOCK(wallet->cs_wallet);
     771                 :             : 
     772   [ #  #  #  #  :           0 :             UniValue options{request.params[0].isNull() ? UniValue::VOBJ : request.params[0]};
             #  #  #  # ]
     773   [ #  #  #  #  :           0 :             const bool active_only{options.exists("active_only") ? options["active_only"].get_bool() : false};
          #  #  #  #  #  
                      # ]
     774   [ #  #  #  #  :           0 :             const bool priv{options.exists("private") ? options["private"].get_bool() : false};
          #  #  #  #  #  
                      # ]
     775         [ #  # ]:           0 :             if (priv) {
     776         [ #  # ]:           0 :                 EnsureWalletIsUnlocked(*wallet);
     777                 :             :             }
     778                 :             : 
     779                 :             : 
     780         [ #  # ]:           0 :             std::set<ScriptPubKeyMan*> spkms;
     781         [ #  # ]:           0 :             if (active_only) {
     782         [ #  # ]:           0 :                 spkms = wallet->GetActiveScriptPubKeyMans();
     783                 :             :             } else {
     784         [ #  # ]:           0 :                 spkms = wallet->GetAllScriptPubKeyMans();
     785                 :             :             }
     786                 :             : 
     787                 :           0 :             std::map<CExtPubKey, std::set<std::tuple<std::string, bool, bool>>> wallet_xpubs;
     788                 :           0 :             std::map<CExtPubKey, CExtKey> wallet_xprvs;
     789         [ #  # ]:           0 :             for (auto* spkm : spkms) {
     790         [ #  # ]:           0 :                 auto* desc_spkm{dynamic_cast<DescriptorScriptPubKeyMan*>(spkm)};
     791         [ #  # ]:           0 :                 CHECK_NONFATAL(desc_spkm);
     792         [ #  # ]:           0 :                 LOCK(desc_spkm->cs_desc_man);
     793         [ #  # ]:           0 :                 WalletDescriptor w_desc = desc_spkm->GetWalletDescriptor();
     794                 :             : 
     795                 :             :                 // Retrieve the pubkeys from the descriptor
     796         [ #  # ]:           0 :                 std::set<CPubKey> desc_pubkeys;
     797                 :           0 :                 std::set<CExtPubKey> desc_xpubs;
     798         [ #  # ]:           0 :                 w_desc.descriptor->GetPubKeys(desc_pubkeys, desc_xpubs);
     799         [ #  # ]:           0 :                 for (const CExtPubKey& xpub : desc_xpubs) {
     800         [ #  # ]:           0 :                     std::string desc_str;
     801         [ #  # ]:           0 :                     bool ok = desc_spkm->GetDescriptorString(desc_str, false);
     802         [ #  # ]:           0 :                     CHECK_NONFATAL(ok);
     803   [ #  #  #  #  :           0 :                     wallet_xpubs[xpub].emplace(desc_str, wallet->IsActiveScriptPubKeyMan(*spkm), desc_spkm->HasPrivKey(xpub.pubkey.GetID()));
          #  #  #  #  #  
                      # ]
     804   [ #  #  #  #  :           0 :                     if (std::optional<CKey> key = priv ? desc_spkm->GetKey(xpub.pubkey.GetID()) : std::nullopt) {
             #  #  #  # ]
     805   [ #  #  #  # ]:           0 :                         wallet_xprvs[xpub] = CExtKey(xpub, *key);
     806                 :           0 :                     }
     807                 :           0 :                 }
     808         [ #  # ]:           0 :             }
     809                 :             : 
     810                 :           0 :             UniValue response(UniValue::VARR);
     811         [ #  # ]:           0 :             for (const auto& [xpub, descs] : wallet_xpubs) {
     812                 :           0 :                 bool has_xprv = false;
     813                 :           0 :                 UniValue descriptors(UniValue::VARR);
     814         [ #  # ]:           0 :                 for (const auto& [desc, active, has_priv] : descs) {
     815                 :           0 :                     UniValue d(UniValue::VOBJ);
     816   [ #  #  #  #  :           0 :                     d.pushKV("desc", desc);
                   #  # ]
     817   [ #  #  #  #  :           0 :                     d.pushKV("active", active);
                   #  # ]
     818                 :           0 :                     has_xprv |= has_priv;
     819                 :             : 
     820         [ #  # ]:           0 :                     descriptors.push_back(std::move(d));
     821                 :           0 :                 }
     822                 :           0 :                 UniValue xpub_info(UniValue::VOBJ);
     823   [ #  #  #  #  :           0 :                 xpub_info.pushKV("xpub", EncodeExtPubKey(xpub));
             #  #  #  # ]
     824   [ #  #  #  #  :           0 :                 xpub_info.pushKV("has_private", has_xprv);
                   #  # ]
     825         [ #  # ]:           0 :                 if (priv) {
     826   [ #  #  #  #  :           0 :                     xpub_info.pushKV("xprv", EncodeExtKey(wallet_xprvs.at(xpub)));
          #  #  #  #  #  
                      # ]
     827                 :             :                 }
     828   [ #  #  #  # ]:           0 :                 xpub_info.pushKV("descriptors", std::move(descriptors));
     829                 :             : 
     830         [ #  # ]:           0 :                 response.push_back(std::move(xpub_info));
     831                 :           0 :             }
     832                 :             : 
     833                 :           0 :             return response;
     834         [ #  # ]:           0 :         },
     835   [ #  #  #  #  :           0 :     };
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
                      # ]
     836   [ #  #  #  #  :           0 : }
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
             #  #  #  #  
                      # ]
     837                 :             : 
     838                 :           0 : static RPCHelpMan createwalletdescriptor()
     839                 :             : {
     840                 :           0 :     return RPCHelpMan{"createwalletdescriptor",
     841                 :             :         "Creates the wallet's descriptor for the given address type. "
     842                 :             :         "The address type must be one that the wallet does not already have a descriptor for."
     843         [ #  # ]:           0 :         + HELP_REQUIRING_PASSPHRASE,
     844                 :             :         {
     845         [ #  # ]:           0 :             {"type", RPCArg::Type::STR, RPCArg::Optional::NO, "The address type the descriptor will produce. Options are \"legacy\", \"p2sh-segwit\", \"bech32\", and \"bech32m\"."},
     846         [ #  # ]:           0 :             {"options", RPCArg::Type::OBJ_NAMED_PARAMS, RPCArg::Optional::OMITTED, "", {
     847         [ #  # ]:           0 :                 {"internal", RPCArg::Type::BOOL, RPCArg::DefaultHint{"Both external and internal will be generated unless this parameter is specified"}, "Whether to only make one descriptor that is internal (if parameter is true) or external (if parameter is false)"},
     848         [ #  # ]:           0 :                 {"hdkey", RPCArg::Type::STR, RPCArg::DefaultHint{"The HD key used by all other active descriptors"}, "The HD key that the wallet knows the private key of, listed using 'gethdkeys', to use for this descriptor's key"},
     849                 :             :             }},
     850                 :             :         },
     851                 :           0 :         RPCResult{
     852                 :             :             RPCResult::Type::OBJ, "", "",
     853                 :             :             {
     854                 :             :                 {RPCResult::Type::ARR, "descs", "The public descriptors that were added to the wallet",
     855                 :             :                     {{RPCResult::Type::STR, "", ""}}
     856                 :             :                 }
     857                 :             :             },
     858   [ #  #  #  #  :           0 :         },
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
             #  #  #  # ]
     859                 :           0 :         RPCExamples{
     860   [ #  #  #  #  :           0 :             HelpExampleCli("createwalletdescriptor", "bech32m")
                   #  # ]
     861   [ #  #  #  #  :           0 :             + HelpExampleRpc("createwalletdescriptor", "bech32m")
             #  #  #  # ]
     862         [ #  # ]:           0 :         },
     863                 :           0 :         [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
     864                 :             :         {
     865                 :           0 :             std::shared_ptr<CWallet> const pwallet = GetWalletForJSONRPCRequest(request);
     866         [ #  # ]:           0 :             if (!pwallet) return UniValue::VNULL;
     867                 :             : 
     868                 :             :             //  Make sure wallet is a descriptor wallet
     869   [ #  #  #  # ]:           0 :             if (!pwallet->IsWalletFlagSet(WALLET_FLAG_DESCRIPTORS)) {
     870   [ #  #  #  # ]:           0 :                 throw JSONRPCError(RPC_WALLET_ERROR, "createwalletdescriptor is not available for non-descriptor wallets");
     871                 :             :             }
     872                 :             : 
     873   [ #  #  #  #  :           0 :             std::optional<OutputType> output_type = ParseOutputType(request.params[0].get_str());
                   #  # ]
     874         [ #  # ]:           0 :             if (!output_type) {
     875   [ #  #  #  #  :           0 :                 throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, strprintf("Unknown address type '%s'", request.params[0].get_str()));
             #  #  #  # ]
     876                 :             :             }
     877                 :             : 
     878   [ #  #  #  #  :           0 :             UniValue options{request.params[1].isNull() ? UniValue::VOBJ : request.params[1]};
             #  #  #  # ]
     879   [ #  #  #  #  :           0 :             UniValue internal_only{options["internal"]};
                   #  # ]
     880   [ #  #  #  #  :           0 :             UniValue hdkey{options["hdkey"]};
                   #  # ]
     881                 :             : 
     882                 :           0 :             std::vector<bool> internals;
     883         [ #  # ]:           0 :             if (internal_only.isNull()) {
     884         [ #  # ]:           0 :                 internals.push_back(false);
     885         [ #  # ]:           0 :                 internals.push_back(true);
     886                 :             :             } else {
     887   [ #  #  #  # ]:           0 :                 internals.push_back(internal_only.get_bool());
     888                 :             :             }
     889                 :             : 
     890         [ #  # ]:           0 :             LOCK(pwallet->cs_wallet);
     891         [ #  # ]:           0 :             EnsureWalletIsUnlocked(*pwallet);
     892                 :             : 
     893         [ #  # ]:           0 :             CExtPubKey xpub;
     894         [ #  # ]:           0 :             if (hdkey.isNull()) {
     895         [ #  # ]:           0 :                 std::set<CExtPubKey> active_xpubs = pwallet->GetActiveHDPubKeys();
     896         [ #  # ]:           0 :                 if (active_xpubs.size() != 1) {
     897   [ #  #  #  # ]:           0 :                     throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Unable to determine which HD key to use from active descriptors. Please specify with 'hdkey'");
     898                 :             :                 }
     899                 :           0 :                 xpub = *active_xpubs.begin();
     900                 :           0 :             } else {
     901   [ #  #  #  # ]:           0 :                 xpub = DecodeExtPubKey(hdkey.get_str());
     902         [ #  # ]:           0 :                 if (!xpub.pubkey.IsValid()) {
     903   [ #  #  #  # ]:           0 :                     throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Unable to parse HD key. Please provide a valid xpub");
     904                 :             :                 }
     905                 :             :             }
     906                 :             : 
     907   [ #  #  #  # ]:           0 :             std::optional<CKey> key = pwallet->GetKey(xpub.pubkey.GetID());
     908         [ #  # ]:           0 :             if (!key) {
     909   [ #  #  #  #  :           0 :                 throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, strprintf("Private key for %s is not known", EncodeExtPubKey(xpub)));
                   #  # ]
     910                 :             :             }
     911         [ #  # ]:           0 :             CExtKey active_hdkey(xpub, *key);
     912                 :             : 
     913                 :           0 :             std::vector<std::reference_wrapper<DescriptorScriptPubKeyMan>> spkms;
     914         [ #  # ]:           0 :             WalletBatch batch{pwallet->GetDatabase()};
     915   [ #  #  #  # ]:           0 :             for (bool internal : internals) {
     916         [ #  # ]:           0 :                 WalletDescriptor w_desc = GenerateWalletDescriptor(xpub, *output_type, internal);
     917         [ #  # ]:           0 :                 uint256 w_id = DescriptorID(*w_desc.descriptor);
     918   [ #  #  #  # ]:           0 :                 if (!pwallet->GetScriptPubKeyMan(w_id)) {
     919   [ #  #  #  # ]:           0 :                     spkms.emplace_back(pwallet->SetupDescriptorScriptPubKeyMan(batch, active_hdkey, *output_type, internal));
     920                 :             :                 }
     921                 :           0 :             }
     922         [ #  # ]:           0 :             if (spkms.empty()) {
     923   [ #  #  #  # ]:           0 :                 throw JSONRPCError(RPC_WALLET_ERROR, "Descriptor already exists");
     924                 :             :             }
     925                 :             : 
     926                 :             :             // Fetch each descspkm from the wallet in order to get the descriptor strings
     927                 :           0 :             UniValue descs{UniValue::VARR};
     928         [ #  # ]:           0 :             for (const auto& spkm : spkms) {
     929         [ #  # ]:           0 :                 std::string desc_str;
     930         [ #  # ]:           0 :                 bool ok = spkm.get().GetDescriptorString(desc_str, false);
     931         [ #  # ]:           0 :                 CHECK_NONFATAL(ok);
     932   [ #  #  #  # ]:           0 :                 descs.push_back(desc_str);
     933                 :           0 :             }
     934                 :           0 :             UniValue out{UniValue::VOBJ};
     935   [ #  #  #  # ]:           0 :             out.pushKV("descs", std::move(descs));
     936                 :           0 :             return out;
     937         [ #  # ]:           0 :         }
     938   [ #  #  #  #  :           0 :     };
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
                #  #  # ]
     939   [ #  #  #  #  :           0 : }
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
                      # ]
     940                 :             : 
     941                 :             : // addresses
     942                 :             : RPCHelpMan getaddressinfo();
     943                 :             : RPCHelpMan getnewaddress();
     944                 :             : RPCHelpMan getrawchangeaddress();
     945                 :             : RPCHelpMan setlabel();
     946                 :             : RPCHelpMan listaddressgroupings();
     947                 :             : RPCHelpMan keypoolrefill();
     948                 :             : RPCHelpMan getaddressesbylabel();
     949                 :             : RPCHelpMan listlabels();
     950                 :             : #ifdef ENABLE_EXTERNAL_SIGNER
     951                 :             : RPCHelpMan walletdisplayaddress();
     952                 :             : #endif // ENABLE_EXTERNAL_SIGNER
     953                 :             : 
     954                 :             : // backup
     955                 :             : RPCHelpMan importprunedfunds();
     956                 :             : RPCHelpMan removeprunedfunds();
     957                 :             : RPCHelpMan importdescriptors();
     958                 :             : RPCHelpMan listdescriptors();
     959                 :             : RPCHelpMan backupwallet();
     960                 :             : RPCHelpMan restorewallet();
     961                 :             : 
     962                 :             : // coins
     963                 :             : RPCHelpMan getreceivedbyaddress();
     964                 :             : RPCHelpMan getreceivedbylabel();
     965                 :             : RPCHelpMan getbalance();
     966                 :             : RPCHelpMan getunconfirmedbalance();
     967                 :             : RPCHelpMan lockunspent();
     968                 :             : RPCHelpMan listlockunspent();
     969                 :             : RPCHelpMan getbalances();
     970                 :             : RPCHelpMan listunspent();
     971                 :             : 
     972                 :             : // encryption
     973                 :             : RPCHelpMan walletpassphrase();
     974                 :             : RPCHelpMan walletpassphrasechange();
     975                 :             : RPCHelpMan walletlock();
     976                 :             : RPCHelpMan encryptwallet();
     977                 :             : 
     978                 :             : // spend
     979                 :             : RPCHelpMan sendtoaddress();
     980                 :             : RPCHelpMan sendmany();
     981                 :             : RPCHelpMan settxfee();
     982                 :             : RPCHelpMan fundrawtransaction();
     983                 :             : RPCHelpMan bumpfee();
     984                 :             : RPCHelpMan psbtbumpfee();
     985                 :             : RPCHelpMan send();
     986                 :             : RPCHelpMan sendall();
     987                 :             : RPCHelpMan walletprocesspsbt();
     988                 :             : RPCHelpMan walletcreatefundedpsbt();
     989                 :             : RPCHelpMan signrawtransactionwithwallet();
     990                 :             : 
     991                 :             : // signmessage
     992                 :             : RPCHelpMan signmessage();
     993                 :             : 
     994                 :             : // transactions
     995                 :             : RPCHelpMan listreceivedbyaddress();
     996                 :             : RPCHelpMan listreceivedbylabel();
     997                 :             : RPCHelpMan listtransactions();
     998                 :             : RPCHelpMan listsinceblock();
     999                 :             : RPCHelpMan gettransaction();
    1000                 :             : RPCHelpMan abandontransaction();
    1001                 :             : RPCHelpMan rescanblockchain();
    1002                 :             : RPCHelpMan abortrescan();
    1003                 :             : 
    1004                 :           0 : std::span<const CRPCCommand> GetWalletRPCCommands()
    1005                 :             : {
    1006                 :           0 :     static const CRPCCommand commands[]{
    1007                 :             :         {"rawtransactions", &fundrawtransaction},
    1008                 :             :         {"wallet", &abandontransaction},
    1009                 :             :         {"wallet", &abortrescan},
    1010                 :             :         {"wallet", &backupwallet},
    1011                 :             :         {"wallet", &bumpfee},
    1012                 :             :         {"wallet", &psbtbumpfee},
    1013                 :             :         {"wallet", &createwallet},
    1014                 :             :         {"wallet", &createwalletdescriptor},
    1015                 :             :         {"wallet", &restorewallet},
    1016                 :             :         {"wallet", &encryptwallet},
    1017                 :             :         {"wallet", &getaddressesbylabel},
    1018                 :             :         {"wallet", &getaddressinfo},
    1019                 :             :         {"wallet", &getbalance},
    1020                 :             :         {"wallet", &gethdkeys},
    1021                 :             :         {"wallet", &getnewaddress},
    1022                 :             :         {"wallet", &getrawchangeaddress},
    1023                 :             :         {"wallet", &getreceivedbyaddress},
    1024                 :             :         {"wallet", &getreceivedbylabel},
    1025                 :             :         {"wallet", &gettransaction},
    1026                 :             :         {"wallet", &getunconfirmedbalance},
    1027                 :             :         {"wallet", &getbalances},
    1028                 :             :         {"wallet", &getwalletinfo},
    1029                 :             :         {"wallet", &importdescriptors},
    1030                 :             :         {"wallet", &importprunedfunds},
    1031                 :             :         {"wallet", &keypoolrefill},
    1032                 :             :         {"wallet", &listaddressgroupings},
    1033                 :             :         {"wallet", &listdescriptors},
    1034                 :             :         {"wallet", &listlabels},
    1035                 :             :         {"wallet", &listlockunspent},
    1036                 :             :         {"wallet", &listreceivedbyaddress},
    1037                 :             :         {"wallet", &listreceivedbylabel},
    1038                 :             :         {"wallet", &listsinceblock},
    1039                 :             :         {"wallet", &listtransactions},
    1040                 :             :         {"wallet", &listunspent},
    1041                 :             :         {"wallet", &listwalletdir},
    1042                 :             :         {"wallet", &listwallets},
    1043                 :             :         {"wallet", &loadwallet},
    1044                 :             :         {"wallet", &lockunspent},
    1045                 :             :         {"wallet", &migratewallet},
    1046                 :             :         {"wallet", &removeprunedfunds},
    1047                 :             :         {"wallet", &rescanblockchain},
    1048                 :             :         {"wallet", &send},
    1049                 :             :         {"wallet", &sendmany},
    1050                 :             :         {"wallet", &sendtoaddress},
    1051                 :             :         {"wallet", &setlabel},
    1052                 :             :         {"wallet", &settxfee},
    1053                 :             :         {"wallet", &setwalletflag},
    1054                 :             :         {"wallet", &signmessage},
    1055                 :             :         {"wallet", &signrawtransactionwithwallet},
    1056                 :             :         {"wallet", &simulaterawtransaction},
    1057                 :             :         {"wallet", &sendall},
    1058                 :             :         {"wallet", &unloadwallet},
    1059                 :             :         {"wallet", &upgradewallet},
    1060                 :             :         {"wallet", &walletcreatefundedpsbt},
    1061                 :             : #ifdef ENABLE_EXTERNAL_SIGNER
    1062                 :             :         {"wallet", &walletdisplayaddress},
    1063                 :             : #endif // ENABLE_EXTERNAL_SIGNER
    1064                 :             :         {"wallet", &walletlock},
    1065                 :             :         {"wallet", &walletpassphrase},
    1066                 :             :         {"wallet", &walletpassphrasechange},
    1067                 :             :         {"wallet", &walletprocesspsbt},
    1068   [ #  #  #  #  :           0 :     };
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
             #  #  #  # ]
    1069                 :           0 :     return commands;
    1070                 :             : }
    1071                 :             : } // namespace wallet
        

Generated by: LCOV version 2.0-1