Bitcoin Core 30.99.0
P2P Digital Currency
mempool.cpp
Go to the documentation of this file.
1// Copyright (c) 2010 Satoshi Nakamoto
2// Copyright (c) 2009-2022 The Bitcoin Core developers
3// Distributed under the MIT software license, see the accompanying
4// file COPYING or http://www.opensource.org/licenses/mit-license.php.
5
6#include <rpc/blockchain.h>
7
9
10#include <chainparams.h>
12#include <core_io.h>
14#include <net_processing.h>
16#include <node/types.h>
17#include <policy/rbf.h>
18#include <policy/settings.h>
20#include <rpc/server.h>
21#include <rpc/server_util.h>
22#include <rpc/util.h>
23#include <txmempool.h>
24#include <univalue.h>
25#include <util/fs.h>
26#include <util/moneystr.h>
27#include <util/strencodings.h>
28#include <util/time.h>
29#include <util/vector.h>
30
31#include <string_view>
32#include <utility>
33
35
41using util::ToString;
42
44{
45 return RPCHelpMan{
46 "sendrawtransaction",
47 "Submit a raw transaction (serialized, hex-encoded) to local node and network.\n"
48 "\nThe transaction will be sent unconditionally to all peers, so using sendrawtransaction\n"
49 "for manual rebroadcast may degrade privacy by leaking the transaction's origin, as\n"
50 "nodes will normally not rebroadcast non-wallet transactions already in their mempool.\n"
51 "\nA specific exception, RPC_TRANSACTION_ALREADY_IN_UTXO_SET, may throw if the transaction cannot be added to the mempool.\n"
52 "\nRelated RPCs: createrawtransaction, signrawtransactionwithkey\n",
53 {
54 {"hexstring", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "The hex string of the raw transaction"},
56 "Reject transactions whose fee rate is higher than the specified value, expressed in " + CURRENCY_UNIT +
57 "/kvB.\nFee rates larger than 1BTC/kvB are rejected.\nSet to 0 to accept any fee rate."},
59 "Reject transactions with provably unspendable outputs (e.g. 'datacarrier' outputs that use the OP_RETURN opcode) greater than the specified value, expressed in " + CURRENCY_UNIT + ".\n"
60 "If burning funds through unspendable outputs is desired, increase this value.\n"
61 "This check is based on heuristics and does not guarantee spendability of outputs.\n"},
62 },
64 RPCResult::Type::STR_HEX, "", "The transaction hash in hex"
65 },
67 "\nCreate a transaction\n"
68 + HelpExampleCli("createrawtransaction", "\"[{\\\"txid\\\" : \\\"mytxid\\\",\\\"vout\\\":0}]\" \"{\\\"myaddress\\\":0.01}\"") +
69 "Sign the transaction, and get back the hex\n"
70 + HelpExampleCli("signrawtransactionwithwallet", "\"myhex\"") +
71 "\nSend the transaction (signed hex)\n"
72 + HelpExampleCli("sendrawtransaction", "\"signedhex\"") +
73 "\nAs a JSON-RPC call\n"
74 + HelpExampleRpc("sendrawtransaction", "\"signedhex\"")
75 },
76 [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
77 {
78 const CAmount max_burn_amount = request.params[2].isNull() ? 0 : AmountFromValue(request.params[2]);
79
81 if (!DecodeHexTx(mtx, request.params[0].get_str())) {
82 throw JSONRPCError(RPC_DESERIALIZATION_ERROR, "TX decode failed. Make sure the tx has at least one input.");
83 }
84
85 for (const auto& out : mtx.vout) {
86 if((out.scriptPubKey.IsUnspendable() || !out.scriptPubKey.HasValidOps()) && out.nValue > max_burn_amount) {
87 throw JSONRPCTransactionError(TransactionError::MAX_BURN_EXCEEDED);
88 }
89 }
90
91 CTransactionRef tx(MakeTransactionRef(std::move(mtx)));
92
93 const CFeeRate max_raw_tx_fee_rate{ParseFeeRate(self.Arg<UniValue>("maxfeerate"))};
94
95 int64_t virtual_size = GetVirtualTransactionSize(*tx);
96 CAmount max_raw_tx_fee = max_raw_tx_fee_rate.GetFee(virtual_size);
97
98 std::string err_string;
100 NodeContext& node = EnsureAnyNodeContext(request.context);
102 tx,
103 err_string,
104 max_raw_tx_fee,
106 /*wait_callback=*/true);
107 if (TransactionError::OK != err) {
108 throw JSONRPCTransactionError(err, err_string);
109 }
110
111 return tx->GetHash().GetHex();
112 },
113 };
114}
115
117{
118 return RPCHelpMan{
119 "testmempoolaccept",
120 "Returns result of mempool acceptance tests indicating if raw transaction(s) (serialized, hex-encoded) would be accepted by mempool.\n"
121 "\nIf multiple transactions are passed in, parents must come before children and package policies apply: the transactions cannot conflict with any mempool transactions or each other.\n"
122 "\nIf one transaction fails, other transactions may not be fully validated (the 'allowed' key will be blank).\n"
123 "\nThe maximum number of transactions allowed is " + ToString(MAX_PACKAGE_COUNT) + ".\n"
124 "\nThis checks if transactions violate the consensus or policy rules.\n"
125 "\nSee sendrawtransaction call.\n",
126 {
127 {"rawtxs", RPCArg::Type::ARR, RPCArg::Optional::NO, "An array of hex strings of raw transactions.",
128 {
130 },
131 },
133 "Reject transactions whose fee rate is higher than the specified value, expressed in " + CURRENCY_UNIT +
134 "/kvB.\nFee rates larger than 1BTC/kvB are rejected.\nSet to 0 to accept any fee rate."},
135 },
136 RPCResult{
137 RPCResult::Type::ARR, "", "The result of the mempool acceptance test for each raw transaction in the input array.\n"
138 "Returns results for each transaction in the same order they were passed in.\n"
139 "Transactions that cannot be fully validated due to failures in other transactions will not contain an 'allowed' result.\n",
140 {
141 {RPCResult::Type::OBJ, "", "",
142 {
143 {RPCResult::Type::STR_HEX, "txid", "The transaction hash in hex"},
144 {RPCResult::Type::STR_HEX, "wtxid", "The transaction witness hash in hex"},
145 {RPCResult::Type::STR, "package-error", /*optional=*/true, "Package validation error, if any (only possible if rawtxs had more than 1 transaction)."},
146 {RPCResult::Type::BOOL, "allowed", /*optional=*/true, "Whether this tx would be accepted to the mempool and pass client-specified maxfeerate. "
147 "If not present, the tx was not fully validated due to a failure in another tx in the list."},
148 {RPCResult::Type::NUM, "vsize", /*optional=*/true, "Virtual transaction size as defined in BIP 141. This is different from actual serialized size for witness transactions as witness data is discounted (only present when 'allowed' is true)"},
149 {RPCResult::Type::OBJ, "fees", /*optional=*/true, "Transaction fees (only present if 'allowed' is true)",
150 {
151 {RPCResult::Type::STR_AMOUNT, "base", "transaction fee in " + CURRENCY_UNIT},
152 {RPCResult::Type::STR_AMOUNT, "effective-feerate", /*optional=*/false, "the effective feerate in " + CURRENCY_UNIT + " per KvB. May differ from the base feerate if, for example, there are modified fees from prioritisetransaction or a package feerate was used."},
153 {RPCResult::Type::ARR, "effective-includes", /*optional=*/false, "transactions whose fees and vsizes are included in effective-feerate.",
154 {RPCResult{RPCResult::Type::STR_HEX, "", "transaction wtxid in hex"},
155 }},
156 }},
157 {RPCResult::Type::STR, "reject-reason", /*optional=*/true, "Rejection reason (only present when 'allowed' is false)"},
158 {RPCResult::Type::STR, "reject-details", /*optional=*/true, "Rejection details (only present when 'allowed' is false and rejection details exist)"},
159 }},
160 }
161 },
163 "\nCreate a transaction\n"
164 + HelpExampleCli("createrawtransaction", "\"[{\\\"txid\\\" : \\\"mytxid\\\",\\\"vout\\\":0}]\" \"{\\\"myaddress\\\":0.01}\"") +
165 "Sign the transaction, and get back the hex\n"
166 + HelpExampleCli("signrawtransactionwithwallet", "\"myhex\"") +
167 "\nTest acceptance of the transaction (signed hex)\n"
168 + HelpExampleCli("testmempoolaccept", R"('["signedhex"]')") +
169 "\nAs a JSON-RPC call\n"
170 + HelpExampleRpc("testmempoolaccept", "[\"signedhex\"]")
171 },
172 [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
173 {
174 const UniValue raw_transactions = request.params[0].get_array();
175 if (raw_transactions.size() < 1 || raw_transactions.size() > MAX_PACKAGE_COUNT) {
177 "Array must contain between 1 and " + ToString(MAX_PACKAGE_COUNT) + " transactions.");
178 }
179
180 const CFeeRate max_raw_tx_fee_rate{ParseFeeRate(self.Arg<UniValue>("maxfeerate"))};
181
182 std::vector<CTransactionRef> txns;
183 txns.reserve(raw_transactions.size());
184 for (const auto& rawtx : raw_transactions.getValues()) {
186 if (!DecodeHexTx(mtx, rawtx.get_str())) {
188 "TX decode failed: " + rawtx.get_str() + " Make sure the tx has at least one input.");
189 }
190 txns.emplace_back(MakeTransactionRef(std::move(mtx)));
191 }
192
193 NodeContext& node = EnsureAnyNodeContext(request.context);
194 CTxMemPool& mempool = EnsureMemPool(node);
196 Chainstate& chainstate = chainman.ActiveChainstate();
197 const PackageMempoolAcceptResult package_result = [&] {
199 if (txns.size() > 1) return ProcessNewPackage(chainstate, mempool, txns, /*test_accept=*/true, /*client_maxfeerate=*/{});
200 return PackageMempoolAcceptResult(txns[0]->GetWitnessHash(),
201 chainman.ProcessTransaction(txns[0], /*test_accept=*/true));
202 }();
203
204 UniValue rpc_result(UniValue::VARR);
205 // We will check transaction fees while we iterate through txns in order. If any transaction fee
206 // exceeds maxfeerate, we will leave the rest of the validation results blank, because it
207 // doesn't make sense to return a validation result for a transaction if its ancestor(s) would
208 // not be submitted.
209 bool exit_early{false};
210 for (const auto& tx : txns) {
211 UniValue result_inner(UniValue::VOBJ);
212 result_inner.pushKV("txid", tx->GetHash().GetHex());
213 result_inner.pushKV("wtxid", tx->GetWitnessHash().GetHex());
215 result_inner.pushKV("package-error", package_result.m_state.ToString());
216 }
217 auto it = package_result.m_tx_results.find(tx->GetWitnessHash());
218 if (exit_early || it == package_result.m_tx_results.end()) {
219 // Validation unfinished. Just return the txid and wtxid.
220 rpc_result.push_back(std::move(result_inner));
221 continue;
222 }
223 const auto& tx_result = it->second;
224 // Package testmempoolaccept doesn't allow transactions to already be in the mempool.
226 if (tx_result.m_result_type == MempoolAcceptResult::ResultType::VALID) {
227 const CAmount fee = tx_result.m_base_fees.value();
228 // Check that fee does not exceed maximum fee
229 const int64_t virtual_size = tx_result.m_vsize.value();
230 const CAmount max_raw_tx_fee = max_raw_tx_fee_rate.GetFee(virtual_size);
231 if (max_raw_tx_fee && fee > max_raw_tx_fee) {
232 result_inner.pushKV("allowed", false);
233 result_inner.pushKV("reject-reason", "max-fee-exceeded");
234 exit_early = true;
235 } else {
236 // Only return the fee and vsize if the transaction would pass ATMP.
237 // These can be used to calculate the feerate.
238 result_inner.pushKV("allowed", true);
239 result_inner.pushKV("vsize", virtual_size);
241 fees.pushKV("base", ValueFromAmount(fee));
242 fees.pushKV("effective-feerate", ValueFromAmount(tx_result.m_effective_feerate.value().GetFeePerK()));
243 UniValue effective_includes_res(UniValue::VARR);
244 for (const auto& wtxid : tx_result.m_wtxids_fee_calculations.value()) {
245 effective_includes_res.push_back(wtxid.ToString());
246 }
247 fees.pushKV("effective-includes", std::move(effective_includes_res));
248 result_inner.pushKV("fees", std::move(fees));
249 }
250 } else {
251 result_inner.pushKV("allowed", false);
252 const TxValidationState state = tx_result.m_state;
254 result_inner.pushKV("reject-reason", "missing-inputs");
255 } else {
256 result_inner.pushKV("reject-reason", state.GetRejectReason());
257 result_inner.pushKV("reject-details", state.ToString());
258 }
259 }
260 rpc_result.push_back(std::move(result_inner));
261 }
262 return rpc_result;
263 },
264 };
265}
266
267static std::vector<RPCResult> MempoolEntryDescription()
268{
269 return {
270 RPCResult{RPCResult::Type::NUM, "vsize", "virtual transaction size as defined in BIP 141. This is different from actual serialized size for witness transactions as witness data is discounted."},
271 RPCResult{RPCResult::Type::NUM, "weight", "transaction weight as defined in BIP 141."},
272 RPCResult{RPCResult::Type::NUM_TIME, "time", "local time transaction entered pool in seconds since 1 Jan 1970 GMT"},
273 RPCResult{RPCResult::Type::NUM, "height", "block height when transaction entered pool"},
274 RPCResult{RPCResult::Type::NUM, "descendantcount", "number of in-mempool descendant transactions (including this one)"},
275 RPCResult{RPCResult::Type::NUM, "descendantsize", "virtual transaction size of in-mempool descendants (including this one)"},
276 RPCResult{RPCResult::Type::NUM, "ancestorcount", "number of in-mempool ancestor transactions (including this one)"},
277 RPCResult{RPCResult::Type::NUM, "ancestorsize", "virtual transaction size of in-mempool ancestors (including this one)"},
278 RPCResult{RPCResult::Type::STR_HEX, "wtxid", "hash of serialized transaction, including witness data"},
280 {
281 RPCResult{RPCResult::Type::STR_AMOUNT, "base", "transaction fee, denominated in " + CURRENCY_UNIT},
282 RPCResult{RPCResult::Type::STR_AMOUNT, "modified", "transaction fee with fee deltas used for mining priority, denominated in " + CURRENCY_UNIT},
283 RPCResult{RPCResult::Type::STR_AMOUNT, "ancestor", "transaction fees of in-mempool ancestors (including this one) with fee deltas used for mining priority, denominated in " + CURRENCY_UNIT},
284 RPCResult{RPCResult::Type::STR_AMOUNT, "descendant", "transaction fees of in-mempool descendants (including this one) with fee deltas used for mining priority, denominated in " + CURRENCY_UNIT},
285 }},
286 RPCResult{RPCResult::Type::ARR, "depends", "unconfirmed transactions used as inputs for this transaction",
287 {RPCResult{RPCResult::Type::STR_HEX, "transactionid", "parent transaction id"}}},
288 RPCResult{RPCResult::Type::ARR, "spentby", "unconfirmed transactions spending outputs from this transaction",
289 {RPCResult{RPCResult::Type::STR_HEX, "transactionid", "child transaction id"}}},
290 RPCResult{RPCResult::Type::BOOL, "bip125-replaceable", "Whether this transaction signals BIP125 replaceability or has an unconfirmed ancestor signaling BIP125 replaceability. (DEPRECATED)\n"},
291 RPCResult{RPCResult::Type::BOOL, "unbroadcast", "Whether this transaction is currently unbroadcast (initial broadcast not yet acknowledged by any peers)"},
292 };
293}
294
295static void entryToJSON(const CTxMemPool& pool, UniValue& info, const CTxMemPoolEntry& e) EXCLUSIVE_LOCKS_REQUIRED(pool.cs)
296{
297 AssertLockHeld(pool.cs);
298
299 info.pushKV("vsize", (int)e.GetTxSize());
300 info.pushKV("weight", (int)e.GetTxWeight());
301 info.pushKV("time", count_seconds(e.GetTime()));
302 info.pushKV("height", (int)e.GetHeight());
303 info.pushKV("descendantcount", e.GetCountWithDescendants());
304 info.pushKV("descendantsize", e.GetSizeWithDescendants());
305 info.pushKV("ancestorcount", e.GetCountWithAncestors());
306 info.pushKV("ancestorsize", e.GetSizeWithAncestors());
307 info.pushKV("wtxid", e.GetTx().GetWitnessHash().ToString());
308
310 fees.pushKV("base", ValueFromAmount(e.GetFee()));
311 fees.pushKV("modified", ValueFromAmount(e.GetModifiedFee()));
312 fees.pushKV("ancestor", ValueFromAmount(e.GetModFeesWithAncestors()));
313 fees.pushKV("descendant", ValueFromAmount(e.GetModFeesWithDescendants()));
314 info.pushKV("fees", std::move(fees));
315
316 const CTransaction& tx = e.GetTx();
317 std::set<std::string> setDepends;
318 for (const CTxIn& txin : tx.vin)
319 {
320 if (pool.exists(txin.prevout.hash))
321 setDepends.insert(txin.prevout.hash.ToString());
322 }
323
324 UniValue depends(UniValue::VARR);
325 for (const std::string& dep : setDepends)
326 {
327 depends.push_back(dep);
328 }
329
330 info.pushKV("depends", std::move(depends));
331
333 for (const CTxMemPoolEntry& child : e.GetMemPoolChildrenConst()) {
334 spent.push_back(child.GetTx().GetHash().ToString());
335 }
336
337 info.pushKV("spentby", std::move(spent));
338
339 // Add opt-in RBF status
340 bool rbfStatus = false;
341 RBFTransactionState rbfState = IsRBFOptIn(tx, pool);
342 if (rbfState == RBFTransactionState::UNKNOWN) {
343 throw JSONRPCError(RPC_MISC_ERROR, "Transaction is not in mempool");
344 } else if (rbfState == RBFTransactionState::REPLACEABLE_BIP125) {
345 rbfStatus = true;
346 }
347
348 info.pushKV("bip125-replaceable", rbfStatus);
349 info.pushKV("unbroadcast", pool.IsUnbroadcastTx(tx.GetHash()));
350}
351
352UniValue MempoolToJSON(const CTxMemPool& pool, bool verbose, bool include_mempool_sequence)
353{
354 if (verbose) {
355 if (include_mempool_sequence) {
356 throw JSONRPCError(RPC_INVALID_PARAMETER, "Verbose results cannot contain mempool sequence values.");
357 }
358 LOCK(pool.cs);
360 for (const CTxMemPoolEntry& e : pool.entryAll()) {
362 entryToJSON(pool, info, e);
363 // Mempool has unique entries so there is no advantage in using
364 // UniValue::pushKV, which checks if the key already exists in O(N).
365 // UniValue::pushKVEnd is used instead which currently is O(1).
366 o.pushKVEnd(e.GetTx().GetHash().ToString(), std::move(info));
367 }
368 return o;
369 } else {
371 uint64_t mempool_sequence;
372 {
373 LOCK(pool.cs);
374 for (const CTxMemPoolEntry& e : pool.entryAll()) {
375 a.push_back(e.GetTx().GetHash().ToString());
376 }
377 mempool_sequence = pool.GetSequence();
378 }
379 if (!include_mempool_sequence) {
380 return a;
381 } else {
383 o.pushKV("txids", std::move(a));
384 o.pushKV("mempool_sequence", mempool_sequence);
385 return o;
386 }
387 }
388}
389
391{
392 return RPCHelpMan{
393 "getrawmempool",
394 "Returns all transaction ids in memory pool as a json array of string transaction ids.\n"
395 "\nHint: use getmempoolentry to fetch a specific transaction from the mempool.\n",
396 {
397 {"verbose", RPCArg::Type::BOOL, RPCArg::Default{false}, "True for a json object, false for array of transaction ids"},
398 {"mempool_sequence", RPCArg::Type::BOOL, RPCArg::Default{false}, "If verbose=false, returns a json object with transaction list and mempool sequence number attached."},
399 },
400 {
401 RPCResult{"for verbose = false",
402 RPCResult::Type::ARR, "", "",
403 {
404 {RPCResult::Type::STR_HEX, "", "The transaction id"},
405 }},
406 RPCResult{"for verbose = true",
408 {
409 {RPCResult::Type::OBJ, "transactionid", "", MempoolEntryDescription()},
410 }},
411 RPCResult{"for verbose = false and mempool_sequence = true",
412 RPCResult::Type::OBJ, "", "",
413 {
414 {RPCResult::Type::ARR, "txids", "",
415 {
416 {RPCResult::Type::STR_HEX, "", "The transaction id"},
417 }},
418 {RPCResult::Type::NUM, "mempool_sequence", "The mempool sequence value."},
419 }},
420 },
422 HelpExampleCli("getrawmempool", "true")
423 + HelpExampleRpc("getrawmempool", "true")
424 },
425 [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
426{
427 bool fVerbose = false;
428 if (!request.params[0].isNull())
429 fVerbose = request.params[0].get_bool();
430
431 bool include_mempool_sequence = false;
432 if (!request.params[1].isNull()) {
433 include_mempool_sequence = request.params[1].get_bool();
434 }
435
436 return MempoolToJSON(EnsureAnyMemPool(request.context), fVerbose, include_mempool_sequence);
437},
438 };
439}
440
442{
443 return RPCHelpMan{
444 "getmempoolancestors",
445 "If txid is in the mempool, returns all in-mempool ancestors.\n",
446 {
447 {"txid", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "The transaction id (must be in mempool)"},
448 {"verbose", RPCArg::Type::BOOL, RPCArg::Default{false}, "True for a json object, false for array of transaction ids"},
449 },
450 {
451 RPCResult{"for verbose = false",
452 RPCResult::Type::ARR, "", "",
453 {{RPCResult::Type::STR_HEX, "", "The transaction id of an in-mempool ancestor transaction"}}},
454 RPCResult{"for verbose = true",
456 {
457 {RPCResult::Type::OBJ, "transactionid", "", MempoolEntryDescription()},
458 }},
459 },
461 HelpExampleCli("getmempoolancestors", "\"mytxid\"")
462 + HelpExampleRpc("getmempoolancestors", "\"mytxid\"")
463 },
464 [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
465{
466 bool fVerbose = false;
467 if (!request.params[1].isNull())
468 fVerbose = request.params[1].get_bool();
469
470 auto txid{Txid::FromUint256(ParseHashV(request.params[0], "txid"))};
471
472 const CTxMemPool& mempool = EnsureAnyMemPool(request.context);
473 LOCK(mempool.cs);
474
475 const auto entry{mempool.GetEntry(txid)};
476 if (entry == nullptr) {
477 throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Transaction not in mempool");
478 }
479
480 auto ancestors{mempool.AssumeCalculateMemPoolAncestors(self.m_name, *entry, CTxMemPool::Limits::NoLimits(), /*fSearchForParents=*/false)};
481
482 if (!fVerbose) {
484 for (CTxMemPool::txiter ancestorIt : ancestors) {
485 o.push_back(ancestorIt->GetTx().GetHash().ToString());
486 }
487 return o;
488 } else {
490 for (CTxMemPool::txiter ancestorIt : ancestors) {
491 const CTxMemPoolEntry &e = *ancestorIt;
493 entryToJSON(mempool, info, e);
494 o.pushKV(e.GetTx().GetHash().ToString(), std::move(info));
495 }
496 return o;
497 }
498},
499 };
500}
501
503{
504 return RPCHelpMan{
505 "getmempooldescendants",
506 "If txid is in the mempool, returns all in-mempool descendants.\n",
507 {
508 {"txid", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "The transaction id (must be in mempool)"},
509 {"verbose", RPCArg::Type::BOOL, RPCArg::Default{false}, "True for a json object, false for array of transaction ids"},
510 },
511 {
512 RPCResult{"for verbose = false",
513 RPCResult::Type::ARR, "", "",
514 {{RPCResult::Type::STR_HEX, "", "The transaction id of an in-mempool descendant transaction"}}},
515 RPCResult{"for verbose = true",
517 {
518 {RPCResult::Type::OBJ, "transactionid", "", MempoolEntryDescription()},
519 }},
520 },
522 HelpExampleCli("getmempooldescendants", "\"mytxid\"")
523 + HelpExampleRpc("getmempooldescendants", "\"mytxid\"")
524 },
525 [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
526{
527 bool fVerbose = false;
528 if (!request.params[1].isNull())
529 fVerbose = request.params[1].get_bool();
530
531 auto txid{Txid::FromUint256(ParseHashV(request.params[0], "txid"))};
532
533 const CTxMemPool& mempool = EnsureAnyMemPool(request.context);
534 LOCK(mempool.cs);
535
536 const auto it{mempool.GetIter(txid)};
537 if (!it) {
538 throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Transaction not in mempool");
539 }
540
541 CTxMemPool::setEntries setDescendants;
542 mempool.CalculateDescendants(*it, setDescendants);
543 // CTxMemPool::CalculateDescendants will include the given tx
544 setDescendants.erase(*it);
545
546 if (!fVerbose) {
548 for (CTxMemPool::txiter descendantIt : setDescendants) {
549 o.push_back(descendantIt->GetTx().GetHash().ToString());
550 }
551
552 return o;
553 } else {
555 for (CTxMemPool::txiter descendantIt : setDescendants) {
556 const CTxMemPoolEntry &e = *descendantIt;
558 entryToJSON(mempool, info, e);
559 o.pushKV(e.GetTx().GetHash().ToString(), std::move(info));
560 }
561 return o;
562 }
563},
564 };
565}
566
568{
569 return RPCHelpMan{
570 "getmempoolentry",
571 "Returns mempool data for given transaction\n",
572 {
573 {"txid", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "The transaction id (must be in mempool)"},
574 },
575 RPCResult{
578 HelpExampleCli("getmempoolentry", "\"mytxid\"")
579 + HelpExampleRpc("getmempoolentry", "\"mytxid\"")
580 },
581 [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
582{
583 auto txid{Txid::FromUint256(ParseHashV(request.params[0], "txid"))};
584
585 const CTxMemPool& mempool = EnsureAnyMemPool(request.context);
586 LOCK(mempool.cs);
587
588 const auto entry{mempool.GetEntry(txid)};
589 if (entry == nullptr) {
590 throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Transaction not in mempool");
591 }
592
594 entryToJSON(mempool, info, *entry);
595 return info;
596},
597 };
598}
599
601{
602 return RPCHelpMan{"gettxspendingprevout",
603 "Scans the mempool to find transactions spending any of the given outputs",
604 {
605 {"outputs", RPCArg::Type::ARR, RPCArg::Optional::NO, "The transaction outputs that we want to check, and within each, the txid (string) vout (numeric).",
606 {
608 {
609 {"txid", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "The transaction id"},
610 {"vout", RPCArg::Type::NUM, RPCArg::Optional::NO, "The output number"},
611 },
612 },
613 },
614 },
615 },
616 RPCResult{
617 RPCResult::Type::ARR, "", "",
618 {
619 {RPCResult::Type::OBJ, "", "",
620 {
621 {RPCResult::Type::STR_HEX, "txid", "the transaction id of the checked output"},
622 {RPCResult::Type::NUM, "vout", "the vout value of the checked output"},
623 {RPCResult::Type::STR_HEX, "spendingtxid", /*optional=*/true, "the transaction id of the mempool transaction spending this output (omitted if unspent)"},
624 }},
625 }
626 },
628 HelpExampleCli("gettxspendingprevout", "\"[{\\\"txid\\\":\\\"a08e6907dbbd3d809776dbfc5d82e371b764ed838b5655e72f463568df1aadf0\\\",\\\"vout\\\":3}]\"")
629 + HelpExampleRpc("gettxspendingprevout", "\"[{\\\"txid\\\":\\\"a08e6907dbbd3d809776dbfc5d82e371b764ed838b5655e72f463568df1aadf0\\\",\\\"vout\\\":3}]\"")
630 },
631 [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
632 {
633 const UniValue& output_params = request.params[0].get_array();
634 if (output_params.empty()) {
635 throw JSONRPCError(RPC_INVALID_PARAMETER, "Invalid parameter, outputs are missing");
636 }
637
638 std::vector<COutPoint> prevouts;
639 prevouts.reserve(output_params.size());
640
641 for (unsigned int idx = 0; idx < output_params.size(); idx++) {
642 const UniValue& o = output_params[idx].get_obj();
643
645 {
646 {"txid", UniValueType(UniValue::VSTR)},
647 {"vout", UniValueType(UniValue::VNUM)},
648 }, /*fAllowNull=*/false, /*fStrict=*/true);
649
650 const Txid txid = Txid::FromUint256(ParseHashO(o, "txid"));
651 const int nOutput{o.find_value("vout").getInt<int>()};
652 if (nOutput < 0) {
653 throw JSONRPCError(RPC_INVALID_PARAMETER, "Invalid parameter, vout cannot be negative");
654 }
655
656 prevouts.emplace_back(txid, nOutput);
657 }
658
659 const CTxMemPool& mempool = EnsureAnyMemPool(request.context);
660 LOCK(mempool.cs);
661
662 UniValue result{UniValue::VARR};
663
664 for (const COutPoint& prevout : prevouts) {
666 o.pushKV("txid", prevout.hash.ToString());
667 o.pushKV("vout", (uint64_t)prevout.n);
668
669 const CTransaction* spendingTx = mempool.GetConflictTx(prevout);
670 if (spendingTx != nullptr) {
671 o.pushKV("spendingtxid", spendingTx->GetHash().ToString());
672 }
673
674 result.push_back(std::move(o));
675 }
676
677 return result;
678 },
679 };
680}
681
683{
684 // Make sure this call is atomic in the pool.
685 LOCK(pool.cs);
687 ret.pushKV("loaded", pool.GetLoadTried());
688 ret.pushKV("size", (int64_t)pool.size());
689 ret.pushKV("bytes", (int64_t)pool.GetTotalTxSize());
690 ret.pushKV("usage", (int64_t)pool.DynamicMemoryUsage());
691 ret.pushKV("total_fee", ValueFromAmount(pool.GetTotalFee()));
692 ret.pushKV("maxmempool", pool.m_opts.max_size_bytes);
693 ret.pushKV("mempoolminfee", ValueFromAmount(std::max(pool.GetMinFee(), pool.m_opts.min_relay_feerate).GetFeePerK()));
694 ret.pushKV("minrelaytxfee", ValueFromAmount(pool.m_opts.min_relay_feerate.GetFeePerK()));
695 ret.pushKV("incrementalrelayfee", ValueFromAmount(pool.m_opts.incremental_relay_feerate.GetFeePerK()));
696 ret.pushKV("unbroadcastcount", uint64_t{pool.GetUnbroadcastTxs().size()});
697 ret.pushKV("fullrbf", true);
698 ret.pushKV("permitbaremultisig", pool.m_opts.permit_bare_multisig);
699 ret.pushKV("maxdatacarriersize", pool.m_opts.max_datacarrier_bytes.value_or(0));
700 return ret;
701}
702
704{
705 return RPCHelpMan{"getmempoolinfo",
706 "Returns details on the active state of the TX memory pool.",
707 {},
708 RPCResult{
709 RPCResult::Type::OBJ, "", "",
710 {
711 {RPCResult::Type::BOOL, "loaded", "True if the initial load attempt of the persisted mempool finished"},
712 {RPCResult::Type::NUM, "size", "Current tx count"},
713 {RPCResult::Type::NUM, "bytes", "Sum of all virtual transaction sizes as defined in BIP 141. Differs from actual serialized size because witness data is discounted"},
714 {RPCResult::Type::NUM, "usage", "Total memory usage for the mempool"},
715 {RPCResult::Type::STR_AMOUNT, "total_fee", "Total fees for the mempool in " + CURRENCY_UNIT + ", ignoring modified fees through prioritisetransaction"},
716 {RPCResult::Type::NUM, "maxmempool", "Maximum memory usage for the mempool"},
717 {RPCResult::Type::STR_AMOUNT, "mempoolminfee", "Minimum fee rate in " + CURRENCY_UNIT + "/kvB for tx to be accepted. Is the maximum of minrelaytxfee and minimum mempool fee"},
718 {RPCResult::Type::STR_AMOUNT, "minrelaytxfee", "Current minimum relay fee for transactions"},
719 {RPCResult::Type::NUM, "incrementalrelayfee", "minimum fee rate increment for mempool limiting or replacement in " + CURRENCY_UNIT + "/kvB"},
720 {RPCResult::Type::NUM, "unbroadcastcount", "Current number of transactions that haven't passed initial broadcast yet"},
721 {RPCResult::Type::BOOL, "fullrbf", "True if the mempool accepts RBF without replaceability signaling inspection (DEPRECATED)"},
722 {RPCResult::Type::BOOL, "permitbaremultisig", "True if the mempool accepts transactions with bare multisig outputs"},
723 {RPCResult::Type::NUM, "maxdatacarriersize", "Maximum number of bytes that can be used by OP_RETURN outputs in the mempool"},
724 }},
726 HelpExampleCli("getmempoolinfo", "")
727 + HelpExampleRpc("getmempoolinfo", "")
728 },
729 [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
730{
731 return MempoolInfoToJSON(EnsureAnyMemPool(request.context));
732},
733 };
734}
735
737{
738 return RPCHelpMan{
739 "importmempool",
740 "Import a mempool.dat file and attempt to add its contents to the mempool.\n"
741 "Warning: Importing untrusted files is dangerous, especially if metadata from the file is taken over.",
742 {
743 {"filepath", RPCArg::Type::STR, RPCArg::Optional::NO, "The mempool file"},
744 {"options",
747 "",
748 {
749 {"use_current_time", RPCArg::Type::BOOL, RPCArg::Default{true},
750 "Whether to use the current system time or use the entry time metadata from the mempool file.\n"
751 "Warning: Importing untrusted metadata may lead to unexpected issues and undesirable behavior."},
752 {"apply_fee_delta_priority", RPCArg::Type::BOOL, RPCArg::Default{false},
753 "Whether to apply the fee delta metadata from the mempool file.\n"
754 "It will be added to any existing fee deltas.\n"
755 "The fee delta can be set by the prioritisetransaction RPC.\n"
756 "Warning: Importing untrusted metadata may lead to unexpected issues and undesirable behavior.\n"
757 "Only set this bool if you understand what it does."},
758 {"apply_unbroadcast_set", RPCArg::Type::BOOL, RPCArg::Default{false},
759 "Whether to apply the unbroadcast set metadata from the mempool file.\n"
760 "Warning: Importing untrusted metadata may lead to unexpected issues and undesirable behavior."},
761 },
762 RPCArgOptions{.oneline_description = "options"}},
763 },
764 RPCResult{RPCResult::Type::OBJ, "", "", std::vector<RPCResult>{}},
765 RPCExamples{HelpExampleCli("importmempool", "/path/to/mempool.dat") + HelpExampleRpc("importmempool", "/path/to/mempool.dat")},
766 [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue {
767 const NodeContext& node{EnsureAnyNodeContext(request.context)};
768
769 CTxMemPool& mempool{EnsureMemPool(node)};
771 Chainstate& chainstate = chainman.ActiveChainstate();
772
773 if (chainman.IsInitialBlockDownload()) {
774 throw JSONRPCError(RPC_CLIENT_IN_INITIAL_DOWNLOAD, "Can only import the mempool after the block download and sync is done.");
775 }
776
777 const fs::path load_path{fs::u8path(self.Arg<std::string_view>("filepath"))};
778 const UniValue& use_current_time{request.params[1]["use_current_time"]};
779 const UniValue& apply_fee_delta{request.params[1]["apply_fee_delta_priority"]};
780 const UniValue& apply_unbroadcast{request.params[1]["apply_unbroadcast_set"]};
782 .use_current_time = use_current_time.isNull() ? true : use_current_time.get_bool(),
783 .apply_fee_delta_priority = apply_fee_delta.isNull() ? false : apply_fee_delta.get_bool(),
784 .apply_unbroadcast_set = apply_unbroadcast.isNull() ? false : apply_unbroadcast.get_bool(),
785 };
786
787 if (!node::LoadMempool(mempool, load_path, chainstate, std::move(opts))) {
788 throw JSONRPCError(RPC_MISC_ERROR, "Unable to import mempool file, see debug.log for details.");
789 }
790
792 return ret;
793 },
794 };
795}
796
798{
799 return RPCHelpMan{
800 "savemempool",
801 "Dumps the mempool to disk. It will fail until the previous dump is fully loaded.\n",
802 {},
803 RPCResult{
804 RPCResult::Type::OBJ, "", "",
805 {
806 {RPCResult::Type::STR, "filename", "the directory and file where the mempool was saved"},
807 }},
809 HelpExampleCli("savemempool", "")
810 + HelpExampleRpc("savemempool", "")
811 },
812 [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
813{
814 const ArgsManager& args{EnsureAnyArgsman(request.context)};
815 const CTxMemPool& mempool = EnsureAnyMemPool(request.context);
816
817 if (!mempool.GetLoadTried()) {
818 throw JSONRPCError(RPC_MISC_ERROR, "The mempool was not loaded yet");
819 }
820
821 const fs::path& dump_path = MempoolPath(args);
822
823 if (!DumpMempool(mempool, dump_path)) {
824 throw JSONRPCError(RPC_MISC_ERROR, "Unable to dump mempool to disk");
825 }
826
828 ret.pushKV("filename", dump_path.utf8string());
829
830 return ret;
831},
832 };
833}
834
835static std::vector<RPCResult> OrphanDescription()
836{
837 return {
838 RPCResult{RPCResult::Type::STR_HEX, "txid", "The transaction hash in hex"},
839 RPCResult{RPCResult::Type::STR_HEX, "wtxid", "The transaction witness hash in hex"},
840 RPCResult{RPCResult::Type::NUM, "bytes", "The serialized transaction size in bytes"},
841 RPCResult{RPCResult::Type::NUM, "vsize", "The virtual transaction size as defined in BIP 141. This is different from actual serialized size for witness transactions as witness data is discounted."},
842 RPCResult{RPCResult::Type::NUM, "weight", "The transaction weight as defined in BIP 141."},
844 {
845 RPCResult{RPCResult::Type::NUM, "peer_id", "Peer ID"},
846 }},
847 };
848}
849
851{
853 o.pushKV("txid", orphan.tx->GetHash().ToString());
854 o.pushKV("wtxid", orphan.tx->GetWitnessHash().ToString());
855 o.pushKV("bytes", orphan.tx->GetTotalSize());
856 o.pushKV("vsize", GetVirtualTransactionSize(*orphan.tx));
857 o.pushKV("weight", GetTransactionWeight(*orphan.tx));
859 for (const auto fromPeer: orphan.announcers) {
860 from.push_back(fromPeer);
861 }
862 o.pushKV("from", from);
863 return o;
864}
865
867{
868 return RPCHelpMan{
869 "getorphantxs",
870 "Shows transactions in the tx orphanage.\n"
871 "\nEXPERIMENTAL warning: this call may be changed in future releases.\n",
872 {
873 {"verbosity", RPCArg::Type::NUM, RPCArg::Default{0}, "0 for an array of txids (may contain duplicates), 1 for an array of objects with tx details, and 2 for details from (1) and tx hex",
875 },
876 {
877 RPCResult{"for verbose = 0",
878 RPCResult::Type::ARR, "", "",
879 {
880 {RPCResult::Type::STR_HEX, "txid", "The transaction hash in hex"},
881 }},
882 RPCResult{"for verbose = 1",
883 RPCResult::Type::ARR, "", "",
884 {
886 }},
887 RPCResult{"for verbose = 2",
888 RPCResult::Type::ARR, "", "",
889 {
890 {RPCResult::Type::OBJ, "", "",
891 Cat<std::vector<RPCResult>>(
893 {{RPCResult::Type::STR_HEX, "hex", "The serialized, hex-encoded transaction data"}}
894 )
895 },
896 }},
897 },
899 HelpExampleCli("getorphantxs", "2")
900 + HelpExampleRpc("getorphantxs", "2")
901 },
902 [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
903 {
904 const NodeContext& node = EnsureAnyNodeContext(request.context);
905 PeerManager& peerman = EnsurePeerman(node);
906 std::vector<node::TxOrphanage::OrphanInfo> orphanage = peerman.GetOrphanTransactions();
907
908 int verbosity{ParseVerbosity(request.params[0], /*default_verbosity=*/0, /*allow_bool*/false)};
909
911
912 if (verbosity == 0) {
913 for (auto const& orphan : orphanage) {
914 ret.push_back(orphan.tx->GetHash().ToString());
915 }
916 } else if (verbosity == 1) {
917 for (auto const& orphan : orphanage) {
918 ret.push_back(OrphanToJSON(orphan));
919 }
920 } else if (verbosity == 2) {
921 for (auto const& orphan : orphanage) {
922 UniValue o{OrphanToJSON(orphan)};
923 o.pushKV("hex", EncodeHexTx(*orphan.tx));
924 ret.push_back(o);
925 }
926 } else {
927 throw JSONRPCError(RPC_INVALID_PARAMETER, "Invalid verbosity value " + ToString(verbosity));
928 }
929
930 return ret;
931 },
932 };
933}
934
936{
937 return RPCHelpMan{"submitpackage",
938 "Submit a package of raw transactions (serialized, hex-encoded) to local node.\n"
939 "The package will be validated according to consensus and mempool policy rules. If any transaction passes, it will be accepted to mempool.\n"
940 "This RPC is experimental and the interface may be unstable. Refer to doc/policy/packages.md for documentation on package policies.\n"
941 "Warning: successful submission does not mean the transactions will propagate throughout the network.\n"
942 ,
943 {
944 {"package", RPCArg::Type::ARR, RPCArg::Optional::NO, "An array of raw transactions.\n"
945 "The package must consist of a transaction with (some, all, or none of) its unconfirmed parents. A single transaction is permitted.\n"
946 "None of the parents may depend on each other. Parents that are already in mempool do not need to be present in the package.\n"
947 "The package must be topologically sorted, with the child being the last element in the array if there are multiple elements.",
948 {
950 },
951 },
953 "Reject transactions whose fee rate is higher than the specified value, expressed in " + CURRENCY_UNIT +
954 "/kvB.\nFee rates larger than 1BTC/kvB are rejected.\nSet to 0 to accept any fee rate."},
956 "Reject transactions with provably unspendable outputs (e.g. 'datacarrier' outputs that use the OP_RETURN opcode) greater than the specified value, expressed in " + CURRENCY_UNIT + ".\n"
957 "If burning funds through unspendable outputs is desired, increase this value.\n"
958 "This check is based on heuristics and does not guarantee spendability of outputs.\n"
959 },
960 },
961 RPCResult{
962 RPCResult::Type::OBJ, "", "",
963 {
964 {RPCResult::Type::STR, "package_msg", "The transaction package result message. \"success\" indicates all transactions were accepted into or are already in the mempool."},
965 {RPCResult::Type::OBJ_DYN, "tx-results", "The transaction results keyed by wtxid. An entry is returned for every submitted wtxid.",
966 {
967 {RPCResult::Type::OBJ, "wtxid", "transaction wtxid", {
968 {RPCResult::Type::STR_HEX, "txid", "The transaction hash in hex"},
969 {RPCResult::Type::STR_HEX, "other-wtxid", /*optional=*/true, "The wtxid of a different transaction with the same txid but different witness found in the mempool. This means the submitted transaction was ignored."},
970 {RPCResult::Type::NUM, "vsize", /*optional=*/true, "Sigops-adjusted virtual transaction size."},
971 {RPCResult::Type::OBJ, "fees", /*optional=*/true, "Transaction fees", {
972 {RPCResult::Type::STR_AMOUNT, "base", "transaction fee in " + CURRENCY_UNIT},
973 {RPCResult::Type::STR_AMOUNT, "effective-feerate", /*optional=*/true, "if the transaction was not already in the mempool, the effective feerate in " + CURRENCY_UNIT + " per KvB. For example, the package feerate and/or feerate with modified fees from prioritisetransaction."},
974 {RPCResult::Type::ARR, "effective-includes", /*optional=*/true, "if effective-feerate is provided, the wtxids of the transactions whose fees and vsizes are included in effective-feerate.",
975 {{RPCResult::Type::STR_HEX, "", "transaction wtxid in hex"},
976 }},
977 }},
978 {RPCResult::Type::STR, "error", /*optional=*/true, "Error string if rejected from mempool, or \"package-not-validated\" when the package aborts before any per-tx processing."},
979 }}
980 }},
981 {RPCResult::Type::ARR, "replaced-transactions", /*optional=*/true, "List of txids of replaced transactions",
982 {
983 {RPCResult::Type::STR_HEX, "", "The transaction id"},
984 }},
985 },
986 },
988 HelpExampleRpc("submitpackage", R"(["raw-parent-tx-1", "raw-parent-tx-2", "raw-child-tx"])") +
989 HelpExampleCli("submitpackage", R"('["raw-tx-without-unconfirmed-parents"]')")
990 },
991 [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
992 {
993 const UniValue raw_transactions = request.params[0].get_array();
994 if (raw_transactions.empty() || raw_transactions.size() > MAX_PACKAGE_COUNT) {
996 "Array must contain between 1 and " + ToString(MAX_PACKAGE_COUNT) + " transactions.");
997 }
998
999 // Fee check needs to be run with chainstate and package context
1000 const CFeeRate max_raw_tx_fee_rate{ParseFeeRate(self.Arg<UniValue>("maxfeerate"))};
1001 std::optional<CFeeRate> client_maxfeerate{max_raw_tx_fee_rate};
1002 // 0-value is special; it's mapped to no sanity check
1003 if (max_raw_tx_fee_rate == CFeeRate(0)) {
1004 client_maxfeerate = std::nullopt;
1005 }
1006
1007 // Burn sanity check is run with no context
1008 const CAmount max_burn_amount = request.params[2].isNull() ? 0 : AmountFromValue(request.params[2]);
1009
1010 std::vector<CTransactionRef> txns;
1011 txns.reserve(raw_transactions.size());
1012 for (const auto& rawtx : raw_transactions.getValues()) {
1014 if (!DecodeHexTx(mtx, rawtx.get_str())) {
1016 "TX decode failed: " + rawtx.get_str() + " Make sure the tx has at least one input.");
1017 }
1018
1019 for (const auto& out : mtx.vout) {
1020 if((out.scriptPubKey.IsUnspendable() || !out.scriptPubKey.HasValidOps()) && out.nValue > max_burn_amount) {
1021 throw JSONRPCTransactionError(TransactionError::MAX_BURN_EXCEEDED);
1022 }
1023 }
1024
1025 txns.emplace_back(MakeTransactionRef(std::move(mtx)));
1026 }
1027 CHECK_NONFATAL(!txns.empty());
1028 if (txns.size() > 1 && !IsChildWithParentsTree(txns)) {
1029 throw JSONRPCTransactionError(TransactionError::INVALID_PACKAGE, "package topology disallowed. not child-with-parents or parents depend on each other.");
1030 }
1031
1032 NodeContext& node = EnsureAnyNodeContext(request.context);
1033 CTxMemPool& mempool = EnsureMemPool(node);
1035 const auto package_result = WITH_LOCK(::cs_main, return ProcessNewPackage(chainstate, mempool, txns, /*test_accept=*/ false, client_maxfeerate));
1036
1037 std::string package_msg = "success";
1038
1039 // First catch package-wide errors, continue if we can
1040 switch(package_result.m_state.GetResult()) {
1042 {
1043 // Belt-and-suspenders check; everything should be successful here
1044 CHECK_NONFATAL(package_result.m_tx_results.size() == txns.size());
1045 for (const auto& tx : txns) {
1046 CHECK_NONFATAL(mempool.exists(tx->GetHash()));
1047 }
1048 break;
1049 }
1051 {
1052 // This only happens with internal bug; user should stop and report
1053 throw JSONRPCTransactionError(TransactionError::MEMPOOL_ERROR,
1054 package_result.m_state.GetRejectReason());
1055 }
1058 {
1059 // Package-wide error we want to return, but we also want to return individual responses
1060 package_msg = package_result.m_state.ToString();
1061 CHECK_NONFATAL(package_result.m_tx_results.size() == txns.size() ||
1062 package_result.m_tx_results.empty());
1063 break;
1064 }
1065 }
1066
1067 size_t num_broadcast{0};
1068 for (const auto& tx : txns) {
1069 // We don't want to re-submit the txn for validation in BroadcastTransaction
1070 if (!mempool.exists(tx->GetHash())) {
1071 continue;
1072 }
1073
1074 // We do not expect an error here; we are only broadcasting things already/still in mempool
1075 std::string err_string;
1076 const auto err = BroadcastTransaction(node,
1077 tx,
1078 err_string,
1079 /*max_tx_fee=*/0,
1081 /*wait_callback=*/true);
1082 if (err != TransactionError::OK) {
1083 throw JSONRPCTransactionError(err,
1084 strprintf("transaction broadcast failed: %s (%d transactions were broadcast successfully)",
1085 err_string, num_broadcast));
1086 }
1087 num_broadcast++;
1088 }
1089
1090 UniValue rpc_result{UniValue::VOBJ};
1091 rpc_result.pushKV("package_msg", package_msg);
1092 UniValue tx_result_map{UniValue::VOBJ};
1093 std::set<Txid> replaced_txids;
1094 for (const auto& tx : txns) {
1095 UniValue result_inner{UniValue::VOBJ};
1096 result_inner.pushKV("txid", tx->GetHash().GetHex());
1097 const auto wtxid_hex = tx->GetWitnessHash().GetHex();
1098 auto it = package_result.m_tx_results.find(tx->GetWitnessHash());
1099 if (it == package_result.m_tx_results.end()) {
1100 // No per-tx result for this wtxid
1101 // Current invariant: per-tx results are all-or-none (every member or empty on package abort).
1102 // If any exist yet this one is missing, it's an unexpected partial map.
1103 CHECK_NONFATAL(package_result.m_tx_results.empty());
1104 result_inner.pushKV("error", "package-not-validated");
1105 tx_result_map.pushKV(wtxid_hex, std::move(result_inner));
1106 continue;
1107 }
1108 const auto& tx_result = it->second;
1109 switch(it->second.m_result_type) {
1111 result_inner.pushKV("other-wtxid", it->second.m_other_wtxid.value().GetHex());
1112 break;
1114 result_inner.pushKV("error", it->second.m_state.ToString());
1115 break;
1118 result_inner.pushKV("vsize", int64_t{it->second.m_vsize.value()});
1120 fees.pushKV("base", ValueFromAmount(it->second.m_base_fees.value()));
1121 if (tx_result.m_result_type == MempoolAcceptResult::ResultType::VALID) {
1122 // Effective feerate is not provided for MEMPOOL_ENTRY transactions even
1123 // though modified fees is known, because it is unknown whether package
1124 // feerate was used when it was originally submitted.
1125 fees.pushKV("effective-feerate", ValueFromAmount(tx_result.m_effective_feerate.value().GetFeePerK()));
1126 UniValue effective_includes_res(UniValue::VARR);
1127 for (const auto& wtxid : tx_result.m_wtxids_fee_calculations.value()) {
1128 effective_includes_res.push_back(wtxid.ToString());
1129 }
1130 fees.pushKV("effective-includes", std::move(effective_includes_res));
1131 }
1132 result_inner.pushKV("fees", std::move(fees));
1133 for (const auto& ptx : it->second.m_replaced_transactions) {
1134 replaced_txids.insert(ptx->GetHash());
1135 }
1136 break;
1137 }
1138 tx_result_map.pushKV(wtxid_hex, std::move(result_inner));
1139 }
1140 rpc_result.pushKV("tx-results", std::move(tx_result_map));
1141 UniValue replaced_list(UniValue::VARR);
1142 for (const auto& txid : replaced_txids) replaced_list.push_back(txid.ToString());
1143 rpc_result.pushKV("replaced-transactions", std::move(replaced_list));
1144 return rpc_result;
1145 },
1146 };
1147}
1148
1150{
1151 static const CRPCCommand commands[]{
1152 {"rawtransactions", &sendrawtransaction},
1153 {"rawtransactions", &testmempoolaccept},
1154 {"blockchain", &getmempoolancestors},
1155 {"blockchain", &getmempooldescendants},
1156 {"blockchain", &getmempoolentry},
1157 {"blockchain", &gettxspendingprevout},
1158 {"blockchain", &getmempoolinfo},
1159 {"blockchain", &getrawmempool},
1160 {"blockchain", &importmempool},
1161 {"blockchain", &savemempool},
1162 {"hidden", &getorphantxs},
1163 {"rawtransactions", &submitpackage},
1164 };
1165 for (const auto& c : commands) {
1166 t.appendCommand(c.name, &c);
1167 }
1168}
int64_t CAmount
Amount in satoshis (Can be negative)
Definition: amount.h:12
int ret
static CAmount AmountFromValue(const UniValue &value)
Definition: bitcoin-tx.cpp:554
ArgsManager & args
Definition: bitcoind.cpp:277
#define CHECK_NONFATAL(condition)
Identity function.
Definition: check.h:109
Fee rate in satoshis per virtualbyte: CAmount / vB the feerate is represented internally as FeeFrac.
Definition: feerate.h:35
CAmount GetFeePerK() const
Return the fee in satoshis for a vsize of 1000 vbytes.
Definition: feerate.h:63
An outpoint - a combination of a transaction hash and an index n into its vout.
Definition: transaction.h:29
Txid hash
Definition: transaction.h:31
RPC command dispatcher.
Definition: server.h:87
The basic transaction that is broadcasted on the network and contained in blocks.
Definition: transaction.h:296
const Txid & GetHash() const LIFETIMEBOUND
Definition: transaction.h:343
const std::vector< CTxIn > vin
Definition: transaction.h:306
An input of a transaction.
Definition: transaction.h:67
COutPoint prevout
Definition: transaction.h:69
CTxMemPoolEntry stores data about the corresponding transaction, as well as data about all in-mempool...
Definition: mempool_entry.h:66
const CTransaction & GetTx() const
const Children & GetMemPoolChildrenConst() const
CTxMemPool stores valid-according-to-the-current-best-chain transactions that may be included in the ...
Definition: txmempool.h:281
setEntries AssumeCalculateMemPoolAncestors(std::string_view calling_fn_name, const CTxMemPoolEntry &entry, const Limits &limits, bool fSearchForParents=true) const EXCLUSIVE_LOCKS_REQUIRED(cs)
Same as CalculateMemPoolAncestors, but always returns a (non-optional) setEntries.
Definition: txmempool.cpp:275
std::optional< txiter > GetIter(const Txid &txid) const EXCLUSIVE_LOCKS_REQUIRED(cs)
Returns an iterator to the given hash, if found.
Definition: txmempool.cpp:962
bool GetLoadTried() const
Definition: txmempool.cpp:1218
RecursiveMutex cs
This mutex needs to be locked when accessing mapTx or other members that are guarded by it.
Definition: txmempool.h:367
CFeeRate GetMinFee(size_t sizelimit) const
Definition: txmempool.cpp:1108
size_t DynamicMemoryUsage() const
Definition: txmempool.cpp:1045
const Options m_opts
Definition: txmempool.h:421
bool exists(const Txid &txid) const
Definition: txmempool.h:630
std::set< txiter, CompareIteratorByHash > setEntries
Definition: txmempool.h:373
uint64_t GetSequence() const EXCLUSIVE_LOCKS_REQUIRED(cs)
Definition: txmempool.h:699
indexed_transaction_set::nth_index< 0 >::type::const_iterator txiter
Definition: txmempool.h:370
const CTransaction * GetConflictTx(const COutPoint &prevout) const EXCLUSIVE_LOCKS_REQUIRED(cs)
Get the transaction in the pool that spends the same prevout.
Definition: txmempool.cpp:956
std::set< Txid > GetUnbroadcastTxs() const
Returns transactions in unbroadcast set.
Definition: txmempool.h:681
void CalculateDescendants(txiter it, setEntries &setDescendants) const EXCLUSIVE_LOCKS_REQUIRED(cs)
Populate setDescendants with all in-mempool descendants of hash.
Definition: txmempool.cpp:572
unsigned long size() const
Definition: txmempool.h:612
std::vector< CTxMemPoolEntryRef > entryAll() const EXCLUSIVE_LOCKS_REQUIRED(cs)
Definition: txmempool.cpp:846
CAmount GetTotalFee() const EXCLUSIVE_LOCKS_REQUIRED(cs)
Definition: txmempool.h:624
uint64_t GetTotalTxSize() const EXCLUSIVE_LOCKS_REQUIRED(cs)
Definition: txmempool.h:618
const CTxMemPoolEntry * GetEntry(const Txid &txid) const LIFETIMEBOUND EXCLUSIVE_LOCKS_REQUIRED(cs)
Definition: txmempool.cpp:872
Chainstate stores and provides an API to update our local knowledge of the current best chain.
Definition: validation.h:532
Provides an interface for creating and interacting with one or two chainstates: an IBD chainstate gen...
Definition: validation.h:899
SnapshotCompletionResult MaybeCompleteSnapshotValidation() EXCLUSIVE_LOCKS_REQUIRED(const CBlockIndex *GetSnapshotBaseBlock() const EXCLUSIVE_LOCKS_REQUIRED(Chainstate ActiveChainstate)() const
Once the background validation chainstate has reached the height which is the base of the UTXO snapsh...
Definition: validation.h:1125
MempoolAcceptResult ProcessTransaction(const CTransactionRef &tx, bool test_accept=false) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Try to add a transaction to the memory pool.
bool IsInitialBlockDownload() const
Check whether we are doing an initial block download (synchronizing from disk or network)
virtual std::vector< node::TxOrphanage::OrphanInfo > GetOrphanTransactions()=0
const std::string m_name
Definition: util.h:496
auto Arg(std::string_view key) const
Helper to get a required or default-valued request argument.
Definition: util.h:444
void push_back(UniValue val)
Definition: univalue.cpp:104
const UniValue & find_value(std::string_view key) const
Definition: univalue.cpp:233
@ VOBJ
Definition: univalue.h:24
@ VSTR
Definition: univalue.h:24
@ VARR
Definition: univalue.h:24
@ VNUM
Definition: univalue.h:24
bool isNull() const
Definition: univalue.h:81
const UniValue & get_obj() const
size_t size() const
Definition: univalue.h:71
const std::vector< UniValue > & getValues() const
bool empty() const
Definition: univalue.h:69
void pushKVEnd(std::string key, UniValue val)
Definition: univalue.cpp:118
Int getInt() const
Definition: univalue.h:140
const UniValue & get_array() const
void pushKV(std::string key, UniValue val)
Definition: univalue.cpp:126
bool get_bool() const
std::string GetRejectReason() const
Definition: validation.h:109
Result GetResult() const
Definition: validation.h:108
std::string ToString() const
Definition: validation.h:111
std::string ToString() const
static transaction_identifier FromUint256(const uint256 &id)
static int32_t GetTransactionWeight(const CTransaction &tx)
Definition: validation.h:132
@ TX_MISSING_INPUTS
transaction was missing some of its inputs
std::string EncodeHexTx(const CTransaction &tx)
Definition: core_write.cpp:143
UniValue ValueFromAmount(const CAmount amount)
Definition: core_write.cpp:26
bool DecodeHexTx(CMutableTransaction &tx, const std::string &hex_tx, bool try_no_witness=false, bool try_witness=true)
Definition: core_read.cpp:196
RecursiveMutex cs_main
Mutex to guard access to validation specific variables, such as reading or changing the chainstate.
Definition: cs_main.cpp:8
const std::string CURRENCY_UNIT
Definition: feerate.h:18
static path u8path(std::string_view utf8_str)
Definition: fs.h:81
uint64_t fee
std::string FormatMoney(const CAmount n)
Money parsing/formatting utilities.
Definition: moneystr.cpp:19
Definition: messages.h:21
static const CAmount DEFAULT_MAX_BURN_AMOUNT
Maximum burn value for sendrawtransaction, submitpackage, and testmempoolaccept RPC calls.
Definition: transaction.h:34
@ MEMPOOL_AND_BROADCAST_TO_ALL
Add the transaction to the mempool and broadcast to all peers for which tx relay is enabled.
TransactionError
Definition: types.h:25
fs::path MempoolPath(const ArgsManager &argsman)
static const CFeeRate DEFAULT_MAX_RAW_TX_FEE_RATE
Maximum fee rate for sendrawtransaction and testmempoolaccept RPC calls.
Definition: transaction.h:28
TransactionError BroadcastTransaction(NodeContext &node, const CTransactionRef tx, std::string &err_string, const CAmount &max_tx_fee, TxBroadcast broadcast_method, bool wait_callback)
Submit a transaction to the mempool and (optionally) relay it to all P2P peers.
Definition: transaction.cpp:34
bool LoadMempool(CTxMemPool &pool, const fs::path &load_path, Chainstate &active_chainstate, ImportMempoolOptions &&opts)
Import the file and attempt to add its contents to the mempool.
bool DumpMempool(const CTxMemPool &pool, const fs::path &dump_path, FopenFn mockable_fopen_function, bool skip_file_commit)
std::string ToString(const T &t)
Locale-independent version of std::to_string.
Definition: string.h:245
is a home for public enum and struct type definitions that are used internally by node code,...
bool IsChildWithParentsTree(const Package &package)
Context-free check that a package IsChildWithParents() and none of the parents depend on each other (...
Definition: packages.cpp:136
static constexpr uint32_t MAX_PACKAGE_COUNT
Default maximum number of transactions in a package.
Definition: packages.h:19
@ PCKG_POLICY
The package itself is invalid (e.g. too many transactions).
@ PCKG_RESULT_UNSET
Initial value. The package has not yet been rejected.
@ PCKG_MEMPOOL_ERROR
Mempool logic error.
@ PCKG_TX
At least one tx is invalid.
RBFTransactionState IsRBFOptIn(const CTransaction &tx, const CTxMemPool &pool)
Determine whether an unconfirmed transaction is signaling opt-in to RBF according to BIP 125 This inv...
Definition: rbf.cpp:24
RBFTransactionState
The rbf state of unconfirmed transactions.
Definition: rbf.h:29
@ UNKNOWN
Unconfirmed tx that does not signal rbf and is not in the mempool.
@ REPLACEABLE_BIP125
Either this tx or a mempool ancestor signals rbf.
int64_t GetVirtualTransactionSize(int64_t nWeight, int64_t nSigOpCost, unsigned int bytes_per_sigop)
Compute the virtual transaction size (weight reinterpreted as bytes).
Definition: policy.cpp:376
static CTransactionRef MakeTransactionRef(Tx &&txIn)
Definition: transaction.h:424
std::shared_ptr< const CTransaction > CTransactionRef
Definition: transaction.h:423
UniValue JSONRPCError(int code, const std::string &message)
Definition: request.cpp:70
static UniValue OrphanToJSON(const node::TxOrphanage::OrphanInfo &orphan)
Definition: mempool.cpp:850
static RPCHelpMan getmempoolinfo()
Definition: mempool.cpp:703
static RPCHelpMan sendrawtransaction()
Definition: mempool.cpp:43
static std::vector< RPCResult > OrphanDescription()
Definition: mempool.cpp:835
static void entryToJSON(const CTxMemPool &pool, UniValue &info, const CTxMemPoolEntry &e) EXCLUSIVE_LOCKS_REQUIRED(pool.cs)
Definition: mempool.cpp:295
static RPCHelpMan importmempool()
Definition: mempool.cpp:736
void RegisterMempoolRPCCommands(CRPCTable &t)
Definition: mempool.cpp:1149
static RPCHelpMan getrawmempool()
Definition: mempool.cpp:390
static RPCHelpMan getmempoolentry()
Definition: mempool.cpp:567
UniValue MempoolInfoToJSON(const CTxMemPool &pool)
Mempool information to JSON.
Definition: mempool.cpp:682
static RPCHelpMan gettxspendingprevout()
Definition: mempool.cpp:600
static std::vector< RPCResult > MempoolEntryDescription()
Definition: mempool.cpp:267
static RPCHelpMan submitpackage()
Definition: mempool.cpp:935
static RPCHelpMan getorphantxs()
Definition: mempool.cpp:866
UniValue MempoolToJSON(const CTxMemPool &pool, bool verbose, bool include_mempool_sequence)
Mempool to JSON.
Definition: mempool.cpp:352
static RPCHelpMan testmempoolaccept()
Definition: mempool.cpp:116
static RPCHelpMan getmempooldescendants()
Definition: mempool.cpp:502
static RPCHelpMan getmempoolancestors()
Definition: mempool.cpp:441
static RPCHelpMan savemempool()
Definition: mempool.cpp:797
@ RPC_MISC_ERROR
General application defined errors.
Definition: protocol.h:40
@ RPC_INVALID_PARAMETER
Invalid, missing or duplicate parameter.
Definition: protocol.h:44
@ RPC_CLIENT_IN_INITIAL_DOWNLOAD
Still downloading initial blocks.
Definition: protocol.h:59
@ RPC_DESERIALIZATION_ERROR
Error parsing or validating structure in raw format.
Definition: protocol.h:46
@ RPC_INVALID_ADDRESS_OR_KEY
Invalid address or key.
Definition: protocol.h:42
std::string HelpExampleCli(const std::string &methodname, const std::string &args)
Definition: util.cpp:183
UniValue JSONRPCTransactionError(TransactionError terr, const std::string &err_string)
Definition: util.cpp:408
CFeeRate ParseFeeRate(const UniValue &json)
Parse a json number or string, denoting BTC/kvB, into a CFeeRate (sat/kvB).
Definition: util.cpp:110
std::string HelpExampleRpc(const std::string &methodname, const std::string &args)
Definition: util.cpp:201
int ParseVerbosity(const UniValue &arg, int default_verbosity, bool allow_bool)
Parses verbosity from provided UniValue.
Definition: util.cpp:83
uint256 ParseHashO(const UniValue &o, std::string_view strKey)
Definition: util.cpp:126
void RPCTypeCheckObj(const UniValue &o, const std::map< std::string, UniValueType > &typesExpected, bool fAllowNull, bool fStrict)
Definition: util.cpp:56
uint256 ParseHashV(const UniValue &v, std::string_view name)
Utilities: convert hex-encoded Values (throws error if not hex).
Definition: util.cpp:117
NodeContext & EnsureAnyNodeContext(const std::any &context)
Definition: server_util.cpp:25
CTxMemPool & EnsureMemPool(const NodeContext &node)
Definition: server_util.cpp:34
PeerManager & EnsurePeerman(const NodeContext &node)
ChainstateManager & EnsureChainman(const NodeContext &node)
Definition: server_util.cpp:74
CTxMemPool & EnsureAnyMemPool(const std::any &context)
Definition: server_util.cpp:42
ArgsManager & EnsureAnyArgsman(const std::any &context)
Definition: server_util.cpp:69
A mutable version of CTransaction.
Definition: transaction.h:378
std::vector< CTxOut > vout
Definition: transaction.h:380
@ DIFFERENT_WITNESS
‍Valid, transaction was already in the mempool.
@ INVALID
‍Fully validated, valid.
Validation result for package mempool acceptance.
Definition: validation.h:233
PackageValidationState m_state
Definition: validation.h:234
std::map< Wtxid, MempoolAcceptResult > m_tx_results
Map from wtxid to finished MempoolAcceptResults.
Definition: validation.h:241
@ STR_HEX
Special type that is a STR with only hex chars.
@ AMOUNT
Special type representing a floating point amount (can be either NUM or STR)
@ OBJ_NAMED_PARAMS
Special type that behaves almost exactly like OBJ, defining an options object with a list of pre-defi...
@ OMITTED
Optional argument for which the default value is omitted from help text for one of two reasons:
@ NO
Required arg.
std::string oneline_description
Should be empty unless it is supposed to override the auto-generated summary line.
Definition: util.h:171
bool skip_type_check
Definition: util.h:170
@ NUM_TIME
Special numeric to denote unix epoch time.
@ OBJ_DYN
Special dictionary with keys that are not literals.
@ STR_HEX
Special string with only hex chars.
@ STR_AMOUNT
Special string to represent a floating point amount.
Wrapper for UniValue::VType, which includes typeAny: Used to denote don't care type.
Definition: util.h:83
static constexpr MemPoolLimits NoLimits()
std::optional< unsigned > max_datacarrier_bytes
A data carrying output is an unspendable output containing data.
CFeeRate incremental_relay_feerate
CFeeRate min_relay_feerate
A fee rate smaller than this is considered zero fee (for relaying, mining and transaction creation)
NodeContext struct containing references to chain state and connection state.
Definition: context.h:56
Allows providing orphan information externally.
Definition: txorphanage.h:44
std::set< NodeId > announcers
Peers added with AddTx or AddAnnouncer.
Definition: txorphanage.h:47
#define AssertLockNotHeld(cs)
Definition: sync.h:142
#define LOCK(cs)
Definition: sync.h:259
#define WITH_LOCK(cs, code)
Run code while locking a mutex.
Definition: sync.h:290
#define EXCLUSIVE_LOCKS_REQUIRED(...)
Definition: threadsafety.h:51
#define strprintf
Format arguments and return the string or write to given std::ostream (see tinyformat::format doc for...
Definition: tinyformat.h:1172
constexpr int64_t count_seconds(std::chrono::seconds t)
Definition: time.h:82
PackageMempoolAcceptResult ProcessNewPackage(Chainstate &active_chainstate, CTxMemPool &pool, const Package &package, bool test_accept, const std::optional< CFeeRate > &client_maxfeerate)
Validate (and maybe submit) a package to the mempool.
AssertLockHeld(pool.cs)