/Users/eugenesiegel/btc/bitcoin/src/wallet/rpc/coins.cpp
| Line | Count | Source (jump to first uncovered line) | 
| 1 |  | // Copyright (c) 2011-2022 The Bitcoin Core developers | 
| 2 |  | // Distributed under the MIT software license, see the accompanying | 
| 3 |  | // file COPYING or http://www.opensource.org/licenses/mit-license.php. | 
| 4 |  |  | 
| 5 |  | #include <core_io.h> | 
| 6 |  | #include <hash.h> | 
| 7 |  | #include <key_io.h> | 
| 8 |  | #include <rpc/util.h> | 
| 9 |  | #include <script/script.h> | 
| 10 |  | #include <util/moneystr.h> | 
| 11 |  | #include <wallet/coincontrol.h> | 
| 12 |  | #include <wallet/receive.h> | 
| 13 |  | #include <wallet/rpc/util.h> | 
| 14 |  | #include <wallet/spend.h> | 
| 15 |  | #include <wallet/wallet.h> | 
| 16 |  |  | 
| 17 |  | #include <univalue.h> | 
| 18 |  |  | 
| 19 |  |  | 
| 20 |  | namespace wallet { | 
| 21 |  | static CAmount GetReceived(const CWallet& wallet, const UniValue& params, bool by_label) EXCLUSIVE_LOCKS_REQUIRED(wallet.cs_wallet) | 
| 22 | 0 | { | 
| 23 | 0 |     std::vector<CTxDestination> addresses; | 
| 24 | 0 |     if (by_label) { | 
| 25 |  |         // Get the set of addresses assigned to label | 
| 26 | 0 |         addresses = wallet.ListAddrBookAddresses(CWallet::AddrBookFilter{LabelFromValue(params[0])}); | 
| 27 | 0 |         if (addresses.empty()) throw JSONRPCError(RPC_WALLET_ERROR, "Label not found in wallet"); | 
| 28 | 0 |     } else { | 
| 29 |  |         // Get the address | 
| 30 | 0 |         CTxDestination dest = DecodeDestination(params[0].get_str()); | 
| 31 | 0 |         if (!IsValidDestination(dest)) { | 
| 32 | 0 |             throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Invalid Bitcoin address"); | 
| 33 | 0 |         } | 
| 34 | 0 |         addresses.emplace_back(dest); | 
| 35 | 0 |     } | 
| 36 |  |  | 
| 37 |  |     // Filter by own scripts only | 
| 38 | 0 |     std::set<CScript> output_scripts; | 
| 39 | 0 |     for (const auto& address : addresses) { | 
| 40 | 0 |         auto output_script{GetScriptForDestination(address)}; | 
| 41 | 0 |         if (wallet.IsMine(output_script)) { | 
| 42 | 0 |             output_scripts.insert(output_script); | 
| 43 | 0 |         } | 
| 44 | 0 |     } | 
| 45 |  | 
 | 
| 46 | 0 |     if (output_scripts.empty()) { | 
| 47 | 0 |         throw JSONRPCError(RPC_WALLET_ERROR, "Address not found in wallet"); | 
| 48 | 0 |     } | 
| 49 |  |  | 
| 50 |  |     // Minimum confirmations | 
| 51 | 0 |     int min_depth = 1; | 
| 52 | 0 |     if (!params[1].isNull()) | 
| 53 | 0 |         min_depth = params[1].getInt<int>(); | 
| 54 |  | 
 | 
| 55 | 0 |     const bool include_immature_coinbase{params[2].isNull() ? false : params[2].get_bool()}; | 
| 56 |  |  | 
| 57 |  |     // Tally | 
| 58 | 0 |     CAmount amount = 0; | 
| 59 | 0 |     for (const std::pair<const uint256, CWalletTx>& wtx_pair : wallet.mapWallet) { | 
| 60 | 0 |         const CWalletTx& wtx = wtx_pair.second; | 
| 61 | 0 |         int depth{wallet.GetTxDepthInMainChain(wtx)}; | 
| 62 | 0 |         if (depth < min_depth | 
| 63 |  |             // Coinbase with less than 1 confirmation is no longer in the main chain | 
| 64 | 0 |             || (wtx.IsCoinBase() && (depth < 1)) | 
| 65 | 0 |             || (wallet.IsTxImmatureCoinBase(wtx) && !include_immature_coinbase)) | 
| 66 | 0 |         { | 
| 67 | 0 |             continue; | 
| 68 | 0 |         } | 
| 69 |  |  | 
| 70 | 0 |         for (const CTxOut& txout : wtx.tx->vout) { | 
| 71 | 0 |             if (output_scripts.count(txout.scriptPubKey) > 0) { | 
| 72 | 0 |                 amount += txout.nValue; | 
| 73 | 0 |             } | 
| 74 | 0 |         } | 
| 75 | 0 |     } | 
| 76 |  | 
 | 
| 77 | 0 |     return amount; | 
| 78 | 0 | } | 
| 79 |  |  | 
| 80 |  |  | 
| 81 |  | RPCHelpMan getreceivedbyaddress() | 
| 82 | 0 | { | 
| 83 | 0 |     return RPCHelpMan{"getreceivedbyaddress", | 
| 84 | 0 |                 "\nReturns the total amount received by the given address in transactions with at least minconf confirmations.\n", | 
| 85 | 0 |                 { | 
| 86 | 0 |                     {"address", RPCArg::Type::STR, RPCArg::Optional::NO, "The bitcoin address for transactions."}, | 
| 87 | 0 |                     {"minconf", RPCArg::Type::NUM, RPCArg::Default{1}, "Only include transactions confirmed at least this many times."}, | 
| 88 | 0 |                     {"include_immature_coinbase", RPCArg::Type::BOOL, RPCArg::Default{false}, "Include immature coinbase transactions."}, | 
| 89 | 0 |                 }, | 
| 90 | 0 |                 RPCResult{ | 
| 91 | 0 |                     RPCResult::Type::STR_AMOUNT, "amount", "The total amount in " + CURRENCY_UNIT + " received at this address." | 
| 92 | 0 |                 }, | 
| 93 | 0 |                 RPCExamples{ | 
| 94 | 0 |             "\nThe amount from transactions with at least 1 confirmation\n" | 
| 95 | 0 |             + HelpExampleCli("getreceivedbyaddress", "\"" + EXAMPLE_ADDRESS[0] + "\"") + | 
| 96 | 0 |             "\nThe amount including unconfirmed transactions, zero confirmations\n" | 
| 97 | 0 |             + HelpExampleCli("getreceivedbyaddress", "\"" + EXAMPLE_ADDRESS[0] + "\" 0") + | 
| 98 | 0 |             "\nThe amount with at least 6 confirmations\n" | 
| 99 | 0 |             + HelpExampleCli("getreceivedbyaddress", "\"" + EXAMPLE_ADDRESS[0] + "\" 6") + | 
| 100 | 0 |             "\nThe amount with at least 6 confirmations including immature coinbase outputs\n" | 
| 101 | 0 |             + HelpExampleCli("getreceivedbyaddress", "\"" + EXAMPLE_ADDRESS[0] + "\" 6 true") + | 
| 102 | 0 |             "\nAs a JSON-RPC call\n" | 
| 103 | 0 |             + HelpExampleRpc("getreceivedbyaddress", "\"" + EXAMPLE_ADDRESS[0] + "\", 6") | 
| 104 | 0 |                 }, | 
| 105 | 0 |         [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue | 
| 106 | 0 | { | 
| 107 | 0 |     const std::shared_ptr<const CWallet> pwallet = GetWalletForJSONRPCRequest(request); | 
| 108 | 0 |     if (!pwallet) return UniValue::VNULL; | 
| 109 |  |  | 
| 110 |  |     // Make sure the results are valid at least up to the most recent block | 
| 111 |  |     // the user could have gotten from another RPC command prior to now | 
| 112 | 0 |     pwallet->BlockUntilSyncedToCurrentChain(); | 
| 113 |  | 
 | 
| 114 | 0 |     LOCK(pwallet->cs_wallet); | Line | Count | Source |  | 257 | 0 | #define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__) | Line | Count | Source |  | 11 | 0 | #define UNIQUE_NAME(name) PASTE2(name, __COUNTER__) | Line | Count | Source |  | 9 | 0 | #define PASTE2(x, y) PASTE(x, y) | Line | Count | Source |  | 8 | 0 | #define PASTE(x, y) x ## y | 
 | 
 | 
 | 
 | 
| 115 |  | 
 | 
| 116 | 0 |     return ValueFromAmount(GetReceived(*pwallet, request.params, /*by_label=*/false)); | 
| 117 | 0 | }, | 
| 118 | 0 |     }; | 
| 119 | 0 | } | 
| 120 |  |  | 
| 121 |  |  | 
| 122 |  | RPCHelpMan getreceivedbylabel() | 
| 123 | 0 | { | 
| 124 | 0 |     return RPCHelpMan{"getreceivedbylabel", | 
| 125 | 0 |                 "\nReturns the total amount received by addresses with <label> in transactions with at least [minconf] confirmations.\n", | 
| 126 | 0 |                 { | 
| 127 | 0 |                     {"label", RPCArg::Type::STR, RPCArg::Optional::NO, "The selected label, may be the default label using \"\"."}, | 
| 128 | 0 |                     {"minconf", RPCArg::Type::NUM, RPCArg::Default{1}, "Only include transactions confirmed at least this many times."}, | 
| 129 | 0 |                     {"include_immature_coinbase", RPCArg::Type::BOOL, RPCArg::Default{false}, "Include immature coinbase transactions."}, | 
| 130 | 0 |                 }, | 
| 131 | 0 |                 RPCResult{ | 
| 132 | 0 |                     RPCResult::Type::STR_AMOUNT, "amount", "The total amount in " + CURRENCY_UNIT + " received for this label." | 
| 133 | 0 |                 }, | 
| 134 | 0 |                 RPCExamples{ | 
| 135 | 0 |             "\nAmount received by the default label with at least 1 confirmation\n" | 
| 136 | 0 |             + HelpExampleCli("getreceivedbylabel", "\"\"") + | 
| 137 | 0 |             "\nAmount received at the tabby label including unconfirmed amounts with zero confirmations\n" | 
| 138 | 0 |             + HelpExampleCli("getreceivedbylabel", "\"tabby\" 0") + | 
| 139 | 0 |             "\nThe amount with at least 6 confirmations\n" | 
| 140 | 0 |             + HelpExampleCli("getreceivedbylabel", "\"tabby\" 6") + | 
| 141 | 0 |             "\nThe amount with at least 6 confirmations including immature coinbase outputs\n" | 
| 142 | 0 |             + HelpExampleCli("getreceivedbylabel", "\"tabby\" 6 true") + | 
| 143 | 0 |             "\nAs a JSON-RPC call\n" | 
| 144 | 0 |             + HelpExampleRpc("getreceivedbylabel", "\"tabby\", 6, true") | 
| 145 | 0 |                 }, | 
| 146 | 0 |         [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue | 
| 147 | 0 | { | 
| 148 | 0 |     const std::shared_ptr<const CWallet> pwallet = GetWalletForJSONRPCRequest(request); | 
| 149 | 0 |     if (!pwallet) return UniValue::VNULL; | 
| 150 |  |  | 
| 151 |  |     // Make sure the results are valid at least up to the most recent block | 
| 152 |  |     // the user could have gotten from another RPC command prior to now | 
| 153 | 0 |     pwallet->BlockUntilSyncedToCurrentChain(); | 
| 154 |  | 
 | 
| 155 | 0 |     LOCK(pwallet->cs_wallet); | Line | Count | Source |  | 257 | 0 | #define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__) | Line | Count | Source |  | 11 | 0 | #define UNIQUE_NAME(name) PASTE2(name, __COUNTER__) | Line | Count | Source |  | 9 | 0 | #define PASTE2(x, y) PASTE(x, y) | Line | Count | Source |  | 8 | 0 | #define PASTE(x, y) x ## y | 
 | 
 | 
 | 
 | 
| 156 |  | 
 | 
| 157 | 0 |     return ValueFromAmount(GetReceived(*pwallet, request.params, /*by_label=*/true)); | 
| 158 | 0 | }, | 
| 159 | 0 |     }; | 
| 160 | 0 | } | 
| 161 |  |  | 
| 162 |  |  | 
| 163 |  | RPCHelpMan getbalance() | 
| 164 | 0 | { | 
| 165 | 0 |     return RPCHelpMan{"getbalance", | 
| 166 | 0 |                 "\nReturns the total available balance.\n" | 
| 167 | 0 |                 "The available balance is what the wallet considers currently spendable, and is\n" | 
| 168 | 0 |                 "thus affected by options which limit spendability such as -spendzeroconfchange.\n", | 
| 169 | 0 |                 { | 
| 170 | 0 |                     {"dummy", RPCArg::Type::STR, RPCArg::Optional::OMITTED, "Remains for backward compatibility. Must be excluded or set to \"*\"."}, | 
| 171 | 0 |                     {"minconf", RPCArg::Type::NUM, RPCArg::Default{0}, "Only include transactions confirmed at least this many times."}, | 
| 172 | 0 |                     {"include_watchonly", RPCArg::Type::BOOL, RPCArg::DefaultHint{"true for watch-only wallets, otherwise false"}, "Also include balance in watch-only addresses (see 'importaddress')"}, | 
| 173 | 0 |                     {"avoid_reuse", RPCArg::Type::BOOL, RPCArg::Default{true}, "(only available if avoid_reuse wallet flag is set) Do not include balance in dirty outputs; addresses are considered dirty if they have previously been used in a transaction."}, | 
| 174 | 0 |                 }, | 
| 175 | 0 |                 RPCResult{ | 
| 176 | 0 |                     RPCResult::Type::STR_AMOUNT, "amount", "The total amount in " + CURRENCY_UNIT + " received for this wallet." | 
| 177 | 0 |                 }, | 
| 178 | 0 |                 RPCExamples{ | 
| 179 | 0 |             "\nThe total amount in the wallet with 0 or more confirmations\n" | 
| 180 | 0 |             + HelpExampleCli("getbalance", "") + | 
| 181 | 0 |             "\nThe total amount in the wallet with at least 6 confirmations\n" | 
| 182 | 0 |             + HelpExampleCli("getbalance", "\"*\" 6") + | 
| 183 | 0 |             "\nAs a JSON-RPC call\n" | 
| 184 | 0 |             + HelpExampleRpc("getbalance", "\"*\", 6") | 
| 185 | 0 |                 }, | 
| 186 | 0 |         [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue | 
| 187 | 0 | { | 
| 188 | 0 |     const std::shared_ptr<const CWallet> pwallet = GetWalletForJSONRPCRequest(request); | 
| 189 | 0 |     if (!pwallet) return UniValue::VNULL; | 
| 190 |  |  | 
| 191 |  |     // Make sure the results are valid at least up to the most recent block | 
| 192 |  |     // the user could have gotten from another RPC command prior to now | 
| 193 | 0 |     pwallet->BlockUntilSyncedToCurrentChain(); | 
| 194 |  | 
 | 
| 195 | 0 |     LOCK(pwallet->cs_wallet); | Line | Count | Source |  | 257 | 0 | #define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__) | Line | Count | Source |  | 11 | 0 | #define UNIQUE_NAME(name) PASTE2(name, __COUNTER__) | Line | Count | Source |  | 9 | 0 | #define PASTE2(x, y) PASTE(x, y) | Line | Count | Source |  | 8 | 0 | #define PASTE(x, y) x ## y | 
 | 
 | 
 | 
 | 
| 196 |  | 
 | 
| 197 | 0 |     const auto dummy_value{self.MaybeArg<std::string>("dummy")}; | 
| 198 | 0 |     if (dummy_value && *dummy_value != "*") { | 
| 199 | 0 |         throw JSONRPCError(RPC_METHOD_DEPRECATED, "dummy first argument must be excluded or set to \"*\"."); | 
| 200 | 0 |     } | 
| 201 |  |  | 
| 202 | 0 |     const auto min_depth{self.Arg<int>("minconf")}; | 
| 203 |  | 
 | 
| 204 | 0 |     bool include_watchonly = ParseIncludeWatchonly(request.params[2], *pwallet); | 
| 205 |  | 
 | 
| 206 | 0 |     bool avoid_reuse = GetAvoidReuseFlag(*pwallet, request.params[3]); | 
| 207 |  | 
 | 
| 208 | 0 |     const auto bal = GetBalance(*pwallet, min_depth, avoid_reuse); | 
| 209 |  | 
 | 
| 210 | 0 |     return ValueFromAmount(bal.m_mine_trusted + (include_watchonly ? bal.m_watchonly_trusted : 0)); | 
| 211 | 0 | }, | 
| 212 | 0 |     }; | 
| 213 | 0 | } | 
| 214 |  |  | 
| 215 |  | RPCHelpMan getunconfirmedbalance() | 
| 216 | 0 | { | 
| 217 | 0 |     return RPCHelpMan{"getunconfirmedbalance", | 
| 218 | 0 |                 "DEPRECATED\nIdentical to getbalances().mine.untrusted_pending\n", | 
| 219 | 0 |                 {}, | 
| 220 | 0 |                 RPCResult{RPCResult::Type::NUM, "", "The balance"}, | 
| 221 | 0 |                 RPCExamples{""}, | 
| 222 | 0 |         [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue | 
| 223 | 0 | { | 
| 224 | 0 |     const std::shared_ptr<const CWallet> pwallet = GetWalletForJSONRPCRequest(request); | 
| 225 | 0 |     if (!pwallet) return UniValue::VNULL; | 
| 226 |  |  | 
| 227 |  |     // Make sure the results are valid at least up to the most recent block | 
| 228 |  |     // the user could have gotten from another RPC command prior to now | 
| 229 | 0 |     pwallet->BlockUntilSyncedToCurrentChain(); | 
| 230 |  | 
 | 
| 231 | 0 |     LOCK(pwallet->cs_wallet); | Line | Count | Source |  | 257 | 0 | #define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__) | Line | Count | Source |  | 11 | 0 | #define UNIQUE_NAME(name) PASTE2(name, __COUNTER__) | Line | Count | Source |  | 9 | 0 | #define PASTE2(x, y) PASTE(x, y) | Line | Count | Source |  | 8 | 0 | #define PASTE(x, y) x ## y | 
 | 
 | 
 | 
 | 
| 232 |  | 
 | 
| 233 | 0 |     return ValueFromAmount(GetBalance(*pwallet).m_mine_untrusted_pending); | 
| 234 | 0 | }, | 
| 235 | 0 |     }; | 
| 236 | 0 | } | 
| 237 |  |  | 
| 238 |  | RPCHelpMan lockunspent() | 
| 239 | 0 | { | 
| 240 | 0 |     return RPCHelpMan{"lockunspent", | 
| 241 | 0 |                 "\nUpdates list of temporarily unspendable outputs.\n" | 
| 242 | 0 |                 "Temporarily lock (unlock=false) or unlock (unlock=true) specified transaction outputs.\n" | 
| 243 | 0 |                 "If no transaction outputs are specified when unlocking then all current locked transaction outputs are unlocked.\n" | 
| 244 | 0 |                 "A locked transaction output will not be chosen by automatic coin selection, when spending bitcoins.\n" | 
| 245 | 0 |                 "Manually selected coins are automatically unlocked.\n" | 
| 246 | 0 |                 "Locks are stored in memory only, unless persistent=true, in which case they will be written to the\n" | 
| 247 | 0 |                 "wallet database and loaded on node start. Unwritten (persistent=false) locks are always cleared\n" | 
| 248 | 0 |                 "(by virtue of process exit) when a node stops or fails. Unlocking will clear both persistent and not.\n" | 
| 249 | 0 |                 "Also see the listunspent call\n", | 
| 250 | 0 |                 { | 
| 251 | 0 |                     {"unlock", RPCArg::Type::BOOL, RPCArg::Optional::NO, "Whether to unlock (true) or lock (false) the specified transactions"}, | 
| 252 | 0 |                     {"transactions", RPCArg::Type::ARR, RPCArg::Default{UniValue::VARR}, "The transaction outputs and within each, the txid (string) vout (numeric).", | 
| 253 | 0 |                         { | 
| 254 | 0 |                             {"", RPCArg::Type::OBJ, RPCArg::Optional::OMITTED, "", | 
| 255 | 0 |                                 { | 
| 256 | 0 |                                     {"txid", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "The transaction id"}, | 
| 257 | 0 |                                     {"vout", RPCArg::Type::NUM, RPCArg::Optional::NO, "The output number"}, | 
| 258 | 0 |                                 }, | 
| 259 | 0 |                             }, | 
| 260 | 0 |                         }, | 
| 261 | 0 |                     }, | 
| 262 | 0 |                     {"persistent", RPCArg::Type::BOOL, RPCArg::Default{false}, "Whether to write/erase this lock in the wallet database, or keep the change in memory only. Ignored for unlocking."}, | 
| 263 | 0 |                 }, | 
| 264 | 0 |                 RPCResult{ | 
| 265 | 0 |                     RPCResult::Type::BOOL, "", "Whether the command was successful or not" | 
| 266 | 0 |                 }, | 
| 267 | 0 |                 RPCExamples{ | 
| 268 | 0 |             "\nList the unspent transactions\n" | 
| 269 | 0 |             + HelpExampleCli("listunspent", "") + | 
| 270 | 0 |             "\nLock an unspent transaction\n" | 
| 271 | 0 |             + HelpExampleCli("lockunspent", "false \"[{\\\"txid\\\":\\\"a08e6907dbbd3d809776dbfc5d82e371b764ed838b5655e72f463568df1aadf0\\\",\\\"vout\\\":1}]\"") + | 
| 272 | 0 |             "\nList the locked transactions\n" | 
| 273 | 0 |             + HelpExampleCli("listlockunspent", "") + | 
| 274 | 0 |             "\nUnlock the transaction again\n" | 
| 275 | 0 |             + HelpExampleCli("lockunspent", "true \"[{\\\"txid\\\":\\\"a08e6907dbbd3d809776dbfc5d82e371b764ed838b5655e72f463568df1aadf0\\\",\\\"vout\\\":1}]\"") + | 
| 276 | 0 |             "\nLock the transaction persistently in the wallet database\n" | 
| 277 | 0 |             + HelpExampleCli("lockunspent", "false \"[{\\\"txid\\\":\\\"a08e6907dbbd3d809776dbfc5d82e371b764ed838b5655e72f463568df1aadf0\\\",\\\"vout\\\":1}]\" true") + | 
| 278 | 0 |             "\nAs a JSON-RPC call\n" | 
| 279 | 0 |             + HelpExampleRpc("lockunspent", "false, \"[{\\\"txid\\\":\\\"a08e6907dbbd3d809776dbfc5d82e371b764ed838b5655e72f463568df1aadf0\\\",\\\"vout\\\":1}]\"") | 
| 280 | 0 |                 }, | 
| 281 | 0 |         [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue | 
| 282 | 0 | { | 
| 283 | 0 |     std::shared_ptr<CWallet> const pwallet = GetWalletForJSONRPCRequest(request); | 
| 284 | 0 |     if (!pwallet) return UniValue::VNULL; | 
| 285 |  |  | 
| 286 |  |     // Make sure the results are valid at least up to the most recent block | 
| 287 |  |     // the user could have gotten from another RPC command prior to now | 
| 288 | 0 |     pwallet->BlockUntilSyncedToCurrentChain(); | 
| 289 |  | 
 | 
| 290 | 0 |     LOCK(pwallet->cs_wallet); | Line | Count | Source |  | 257 | 0 | #define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__) | Line | Count | Source |  | 11 | 0 | #define UNIQUE_NAME(name) PASTE2(name, __COUNTER__) | Line | Count | Source |  | 9 | 0 | #define PASTE2(x, y) PASTE(x, y) | Line | Count | Source |  | 8 | 0 | #define PASTE(x, y) x ## y | 
 | 
 | 
 | 
 | 
| 291 |  | 
 | 
| 292 | 0 |     bool fUnlock = request.params[0].get_bool(); | 
| 293 |  | 
 | 
| 294 | 0 |     const bool persistent{request.params[2].isNull() ? false : request.params[2].get_bool()}; | 
| 295 |  | 
 | 
| 296 | 0 |     if (request.params[1].isNull()) { | 
| 297 | 0 |         if (fUnlock) { | 
| 298 | 0 |             if (!pwallet->UnlockAllCoins()) | 
| 299 | 0 |                 throw JSONRPCError(RPC_WALLET_ERROR, "Unlocking coins failed"); | 
| 300 | 0 |         } | 
| 301 | 0 |         return true; | 
| 302 | 0 |     } | 
| 303 |  |  | 
| 304 | 0 |     const UniValue& output_params = request.params[1].get_array(); | 
| 305 |  |  | 
| 306 |  |     // Create and validate the COutPoints first. | 
| 307 |  | 
 | 
| 308 | 0 |     std::vector<COutPoint> outputs; | 
| 309 | 0 |     outputs.reserve(output_params.size()); | 
| 310 |  | 
 | 
| 311 | 0 |     for (unsigned int idx = 0; idx < output_params.size(); idx++) { | 
| 312 | 0 |         const UniValue& o = output_params[idx].get_obj(); | 
| 313 |  | 
 | 
| 314 | 0 |         RPCTypeCheckObj(o, | 
| 315 | 0 |             { | 
| 316 | 0 |                 {"txid", UniValueType(UniValue::VSTR)}, | 
| 317 | 0 |                 {"vout", UniValueType(UniValue::VNUM)}, | 
| 318 | 0 |             }); | 
| 319 |  | 
 | 
| 320 | 0 |         const Txid txid = Txid::FromUint256(ParseHashO(o, "txid")); | 
| 321 | 0 |         const int nOutput = o.find_value("vout").getInt<int>(); | 
| 322 | 0 |         if (nOutput < 0) { | 
| 323 | 0 |             throw JSONRPCError(RPC_INVALID_PARAMETER, "Invalid parameter, vout cannot be negative"); | 
| 324 | 0 |         } | 
| 325 |  |  | 
| 326 | 0 |         const COutPoint outpt(txid, nOutput); | 
| 327 |  | 
 | 
| 328 | 0 |         const auto it = pwallet->mapWallet.find(outpt.hash); | 
| 329 | 0 |         if (it == pwallet->mapWallet.end()) { | 
| 330 | 0 |             throw JSONRPCError(RPC_INVALID_PARAMETER, "Invalid parameter, unknown transaction"); | 
| 331 | 0 |         } | 
| 332 |  |  | 
| 333 | 0 |         const CWalletTx& trans = it->second; | 
| 334 |  | 
 | 
| 335 | 0 |         if (outpt.n >= trans.tx->vout.size()) { | 
| 336 | 0 |             throw JSONRPCError(RPC_INVALID_PARAMETER, "Invalid parameter, vout index out of bounds"); | 
| 337 | 0 |         } | 
| 338 |  |  | 
| 339 | 0 |         if (pwallet->IsSpent(outpt)) { | 
| 340 | 0 |             throw JSONRPCError(RPC_INVALID_PARAMETER, "Invalid parameter, expected unspent output"); | 
| 341 | 0 |         } | 
| 342 |  |  | 
| 343 | 0 |         const bool is_locked = pwallet->IsLockedCoin(outpt); | 
| 344 |  | 
 | 
| 345 | 0 |         if (fUnlock && !is_locked) { | 
| 346 | 0 |             throw JSONRPCError(RPC_INVALID_PARAMETER, "Invalid parameter, expected locked output"); | 
| 347 | 0 |         } | 
| 348 |  |  | 
| 349 | 0 |         if (!fUnlock && is_locked && !persistent) { | 
| 350 | 0 |             throw JSONRPCError(RPC_INVALID_PARAMETER, "Invalid parameter, output already locked"); | 
| 351 | 0 |         } | 
| 352 |  |  | 
| 353 | 0 |         outputs.push_back(outpt); | 
| 354 | 0 |     } | 
| 355 |  |  | 
| 356 | 0 |     std::unique_ptr<WalletBatch> batch = nullptr; | 
| 357 |  |     // Unlock is always persistent | 
| 358 | 0 |     if (fUnlock || persistent) batch = std::make_unique<WalletBatch>(pwallet->GetDatabase()); | 
| 359 |  |  | 
| 360 |  |     // Atomically set (un)locked status for the outputs. | 
| 361 | 0 |     for (const COutPoint& outpt : outputs) { | 
| 362 | 0 |         if (fUnlock) { | 
| 363 | 0 |             if (!pwallet->UnlockCoin(outpt, batch.get())) throw JSONRPCError(RPC_WALLET_ERROR, "Unlocking coin failed"); | 
| 364 | 0 |         } else { | 
| 365 | 0 |             if (!pwallet->LockCoin(outpt, batch.get())) throw JSONRPCError(RPC_WALLET_ERROR, "Locking coin failed"); | 
| 366 | 0 |         } | 
| 367 | 0 |     } | 
| 368 |  |  | 
| 369 | 0 |     return true; | 
| 370 | 0 | }, | 
| 371 | 0 |     }; | 
| 372 | 0 | } | 
| 373 |  |  | 
| 374 |  | RPCHelpMan listlockunspent() | 
| 375 | 0 | { | 
| 376 | 0 |     return RPCHelpMan{"listlockunspent", | 
| 377 | 0 |                 "\nReturns list of temporarily unspendable outputs.\n" | 
| 378 | 0 |                 "See the lockunspent call to lock and unlock transactions for spending.\n", | 
| 379 | 0 |                 {}, | 
| 380 | 0 |                 RPCResult{ | 
| 381 | 0 |                     RPCResult::Type::ARR, "", "", | 
| 382 | 0 |                     { | 
| 383 | 0 |                         {RPCResult::Type::OBJ, "", "", | 
| 384 | 0 |                         { | 
| 385 | 0 |                             {RPCResult::Type::STR_HEX, "txid", "The transaction id locked"}, | 
| 386 | 0 |                             {RPCResult::Type::NUM, "vout", "The vout value"}, | 
| 387 | 0 |                         }}, | 
| 388 | 0 |                     } | 
| 389 | 0 |                 }, | 
| 390 | 0 |                 RPCExamples{ | 
| 391 | 0 |             "\nList the unspent transactions\n" | 
| 392 | 0 |             + HelpExampleCli("listunspent", "") + | 
| 393 | 0 |             "\nLock an unspent transaction\n" | 
| 394 | 0 |             + HelpExampleCli("lockunspent", "false \"[{\\\"txid\\\":\\\"a08e6907dbbd3d809776dbfc5d82e371b764ed838b5655e72f463568df1aadf0\\\",\\\"vout\\\":1}]\"") + | 
| 395 | 0 |             "\nList the locked transactions\n" | 
| 396 | 0 |             + HelpExampleCli("listlockunspent", "") + | 
| 397 | 0 |             "\nUnlock the transaction again\n" | 
| 398 | 0 |             + HelpExampleCli("lockunspent", "true \"[{\\\"txid\\\":\\\"a08e6907dbbd3d809776dbfc5d82e371b764ed838b5655e72f463568df1aadf0\\\",\\\"vout\\\":1}]\"") + | 
| 399 | 0 |             "\nAs a JSON-RPC call\n" | 
| 400 | 0 |             + HelpExampleRpc("listlockunspent", "") | 
| 401 | 0 |                 }, | 
| 402 | 0 |         [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue | 
| 403 | 0 | { | 
| 404 | 0 |     const std::shared_ptr<const CWallet> pwallet = GetWalletForJSONRPCRequest(request); | 
| 405 | 0 |     if (!pwallet) return UniValue::VNULL; | 
| 406 |  |  | 
| 407 | 0 |     LOCK(pwallet->cs_wallet); | Line | Count | Source |  | 257 | 0 | #define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__) | Line | Count | Source |  | 11 | 0 | #define UNIQUE_NAME(name) PASTE2(name, __COUNTER__) | Line | Count | Source |  | 9 | 0 | #define PASTE2(x, y) PASTE(x, y) | Line | Count | Source |  | 8 | 0 | #define PASTE(x, y) x ## y | 
 | 
 | 
 | 
 | 
| 408 |  | 
 | 
| 409 | 0 |     std::vector<COutPoint> vOutpts; | 
| 410 | 0 |     pwallet->ListLockedCoins(vOutpts); | 
| 411 |  | 
 | 
| 412 | 0 |     UniValue ret(UniValue::VARR); | 
| 413 |  | 
 | 
| 414 | 0 |     for (const COutPoint& outpt : vOutpts) { | 
| 415 | 0 |         UniValue o(UniValue::VOBJ); | 
| 416 |  | 
 | 
| 417 | 0 |         o.pushKV("txid", outpt.hash.GetHex()); | 
| 418 | 0 |         o.pushKV("vout", (int)outpt.n); | 
| 419 | 0 |         ret.push_back(std::move(o)); | 
| 420 | 0 |     } | 
| 421 |  | 
 | 
| 422 | 0 |     return ret; | 
| 423 | 0 | }, | 
| 424 | 0 |     }; | 
| 425 | 0 | } | 
| 426 |  |  | 
| 427 |  | RPCHelpMan getbalances() | 
| 428 | 0 | { | 
| 429 | 0 |     return RPCHelpMan{ | 
| 430 | 0 |         "getbalances", | 
| 431 | 0 |         "Returns an object with all balances in " + CURRENCY_UNIT + ".\n", | 
| 432 | 0 |         {}, | 
| 433 | 0 |         RPCResult{ | 
| 434 | 0 |             RPCResult::Type::OBJ, "", "", | 
| 435 | 0 |             { | 
| 436 | 0 |                 {RPCResult::Type::OBJ, "mine", "balances from outputs that the wallet can sign", | 
| 437 | 0 |                 { | 
| 438 | 0 |                     {RPCResult::Type::STR_AMOUNT, "trusted", "trusted balance (outputs created by the wallet or confirmed outputs)"}, | 
| 439 | 0 |                     {RPCResult::Type::STR_AMOUNT, "untrusted_pending", "untrusted pending balance (outputs created by others that are in the mempool)"}, | 
| 440 | 0 |                     {RPCResult::Type::STR_AMOUNT, "immature", "balance from immature coinbase outputs"}, | 
| 441 | 0 |                     {RPCResult::Type::STR_AMOUNT, "used", /*optional=*/true, "(only present if avoid_reuse is set) balance from coins sent to addresses that were previously spent from (potentially privacy violating)"}, | 
| 442 | 0 |                 }}, | 
| 443 | 0 |                 RESULT_LAST_PROCESSED_BLOCK, | 
| 444 | 0 |             } | 
| 445 | 0 |             }, | 
| 446 | 0 |         RPCExamples{ | 
| 447 | 0 |             HelpExampleCli("getbalances", "") + | 
| 448 | 0 |             HelpExampleRpc("getbalances", "")}, | 
| 449 | 0 |         [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue | 
| 450 | 0 | { | 
| 451 | 0 |     const std::shared_ptr<const CWallet> rpc_wallet = GetWalletForJSONRPCRequest(request); | 
| 452 | 0 |     if (!rpc_wallet) return UniValue::VNULL; | 
| 453 | 0 |     const CWallet& wallet = *rpc_wallet; | 
| 454 |  |  | 
| 455 |  |     // Make sure the results are valid at least up to the most recent block | 
| 456 |  |     // the user could have gotten from another RPC command prior to now | 
| 457 | 0 |     wallet.BlockUntilSyncedToCurrentChain(); | 
| 458 |  | 
 | 
| 459 | 0 |     LOCK(wallet.cs_wallet); | Line | Count | Source |  | 257 | 0 | #define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__) | Line | Count | Source |  | 11 | 0 | #define UNIQUE_NAME(name) PASTE2(name, __COUNTER__) | Line | Count | Source |  | 9 | 0 | #define PASTE2(x, y) PASTE(x, y) | Line | Count | Source |  | 8 | 0 | #define PASTE(x, y) x ## y | 
 | 
 | 
 | 
 | 
| 460 |  | 
 | 
| 461 | 0 |     const auto bal = GetBalance(wallet); | 
| 462 | 0 |     UniValue balances{UniValue::VOBJ}; | 
| 463 | 0 |     { | 
| 464 | 0 |         UniValue balances_mine{UniValue::VOBJ}; | 
| 465 | 0 |         balances_mine.pushKV("trusted", ValueFromAmount(bal.m_mine_trusted)); | 
| 466 | 0 |         balances_mine.pushKV("untrusted_pending", ValueFromAmount(bal.m_mine_untrusted_pending)); | 
| 467 | 0 |         balances_mine.pushKV("immature", ValueFromAmount(bal.m_mine_immature)); | 
| 468 | 0 |         if (wallet.IsWalletFlagSet(WALLET_FLAG_AVOID_REUSE)) { | 
| 469 |  |             // If the AVOID_REUSE flag is set, bal has been set to just the un-reused address balance. Get | 
| 470 |  |             // the total balance, and then subtract bal to get the reused address balance. | 
| 471 | 0 |             const auto full_bal = GetBalance(wallet, 0, false); | 
| 472 | 0 |             balances_mine.pushKV("used", ValueFromAmount(full_bal.m_mine_trusted + full_bal.m_mine_untrusted_pending - bal.m_mine_trusted - bal.m_mine_untrusted_pending)); | 
| 473 | 0 |         } | 
| 474 | 0 |         balances.pushKV("mine", std::move(balances_mine)); | 
| 475 | 0 |     } | 
| 476 | 0 |     AppendLastProcessedBlock(balances, wallet); | 
| 477 | 0 |     return balances; | 
| 478 | 0 | }, | 
| 479 | 0 |     }; | 
| 480 | 0 | } | 
| 481 |  |  | 
| 482 |  | RPCHelpMan listunspent() | 
| 483 | 0 | { | 
| 484 | 0 |     return RPCHelpMan{ | 
| 485 | 0 |                 "listunspent", | 
| 486 | 0 |                 "\nReturns array of unspent transaction outputs\n" | 
| 487 | 0 |                 "with between minconf and maxconf (inclusive) confirmations.\n" | 
| 488 | 0 |                 "Optionally filter to only include txouts paid to specified addresses.\n", | 
| 489 | 0 |                 { | 
| 490 | 0 |                     {"minconf", RPCArg::Type::NUM, RPCArg::Default{1}, "The minimum confirmations to filter"}, | 
| 491 | 0 |                     {"maxconf", RPCArg::Type::NUM, RPCArg::Default{9999999}, "The maximum confirmations to filter"}, | 
| 492 | 0 |                     {"addresses", RPCArg::Type::ARR, RPCArg::Default{UniValue::VARR}, "The bitcoin addresses to filter", | 
| 493 | 0 |                         { | 
| 494 | 0 |                             {"address", RPCArg::Type::STR, RPCArg::Optional::OMITTED, "bitcoin address"}, | 
| 495 | 0 |                         }, | 
| 496 | 0 |                     }, | 
| 497 | 0 |                     {"include_unsafe", RPCArg::Type::BOOL, RPCArg::Default{true}, "Include outputs that are not safe to spend\n" | 
| 498 | 0 |                               "See description of \"safe\" attribute below."}, | 
| 499 | 0 |                     {"query_options", RPCArg::Type::OBJ_NAMED_PARAMS, RPCArg::Optional::OMITTED, "", | 
| 500 | 0 |                         { | 
| 501 | 0 |                             {"minimumAmount", RPCArg::Type::AMOUNT, RPCArg::Default{FormatMoney(0)}, "Minimum value of each UTXO in " + CURRENCY_UNIT + ""}, | 
| 502 | 0 |                             {"maximumAmount", RPCArg::Type::AMOUNT, RPCArg::DefaultHint{"unlimited"}, "Maximum value of each UTXO in " + CURRENCY_UNIT + ""}, | 
| 503 | 0 |                             {"maximumCount", RPCArg::Type::NUM, RPCArg::DefaultHint{"unlimited"}, "Maximum number of UTXOs"}, | 
| 504 | 0 |                             {"minimumSumAmount", RPCArg::Type::AMOUNT, RPCArg::DefaultHint{"unlimited"}, "Minimum sum value of all UTXOs in " + CURRENCY_UNIT + ""}, | 
| 505 | 0 |                             {"include_immature_coinbase", RPCArg::Type::BOOL, RPCArg::Default{false}, "Include immature coinbase UTXOs"} | 
| 506 | 0 |                         }, | 
| 507 | 0 |                         RPCArgOptions{.oneline_description="query_options"}}, | 
| 508 | 0 |                 }, | 
| 509 | 0 |                 RPCResult{ | 
| 510 | 0 |                     RPCResult::Type::ARR, "", "", | 
| 511 | 0 |                     { | 
| 512 | 0 |                         {RPCResult::Type::OBJ, "", "", | 
| 513 | 0 |                         { | 
| 514 | 0 |                             {RPCResult::Type::STR_HEX, "txid", "the transaction id"}, | 
| 515 | 0 |                             {RPCResult::Type::NUM, "vout", "the vout value"}, | 
| 516 | 0 |                             {RPCResult::Type::STR, "address", /*optional=*/true, "the bitcoin address"}, | 
| 517 | 0 |                             {RPCResult::Type::STR, "label", /*optional=*/true, "The associated label, or \"\" for the default label"}, | 
| 518 | 0 |                             {RPCResult::Type::STR, "scriptPubKey", "the output script"}, | 
| 519 | 0 |                             {RPCResult::Type::STR_AMOUNT, "amount", "the transaction output amount in " + CURRENCY_UNIT}, | 
| 520 | 0 |                             {RPCResult::Type::NUM, "confirmations", "The number of confirmations"}, | 
| 521 | 0 |                             {RPCResult::Type::NUM, "ancestorcount", /*optional=*/true, "The number of in-mempool ancestor transactions, including this one (if transaction is in the mempool)"}, | 
| 522 | 0 |                             {RPCResult::Type::NUM, "ancestorsize", /*optional=*/true, "The virtual transaction size of in-mempool ancestors, including this one (if transaction is in the mempool)"}, | 
| 523 | 0 |                             {RPCResult::Type::STR_AMOUNT, "ancestorfees", /*optional=*/true, "The total fees of in-mempool ancestors (including this one) with fee deltas used for mining priority in " + CURRENCY_ATOM + " (if transaction is in the mempool)"}, | 
| 524 | 0 |                             {RPCResult::Type::STR_HEX, "redeemScript", /*optional=*/true, "The redeem script if the output script is P2SH"}, | 
| 525 | 0 |                             {RPCResult::Type::STR, "witnessScript", /*optional=*/true, "witness script if the output script is P2WSH or P2SH-P2WSH"}, | 
| 526 | 0 |                             {RPCResult::Type::BOOL, "spendable", "Whether we have the private keys to spend this output"}, | 
| 527 | 0 |                             {RPCResult::Type::BOOL, "solvable", "Whether we know how to spend this output, ignoring the lack of keys"}, | 
| 528 | 0 |                             {RPCResult::Type::BOOL, "reused", /*optional=*/true, "(only present if avoid_reuse is set) Whether this output is reused/dirty (sent to an address that was previously spent from)"}, | 
| 529 | 0 |                             {RPCResult::Type::STR, "desc", /*optional=*/true, "(only when solvable) A descriptor for spending this output"}, | 
| 530 | 0 |                             {RPCResult::Type::ARR, "parent_descs", /*optional=*/false, "List of parent descriptors for the output script of this coin.", { | 
| 531 | 0 |                                 {RPCResult::Type::STR, "desc", "The descriptor string."}, | 
| 532 | 0 |                             }}, | 
| 533 | 0 |                             {RPCResult::Type::BOOL, "safe", "Whether this output is considered safe to spend. Unconfirmed transactions\n" | 
| 534 | 0 |                                                             "from outside keys and unconfirmed replacement transactions are considered unsafe\n" | 
| 535 | 0 |                                                             "and are not eligible for spending by fundrawtransaction and sendtoaddress."}, | 
| 536 | 0 |                         }}, | 
| 537 | 0 |                     } | 
| 538 | 0 |                 }, | 
| 539 | 0 |                 RPCExamples{ | 
| 540 | 0 |                     HelpExampleCli("listunspent", "") | 
| 541 | 0 |             + HelpExampleCli("listunspent", "6 9999999 \"[\\\"" + EXAMPLE_ADDRESS[0] + "\\\",\\\"" + EXAMPLE_ADDRESS[1] + "\\\"]\"") | 
| 542 | 0 |             + HelpExampleRpc("listunspent", "6, 9999999 \"[\\\"" + EXAMPLE_ADDRESS[0] + "\\\",\\\"" + EXAMPLE_ADDRESS[1] + "\\\"]\"") | 
| 543 | 0 |             + HelpExampleCli("listunspent", "6 9999999 '[]' true '{ \"minimumAmount\": 0.005 }'") | 
| 544 | 0 |             + HelpExampleRpc("listunspent", "6, 9999999, [] , true, { \"minimumAmount\": 0.005 } ") | 
| 545 | 0 |                 }, | 
| 546 | 0 |         [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue | 
| 547 | 0 | { | 
| 548 | 0 |     const std::shared_ptr<const CWallet> pwallet = GetWalletForJSONRPCRequest(request); | 
| 549 | 0 |     if (!pwallet) return UniValue::VNULL; | 
| 550 |  |  | 
| 551 | 0 |     int nMinDepth = 1; | 
| 552 | 0 |     if (!request.params[0].isNull()) { | 
| 553 | 0 |         nMinDepth = request.params[0].getInt<int>(); | 
| 554 | 0 |     } | 
| 555 |  | 
 | 
| 556 | 0 |     int nMaxDepth = 9999999; | 
| 557 | 0 |     if (!request.params[1].isNull()) { | 
| 558 | 0 |         nMaxDepth = request.params[1].getInt<int>(); | 
| 559 | 0 |     } | 
| 560 |  | 
 | 
| 561 | 0 |     std::set<CTxDestination> destinations; | 
| 562 | 0 |     if (!request.params[2].isNull()) { | 
| 563 | 0 |         UniValue inputs = request.params[2].get_array(); | 
| 564 | 0 |         for (unsigned int idx = 0; idx < inputs.size(); idx++) { | 
| 565 | 0 |             const UniValue& input = inputs[idx]; | 
| 566 | 0 |             CTxDestination dest = DecodeDestination(input.get_str()); | 
| 567 | 0 |             if (!IsValidDestination(dest)) { | 
| 568 | 0 |                 throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, std::string("Invalid Bitcoin address: ") + input.get_str()); | 
| 569 | 0 |             } | 
| 570 | 0 |             if (!destinations.insert(dest).second) { | 
| 571 | 0 |                 throw JSONRPCError(RPC_INVALID_PARAMETER, std::string("Invalid parameter, duplicated address: ") + input.get_str()); | 
| 572 | 0 |             } | 
| 573 | 0 |         } | 
| 574 | 0 |     } | 
| 575 |  |  | 
| 576 | 0 |     bool include_unsafe = true; | 
| 577 | 0 |     if (!request.params[3].isNull()) { | 
| 578 | 0 |         include_unsafe = request.params[3].get_bool(); | 
| 579 | 0 |     } | 
| 580 |  | 
 | 
| 581 | 0 |     CoinFilterParams filter_coins; | 
| 582 | 0 |     filter_coins.min_amount = 0; | 
| 583 |  | 
 | 
| 584 | 0 |     if (!request.params[4].isNull()) { | 
| 585 | 0 |         const UniValue& options = request.params[4].get_obj(); | 
| 586 |  | 
 | 
| 587 | 0 |         RPCTypeCheckObj(options, | 
| 588 | 0 |             { | 
| 589 | 0 |                 {"minimumAmount", UniValueType()}, | 
| 590 | 0 |                 {"maximumAmount", UniValueType()}, | 
| 591 | 0 |                 {"minimumSumAmount", UniValueType()}, | 
| 592 | 0 |                 {"maximumCount", UniValueType(UniValue::VNUM)}, | 
| 593 | 0 |                 {"include_immature_coinbase", UniValueType(UniValue::VBOOL)} | 
| 594 | 0 |             }, | 
| 595 | 0 |             true, true); | 
| 596 |  | 
 | 
| 597 | 0 |         if (options.exists("minimumAmount")) | 
| 598 | 0 |             filter_coins.min_amount = AmountFromValue(options["minimumAmount"]); | 
| 599 |  | 
 | 
| 600 | 0 |         if (options.exists("maximumAmount")) | 
| 601 | 0 |             filter_coins.max_amount = AmountFromValue(options["maximumAmount"]); | 
| 602 |  | 
 | 
| 603 | 0 |         if (options.exists("minimumSumAmount")) | 
| 604 | 0 |             filter_coins.min_sum_amount = AmountFromValue(options["minimumSumAmount"]); | 
| 605 |  | 
 | 
| 606 | 0 |         if (options.exists("maximumCount")) | 
| 607 | 0 |             filter_coins.max_count = options["maximumCount"].getInt<int64_t>(); | 
| 608 |  | 
 | 
| 609 | 0 |         if (options.exists("include_immature_coinbase")) { | 
| 610 | 0 |             filter_coins.include_immature_coinbase = options["include_immature_coinbase"].get_bool(); | 
| 611 | 0 |         } | 
| 612 | 0 |     } | 
| 613 |  |  | 
| 614 |  |     // Make sure the results are valid at least up to the most recent block | 
| 615 |  |     // the user could have gotten from another RPC command prior to now | 
| 616 | 0 |     pwallet->BlockUntilSyncedToCurrentChain(); | 
| 617 |  | 
 | 
| 618 | 0 |     UniValue results(UniValue::VARR); | 
| 619 | 0 |     std::vector<COutput> vecOutputs; | 
| 620 | 0 |     { | 
| 621 | 0 |         CCoinControl cctl; | 
| 622 | 0 |         cctl.m_avoid_address_reuse = false; | 
| 623 | 0 |         cctl.m_min_depth = nMinDepth; | 
| 624 | 0 |         cctl.m_max_depth = nMaxDepth; | 
| 625 | 0 |         cctl.m_include_unsafe_inputs = include_unsafe; | 
| 626 | 0 |         LOCK(pwallet->cs_wallet); | Line | Count | Source |  | 257 | 0 | #define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__) | Line | Count | Source |  | 11 | 0 | #define UNIQUE_NAME(name) PASTE2(name, __COUNTER__) | Line | Count | Source |  | 9 | 0 | #define PASTE2(x, y) PASTE(x, y) | Line | Count | Source |  | 8 | 0 | #define PASTE(x, y) x ## y | 
 | 
 | 
 | 
 | 
| 627 | 0 |         vecOutputs = AvailableCoinsListUnspent(*pwallet, &cctl, filter_coins).All(); | 
| 628 | 0 |     } | 
| 629 |  | 
 | 
| 630 | 0 |     LOCK(pwallet->cs_wallet); | Line | Count | Source |  | 257 | 0 | #define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__) | Line | Count | Source |  | 11 | 0 | #define UNIQUE_NAME(name) PASTE2(name, __COUNTER__) | Line | Count | Source |  | 9 | 0 | #define PASTE2(x, y) PASTE(x, y) | Line | Count | Source |  | 8 | 0 | #define PASTE(x, y) x ## y | 
 | 
 | 
 | 
 | 
| 631 |  | 
 | 
| 632 | 0 |     const bool avoid_reuse = pwallet->IsWalletFlagSet(WALLET_FLAG_AVOID_REUSE); | 
| 633 |  | 
 | 
| 634 | 0 |     for (const COutput& out : vecOutputs) { | 
| 635 | 0 |         CTxDestination address; | 
| 636 | 0 |         const CScript& scriptPubKey = out.txout.scriptPubKey; | 
| 637 | 0 |         bool fValidAddress = ExtractDestination(scriptPubKey, address); | 
| 638 | 0 |         bool reused = avoid_reuse && pwallet->IsSpentKey(scriptPubKey); | 
| 639 |  | 
 | 
| 640 | 0 |         if (destinations.size() && (!fValidAddress || !destinations.count(address))) | 
| 641 | 0 |             continue; | 
| 642 |  |  | 
| 643 | 0 |         UniValue entry(UniValue::VOBJ); | 
| 644 | 0 |         entry.pushKV("txid", out.outpoint.hash.GetHex()); | 
| 645 | 0 |         entry.pushKV("vout", (int)out.outpoint.n); | 
| 646 |  | 
 | 
| 647 | 0 |         if (fValidAddress) { | 
| 648 | 0 |             entry.pushKV("address", EncodeDestination(address)); | 
| 649 |  | 
 | 
| 650 | 0 |             const auto* address_book_entry = pwallet->FindAddressBookEntry(address); | 
| 651 | 0 |             if (address_book_entry) { | 
| 652 | 0 |                 entry.pushKV("label", address_book_entry->GetLabel()); | 
| 653 | 0 |             } | 
| 654 |  | 
 | 
| 655 | 0 |             std::unique_ptr<SigningProvider> provider = pwallet->GetSolvingProvider(scriptPubKey); | 
| 656 | 0 |             if (provider) { | 
| 657 | 0 |                 if (scriptPubKey.IsPayToScriptHash()) { | 
| 658 | 0 |                     const CScriptID hash = ToScriptID(std::get<ScriptHash>(address)); | 
| 659 | 0 |                     CScript redeemScript; | 
| 660 | 0 |                     if (provider->GetCScript(hash, redeemScript)) { | 
| 661 | 0 |                         entry.pushKV("redeemScript", HexStr(redeemScript)); | 
| 662 |  |                         // Now check if the redeemScript is actually a P2WSH script | 
| 663 | 0 |                         CTxDestination witness_destination; | 
| 664 | 0 |                         if (redeemScript.IsPayToWitnessScriptHash()) { | 
| 665 | 0 |                             bool extracted = ExtractDestination(redeemScript, witness_destination); | 
| 666 | 0 |                             CHECK_NONFATAL(extracted); | Line | Count | Source |  | 103 | 0 |     inline_check_non_fatal(condition, __FILE__, __LINE__, __func__, #condition) | 
 | 
| 667 |  |                             // Also return the witness script | 
| 668 | 0 |                             const WitnessV0ScriptHash& whash = std::get<WitnessV0ScriptHash>(witness_destination); | 
| 669 | 0 |                             CScriptID id{RIPEMD160(whash)}; | 
| 670 | 0 |                             CScript witnessScript; | 
| 671 | 0 |                             if (provider->GetCScript(id, witnessScript)) { | 
| 672 | 0 |                                 entry.pushKV("witnessScript", HexStr(witnessScript)); | 
| 673 | 0 |                             } | 
| 674 | 0 |                         } | 
| 675 | 0 |                     } | 
| 676 | 0 |                 } else if (scriptPubKey.IsPayToWitnessScriptHash()) { | 
| 677 | 0 |                     const WitnessV0ScriptHash& whash = std::get<WitnessV0ScriptHash>(address); | 
| 678 | 0 |                     CScriptID id{RIPEMD160(whash)}; | 
| 679 | 0 |                     CScript witnessScript; | 
| 680 | 0 |                     if (provider->GetCScript(id, witnessScript)) { | 
| 681 | 0 |                         entry.pushKV("witnessScript", HexStr(witnessScript)); | 
| 682 | 0 |                     } | 
| 683 | 0 |                 } | 
| 684 | 0 |             } | 
| 685 | 0 |         } | 
| 686 |  | 
 | 
| 687 | 0 |         entry.pushKV("scriptPubKey", HexStr(scriptPubKey)); | 
| 688 | 0 |         entry.pushKV("amount", ValueFromAmount(out.txout.nValue)); | 
| 689 | 0 |         entry.pushKV("confirmations", out.depth); | 
| 690 | 0 |         if (!out.depth) { | 
| 691 | 0 |             size_t ancestor_count, descendant_count, ancestor_size; | 
| 692 | 0 |             CAmount ancestor_fees; | 
| 693 | 0 |             pwallet->chain().getTransactionAncestry(out.outpoint.hash, ancestor_count, descendant_count, &ancestor_size, &ancestor_fees); | 
| 694 | 0 |             if (ancestor_count) { | 
| 695 | 0 |                 entry.pushKV("ancestorcount", uint64_t(ancestor_count)); | 
| 696 | 0 |                 entry.pushKV("ancestorsize", uint64_t(ancestor_size)); | 
| 697 | 0 |                 entry.pushKV("ancestorfees", uint64_t(ancestor_fees)); | 
| 698 | 0 |             } | 
| 699 | 0 |         } | 
| 700 | 0 |         entry.pushKV("spendable", out.spendable); | 
| 701 | 0 |         entry.pushKV("solvable", out.solvable); | 
| 702 | 0 |         if (out.solvable) { | 
| 703 | 0 |             std::unique_ptr<SigningProvider> provider = pwallet->GetSolvingProvider(scriptPubKey); | 
| 704 | 0 |             if (provider) { | 
| 705 | 0 |                 auto descriptor = InferDescriptor(scriptPubKey, *provider); | 
| 706 | 0 |                 entry.pushKV("desc", descriptor->ToString()); | 
| 707 | 0 |             } | 
| 708 | 0 |         } | 
| 709 | 0 |         PushParentDescriptors(*pwallet, scriptPubKey, entry); | 
| 710 | 0 |         if (avoid_reuse) entry.pushKV("reused", reused); | 
| 711 | 0 |         entry.pushKV("safe", out.safe); | 
| 712 | 0 |         results.push_back(std::move(entry)); | 
| 713 | 0 |     } | 
| 714 |  | 
 | 
| 715 | 0 |     return results; | 
| 716 | 0 | }, | 
| 717 | 0 |     }; | 
| 718 | 0 | } | 
| 719 |  | } // namespace wallet |