10 #include <chainparams.h> 26 #include <txmempool.h> 31 #include <util/time.h> 35 #include <string_view> 51 "Submit a raw transaction (serialized, hex-encoded) to the network.\n" 53 "\nIf -privatebroadcast is disabled, then the transaction will be put into the\n" 54 "local mempool of the node and will be sent unconditionally to all currently\n" 55 "connected peers, so using sendrawtransaction for manual rebroadcast will degrade\n" 56 "privacy by leaking the transaction's origin, as nodes will normally not\n" 57 "rebroadcast non-wallet transactions already in their mempool.\n" 59 "\nIf -privatebroadcast is enabled, then the transaction will be sent only via\n" 60 "dedicated, short-lived connections to Tor or I2P peers or IPv4/IPv6 peers\n" 61 "via the Tor network. This conceals the transaction's origin. The transaction\n" 62 "will only enter the local mempool when it is received back from the network.\n" 64 "\nA specific exception, RPC_TRANSACTION_ALREADY_IN_UTXO_SET, may throw if the transaction cannot be added to the mempool.\n" 66 "\nRelated RPCs: createrawtransaction, signrawtransactionwithkey\n",
70 "Reject transactions whose fee rate is higher than the specified value, expressed in " +
CURRENCY_UNIT +
71 "/kvB.\nFee rates larger than 1BTC/kvB are rejected.\nSet to 0 to accept any fee rate."},
73 "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" 74 "If burning funds through unspendable outputs is desired, increase this value.\n" 75 "This check is based on heuristics and does not guarantee spendability of outputs.\n"},
81 "\nCreate a transaction\n" 82 +
HelpExampleCli(
"createrawtransaction",
"\"[{\\\"txid\\\" : \\\"mytxid\\\",\\\"vout\\\":0}]\" \"{\\\"myaddress\\\":0.01}\"") +
83 "Sign the transaction, and get back the hex\n" 85 "\nSend the transaction (signed hex)\n" 87 "\nAs a JSON-RPC call\n" 95 if (!
DecodeHexTx(mtx, request.params[0].get_str())) {
99 for (
const auto&
out : mtx.
vout) {
100 if((
out.scriptPubKey.IsUnspendable() || !
out.scriptPubKey.HasValidOps()) &&
out.nValue > max_burn_amount) {
110 CAmount max_raw_tx_fee = max_raw_tx_fee_rate.GetFee(virtual_size);
112 std::string err_string;
116 if (private_broadcast_enabled &&
120 "-privatebroadcast is enabled, but none of the Tor or I2P networks is " 121 "reachable. Maybe the location of the Tor proxy couldn't be retrieved " 122 "from the Tor daemon at startup. Check whether the Tor daemon is running " 123 "and that -torcontrol, -torpassword and -i2psam are configured properly.");
133 if (TransactionError::OK != err) {
137 return tx->GetHash().GetHex();
145 "getprivatebroadcastinfo",
146 "Returns information about transactions that are currently being privately broadcast.\n",
158 {
RPCResult::Type::ARR,
"peers",
"Per-peer send and acknowledgment information for this transaction",
163 {
RPCResult::Type::NUM_TIME,
"sent",
"The time this transaction was picked for sending to this peer via private broadcast (seconds since epoch)"},
164 {
RPCResult::Type::NUM_TIME,
"received",
true,
"The time this peer acknowledged reception of the transaction (seconds since epoch)"},
181 for (
const auto& tx_info : txs) {
183 o.
pushKV(
"txid", tx_info.tx->GetHash().ToString());
184 o.
pushKV(
"wtxid", tx_info.tx->GetWitnessHash().ToString());
187 for (
const auto& peer : tx_info.peers) {
189 p.
pushKV(
"address", peer.address.ToStringAddrPort());
190 p.
pushKV(
"sent", TicksSinceEpoch<std::chrono::seconds>(peer.sent));
191 if (peer.received.has_value()) {
192 p.
pushKV(
"received", TicksSinceEpoch<std::chrono::seconds>(*peer.received));
196 o.
pushKV(
"peers", std::move(peers));
201 ret.pushKV(
"transactions", std::move(transactions));
210 "abortprivatebroadcast",
211 "Abort private broadcast attempts for a transaction currently being privately broadcast.\n" 212 "The transaction will be removed from the private broadcast queue.\n",
215 "If the provided id matches a txid that corresponds to multiple transactions with different wtxids, multiple transactions will be removed and returned."},
220 {
RPCResult::Type::ARR,
"removed_transactions",
"Transactions removed from the private broadcast queue",
243 if (removed_txs.empty()) {
248 for (
const auto& tx : removed_txs) {
250 o.
pushKV(
"txid", tx->GetHash().ToString());
251 o.
pushKV(
"wtxid", tx->GetWitnessHash().ToString());
253 removed_transactions.
push_back(std::move(o));
256 ret.pushKV(
"removed_transactions", std::move(removed_transactions));
266 "Returns result of mempool acceptance tests indicating if raw transaction(s) (serialized, hex-encoded) would be accepted by mempool.\n" 267 "\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" 268 "\nIf one transaction fails, other transactions may not be fully validated (the 'allowed' key will be blank).\n" 270 "\nThis checks if transactions violate the consensus or policy rules.\n" 271 "\nSee sendrawtransaction call.\n",
279 "Reject transactions whose fee rate is higher than the specified value, expressed in " +
CURRENCY_UNIT +
280 "/kvB.\nFee rates larger than 1BTC/kvB are rejected.\nSet to 0 to accept any fee rate."},
283 RPCResult::Type::ARR,
"",
"The result of the mempool acceptance test for each raw transaction in the input array.\n" 284 "Returns results for each transaction in the same order they were passed in.\n" 285 "Transactions that cannot be fully validated due to failures in other transactions will not contain an 'allowed' result.\n",
291 {
RPCResult::Type::STR,
"package-error",
true,
"Package validation error, if any (only possible if rawtxs had more than 1 transaction)."},
292 {
RPCResult::Type::BOOL,
"allowed",
true,
"Whether this tx would be accepted to the mempool and pass client-specified maxfeerate. " 293 "If not present, the tx was not fully validated due to a failure in another tx in the list."},
294 {
RPCResult::Type::NUM,
"vsize",
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)"},
298 {
RPCResult::Type::STR_AMOUNT,
"effective-feerate",
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."},
299 {
RPCResult::Type::ARR,
"effective-includes",
false,
"transactions whose fees and vsizes are included in effective-feerate.",
303 {
RPCResult::Type::STR,
"reject-reason",
true,
"Rejection reason (only present when 'allowed' is false)"},
304 {
RPCResult::Type::STR,
"reject-details",
true,
"Rejection details (only present when 'allowed' is false and rejection details exist)"},
309 "\nCreate a transaction\n" 310 +
HelpExampleCli(
"createrawtransaction",
"\"[{\\\"txid\\\" : \\\"mytxid\\\",\\\"vout\\\":0}]\" \"{\\\"myaddress\\\":0.01}\"") +
311 "Sign the transaction, and get back the hex\n" 313 "\nTest acceptance of the transaction (signed hex)\n" 315 "\nAs a JSON-RPC call\n" 328 std::vector<CTransactionRef> txns;
329 txns.reserve(raw_transactions.
size());
330 for (
const auto& rawtx : raw_transactions.
getValues()) {
334 "TX decode failed: " + rawtx.get_str() +
" Make sure the tx has at least one input.");
345 if (txns.size() > 1)
return ProcessNewPackage(chainstate, mempool, txns,
true, {});
355 bool exit_early{
false};
356 for (
const auto& tx : txns) {
358 result_inner.
pushKV(
"txid", tx->GetHash().GetHex());
359 result_inner.
pushKV(
"wtxid", tx->GetWitnessHash().GetHex());
363 auto it = package_result.
m_tx_results.find(tx->GetWitnessHash());
364 if (exit_early || it == package_result.
m_tx_results.end()) {
366 rpc_result.
push_back(std::move(result_inner));
369 const auto& tx_result = it->second;
373 const CAmount fee = tx_result.m_base_fees.value();
375 const int64_t virtual_size = tx_result.m_vsize.value();
376 const CAmount max_raw_tx_fee = max_raw_tx_fee_rate.GetFee(virtual_size);
377 if (max_raw_tx_fee &&
fee > max_raw_tx_fee) {
378 result_inner.
pushKV(
"allowed",
false);
379 result_inner.
pushKV(
"reject-reason",
"max-fee-exceeded");
384 result_inner.
pushKV(
"allowed",
true);
385 result_inner.
pushKV(
"vsize", virtual_size);
388 fees.
pushKV(
"effective-feerate",
ValueFromAmount(tx_result.m_effective_feerate.value().GetFeePerK()));
390 for (
const auto& wtxid : tx_result.m_wtxids_fee_calculations.value()) {
391 effective_includes_res.
push_back(wtxid.ToString());
393 fees.
pushKV(
"effective-includes", std::move(effective_includes_res));
394 result_inner.
pushKV(
"fees", std::move(fees));
397 result_inner.
pushKV(
"allowed",
false);
400 result_inner.
pushKV(
"reject-reason",
"missing-inputs");
406 rpc_result.
push_back(std::move(result_inner));
434 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."},
442 RPCResult{
RPCResult::Type::NUM,
"chunkweight",
"sigops-adjusted weight (as defined in BIP 141 and modified by '-bytespersigop') of this transaction's chunk"},
456 RPCResult{
RPCResult::Type::BOOL,
"bip125-replaceable",
"Whether this transaction signals BIP125 replaceability or has an unconfirmed ancestor signaling BIP125 replaceability. (DEPRECATED)\n"},
457 RPCResult{
RPCResult::Type::BOOL,
"unbroadcast",
"Whether this transaction is currently unbroadcast (initial broadcast not yet acknowledged by any peers)"},
465 chunk.
pushKV(
"chunkweight", chunk_feerate.
size);
467 for (
const auto& chunk_tx : chunk_txs) {
468 chunk_txids.
push_back(chunk_tx->GetTx().GetHash().ToString());
470 chunk.
pushKV(
"txs", std::move(chunk_txids));
478 for (
const auto& tx : cluster) {
479 total_weight += tx->GetAdjustedWeight();
481 info.pushKV(
"clusterweight", total_weight);
482 info.pushKV(
"txcount", cluster.size());
487 FeePerWeight current_chunk_feerate = pool.GetMainChunkFeerate(*cluster[0]);
488 std::vector<const CTxMemPoolEntry *> current_chunk;
489 current_chunk.reserve(cluster.size());
492 for (
const auto& tx : cluster) {
493 if (current_chunk_feerate.
size == 0) {
496 AppendChunkInfo(all_chunks, pool.GetMainChunkFeerate(*current_chunk[0]), current_chunk);
497 current_chunk.clear();
498 current_chunk_feerate = pool.GetMainChunkFeerate(*tx);
500 current_chunk.push_back(tx);
501 current_chunk_feerate.
size -= tx->GetAdjustedWeight();
503 AppendChunkInfo(all_chunks, pool.GetMainChunkFeerate(*current_chunk[0]), current_chunk);
504 current_chunk.clear();
505 info.pushKV(
"chunks", std::move(all_chunks));
512 auto [ancestor_count, ancestor_size, ancestor_fees] = pool.CalculateAncestorData(e);
513 auto [descendant_count, descendant_size, descendant_fees] = pool.CalculateDescendantData(e);
515 info.pushKV(
"vsize", e.GetTxSize());
516 info.pushKV(
"weight", e.GetTxWeight());
518 info.pushKV(
"height", e.GetHeight());
519 info.pushKV(
"descendantcount", descendant_count);
520 info.pushKV(
"descendantsize", descendant_size);
521 info.pushKV(
"ancestorcount", ancestor_count);
522 info.pushKV(
"ancestorsize", ancestor_size);
523 info.pushKV(
"wtxid", e.GetTx().GetWitnessHash().ToString());
524 auto feerate = pool.GetMainChunkFeerate(e);
525 info.pushKV(
"chunkweight", feerate.size);
533 info.pushKV(
"fees", std::move(fees));
536 std::set<std::string> setDepends;
544 for (
const std::string& dep : setDepends)
549 info.pushKV(
"depends", std::move(depends));
553 spent.
push_back(child.GetTx().GetHash().ToString());
556 info.pushKV(
"spentby", std::move(spent));
559 bool rbfStatus =
false;
567 info.pushKV(
"bip125-replaceable", rbfStatus);
568 info.pushKV(
"unbroadcast", pool.IsUnbroadcastTx(tx.
GetHash()));
574 if (include_mempool_sequence) {
585 o.
pushKVEnd(e.GetTx().GetHash().ToString(), std::move(info));
590 uint64_t mempool_sequence;
594 a.
push_back(e.GetTx().GetHash().ToString());
598 if (!include_mempool_sequence) {
602 o.
pushKV(
"txids", std::move(a));
603 o.
pushKV(
"mempool_sequence", mempool_sequence);
612 "Returns the feerate diagram for the whole mempool.",
641 for (
auto f : diagram) {
643 o.
pushKV(
"weight", f.size);
656 "Returns all transaction ids in memory pool as a json array of string transaction ids.\n" 657 "\nHint: use getmempoolentry to fetch a specific transaction from the mempool.\n",
660 {
"mempool_sequence",
RPCArg::Type::BOOL,
RPCArg::Default{
false},
"If verbose=false, returns a json object with transaction list and mempool sequence number attached."},
673 RPCResult{
"for verbose = false and mempool_sequence = true",
689 bool fVerbose =
false;
690 if (!request.params[0].isNull())
691 fVerbose = request.params[0].
get_bool();
693 bool include_mempool_sequence =
false;
694 if (!request.params[1].isNull()) {
695 include_mempool_sequence = request.params[1].get_bool();
706 "getmempoolancestors",
707 "If txid is in the mempool, returns all in-mempool ancestors.\n",
728 bool fVerbose =
false;
729 if (!request.params[1].isNull())
730 fVerbose = request.params[1].
get_bool();
737 const auto entry{mempool.
GetEntry(txid)};
738 if (entry ==
nullptr) {
747 o.
push_back(ancestorIt->GetTx().GetHash().ToString());
767 "getmempooldescendants",
768 "If txid is in the mempool, returns all in-mempool descendants.\n",
789 bool fVerbose =
false;
790 if (!request.params[1].isNull())
791 fVerbose = request.params[1].
get_bool();
798 const auto it{mempool.
GetIter(txid)};
806 setDescendants.erase(*it);
811 o.
push_back(descendantIt->GetTx().GetHash().ToString());
832 "Returns mempool data for given cluster\n",
850 const auto entry{mempool.
GetEntry(txid)};
851 if (entry ==
nullptr) {
868 "Returns mempool data for given transaction\n",
885 const auto entry{mempool.
GetEntry(txid)};
886 if (entry ==
nullptr) {
900 "Scans the mempool (and the txospenderindex, if available) to find transactions spending any of the given outputs",
914 {
"mempool_only",
RPCArg::Type::BOOL,
RPCArg::DefaultHint{
"true if txospenderindex unavailable, otherwise false"},
"If false and mempool lacks a relevant spend, use txospenderindex (throws an exception if not available)."},
926 {
RPCResult::Type::STR_HEX,
"spendingtxid",
true,
"the transaction id of the mempool transaction spending this output (omitted if unspent)"},
927 {
RPCResult::Type::STR_HEX,
"spendingtx",
true,
"the transaction spending this output (only if return_spending_tx is set, omitted if unspent)"},
928 {
RPCResult::Type::STR_HEX,
"blockhash",
true,
"the hash of the spending block (omitted if unspent or the spending tx is not confirmed)"},
933 HelpExampleCli(
"gettxspendingprevout",
"\"[{\\\"txid\\\":\\\"a08e6907dbbd3d809776dbfc5d82e371b764ed838b5655e72f463568df1aadf0\\\",\\\"vout\\\":3}]\"")
934 +
HelpExampleRpc(
"gettxspendingprevout",
"\"[{\\\"txid\\\":\\\"a08e6907dbbd3d809776dbfc5d82e371b764ed838b5655e72f463568df1aadf0\\\",\\\"vout\\\":3}]\"")
935 +
HelpExampleCliNamed(
"gettxspendingprevout", {{
"outputs",
"[{\"txid\":\"a08e6907dbbd3d809776dbfc5d82e371b764ed838b5655e72f463568df1aadf0\",\"vout\":3}]"}, {
"return_spending_tx",
true}})
940 if (output_params.
empty()) {
950 const bool mempool_only{options.exists(
"mempool_only") ? options[
"mempool_only"].get_bool() : !
g_txospenderindex};
951 const bool return_spending_tx{options.exists(
"return_spending_tx") ? options[
"return_spending_tx"].get_bool() :
false};
958 std::vector<Entry> prevouts;
961 for (
unsigned int idx = 0; idx < output_params.
size(); idx++) {
979 bool missing_from_mempool{
false};
983 for (
auto& entry : prevouts) {
985 if (spendingTx !=
nullptr) {
988 if (return_spending_tx) {
991 entry.output = std::move(o);
993 missing_from_mempool =
true;
1001 for (
auto& entry : prevouts) {
1002 if (!entry.output.isNull()) {
1003 result.push_back(std::move(entry.output));
1009 }
else if (!txospenderindex_ready) {
1010 throw JSONRPCError(
RPC_MISC_ERROR,
strprintf(
"No spending tx for the outpoint %s:%d in mempool, and txospenderindex is unavailable.", entry.prevout.hash.GetHex(), entry.prevout.n));
1017 if (spender.value()) {
1018 o.
pushKV(
"spendingtxid", spender.value()->tx->GetHash().GetHex());
1019 o.pushKV(
"blockhash", spender.value()->block_hash.GetHex());
1020 if (return_spending_tx) {
1021 o.pushKV(
"spendingtx",
EncodeHexTx(*spender.value()->tx));
1025 result.push_back(std::move(o));
1039 ret.pushKV(
"size", pool.
size());
1048 ret.pushKV(
"fullrbf",
true);
1053 ret.pushKV(
"optimal", pool.m_txgraph->DoWork(0));
1060 "Returns details on the active state of the TX memory pool.",
1065 {
RPCResult::Type::BOOL,
"loaded",
"True if the initial load attempt of the persisted mempool finished"},
1067 {
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"},
1074 {
RPCResult::Type::NUM,
"unbroadcastcount",
"Current number of transactions that haven't passed initial broadcast yet"},
1075 {
RPCResult::Type::BOOL,
"fullrbf",
"True if the mempool accepts RBF without replaceability signaling inspection (DEPRECATED)"},
1076 {
RPCResult::Type::BOOL,
"permitbaremultisig",
"True if the mempool accepts transactions with bare multisig outputs"},
1077 {
RPCResult::Type::NUM,
"maxdatacarriersize",
"Maximum number of bytes that can be used by OP_RETURN outputs in the mempool"},
1078 {
RPCResult::Type::NUM,
"limitclustercount",
"Maximum number of transactions that can be in a cluster (configured by -limitclustercount)"},
1079 {
RPCResult::Type::NUM,
"limitclustersize",
"Maximum size of a cluster in virtual bytes (configured by -limitclustersize)"},
1097 "Import a mempool.dat file and attempt to add its contents to the mempool.\n" 1098 "Warning: Importing untrusted files is dangerous, especially if metadata from the file is taken over.",
1107 "Whether to use the current system time or use the entry time metadata from the mempool file.\n" 1108 "Warning: Importing untrusted metadata may lead to unexpected issues and undesirable behavior."},
1110 "Whether to apply the fee delta metadata from the mempool file.\n" 1111 "It will be added to any existing fee deltas.\n" 1112 "The fee delta can be set by the prioritisetransaction RPC.\n" 1113 "Warning: Importing untrusted metadata may lead to unexpected issues and undesirable behavior.\n" 1114 "Only set this bool if you understand what it does."},
1116 "Whether to apply the unbroadcast set metadata from the mempool file.\n" 1117 "Warning: Importing untrusted metadata may lead to unexpected issues and undesirable behavior."},
1135 const UniValue& use_current_time{request.params[1][
"use_current_time"]};
1136 const UniValue& apply_fee_delta{request.params[1][
"apply_fee_delta_priority"]};
1137 const UniValue& apply_unbroadcast{request.params[1][
"apply_unbroadcast_set"]};
1139 .
use_current_time = use_current_time.isNull() ? true : use_current_time.get_bool(),
1140 .apply_fee_delta_priority = apply_fee_delta.isNull() ? false : apply_fee_delta.get_bool(),
1141 .apply_unbroadcast_set = apply_unbroadcast.isNull() ? false : apply_unbroadcast.get_bool(),
1158 "Dumps the mempool to disk. It will fail until the previous dump is fully loaded.\n",
1198 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."},
1210 o.
pushKV(
"txid", orphan.
tx->GetHash().ToString());
1211 o.
pushKV(
"wtxid", orphan.
tx->GetWitnessHash().ToString());
1212 o.
pushKV(
"bytes", orphan.
tx->ComputeTotalSize());
1216 for (
const auto fromPeer: orphan.
announcers) {
1227 "Shows transactions in the tx orphanage.\n" 1228 "\nEXPERIMENTAL warning: this call may be changed in future releases.\n",
1230 {
"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",
1248 Cat<std::vector<RPCResult>>(
1269 if (verbosity == 0) {
1270 for (
auto const& orphan : orphanage) {
1271 ret.push_back(orphan.tx->GetHash().ToString());
1273 }
else if (verbosity == 1) {
1274 for (
auto const& orphan : orphanage) {
1277 }
else if (verbosity == 2) {
1278 for (
auto const& orphan : orphanage) {
1295 "Submit a package of raw transactions (serialized, hex-encoded) to local node.\n" 1296 "The package will be validated according to consensus and mempool policy rules. If any transaction passes, it will be accepted to mempool.\n" 1297 "This RPC is experimental and the interface may be unstable. Refer to doc/policy/packages.md for documentation on package policies.\n" 1298 "Warning: successful submission does not mean the transactions will propagate throughout the network.\n" 1302 "The package must consist of a transaction with (some, all, or none of) its unconfirmed parents. A single transaction is permitted.\n" 1303 "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" 1304 "The package must be topologically sorted, with the child being the last element in the array if there are multiple elements.",
1310 "Reject transactions whose fee rate is higher than the specified value, expressed in " +
CURRENCY_UNIT +
1311 "/kvB.\nFee rates larger than 1BTC/kvB are rejected.\nSet to 0 to accept any fee rate."},
1313 "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" 1314 "If burning funds through unspendable outputs is desired, increase this value.\n" 1315 "This check is based on heuristics and does not guarantee spendability of outputs.\n" 1321 {
RPCResult::Type::STR,
"package_msg",
"The transaction package result message. \"success\" indicates all transactions were accepted into or are already in the mempool."},
1322 {
RPCResult::Type::OBJ_DYN,
"tx-results",
"The transaction results keyed by wtxid. An entry is returned for every submitted wtxid.",
1326 {
RPCResult::Type::STR_HEX,
"other-wtxid",
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."},
1330 {
RPCResult::Type::STR_AMOUNT,
"effective-feerate",
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."},
1331 {
RPCResult::Type::ARR,
"effective-includes",
true,
"if effective-feerate is provided, the wtxids of the transactions whose fees and vsizes are included in effective-feerate.",
1335 {
RPCResult::Type::STR,
"error",
true,
"Error string if rejected from mempool, or \"package-not-validated\" when the package aborts before any per-tx processing."},
1338 {
RPCResult::Type::ARR,
"replaced-transactions",
true,
"List of txids of replaced transactions",
1345 HelpExampleRpc(
"submitpackage", R
"(["raw-parent-tx-1", "raw-parent-tx-2", "raw-child-tx"])") + 1346 HelpExampleCli("submitpackage", R
"('["raw-tx-without-unconfirmed-parents"]')") 1358 std::optional<CFeeRate> client_maxfeerate{max_raw_tx_fee_rate};
1360 if (max_raw_tx_fee_rate ==
CFeeRate(0)) {
1361 client_maxfeerate = std::nullopt;
1367 std::vector<CTransactionRef> txns;
1368 txns.reserve(raw_transactions.
size());
1369 for (
const auto& rawtx : raw_transactions.
getValues()) {
1373 "TX decode failed: " + rawtx.get_str() +
" Make sure the tx has at least one input.");
1376 for (
const auto&
out : mtx.
vout) {
1377 if((
out.scriptPubKey.IsUnspendable() || !
out.scriptPubKey.HasValidOps()) &&
out.nValue > max_burn_amount) {
1386 throw JSONRPCTransactionError(TransactionError::INVALID_PACKAGE,
"package topology disallowed. not child-with-parents or parents depend on each other.");
1394 std::string package_msg =
"success";
1397 switch(package_result.m_state.GetResult()) {
1401 CHECK_NONFATAL(package_result.m_tx_results.size() == txns.size());
1402 for (
const auto& tx : txns) {
1411 package_result.m_state.GetRejectReason());
1417 package_msg = package_result.m_state.ToString();
1418 CHECK_NONFATAL(package_result.m_tx_results.size() == txns.size() ||
1419 package_result.m_tx_results.empty());
1424 size_t num_broadcast{0};
1425 for (
const auto& tx : txns) {
1427 if (!mempool.
exists(tx->GetHash())) {
1432 std::string err_string;
1439 if (err != TransactionError::OK) {
1441 strprintf(
"transaction broadcast failed: %s (%d transactions were broadcast successfully)",
1442 err_string, num_broadcast));
1448 rpc_result.pushKV(
"package_msg", package_msg);
1450 std::set<Txid> replaced_txids;
1451 for (
const auto& tx : txns) {
1453 result_inner.pushKV(
"txid", tx->GetHash().GetHex());
1454 const auto wtxid_hex = tx->GetWitnessHash().GetHex();
1455 auto it = package_result.m_tx_results.find(tx->GetWitnessHash());
1456 if (it == package_result.m_tx_results.end()) {
1461 result_inner.pushKV(
"error",
"package-not-validated");
1462 tx_result_map.pushKV(wtxid_hex, std::move(result_inner));
1465 const auto& tx_result = it->second;
1466 switch(it->second.m_result_type) {
1468 result_inner.pushKV(
"other-wtxid", it->second.m_other_wtxid.value().GetHex());
1471 result_inner.pushKV(
"error", it->second.m_state.ToString());
1475 result_inner.pushKV(
"vsize", it->second.m_vsize.value());
1482 fees.
pushKV(
"effective-feerate",
ValueFromAmount(tx_result.m_effective_feerate.value().GetFeePerK()));
1484 for (
const auto& wtxid : tx_result.m_wtxids_fee_calculations.value()) {
1485 effective_includes_res.
push_back(wtxid.ToString());
1487 fees.
pushKV(
"effective-includes", std::move(effective_includes_res));
1489 result_inner.pushKV(
"fees", std::move(fees));
1490 for (
const auto& ptx : it->second.m_replaced_transactions) {
1491 replaced_txids.insert(ptx->GetHash());
1495 tx_result_map.pushKV(wtxid_hex, std::move(result_inner));
1497 rpc_result.pushKV(
"tx-results", std::move(tx_result_map));
1499 for (
const auto& txid : replaced_txids) replaced_list.
push_back(txid.ToString());
1500 rpc_result.pushKV(
"replaced-transactions", std::move(replaced_list));
1526 for (
const auto& c : commands) {
1527 t.appendCommand(c.name, &c);
std::shared_ptr< const CTransaction > CTransactionRef
static constexpr bool DEFAULT_PRIVATE_BROADCAST
Default for -privatebroadcast.
void RegisterMempoolRPCCommands(CRPCTable &t)
void push_back(UniValue val)
UniValue MempoolToJSON(const CTxMemPool &pool, bool verbose, bool include_mempool_sequence)
Mempool to JSON.
static RPCHelpMan getrawmempool()
const std::vector< UniValue > & getValues() const
constexpr int64_t count_seconds(std::chrono::seconds t)
uint256 ParseHashO(const UniValue &o, std::string_view strKey)
indexed_transaction_set::nth_index< 0 >::type::const_iterator txiter
static RPCHelpMan getmempoolancestors()
UniValue ValueFromAmount(const CAmount amount)
void RPCTypeCheckObj(const UniValue &o, const std::map< std::string, UniValueType > &typesExpected, bool fAllowNull, bool fStrict)
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...
The package itself is invalid (e.g. too many transactions).
Valid, transaction was already in the mempool.
Either this tx or a mempool ancestor signals rbf.
std::vector< const CTxMemPoolEntry * > GetCluster(Txid txid) const EXCLUSIVE_LOCKS_REQUIRED(cs)
CTxMemPool & EnsureMemPool(const NodeContext &node)
std::optional< unsigned > max_datacarrier_bytes
A data carrying output is an unspendable output containing data.
unsigned cluster_count
The maximum number of transactions in a cluster.
Omit the mempool and directly send the transaction via a few dedicated connections to peers on privac...
static void clusterToJSON(const CTxMemPool &pool, UniValue &info, std::vector< const CTxMemPoolEntry *> cluster) EXCLUSIVE_LOCKS_REQUIRED(pool.cs)
Interface for managing multiple Chainstate objects, where each chainstate is associated with chainsta...
static RPCHelpMan getmempoolinfo()
Unconfirmed tx that does not signal rbf and is not in the mempool.
void CalculateDescendants(txiter it, setEntries &setDescendants) const EXCLUSIVE_LOCKS_REQUIRED(cs)
Populate setDescendants with all in-mempool descendants of given transaction.
size_t DynamicMemoryUsage() const
virtual std::vector< PrivateBroadcast::TxBroadcastInfo > GetPrivateBroadcastInfo() const =0
Get info about transactions currently being privately broadcast.
transaction was missing some of its inputs
#define CHECK_NONFATAL(condition)
Identity function.
int64_t cluster_size_vbytes
The maximum allowed size in virtual bytes of a cluster.
Special type that behaves almost exactly like OBJ, defining an options object with a list of pre-defi...
static RPCHelpMan abortprivatebroadcast()
unsigned long size() const
UniValue MempoolInfoToJSON(const CTxMemPool &pool)
Mempool information to JSON.
uint64_t GetSequence() const EXCLUSIVE_LOCKS_REQUIRED(cs)
bool IsChildWithParentsTree(const Package &package)
Context-free check that a package IsChildWithParents() and none of the parents depend on each other (...
std::set< txiter, CompareIteratorByHash > setEntries
void pushKVEnd(std::string key, UniValue val)
const UniValue & get_array() const
bool Contains(Network net) const EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
Add the transaction to the mempool and broadcast to all peers for which tx relay is enabled...
static void entryToJSON(const CTxMemPool &pool, UniValue &info, const CTxMemPoolEntry &e) EXCLUSIVE_LOCKS_REQUIRED(pool.cs)
bool GetBoolArg(const std::string &strArg, bool fDefault) const
Return boolean argument or default value.
int64_t GetVirtualTransactionSize(int64_t nWeight, int64_t nSigOpCost, unsigned int bytes_per_sigop)
Compute the virtual transaction size (weight reinterpreted as bytes).
std::map< Wtxid, MempoolAcceptResult > m_tx_results
Map from wtxid to finished MempoolAcceptResults.
bool IsInitialBlockDownload() const noexcept
Check whether we are doing an initial block download (synchronizing from disk or network) ...
std::optional< txiter > GetIter(const Txid &txid) const EXCLUSIVE_LOCKS_REQUIRED(cs)
Returns an iterator to the given hash, if found.
Tagged wrapper around FeeFrac to avoid unit confusion.
std::set< NodeId > announcers
Peers added with AddTx or AddAnnouncer.
static int32_t GetTransactionWeight(const CTransaction &tx)
static RPCHelpMan getorphantxs()
std::set< Txid > GetUnbroadcastTxs() const
Returns transactions in unbroadcast set.
void reserve(size_t new_cap)
const std::vector< CTxIn > vin
Invalid, missing or duplicate parameter.
bool DumpMempool(const CTxMemPool &pool, const fs::path &dump_path, FopenFn mockable_fopen_function, bool skip_file_commit)
static UniValue OrphanToJSON(const node::TxOrphanage::OrphanInfo &orphan)
RBFTransactionState
The rbf state of unconfirmed transactions.
CFeeRate ParseFeeRate(const UniValue &json)
Parse a json number or string, denoting BTC/kvB, into a CFeeRate (sat/kvB).
CTxMemPoolEntry stores data about the corresponding transaction, as well as data about all in-mempool...
static RPCHelpMan importmempool()
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.
int64_t CAmount
Amount in satoshis (Can be negative)
const UniValue & find_value(std::string_view key) const
std::string oneline_description
Should be empty unless it is supposed to override the auto-generated summary line.
Special type that is a STR with only hex chars.
NodeContext struct containing references to chain state and connection state.
std::string HelpExampleRpc(const std::string &methodname, const std::string &args)
void AppendChunkInfo(UniValue &all_chunks, FeePerWeight chunk_feerate, std::vector< const CTxMemPoolEntry *> chunk_txs)
static RPCHelpMan getmempoolcluster()
UniValue JSONRPCError(int code, const std::string &message)
virtual std::vector< node::TxOrphanage::OrphanInfo > GetOrphanTransactions()=0
Special string with only hex chars.
static CAmount AmountFromValue(const UniValue &value)
Chainstate stores and provides an API to update our local knowledge of the current best chain...
static RPCHelpMan getprivatebroadcastinfo()
static RPCHelpMan submitpackage()
Validation result for package mempool acceptance.
An input of a transaction.
std::unique_ptr< TxoSpenderIndex > g_txospenderindex
The global txo spender index. May be null.
std::string ToString() const
setEntries CalculateMemPoolAncestors(const CTxMemPoolEntry &entry) const EXCLUSIVE_LOCKS_REQUIRED(cs)
Calculate all in-mempool ancestors of entry (not including the tx itself)
std::string ToString() const
const std::string CURRENCY_UNIT
static RPCHelpMan savemempool()
General application defined errors.
std::string DefaultHint
Hint for default value.
std::string HelpExampleCliNamed(const std::string &methodname, const RPCArgList &args)
static std::vector< RPCResult > OrphanDescription()
uint64_t GetTotalTxSize() const EXCLUSIVE_LOCKS_REQUIRED(cs)
CFeeRate min_relay_feerate
A fee rate smaller than this is considered zero fee (for relaying, mining and transaction creation) ...
CTxMemPool & EnsureAnyMemPool(const std::any &context)
At least one tx is invalid.
static RPCHelpMan getmempooldescendants()
const CTransaction * GetConflictTx(const COutPoint &prevout) const EXCLUSIVE_LOCKS_REQUIRED(cs)
Get the transaction in the pool that spends the same prevout.
An outpoint - a combination of a transaction hash and an index n into its vout.
std::string HelpExampleCli(const std::string &methodname, const std::string &args)
std::vector< CTxOut > vout
CFeeRate GetMinFee(size_t sizelimit) const
static RPCHelpMan sendrawtransaction()
Special numeric to denote unix epoch time.
#define WITH_LOCK(cs, code)
Run code while locking a mutex.
static RPCHelpMan gettxspendingprevout()
static RPCHelpMan getmempoolfeeratediagram()
static RPCHelpMan getmempoolentry()
bool exists(const Txid &txid) const
std::vector< FeePerWeight > GetFeerateDiagram() const EXCLUSIVE_LOCKS_REQUIRED(cs)
bool GetLoadTried() const
static CTransactionRef MakeTransactionRef(Tx &&txIn)
bool permit_bare_multisig
std::string FormatMoney(const CAmount n)
Money parsing/formatting utilities.
PackageValidationState m_state
Optional argument for which the default value is omitted from help text for one of two reasons: ...
const CTxMemPoolEntry * GetEntry(const Txid &txid) const LIFETIMEBOUND EXCLUSIVE_LOCKS_REQUIRED(cs)
#define EXCLUSIVE_LOCKS_REQUIRED(...)
std::string utf8string() const
Return a UTF-8 representation of the path as a std::string, for compatibility with code using std::st...
CTxMemPool stores valid-according-to-the-current-best-chain transactions that may be included in the ...
int ParseVerbosity(const UniValue &arg, int default_verbosity, bool allow_bool)
Parses verbosity from provided UniValue.
Special string to represent a floating point amount.
static const CAmount DEFAULT_MAX_BURN_AMOUNT
Maximum burn value for sendrawtransaction, submitpackage, and testmempoolaccept RPC calls...
void pushKV(std::string key, UniValue val)
static transaction_identifier FromUint256(const uint256 &id)
uint256 ParseHashV(const UniValue &v, std::string_view name)
Utilities: convert hex-encoded Values (throws error if not hex).
const CTransaction & GetTx() const
fs::path MempoolPath(const ArgsManager &argsman)
ChainstateManager & EnsureChainman(const NodeContext &node)
CAmount GetTotalFee() const EXCLUSIVE_LOCKS_REQUIRED(cs)
std::string EncodeHexTx(const CTransaction &tx)
const UniValue & get_obj() const
bool DecodeHexTx(CMutableTransaction &tx, const std::string &hex_tx, bool try_no_witness, bool try_witness)
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.
Special type representing a floating point amount (can be either NUM or STR)
static RPCHelpMan testmempoolaccept()
Fee rate in satoshis per virtualbyte: CAmount / vB the feerate is represented internally as FeeFrac...
ArgsManager & EnsureAnyArgsman(const std::any &context)
#define AssertLockNotHeld(cs)
PeerManager & EnsurePeerman(const NodeContext &node)
A mutable version of CTransaction.
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.
static path u8path(std::string_view utf8_str)
Allows providing orphan information externally.
static constexpr uint32_t MAX_PACKAGE_COUNT
Default maximum number of transactions in a package.
The basic transaction that is broadcasted on the network and contained in blocks. ...
UniValue JSONRPCTransactionError(TransactionError terr, const std::string &err_string)
Chainstate & ActiveChainstate() const
Alternatives to CurrentChainstate() used by older code to query latest chainstate information without...
Special dictionary with keys that are not literals.
std::string GetRejectReason() const
Still downloading initial blocks.
is a home for public enum and struct type definitions that are used internally by node code...
static const CFeeRate DEFAULT_MAX_RAW_TX_FEE_RATE
Maximum fee rate for sendrawtransaction and testmempoolaccept RPC calls.
Initial value. The package has not yet been rejected.
Path class wrapper to block calls to the fs::path(std::string) implicit constructor and the fs::path:...
RecursiveMutex cs_main
Mutex to guard access to validation specific variables, such as reading or changing the chainstate...
static std::vector< RPCResult > ClusterDescription()
CAmount GetFeePerK() const
Return the fee in satoshis for a vsize of 1000 vbytes.
virtual std::vector< CTransactionRef > AbortPrivateBroadcast(const uint256 &id)=0
Abort private broadcast attempts for transactions currently being privately broadcast.
Wrapper for UniValue::VType, which includes typeAny: Used to denote don't care type.
std::vector< CTxMemPoolEntryRef > entryAll() const EXCLUSIVE_LOCKS_REQUIRED(cs)
std::string ToString(const T &t)
Locale-independent version of std::to_string.
NodeContext & EnsureAnyNodeContext(const std::any &context)
RecursiveMutex cs
This mutex needs to be locked when accessing mapTx or other members that are guarded by it...
Error parsing or validating structure in raw format.
const Txid & GetHash() const LIFETIMEBOUND
CFeeRate incremental_relay_feerate
ReachableNets g_reachable_nets
static std::vector< RPCResult > MempoolEntryDescription()
MempoolAcceptResult ProcessTransaction(const CTransactionRef &tx, bool test_accept=false) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Try to add a transaction to the memory pool.