LCOV - code coverage report
Current view: top level - src/wallet/rpc - spend.cpp (source / functions) Coverage Total Hit
Test: total_coverage.info Lines: 97.2 % 1102 1071
Test Date: 2026-06-27 07:20:58 Functions: 100.0 % 32 32
Branches: 52.5 % 4683 2460

             Branch data     Line data    Source code
       1                 :             : // Copyright (c) 2011-present The Bitcoin Core developers
       2                 :             : // Distributed under the MIT software license, see the accompanying
       3                 :             : // file COPYING or http://www.opensource.org/licenses/mit-license.php.
       4                 :             : 
       5                 :             : #include <common/messages.h>
       6                 :             : #include <consensus/validation.h>
       7                 :             : #include <core_io.h>
       8                 :             : #include <key_io.h>
       9                 :             : #include <node/types.h>
      10                 :             : #include <policy/policy.h>
      11                 :             : #include <policy/truc_policy.h>
      12                 :             : #include <rpc/rawtransaction_util.h>
      13                 :             : #include <rpc/util.h>
      14                 :             : #include <script/script.h>
      15                 :             : #include <util/rbf.h>
      16                 :             : #include <util/translation.h>
      17                 :             : #include <util/vector.h>
      18                 :             : #include <wallet/coincontrol.h>
      19                 :             : #include <wallet/feebumper.h>
      20                 :             : #include <wallet/fees.h>
      21                 :             : #include <wallet/rpc/util.h>
      22                 :             : #include <wallet/spend.h>
      23                 :             : #include <wallet/wallet.h>
      24                 :             : 
      25                 :             : #include <univalue.h>
      26                 :             : 
      27                 :             : using common::FeeModeFromString;
      28                 :             : using common::FeeModesDetail;
      29                 :             : using common::InvalidEstimateModeErrorMessage;
      30                 :             : using common::StringForFeeReason;
      31                 :             : using common::TransactionErrorString;
      32                 :             : using node::TransactionError;
      33                 :             : 
      34                 :             : namespace wallet {
      35                 :        1953 : std::vector<CRecipient> CreateRecipients(const std::vector<std::pair<CTxDestination, CAmount>>& outputs, const std::set<int>& subtract_fee_outputs)
      36                 :             : {
      37                 :        1953 :     std::vector<CRecipient> recipients;
      38   [ -  +  +  + ]:       22986 :     for (size_t i = 0; i < outputs.size(); ++i) {
      39   [ +  -  +  - ]:       21033 :         const auto& [destination, amount] = outputs.at(i);
      40         [ +  - ]:       42066 :         CRecipient recipient{destination, amount, subtract_fee_outputs.contains(i)};
      41         [ +  - ]:       21033 :         recipients.push_back(recipient);
      42                 :       21033 :     }
      43                 :        1953 :     return recipients;
      44                 :           0 : }
      45                 :             : 
      46                 :         394 : static void InterpretFeeEstimationInstructions(const UniValue& conf_target, const UniValue& estimate_mode, const UniValue& fee_rate, UniValue& options)
      47                 :             : {
      48   [ +  -  +  -  :        1159 :     if (options.exists("conf_target") || options.exists("estimate_mode")) {
          +  +  +  -  +  
          -  +  +  +  +  
                   -  - ]
      49   [ +  +  -  + ]:          24 :         if (!conf_target.isNull() || !estimate_mode.isNull()) {
      50   [ +  -  +  - ]:           6 :             throw JSONRPCError(RPC_INVALID_PARAMETER, "Pass conf_target and estimate_mode either as arguments or in the options object, but not both");
      51                 :             :         }
      52                 :             :     } else {
      53   [ +  -  +  - ]:         740 :         options.pushKV("conf_target", conf_target);
      54   [ +  -  +  - ]:         740 :         options.pushKV("estimate_mode", estimate_mode);
      55                 :             :     }
      56         [ +  + ]:         782 :     if (options.exists("fee_rate")) {
      57         [ +  + ]:          28 :         if (!fee_rate.isNull()) {
      58   [ +  -  +  - ]:           2 :             throw JSONRPCError(RPC_INVALID_PARAMETER, "Pass the fee_rate either as an argument, or in the options object, but not both");
      59                 :             :         }
      60                 :             :     } else {
      61   [ +  -  +  - ]:         726 :         options.pushKV("fee_rate", fee_rate);
      62                 :             :     }
      63   [ +  -  +  -  :         838 :     if (!options["conf_target"].isNull() && (options["estimate_mode"].isNull() || (options["estimate_mode"].get_str() == "unset"))) {
          +  +  +  -  +  
          -  +  -  +  -  
          +  -  +  -  -  
          +  +  +  -  +  
             -  -  -  - ]
      64   [ #  #  #  # ]:           0 :         throw JSONRPCError(RPC_INVALID_PARAMETER, "Specify estimate_mode");
      65                 :             :     }
      66                 :         390 : }
      67                 :             : 
      68                 :         750 : std::set<int> InterpretSubtractFeeFromOutputInstructions(const UniValue& sffo_instructions, const std::vector<std::string>& destinations)
      69                 :             : {
      70         [ +  + ]:         750 :     std::set<int> sffo_set;
      71         [ +  + ]:         750 :     if (sffo_instructions.isNull()) return sffo_set;
      72                 :             : 
      73   [ +  -  +  + ]:         200 :     for (const auto& sffo : sffo_instructions.getValues()) {
      74                 :         105 :         int pos{-1};
      75         [ +  + ]:         105 :         if (sffo.isStr()) {
      76         [ +  - ]:           9 :             auto it = find(destinations.begin(), destinations.end(), sffo.get_str());
      77   [ +  +  +  -  :          10 :             if (it == destinations.end()) throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Invalid parameter 'subtract fee from output', destination %s not found in tx outputs", sffo.get_str()));
             +  -  +  - ]
      78                 :           8 :             pos = it - destinations.begin();
      79         [ +  + ]:          96 :         } else if (sffo.isNum()) {
      80         [ +  - ]:          95 :             pos = sffo.getInt<int>();
      81                 :             :         } else {
      82   [ +  -  +  -  :           2 :             throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Invalid parameter 'subtract fee from output', invalid value type: %s", uvTypeName(sffo.type())));
                   +  - ]
      83                 :             :         }
      84                 :             : 
      85         [ +  + ]:         103 :         if (sffo_set.contains(pos))
      86   [ +  -  +  - ]:           4 :             throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Invalid parameter 'subtract fee from output', duplicated position: %d", pos));
      87         [ +  + ]:         101 :         if (pos < 0)
      88   [ +  -  +  - ]:           2 :             throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Invalid parameter 'subtract fee from output', negative position: %d", pos));
      89   [ -  +  +  + ]:         100 :         if (pos >= int(destinations.size()))
      90   [ +  -  +  - ]:           2 :             throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Invalid parameter 'subtract fee from output', position too large: %d", pos));
      91         [ +  - ]:          99 :         sffo_set.insert(pos);
      92                 :             :     }
      93                 :             :     return sffo_set;
      94                 :           6 : }
      95                 :             : 
      96                 :         249 : static UniValue FinishTransaction(const std::shared_ptr<CWallet> pwallet, const UniValue& options, CMutableTransaction& rawTx)
      97                 :             : {
      98                 :         498 :     bool can_anti_fee_snipe = !options.exists("locktime");
      99                 :             : 
     100         [ +  + ]:        1663 :     for (const CTxIn& tx_in : rawTx.vin) {
     101                 :             :         // Checks sequence values consistent with DiscourageFeeSniping
     102   [ +  +  +  + ]:        1427 :         can_anti_fee_snipe = can_anti_fee_snipe && (tx_in.nSequence == CTxIn::MAX_SEQUENCE_NONFINAL || tx_in.nSequence == MAX_BIP125_RBF_SEQUENCE);
     103                 :             :     }
     104                 :             : 
     105         [ +  + ]:         249 :     if (can_anti_fee_snipe) {
     106                 :         238 :         LOCK(pwallet->cs_wallet);
     107                 :         238 :         FastRandomContext rng_fast;
     108         [ +  - ]:         238 :         DiscourageFeeSniping(rawTx, rng_fast, pwallet->chain(), pwallet->GetLastBlockHash(), pwallet->GetLastBlockHeight());
     109         [ +  - ]:         476 :     }
     110                 :             : 
     111                 :             :     // Make a blank psbt
     112                 :         249 :     PartiallySignedTransaction psbtx(rawTx, /*version=*/2);
     113                 :             : 
     114                 :             :     // First fill transaction with our data without signing,
     115                 :             :     // so external signers are not asked to sign more than once.
     116                 :         249 :     bool complete;
     117         [ +  - ]:         249 :     pwallet->FillPSBT(psbtx, {.sign = false, .bip32_derivs = true}, complete);
     118         [ +  - ]:         249 :     const auto err{pwallet->FillPSBT(psbtx, {.sign = true, .bip32_derivs = false}, complete)};
     119         [ +  + ]:         249 :     if (err) {
     120         [ +  - ]:           1 :         throw JSONRPCPSBTError(*err);
     121                 :             :     }
     122                 :             : 
     123         [ +  - ]:         248 :     CMutableTransaction mtx;
     124         [ +  - ]:         248 :     complete = FinalizeAndExtractPSBT(psbtx, mtx);
     125                 :             : 
     126                 :         248 :     UniValue result(UniValue::VOBJ);
     127                 :             : 
     128   [ +  -  +  -  :         269 :     const bool psbt_opt_in{options.exists("psbt") && options["psbt"].get_bool()};
          +  +  +  -  +  
          -  +  -  +  -  
                   -  - ]
     129   [ +  -  +  +  :         534 :     bool add_to_wallet{options.exists("add_to_wallet") ? options["add_to_wallet"].get_bool() : true};
          +  -  +  -  +  
                      - ]
     130   [ +  +  +  +  :         248 :     if (psbt_opt_in || !complete || !add_to_wallet) {
                   +  + ]
     131                 :             :         // Serialize the PSBT
     132                 :          63 :         DataStream ssTx{};
     133         [ +  - ]:          63 :         ssTx << psbtx;
     134   [ +  -  +  -  :         189 :         result.pushKV("psbt", EncodeBase64(ssTx.str()));
          +  -  +  -  +  
                      - ]
     135                 :          63 :     }
     136                 :             : 
     137         [ +  + ]:         248 :     if (complete) {
     138   [ +  -  +  - ]:         222 :         std::string hex{EncodeHexTx(CTransaction(mtx))};
     139         [ +  - ]:         222 :         CTransactionRef tx(MakeTransactionRef(std::move(mtx)));
     140   [ +  -  +  -  :         444 :         result.pushKV("txid", tx->GetHash().GetHex());
             +  -  +  - ]
     141         [ +  + ]:         222 :         if (add_to_wallet && !psbt_opt_in) {
     142   [ +  -  +  - ]:         555 :             pwallet->CommitTransaction(tx, {}, /*orderForm=*/{});
     143                 :             :         } else {
     144   [ +  -  +  -  :          74 :             result.pushKV("hex", hex);
                   +  - ]
     145                 :             :         }
     146                 :         222 :     }
     147   [ +  -  +  -  :         496 :     result.pushKV("complete", complete);
                   +  - ]
     148                 :             : 
     149                 :         496 :     return result;
     150                 :         249 : }
     151                 :             : 
     152                 :         390 : static void PreventOutdatedOptions(const UniValue& options)
     153                 :             : {
     154         [ +  + ]:         780 :     if (options.exists("feeRate")) {
     155   [ +  -  +  - ]:           3 :         throw JSONRPCError(RPC_INVALID_PARAMETER, "Use fee_rate (" + CURRENCY_ATOM + "/vB) instead of feeRate");
     156                 :             :     }
     157         [ -  + ]:         778 :     if (options.exists("changeAddress")) {
     158   [ #  #  #  # ]:           0 :         throw JSONRPCError(RPC_INVALID_PARAMETER, "Use change_address instead of changeAddress");
     159                 :             :     }
     160         [ -  + ]:         778 :     if (options.exists("changePosition")) {
     161   [ #  #  #  # ]:           0 :         throw JSONRPCError(RPC_INVALID_PARAMETER, "Use change_position instead of changePosition");
     162                 :             :     }
     163         [ -  + ]:         778 :     if (options.exists("lockUnspents")) {
     164   [ #  #  #  # ]:           0 :         throw JSONRPCError(RPC_INVALID_PARAMETER, "Use lock_unspents instead of lockUnspents");
     165                 :             :     }
     166         [ -  + ]:         778 :     if (options.exists("subtractFeeFromOutputs")) {
     167   [ #  #  #  # ]:           0 :         throw JSONRPCError(RPC_INVALID_PARAMETER, "Use subtract_fee_from_outputs instead of subtractFeeFromOutputs");
     168                 :             :     }
     169                 :         389 : }
     170                 :             : 
     171                 :        1264 : UniValue SendMoney(CWallet& wallet, const CCoinControl &coin_control, std::vector<CRecipient> &recipients, mapValue_t map_value, bool verbose)
     172                 :             : {
     173                 :        1264 :     EnsureWalletIsUnlocked(wallet);
     174                 :             : 
     175                 :             :     // This function is only used by sendtoaddress and sendmany.
     176                 :             :     // This should always try to sign, if we don't have (all) private keys, don't
     177                 :             :     // try to do anything here.
     178         [ +  + ]:        1264 :     if (wallet.IsWalletFlagSet(WALLET_FLAG_EXTERNAL_SIGNER)) {
     179   [ +  -  +  - ]:           4 :         throw JSONRPCError(RPC_WALLET_ERROR, "Error: sendtoaddress and sendmany are not supported for wallets with external signers; use send instead");
     180                 :             :     }
     181         [ +  + ]:        1262 :     if (wallet.IsWalletFlagSet(WALLET_FLAG_DISABLE_PRIVATE_KEYS)) {
     182   [ +  -  +  - ]:           2 :         throw JSONRPCError(RPC_WALLET_ERROR, "Error: Private keys are disabled for this wallet");
     183                 :             :     }
     184                 :             : 
     185                 :             :     // Shuffle recipient list
     186                 :        1261 :     std::shuffle(recipients.begin(), recipients.end(), FastRandomContext());
     187                 :             : 
     188                 :             :     // Send
     189                 :        1261 :     auto res = CreateTransaction(wallet, recipients, /*change_pos=*/std::nullopt, coin_control, true);
     190         [ +  + ]:        1261 :     if (!res) {
     191   [ +  -  +  - ]:          34 :         throw JSONRPCError(RPC_WALLET_INSUFFICIENT_FUNDS, util::ErrorString(res).original);
     192                 :             :     }
     193                 :        1244 :     const CTransactionRef& tx = res->tx;
     194   [ +  -  +  - ]:        3732 :     wallet.CommitTransaction(tx, std::move(map_value), /*orderForm=*/{});
     195         [ +  + ]:        1244 :     if (verbose) {
     196                 :           6 :         UniValue entry(UniValue::VOBJ);
     197   [ +  -  +  -  :          12 :         entry.pushKV("txid", tx->GetHash().GetHex());
             +  -  +  - ]
     198   [ +  -  +  -  :          12 :         entry.pushKV("fee_reason", StringForFeeReason(res->fee_calc.reason));
             +  -  +  - ]
     199                 :           6 :         return entry;
     200                 :           0 :     }
     201   [ +  -  +  - ]:        1238 :     return tx->GetHash().GetHex();
     202                 :        1244 : }
     203                 :             : 
     204                 :             : 
     205                 :             : /**
     206                 :             :  * Update coin control with fee estimation based on the given parameters
     207                 :             :  *
     208                 :             :  * @param[in]     wallet            Wallet reference
     209                 :             :  * @param[in,out] cc                Coin control to be updated
     210                 :             :  * @param[in]     conf_target       UniValue integer; confirmation target in blocks, values between 1 and 1008 are valid per policy/fees/block_policy_estimator.h;
     211                 :             :  * @param[in]     estimate_mode     UniValue string; fee estimation mode, valid values are "unset", "economical" or "conservative";
     212                 :             :  * @param[in]     fee_rate          UniValue real; fee rate in sat/vB;
     213                 :             :  *                                      if present, both conf_target and estimate_mode must either be null, or "unset"
     214                 :             :  * @param[in]     override_min_fee  bool; whether to set fOverrideFeeRate to true to disable minimum fee rate checks and instead
     215                 :             :  *                                      verify only that fee_rate is greater than 0
     216                 :             :  * @throws a JSONRPCError if conf_target, estimate_mode, or fee_rate contain invalid values or are in conflict
     217                 :             :  */
     218                 :        2132 : static void SetFeeEstimateMode(const CWallet& wallet, CCoinControl& cc, const UniValue& conf_target, const UniValue& estimate_mode, const UniValue& fee_rate, bool override_min_fee)
     219                 :             : {
     220         [ +  + ]:        2132 :     if (!fee_rate.isNull()) {
     221         [ +  + ]:         548 :         if (!conf_target.isNull()) {
     222   [ +  -  +  - ]:           6 :             throw JSONRPCError(RPC_INVALID_PARAMETER, "Cannot specify both conf_target and fee_rate. Please provide either a confirmation target in blocks for automatic fee estimation, or an explicit fee rate.");
     223                 :             :         }
     224   [ +  +  +  + ]:         545 :         if (!estimate_mode.isNull() && estimate_mode.get_str() != "unset") {
     225   [ +  -  +  - ]:           6 :             throw JSONRPCError(RPC_INVALID_PARAMETER, "Cannot specify both estimate_mode and fee_rate");
     226                 :             :         }
     227                 :             :         // Fee rates in sat/vB cannot represent more than 3 significant digits.
     228         [ -  + ]:         542 :         cc.m_feerate = CFeeRate{AmountFromValue(fee_rate, /*decimals=*/3)};
     229         [ +  + ]:         466 :         if (override_min_fee) cc.fOverrideFeeRate = true;
     230                 :             :         // Default RBF to true for explicit fee_rate, if unset.
     231         [ +  + ]:         466 :         if (!cc.m_signal_bip125_rbf) cc.m_signal_bip125_rbf = true;
     232                 :         466 :         return;
     233                 :             :     }
     234   [ +  +  -  +  :        1584 :     if (!estimate_mode.isNull() && !FeeModeFromString(estimate_mode.get_str(), cc.m_fee_mode)) {
                   +  + ]
     235   [ +  -  +  - ]:          54 :         throw JSONRPCError(RPC_INVALID_PARAMETER, InvalidEstimateModeErrorMessage());
     236                 :             :     }
     237         [ +  + ]:        1557 :     if (!conf_target.isNull()) {
     238                 :          32 :         cc.m_confirm_target = ParseConfirmTarget(conf_target, wallet.chain().estimateMaxBlocks());
     239                 :             :     }
     240                 :             : }
     241                 :             : 
     242                 :        2052 : RPCMethod sendtoaddress()
     243                 :             : {
     244                 :        2052 :     return RPCMethod{
     245                 :        2052 :         "sendtoaddress",
     246                 :        2052 :         "Send an amount to a given address." +
     247         [ +  - ]:        2052 :         HELP_REQUIRING_PASSPHRASE,
     248                 :             :                 {
     249   [ +  -  +  - ]:        4104 :                     {"address", RPCArg::Type::STR, RPCArg::Optional::NO, "The bitcoin address to send to."},
     250   [ +  -  +  - ]:        4104 :                     {"amount", RPCArg::Type::AMOUNT, RPCArg::Optional::NO, "The amount in " + CURRENCY_UNIT + " to send. eg 0.1"},
     251   [ +  -  +  - ]:        4104 :                     {"comment", RPCArg::Type::STR, RPCArg::Optional::OMITTED, "A comment used to store what the transaction is for.\n"
     252                 :             :                                          "This is not part of the transaction, just kept in your wallet."},
     253   [ +  -  +  - ]:        4104 :                     {"comment_to", RPCArg::Type::STR, RPCArg::Optional::OMITTED, "A comment to store the name of the person or organization\n"
     254                 :             :                                          "to which you're sending the transaction. This is not part of the \n"
     255                 :             :                                          "transaction, just kept in your wallet."},
     256   [ +  -  +  -  :        6156 :                     {"subtractfeefromamount", RPCArg::Type::BOOL, RPCArg::Default{false}, "The fee will be deducted from the amount being sent.\n"
                   +  - ]
     257                 :             :                                          "The recipient will receive less bitcoins than you enter in the amount field."},
     258   [ +  -  +  -  :        6156 :                     {"replaceable", RPCArg::Type::BOOL, RPCArg::DefaultHint{"wallet default"}, "Signal that this transaction can be replaced by a transaction (BIP 125)"},
                   +  - ]
     259   [ +  -  +  -  :        6156 :                     {"conf_target", RPCArg::Type::NUM, RPCArg::DefaultHint{"wallet -txconfirmtarget"}, "Confirmation target in blocks"},
                   +  - ]
     260   [ +  -  +  - ]:        4104 :                     {"estimate_mode", RPCArg::Type::STR, RPCArg::Default{"unset"}, "The fee estimate mode, must be one of (case insensitive):\n"
     261   [ +  -  +  -  :        4104 :                       + FeeModesDetail(std::string("economical mode is used if the transaction is replaceable;\notherwise, conservative mode is used"))},
                   +  - ]
     262   [ +  -  +  -  :        6156 :                     {"avoid_reuse", RPCArg::Type::BOOL, RPCArg::Default{true}, "(only available if avoid_reuse wallet flag is set) Avoid spending from dirty addresses; addresses are considered\n"
                   +  - ]
     263                 :             :                                          "dirty if they have previously been used in a transaction. If true, this also activates avoidpartialspends, grouping outputs by their addresses."},
     264   [ +  -  +  -  :        6156 :                     {"fee_rate", RPCArg::Type::AMOUNT, RPCArg::DefaultHint{"not set, fall back to wallet fee estimation"}, "Specify a fee rate in " + CURRENCY_ATOM + "/vB."},
                   +  - ]
     265   [ +  -  +  -  :        6156 :                     {"verbose", RPCArg::Type::BOOL, RPCArg::Default{false}, "If true, return extra information about the transaction."},
                   +  - ]
     266                 :             :                 },
     267                 :             :                 {
     268         [ +  - ]:        2052 :                     RPCResult{"if verbose is not set or set to false",
     269   [ +  -  +  - ]:        4104 :                         RPCResult::Type::STR_HEX, "txid", "The transaction id."
     270                 :        2052 :                     },
     271         [ +  - ]:        4104 :                     RPCResult{"if verbose is set to true",
     272   [ +  -  +  - ]:        4104 :                         RPCResult::Type::OBJ, "", "",
     273                 :             :                         {
     274   [ +  -  +  - ]:        4104 :                             {RPCResult::Type::STR_HEX, "txid", "The transaction id."},
     275   [ +  -  +  - ]:        4104 :                             {RPCResult::Type::STR, "fee_reason", "The transaction fee reason."}
     276                 :             :                         },
     277   [ +  -  +  +  :       12312 :                     },
                   -  - ]
     278                 :             :                 },
     279                 :        2052 :                 RPCExamples{
     280                 :             :                     "\nSend 0.1 BTC\n"
     281   [ +  -  +  -  :        6156 :                     + HelpExampleCli("sendtoaddress", "\"" + EXAMPLE_ADDRESS[0] + "\" 0.1") +
             +  -  +  - ]
     282                 :        2052 :                     "\nSend 0.1 BTC with a confirmation target of 6 blocks in economical fee estimate mode using positional arguments\n"
     283   [ +  -  +  -  :       10260 :                     + HelpExampleCli("sendtoaddress", "\"" + EXAMPLE_ADDRESS[0] + "\" 0.1 \"donation\" \"sean's outpost\" false true 6 economical") +
             +  -  +  - ]
     284         [ +  - ]:        4104 :                     "\nSend 0.1 BTC with a fee rate of 1.1 " + CURRENCY_ATOM + "/vB, subtract fee from amount, BIP125-replaceable, using positional arguments\n"
     285   [ +  -  +  -  :       10260 :                     + HelpExampleCli("sendtoaddress", "\"" + EXAMPLE_ADDRESS[0] + "\" 0.1 \"drinks\" \"room77\" true true null \"unset\" null 1.1") +
             +  -  +  - ]
     286                 :        2052 :                     "\nSend 0.2 BTC with a confirmation target of 6 blocks in economical fee estimate mode using named arguments\n"
     287   [ +  -  +  -  :       10260 :                     + HelpExampleCli("-named sendtoaddress", "address=\"" + EXAMPLE_ADDRESS[0] + "\" amount=0.2 conf_target=6 estimate_mode=\"economical\"") +
             +  -  +  - ]
     288         [ +  - ]:        4104 :                     "\nSend 0.5 BTC with a fee rate of 25 " + CURRENCY_ATOM + "/vB using named arguments\n"
     289   [ +  -  +  -  :       10260 :                     + HelpExampleCli("-named sendtoaddress", "address=\"" + EXAMPLE_ADDRESS[0] + "\" amount=0.5 fee_rate=25")
             +  -  +  - ]
     290   [ +  -  +  -  :        8208 :                     + HelpExampleCli("-named sendtoaddress", "address=\"" + EXAMPLE_ADDRESS[0] + "\" amount=0.5 fee_rate=25 subtractfeefromamount=false replaceable=true avoid_reuse=true comment=\"2 pizzas\" comment_to=\"jeremy\" verbose=true")
                   +  - ]
     291         [ +  - ]:        2052 :                 },
     292                 :        2052 :         [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
     293                 :             : {
     294                 :        1213 :     std::shared_ptr<CWallet> const pwallet = GetWalletForJSONRPCRequest(request);
     295         [ -  + ]:        1213 :     if (!pwallet) return UniValue::VNULL;
     296                 :             : 
     297                 :             :     // Make sure the results are valid at least up to the most recent block
     298                 :             :     // the user could have gotten from another RPC command prior to now
     299         [ +  - ]:        1213 :     pwallet->BlockUntilSyncedToCurrentChain();
     300                 :             : 
     301         [ +  - ]:        1213 :     LOCK(pwallet->cs_wallet);
     302                 :             : 
     303                 :             :     // Wallet comments
     304         [ +  - ]:        1213 :     mapValue_t mapValue;
     305   [ +  -  +  +  :        1213 :     if (!request.params[2].isNull() && !request.params[2].get_str().empty())
          +  -  +  -  +  
                      + ]
     306   [ +  -  +  -  :           2 :         mapValue["comment"] = request.params[2].get_str();
          +  -  +  -  +  
                      - ]
     307   [ +  -  +  +  :        1213 :     if (!request.params[3].isNull() && !request.params[3].get_str().empty())
          +  -  +  -  +  
                      + ]
     308   [ +  -  +  -  :           2 :         mapValue["to"] = request.params[3].get_str();
          +  -  +  -  +  
                      - ]
     309                 :             : 
     310         [ +  - ]:        1213 :     CCoinControl coin_control;
     311   [ +  -  +  + ]:        1213 :     if (!request.params[5].isNull()) {
     312   [ +  -  +  - ]:           2 :         coin_control.m_signal_bip125_rbf = request.params[5].get_bool();
     313                 :             :     }
     314                 :             : 
     315   [ +  -  +  - ]:        1213 :     coin_control.m_avoid_address_reuse = GetAvoidReuseFlag(*pwallet, request.params[8]);
     316                 :             :     // We also enable partial spend avoidance if reuse avoidance is set.
     317                 :        1213 :     coin_control.m_avoid_partial_spends |= coin_control.m_avoid_address_reuse;
     318                 :             : 
     319   [ +  -  +  -  :        1213 :     SetFeeEstimateMode(*pwallet, coin_control, /*conf_target=*/request.params[6], /*estimate_mode=*/request.params[7], /*fee_rate=*/request.params[9], /*override_min_fee=*/false);
             +  -  +  - ]
     320                 :             : 
     321         [ +  + ]:        1213 :     EnsureWalletIsUnlocked(*pwallet);
     322                 :             : 
     323                 :        1212 :     UniValue address_amounts(UniValue::VOBJ);
     324   [ +  -  +  -  :        1212 :     const std::string address = request.params[0].get_str();
                   -  + ]
     325   [ +  -  +  -  :        3636 :     address_amounts.pushKV(address, request.params[1]);
                   +  - ]
     326                 :             : 
     327         [ +  - ]:        1212 :     std::set<int> sffo_set;
     328   [ +  -  +  +  :        1212 :     if (!request.params[4].isNull() && request.params[4].get_bool()) {
          +  -  +  -  +  
                      + ]
     329         [ +  - ]:         216 :         sffo_set.insert(0);
     330                 :             :     }
     331                 :             : 
     332   [ +  +  +  - ]:        1212 :     std::vector<CRecipient> recipients{CreateRecipients(ParseOutputs(address_amounts), sffo_set)};
     333   [ +  -  +  +  :        1210 :     const bool verbose{request.params[10].isNull() ? false : request.params[10].get_bool()};
             +  -  +  - ]
     334                 :             : 
     335   [ +  -  +  + ]:        1217 :     return SendMoney(*pwallet, coin_control, recipients, mapValue, verbose);
     336         [ +  - ]:        3664 : },
     337   [ +  -  +  -  :       41040 :     };
          +  -  +  -  +  
          +  +  +  -  -  
                   -  - ]
     338   [ +  -  +  -  :       61560 : }
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  -  -  -  -  
                   -  - ]
     339                 :             : 
     340                 :         922 : RPCMethod sendmany()
     341                 :             : {
     342                 :         922 :     return RPCMethod{"sendmany",
     343                 :         922 :         "Send multiple times. Amounts are double-precision floating point numbers." +
     344         [ +  - ]:         922 :         HELP_REQUIRING_PASSPHRASE,
     345                 :             :                 {
     346   [ +  -  +  -  :        2766 :                     {"dummy", RPCArg::Type::STR, RPCArg::Default{"\"\""}, "Must be set to \"\" for backwards compatibility.",
                   +  - ]
     347         [ +  - ]:        1844 :                      RPCArgOptions{
     348                 :             :                          .oneline_description = "\"\"",
     349                 :             :                      }},
     350   [ +  -  +  - ]:        1844 :                     {"amounts", RPCArg::Type::OBJ_USER_KEYS, RPCArg::Optional::NO, "The addresses and amounts",
     351                 :             :                         {
     352   [ +  -  +  - ]:        1844 :                             {"address", RPCArg::Type::AMOUNT, RPCArg::Optional::NO, "The bitcoin address is the key, the numeric amount (can be string) in " + CURRENCY_UNIT + " is the value"},
     353                 :             :                         },
     354                 :             :                     },
     355   [ +  -  +  - ]:        1844 :                     {"minconf", RPCArg::Type::NUM, RPCArg::Optional::OMITTED, "Ignored dummy value"},
     356   [ +  -  +  - ]:        1844 :                     {"comment", RPCArg::Type::STR, RPCArg::Optional::OMITTED, "A comment"},
     357   [ +  -  +  - ]:        1844 :                     {"subtractfeefrom", RPCArg::Type::ARR, RPCArg::Optional::OMITTED, "The addresses.\n"
     358                 :             :                                        "The fee will be equally deducted from the amount of each selected address.\n"
     359                 :             :                                        "Those recipients will receive less bitcoins than you enter in their corresponding amount field.\n"
     360                 :             :                                        "If no addresses are specified here, the sender pays the fee.",
     361                 :             :                         {
     362   [ +  -  +  - ]:        1844 :                             {"address", RPCArg::Type::STR, RPCArg::Optional::OMITTED, "Subtract fee from this address"},
     363                 :             :                         },
     364                 :             :                     },
     365   [ +  -  +  -  :        2766 :                     {"replaceable", RPCArg::Type::BOOL, RPCArg::DefaultHint{"wallet default"}, "Signal that this transaction can be replaced by a transaction (BIP 125)"},
                   +  - ]
     366   [ +  -  +  -  :        2766 :                     {"conf_target", RPCArg::Type::NUM, RPCArg::DefaultHint{"wallet -txconfirmtarget"}, "Confirmation target in blocks"},
                   +  - ]
     367   [ +  -  +  - ]:        1844 :                     {"estimate_mode", RPCArg::Type::STR, RPCArg::Default{"unset"}, "The fee estimate mode, must be one of (case insensitive):\n"
     368   [ +  -  +  -  :        1844 :                       + FeeModesDetail(std::string("economical mode is used if the transaction is replaceable;\notherwise, conservative mode is used"))},
                   +  - ]
     369   [ +  -  +  -  :        2766 :                     {"fee_rate", RPCArg::Type::AMOUNT, RPCArg::DefaultHint{"not set, fall back to wallet fee estimation"}, "Specify a fee rate in " + CURRENCY_ATOM + "/vB."},
                   +  - ]
     370   [ +  -  +  -  :        2766 :                     {"verbose", RPCArg::Type::BOOL, RPCArg::Default{false}, "If true, return extra information about the transaction."},
                   +  - ]
     371                 :             :                 },
     372                 :             :                 {
     373         [ +  - ]:         922 :                     RPCResult{"if verbose is not set or set to false",
     374   [ +  -  +  - ]:        1844 :                         RPCResult::Type::STR_HEX, "txid", "The transaction id for the send. Only 1 transaction is created regardless of\n"
     375                 :             :                 "the number of addresses."
     376                 :         922 :                     },
     377         [ +  - ]:        1844 :                     RPCResult{"if verbose is set to true",
     378   [ +  -  +  - ]:        1844 :                         RPCResult::Type::OBJ, "", "",
     379                 :             :                         {
     380   [ +  -  +  - ]:        1844 :                             {RPCResult::Type::STR_HEX, "txid", "The transaction id for the send. Only 1 transaction is created regardless of\n"
     381                 :             :                 "the number of addresses."},
     382   [ +  -  +  - ]:        1844 :                             {RPCResult::Type::STR, "fee_reason", "The transaction fee reason."}
     383                 :             :                         },
     384   [ +  -  +  +  :        5532 :                     },
                   -  - ]
     385                 :             :                 },
     386                 :         922 :                 RPCExamples{
     387                 :             :             "\nSend two amounts to two different addresses:\n"
     388   [ +  -  +  -  :        3688 :             + HelpExampleCli("sendmany", "\"\" \"{\\\"" + EXAMPLE_ADDRESS[0] + "\\\":0.01,\\\"" + EXAMPLE_ADDRESS[1] + "\\\":0.02}\"") +
          +  -  +  -  +  
                      - ]
     389                 :         922 :             "\nSend two amounts to two different addresses setting the confirmation and comment:\n"
     390   [ +  -  +  -  :        5532 :             + HelpExampleCli("sendmany", "\"\" \"{\\\"" + EXAMPLE_ADDRESS[0] + "\\\":0.01,\\\"" + EXAMPLE_ADDRESS[1] + "\\\":0.02}\" 6 \"testing\"") +
          +  -  +  -  +  
                      - ]
     391                 :         922 :             "\nSend two amounts to two different addresses, subtract fee from amount:\n"
     392   [ +  -  +  -  :        7376 :             + HelpExampleCli("sendmany", "\"\" \"{\\\"" + EXAMPLE_ADDRESS[0] + "\\\":0.01,\\\"" + EXAMPLE_ADDRESS[1] + "\\\":0.02}\" 1 \"\" \"[\\\"" + EXAMPLE_ADDRESS[0] + "\\\",\\\"" + EXAMPLE_ADDRESS[1] + "\\\"]\"") +
          +  -  +  -  +  
             -  +  -  +  
                      - ]
     393                 :         922 :             "\nAs a JSON-RPC call\n"
     394   [ +  -  +  -  :        4610 :             + HelpExampleRpc("sendmany", "\"\", {\"" + EXAMPLE_ADDRESS[0] + "\":0.01,\"" + EXAMPLE_ADDRESS[1] + "\":0.02}, 6, \"testing\"")
             +  -  +  - ]
     395         [ +  - ]:         922 :                 },
     396                 :         922 :         [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
     397                 :             : {
     398                 :          83 :     std::shared_ptr<CWallet> const pwallet = GetWalletForJSONRPCRequest(request);
     399         [ -  + ]:          83 :     if (!pwallet) return UniValue::VNULL;
     400                 :             : 
     401                 :             :     // Make sure the results are valid at least up to the most recent block
     402                 :             :     // the user could have gotten from another RPC command prior to now
     403         [ +  - ]:          83 :     pwallet->BlockUntilSyncedToCurrentChain();
     404                 :             : 
     405         [ +  - ]:          83 :     LOCK(pwallet->cs_wallet);
     406                 :             : 
     407   [ +  -  +  +  :          83 :     if (!request.params[0].isNull() && !request.params[0].get_str().empty()) {
          +  -  +  -  -  
                      + ]
     408   [ #  #  #  # ]:           0 :         throw JSONRPCError(RPC_INVALID_PARAMETER, "Dummy value must be set to \"\"");
     409                 :             :     }
     410   [ +  -  +  -  :          83 :     UniValue sendTo = request.params[1].get_obj();
                   +  - ]
     411                 :             : 
     412         [ +  - ]:          83 :     mapValue_t mapValue;
     413   [ +  -  +  +  :          83 :     if (!request.params[3].isNull() && !request.params[3].get_str().empty())
          +  -  +  -  -  
                      + ]
     414   [ #  #  #  #  :           0 :         mapValue["comment"] = request.params[3].get_str();
          #  #  #  #  #  
                      # ]
     415                 :             : 
     416         [ +  - ]:          83 :     CCoinControl coin_control;
     417   [ +  -  -  + ]:          83 :     if (!request.params[5].isNull()) {
     418   [ #  #  #  # ]:           0 :         coin_control.m_signal_bip125_rbf = request.params[5].get_bool();
     419                 :             :     }
     420                 :             : 
     421   [ +  -  +  -  :          83 :     SetFeeEstimateMode(*pwallet, coin_control, /*conf_target=*/request.params[6], /*estimate_mode=*/request.params[7], /*fee_rate=*/request.params[8], /*override_min_fee=*/false);
             +  -  +  + ]
     422                 :             : 
     423                 :          60 :     std::vector<CRecipient> recipients = CreateRecipients(
     424         [ +  - ]:         108 :             ParseOutputs(sendTo),
     425   [ +  -  +  -  :          60 :             InterpretSubtractFeeFromOutputInstructions(request.params[4], sendTo.getKeys())
                   +  + ]
     426         [ +  - ]:          54 :     );
     427   [ +  -  +  +  :          54 :     const bool verbose{request.params[9].isNull() ? false : request.params[9].get_bool()};
             +  -  +  - ]
     428                 :             : 
     429         [ +  + ]:          67 :     return SendMoney(*pwallet, coin_control, recipients, std::move(mapValue), verbose);
     430         [ +  - ]:         262 : },
     431   [ +  -  +  -  :       23050 :     };
          +  -  +  -  +  
          -  +  -  +  +  
          +  +  +  +  +  
          +  -  -  -  -  
             -  -  -  - ]
     432   [ +  -  +  -  :       28582 : }
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  -  -  
             -  -  -  - ]
     433                 :             : 
     434                 :             : // Only includes key documentation where the key is snake_case in all RPC methods. MixedCase keys can be added later.
     435                 :        4225 : static std::vector<RPCArg> FundTxDoc(bool solving_data = true)
     436                 :             : {
     437                 :        4225 :     std::vector<RPCArg> args = {
     438   [ +  -  +  -  :       12675 :         {"conf_target", RPCArg::Type::NUM, RPCArg::DefaultHint{"wallet -txconfirmtarget"}, "Confirmation target in blocks", RPCArgOptions{.also_positional = true}},
             +  -  +  - ]
     439   [ +  -  +  - ]:        8450 :         {"estimate_mode", RPCArg::Type::STR, RPCArg::Default{"unset"}, "The fee estimate mode, must be one of (case insensitive):\n"
     440   [ +  -  +  -  :        8450 :           + FeeModesDetail(std::string("economical mode is used if the transaction is replaceable;\notherwise, conservative mode is used")), RPCArgOptions{.also_positional = true}},
             +  -  +  - ]
     441                 :             :         {
     442   [ +  -  +  -  :       12675 :             "replaceable", RPCArg::Type::BOOL, RPCArg::DefaultHint{"wallet default"}, "Marks this transaction as BIP125-replaceable.\n"
                   +  - ]
     443                 :             :             "Allows this transaction to be replaced by a transaction with higher fees"
     444                 :             :         },
     445   [ +  -  +  +  :       21125 :     };
                   -  - ]
     446         [ +  - ]:        4225 :     if (solving_data) {
     447   [ +  -  +  -  :       71825 :         args.push_back({"solving_data", RPCArg::Type::OBJ, RPCArg::Optional::OMITTED, "Keys and scripts needed for producing a final transaction with a dummy signature.\n"
          +  -  +  -  +  
          -  +  -  +  -  
          +  +  +  +  +  
          +  +  +  -  -  
          -  -  -  -  -  
                      - ]
     448                 :             :         "Used for fee estimation during coin selection.",
     449                 :             :             {
     450                 :             :                 {
     451   [ +  -  +  - ]:        8450 :                     "pubkeys", RPCArg::Type::ARR, RPCArg::Default{UniValue::VARR}, "Public keys involved in this transaction.",
     452                 :             :                     {
     453   [ +  -  +  - ]:        8450 :                         {"pubkey", RPCArg::Type::STR_HEX, RPCArg::Optional::OMITTED, "A public key"},
     454                 :             :                     }
     455                 :             :                 },
     456                 :             :                 {
     457   [ +  -  +  - ]:        8450 :                     "scripts", RPCArg::Type::ARR, RPCArg::Default{UniValue::VARR}, "Scripts involved in this transaction.",
     458                 :             :                     {
     459   [ +  -  +  - ]:        8450 :                         {"script", RPCArg::Type::STR_HEX, RPCArg::Optional::OMITTED, "A script"},
     460                 :             :                     }
     461                 :             :                 },
     462                 :             :                 {
     463   [ +  -  +  - ]:        8450 :                     "descriptors", RPCArg::Type::ARR, RPCArg::Default{UniValue::VARR}, "Descriptors that provide solving data for this transaction.",
     464                 :             :                     {
     465   [ +  -  +  - ]:        8450 :                         {"descriptor", RPCArg::Type::STR, RPCArg::Optional::OMITTED, "A descriptor"},
     466                 :             :                     }
     467                 :             :                 },
     468                 :             :             }
     469                 :             :         });
     470                 :             :     }
     471                 :        4225 :     return args;
     472   [ +  -  +  -  :       71825 : }
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
             -  -  -  -  
                      - ]
     473                 :             : 
     474                 :         687 : CreatedTransactionResult FundTransaction(CWallet& wallet, const CMutableTransaction& tx, const std::vector<CRecipient>& recipients, const UniValue& options, CCoinControl& coinControl, bool override_min_fee)
     475                 :             : {
     476                 :             :     // We want to make sure tx.vout is not used now that we are passing outputs as a vector of recipients.
     477                 :             :     // This sets us up to remove tx completely in a future PR in favor of passing the inputs directly.
     478                 :         687 :     CHECK_NONFATAL(tx.vout.empty());
     479                 :             :     // Make sure the results are valid at least up to the most recent block
     480                 :             :     // the user could have gotten from another RPC command prior to now
     481                 :         687 :     wallet.BlockUntilSyncedToCurrentChain();
     482                 :             : 
     483                 :         687 :     std::optional<unsigned int> change_position;
     484                 :         687 :     bool lockUnspents = false;
     485         [ +  + ]:         687 :     if (!options.isNull()) {
     486         [ +  + ]:         652 :         if (options.type() == UniValue::VBOOL) {
     487                 :             :             // backward compatibility bool only fallback, does nothing
     488                 :             :         } else {
     489   [ +  +  +  +  :       18902 :             RPCTypeCheckObj(options,
                   +  + ]
     490                 :             :                 {
     491         [ +  - ]:         651 :                     {"add_inputs", UniValueType(UniValue::VBOOL)},
     492         [ +  - ]:         651 :                     {"include_unsafe", UniValueType(UniValue::VBOOL)},
     493         [ +  - ]:         651 :                     {"add_to_wallet", UniValueType(UniValue::VBOOL)},
     494         [ +  - ]:         651 :                     {"changeAddress", UniValueType(UniValue::VSTR)},
     495         [ +  - ]:         651 :                     {"change_address", UniValueType(UniValue::VSTR)},
     496         [ +  - ]:         651 :                     {"changePosition", UniValueType(UniValue::VNUM)},
     497         [ +  - ]:         651 :                     {"change_position", UniValueType(UniValue::VNUM)},
     498         [ +  - ]:         651 :                     {"change_type", UniValueType(UniValue::VSTR)},
     499         [ +  - ]:         651 :                     {"includeWatching", UniValueType(UniValue::VBOOL)},
     500         [ +  - ]:         651 :                     {"include_watching", UniValueType(UniValue::VBOOL)},
     501         [ +  - ]:         651 :                     {"inputs", UniValueType(UniValue::VARR)},
     502         [ +  - ]:         651 :                     {"lockUnspents", UniValueType(UniValue::VBOOL)},
     503         [ +  - ]:         651 :                     {"lock_unspents", UniValueType(UniValue::VBOOL)},
     504         [ +  - ]:         651 :                     {"locktime", UniValueType(UniValue::VNUM)},
     505         [ +  - ]:         651 :                     {"fee_rate", UniValueType()}, // will be checked by AmountFromValue() in SetFeeEstimateMode()
     506         [ +  - ]:         651 :                     {"feeRate", UniValueType()}, // will be checked by AmountFromValue() below
     507         [ +  - ]:         651 :                     {"psbt", UniValueType(UniValue::VBOOL)},
     508         [ +  - ]:         651 :                     {"solving_data", UniValueType(UniValue::VOBJ)},
     509         [ +  - ]:         651 :                     {"subtractFeeFromOutputs", UniValueType(UniValue::VARR)},
     510         [ +  - ]:         651 :                     {"subtract_fee_from_outputs", UniValueType(UniValue::VARR)},
     511         [ +  - ]:         651 :                     {"replaceable", UniValueType(UniValue::VBOOL)},
     512         [ +  - ]:         651 :                     {"conf_target", UniValueType(UniValue::VNUM)},
     513         [ +  - ]:         651 :                     {"estimate_mode", UniValueType(UniValue::VSTR)},
     514         [ +  - ]:         651 :                     {"minconf", UniValueType(UniValue::VNUM)},
     515         [ +  - ]:         651 :                     {"maxconf", UniValueType(UniValue::VNUM)},
     516         [ +  - ]:         651 :                     {"input_weights", UniValueType(UniValue::VARR)},
     517         [ +  - ]:         651 :                     {"max_tx_weight", UniValueType(UniValue::VNUM)},
     518                 :             :                 },
     519                 :             :                 true, true);
     520                 :             : 
     521         [ +  + ]:        1256 :             if (options.exists("add_inputs")) {
     522   [ +  -  +  - ]:         174 :                 coinControl.m_allow_other_inputs = options["add_inputs"].get_bool();
     523                 :             :             }
     524                 :             : 
     525   [ +  -  +  -  :        1875 :             if (options.exists("changeAddress") || options.exists("change_address")) {
          +  +  +  -  +  
          -  +  +  +  +  
                   -  - ]
     526   [ +  +  +  -  :          52 :                 const std::string change_address_str = (options.exists("change_address") ? options["change_address"] : options["changeAddress"]).get_str();
          +  -  +  -  +  
          -  +  -  -  +  
          +  +  +  +  -  
                -  -  - ]
     527         [ +  - ]:          13 :                 CTxDestination dest = DecodeDestination(change_address_str);
     528                 :             : 
     529   [ +  -  +  + ]:          13 :                 if (!IsValidDestination(dest)) {
     530   [ +  -  +  - ]:           4 :                     throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Change address must be a valid bitcoin address");
     531                 :             :                 }
     532                 :             : 
     533         [ +  - ]:          22 :                 coinControl.destChange = dest;
     534                 :          15 :             }
     535                 :             : 
     536   [ +  -  +  -  :        1851 :             if (options.exists("changePosition") || options.exists("change_position")) {
          +  +  +  -  +  
          -  +  +  +  +  
                   -  - ]
     537   [ +  +  +  -  :         180 :                 int pos = (options.exists("change_position") ? options["change_position"] : options["changePosition"]).getInt<int>();
          +  -  +  -  +  
          -  +  -  +  +  
          +  +  -  -  -  
                      - ]
     538   [ +  -  -  +  :          60 :                 if (pos < 0 || (unsigned int)pos > recipients.size()) {
                   +  + ]
     539   [ +  -  +  - ]:           2 :                     throw JSONRPCError(RPC_INVALID_PARAMETER, "changePosition out of bounds");
     540                 :             :                 }
     541                 :          59 :                 change_position = (unsigned int)pos;
     542                 :             :             }
     543                 :             : 
     544         [ +  + ]:        1250 :             if (options.exists("change_type")) {
     545   [ +  -  +  -  :         296 :                 if (options.exists("changeAddress") || options.exists("change_address")) {
          +  +  +  -  +  
          -  +  -  +  +  
                   -  - ]
     546   [ +  -  +  - ]:           2 :                     throw JSONRPCError(RPC_INVALID_PARAMETER, "Cannot specify both change address and address type options");
     547                 :             :                 }
     548   [ +  -  +  +  :         196 :                 if (std::optional<OutputType> parsed = ParseOutputType(options["change_type"].get_str())) {
          -  +  +  -  +  
                      + ]
     549         [ -  + ]:         192 :                     coinControl.m_change_type.emplace(parsed.value());
     550                 :             :                 } else {
     551   [ +  -  +  -  :           2 :                     throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, strprintf("Unknown change type '%s'", options["change_type"].get_str()));
          +  -  +  -  +  
                      - ]
     552                 :             :                 }
     553                 :             :             }
     554                 :             : 
     555   [ +  -  +  -  :        1863 :             if (options.exists("lockUnspents") || options.exists("lock_unspents")) {
          +  +  +  -  +  
          -  +  +  +  +  
                   -  - ]
     556   [ +  +  +  -  :          16 :                 lockUnspents = (options.exists("lock_unspents") ? options["lock_unspents"] : options["lockUnspents"]).get_bool();
          +  -  +  -  +  
          -  +  -  +  +  
          +  +  -  -  -  
                      - ]
     557                 :             :             }
     558                 :             : 
     559         [ +  + ]:        1244 :             if (options.exists("include_unsafe")) {
     560   [ +  -  +  - ]:          46 :                 coinControl.m_include_unsafe_inputs = options["include_unsafe"].get_bool();
     561                 :             :             }
     562                 :             : 
     563         [ +  + ]:        1244 :             if (options.exists("feeRate")) {
     564         [ +  + ]:         116 :                 if (options.exists("fee_rate")) {
     565   [ +  -  +  -  :           8 :                     throw JSONRPCError(RPC_INVALID_PARAMETER, "Cannot specify both fee_rate (" + CURRENCY_ATOM + "/vB) and feeRate (" + CURRENCY_UNIT + "/kvB)");
                   +  - ]
     566                 :             :                 }
     567         [ +  + ]:         112 :                 if (options.exists("conf_target")) {
     568   [ +  -  +  - ]:           4 :                     throw JSONRPCError(RPC_INVALID_PARAMETER, "Cannot specify both conf_target and feeRate. Please provide either a confirmation target in blocks for automatic fee estimation, or an explicit fee rate.");
     569                 :             :                 }
     570         [ +  + ]:         108 :                 if (options.exists("estimate_mode")) {
     571   [ +  -  +  - ]:           4 :                     throw JSONRPCError(RPC_INVALID_PARAMETER, "Cannot specify both estimate_mode and feeRate");
     572                 :             :                 }
     573   [ +  -  +  +  :         104 :                 coinControl.m_feerate = CFeeRate(AmountFromValue(options["feeRate"]));
                   -  + ]
     574                 :          36 :                 coinControl.fOverrideFeeRate = true;
     575                 :             :             }
     576                 :             : 
     577         [ +  + ]:        1200 :             if (options.exists("replaceable")) {
     578   [ +  -  +  - ]:           5 :                 coinControl.m_signal_bip125_rbf = options["replaceable"].get_bool();
     579                 :             :             }
     580                 :             : 
     581         [ +  + ]:        1200 :             if (options.exists("minconf")) {
     582   [ +  -  +  - ]:           9 :                 coinControl.m_min_depth = options["minconf"].getInt<int>();
     583                 :             : 
     584         [ +  + ]:           9 :                 if (coinControl.m_min_depth < 0) {
     585   [ +  -  +  - ]:           2 :                     throw JSONRPCError(RPC_INVALID_PARAMETER, "Negative minconf");
     586                 :             :                 }
     587                 :             :             }
     588                 :             : 
     589         [ +  + ]:        1198 :             if (options.exists("maxconf")) {
     590   [ +  -  +  - ]:           4 :                 coinControl.m_max_depth = options["maxconf"].getInt<int>();
     591                 :             : 
     592         [ -  + ]:           4 :                 if (coinControl.m_max_depth < coinControl.m_min_depth) {
     593   [ #  #  #  # ]:           0 :                     throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("maxconf can't be lower than minconf: %d < %d", coinControl.m_max_depth, coinControl.m_min_depth));
     594                 :             :                 }
     595                 :             :             }
     596   [ +  -  +  -  :        1392 :             SetFeeEstimateMode(wallet, coinControl, options["conf_target"], options["estimate_mode"], options["fee_rate"], override_min_fee);
          +  -  +  -  +  
                -  +  + ]
     597                 :             :         }
     598                 :             :     }
     599                 :             : 
     600         [ +  + ]:        1076 :     if (options.exists("solving_data")) {
     601   [ +  -  +  -  :          14 :         const UniValue solving_data = options["solving_data"].get_obj();
                   +  - ]
     602   [ +  -  +  + ]:          28 :         if (solving_data.exists("pubkeys")) {
     603   [ +  -  +  -  :           8 :             for (const UniValue& pk_univ : solving_data["pubkeys"].get_array().getValues()) {
          +  -  +  -  +  
                      + ]
     604   [ +  -  +  + ]:           5 :                 const CPubKey pubkey = HexToPubKey(pk_univ.get_str());
     605   [ +  -  +  - ]:           3 :                 coinControl.m_external_provider.pubkeys.emplace(pubkey.GetID(), pubkey);
     606                 :             :                 // Add witness script for pubkeys
     607   [ +  -  +  - ]:           3 :                 const CScript wit_script = GetScriptForDestination(WitnessV0KeyHash(pubkey));
     608   [ +  -  +  - ]:           3 :                 coinControl.m_external_provider.scripts.emplace(CScriptID(wit_script), wit_script);
     609                 :           3 :             }
     610                 :             :         }
     611                 :             : 
     612   [ +  -  +  + ]:          24 :         if (solving_data.exists("scripts")) {
     613   [ +  -  +  -  :           9 :             for (const UniValue& script_univ : solving_data["scripts"].get_array().getValues()) {
          +  -  +  -  +  
                      + ]
     614         [ +  - ]:           6 :                 const std::string& script_str = script_univ.get_str();
     615   [ -  +  +  -  :           6 :                 if (!IsHex(script_str)) {
                   +  + ]
     616   [ +  -  +  - ]:           2 :                     throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, strprintf("'%s' is not hex", script_str));
     617                 :             :                 }
     618   [ -  +  +  - ]:           5 :                 std::vector<unsigned char> script_data(ParseHex(script_str));
     619                 :           5 :                 const CScript script(script_data.begin(), script_data.end());
     620   [ +  -  +  - ]:           5 :                 coinControl.m_external_provider.scripts.emplace(CScriptID(script), script);
     621                 :           5 :             }
     622                 :             :         }
     623                 :             : 
     624   [ +  -  +  + ]:          22 :         if (solving_data.exists("descriptors")) {
     625   [ +  -  +  -  :          15 :             for (const UniValue& desc_univ : solving_data["descriptors"].get_array().getValues()) {
          +  -  +  -  +  
                      + ]
     626         [ +  - ]:           8 :                 const std::string& desc_str  = desc_univ.get_str();
     627                 :           8 :                 FlatSigningProvider desc_out;
     628         [ -  + ]:           8 :                 std::string error;
     629                 :           8 :                 std::vector<CScript> scripts_temp;
     630   [ -  +  +  - ]:           8 :                 auto descs = Parse(desc_str, desc_out, error, true);
     631         [ +  + ]:           8 :                 if (descs.empty()) {
     632   [ +  -  +  - ]:           2 :                     throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Unable to parse descriptor '%s': %s", desc_str, error));
     633                 :             :                 }
     634         [ +  + ]:          14 :                 for (auto& desc : descs) {
     635         [ +  - ]:           7 :                     desc->Expand(0, desc_out, scripts_temp, desc_out);
     636                 :             :                 }
     637         [ +  - ]:           7 :                 coinControl.m_external_provider.Merge(std::move(desc_out));
     638                 :          10 :             }
     639                 :             :         }
     640                 :          14 :     }
     641                 :             : 
     642         [ +  + ]:        1068 :     if (options.exists("input_weights")) {
     643   [ +  -  +  -  :        2625 :         for (const UniValue& input : options["input_weights"].get_array().getValues()) {
             +  -  +  + ]
     644                 :        2491 :             Txid txid = Txid::FromUint256(ParseHashO(input, "txid"));
     645                 :             : 
     646                 :        2491 :             const UniValue& vout_v = input.find_value("vout");
     647         [ +  + ]:        2491 :             if (!vout_v.isNum()) {
     648   [ +  -  +  - ]:           2 :                 throw JSONRPCError(RPC_INVALID_PARAMETER, "Invalid parameter, missing vout key");
     649                 :             :             }
     650                 :        2490 :             int vout = vout_v.getInt<int>();
     651         [ +  + ]:        2490 :             if (vout < 0) {
     652   [ +  -  +  - ]:           2 :                 throw JSONRPCError(RPC_INVALID_PARAMETER, "Invalid parameter, vout cannot be negative");
     653                 :             :             }
     654                 :             : 
     655                 :        2489 :             const UniValue& weight_v = input.find_value("weight");
     656         [ +  + ]:        2489 :             if (!weight_v.isNum()) {
     657   [ +  -  +  - ]:           2 :                 throw JSONRPCError(RPC_INVALID_PARAMETER, "Invalid parameter, missing weight key");
     658                 :             :             }
     659                 :        2488 :             int64_t weight = weight_v.getInt<int64_t>();
     660                 :        2488 :             const int64_t min_input_weight = GetTransactionInputWeight(CTxIn());
     661                 :        2488 :             CHECK_NONFATAL(min_input_weight == 165);
     662         [ +  + ]:        2488 :             if (weight < min_input_weight) {
     663   [ +  -  +  - ]:           4 :                 throw JSONRPCError(RPC_INVALID_PARAMETER, "Invalid parameter, weight cannot be less than 165 (41 bytes (size of outpoint + sequence + empty scriptSig) * 4 (witness scaling factor)) + 1 (empty witness)");
     664                 :             :             }
     665         [ +  + ]:        2486 :             if (weight > MAX_STANDARD_TX_WEIGHT) {
     666   [ +  -  +  - ]:           2 :                 throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Invalid parameter, weight cannot be greater than the maximum standard tx weight of %d", MAX_STANDARD_TX_WEIGHT));
     667                 :             :             }
     668                 :             : 
     669                 :        2485 :             coinControl.SetInputWeight(COutPoint(txid, vout), weight);
     670                 :             :         }
     671                 :             :     }
     672                 :             : 
     673         [ +  + ]:        1056 :     if (options.exists("max_tx_weight")) {
     674   [ +  -  +  - ]:          10 :         coinControl.m_max_tx_weight = options["max_tx_weight"].getInt<int>();
     675                 :             :     }
     676                 :             : 
     677         [ +  + ]:         528 :     if (tx.version == TRUC_VERSION) {
     678   [ +  +  -  + ]:          36 :         if (!coinControl.m_max_tx_weight.has_value() || coinControl.m_max_tx_weight.value() > TRUC_MAX_WEIGHT) {
     679         [ -  + ]:          34 :             coinControl.m_max_tx_weight = TRUC_MAX_WEIGHT;
     680                 :             :         }
     681                 :             :     }
     682                 :             : 
     683         [ -  + ]:         528 :     if (recipients.empty())
     684   [ #  #  #  # ]:           0 :         throw JSONRPCError(RPC_INVALID_PARAMETER, "TX must have at least one output");
     685                 :             : 
     686         [ +  - ]:         528 :     auto txr = FundTransaction(wallet, tx, recipients, change_position, lockUnspents, coinControl);
     687         [ +  + ]:         528 :     if (!txr) {
     688   [ +  -  +  - ]:         182 :         throw JSONRPCError(RPC_WALLET_ERROR, ErrorString(txr).original);
     689                 :             :     }
     690                 :         437 :     return *txr;
     691   [ +  -  +  -  :        1111 : }
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
             +  -  -  + ]
     692                 :             : 
     693                 :         445 : static void SetOptionsInputWeights(const UniValue& inputs, UniValue& options)
     694                 :             : {
     695         [ +  + ]:         890 :     if (options.exists("input_weights")) {
     696   [ +  -  +  - ]:           4 :         throw JSONRPCError(RPC_INVALID_PARAMETER, "Input weights should be specified in inputs rather than in options.");
     697                 :             :     }
     698   [ -  +  +  + ]:         443 :     if (inputs.size() == 0) {
     699                 :             :         return;
     700                 :             :     }
     701                 :         170 :     UniValue weights(UniValue::VARR);
     702   [ +  -  +  + ]:        2955 :     for (const UniValue& input : inputs.getValues()) {
     703   [ +  -  +  + ]:        5570 :         if (input.exists("weight")) {
     704   [ +  -  +  - ]:           6 :             weights.push_back(input);
     705                 :             :         }
     706                 :             :     }
     707   [ +  -  +  - ]:         340 :     options.pushKV("input_weights", std::move(weights));
     708                 :         170 : }
     709                 :             : 
     710                 :        1083 : RPCMethod fundrawtransaction()
     711                 :             : {
     712                 :        1083 :     return RPCMethod{
     713                 :        1083 :         "fundrawtransaction",
     714         [ +  - ]:        2166 :         "If the transaction has no inputs, they will be automatically selected to meet its out value.\n"
     715                 :             :                 "It will add at most one change output to the outputs.\n"
     716                 :             :                 "No existing outputs will be modified unless \"subtractFeeFromOutputs\" is specified.\n"
     717                 :             :                 "Note that inputs which were signed may need to be resigned after completion since in/outputs have been added.\n"
     718                 :             :                 "The inputs added will not be signed, use signrawtransactionwithkey\n"
     719                 :             :                 "or signrawtransactionwithwallet for that.\n"
     720                 :             :                 "All existing inputs must either have their previous output transaction be in the wallet\n"
     721                 :             :                 "or be in the UTXO set. Solving data must be provided for non-wallet inputs.\n"
     722                 :             :                 "Note that all inputs selected must be of standard form and P2SH scripts must be\n"
     723                 :             :                 "in the wallet using importdescriptors (to calculate fees).\n"
     724                 :             :                 "You can see whether this is the case by checking the \"solvable\" field in the listunspent output.\n"
     725                 :             :                 "Note that if specifying an exact fee rate, the resulting transaction may have a higher fee rate\n"
     726                 :             :                 "if the transaction has unconfirmed inputs. This is because the wallet will attempt to make the\n"
     727                 :             :                 "entire package have the given fee rate, not the resulting transaction.\n",
     728                 :             :                 {
     729   [ +  -  +  - ]:        2166 :                     {"hexstring", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "The hex string of the raw transaction"},
     730   [ +  -  +  - ]:        2166 :                     {"options", RPCArg::Type::OBJ_NAMED_PARAMS, RPCArg::Optional::OMITTED, "",
     731   [ +  -  +  -  :       29241 :                         Cat<std::vector<RPCArg>>(
          +  -  +  -  +  
          -  +  +  +  +  
          +  +  +  +  -  
          -  -  -  -  -  
                   -  - ]
     732                 :             :                         {
     733   [ +  -  +  -  :        3249 :                             {"add_inputs", RPCArg::Type::BOOL, RPCArg::Default{true}, "For a transaction with existing inputs, automatically include more if they are not enough."},
                   +  - ]
     734   [ +  -  +  -  :        3249 :                             {"include_unsafe", RPCArg::Type::BOOL, RPCArg::Default{false}, "Include inputs that are not safe to spend (unconfirmed transactions from outside keys and unconfirmed replacement transactions).\n"
                   +  - ]
     735                 :             :                                                           "Warning: the resulting transaction may become invalid if one of the unsafe inputs disappears.\n"
     736                 :             :                                                           "If that happens, you will need to fund the transaction with different inputs and republish it."},
     737   [ +  -  +  -  :        3249 :                             {"minconf", RPCArg::Type::NUM, RPCArg::Default{0}, "If add_inputs is specified, require inputs with at least this many confirmations."},
                   +  - ]
     738   [ +  -  +  - ]:        2166 :                             {"maxconf", RPCArg::Type::NUM, RPCArg::Optional::OMITTED, "If add_inputs is specified, require inputs with at most this many confirmations."},
     739   [ +  -  +  -  :        3249 :                             {"changeAddress", RPCArg::Type::STR, RPCArg::DefaultHint{"automatic"}, "The bitcoin address to receive the change"},
                   +  - ]
     740   [ +  -  +  -  :        3249 :                             {"changePosition", RPCArg::Type::NUM, RPCArg::DefaultHint{"random"}, "The index of the change output"},
                   +  - ]
     741   [ +  -  +  -  :        3249 :                             {"change_type", RPCArg::Type::STR, RPCArg::DefaultHint{"set by -changetype"}, "The output type to use. Only valid if changeAddress is not specified. Options are " + FormatAllOutputTypes() + "."},
             +  -  +  - ]
     742   [ +  -  +  -  :        3249 :                             {"includeWatching", RPCArg::Type::BOOL, RPCArg::Default{false}, "(DEPRECATED) No longer used"},
                   +  - ]
     743   [ +  -  +  -  :        3249 :                             {"lockUnspents", RPCArg::Type::BOOL, RPCArg::Default{false}, "Lock selected unspent outputs"},
                   +  - ]
     744   [ +  -  +  -  :        3249 :                             {"fee_rate", RPCArg::Type::AMOUNT, RPCArg::DefaultHint{"not set, fall back to wallet fee estimation"}, "Specify a fee rate in " + CURRENCY_ATOM + "/vB."},
                   +  - ]
     745   [ +  -  +  -  :        3249 :                             {"feeRate", RPCArg::Type::AMOUNT, RPCArg::DefaultHint{"not set, fall back to wallet fee estimation"}, "Specify a fee rate in " + CURRENCY_UNIT + "/kvB."},
                   +  - ]
     746   [ +  -  +  - ]:        2166 :                             {"subtractFeeFromOutputs", RPCArg::Type::ARR, RPCArg::Default{UniValue::VARR}, "The integers.\n"
     747                 :             :                                                           "The fee will be equally deducted from the amount of each specified output.\n"
     748                 :             :                                                           "Those recipients will receive less bitcoins than you enter in their corresponding amount field.\n"
     749                 :             :                                                           "If no outputs are specified here, the sender pays the fee.",
     750                 :             :                                 {
     751   [ +  -  +  - ]:        2166 :                                     {"vout_index", RPCArg::Type::NUM, RPCArg::Optional::OMITTED, "The zero-based output index, before a change output is added."},
     752                 :             :                                 },
     753                 :             :                             },
     754   [ +  -  +  - ]:        2166 :                             {"input_weights", RPCArg::Type::ARR, RPCArg::Optional::OMITTED, "Inputs and their corresponding weights",
     755                 :             :                                 {
     756   [ +  -  +  - ]:        2166 :                                     {"", RPCArg::Type::OBJ, RPCArg::Optional::OMITTED, "",
     757                 :             :                                         {
     758   [ +  -  +  - ]:        2166 :                                             {"txid", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "The transaction id"},
     759   [ +  -  +  - ]:        2166 :                                             {"vout", RPCArg::Type::NUM, RPCArg::Optional::NO, "The output index"},
     760   [ +  -  +  - ]:        2166 :                                             {"weight", RPCArg::Type::NUM, RPCArg::Optional::NO, "The maximum weight for this input, "
     761                 :             :                                                 "including the weight of the outpoint and sequence number. "
     762                 :             :                                                 "Note that serialized signature sizes are not guaranteed to be consistent, "
     763                 :             :                                                 "so the maximum DER signatures size of 73 bytes should be used when considering ECDSA signatures."
     764                 :             :                                                 "Remember to convert serialized sizes to weight units when necessary."},
     765                 :             :                                         },
     766                 :             :                                     },
     767                 :             :                                 },
     768                 :             :                              },
     769   [ +  -  +  -  :        3249 :                             {"max_tx_weight", RPCArg::Type::NUM, RPCArg::Default{MAX_STANDARD_TX_WEIGHT}, "The maximum acceptable transaction weight.\n"
                   +  - ]
     770                 :             :                                                           "Transaction building will fail if this can not be satisfied."},
     771                 :             :                         },
     772         [ +  - ]:        2166 :                         FundTxDoc()),
     773         [ +  - ]:        2166 :                         RPCArgOptions{
     774                 :             :                             .skip_type_check = true,
     775                 :             :                             .oneline_description = "options",
     776                 :             :                         }},
     777   [ +  -  +  -  :        3249 :                     {"iswitness", RPCArg::Type::BOOL, RPCArg::DefaultHint{"depends on heuristic tests"}, "Whether the transaction hex is a serialized witness transaction.\n"
                   +  - ]
     778                 :             :                         "If iswitness is not present, heuristic tests will be used in decoding.\n"
     779                 :             :                         "If true, only witness deserialization will be tried.\n"
     780                 :             :                         "If false, only non-witness deserialization will be tried.\n"
     781                 :             :                         "This boolean should reflect whether the transaction has inputs\n"
     782                 :             :                         "(e.g. fully valid, or on-chain transactions), if known by the caller."
     783                 :             :                     },
     784                 :             :                 },
     785         [ +  - ]:        2166 :                 RPCResult{
     786   [ +  -  +  - ]:        2166 :                     RPCResult::Type::OBJ, "", "",
     787                 :             :                     {
     788   [ +  -  +  - ]:        2166 :                         {RPCResult::Type::STR_HEX, "hex", "The resulting raw transaction (hex-encoded string)"},
     789   [ +  -  +  - ]:        2166 :                         {RPCResult::Type::STR_AMOUNT, "fee", "Fee in " + CURRENCY_UNIT + " the resulting transaction pays"},
     790   [ +  -  +  - ]:        2166 :                         {RPCResult::Type::NUM, "changepos", "The position of the added change output, or -1"},
     791                 :             :                     }
     792   [ +  -  +  -  :        8664 :                                 },
             +  +  -  - ]
     793                 :        1083 :                                 RPCExamples{
     794                 :             :                             "\nCreate a transaction with no inputs\n"
     795   [ +  -  +  -  :        2166 :                             + HelpExampleCli("createrawtransaction", "\"[]\" \"{\\\"myaddress\\\":0.01}\"") +
             +  -  +  - ]
     796                 :        1083 :                             "\nAdd sufficient unsigned inputs to meet the output value\n"
     797   [ +  -  +  -  :        4332 :                             + HelpExampleCli("fundrawtransaction", "\"rawtransactionhex\"") +
             +  -  +  - ]
     798                 :        1083 :                             "\nSign the transaction\n"
     799   [ +  -  +  -  :        4332 :                             + HelpExampleCli("signrawtransactionwithwallet", "\"fundedtransactionhex\"") +
             +  -  +  - ]
     800                 :        1083 :                             "\nSend the transaction\n"
     801   [ +  -  +  -  :        4332 :                             + HelpExampleCli("sendrawtransaction", "\"signedtransactionhex\"")
             +  -  +  - ]
     802         [ +  - ]:        1083 :                                 },
     803                 :        1083 :         [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
     804                 :             : {
     805                 :         244 :     std::shared_ptr<CWallet> const pwallet = GetWalletForJSONRPCRequest(request);
     806         [ -  + ]:         244 :     if (!pwallet) return UniValue::VNULL;
     807                 :             : 
     808                 :             :     // parse hex string from parameter
     809         [ +  - ]:         244 :     CMutableTransaction tx;
     810   [ +  -  -  +  :         244 :     bool try_witness = request.params[2].isNull() ? true : request.params[2].get_bool();
             -  -  -  - ]
     811   [ +  -  -  +  :         244 :     bool try_no_witness = request.params[2].isNull() ? true : !request.params[2].get_bool();
             -  -  -  - ]
     812   [ +  -  +  -  :         244 :     if (!DecodeHexTx(tx, request.params[0].get_str(), try_no_witness, try_witness)) {
             +  -  -  + ]
     813   [ #  #  #  # ]:           0 :         throw JSONRPCError(RPC_DESERIALIZATION_ERROR, "TX decode failed");
     814                 :             :     }
     815   [ +  -  +  - ]:         244 :     UniValue options = request.params[1];
     816                 :         244 :     std::vector<std::pair<CTxDestination, CAmount>> destinations;
     817         [ +  + ]:       11711 :     for (const auto& tx_out : tx.vout) {
     818                 :       11467 :         CTxDestination dest;
     819         [ +  - ]:       11467 :         ExtractDestination(tx_out.scriptPubKey, dest);
     820         [ +  - ]:       11467 :         destinations.emplace_back(dest, tx_out.nValue);
     821                 :       11467 :     }
     822   [ +  -  +  - ]:         488 :     std::vector<std::string> dummy(destinations.size(), "dummy");
     823                 :         244 :     std::vector<CRecipient> recipients = CreateRecipients(
     824                 :             :             destinations,
     825   [ +  -  +  -  :         488 :             InterpretSubtractFeeFromOutputInstructions(options["subtractFeeFromOutputs"], dummy)
                   +  - ]
     826         [ +  - ]:         244 :     );
     827         [ +  - ]:         244 :     CCoinControl coin_control;
     828                 :             :     // Automatically select (additional) coins. Can be overridden by options.add_inputs.
     829                 :         244 :     coin_control.m_allow_other_inputs = true;
     830                 :             :     // Clear tx.vout since it is not meant to be used now that we are passing outputs directly.
     831                 :             :     // This sets us up for a future PR to completely remove tx from the function signature in favor of passing inputs directly
     832                 :         244 :     tx.vout.clear();
     833         [ +  + ]:         244 :     auto txr = FundTransaction(*pwallet, tx, recipients, options, coin_control, /*override_min_fee=*/true);
     834                 :             : 
     835                 :         145 :     UniValue result(UniValue::VOBJ);
     836   [ +  -  +  -  :         290 :     result.pushKV("hex", EncodeHexTx(*txr.tx));
             +  -  +  - ]
     837   [ +  -  +  -  :         290 :     result.pushKV("fee", ValueFromAmount(txr.fee));
                   +  - ]
     838   [ +  +  +  -  :         290 :     result.pushKV("changepos", txr.change_pos ? (int)*txr.change_pos : -1);
             +  -  +  - ]
     839                 :             : 
     840                 :         145 :     return result;
     841         [ +  - ]:        1174 : },
     842   [ +  -  +  -  :        7581 :     };
             +  +  -  - ]
     843   [ +  -  +  -  :       53067 : }
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  -  -  -  -  
             -  -  -  - ]
     844                 :             : 
     845                 :        1170 : RPCMethod signrawtransactionwithwallet()
     846                 :             : {
     847                 :        1170 :     return RPCMethod{
     848                 :        1170 :         "signrawtransactionwithwallet",
     849                 :             :         "Sign inputs for raw transaction (serialized, hex-encoded).\n"
     850                 :             :                 "The second optional argument (may be null) is an array of previous transaction outputs that\n"
     851                 :        1170 :                 "this transaction depends on but may not yet be in the block chain." +
     852         [ +  - ]:        1170 :         HELP_REQUIRING_PASSPHRASE,
     853                 :             :                 {
     854   [ +  -  +  - ]:        2340 :                     {"hexstring", RPCArg::Type::STR, RPCArg::Optional::NO, "The transaction hex string"},
     855   [ +  -  +  - ]:        2340 :                     {"prevtxs", RPCArg::Type::ARR, RPCArg::Optional::OMITTED, "The previous dependent transaction outputs",
     856                 :             :                         {
     857   [ +  -  +  - ]:        2340 :                             {"", RPCArg::Type::OBJ, RPCArg::Optional::OMITTED, "",
     858                 :             :                                 {
     859   [ +  -  +  - ]:        2340 :                                     {"txid", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "The transaction id"},
     860   [ +  -  +  - ]:        2340 :                                     {"vout", RPCArg::Type::NUM, RPCArg::Optional::NO, "The output number"},
     861   [ +  -  +  - ]:        2340 :                                     {"scriptPubKey", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "The output script"},
     862   [ +  -  +  - ]:        2340 :                                     {"redeemScript", RPCArg::Type::STR_HEX, RPCArg::Optional::OMITTED, "(required for P2SH) redeem script"},
     863   [ +  -  +  - ]:        2340 :                                     {"witnessScript", RPCArg::Type::STR_HEX, RPCArg::Optional::OMITTED, "(required for P2WSH or P2SH-P2WSH) witness script"},
     864   [ +  -  +  - ]:        2340 :                                     {"amount", RPCArg::Type::AMOUNT, RPCArg::Optional::OMITTED, "(required for Segwit inputs) the amount spent"},
     865                 :             :                                 },
     866                 :             :                             },
     867                 :             :                         },
     868                 :             :                     },
     869   [ +  -  +  -  :        3510 :                     {"sighashtype", RPCArg::Type::STR, RPCArg::Default{"DEFAULT for Taproot, ALL otherwise"}, "The signature hash type. Must be one of\n"
                   +  - ]
     870                 :             :             "       \"DEFAULT\"\n"
     871                 :             :             "       \"ALL\"\n"
     872                 :             :             "       \"NONE\"\n"
     873                 :             :             "       \"SINGLE\"\n"
     874                 :             :             "       \"ALL|ANYONECANPAY\"\n"
     875                 :             :             "       \"NONE|ANYONECANPAY\"\n"
     876                 :             :             "       \"SINGLE|ANYONECANPAY\""},
     877                 :             :                 },
     878         [ +  - ]:        2340 :                 RPCResult{
     879   [ +  -  +  - ]:        2340 :                     RPCResult::Type::OBJ, "", "",
     880                 :             :                     {
     881   [ +  -  +  - ]:        2340 :                         {RPCResult::Type::STR_HEX, "hex", "The hex-encoded raw transaction with signature(s)"},
     882   [ +  -  +  - ]:        2340 :                         {RPCResult::Type::BOOL, "complete", "If the transaction has a complete set of signatures"},
     883   [ +  -  +  - ]:        2340 :                         {RPCResult::Type::ARR, "errors", /*optional=*/true, "Script verification errors (if there are any)",
     884                 :             :                         {
     885   [ +  -  +  - ]:        2340 :                             {RPCResult::Type::OBJ, "", "",
     886                 :             :                             {
     887   [ +  -  +  - ]:        2340 :                                 {RPCResult::Type::STR_HEX, "txid", "The hash of the referenced, previous transaction"},
     888   [ +  -  +  - ]:        2340 :                                 {RPCResult::Type::NUM, "vout", "The index of the output to spent and used as input"},
     889   [ +  -  +  - ]:        2340 :                                 {RPCResult::Type::ARR, "witness", "",
     890                 :             :                                 {
     891   [ +  -  +  - ]:        2340 :                                     {RPCResult::Type::STR_HEX, "witness", ""},
     892                 :             :                                 }},
     893   [ +  -  +  - ]:        2340 :                                 {RPCResult::Type::STR_HEX, "scriptSig", "The hex-encoded signature script"},
     894   [ +  -  +  - ]:        2340 :                                 {RPCResult::Type::NUM, "sequence", "Script sequence number"},
     895   [ +  -  +  - ]:        3510 :                                 {RPCResult::Type::STR, "error", "Verification or signing error related to the input"},
     896                 :             :                             }},
     897                 :             :                         }},
     898                 :             :                     }
     899   [ +  -  +  -  :       31590 :                 },
          +  -  +  -  +  
          -  +  +  +  +  
          +  +  +  +  -  
          -  -  -  -  -  
                   -  - ]
     900                 :        1170 :                 RPCExamples{
     901   [ +  -  +  -  :        2340 :                     HelpExampleCli("signrawtransactionwithwallet", "\"myhex\"")
                   +  - ]
     902   [ +  -  +  -  :        4680 :             + HelpExampleRpc("signrawtransactionwithwallet", "\"myhex\"")
             +  -  +  - ]
     903         [ +  - ]:        1170 :                 },
     904                 :        1170 :         [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
     905                 :             : {
     906         [ -  + ]:         331 :     const std::shared_ptr<const CWallet> pwallet = GetWalletForJSONRPCRequest(request);
     907         [ -  + ]:         331 :     if (!pwallet) return UniValue::VNULL;
     908                 :             : 
     909         [ +  - ]:         331 :     CMutableTransaction mtx;
     910   [ +  -  +  -  :         331 :     if (!DecodeHexTx(mtx, request.params[0].get_str())) {
             +  -  -  + ]
     911   [ #  #  #  # ]:           0 :         throw JSONRPCError(RPC_DESERIALIZATION_ERROR, "TX decode failed. Make sure the tx has at least one input.");
     912                 :             :     }
     913                 :             : 
     914                 :             :     // Sign the transaction
     915         [ +  - ]:         331 :     LOCK(pwallet->cs_wallet);
     916         [ +  + ]:         331 :     EnsureWalletIsUnlocked(*pwallet);
     917                 :             : 
     918                 :             :     // Fetch previous transactions (inputs):
     919                 :         330 :     std::map<COutPoint, Coin> coins;
     920         [ +  + ]:        1118 :     for (const CTxIn& txin : mtx.vin) {
     921         [ +  - ]:         788 :         coins[txin.prevout]; // Create empty map entry keyed by prevout.
     922                 :             :     }
     923         [ +  - ]:         330 :     pwallet->chain().findCoins(coins);
     924                 :             : 
     925                 :             :     // Parse the prevtxs array
     926   [ +  -  +  + ]:         330 :     ParsePrevouts(request.params[1], nullptr, coins);
     927                 :             : 
     928   [ +  -  +  + ]:         321 :     std::optional<int> nHashType = ParseSighashString(request.params[2]);
     929         [ +  + ]:         320 :     if (!nHashType) {
     930                 :         317 :         nHashType = SIGHASH_DEFAULT;
     931                 :             :     }
     932                 :             : 
     933                 :             :     // Script verification errors
     934         [ +  - ]:         320 :     std::map<int, bilingual_str> input_errors;
     935                 :             : 
     936         [ +  - ]:         320 :     bool complete = pwallet->SignTransaction(mtx, coins, *nHashType, input_errors);
     937                 :         320 :     UniValue result(UniValue::VOBJ);
     938         [ +  + ]:         320 :     SignTransactionResultToJSON(mtx, complete, coins, input_errors, result);
     939                 :         318 :     return result;
     940         [ +  - ]:        1314 : },
     941   [ +  -  +  -  :       21060 :     };
          +  -  +  -  +  
          +  +  +  +  +  
          -  -  -  -  -  
                      - ]
     942   [ +  -  +  -  :       49140 : }
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  -  -  
          -  -  -  -  -  
                      - ]
     943                 :             : 
     944                 :             : // Definition of allowed formats of specifying transaction outputs in
     945                 :             : // `bumpfee`, `psbtbumpfee`, `send` and `walletcreatefundedpsbt` RPCs.
     946                 :        3968 : static std::vector<RPCArg> OutputsDoc()
     947                 :             : {
     948                 :        3968 :     return
     949                 :             :     {
     950   [ +  -  +  - ]:        7936 :         {"", RPCArg::Type::OBJ_USER_KEYS, RPCArg::Optional::OMITTED, "",
     951                 :             :             {
     952         [ +  - ]:        7936 :                 {"address", RPCArg::Type::AMOUNT, RPCArg::Optional::NO, "A key-value pair. The key (string) is the bitcoin address,\n"
     953         [ +  - ]:        7936 :                          "the value (float or string) is the amount in " + CURRENCY_UNIT + ""},
     954                 :             :             },
     955                 :             :         },
     956   [ +  -  +  - ]:        7936 :         {"", RPCArg::Type::OBJ, RPCArg::Optional::OMITTED, "",
     957                 :             :             {
     958   [ +  -  +  - ]:        7936 :                 {"data", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "A key-value pair. The key must be \"data\", the value is hex-encoded data that becomes a part of an OP_RETURN output"},
     959                 :             :             },
     960                 :             :         },
     961   [ +  -  +  -  :       43648 :     };
          +  -  +  +  +  
          +  +  +  -  -  
             -  -  -  - ]
     962   [ +  -  +  -  :       31744 : }
          +  -  +  -  -  
                      - ]
     963                 :             : 
     964                 :        1839 : static RPCMethod bumpfee_helper(std::string method_name)
     965                 :             : {
     966                 :        1839 :     const bool want_psbt = method_name == "psbtbumpfee";
     967                 :        1839 :     const std::string incremental_fee{CFeeRate(DEFAULT_INCREMENTAL_RELAY_FEE).ToString(FeeRateFormat::SAT_VB)};
     968                 :             : 
     969         [ -  + ]:        1839 :     return RPCMethod{method_name,
     970                 :             :         "Bumps the fee of a transaction T, replacing it with a new transaction B.\n"
     971   [ +  -  +  - ]:        3678 :         + std::string(want_psbt ? "Returns a PSBT instead of creating and signing a new transaction.\n" : "") +
     972                 :             :         "A transaction with the given txid must be in the wallet.\n"
     973                 :             :         "The command will pay the additional fee by reducing change outputs or adding inputs when necessary.\n"
     974                 :             :         "It may add a new change output if one does not already exist.\n"
     975                 :             :         "All inputs in the original transaction will be included in the replacement transaction.\n"
     976                 :             :         "The command will fail if the wallet or mempool contains a transaction that spends one of T's outputs.\n"
     977                 :             :         "By default, the new fee will be calculated automatically using the estimatesmartfee RPC.\n"
     978                 :             :         "The user can specify a confirmation target for estimatesmartfee.\n"
     979         [ +  - ]:        3678 :         "Alternatively, the user can specify a fee rate in " + CURRENCY_ATOM + "/vB for the new transaction.\n"
     980                 :             :         "At a minimum, the new fee rate must be high enough to pay an additional new relay fee (incrementalfee\n"
     981                 :             :         "returned by getnetworkinfo) to enter the node's mempool.\n"
     982   [ +  -  +  - ]:        5517 :         "* WARNING: before version 0.21, fee_rate was in " + CURRENCY_UNIT + "/kvB. As of 0.21, fee_rate is in " + CURRENCY_ATOM + "/vB. *\n",
     983                 :             :         {
     984   [ +  -  +  - ]:        3678 :             {"txid", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "The txid to be bumped"},
     985   [ +  -  +  - ]:        3678 :             {"options", RPCArg::Type::OBJ_NAMED_PARAMS, RPCArg::Optional::OMITTED, "",
     986   [ +  -  +  -  :       14712 :                 Cat(
             +  +  -  - ]
     987                 :             :                 {
     988   [ +  -  +  -  :        5517 :                     {"conf_target", RPCArg::Type::NUM, RPCArg::DefaultHint{"wallet -txconfirmtarget"}, "Confirmation target in blocks\n"},
                   +  - ]
     989   [ +  -  +  - ]:        3678 :                     {"fee_rate", RPCArg::Type::AMOUNT, RPCArg::DefaultHint{"not set, fall back to wallet fee estimation"},
     990         [ +  - ]:        3678 :                              "\nSpecify a fee rate in " + CURRENCY_ATOM + "/vB instead of relying on the built-in fee estimator.\n"
     991         [ +  - ]:        3678 :                              "Must be at least " + incremental_fee + " higher than the current transaction fee rate.\n"
     992   [ +  -  +  - ]:        5517 :                              "WARNING: before version 0.21, fee_rate was in " + CURRENCY_UNIT + "/kvB. As of 0.21, fee_rate is in " + CURRENCY_ATOM + "/vB.\n"},
     993   [ +  -  +  - ]:        3678 :                     {"replaceable", RPCArg::Type::BOOL, RPCArg::Default{true},
     994         [ +  - ]:        3678 :                              "Whether the new transaction should be\n"
     995                 :             :                              "marked bip-125 replaceable. If true, the sequence numbers in the transaction will\n"
     996                 :             :                              "be set to 0xfffffffd. If false, any input sequence numbers in the\n"
     997                 :             :                              "transaction will be set to 0xfffffffe\n"
     998                 :             :                              "so the new transaction will not be explicitly bip-125 replaceable (though it may\n"
     999                 :             :                              "still be replaceable in practice, for example if it has unconfirmed ancestors which\n"
    1000                 :             :                              "are replaceable).\n"},
    1001   [ +  -  +  - ]:        3678 :                     {"estimate_mode", RPCArg::Type::STR, RPCArg::Default{"unset"}, "The fee estimate mode, must be one of (case insensitive):\n"
    1002   [ +  -  +  -  :        3678 :                               + FeeModesDetail(std::string("economical mode is used if the transaction is replaceable;\notherwise, conservative mode is used"))},
                   +  - ]
    1003   [ +  -  +  - ]:        3678 :                     {"outputs", RPCArg::Type::ARR, RPCArg::Default{UniValue::VARR}, "The outputs specified as key-value pairs.\n"
    1004                 :             :                              "Each key may only appear once, i.e. there can only be one 'data' output, and no address may be duplicated.\n"
    1005                 :             :                              "At least one output of either type must be specified.\n"
    1006                 :             :                              "Cannot be provided if 'original_change_index' is specified.",
    1007         [ +  - ]:        3678 :                         OutputsDoc(),
    1008         [ +  - ]:        3678 :                         RPCArgOptions{.skip_type_check = true}},
    1009   [ +  -  +  -  :        5517 :                     {"original_change_index", RPCArg::Type::NUM, RPCArg::DefaultHint{"not set, detect change automatically"}, "The 0-based index of the change output on the original transaction. "
                   +  - ]
    1010                 :             :                                                                                                                             "The indicated output will be recycled into the new change output on the bumped transaction. "
    1011                 :             :                                                                                                                             "The remainder after paying the recipients and fees will be sent to the output script of the "
    1012                 :             :                                                                                                                             "original change output. The change output’s amount can increase if bumping the transaction "
    1013                 :             :                                                                                                                             "adds new inputs, otherwise it will decrease. Cannot be used in combination with the 'outputs' option."},
    1014                 :             :                 },
    1015   [ +  +  +  -  :        9762 :                 want_psbt ? std::vector<RPCArg>{{"psbt_version", RPCArg::Type::NUM, RPCArg::Default(2), "The PSBT version number to use."}} : std::vector<RPCArg>()
          +  -  +  -  +  
          -  +  +  +  +  
          +  +  +  +  +  
          +  +  +  -  -  
          -  -  -  -  -  
             -  -  -  -  
                      - ]
    1016                 :             :                 ),
    1017         [ +  - ]:        3678 :                 RPCArgOptions{.oneline_description="options"}},
    1018                 :             :         },
    1019         [ +  - ]:        3678 :         RPCResult{
    1020   [ +  -  +  -  :       25746 :             RPCResult::Type::OBJ, "", "", Cat(
          +  -  +  -  +  
          -  +  +  +  +  
             -  -  -  - ]
    1021   [ +  +  +  -  :        9195 :                 want_psbt ?
          +  -  +  +  +  
          +  +  +  +  +  
          -  -  -  -  -  
                -  -  - ]
    1022   [ +  -  +  -  :        5235 :                 std::vector<RPCResult>{{RPCResult::Type::STR, "psbt", "The base64-encoded unsigned PSBT of the new transaction."}} :
          +  +  +  +  +  
          +  -  -  -  -  
                   -  - ]
    1023   [ +  -  +  -  :        5799 :                 std::vector<RPCResult>{{RPCResult::Type::STR_HEX, "txid", "The id of the new transaction."}},
          +  +  +  +  +  
          +  -  -  -  -  
                   -  - ]
    1024                 :             :             {
    1025   [ +  -  +  - ]:        3678 :                 {RPCResult::Type::STR_AMOUNT, "origfee", "The fee of the replaced transaction."},
    1026   [ +  -  +  - ]:        3678 :                 {RPCResult::Type::STR_AMOUNT, "fee", "The fee of the new transaction."},
    1027   [ +  -  +  - ]:        3678 :                 {RPCResult::Type::ARR, "errors", "Errors encountered during processing (may be empty).",
    1028                 :             :                 {
    1029   [ +  -  +  - ]:        3678 :                     {RPCResult::Type::STR, "", ""},
    1030                 :             :                 }},
    1031                 :             :             })
    1032         [ +  - ]:        3678 :         },
    1033                 :        1839 :         RPCExamples{
    1034   [ +  +  +  -  :        4668 :     "\nBump the fee, get the new transaction\'s " + std::string(want_psbt ? "psbt" : "txid") + "\n" +
                   +  - ]
    1035   [ +  -  +  -  :        5517 :             HelpExampleCli(method_name, "<txid>")
                   +  - ]
    1036                 :        1839 :         },
    1037         [ +  - ]:        3678 :         [want_psbt](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
    1038                 :             : {
    1039                 :         161 :     std::shared_ptr<CWallet> const pwallet = GetWalletForJSONRPCRequest(request);
    1040         [ -  + ]:         161 :     if (!pwallet) return UniValue::VNULL;
    1041                 :             : 
    1042   [ +  -  +  +  :         161 :     if (pwallet->IsWalletFlagSet(WALLET_FLAG_DISABLE_PRIVATE_KEYS) && !pwallet->IsWalletFlagSet(WALLET_FLAG_EXTERNAL_SIGNER) && !want_psbt) {
          +  -  +  +  +  
                      + ]
    1043   [ +  -  +  - ]:           2 :         throw JSONRPCError(RPC_WALLET_ERROR, "bumpfee is not available with wallets that have private keys disabled. Use psbtbumpfee instead.");
    1044                 :             :     }
    1045                 :             : 
    1046   [ +  -  +  - ]:         160 :     Txid hash{Txid::FromUint256(ParseHashV(request.params[0], "txid"))};
    1047                 :             : 
    1048         [ +  - ]:         160 :     CCoinControl coin_control;
    1049                 :             :     // optional parameters
    1050                 :         160 :     coin_control.m_signal_bip125_rbf = true;
    1051                 :         160 :     std::vector<CTxOut> outputs;
    1052                 :             : 
    1053                 :         160 :     std::optional<uint32_t> original_change_index;
    1054                 :             : 
    1055                 :         160 :     uint32_t psbt_version = 2;
    1056                 :             : 
    1057   [ +  -  +  + ]:         160 :     if (!request.params[1].isNull()) {
    1058   [ +  -  +  - ]:          70 :         UniValue options = request.params[1];
    1059   [ +  +  +  +  :         704 :         RPCTypeCheckObj(options,
                   +  + ]
    1060                 :             :             {
    1061         [ +  - ]:          70 :                 {"confTarget", UniValueType(UniValue::VNUM)},
    1062         [ +  - ]:          70 :                 {"conf_target", UniValueType(UniValue::VNUM)},
    1063         [ +  - ]:          70 :                 {"fee_rate", UniValueType()}, // will be checked by AmountFromValue() in SetFeeEstimateMode()
    1064         [ +  - ]:          70 :                 {"replaceable", UniValueType(UniValue::VBOOL)},
    1065         [ +  - ]:          70 :                 {"estimate_mode", UniValueType(UniValue::VSTR)},
    1066         [ +  - ]:          70 :                 {"outputs", UniValueType()}, // will be checked by AddOutputs()
    1067         [ +  - ]:          70 :                 {"original_change_index", UniValueType(UniValue::VNUM)},
    1068         [ +  - ]:          70 :                 {"psbt_version", UniValueType(UniValue::VNUM)},
    1069                 :             :             },
    1070                 :             :             true, true);
    1071                 :             : 
    1072   [ +  -  +  -  :         133 :         if (options.exists("confTarget") && options.exists("conf_target")) {
          +  +  +  -  +  
          -  +  -  +  +  
                   -  - ]
    1073   [ +  -  +  - ]:           2 :             throw JSONRPCError(RPC_INVALID_PARAMETER, "confTarget and conf_target options should not both be set. Use conf_target (confTarget is deprecated).");
    1074                 :             :         }
    1075                 :             : 
    1076   [ +  -  -  +  :         195 :         auto conf_target = options.exists("confTarget") ? options["confTarget"] : options["conf_target"];
          -  -  -  -  +  
          -  +  -  +  -  
          +  -  -  +  -  
                -  -  - ]
    1077                 :             : 
    1078   [ +  -  +  + ]:         130 :         if (options.exists("replaceable")) {
    1079   [ +  -  +  -  :           1 :             coin_control.m_signal_bip125_rbf = options["replaceable"].get_bool();
                   +  - ]
    1080                 :             :         }
    1081   [ +  -  +  -  :         149 :         SetFeeEstimateMode(*pwallet, coin_control, conf_target, options["estimate_mode"], options["fee_rate"], /*override_min_fee=*/false);
          +  -  +  -  +  
                      + ]
    1082                 :             : 
    1083                 :             :         // Prepare new outputs by creating a temporary tx and calling AddOutputs().
    1084   [ +  -  +  -  :          46 :         if (!options["outputs"].isNull()) {
                   +  + ]
    1085   [ +  -  +  -  :          31 :             if (options["outputs"].isArray() && options["outputs"].empty()) {
          +  +  +  -  +  
          -  -  +  +  +  
             +  +  -  - ]
    1086   [ +  -  +  - ]:           2 :                 throw JSONRPCError(RPC_INVALID_PARAMETER, "Invalid parameter, output argument cannot be an empty array");
    1087                 :             :             }
    1088         [ +  - ]:          10 :             CMutableTransaction tempTx;
    1089   [ +  -  +  -  :          12 :             AddOutputs(tempTx, options["outputs"]);
                   +  + ]
    1090         [ +  - ]:           8 :             outputs = tempTx.vout;
    1091                 :          10 :         }
    1092                 :             : 
    1093   [ +  -  +  + ]:          86 :         if (options.exists("original_change_index")) {
    1094   [ +  -  +  -  :           7 :             original_change_index = options["original_change_index"].getInt<uint32_t>();
                   +  + ]
    1095                 :             :         }
    1096                 :             : 
    1097   [ +  -  +  + ]:          84 :         if (options.exists("psbt_version")) {
    1098   [ +  -  +  -  :           3 :             psbt_version = options["psbt_version"].getInt<uint32_t>();
                   +  - ]
    1099                 :             :         }
    1100         [ +  + ]:          42 :         if (psbt_version != 2 && psbt_version != 0) {
    1101   [ +  -  +  - ]:           2 :             throw JSONRPCError(RPC_INVALID_PARAMETER, "The PSBT version can only be 2 or 0");
    1102                 :             :         }
    1103                 :          94 :     }
    1104                 :             : 
    1105                 :             :     // Make sure the results are valid at least up to the most recent block
    1106                 :             :     // the user could have gotten from another RPC command prior to now
    1107         [ +  - ]:         131 :     pwallet->BlockUntilSyncedToCurrentChain();
    1108                 :             : 
    1109         [ +  - ]:         131 :     LOCK(pwallet->cs_wallet);
    1110                 :             : 
    1111         [ +  + ]:         131 :     EnsureWalletIsUnlocked(*pwallet);
    1112                 :             : 
    1113                 :             : 
    1114                 :         130 :     std::vector<bilingual_str> errors;
    1115                 :         130 :     CAmount old_fee;
    1116                 :         130 :     CAmount new_fee;
    1117         [ +  - ]:         130 :     CMutableTransaction mtx;
    1118                 :             :     // Targeting feerate bump.
    1119                 :         260 :     [&](){
    1120   [ -  -  +  +  :         130 :         switch (feebumper::CreateRateBumpTransaction(*pwallet, hash, coin_control, errors, old_fee, new_fee, mtx, /*require_mine=*/ !want_psbt, outputs, original_change_index)) {
                +  -  + ]
    1121                 :         107 :             case feebumper::Result::OK:
    1122                 :         107 :                 return;
    1123                 :           0 :             case feebumper::Result::INVALID_ADDRESS_OR_KEY:
    1124         [ #  # ]:           0 :                 throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, errors[0].original);
    1125                 :           0 :             case feebumper::Result::INVALID_REQUEST:
    1126         [ #  # ]:           0 :                 throw JSONRPCError(RPC_INVALID_REQUEST, errors[0].original);
    1127                 :          16 :             case feebumper::Result::INVALID_PARAMETER:
    1128         [ +  - ]:          16 :                 throw JSONRPCError(RPC_INVALID_PARAMETER, errors[0].original);
    1129                 :           6 :             case feebumper::Result::WALLET_ERROR:
    1130         [ +  - ]:           6 :                 throw JSONRPCError(RPC_WALLET_ERROR, errors[0].original);
    1131                 :           1 :             case feebumper::Result::MISC_ERROR:
    1132         [ +  - ]:           1 :                 throw JSONRPCError(RPC_MISC_ERROR, errors[0].original);
    1133                 :             :         } // no default case, so the compiler can warn about missing cases
    1134         [ #  # ]:           0 :         NONFATAL_UNREACHABLE();
    1135         [ +  + ]:         130 :     }();
    1136                 :             : 
    1137                 :         107 :     UniValue result(UniValue::VOBJ);
    1138                 :             : 
    1139                 :             :     // For bumpfee, return the new transaction id.
    1140                 :             :     // For psbtbumpfee, return the base64-encoded unsigned PSBT of the new transaction.
    1141         [ +  + ]:         107 :     if (!want_psbt) {
    1142   [ +  -  -  + ]:          98 :         if (!feebumper::SignTransaction(*pwallet, mtx)) {
    1143   [ #  #  #  # ]:           0 :             if (pwallet->IsWalletFlagSet(WALLET_FLAG_EXTERNAL_SIGNER)) {
    1144   [ #  #  #  # ]:           0 :                 throw JSONRPCError(RPC_WALLET_ERROR, "Transaction incomplete. Try psbtbumpfee instead.");
    1145                 :             :             }
    1146   [ #  #  #  # ]:           0 :             throw JSONRPCError(RPC_WALLET_ERROR, "Can't sign transaction.");
    1147                 :             :         }
    1148                 :             : 
    1149         [ +  - ]:          98 :         Txid txid;
    1150   [ +  -  -  + ]:          98 :         if (feebumper::CommitTransaction(*pwallet, hash, std::move(mtx), errors, txid) != feebumper::Result::OK) {
    1151         [ #  # ]:           0 :             throw JSONRPCError(RPC_WALLET_ERROR, errors[0].original);
    1152                 :             :         }
    1153                 :             : 
    1154   [ +  -  +  -  :         196 :         result.pushKV("txid", txid.GetHex());
             +  -  +  - ]
    1155                 :             :     } else {
    1156         [ +  - ]:           9 :         PartiallySignedTransaction psbtx(mtx, psbt_version);
    1157                 :           9 :         bool complete = false;
    1158         [ +  - ]:           9 :         const auto err{pwallet->FillPSBT(psbtx, {.sign = false, .bip32_derivs = true}, complete)};
    1159         [ +  - ]:           9 :         CHECK_NONFATAL(!err);
    1160         [ +  - ]:           9 :         CHECK_NONFATAL(!complete);
    1161                 :           9 :         DataStream ssTx{};
    1162         [ +  - ]:           9 :         ssTx << psbtx;
    1163   [ +  -  +  -  :          27 :         result.pushKV("psbt", EncodeBase64(ssTx.str()));
          +  -  +  -  +  
                      - ]
    1164                 :           9 :     }
    1165                 :             : 
    1166   [ +  -  +  -  :         214 :     result.pushKV("origfee", ValueFromAmount(old_fee));
                   +  - ]
    1167   [ +  -  +  -  :         214 :     result.pushKV("fee", ValueFromAmount(new_fee));
                   +  - ]
    1168                 :         107 :     UniValue result_errors(UniValue::VARR);
    1169         [ -  + ]:         107 :     for (const bilingual_str& error : errors) {
    1170   [ #  #  #  # ]:           0 :         result_errors.push_back(error.original);
    1171                 :             :     }
    1172   [ +  -  +  - ]:         214 :     result.pushKV("errors", std::move(result_errors));
    1173                 :             : 
    1174                 :         107 :     return result;
    1175   [ +  -  +  -  :         631 : },
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
                -  -  + ]
    1176   [ +  +  +  -  :       12873 :     };
          +  -  +  +  -  
                      - ]
    1177   [ +  -  +  -  :       53190 : }
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  +  +  +  
          +  +  -  -  -  
          -  -  -  -  -  
             -  -  -  - ]
    1178                 :             : 
    1179         [ +  - ]:        1980 : RPCMethod bumpfee() { return bumpfee_helper("bumpfee"); }
    1180         [ +  - ]:        1698 : RPCMethod psbtbumpfee() { return bumpfee_helper("psbtbumpfee"); }
    1181                 :             : 
    1182                 :        1059 : RPCMethod send()
    1183                 :             : {
    1184                 :        1059 :     return RPCMethod{
    1185                 :        1059 :         "send",
    1186         [ +  - ]:        2118 :         "Send a transaction.\n",
    1187                 :             :         {
    1188   [ +  -  +  - ]:        2118 :             {"outputs", RPCArg::Type::ARR, RPCArg::Optional::NO, "The outputs specified as key-value pairs.\n"
    1189                 :             :                     "Each key may only appear once, i.e. there can only be one 'data' output, and no address may be duplicated.\n"
    1190                 :             :                     "At least one output of either type must be specified.\n"
    1191                 :             :                     "For convenience, a dictionary, which holds the key-value pairs directly, is also accepted.",
    1192         [ +  - ]:        2118 :                 OutputsDoc(),
    1193         [ +  - ]:        2118 :                 RPCArgOptions{.skip_type_check = true}},
    1194   [ +  -  +  -  :        3177 :             {"conf_target", RPCArg::Type::NUM, RPCArg::DefaultHint{"wallet -txconfirmtarget"}, "Confirmation target in blocks"},
                   +  - ]
    1195   [ +  -  +  - ]:        2118 :             {"estimate_mode", RPCArg::Type::STR, RPCArg::Default{"unset"}, "The fee estimate mode, must be one of (case insensitive):\n"
    1196   [ +  -  +  -  :        2118 :               + FeeModesDetail(std::string("economical mode is used if the transaction is replaceable;\notherwise, conservative mode is used"))},
                   +  - ]
    1197   [ +  -  +  -  :        3177 :             {"fee_rate", RPCArg::Type::AMOUNT, RPCArg::DefaultHint{"not set, fall back to wallet fee estimation"}, "Specify a fee rate in " + CURRENCY_ATOM + "/vB."},
                   +  - ]
    1198   [ +  -  +  - ]:        2118 :             {"options", RPCArg::Type::OBJ_NAMED_PARAMS, RPCArg::Optional::OMITTED, "",
    1199   [ +  -  +  -  :       31770 :                 Cat<std::vector<RPCArg>>(
          +  -  +  -  +  
          -  +  +  +  +  
          +  +  +  +  -  
          -  -  -  -  -  
                   -  - ]
    1200                 :             :                 {
    1201   [ +  -  +  -  :        3177 :                     {"add_inputs", RPCArg::Type::BOOL, RPCArg::DefaultHint{"false when \"inputs\" are specified, true otherwise"},"Automatically include coins from the wallet to cover the target amount.\n"},
                   +  - ]
    1202   [ +  -  +  -  :        3177 :                     {"include_unsafe", RPCArg::Type::BOOL, RPCArg::Default{false}, "Include inputs that are not safe to spend (unconfirmed transactions from outside keys and unconfirmed replacement transactions).\n"
                   +  - ]
    1203                 :             :                                                           "Warning: the resulting transaction may become invalid if one of the unsafe inputs disappears.\n"
    1204                 :             :                                                           "If that happens, you will need to fund the transaction with different inputs and republish it."},
    1205   [ +  -  +  -  :        3177 :                     {"minconf", RPCArg::Type::NUM, RPCArg::Default{0}, "If add_inputs is specified, require inputs with at least this many confirmations."},
                   +  - ]
    1206   [ +  -  +  - ]:        2118 :                     {"maxconf", RPCArg::Type::NUM, RPCArg::Optional::OMITTED, "If add_inputs is specified, require inputs with at most this many confirmations."},
    1207   [ +  -  +  -  :        3177 :                     {"add_to_wallet", RPCArg::Type::BOOL, RPCArg::Default{true}, "When false, returns a serialized transaction which will not be added to the wallet or broadcast"},
                   +  - ]
    1208   [ +  -  +  -  :        3177 :                     {"change_address", RPCArg::Type::STR, RPCArg::DefaultHint{"automatic"}, "The bitcoin address to receive the change"},
                   +  - ]
    1209   [ +  -  +  -  :        3177 :                     {"change_position", RPCArg::Type::NUM, RPCArg::DefaultHint{"random"}, "The index of the change output"},
                   +  - ]
    1210   [ +  -  +  -  :        3177 :                     {"change_type", RPCArg::Type::STR, RPCArg::DefaultHint{"set by -changetype"}, "The output type to use. Only valid if change_address is not specified. Options are " + FormatAllOutputTypes() + "."},
             +  -  +  - ]
    1211   [ +  -  +  -  :        4236 :                     {"fee_rate", RPCArg::Type::AMOUNT, RPCArg::DefaultHint{"not set, fall back to wallet fee estimation"}, "Specify a fee rate in " + CURRENCY_ATOM + "/vB.", RPCArgOptions{.also_positional = true}},
             +  -  +  - ]
    1212   [ +  -  +  -  :        3177 :                     {"include_watching", RPCArg::Type::BOOL, RPCArg::Default{"false"}, "(DEPRECATED) No longer used"},
                   +  - ]
    1213   [ +  -  +  - ]:        2118 :                     {"inputs", RPCArg::Type::ARR, RPCArg::Default{UniValue::VARR}, "Specify inputs instead of adding them automatically.",
    1214                 :             :                         {
    1215   [ +  -  +  - ]:        2118 :                           {"", RPCArg::Type::OBJ, RPCArg::Optional::OMITTED, "", {
    1216   [ +  -  +  - ]:        2118 :                             {"txid", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "The transaction id"},
    1217   [ +  -  +  - ]:        2118 :                             {"vout", RPCArg::Type::NUM, RPCArg::Optional::NO, "The output number"},
    1218   [ +  -  +  -  :        3177 :                             {"sequence", RPCArg::Type::NUM, RPCArg::DefaultHint{"depends on the value of the 'replaceable' and 'locktime' arguments"}, "The sequence number"},
                   +  - ]
    1219   [ +  -  +  -  :        3177 :                             {"weight", RPCArg::Type::NUM, RPCArg::DefaultHint{"Calculated from wallet and solving data"}, "The maximum weight for this input, "
                   +  - ]
    1220                 :             :                                         "including the weight of the outpoint and sequence number. "
    1221                 :             :                                         "Note that signature sizes are not guaranteed to be consistent, "
    1222                 :             :                                         "so the maximum DER signatures size of 73 bytes should be used when considering ECDSA signatures."
    1223                 :             :                                         "Remember to convert serialized sizes to weight units when necessary."},
    1224                 :             :                           }},
    1225                 :             :                         },
    1226                 :             :                     },
    1227   [ +  -  +  -  :        3177 :                     {"locktime", RPCArg::Type::NUM, RPCArg::DefaultHint{"locktime close to block height to prevent fee sniping"}, "Raw locktime. Non-0 value also locktime-activates inputs"},
                   +  - ]
    1228   [ +  -  +  -  :        3177 :                     {"lock_unspents", RPCArg::Type::BOOL, RPCArg::Default{false}, "Lock selected unspent outputs"},
                   +  - ]
    1229   [ +  -  +  -  :        3177 :                     {"psbt", RPCArg::Type::BOOL,  RPCArg::DefaultHint{"automatic"}, "Always return a PSBT, implies add_to_wallet=false."},
                   +  - ]
    1230   [ +  -  +  - ]:        2118 :                     {"subtract_fee_from_outputs", RPCArg::Type::ARR, RPCArg::Default{UniValue::VARR}, "Outputs to subtract the fee from, specified as integer indices.\n"
    1231                 :             :                     "The fee will be equally deducted from the amount of each specified output.\n"
    1232                 :             :                     "Those recipients will receive less bitcoins than you enter in their corresponding amount field.\n"
    1233                 :             :                     "If no outputs are specified here, the sender pays the fee.",
    1234                 :             :                         {
    1235   [ +  -  +  - ]:        2118 :                             {"vout_index", RPCArg::Type::NUM, RPCArg::Optional::OMITTED, "The zero-based output index, before a change output is added."},
    1236                 :             :                         },
    1237                 :             :                     },
    1238   [ +  -  +  -  :        3177 :                     {"max_tx_weight", RPCArg::Type::NUM, RPCArg::Default{MAX_STANDARD_TX_WEIGHT}, "The maximum acceptable transaction weight.\n"
                   +  - ]
    1239                 :             :                                                   "Transaction building will fail if this can not be satisfied."},
    1240                 :             :                 },
    1241         [ +  - ]:        2118 :                 FundTxDoc()),
    1242         [ +  - ]:        2118 :                 RPCArgOptions{.oneline_description="options"}},
    1243   [ +  -  +  -  :        3177 :                 {"version", RPCArg::Type::NUM, RPCArg::Default{DEFAULT_WALLET_TX_VERSION}, "Transaction version"},
                   +  - ]
    1244                 :             :         },
    1245         [ +  - ]:        2118 :         RPCResult{
    1246   [ +  -  +  - ]:        2118 :             RPCResult::Type::OBJ, "", "",
    1247                 :             :                 {
    1248   [ +  -  +  - ]:        2118 :                     {RPCResult::Type::BOOL, "complete", "If the transaction has a complete set of signatures"},
    1249   [ +  -  +  - ]:        2118 :                     {RPCResult::Type::STR_HEX, "txid", /*optional=*/true, "The transaction id for the send. Only 1 transaction is created regardless of the number of addresses."},
    1250   [ +  -  +  - ]:        2118 :                     {RPCResult::Type::STR_HEX, "hex", /*optional=*/true, "If add_to_wallet is false, the hex-encoded raw transaction with signature(s)"},
    1251   [ +  -  +  - ]:        2118 :                     {RPCResult::Type::STR, "psbt", /*optional=*/true, "If more signatures are needed, or if add_to_wallet is false, the base64-encoded (partially) signed transaction"}
    1252                 :             :                 }
    1253   [ +  -  +  -  :       10590 :         },
             +  +  -  - ]
    1254                 :        1059 :         RPCExamples{""
    1255                 :             :         "\nSend 0.1 BTC with a confirmation target of 6 blocks in economical fee estimate mode\n"
    1256   [ +  -  +  -  :        3177 :         + HelpExampleCli("send", "'{\"" + EXAMPLE_ADDRESS[0] + "\": 0.1}' 6 economical\n") +
             +  -  +  - ]
    1257         [ +  - ]:        2118 :         "Send 0.2 BTC with a fee rate of 1.1 " + CURRENCY_ATOM + "/vB using positional arguments\n"
    1258   [ +  -  +  -  :        5295 :         + HelpExampleCli("send", "'{\"" + EXAMPLE_ADDRESS[0] + "\": 0.2}' null \"unset\" 1.1\n") +
             +  -  +  - ]
    1259         [ +  - ]:        2118 :         "Send 0.2 BTC with a fee rate of 1 " + CURRENCY_ATOM + "/vB using the options argument\n"
    1260   [ +  -  +  -  :        5295 :         + HelpExampleCli("send", "'{\"" + EXAMPLE_ADDRESS[0] + "\": 0.2}' null \"unset\" null '{\"fee_rate\": 1}'\n") +
             +  -  +  - ]
    1261         [ +  - ]:        2118 :         "Send 0.3 BTC with a fee rate of 25 " + CURRENCY_ATOM + "/vB using named arguments\n"
    1262   [ +  -  +  -  :        5295 :         + HelpExampleCli("-named send", "outputs='{\"" + EXAMPLE_ADDRESS[0] + "\": 0.3}' fee_rate=25\n") +
             +  -  +  - ]
    1263                 :        1059 :         "Create a transaction that should confirm the next block, with a specific input, and return result without adding to wallet or broadcasting to the network\n"
    1264   [ +  -  +  -  :        5295 :         + HelpExampleCli("send", "'{\"" + EXAMPLE_ADDRESS[0] + "\": 0.1}' 1 economical null '{\"add_to_wallet\": false, \"inputs\": [{\"txid\":\"a08e6907dbbd3d809776dbfc5d82e371b764ed838b5655e72f463568df1aadf0\", \"vout\":1}]}'")
             +  -  +  - ]
    1265         [ +  - ]:        1059 :         },
    1266                 :        1059 :         [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
    1267                 :             :         {
    1268                 :         220 :             std::shared_ptr<CWallet> const pwallet = GetWalletForJSONRPCRequest(request);
    1269         [ -  + ]:         220 :             if (!pwallet) return UniValue::VNULL;
    1270                 :             : 
    1271   [ +  -  +  +  :         220 :             UniValue options{request.params[4].isNull() ? UniValue::VOBJ : request.params[4]};
             +  -  +  - ]
    1272   [ +  -  +  -  :         220 :             InterpretFeeEstimationInstructions(/*conf_target=*/request.params[1], /*estimate_mode=*/request.params[2], /*fee_rate=*/request.params[3], options);
             +  -  +  + ]
    1273         [ +  + ]:         216 :             PreventOutdatedOptions(options);
    1274                 :             : 
    1275                 :             : 
    1276   [ +  -  -  +  :         430 :             bool rbf{options.exists("replaceable") ? options["replaceable"].get_bool() : pwallet->m_signal_rbf};
          -  -  -  -  -  
                      - ]
    1277                 :         215 :             UniValue outputs(UniValue::VOBJ);
    1278   [ +  -  +  - ]:         215 :             outputs = NormalizeOutputs(request.params[0]);
    1279                 :         215 :             std::vector<CRecipient> recipients = CreateRecipients(
    1280         [ +  + ]:         429 :                     ParseOutputs(outputs),
    1281   [ +  -  +  -  :         431 :                     InterpretSubtractFeeFromOutputInstructions(options["subtract_fee_from_outputs"], outputs.getKeys())
             +  -  +  - ]
    1282         [ +  - ]:         214 :             );
    1283         [ +  - ]:         214 :             CCoinControl coin_control;
    1284         [ +  - ]:         214 :             coin_control.m_version = self.Arg<uint32_t>("version");
    1285   [ +  -  +  -  :         428 :             CMutableTransaction rawTx = ConstructTransaction(options["inputs"], request.params[0], options["locktime"], rbf, coin_control.m_version);
          +  -  +  -  +  
                -  +  - ]
    1286                 :             :             // Automatically select coins, unless at least one is manually selected. Can
    1287                 :             :             // be overridden by options.add_inputs.
    1288         [ -  + ]:         214 :             coin_control.m_allow_other_inputs = rawTx.vin.size() == 0;
    1289   [ +  -  -  + ]:         428 :             if (options.exists("max_tx_weight")) {
    1290   [ #  #  #  #  :           0 :                 coin_control.m_max_tx_weight = options["max_tx_weight"].getInt<int>();
                   #  # ]
    1291                 :             :             }
    1292                 :             : 
    1293   [ +  -  +  -  :         215 :             SetOptionsInputWeights(options["inputs"], options);
                   +  + ]
    1294                 :             :             // Clear tx.vout since it is not meant to be used now that we are passing outputs directly.
    1295                 :             :             // This sets us up for a future PR to completely remove tx from the function signature in favor of passing inputs directly
    1296                 :         213 :             rawTx.vout.clear();
    1297         [ +  + ]:         213 :             auto txr = FundTransaction(*pwallet, rawTx, recipients, options, coin_control, /*override_min_fee=*/false);
    1298                 :             : 
    1299         [ +  - ]:         130 :             CMutableTransaction tx = CMutableTransaction(*txr.tx);
    1300   [ +  -  +  + ]:         390 :             return FinishTransaction(pwallet, options, tx);
    1301         [ +  - ]:         865 :         }
    1302   [ +  -  +  -  :       10590 :     };
             +  +  -  - ]
    1303   [ +  -  +  -  :       64599 : }
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          -  -  -  -  -  
                -  -  - ]
    1304                 :             : 
    1305                 :        1013 : RPCMethod sendall()
    1306                 :             : {
    1307                 :        1013 :     return RPCMethod{"sendall",
    1308         [ +  - ]:        2026 :         "Spend the value of all (or specific) confirmed UTXOs and unconfirmed change in the wallet to one or more recipients.\n"
    1309                 :             :         "Unconfirmed inbound UTXOs and locked UTXOs will not be spent. Sendall will respect the avoid_reuse wallet flag.\n"
    1310                 :             :         "If your wallet contains many small inputs, either because it received tiny payments or as a result of accumulating change, consider using `send_max` to exclude inputs that are worth less than the fees needed to spend them.\n",
    1311                 :             :         {
    1312   [ +  -  +  - ]:        2026 :             {"recipients", RPCArg::Type::ARR, RPCArg::Optional::NO, "The sendall destinations. Each address may only appear once.\n"
    1313                 :             :                 "Optionally some recipients can be specified with an amount to perform payments, but at least one address must appear without a specified amount.\n",
    1314                 :             :                 {
    1315   [ +  -  +  - ]:        2026 :                     {"address", RPCArg::Type::STR, RPCArg::Optional::NO, "A bitcoin address which receives an equal share of the unspecified amount."},
    1316   [ +  -  +  - ]:        2026 :                     {"", RPCArg::Type::OBJ_USER_KEYS, RPCArg::Optional::OMITTED, "",
    1317                 :             :                         {
    1318   [ +  -  +  - ]:        2026 :                             {"address", RPCArg::Type::AMOUNT, RPCArg::Optional::NO, "A key-value pair. The key (string) is the bitcoin address, the value (float or string) is the amount in " + CURRENCY_UNIT + ""},
    1319                 :             :                         },
    1320                 :             :                     },
    1321                 :             :                 },
    1322                 :             :             },
    1323   [ +  -  +  -  :        3039 :             {"conf_target", RPCArg::Type::NUM, RPCArg::DefaultHint{"wallet -txconfirmtarget"}, "Confirmation target in blocks"},
                   +  - ]
    1324   [ +  -  +  - ]:        2026 :             {"estimate_mode", RPCArg::Type::STR, RPCArg::Default{"unset"}, "The fee estimate mode, must be one of (case insensitive):\n"
    1325   [ +  -  +  -  :        2026 :               + FeeModesDetail(std::string("economical mode is used if the transaction is replaceable;\notherwise, conservative mode is used"))},
                   +  - ]
    1326   [ +  -  +  -  :        3039 :             {"fee_rate", RPCArg::Type::AMOUNT, RPCArg::DefaultHint{"not set, fall back to wallet fee estimation"}, "Specify a fee rate in " + CURRENCY_ATOM + "/vB."},
                   +  - ]
    1327                 :             :             {
    1328   [ +  -  +  - ]:        2026 :                 "options", RPCArg::Type::OBJ_NAMED_PARAMS, RPCArg::Optional::OMITTED, "",
    1329   [ +  -  +  -  :       21273 :                 Cat<std::vector<RPCArg>>(
          +  -  +  -  +  
          +  +  +  +  +  
          -  -  -  -  -  
                      - ]
    1330                 :             :                     {
    1331   [ +  -  +  -  :        3039 :                         {"add_to_wallet", RPCArg::Type::BOOL, RPCArg::Default{true}, "When false, returns the serialized transaction without broadcasting or adding it to the wallet"},
                   +  - ]
    1332   [ +  -  +  -  :        4052 :                         {"fee_rate", RPCArg::Type::AMOUNT, RPCArg::DefaultHint{"not set, fall back to wallet fee estimation"}, "Specify a fee rate in " + CURRENCY_ATOM + "/vB.", RPCArgOptions{.also_positional = true}},
             +  -  +  - ]
    1333   [ +  -  +  -  :        3039 :                         {"include_watching", RPCArg::Type::BOOL, RPCArg::Default{false}, "(DEPRECATED) No longer used"},
                   +  - ]
    1334   [ +  -  +  - ]:        2026 :                         {"inputs", RPCArg::Type::ARR, RPCArg::Default{UniValue::VARR}, "Use exactly the specified inputs to build the transaction. Specifying inputs is incompatible with the send_max, minconf, and maxconf options.",
    1335                 :             :                             {
    1336   [ +  -  +  - ]:        2026 :                                 {"", RPCArg::Type::OBJ, RPCArg::Optional::OMITTED, "",
    1337                 :             :                                     {
    1338   [ +  -  +  - ]:        2026 :                                         {"txid", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "The transaction id"},
    1339   [ +  -  +  - ]:        2026 :                                         {"vout", RPCArg::Type::NUM, RPCArg::Optional::NO, "The output number"},
    1340   [ +  -  +  -  :        3039 :                                         {"sequence", RPCArg::Type::NUM, RPCArg::DefaultHint{"depends on the value of the 'replaceable' and 'locktime' arguments"}, "The sequence number"},
                   +  - ]
    1341                 :             :                                     },
    1342                 :             :                                 },
    1343                 :             :                             },
    1344                 :             :                         },
    1345   [ +  -  +  -  :        3039 :                         {"locktime", RPCArg::Type::NUM, RPCArg::DefaultHint{"locktime close to block height to prevent fee sniping"}, "Raw locktime. Non-0 value also locktime-activates inputs"},
                   +  - ]
    1346   [ +  -  +  -  :        3039 :                         {"lock_unspents", RPCArg::Type::BOOL, RPCArg::Default{false}, "Lock selected unspent outputs"},
                   +  - ]
    1347   [ +  -  +  -  :        3039 :                         {"psbt", RPCArg::Type::BOOL,  RPCArg::DefaultHint{"automatic"}, "Always return a PSBT, implies add_to_wallet=false."},
                   +  - ]
    1348   [ +  -  +  -  :        3039 :                         {"send_max", RPCArg::Type::BOOL, RPCArg::Default{false}, "When true, only use UTXOs that can pay for their own fees to maximize the output amount. When 'false' (default), no UTXO is left behind. send_max is incompatible with providing specific inputs."},
                   +  - ]
    1349   [ +  -  +  -  :        3039 :                         {"minconf", RPCArg::Type::NUM, RPCArg::Default{0}, "Require inputs with at least this many confirmations."},
                   +  - ]
    1350   [ +  -  +  - ]:        2026 :                         {"maxconf", RPCArg::Type::NUM, RPCArg::Optional::OMITTED, "Require inputs with at most this many confirmations."},
    1351   [ +  -  +  -  :        3039 :                         {"version", RPCArg::Type::NUM, RPCArg::Default{DEFAULT_WALLET_TX_VERSION}, "Transaction version"},
                   +  - ]
    1352                 :             :                     },
    1353         [ +  - ]:        2026 :                     FundTxDoc()
    1354                 :             :                 ),
    1355         [ +  - ]:        2026 :                 RPCArgOptions{.oneline_description="options"}
    1356                 :             :             },
    1357                 :             :         },
    1358         [ +  - ]:        2026 :         RPCResult{
    1359   [ +  -  +  - ]:        2026 :             RPCResult::Type::OBJ, "", "",
    1360                 :             :                 {
    1361   [ +  -  +  - ]:        2026 :                     {RPCResult::Type::BOOL, "complete", "If the transaction has a complete set of signatures"},
    1362   [ +  -  +  - ]:        2026 :                     {RPCResult::Type::STR_HEX, "txid", /*optional=*/true, "The transaction id for the send. Only 1 transaction is created regardless of the number of addresses."},
    1363   [ +  -  +  - ]:        2026 :                     {RPCResult::Type::STR_HEX, "hex", /*optional=*/true, "If add_to_wallet is false, the hex-encoded raw transaction with signature(s)"},
    1364   [ +  -  +  - ]:        2026 :                     {RPCResult::Type::STR, "psbt", /*optional=*/true, "If more signatures are needed, or if add_to_wallet is false, the base64-encoded (partially) signed transaction"}
    1365                 :             :                 }
    1366   [ +  -  +  -  :       10130 :         },
             +  +  -  - ]
    1367                 :        1013 :         RPCExamples{""
    1368         [ +  - ]:        2026 :         "\nSpend all UTXOs from the wallet with a fee rate of 1 " + CURRENCY_ATOM + "/vB using named arguments\n"
    1369   [ +  -  +  -  :        5065 :         + HelpExampleCli("-named sendall", "recipients='[\"" + EXAMPLE_ADDRESS[0] + "\"]' fee_rate=1\n") +
             +  -  +  - ]
    1370         [ +  - ]:        2026 :         "Spend all UTXOs with a fee rate of 1.1 " + CURRENCY_ATOM + "/vB using positional arguments\n"
    1371   [ +  -  +  -  :        5065 :         + HelpExampleCli("sendall", "'[\"" + EXAMPLE_ADDRESS[0] + "\"]' null \"unset\" 1.1\n") +
             +  -  +  - ]
    1372         [ +  - ]:        2026 :         "Spend all UTXOs split into equal amounts to two addresses with a fee rate of 1.5 " + CURRENCY_ATOM + "/vB using the options argument\n"
    1373   [ +  -  +  -  :        6078 :         + HelpExampleCli("sendall", "'[\"" + EXAMPLE_ADDRESS[0] + "\", \"" + EXAMPLE_ADDRESS[1] + "\"]' null \"unset\" null '{\"fee_rate\": 1.5}'\n") +
          +  -  +  -  +  
                      - ]
    1374         [ +  - ]:        2026 :         "Leave dust UTXOs in wallet, spend only UTXOs with positive effective value with a fee rate of 10 " + CURRENCY_ATOM + "/vB using the options argument\n"
    1375   [ +  -  +  -  :        5065 :         + HelpExampleCli("sendall", "'[\"" + EXAMPLE_ADDRESS[0] + "\"]' null \"unset\" null '{\"fee_rate\": 10, \"send_max\": true}'\n") +
             +  -  +  - ]
    1376   [ +  -  +  - ]:        3039 :         "Spend all UTXOs with a fee rate of 1.3 " + CURRENCY_ATOM + "/vB using named arguments and sending a 0.25 " + CURRENCY_UNIT + " to another recipient\n"
    1377   [ +  -  +  -  :        6078 :         + HelpExampleCli("-named sendall", "recipients='[{\"" + EXAMPLE_ADDRESS[1] + "\": 0.25}, \""+ EXAMPLE_ADDRESS[0] + "\"]' fee_rate=1.3\n")
          +  -  +  -  +  
                      - ]
    1378         [ +  - ]:        1013 :         },
    1379                 :        1013 :         [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
    1380                 :             :         {
    1381                 :         174 :             std::shared_ptr<CWallet> const pwallet{GetWalletForJSONRPCRequest(request)};
    1382         [ -  + ]:         174 :             if (!pwallet) return UniValue::VNULL;
    1383                 :             :             // Make sure the results are valid at least up to the most recent block
    1384                 :             :             // the user could have gotten from another RPC command prior to now
    1385         [ +  - ]:         174 :             pwallet->BlockUntilSyncedToCurrentChain();
    1386                 :             : 
    1387   [ +  -  +  +  :         174 :             UniValue options{request.params[4].isNull() ? UniValue::VOBJ : request.params[4]};
             +  -  +  - ]
    1388   [ +  -  +  -  :         174 :             InterpretFeeEstimationInstructions(/*conf_target=*/request.params[1], /*estimate_mode=*/request.params[2], /*fee_rate=*/request.params[3], options);
             +  -  +  - ]
    1389         [ +  - ]:         174 :             PreventOutdatedOptions(options);
    1390                 :             : 
    1391                 :             : 
    1392                 :         174 :             std::set<std::string> addresses_without_amount;
    1393                 :         174 :             UniValue recipient_key_value_pairs(UniValue::VARR);
    1394         [ +  - ]:         174 :             const UniValue& recipients{request.params[0]};
    1395   [ -  +  +  + ]:        1057 :             for (unsigned int i = 0; i < recipients.size(); ++i) {
    1396         [ +  - ]:         883 :                 const UniValue& recipient{recipients[i]};
    1397         [ +  + ]:         883 :                 if (recipient.isStr()) {
    1398                 :         874 :                     UniValue rkvp(UniValue::VOBJ);
    1399   [ +  -  +  -  :        2622 :                     rkvp.pushKV(recipient.get_str(), 0);
             -  +  +  - ]
    1400         [ +  - ]:         874 :                     recipient_key_value_pairs.push_back(std::move(rkvp));
    1401   [ +  -  +  - ]:         874 :                     addresses_without_amount.insert(recipient.get_str());
    1402                 :         874 :                 } else {
    1403   [ +  -  +  - ]:           9 :                     recipient_key_value_pairs.push_back(recipient);
    1404                 :             :                 }
    1405                 :             :             }
    1406                 :             : 
    1407         [ +  + ]:         174 :             if (addresses_without_amount.size() == 0) {
    1408   [ +  -  +  - ]:           4 :                 throw JSONRPCError(RPC_INVALID_PARAMETER, "Must provide at least one address without a specified amount");
    1409                 :             :             }
    1410                 :             : 
    1411         [ +  - ]:         172 :             CCoinControl coin_control;
    1412                 :             : 
    1413   [ +  -  +  -  :         344 :             SetFeeEstimateMode(*pwallet, coin_control, options["conf_target"], options["estimate_mode"], options["fee_rate"], /*override_min_fee=*/false);
          +  -  +  -  +  
             -  +  -  +  
                      - ]
    1414                 :             : 
    1415   [ +  -  +  + ]:         344 :             if (options.exists("minconf")) {
    1416   [ +  -  +  -  :           7 :                 if (options["minconf"].getInt<int>() < 0)
             +  -  +  + ]
    1417                 :             :                 {
    1418   [ +  -  +  -  :           2 :                     throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Invalid minconf (minconf cannot be negative): %s", options["minconf"].getInt<int>()));
          +  -  +  -  +  
                      - ]
    1419                 :             :                 }
    1420                 :             : 
    1421   [ +  -  +  -  :           6 :                 coin_control.m_min_depth = options["minconf"].getInt<int>();
                   +  - ]
    1422                 :             :             }
    1423                 :             : 
    1424   [ +  -  +  + ]:         342 :             if (options.exists("maxconf")) {
    1425   [ +  -  +  -  :           2 :                 coin_control.m_max_depth = options["maxconf"].getInt<int>();
                   +  - ]
    1426                 :             : 
    1427         [ -  + ]:           2 :                 if (coin_control.m_max_depth < coin_control.m_min_depth) {
    1428   [ #  #  #  # ]:           0 :                     throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("maxconf can't be lower than minconf: %d < %d", coin_control.m_max_depth, coin_control.m_min_depth));
    1429                 :             :                 }
    1430                 :             :             }
    1431                 :             : 
    1432   [ +  -  +  + ]:         342 :             if (options.exists("version")) {
    1433   [ +  -  +  -  :           9 :                 coin_control.m_version = options["version"].getInt<decltype(coin_control.m_version)>();
                   +  - ]
    1434                 :             :             }
    1435                 :             : 
    1436         [ +  + ]:         171 :             if (coin_control.m_version == TRUC_VERSION) {
    1437         [ -  + ]:           7 :                 coin_control.m_max_tx_weight = TRUC_MAX_WEIGHT;
    1438                 :             :             } else {
    1439                 :         164 :                 coin_control.m_max_tx_weight = MAX_STANDARD_TX_WEIGHT;
    1440                 :             :             }
    1441                 :             : 
    1442   [ +  -  +  +  :         342 :             const bool rbf{options.exists("replaceable") ? options["replaceable"].get_bool() : pwallet->m_signal_rbf};
          +  -  +  -  +  
                      - ]
    1443                 :             : 
    1444                 :         171 :             FeeCalculation fee_calc_out;
    1445         [ +  - ]:         171 :             CFeeRate fee_rate{GetMinimumFeeRate(*pwallet, coin_control, &fee_calc_out)};
    1446                 :             :             // Do not, ever, assume that it's fine to change the fee rate if the user has explicitly
    1447                 :             :             // provided one
    1448   [ +  +  +  + ]:         171 :             if (coin_control.m_feerate && fee_rate > *coin_control.m_feerate) {
    1449                 :           1 :                 const auto feerate_format = FeeRateFormat::SAT_VB;
    1450                 :           1 :                 auto msg{strprintf("Fee rate (%s) is lower than the minimum fee rate setting (%s).",
    1451         [ +  - ]:           2 :                     coin_control.m_feerate->ToString(feerate_format),
    1452   [ +  -  +  - ]:           2 :                     fee_rate.ToString(feerate_format))};
    1453         [ +  - ]:           1 :                 if (fee_calc_out.reason == FeeReason::REQUIRED) {
    1454         [ +  - ]:           3 :                     msg += strprintf("\nConsider modifying -mintxfee (%s) or -minrelaytxfee (%s).",
    1455         [ +  - ]:           2 :                         pwallet->m_min_fee.ToString(feerate_format),
    1456   [ +  -  +  - ]:           3 :                         pwallet->chain().relayMinFee().ToString(feerate_format));
    1457                 :             :                 }
    1458         [ +  - ]:           1 :                 throw JSONRPCError(RPC_INVALID_PARAMETER, msg);
    1459                 :           1 :             }
    1460   [ +  +  +  - ]:         170 :             if (fee_calc_out.reason == FeeReason::FALLBACK && !pwallet->m_allow_fallback_fee) {
    1461                 :             :                 // eventually allow a fallback fee
    1462   [ #  #  #  # ]:           0 :                 throw JSONRPCError(RPC_WALLET_ERROR, "Fee estimation failed. Fallbackfee is disabled. Wait a few blocks or enable -fallbackfee.");
    1463                 :             :             }
    1464                 :             : 
    1465   [ +  -  +  -  :         341 :             CMutableTransaction rawTx{ConstructTransaction(options["inputs"], recipient_key_value_pairs, options["locktime"], rbf, coin_control.m_version)};
          +  -  +  -  +  
                      + ]
    1466         [ +  - ]:         169 :             LOCK(pwallet->cs_wallet);
    1467                 :             : 
    1468                 :         169 :             CAmount total_input_value(0);
    1469   [ +  -  +  +  :         340 :             bool send_max{options.exists("send_max") ? options["send_max"].get_bool() : false};
          +  -  +  -  +  
                      - ]
    1470   [ +  -  +  -  :         355 :             if (options.exists("inputs") && options.exists("send_max")) {
          +  +  +  -  +  
          -  +  +  +  +  
                   -  - ]
    1471   [ +  -  +  - ]:           2 :                 throw JSONRPCError(RPC_INVALID_PARAMETER, "Cannot combine send_max with specific inputs.");
    1472   [ +  -  +  -  :         367 :             } else if (options.exists("inputs") && (options.exists("minconf") || options.exists("maxconf"))) {
          +  +  +  -  +  
          -  +  +  +  -  
          +  -  -  +  +  
          +  +  +  -  -  
                   -  - ]
    1473   [ +  -  +  - ]:           2 :                 throw JSONRPCError(RPC_INVALID_PARAMETER, "Cannot combine minconf or maxconf with specific inputs.");
    1474   [ +  -  +  + ]:         334 :             } else if (options.exists("inputs")) {
    1475         [ +  + ]:          27 :                 for (const CTxIn& input : rawTx.vin) {
    1476   [ +  -  +  + ]:          16 :                     if (pwallet->IsSpent(input.prevout)) {
    1477   [ +  -  +  -  :           4 :                         throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Input not available. UTXO (%s:%d) was already spent.", input.prevout.hash.ToString(), input.prevout.n));
                   +  - ]
    1478                 :             :                     }
    1479         [ +  - ]:          14 :                     const CWalletTx* tx{pwallet->GetWalletTx(input.prevout.hash)};
    1480   [ +  +  -  +  :          14 :                     if (!tx || input.prevout.n >= tx->tx->vout.size() || !pwallet->IsMine(tx->tx->vout[input.prevout.n])) {
          +  +  +  -  +  
                      - ]
    1481   [ +  -  +  -  :           4 :                         throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Input not found. UTXO (%s:%d) is not part of wallet.", input.prevout.hash.ToString(), input.prevout.n));
                   +  - ]
    1482                 :             :                     }
    1483   [ +  -  +  + ]:          12 :                     if (pwallet->GetTxDepthInMainChain(*tx) == 0) {
    1484   [ -  +  -  - ]:           5 :                         if (tx->tx->version == TRUC_VERSION && coin_control.m_version != TRUC_VERSION) {
    1485   [ #  #  #  # ]:           0 :                             throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Can't spend unconfirmed version 3 pre-selected input with a version %d tx", coin_control.m_version));
    1486   [ -  +  -  - ]:           5 :                         } else if (coin_control.m_version == TRUC_VERSION && tx->tx->version != TRUC_VERSION) {
    1487   [ #  #  #  # ]:           0 :                             throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Can't spend unconfirmed version %d pre-selected input with a version 3 tx", tx->tx->version));
    1488                 :             :                         }
    1489                 :             :                     }
    1490                 :          12 :                     total_input_value += tx->tx->vout[input.prevout.n].nValue;
    1491                 :             :                 }
    1492                 :             :             } else {
    1493                 :         152 :                 CoinFilterParams coins_params;
    1494                 :         152 :                 coins_params.min_amount = 0;
    1495   [ +  -  +  -  :        2978 :                 for (const COutput& output : AvailableCoins(*pwallet, &coin_control, fee_rate, coins_params).All()) {
                   +  + ]
    1496   [ +  +  +  -  :        2826 :                     if (send_max && fee_rate.GetFee(output.input_bytes) > output.txout.nValue) {
                   +  + ]
    1497                 :           2 :                         continue;
    1498                 :             :                     }
    1499                 :             :                     // we are spending an unconfirmed TRUC transaction, so lower max weight
    1500   [ +  +  +  + ]:        2824 :                     if (output.depth == 0 && coin_control.m_version == TRUC_VERSION) {
    1501         [ +  - ]:           4 :                         coin_control.m_max_tx_weight = TRUC_CHILD_MAX_WEIGHT;
    1502                 :             :                     }
    1503   [ +  +  +  - ]:        2827 :                     CTxIn input(output.outpoint.hash, output.outpoint.n, CScript(), rbf ? MAX_BIP125_RBF_SEQUENCE : CTxIn::MAX_SEQUENCE_NONFINAL);
    1504         [ +  - ]:        2824 :                     rawTx.vin.push_back(input);
    1505                 :        2824 :                     total_input_value += output.txout.nValue;
    1506                 :        2976 :                 }
    1507                 :             :             }
    1508                 :             : 
    1509                 :         163 :             std::vector<COutPoint> outpoints_spent;
    1510   [ -  +  +  - ]:         163 :             outpoints_spent.reserve(rawTx.vin.size());
    1511                 :             : 
    1512         [ +  + ]:        2999 :             for (const CTxIn& tx_in : rawTx.vin) {
    1513         [ +  - ]:        2836 :                 outpoints_spent.push_back(tx_in.prevout);
    1514                 :             :             }
    1515                 :             : 
    1516                 :             :             // estimate final size of tx
    1517   [ +  -  +  - ]:         163 :             const TxSize tx_size{CalculateMaximumSignedTxSize(CTransaction(rawTx), pwallet.get())};
    1518         [ +  + ]:         163 :             if (tx_size.vsize == -1) {
    1519   [ +  -  +  - ]:           4 :                 throw JSONRPCError(RPC_WALLET_ERROR, "Unable to determine the size of the transaction, the wallet contains unsolvable descriptors");
    1520                 :             :             }
    1521         [ +  - ]:         161 :             const CAmount fee_from_size{fee_rate.GetFee(tx_size.vsize)};
    1522         [ +  - ]:         161 :             const std::optional<CAmount> total_bump_fees{pwallet->chain().calculateCombinedBumpFee(outpoints_spent, fee_rate)};
    1523         [ +  - ]:         161 :             CAmount effective_value = total_input_value - fee_from_size - total_bump_fees.value_or(0);
    1524                 :             : 
    1525         [ +  + ]:         161 :             if (fee_from_size > pwallet->m_default_max_tx_fee) {
    1526   [ +  -  +  - ]:           2 :                 throw JSONRPCError(RPC_WALLET_ERROR, TransactionErrorString(TransactionError::MAX_FEE_EXCEEDED).original);
    1527                 :             :             }
    1528                 :             : 
    1529         [ +  + ]:         160 :             if (effective_value <= 0) {
    1530         [ -  + ]:          33 :                 if (send_max) {
    1531   [ #  #  #  # ]:           0 :                     throw JSONRPCError(RPC_WALLET_INSUFFICIENT_FUNDS, "Total value of UTXO pool too low to pay for transaction, try using lower feerate.");
    1532                 :             :                 } else {
    1533   [ +  -  +  - ]:          66 :                     throw JSONRPCError(RPC_WALLET_INSUFFICIENT_FUNDS, "Total value of UTXO pool too low to pay for transaction. Try using lower feerate or excluding uneconomic UTXOs with 'send_max' option.");
    1534                 :             :                 }
    1535                 :             :             }
    1536                 :             : 
    1537                 :             :             // If this transaction is too large, e.g. because the wallet has many UTXOs, it will be rejected by the node's mempool.
    1538         [ +  - ]:         127 :             if (tx_size.weight > coin_control.m_max_tx_weight) {
    1539   [ +  -  +  - ]:           6 :                 throw JSONRPCError(RPC_WALLET_ERROR, "Transaction too large.");
    1540                 :             :             }
    1541                 :             : 
    1542                 :         124 :             CAmount output_amounts_claimed{0};
    1543         [ +  + ]:         609 :             for (const CTxOut& out : rawTx.vout) {
    1544                 :         485 :                 output_amounts_claimed += out.nValue;
    1545                 :             :             }
    1546                 :             : 
    1547         [ +  + ]:         124 :             if (output_amounts_claimed > total_input_value) {
    1548   [ +  -  +  - ]:           2 :                 throw JSONRPCError(RPC_WALLET_INSUFFICIENT_FUNDS, "Assigned more value to outputs than available funds.");
    1549                 :             :             }
    1550                 :             : 
    1551                 :         123 :             const CAmount remainder{effective_value - output_amounts_claimed};
    1552         [ +  + ]:         123 :             if (remainder < 0) {
    1553   [ +  -  +  - ]:           2 :                 throw JSONRPCError(RPC_WALLET_INSUFFICIENT_FUNDS, "Insufficient funds for fees after creating specified outputs.");
    1554                 :             :             }
    1555                 :             : 
    1556                 :         122 :             const CAmount per_output_without_amount{remainder / (long)addresses_without_amount.size()};
    1557                 :             : 
    1558                 :         122 :             bool gave_remaining_to_first{false};
    1559         [ +  + ]:         599 :             for (CTxOut& out : rawTx.vout) {
    1560                 :         480 :                 CTxDestination dest;
    1561         [ +  - ]:         480 :                 ExtractDestination(out.scriptPubKey, dest);
    1562         [ +  - ]:         480 :                 std::string addr{EncodeDestination(dest)};
    1563         [ +  + ]:         480 :                 if (addresses_without_amount.contains(addr)) {
    1564                 :         474 :                     out.nValue = per_output_without_amount;
    1565         [ +  + ]:         474 :                     if (!gave_remaining_to_first) {
    1566                 :         121 :                         out.nValue += remainder % addresses_without_amount.size();
    1567                 :         121 :                         gave_remaining_to_first = true;
    1568                 :             :                     }
    1569   [ +  -  +  -  :         474 :                     if (IsDust(out, pwallet->chain().relayDustFee())) {
                   +  + ]
    1570                 :             :                         // Dynamically generated output amount is dust
    1571   [ +  -  +  - ]:           4 :                         throw JSONRPCError(RPC_WALLET_INSUFFICIENT_FUNDS, "Dynamically assigned remainder results in dust output.");
    1572                 :             :                     }
    1573                 :             :                 } else {
    1574   [ +  -  +  -  :           6 :                     if (IsDust(out, pwallet->chain().relayDustFee())) {
                   +  + ]
    1575                 :             :                         // Specified output amount is dust
    1576   [ +  -  +  - ]:           2 :                         throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Specified output amount to %s is below dust threshold.", addr));
    1577                 :             :                     }
    1578                 :             :                 }
    1579                 :         483 :             }
    1580                 :             : 
    1581   [ +  -  +  +  :         240 :             const bool lock_unspents{options.exists("lock_unspents") ? options["lock_unspents"].get_bool() : false};
          +  -  +  -  +  
                      - ]
    1582         [ +  + ]:         119 :             if (lock_unspents) {
    1583         [ +  + ]:           4 :                 for (const CTxIn& txin : rawTx.vin) {
    1584         [ +  - ]:           2 :                     pwallet->LockCoin(txin.prevout, /*persist=*/false);
    1585                 :             :                 }
    1586                 :             :             }
    1587                 :             : 
    1588   [ +  -  +  - ]:         357 :             return FinishTransaction(pwallet, options, rawTx);
    1589         [ +  - ]:         733 :         }
    1590   [ +  -  +  -  :       16208 :     };
          +  -  +  -  +  
          +  +  +  +  +  
          -  -  -  -  -  
                      - ]
    1591   [ +  -  +  -  :       52676 : }
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          -  -  -  -  -  
             -  -  -  -  
                      - ]
    1592                 :             : 
    1593                 :        1459 : RPCMethod walletprocesspsbt()
    1594                 :             : {
    1595                 :        1459 :     return RPCMethod{
    1596                 :        1459 :         "walletprocesspsbt",
    1597                 :             :         "Update a PSBT with input information from our wallet and then sign inputs\n"
    1598                 :        1459 :                 "that we can sign for." +
    1599         [ +  - ]:        1459 :         HELP_REQUIRING_PASSPHRASE,
    1600                 :             :                 {
    1601   [ +  -  +  - ]:        2918 :                     {"psbt", RPCArg::Type::STR, RPCArg::Optional::NO, "The transaction base64 string"},
    1602   [ +  -  +  -  :        4377 :                     {"sign", RPCArg::Type::BOOL, RPCArg::Default{true}, "Also sign the transaction when updating (requires wallet to be unlocked)"},
                   +  - ]
    1603   [ +  -  +  -  :        4377 :                     {"sighashtype", RPCArg::Type::STR, RPCArg::Default{"DEFAULT for Taproot, ALL otherwise"}, "The signature hash type to sign with if not specified by the PSBT. Must be one of\n"
                   +  - ]
    1604                 :             :             "       \"DEFAULT\"\n"
    1605                 :             :             "       \"ALL\"\n"
    1606                 :             :             "       \"NONE\"\n"
    1607                 :             :             "       \"SINGLE\"\n"
    1608                 :             :             "       \"ALL|ANYONECANPAY\"\n"
    1609                 :             :             "       \"NONE|ANYONECANPAY\"\n"
    1610                 :             :             "       \"SINGLE|ANYONECANPAY\""},
    1611   [ +  -  +  -  :        4377 :                     {"bip32derivs", RPCArg::Type::BOOL, RPCArg::Default{true}, "Include BIP 32 derivation paths for public keys if we know them"},
                   +  - ]
    1612   [ +  -  +  -  :        4377 :                     {"finalize", RPCArg::Type::BOOL, RPCArg::Default{true}, "Also finalize inputs if possible"},
                   +  - ]
    1613                 :             :                 },
    1614         [ +  - ]:        2918 :                 RPCResult{
    1615   [ +  -  +  - ]:        2918 :                     RPCResult::Type::OBJ, "", "",
    1616                 :             :                     {
    1617   [ +  -  +  - ]:        2918 :                         {RPCResult::Type::STR, "psbt", "The base64-encoded partially signed transaction"},
    1618   [ +  -  +  - ]:        2918 :                         {RPCResult::Type::BOOL, "complete", "If the transaction has a complete set of signatures"},
    1619   [ +  -  +  - ]:        2918 :                         {RPCResult::Type::STR_HEX, "hex", /*optional=*/true, "The hex-encoded network transaction if complete"},
    1620                 :             :                     }
    1621   [ +  -  +  -  :       11672 :                 },
             +  +  -  - ]
    1622                 :        1459 :                 RPCExamples{
    1623   [ +  -  +  -  :        2918 :                     HelpExampleCli("walletprocesspsbt", "\"psbt\"")
                   +  - ]
    1624         [ +  - ]:        1459 :                 },
    1625                 :        1459 :         [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
    1626                 :             : {
    1627         [ -  + ]:         620 :     const std::shared_ptr<const CWallet> pwallet = GetWalletForJSONRPCRequest(request);
    1628         [ -  + ]:         620 :     if (!pwallet) return UniValue::VNULL;
    1629                 :             : 
    1630         [ +  - ]:         620 :     const CWallet& wallet{*pwallet};
    1631                 :             :     // Make sure the results are valid at least up to the most recent block
    1632                 :             :     // the user could have gotten from another RPC command prior to now
    1633         [ +  - ]:         620 :     wallet.BlockUntilSyncedToCurrentChain();
    1634                 :             : 
    1635                 :             :     // Unserialize the transaction
    1636   [ +  -  +  -  :         620 :     util::Result<PartiallySignedTransaction> psbt_res = DecodeBase64PSBT(request.params[0].get_str());
                   +  - ]
    1637         [ +  + ]:         620 :     if (!psbt_res) {
    1638   [ +  -  +  -  :           6 :         throw JSONRPCError(RPC_DESERIALIZATION_ERROR, strprintf("TX decode failed %s", util::ErrorString(psbt_res).original));
                   +  - ]
    1639                 :             :     }
    1640         [ +  - ]:         618 :     PartiallySignedTransaction psbtx = *psbt_res;
    1641                 :             : 
    1642                 :             :     // Get the sighash type
    1643   [ +  -  +  + ]:         618 :     std::optional<int> nHashType = ParseSighashString(request.params[2]);
    1644                 :             : 
    1645                 :             :     // Fill transaction with our data and also sign
    1646   [ +  -  +  +  :         617 :     bool sign = request.params[1].isNull() ? true : request.params[1].get_bool();
             +  -  +  - ]
    1647   [ +  -  +  +  :         617 :     bool bip32derivs = request.params[3].isNull() ? true : request.params[3].get_bool();
             +  -  +  - ]
    1648   [ +  -  +  +  :         617 :     bool finalize = request.params[4].isNull() ? true : request.params[4].get_bool();
             +  -  +  - ]
    1649                 :         617 :     bool complete = true;
    1650                 :             : 
    1651   [ +  +  +  + ]:         617 :     if (sign) EnsureWalletIsUnlocked(*pwallet);
    1652                 :             : 
    1653         [ +  - ]:         616 :     const auto err{wallet.FillPSBT(psbtx, {.sign = sign, .sighash_type = nHashType, .finalize = finalize, .bip32_derivs = bip32derivs}, complete)};
    1654         [ +  + ]:         616 :     if (err) {
    1655         [ +  - ]:           7 :         throw JSONRPCPSBTError(*err);
    1656                 :             :     }
    1657                 :             : 
    1658                 :         609 :     UniValue result(UniValue::VOBJ);
    1659                 :         609 :     DataStream ssTx{};
    1660         [ +  - ]:         609 :     ssTx << psbtx;
    1661   [ +  -  +  -  :        1827 :     result.pushKV("psbt", EncodeBase64(ssTx.str()));
          +  -  +  -  +  
                      - ]
    1662   [ +  -  +  -  :        1218 :     result.pushKV("complete", complete);
                   +  - ]
    1663         [ +  + ]:         609 :     if (complete) {
    1664         [ +  - ]:          40 :         CMutableTransaction mtx;
    1665                 :             :         // Returns true if complete, which we already think it is.
    1666   [ +  -  +  - ]:          40 :         CHECK_NONFATAL(FinalizeAndExtractPSBT(psbtx, mtx));
    1667                 :          40 :         DataStream ssTx_final;
    1668         [ +  - ]:          40 :         ssTx_final << TX_WITH_WITNESS(mtx);
    1669   [ -  +  +  -  :          80 :         result.pushKV("hex", HexStr(ssTx_final));
          +  -  +  -  +  
                      - ]
    1670                 :          80 :     }
    1671                 :             : 
    1672                 :         609 :     return result;
    1673                 :        1238 : },
    1674   [ +  -  +  -  :       13131 :     };
             +  +  -  - ]
    1675   [ +  -  +  -  :       23344 : }
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  -  -  -  
                      - ]
    1676                 :             : 
    1677                 :        1070 : RPCMethod walletcreatefundedpsbt()
    1678                 :             : {
    1679                 :        1070 :     return RPCMethod{
    1680                 :        1070 :         "walletcreatefundedpsbt",
    1681         [ +  - ]:        2140 :         "Creates and funds a transaction in the Partially Signed Transaction format.\n"
    1682                 :             :                 "Implements the Creator and Updater roles.\n"
    1683                 :             :                 "All existing inputs must either have their previous output transaction be in the wallet\n"
    1684                 :             :                 "or be in the UTXO set. Solving data must be provided for non-wallet inputs.\n",
    1685                 :             :                 {
    1686   [ +  -  +  - ]:        2140 :                     {"inputs", RPCArg::Type::ARR, RPCArg::Optional::OMITTED, "Leave empty to add inputs automatically. See add_inputs option.",
    1687                 :             :                         {
    1688   [ +  -  +  - ]:        2140 :                             {"", RPCArg::Type::OBJ, RPCArg::Optional::OMITTED, "",
    1689                 :             :                                 {
    1690   [ +  -  +  - ]:        2140 :                                     {"txid", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "The transaction id"},
    1691   [ +  -  +  - ]:        2140 :                                     {"vout", RPCArg::Type::NUM, RPCArg::Optional::NO, "The output number"},
    1692   [ +  -  +  -  :        3210 :                                     {"sequence", RPCArg::Type::NUM, RPCArg::DefaultHint{"depends on the value of the 'locktime' and 'options.replaceable' arguments"}, "The sequence number"},
                   +  - ]
    1693   [ +  -  +  -  :        3210 :                                     {"weight", RPCArg::Type::NUM, RPCArg::DefaultHint{"Calculated from wallet and solving data"}, "The maximum weight for this input, "
                   +  - ]
    1694                 :             :                                         "including the weight of the outpoint and sequence number. "
    1695                 :             :                                         "Note that signature sizes are not guaranteed to be consistent, "
    1696                 :             :                                         "so the maximum DER signatures size of 73 bytes should be used when considering ECDSA signatures."
    1697                 :             :                                         "Remember to convert serialized sizes to weight units when necessary."},
    1698                 :             :                                 },
    1699                 :             :                             },
    1700                 :             :                         },
    1701                 :             :                         },
    1702   [ +  -  +  - ]:        2140 :                     {"outputs", RPCArg::Type::ARR, RPCArg::Optional::NO, "The outputs specified as key-value pairs.\n"
    1703                 :             :                             "Each key may only appear once, i.e. there can only be one 'data' output, and no address may be duplicated.\n"
    1704                 :             :                             "At least one output of either type must be specified.\n"
    1705                 :             :                             "For compatibility reasons, a dictionary, which holds the key-value pairs directly, is also\n"
    1706                 :             :                             "accepted as second parameter.",
    1707         [ +  - ]:        2140 :                         OutputsDoc(),
    1708         [ +  - ]:        2140 :                         RPCArgOptions{.skip_type_check = true}},
    1709   [ +  -  +  -  :        3210 :                     {"locktime", RPCArg::Type::NUM, RPCArg::Default{0}, "Raw locktime. Non-0 value also locktime-activates inputs"},
                   +  - ]
    1710   [ +  -  +  - ]:        2140 :                     {"options", RPCArg::Type::OBJ_NAMED_PARAMS, RPCArg::Optional::OMITTED, "",
    1711   [ +  -  +  -  :       19260 :                         Cat<std::vector<RPCArg>>(
          +  -  +  +  +  
             +  -  -  -  
                      - ]
    1712                 :             :                         {
    1713   [ +  -  +  -  :        3210 :                             {"add_inputs", RPCArg::Type::BOOL, RPCArg::DefaultHint{"false when \"inputs\" are specified, true otherwise"}, "Automatically include coins from the wallet to cover the target amount.\n"},
                   +  - ]
    1714   [ +  -  +  -  :        3210 :                             {"include_unsafe", RPCArg::Type::BOOL, RPCArg::Default{false}, "Include inputs that are not safe to spend (unconfirmed transactions from outside keys and unconfirmed replacement transactions).\n"
                   +  - ]
    1715                 :             :                                                           "Warning: the resulting transaction may become invalid if one of the unsafe inputs disappears.\n"
    1716                 :             :                                                           "If that happens, you will need to fund the transaction with different inputs and republish it."},
    1717   [ +  -  +  -  :        3210 :                             {"minconf", RPCArg::Type::NUM, RPCArg::Default{0}, "If add_inputs is specified, require inputs with at least this many confirmations."},
                   +  - ]
    1718   [ +  -  +  - ]:        2140 :                             {"maxconf", RPCArg::Type::NUM, RPCArg::Optional::OMITTED, "If add_inputs is specified, require inputs with at most this many confirmations."},
    1719   [ +  -  +  -  :        3210 :                             {"changeAddress", RPCArg::Type::STR, RPCArg::DefaultHint{"automatic"}, "The bitcoin address to receive the change"},
                   +  - ]
    1720   [ +  -  +  -  :        3210 :                             {"changePosition", RPCArg::Type::NUM, RPCArg::DefaultHint{"random"}, "The index of the change output"},
                   +  - ]
    1721   [ +  -  +  -  :        3210 :                             {"change_type", RPCArg::Type::STR, RPCArg::DefaultHint{"set by -changetype"}, "The output type to use. Only valid if changeAddress is not specified. Options are " + FormatAllOutputTypes() + "."},
             +  -  +  - ]
    1722   [ +  -  +  -  :        3210 :                             {"includeWatching", RPCArg::Type::BOOL, RPCArg::Default{false}, "(DEPRECATED) No longer used"},
                   +  - ]
    1723   [ +  -  +  -  :        3210 :                             {"lockUnspents", RPCArg::Type::BOOL, RPCArg::Default{false}, "Lock selected unspent outputs"},
                   +  - ]
    1724   [ +  -  +  -  :        3210 :                             {"fee_rate", RPCArg::Type::AMOUNT, RPCArg::DefaultHint{"not set, fall back to wallet fee estimation"}, "Specify a fee rate in " + CURRENCY_ATOM + "/vB."},
                   +  - ]
    1725   [ +  -  +  -  :        3210 :                             {"feeRate", RPCArg::Type::AMOUNT, RPCArg::DefaultHint{"not set, fall back to wallet fee estimation"}, "Specify a fee rate in " + CURRENCY_UNIT + "/kvB."},
                   +  - ]
    1726   [ +  -  +  - ]:        2140 :                             {"subtractFeeFromOutputs", RPCArg::Type::ARR, RPCArg::Default{UniValue::VARR}, "The outputs to subtract the fee from.\n"
    1727                 :             :                                                           "The fee will be equally deducted from the amount of each specified output.\n"
    1728                 :             :                                                           "Those recipients will receive less bitcoins than you enter in their corresponding amount field.\n"
    1729                 :             :                                                           "If no outputs are specified here, the sender pays the fee.",
    1730                 :             :                                 {
    1731   [ +  -  +  - ]:        2140 :                                     {"vout_index", RPCArg::Type::NUM, RPCArg::Optional::OMITTED, "The zero-based output index, before a change output is added."},
    1732                 :             :                                 },
    1733                 :             :                             },
    1734   [ +  -  +  -  :        3210 :                             {"max_tx_weight", RPCArg::Type::NUM, RPCArg::Default{MAX_STANDARD_TX_WEIGHT}, "The maximum acceptable transaction weight.\n"
                   +  - ]
    1735                 :             :                                                           "Transaction building will fail if this can not be satisfied."},
    1736                 :             :                         },
    1737         [ +  - ]:        2140 :                         FundTxDoc()),
    1738         [ +  - ]:        2140 :                         RPCArgOptions{.oneline_description="options"}},
    1739   [ +  -  +  -  :        3210 :                     {"bip32derivs", RPCArg::Type::BOOL, RPCArg::Default{true}, "Include BIP 32 derivation paths for public keys if we know them"},
                   +  - ]
    1740   [ +  -  +  -  :        3210 :                     {"version", RPCArg::Type::NUM, RPCArg::Default{DEFAULT_WALLET_TX_VERSION}, "Transaction version"},
                   +  - ]
    1741   [ +  -  +  -  :        3210 :                     {"psbt_version", RPCArg::Type::NUM, RPCArg::Default(2), "The PSBT version number to use."},
                   +  - ]
    1742                 :             :                 },
    1743         [ +  - ]:        2140 :                 RPCResult{
    1744   [ +  -  +  - ]:        2140 :                     RPCResult::Type::OBJ, "", "",
    1745                 :             :                     {
    1746   [ +  -  +  - ]:        2140 :                         {RPCResult::Type::STR, "psbt", "The resulting raw transaction (base64-encoded string)"},
    1747   [ +  -  +  - ]:        2140 :                         {RPCResult::Type::STR_AMOUNT, "fee", "Fee in " + CURRENCY_UNIT + " the resulting transaction pays"},
    1748   [ +  -  +  - ]:        2140 :                         {RPCResult::Type::NUM, "changepos", "The position of the added change output, or -1"},
    1749                 :             :                     }
    1750   [ +  -  +  -  :        8560 :                                 },
             +  +  -  - ]
    1751                 :        1070 :                                 RPCExamples{
    1752                 :             :                             "\nCreate a PSBT with automatically picked inputs that sends 0.5 BTC to an address and has a fee rate of 2 sat/vB:\n"
    1753   [ +  -  +  -  :        3210 :                             + HelpExampleCli("walletcreatefundedpsbt", "\"[]\" \"[{\\\"" + EXAMPLE_ADDRESS[0] + "\\\":0.5}]\" 0 \"{\\\"add_inputs\\\":true,\\\"fee_rate\\\":2}\"")
             +  -  +  - ]
    1754                 :        1070 :                             + "\nCreate the same PSBT as the above one instead using named arguments:\n"
    1755   [ +  -  +  -  :        5350 :                             + HelpExampleCli("-named walletcreatefundedpsbt", "outputs=\"[{\\\"" + EXAMPLE_ADDRESS[0] + "\\\":0.5}]\" add_inputs=true fee_rate=2")
             +  -  +  - ]
    1756         [ +  - ]:        1070 :                                 },
    1757                 :        1070 :         [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
    1758                 :             : {
    1759                 :         231 :     std::shared_ptr<CWallet> const pwallet = GetWalletForJSONRPCRequest(request);
    1760         [ -  + ]:         231 :     if (!pwallet) return UniValue::VNULL;
    1761                 :             : 
    1762         [ +  - ]:         231 :     CWallet& wallet{*pwallet};
    1763                 :             :     // Make sure the results are valid at least up to the most recent block
    1764                 :             :     // the user could have gotten from another RPC command prior to now
    1765         [ +  - ]:         231 :     wallet.BlockUntilSyncedToCurrentChain();
    1766                 :             : 
    1767   [ +  -  +  +  :         231 :     UniValue options{request.params[3].isNull() ? UniValue::VOBJ : request.params[3]};
             +  -  +  - ]
    1768                 :             : 
    1769         [ +  - ]:         231 :     CCoinControl coin_control;
    1770         [ +  - ]:         231 :     coin_control.m_version = self.Arg<uint32_t>("version");
    1771                 :             : 
    1772   [ +  -  +  - ]:         231 :     const UniValue &replaceable_arg = options["replaceable"];
    1773   [ +  +  +  - ]:         231 :     const bool rbf{replaceable_arg.isNull() ? wallet.m_signal_rbf : replaceable_arg.get_bool()};
    1774   [ +  -  +  -  :         231 :     CMutableTransaction rawTx = ConstructTransaction(request.params[0], request.params[1], request.params[2], rbf, coin_control.m_version);
             +  -  +  - ]
    1775                 :         231 :     UniValue outputs(UniValue::VOBJ);
    1776   [ +  -  +  - ]:         231 :     outputs = NormalizeOutputs(request.params[1]);
    1777                 :         231 :     std::vector<CRecipient> recipients = CreateRecipients(
    1778         [ +  - ]:         462 :             ParseOutputs(outputs),
    1779   [ +  -  +  -  :         462 :             InterpretSubtractFeeFromOutputInstructions(options["subtractFeeFromOutputs"], outputs.getKeys())
             +  -  +  - ]
    1780         [ +  - ]:         231 :     );
    1781                 :             :     // Automatically select coins, unless at least one is manually selected. Can
    1782                 :             :     // be overridden by options.add_inputs.
    1783         [ -  + ]:         231 :     coin_control.m_allow_other_inputs = rawTx.vin.size() == 0;
    1784   [ +  -  +  + ]:         231 :     SetOptionsInputWeights(request.params[0], options);
    1785                 :             :     // Clear tx.vout since it is not meant to be used now that we are passing outputs directly.
    1786                 :             :     // This sets us up for a future PR to completely remove tx from the function signature in favor of passing inputs directly
    1787                 :         230 :     rawTx.vout.clear();
    1788         [ +  + ]:         230 :     auto txr = FundTransaction(wallet, rawTx, recipients, options, coin_control, /*override_min_fee=*/true);
    1789                 :             : 
    1790                 :             :     // Make a blank psbt
    1791                 :         162 :     uint32_t psbt_version = 2;
    1792   [ +  -  +  + ]:         162 :     if (!request.params[6].isNull()) {
    1793   [ +  -  +  - ]:           5 :         psbt_version = request.params[6].getInt<int>();
    1794                 :             :     }
    1795         [ +  + ]:         162 :     if (psbt_version != 2 && psbt_version != 0) {
    1796   [ +  -  +  - ]:           2 :         throw JSONRPCError(RPC_INVALID_PARAMETER, "The PSBT version can only be 2 or 0");
    1797                 :             :     }
    1798                 :             : 
    1799   [ +  -  +  - ]:         161 :     PartiallySignedTransaction psbtx(CMutableTransaction(*txr.tx), psbt_version);
    1800                 :             : 
    1801                 :             :     // Fill transaction with out data but don't sign
    1802   [ +  -  +  +  :         161 :     bool bip32derivs = request.params[4].isNull() ? true : request.params[4].get_bool();
             +  -  +  - ]
    1803                 :         161 :     bool complete = true;
    1804         [ +  - ]:         161 :     const auto err{wallet.FillPSBT(psbtx, {.sign = false, .bip32_derivs = bip32derivs}, complete)};
    1805         [ -  + ]:         161 :     if (err) {
    1806         [ #  # ]:           0 :         throw JSONRPCPSBTError(*err);
    1807                 :             :     }
    1808                 :             : 
    1809                 :             :     // Serialize the PSBT
    1810                 :         161 :     DataStream ssTx{};
    1811         [ +  - ]:         161 :     ssTx << psbtx;
    1812                 :             : 
    1813                 :         161 :     UniValue result(UniValue::VOBJ);
    1814   [ +  -  +  -  :         483 :     result.pushKV("psbt", EncodeBase64(ssTx.str()));
          +  -  +  -  +  
                      - ]
    1815   [ +  -  +  -  :         322 :     result.pushKV("fee", ValueFromAmount(txr.fee));
                   +  - ]
    1816   [ +  +  +  -  :         322 :     result.pushKV("changepos", txr.change_pos ? (int)*txr.change_pos : -1);
             +  -  +  - ]
    1817                 :         161 :     return result;
    1818         [ +  - ]:         924 : },
    1819   [ +  -  +  -  :       21400 :     };
          +  -  +  -  +  
          +  +  +  +  +  
          -  -  -  -  -  
                      - ]
    1820   [ +  -  +  -  :       59920 : }
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  -  
          -  -  -  -  -  
                   -  - ]
    1821                 :             : } // namespace wallet
        

Generated by: LCOV version 2.0-1