8#include <blockfilter.h>
10#include <chainparams.h>
56#include <versionbits.h>
60#include <condition_variable>
82 const std::function<
void()>& interruption_point = {})
92 const fs::path& temppath,
93 const std::function<
void()>& interruption_point = {});
100 const fs::path& path,
101 const fs::path& tmppath,
108 int nShift = (blockindex.
nBits >> 24) & 0xff;
110 (double)0x0000ffff / (
double)(blockindex.
nBits & 0x00ffffff);
129 if (next && next->
pprev == &blockindex) {
133 return &blockindex == &tip ? 1 : -1;
142 const int height{param.
getInt<
int>()};
146 const int current_tip{active_chain.
Height()};
147 if (height > current_tip) {
151 return active_chain[height];
173 result.
pushKV(
"confirmations", confirmations);
187 if (blockindex.
pprev)
198 const CTxIn& vin_0{coinbase_tx.
vin[0]};
202 coinbase_tx_obj.
pushKV(
"sequence", vin_0.nSequence);
203 coinbase_tx_obj.
pushKV(
"coinbase",
HexStr(vin_0.scriptSig));
204 const auto& witness_stack{vin_0.scriptWitness.stack};
205 if (!witness_stack.empty()) {
207 coinbase_tx_obj.
pushKV(
"witness",
HexStr(witness_stack[0]));
209 return coinbase_tx_obj;
236 const bool is_not_pruned{
WITH_LOCK(
::cs_main,
return !blockman.IsBlockPruned(blockindex))};
238 if (have_undo && !blockman.
ReadBlockUndo(blockUndo, blockindex)) {
239 throw JSONRPCError(
RPC_INTERNAL_ERROR,
"Undo data expected but can't be read. This could be due to disk corruption or a conflict with a pruning event.");
241 for (
size_t i = 0; i < block.
vtx.size(); ++i) {
244 const CTxUndo* txundo = (have_undo && i > 0) ? &blockUndo.
vtxundo.at(i - 1) :
nullptr;
252 result.
pushKV(
"tx", std::move(txs));
261 "Returns the height of the most-work fully-validated chain.\n"
262 "The genesis block has height 0.\n",
283 "Returns the hash of the best (tip) block in the most-work fully-validated chain.\n",
304 "Waits for any new block and returns useful info about it.\n"
305 "\nReturns the current block on timeout or exit.\n"
306 "\nMake sure to use no RPC timeout (bitcoin-cli -rpcclienttimeout=0)",
324 if (!request.params[0].isNull())
325 timeout = request.params[0].
getInt<
int>();
339 uint256 tip_hash{request.params[1].isNull()
341 :
ParseHashV(request.params[1],
"current_tip")};
345 std::optional<BlockRef> block = timeout ? miner.
waitTipChanged(tip_hash, std::chrono::milliseconds(timeout)) :
349 if (block) current_block = *block;
352 ret.pushKV(
"hash", current_block.hash.GetHex());
353 ret.pushKV(
"height", current_block.height);
363 "Waits for a specific new block and returns useful info about it.\n"
364 "\nReturns the current block on timeout or exit.\n"
365 "\nMake sure to use no RPC timeout (bitcoin-cli -rpcclienttimeout=0)",
377 HelpExampleCli(
"waitforblock",
"\"0000000000079f8ef3d2c688c244eb7a4570b24c9ed7b4a8c619eb02596f8862\" 1000")
378 +
HelpExampleRpc(
"waitforblock",
"\"0000000000079f8ef3d2c688c244eb7a4570b24c9ed7b4a8c619eb02596f8862\", 1000")
386 if (!request.params[1].isNull())
387 timeout = request.params[1].getInt<
int>();
396 const auto deadline{std::chrono::steady_clock::now() + 1ms * timeout};
397 while (current_block.hash != hash) {
398 std::optional<BlockRef> block;
400 auto now{std::chrono::steady_clock::now()};
401 if (now >= deadline)
break;
409 current_block = *block;
413 ret.pushKV(
"hash", current_block.hash.GetHex());
414 ret.pushKV(
"height", current_block.height);
423 "waitforblockheight",
424 "Waits for (at least) block height and returns the height and hash\n"
425 "of the current tip.\n"
426 "\nReturns the current block on timeout or exit.\n"
427 "\nMake sure to use no RPC timeout (bitcoin-cli -rpcclienttimeout=0)",
446 int height = request.params[0].
getInt<
int>();
448 if (!request.params[1].isNull())
449 timeout = request.params[1].getInt<
int>();
458 const auto deadline{std::chrono::steady_clock::now() + 1ms * timeout};
460 while (current_block.height < height) {
461 std::optional<BlockRef> block;
463 auto now{std::chrono::steady_clock::now()};
464 if (now >= deadline)
break;
472 current_block = *block;
476 ret.pushKV(
"hash", current_block.hash.GetHex());
477 ret.pushKV(
"height", current_block.height);
486 "syncwithvalidationinterfacequeue",
487 "Waits for the validation interface queue to catch up on everything that was there when we entered this function.\n",
507 "Returns the proof-of-work difficulty as a multiple of the minimum difficulty.\n",
510 RPCResult::Type::NUM,
"",
"the proof-of-work difficulty as a multiple of the minimum difficulty."},
528 "Attempt to fetch block from a given peer.\n\n"
529 "We must have the header for this block, e.g. using submitheader.\n"
530 "The block will not have any undo data which can limit the usage of the block data in a context where the undo data is needed.\n"
531 "Subsequent calls for the same block may cause the response from the previous peer to be ignored.\n"
532 "Peers generally ignore requests for a stale block that they never fully verified, or one that is more than a month old.\n"
533 "When a peer does not respond with a block, we will disconnect.\n"
534 "Note: The block could be re-pruned as soon as it is received.\n\n"
535 "Returns an empty JSON object if the request was successfully scheduled.",
542 HelpExampleCli(
"getblockfrompeer",
"\"00000000c937983704a73af28acdec37b049d214adbda81d7e2a3dd146f6ed09\" 0")
543 +
HelpExampleRpc(
"getblockfrompeer",
"\"00000000c937983704a73af28acdec37b049d214adbda81d7e2a3dd146f6ed09\" 0")
552 const NodeId peer_id{request.params[1].getInt<int64_t>()};
563 throw JSONRPCError(
RPC_MISC_ERROR,
"In prune mode, only blocks that the node has already synced previously can be fetched from a peer");
567 if (block_has_data) {
571 if (
const auto res{peerman.
FetchBlock(peer_id, *index)}; !res) {
583 "Returns hash of block in best-block-chain at height provided.\n",
599 int nHeight = request.params[0].getInt<
int>();
600 if (nHeight < 0 || nHeight > active_chain.
Height())
613 "If verbose is false, returns a string that is serialized, hex-encoded data for blockheader 'hash'.\n"
614 "If verbose is true, returns an Object with information about blockheader <hash>.\n",
624 {
RPCResult::Type::NUM,
"confirmations",
"The number of confirmations, or -1 if the block is not on the main chain"},
644 HelpExampleCli(
"getblockheader",
"\"00000000c937983704a73af28acdec37b049d214adbda81d7e2a3dd146f6ed09\"")
645 +
HelpExampleRpc(
"getblockheader",
"\"00000000c937983704a73af28acdec37b049d214adbda81d7e2a3dd146f6ed09\"")
651 bool fVerbose =
true;
652 if (!request.params[1].isNull())
653 fVerbose = request.params[1].get_bool();
672 std::string strHex =
HexStr(ssBlock);
685 if (!(blockindex.nStatus & flag)) {
686 if (blockman.IsBlockPruned(blockindex)) {
689 if (check_for_undo) {
704 if (!blockman.
ReadBlock(block, blockindex)) {
733 if (blockindex.
nHeight == 0)
return blockUndo;
749 auto fields = std::vector<RPCResult>{
751 {
RPCResult::Type::NUM,
"confirmations",
"The number of confirmations, or -1 if the block is not on the main chain"},
761 {
RPCResult::Type::STR_HEX,
"witness",
true,
"The coinbase input's first (and only) witness stack element, if present"},
768 fields.push_back(std::move(tx_result));
772 fields.emplace_back(
RPCResult::Type::STR_HEX,
"bits",
"nBits: compact representation of the block difficulty target");
775 fields.emplace_back(
RPCResult::Type::STR_HEX,
"chainwork",
"Expected number of hashes required to produce the chain up to this block (in hex)");
777 fields.emplace_back(
RPCResult::Type::STR_HEX,
"previousblockhash",
true,
"The hash of the previous block (if available)");
781 std::vector<RPCResult> new_fields;
782 new_fields.reserve(fields.size());
784 for (
const auto& f : fields) {
785 if (f.m_key_name ==
"tx") {
786 new_fields.push_back(f);
792 new_fields.emplace_back(f, std::move(eopts));
797 new_fields.emplace_back(f, std::move(eopts));
800 fields = std::move(new_fields);
809 "If verbosity is 0, returns a string that is serialized, hex-encoded data for block 'hash'.\n"
810 "If verbosity is 1, returns an Object with information about block <hash>.\n"
811 "If verbosity is 2, returns an Object with information about block <hash> and information about each transaction.\n"
812 "If verbosity is 3, returns an Object with information about block <hash> and information about each transaction, including prevout information for inputs (only for unpruned blocks in the current best chain).\n",
815 {
"verbosity|verbose",
RPCArg::Type::NUM,
RPCArg::Default{1},
"0 for hex-encoded data, 1 for a JSON object, 2 for JSON object with transaction data, and 3 for JSON object with transaction data including prevout information for inputs",
828 .elision_summary =
"The transactions in the format of the getrawtransaction RPC. Different from verbosity = 1 \"tx\" result",
829 .fee =
true, .hex =
true,
830 .fee_doc =
"The transaction fee in " +
CURRENCY_UNIT +
", omitted if block undo data is not available"})},
831 }},
"Same output as verbosity = 1")},
838 .prevout_optional =
true,
842 .prevout_doc =
"(Only if undo information is available)",
843 .vin_inner_elision =
"The same output as verbosity = 2"})},
844 }},
"Same output as verbosity = 2")},
847 HelpExampleCli(
"getblock",
"\"00000000c937983704a73af28acdec37b049d214adbda81d7e2a3dd146f6ed09\"")
848 +
HelpExampleRpc(
"getblock",
"\"00000000c937983704a73af28acdec37b049d214adbda81d7e2a3dd146f6ed09\"")
871 if (verbosity <= 0) {
872 return HexStr(block_data);
879 if (verbosity == 1) {
881 }
else if (verbosity == 2) {
903 if (!first_block || !chain_tip)
return std::nullopt;
908 const auto& first_unpruned{blockman.GetFirstBlock(*chain_tip,
BLOCK_HAVE_MASK, first_block)};
909 if (&first_unpruned == first_block) {
921 "Attempts to delete block and undo data up to a specified height or timestamp, if eligible for pruning.\n"
922 "Requires `-prune` to be enabled at startup. While pruned data may be re-fetched in some cases (e.g., via `getblockfrompeer`), local deletion is irreversible.\n",
925 " to prune blocks whose block time is at least 2 hours older than the provided timestamp."},
944 int heightParam = request.params[0].getInt<
int>();
945 if (heightParam < 0) {
951 if (heightParam > 1000000000) {
960 unsigned int height = (
unsigned int) heightParam;
961 unsigned int chainHeight = (
unsigned int) active_chain.
Height();
964 }
else if (height > chainHeight) {
967 LogDebug(
BCLog::RPC,
"Attempt to prune blocks close to the tip. Retaining the minimum number of blocks.\n");
979 if (hash_type_input ==
"hash_serialized_3") {
980 return CoinStatsHashType::HASH_SERIALIZED;
981 }
else if (hash_type_input ==
"muhash") {
982 return CoinStatsHashType::MUHASH;
983 }
else if (hash_type_input ==
"none") {
997 const std::function<
void()>& interruption_point = {},
999 bool index_requested =
true)
1024 "Returns statistics about the unspent transaction output set.\n"
1025 "Note this call may take some time if you are not using coinstatsindex.\n",
1027 {
"hash_type",
RPCArg::Type::STR,
RPCArg::Default{
"hash_serialized_3"},
"Which UTXO set hash should be calculated. Options: 'hash_serialized_3' (the legacy algorithm), 'muhash', 'none'."},
1031 .type_str = {
"",
"string or numeric"},
1041 {
RPCResult::Type::NUM,
"bogosize",
"Database-independent, meaningless metric indicating the UTXO set size"},
1042 {
RPCResult::Type::STR_HEX,
"hash_serialized_3",
true,
"The serialized hash (only present if 'hash_serialized_3' hash_type is chosen)"},
1044 {
RPCResult::Type::NUM,
"transactions",
true,
"The number of transactions with unspent outputs (not available when coinstatsindex is used)"},
1045 {
RPCResult::Type::NUM,
"disk_size",
true,
"The estimated size of the chainstate on disk (not available when coinstatsindex is used)"},
1047 {
RPCResult::Type::STR_AMOUNT,
"total_unspendable_amount",
true,
"The total amount of coins permanently excluded from the UTXO set (only available if coinstatsindex is used)"},
1048 {
RPCResult::Type::OBJ,
"block_info",
true,
"Info on amounts in the block at this block height (only available if coinstatsindex is used)",
1067 HelpExampleCli("gettxoutsetinfo", R
"("none" '"00000000c937983704a73af28acdec37b049d214adbda81d7e2a3dd146f6ed09"')") +
1068 HelpExampleCli("-named gettxoutsetinfo", R
"(hash_type='muhash' use_index='false')") +
1072 HelpExampleRpc("gettxoutsetinfo", R
"("none", "00000000c937983704a73af28acdec37b049d214adbda81d7e2a3dd146f6ed09")")
1079 bool index_requested = request.params[2].isNull() || request.params[2].get_bool();
1090 coins_view = &active_chainstate.
CoinsDB();
1095 if (!request.params[1].isNull()) {
1100 if (hash_type == CoinStatsHashType::HASH_SERIALIZED) {
1104 if (!index_requested) {
1116 if (pindex && pindex->
nHeight > summary.best_block_height) {
1122 const std::optional<CCoinsStats> maybe_stats =
GetUTXOStats(coins_view, *blockman, hash_type,
node.rpc_interruption_point, pindex, index_requested);
1123 if (maybe_stats.has_value()) {
1129 if (hash_type == CoinStatsHashType::HASH_SERIALIZED) {
1132 if (hash_type == CoinStatsHashType::MUHASH) {
1144 const std::optional<CCoinsStats> maybe_prev_stats =
GetUTXOStats(coins_view, *blockman, hash_type,
node.rpc_interruption_point, block_index.
pprev, index_requested);
1145 if (!maybe_prev_stats) {
1148 prev_stats = maybe_prev_stats.value();
1155 CAmount prev_block_total_unspendable_amount = prev_stats.total_unspendables_genesis_block +
1156 prev_stats.total_unspendables_bip30 +
1157 prev_stats.total_unspendables_scripts +
1158 prev_stats.total_unspendables_unclaimed_rewards;
1160 ret.pushKV(
"total_unspendable_amount",
ValueFromAmount(block_total_unspendable_amount));
1173 block_info.
pushKV(
"unspendable",
ValueFromAmount(block_total_unspendable_amount - prev_block_total_unspendable_amount));
1180 block_info.
pushKV(
"unspendables", std::move(unspendables));
1182 ret.pushKV(
"block_info", std::move(block_info));
1196 "Returns details about an unspent transaction output.\n",
1200 {
"include_mempool",
RPCArg::Type::BOOL,
RPCArg::Default{
true},
"Whether to include the mempool. Note that an unspent output that is spent in the mempool won't appear."},
1213 {
RPCResult::Type::STR,
"address",
true,
"The Bitcoin address (only if a well-defined address exists)"},
1219 "\nGet unspent transactions\n"
1221 "\nView the details\n"
1223 "\nAs a JSON-RPC call\n"
1235 COutPoint out{hash, request.params[1].getInt<uint32_t>()};
1236 bool fMempool =
true;
1237 if (!request.params[2].isNull())
1238 fMempool = request.params[2].get_bool();
1243 std::optional<Coin> coin;
1257 ret.pushKV(
"confirmations", 0);
1259 ret.pushKV(
"confirmations", pindex->
nHeight - coin->nHeight + 1);
1264 ret.pushKV(
"scriptPubKey", std::move(o));
1265 ret.pushKV(
"coinbase", coin->IsCoinBase());
1276 "Verifies blockchain database.\n",
1283 RPCResult::Type::BOOL,
"",
"Verification finished successfully. If false, check debug log for reason."},
1291 const int check_depth{request.params[1].isNull() ?
DEFAULT_CHECKBLOCKS : request.params[1].getInt<
int>()};
1310 rv.
pushKV(
"type",
"buried");
1322 if (blockindex ==
nullptr)
return;
1329 if (info.stats.has_value()) {
1330 bip9.
pushKV(
"bit", depparams.bit);
1332 bip9.
pushKV(
"start_time", depparams.nStartTime);
1333 bip9.
pushKV(
"timeout", depparams.nTimeout);
1334 bip9.
pushKV(
"min_activation_height", depparams.min_activation_height);
1337 bip9.
pushKV(
"status", info.current_state);
1338 bip9.
pushKV(
"since", info.since);
1339 bip9.
pushKV(
"status_next", info.next_state);
1342 if (info.stats.has_value()) {
1344 statsUV.
pushKV(
"period", info.stats->period);
1345 statsUV.
pushKV(
"elapsed", info.stats->elapsed);
1346 statsUV.
pushKV(
"count", info.stats->count);
1347 if (info.stats->threshold > 0 || info.stats->possible) {
1348 statsUV.
pushKV(
"threshold", info.stats->threshold);
1349 statsUV.
pushKV(
"possible", info.stats->possible);
1351 bip9.
pushKV(
"statistics", std::move(statsUV));
1354 sig.reserve(info.signalling_blocks.size());
1355 for (
const bool s : info.signalling_blocks) {
1356 sig.push_back(
s ?
'#' :
'-');
1358 bip9.
pushKV(
"signalling", sig);
1362 rv.
pushKV(
"type",
"bip9");
1363 bool is_active =
false;
1364 if (info.active_since.has_value()) {
1365 rv.
pushKV(
"height", *info.active_since);
1366 is_active = (*info.active_since <= blockindex->
nHeight + 1);
1368 rv.
pushKV(
"active", is_active);
1377 "Returns an object containing various state info regarding blockchain processing.\n",
1383 {
RPCResult::Type::NUM,
"blocks",
"the height of the most-work fully-validated chain. The genesis block has height 0"},
1392 {
RPCResult::Type::BOOL,
"initialblockdownload",
"(debug information) estimate of whether this node is in Initial Block Download mode"},
1393 {
RPCResult::Type::OBJ,
"backgroundvalidation",
true,
"state info regarding background validation process",
1395 {
RPCResult::Type::NUM,
"snapshotheight",
"the height of the snapshot block. Background validation verifies the chain from genesis up to this height"},
1396 {
RPCResult::Type::NUM,
"blocks",
"the height of the most-work background fully-validated chain. The genesis block has height 0"},
1397 {
RPCResult::Type::STR,
"bestblockhash",
"the hash of the currently best block validated in the background"},
1399 {
RPCResult::Type::NUM,
"verificationprogress",
"estimate of background verification progress [0..1]"},
1403 {
RPCResult::Type::NUM,
"size_on_disk",
"the estimated size of the block and undo files on disk"},
1405 {
RPCResult::Type::NUM,
"pruneheight",
true,
"the first block unpruned, all previous blocks were pruned (only present if pruning is enabled)"},
1406 {
RPCResult::Type::BOOL,
"automatic_pruning",
true,
"whether automatic pruning is enabled (only present if pruning is enabled)"},
1407 {
RPCResult::Type::NUM,
"prune_target_size",
true,
"the target size used by pruning (only present if automatic pruning is enabled)"},
1408 {
RPCResult::Type::STR_HEX,
"signet_challenge",
true,
"the block challenge (aka. block script), in hexadecimal (only present if the current network is a signet)"},
1411 RPCResult{
RPCResult::Type::ARR,
"warnings",
"any network and blockchain warnings (run with `-deprecatedrpc=warnings` to return the latest warning as a single string)",
1429 const int height{tip.
nHeight};
1432 obj.
pushKV(
"blocks", height);
1433 obj.
pushKV(
"headers", chainman.m_best_header ? chainman.m_best_header->nHeight : -1);
1442 auto historical_blocks{chainman.GetHistoricalBlockRange()};
1443 if (historical_blocks) {
1447 background_validation.
pushKV(
"snapshotheight", btarget.nHeight);
1448 background_validation.
pushKV(
"blocks", btip.nHeight);
1449 background_validation.
pushKV(
"bestblockhash", btip.GetBlockHash().GetHex());
1450 background_validation.
pushKV(
"mediantime", btip.GetMedianTimePast());
1451 background_validation.
pushKV(
"chainwork", btip.nChainWork.GetHex());
1453 obj.
pushKV(
"backgroundvalidation", std::move(background_validation));
1460 obj.
pushKV(
"pruneheight", prune_height ? prune_height.value() + 1 : 0);
1463 obj.
pushKV(
"automatic_pruning", automatic_pruning);
1464 if (automatic_pruning) {
1469 const std::vector<uint8_t>& signet_challenge =
1471 obj.
pushKV(
"signet_challenge",
HexStr(signet_challenge));
1482const std::vector<RPCResult> RPCHelpForDeployment{
1484 {
RPCResult::Type::NUM,
"height",
true,
"height of the first block which the rules are or will be enforced (only for \"buried\" type, or \"bip9\" type with \"active\" status)"},
1485 {
RPCResult::Type::BOOL,
"active",
"true if the rules are enforced for the mempool and the next block"},
1488 {
RPCResult::Type::NUM,
"bit",
true,
"the bit (0-28) in the block version field used to signal this softfork (only for \"started\" and \"locked_in\" status)"},
1489 {
RPCResult::Type::NUM_TIME,
"start_time",
"the minimum median time past of a block at which the bit gains its meaning"},
1490 {
RPCResult::Type::NUM_TIME,
"timeout",
"the median time past of a block at which the deployment is considered failed if not yet locked in"},
1491 {
RPCResult::Type::NUM,
"min_activation_height",
"minimum height of blocks for which the rules may be enforced"},
1492 {
RPCResult::Type::STR,
"status",
"status of deployment at specified block (one of \"defined\", \"started\", \"locked_in\", \"active\", \"failed\")"},
1495 {
RPCResult::Type::OBJ,
"statistics",
true,
"numeric statistics about signalling for a softfork (only for \"started\" and \"locked_in\" status)",
1498 {
RPCResult::Type::NUM,
"threshold",
true,
"the number of blocks with the version bit set required to activate the feature (only for \"started\" status)"},
1499 {
RPCResult::Type::NUM,
"elapsed",
"the number of blocks elapsed since the beginning of the current period"},
1500 {
RPCResult::Type::NUM,
"count",
"the number of blocks with the version bit set in the current period"},
1501 {
RPCResult::Type::BOOL,
"possible",
true,
"returns false if there are not enough blocks left in this period to pass activation threshold (only for \"started\" status)"},
1503 {
RPCResult::Type::STR,
"signalling",
true,
"indicates blocks that signalled with a # and blocks that did not with a -"},
1523 "Returns an object containing various state info regarding deployments of consensus changes.\n"
1524 "Consensus changes for which the new rules are enforced from genesis are not listed in \"deployments\".",
1548 if (request.params[0].isNull()) {
1564 uv_flagnames.
push_backV(flagnames.begin(), flagnames.end());
1565 deploymentinfo.
pushKV(
"script_flags", uv_flagnames);
1567 deploymentinfo.
pushKV(
"deployments", DeploymentInfo(blockindex, chainman));
1568 return deploymentinfo;
1591 "Return information about all known tips in the block tree,"
1592 " including the main chain as well as orphaned branches.\n",
1600 {
RPCResult::Type::NUM,
"branchlen",
"zero for main chain, otherwise length of branch connecting the tip to the main chain"},
1602 "Possible values for status:\n"
1603 "1. \"invalid\" This branch contains at least one invalid block\n"
1604 "2. \"headers-only\" Not all blocks for this branch are available, but the headers are valid\n"
1605 "3. \"valid-headers\" All blocks are available for this branch, but they were never fully validated\n"
1606 "4. \"valid-fork\" This branch is not part of the active chain, but is fully validated\n"
1607 "5. \"active\" This is the tip of the active main chain, which is certainly valid"},
1626 std::set<const CBlockIndex*, CompareBlocksByHeight> setTips;
1627 std::set<const CBlockIndex*> setOrphans;
1628 std::set<const CBlockIndex*> setPrevs;
1630 for (
const auto& [
_, block_index] : chainman.
BlockIndex()) {
1631 if (!active_chain.
Contains(block_index)) {
1632 setOrphans.insert(&block_index);
1633 setPrevs.insert(block_index.pprev);
1637 for (std::set<const CBlockIndex*>::iterator it = setOrphans.begin(); it != setOrphans.end(); ++it) {
1638 if (setPrevs.erase(*it) == 0) {
1639 setTips.insert(*it);
1644 setTips.insert(active_chain.
Tip());
1651 obj.
pushKV(
"height", block->nHeight);
1652 obj.
pushKV(
"hash", block->phashBlock->GetHex());
1654 const int branchLen = block->nHeight - active_chain.
FindFork(*block)->
nHeight;
1655 obj.
pushKV(
"branchlen", branchLen);
1658 if (active_chain.
Contains(*block)) {
1664 }
else if (!block->HaveNumChainTxs()) {
1666 status =
"headers-only";
1669 status =
"valid-fork";
1672 status =
"valid-headers";
1677 obj.
pushKV(
"status", status);
1691 "Treats a block as if it were received before others with the same work.\n"
1692 "\nA later preciousblock call can override the effect of an earlier one.\n"
1693 "\nThe effects of preciousblock are not retained across restarts.\n",
1753 "Permanently marks a block as invalid, as if it violated a consensus rule.\n",
1783 chainman.RecalculateBestHeader();
1798 "Removes invalidity status of a block, its ancestors and its descendants, reconsider them for activation.\n"
1799 "This can be used to undo the effects of invalidateblock.\n",
1824 "Compute statistics about the total number and rate of transactions in the chain.\n",
1834 "The total number of transactions in the chain up to that point, if known. "
1835 "It may be unknown when using assumeutxo."},
1837 {
RPCResult::Type::NUM,
"window_final_block_height",
"The height of the final block in the window."},
1839 {
RPCResult::Type::NUM,
"window_interval",
true,
"The elapsed time in the window in seconds. Only returned if \"window_block_count\" is > 0"},
1841 "The number of transactions in the window. "
1842 "Only returned if \"window_block_count\" is > 0 and if txcount exists for the start and end of the window."},
1844 "The average rate of transactions per second in the window. "
1845 "Only returned if \"window_interval\" is > 0 and if window_tx_count exists."},
1857 if (request.params[1].isNull()) {
1874 if (request.params[0].isNull()) {
1875 blockcount = std::max(0, std::min(blockcount, pindex->
nHeight - 1));
1877 blockcount = request.params[0].getInt<
int>();
1879 if (blockcount < 0 || (blockcount > 0 && blockcount >= pindex->
nHeight)) {
1885 const int64_t nTimeDiff{pindex->
GetMedianTimePast() - past_block.GetMedianTimePast()};
1893 ret.pushKV(
"window_final_block_height", pindex->
nHeight);
1894 ret.pushKV(
"window_block_count", blockcount);
1895 if (blockcount > 0) {
1896 ret.pushKV(
"window_interval", nTimeDiff);
1898 const auto window_tx_count = pindex->
m_chain_tx_count - past_block.m_chain_tx_count;
1899 ret.pushKV(
"window_tx_count", window_tx_count);
1900 if (nTimeDiff > 0) {
1901 ret.pushKV(
"txrate",
double(window_tx_count) / nTimeDiff);
1914 size_t size = scores.size();
1919 std::sort(scores.begin(), scores.end());
1920 if (size % 2 == 0) {
1921 return (scores[size / 2 - 1] + scores[size / 2]) / 2;
1923 return scores[size / 2];
1929 if (scores.empty()) {
1933 std::sort(scores.begin(), scores.end());
1937 total_weight / 10.0, total_weight / 4.0, total_weight / 2.0, (total_weight * 3.0) / 4.0, (total_weight * 9.0) / 10.0
1940 int64_t next_percentile_index = 0;
1941 int64_t cumulative_weight = 0;
1942 for (
const auto& element : scores) {
1943 cumulative_weight += element.second;
1944 while (next_percentile_index < NUM_GETBLOCKSTATS_PERCENTILES && cumulative_weight >= weights[next_percentile_index]) {
1945 result[next_percentile_index] = element.first;
1946 ++next_percentile_index;
1952 result[i] = scores.back().first;
1957static inline bool SetHasKeys(
const std::set<T>& set) {
return false;}
1958template<
typename T,
typename Tk,
typename... Args>
1959static inline bool SetHasKeys(
const std::set<T>& set,
const Tk& key,
const Args&...
args)
1971 "Compute per block statistics for a given window. All amounts are in satoshis.\n"
1972 "It won't work for some heights with pruning.\n",
1977 .type_str = {
"",
"string or numeric"},
1993 {
RPCResult::Type::ARR_FIXED,
"feerate_percentiles",
true,
"Feerates at the 10th, 25th, 50th, 75th, and 90th percentile weight unit (in satoshis per virtual byte)",
2018 {
RPCResult::Type::NUM,
"total_out",
true,
"Total amount in all outputs (excluding coinbase and thus reward [ie subsidy + totalfee])"},
2023 {
RPCResult::Type::NUM,
"utxo_increase",
true,
"The increase/decrease in the number of unspent outputs (not discounting op_return and similar)"},
2024 {
RPCResult::Type::NUM,
"utxo_size_inc",
true,
"The increase/decrease in size for the utxo index (not discounting op_return and similar)"},
2025 {
RPCResult::Type::NUM,
"utxo_increase_actual",
true,
"The increase/decrease in the number of unspent outputs, not counting unspendables"},
2026 {
RPCResult::Type::NUM,
"utxo_size_inc_actual",
true,
"The increase/decrease in size for the utxo index, not counting unspendables"},
2029 HelpExampleCli(
"getblockstats", R
"('"00000000c937983704a73af28acdec37b049d214adbda81d7e2a3dd146f6ed09"' '["minfeerate","avgfeerate"]')") +
2030 HelpExampleCli("getblockstats", R
"(1000 '["minfeerate","avgfeerate"]')") +
2031 HelpExampleRpc("getblockstats", R
"("00000000c937983704a73af28acdec37b049d214adbda81d7e2a3dd146f6ed09", ["minfeerate","avgfeerate"])") +
2032 HelpExampleRpc("getblockstats", R
"(1000, ["minfeerate","avgfeerate"])")
2039 std::set<std::string> stats;
2040 if (!request.params[1].isNull()) {
2042 for (
unsigned int i = 0; i < stats_univalue.
size(); i++) {
2043 const std::string stat = stats_univalue[i].
get_str();
2051 const bool do_all = stats.size() == 0;
2052 const bool do_mediantxsize = do_all || stats.contains(
"mediantxsize");
2053 const bool do_medianfee = do_all || stats.contains(
"medianfee");
2054 const bool do_feerate_percentiles = do_all || stats.contains(
"feerate_percentiles");
2055 const bool loop_inputs = do_all || do_medianfee || do_feerate_percentiles ||
2056 SetHasKeys(stats,
"utxo_increase",
"utxo_increase_actual",
"utxo_size_inc",
"utxo_size_inc_actual",
"totalfee",
"avgfee",
"avgfeerate",
"minfee",
"maxfee",
"minfeerate",
"maxfeerate");
2057 const bool loop_outputs = do_all || loop_inputs || stats.contains(
"total_out");
2058 const bool do_calculate_size = do_mediantxsize ||
2059 SetHasKeys(stats,
"total_size",
"avgtxsize",
"mintxsize",
"maxtxsize",
"swtotal_size");
2060 const bool do_calculate_weight = do_all ||
SetHasKeys(stats,
"total_weight",
"avgfeerate",
"swtotal_weight",
"avgfeerate",
"feerate_percentiles",
"minfeerate",
"maxfeerate");
2061 const bool do_calculate_sw = do_all ||
SetHasKeys(stats,
"swtxs",
"swtotal_size",
"swtotal_weight");
2070 int64_t maxtxsize = 0;
2072 int64_t outputs = 0;
2073 int64_t swtotal_size = 0;
2074 int64_t swtotal_weight = 0;
2076 int64_t total_size = 0;
2077 int64_t total_weight = 0;
2079 int64_t utxo_size_inc = 0;
2080 int64_t utxo_size_inc_actual = 0;
2081 std::vector<CAmount> fee_array;
2082 std::vector<std::pair<CAmount, int64_t>> feerate_array;
2083 std::vector<int64_t> txsize_array;
2085 for (
size_t i = 0; i < block.
vtx.size(); ++i) {
2086 const auto& tx = block.
vtx.at(i);
2087 outputs += tx->vout.size();
2092 tx_total_out +=
out.nValue;
2095 utxo_size_inc += out_size;
2101 if (
out.scriptPubKey.IsUnspendable())
continue;
2104 utxo_size_inc_actual += out_size;
2108 if (tx->IsCoinBase()) {
2112 inputs += tx->vin.size();
2113 total_out += tx_total_out;
2115 int64_t tx_size = 0;
2116 if (do_calculate_size) {
2118 tx_size = tx->ComputeTotalSize();
2119 if (do_mediantxsize) {
2120 txsize_array.push_back(tx_size);
2122 maxtxsize = std::max(maxtxsize, tx_size);
2123 mintxsize = std::min(mintxsize, tx_size);
2124 total_size += tx_size;
2128 if (do_calculate_weight) {
2130 total_weight += weight;
2133 if (do_calculate_sw && tx->HasWitness()) {
2135 swtotal_size += tx_size;
2136 swtotal_weight += weight;
2141 const auto& txundo = blockUndo.
vtxundo.at(i - 1);
2142 for (
const Coin& coin: txundo.vprevout) {
2145 tx_total_in += prevoutput.
nValue;
2147 utxo_size_inc -= prevout_size;
2148 utxo_size_inc_actual -= prevout_size;
2151 CAmount txfee = tx_total_in - tx_total_out;
2154 fee_array.push_back(txfee);
2156 maxfee = std::max(maxfee, txfee);
2157 minfee = std::min(minfee, txfee);
2162 if (do_feerate_percentiles) {
2163 feerate_array.emplace_back(feerate, weight);
2165 maxfeerate = std::max(maxfeerate, feerate);
2166 minfeerate = std::min(minfeerate, feerate);
2175 feerates_res.
push_back(feerate_percentiles[i]);
2179 ret_all.
pushKV(
"avgfee", (block.
vtx.size() > 1) ? totalfee / (block.
vtx.size() - 1) : 0);
2181 ret_all.
pushKV(
"avgtxsize", (block.
vtx.size() > 1) ? total_size / (block.
vtx.size() - 1) : 0);
2183 ret_all.
pushKV(
"feerate_percentiles", std::move(feerates_res));
2185 ret_all.
pushKV(
"ins", inputs);
2186 ret_all.
pushKV(
"maxfee", maxfee);
2187 ret_all.
pushKV(
"maxfeerate", maxfeerate);
2188 ret_all.
pushKV(
"maxtxsize", maxtxsize);
2193 ret_all.
pushKV(
"minfeerate", (minfeerate ==
MAX_MONEY) ? 0 : minfeerate);
2195 ret_all.
pushKV(
"outs", outputs);
2197 ret_all.
pushKV(
"swtotal_size", swtotal_size);
2198 ret_all.
pushKV(
"swtotal_weight", swtotal_weight);
2199 ret_all.
pushKV(
"swtxs", swtxs);
2201 ret_all.
pushKV(
"total_out", total_out);
2202 ret_all.
pushKV(
"total_size", total_size);
2203 ret_all.
pushKV(
"total_weight", total_weight);
2204 ret_all.
pushKV(
"totalfee", totalfee);
2205 ret_all.
pushKV(
"txs", block.
vtx.size());
2206 ret_all.
pushKV(
"utxo_increase", outputs - inputs);
2207 ret_all.
pushKV(
"utxo_size_inc", utxo_size_inc);
2208 ret_all.
pushKV(
"utxo_increase_actual", utxos - inputs);
2209 ret_all.
pushKV(
"utxo_size_inc_actual", utxo_size_inc_actual);
2216 for (
const std::string& stat : stats) {
2217 const UniValue& value = ret_all[stat];
2221 ret.pushKV(stat, value);
2230bool FindScriptPubKey(std::atomic<int>& scan_progress,
const std::atomic<bool>& should_abort, int64_t&
count,
CCoinsViewCursor* cursor,
const std::set<CScript>& needles, std::map<COutPoint, Coin>& out_results, std::function<
void()>& interruption_point)
2234 while (cursor->
Valid()) {
2237 if (!cursor->
GetKey(key) || !cursor->
GetValue(coin))
return false;
2238 if (++
count % 8192 == 0) {
2239 interruption_point();
2245 if (
count % 256 == 0) {
2248 scan_progress = (int)(high * 100.0 / 65536.0 + 0.5);
2251 out_results.emplace(key, coin);
2255 scan_progress = 100;
2291 "\"start\" for starting a scan\n"
2292 "\"abort\" for aborting the current scan (returns true when abort was successful)\n"
2293 "\"status\" for progress report (in %) of the current scan"
2306 "Every scan object is either a string descriptor or an object:",
2316 "True if scan will be aborted (not necessarily before this RPC returns), or false if there is no scan to abort"
2319 "when action=='status' and no scan is in progress - possibly already completed",
RPCResult::Type::NONE,
"",
""
2330 const std::string EXAMPLE_DESCRIPTOR_RAW =
"raw(76a91411b366edfc0a8b66feebae5c2e25a7b6a5d1cf3188ac)#fm24fxxy";
2334 "Scans the unspent transaction output set for entries that match certain output descriptors.\n"
2335 "Examples of output descriptors are:\n"
2336 " addr(<address>) Outputs whose output script corresponds to the specified address (does not include P2PK)\n"
2337 " raw(<hex script>) Outputs whose output script equals the specified hex-encoded bytes\n"
2338 " combo(<pubkey>) P2PK, P2PKH, P2WPKH, and P2SH-P2WPKH outputs for the given pubkey\n"
2339 " pkh(<pubkey>) P2PKH outputs for the given pubkey\n"
2340 " sh(multi(<n>,<pubkey>,<pubkey>,...)) P2SH-multisig outputs for the given threshold and pubkeys\n"
2341 " tr(<pubkey>) P2TR\n"
2342 " tr(<pubkey>,{pk(<pubkey>)}) P2TR with single fallback pubkey in tapscript\n"
2343 " rawtr(<pubkey>) P2TR with the specified key as output key rather than inner\n"
2344 " wsh(and_v(v:pk(<pubkey>),after(2))) P2WSH miniscript with mandatory pubkey and a timelock\n"
2345 "\nIn the above, <pubkey> either refers to a fixed public key in hexadecimal notation, or to an xpub/xprv optionally followed by one\n"
2346 "or more path elements separated by \"/\", and optionally ending in \"/*\" (unhardened), or \"/*'\" or \"/*h\" (hardened) to specify all\n"
2347 "unhardened or hardened child keys.\n"
2348 "In the latter case, a range needs to be specified by below if different from 1000.\n"
2349 "For more information on output descriptors, see the documentation in the doc/descriptors.md file.\n",
2372 {
RPCResult::Type::NUM,
"confirmations",
"Number of confirmations of the unspent transaction output when the scan was done"},
2382 HelpExampleCli(
"scantxoutset",
"start \'[\"" + EXAMPLE_DESCRIPTOR_RAW +
"\"]\'") +
2385 HelpExampleRpc(
"scantxoutset",
"\"start\", [\"" + EXAMPLE_DESCRIPTOR_RAW +
"\"]") +
2392 const auto action{self.
Arg<std::string_view>(
"action")};
2393 if (action ==
"status") {
2401 }
else if (action ==
"abort") {
2410 }
else if (action ==
"start") {
2416 if (request.params.size() < 2) {
2420 std::set<CScript> needles;
2421 std::map<CScript, std::string> descriptors;
2431 descriptors.emplace(std::move(
script), std::move(inferred));
2437 std::vector<CTxOut> input_txos;
2438 std::map<COutPoint, Coin> coins;
2441 std::unique_ptr<CCoinsViewCursor> pcursor;
2453 result.
pushKV(
"success", res);
2458 for (
const auto& it : coins) {
2460 const Coin& coin = it.second;
2463 input_txos.push_back(txo);
2468 unspent.
pushKV(
"vout", outpoint.
n);
2474 unspent.
pushKV(
"blockhash", coinb_block.GetBlockHash().GetHex());
2479 result.
pushKV(
"unspents", std::move(unspents));
2523 for (
const auto& tx : block.vtx) {
2524 if (std::any_of(tx->vout.cbegin(), tx->vout.cend(), [&](
const auto& txout) {
2525 return needles.contains(std::vector<unsigned char>(txout.scriptPubKey.begin(), txout.scriptPubKey.end()));
2531 for (
const auto& txundo : block_undo.vtxundo) {
2532 if (std::any_of(txundo.vprevout.cbegin(), txundo.vprevout.cend(), [&](
const auto& coin) {
2533 return needles.contains(std::vector<unsigned char>(coin.out.scriptPubKey.begin(), coin.out.scriptPubKey.end()));
2546 "Return relevant blockhashes for given descriptors (requires blockfilterindex).\n"
2547 "This call may take several minutes. Make sure to use no RPC timeout (bitcoin-cli -rpcclienttimeout=0)",
2556 {
"filter_false_positives",
RPCArg::Type::BOOL,
RPCArg::Default{
false},
"Filter false positives (slower and may fail on pruned nodes). Otherwise they may occur at a rate of 1/M"},
2578 HelpExampleCli(
"scanblocks",
"start '[\"addr(bcrt1q4u4nsgk6ug0sqz7r3rj9tykjxrsl0yy4d0wwte)\"]' 300000") +
2579 HelpExampleCli(
"scanblocks",
"start '[\"addr(bcrt1q4u4nsgk6ug0sqz7r3rj9tykjxrsl0yy4d0wwte)\"]' 100 150 basic") +
2581 HelpExampleRpc(
"scanblocks",
"\"start\", [\"addr(bcrt1q4u4nsgk6ug0sqz7r3rj9tykjxrsl0yy4d0wwte)\"], 300000") +
2582 HelpExampleRpc(
"scanblocks",
"\"start\", [\"addr(bcrt1q4u4nsgk6ug0sqz7r3rj9tykjxrsl0yy4d0wwte)\"], 100, 150, \"basic\"") +
2588 auto action{self.
Arg<std::string_view>(
"action")};
2589 if (action ==
"status") {
2598 }
else if (action ==
"abort") {
2607 }
else if (action ==
"start") {
2612 auto filtertype_name{self.
Arg<std::string_view>(
"filtertype")};
2620 bool filter_false_positives{options.exists(
"filter_false_positives") ? options[
"filter_false_positives"].get_bool() :
false};
2636 start_index = active_chain.
Genesis();
2637 stop_block = active_chain.
Tip();
2638 if (!request.params[2].isNull()) {
2639 start_index = active_chain[request.params[2].getInt<
int>()];
2644 if (!request.params[3].isNull()) {
2645 stop_block = active_chain[request.params[3].getInt<
int>()];
2664 const int amount_per_chunk = 10000;
2665 std::vector<BlockFilter> filters;
2666 int start_block_height = start_index->
nHeight;
2667 const int total_blocks_to_process = stop_block->
nHeight - start_block_height;
2672 bool completed =
true;
2676 node.rpc_interruption_point();
2683 int start_block = !end_range ? start_index->
nHeight : start_index->
nHeight + 1;
2684 end_range = (start_block + amount_per_chunk < stop_block->
nHeight) ?
2691 if (filter.GetFilter().MatchAny(needle_set)) {
2692 if (filter_false_positives) {
2701 blocks.
push_back(filter.GetBlockHash().GetHex());
2705 start_index = end_range;
2708 int blocks_processed = end_range->
nHeight - start_block_height;
2709 if (total_blocks_to_process > 0) {
2717 }
while (start_index != stop_block);
2719 ret.pushKV(
"from_height", start_block_height);
2720 ret.pushKV(
"to_height", start_index->
nHeight);
2721 ret.pushKV(
"relevant_blocks", std::move(blocks));
2722 ret.pushKV(
"completed", completed);
2734 "getdescriptoractivity",
2735 "Get spend and receive activity associated with a set of descriptors for a set of blocks. "
2736 "This command pairs well with the `relevant_blocks` output of `scanblocks()`.\n"
2737 "This call may take several minutes. If you encounter timeouts, try specifying no RPC timeout (bitcoin-cli -rpcclienttimeout=0)",
2775 }, {.skip_type_check=
true}, },
2779 HelpExampleCli(
"getdescriptoractivity",
"'[\"000000000000000000001347062c12fded7c528943c8ce133987e2e2f5a840ee\"]' '[\"addr(bc1qzl6nsgqzu89a66l50cvwapnkw5shh23zarqkw9)\"]'")
2788 struct CompareByHeightAscending {
2794 std::set<const CBlockIndex*, CompareByHeightAscending> blockindexes_sorted;
2808 blockindexes_sorted.insert(pindex);
2812 std::set<CScript> scripts_to_watch;
2820 scripts_to_watch.insert(
script);
2824 const auto AddSpend = [&](
2836 event.pushKV(
"type",
"spend");
2839 event.pushKV(
"blockhash", index->GetBlockHash().ToString());
2840 event.pushKV(
"height", index->nHeight);
2842 event.pushKV(
"spend_txid", tx->GetHash().ToString());
2843 event.pushKV(
"spend_vin", vin);
2844 event.pushKV(
"prevout_txid", txin.prevout.hash.ToString());
2845 event.pushKV(
"prevout_vout", txin.prevout.n);
2846 event.pushKV(
"prevout_spk", spkUv);
2856 event.pushKV(
"type",
"receive");
2859 event.pushKV(
"blockhash", index->GetBlockHash().ToString());
2860 event.pushKV(
"height", index->nHeight);
2862 event.pushKV(
"txid", tx->GetHash().ToString());
2863 event.pushKV(
"vout", vout);
2864 event.pushKV(
"output_spk", spkUv);
2876 for (
const CBlockIndex* blockindex : blockindexes_sorted) {
2880 for (
size_t i = 0; i < block.vtx.size(); ++i) {
2881 const auto& tx = block.vtx.at(i);
2883 if (!tx->IsCoinBase()) {
2885 const auto& txundo = block_undo.
vtxundo.at(i - 1);
2887 for (
size_t vin_idx = 0; vin_idx < tx->vin.size(); ++vin_idx) {
2888 const auto& coin = txundo.vprevout.at(vin_idx);
2889 const auto& txin = tx->vin.at(vin_idx);
2897 for (
size_t vout_idx = 0; vout_idx < tx->vout.size(); ++vout_idx) {
2898 const auto& vout = tx->vout.at(vout_idx);
2899 if (scripts_to_watch.contains(vout.scriptPubKey)) {
2900 activity.
push_back(AddReceive(vout, blockindex, vout_idx, tx));
2906 bool search_mempool =
true;
2907 if (!request.params[2].isNull()) {
2908 search_mempool = request.params[2].get_bool();
2911 if (search_mempool) {
2918 const auto& tx = e.GetSharedTx();
2920 for (
size_t vin_idx = 0; vin_idx < tx->vin.size(); ++vin_idx) {
2923 const auto& txin = tx->vin.at(vin_idx);
2924 std::optional<Coin> coin = coins_view.
GetCoin(txin.prevout);
2932 if (txin.prevout.n >= prev_tx->vout.size()) {
2933 throw std::runtime_error(
"Invalid output index");
2935 const CTxOut&
out = prev_tx->vout[txin.prevout.n];
2936 scriptPubKey =
out.scriptPubKey;
2941 scriptPubKey =
out.scriptPubKey;
2945 if (scripts_to_watch.contains(scriptPubKey)) {
2948 scriptPubKey, value, tx, vin_idx, txin,
nullptr));
2952 for (
size_t vout_idx = 0; vout_idx < tx->vout.size(); ++vout_idx) {
2953 const auto& vout = tx->vout.at(vout_idx);
2954 if (scripts_to_watch.contains(vout.scriptPubKey)) {
2955 activity.
push_back(AddReceive(vout,
nullptr, vout_idx, tx));
2961 ret.pushKV(
"activity", activity);
2971 "Retrieve a BIP 157 content filter for a particular block.\n",
2983 HelpExampleCli(
"getblockfilter",
"\"00000000c937983704a73af28acdec37b049d214adbda81d7e2a3dd146f6ed09\" \"basic\"") +
2984 HelpExampleRpc(
"getblockfilter",
"\"00000000c937983704a73af28acdec37b049d214adbda81d7e2a3dd146f6ed09\", \"basic\"")
2989 auto filtertype_name{self.
Arg<std::string_view>(
"filtertype")};
3002 bool block_was_connected;
3013 bool index_ready = index->BlockUntilSyncedToCurrentChain();
3020 std::string errmsg =
"Filter not found.";
3022 if (!block_was_connected) {
3024 errmsg +=
" Block was not connected to active chain.";
3025 }
else if (!index_ready) {
3027 errmsg +=
" Block filters are still in the process of being indexed.";
3030 errmsg +=
" This error is unexpected and indicates index corruption.";
3038 ret.pushKV(
"header", filter_header.
GetHex());
3050 static constexpr const char*
LOCK_NAME{
"dumptxoutset-rollback"};
3076 "Write the serialized UTXO set to a file. This can be used in loadtxoutset afterwards if this snapshot height is supported in the chainparams as well.\n"
3077 "This creates a temporary UTXO database when rolling back, keeping the main chain intact. Should the node experience an unclean shutdown the temporary database may need to be removed from the datadir manually.\n"
3078 "For deep rollbacks, make sure to use no RPC timeout (bitcoin-cli -rpcclienttimeout=0) as it may take several minutes.",
3081 {
"type",
RPCArg::Type::STR,
RPCArg::Default(
""),
"The type of snapshot to create. Can be \"latest\" to create a snapshot of the current UTXO set or \"rollback\" to temporarily roll back the state of the node to a historical block before creating the snapshot of a historical UTXO set. This parameter can be omitted if a separate \"rollback\" named parameter is specified indicating the height or hash of a specific historical block. If \"rollback\" is specified and separate \"rollback\" named parameter is not specified, this will roll back to the latest valid snapshot block that can currently be loaded with loadtxoutset."},
3085 "Height or hash of the block to roll back to before creating the snapshot. Note: The further this number is from the tip, the longer this process will take. Consider setting a higher -rpcclienttimeout value in this case.",
3087 {
"in_memory",
RPCArg::Type::BOOL,
RPCArg::Default{
false},
"If true, the temporary UTXO-set database used during rollback is kept entirely in memory. This can significantly speed up the process but requires sufficient free RAM (over 10 GB on mainnet)."},
3099 {
RPCResult::Type::NUM,
"nchaintx",
"the number of transactions in the chain up to and including the base block"},
3103 HelpExampleCli(
"-rpcclienttimeout=0 dumptxoutset",
"utxo.dat latest") +
3104 HelpExampleCli(
"-rpcclienttimeout=0 dumptxoutset",
"utxo.dat rollback") +
3105 HelpExampleCli(
"-rpcclienttimeout=0 -named dumptxoutset", R
"(utxo.dat rollback=853456)") +
3106 HelpExampleCli("-rpcclienttimeout=0 -named dumptxoutset", R
"(utxo.dat rollback=853456 in_memory=true)")
3113 const auto snapshot_type{self.
Arg<std::string_view>(
"type")};
3115 if (options.exists(
"rollback")) {
3116 if (!snapshot_type.empty() && snapshot_type !=
"rollback") {
3120 }
else if (snapshot_type ==
"rollback") {
3121 auto snapshot_heights =
node.chainman->GetParams().GetAvailableSnapshotHeights();
3123 auto max_height = std::max_element(snapshot_heights.begin(), snapshot_heights.end());
3125 }
else if (snapshot_type ==
"latest") {
3133 const auto path_info{fs::status(path)};
3136 const fs::path temppath = fs::is_fifo(path_info) ? path : path +
".incomplete";
3138 if (
fs::exists(path_info) && !fs::is_fifo(path_info)) {
3141 path.utf8string() +
" already exists. If you are sure this is what you want, "
3142 "move it out of the way first");
3147 if (afile.IsNull()) {
3150 "Couldn't open file " + temppath.utf8string() +
" for writing.");
3155 if (target_index == tip) {
3161 std::optional<TemporaryPruneLock> temp_prune_lock;
3162 if (
node.chainman->m_blockman.IsPruneMode()) {
3166 if (first_block.nHeight > target_index->nHeight) {
3169 temp_prune_lock.emplace(
node.chainman->m_blockman, target_index->nHeight);
3172 const bool in_memory{options.exists(
"in_memory") ? options[
"in_memory"].get_bool() :
false};
3182 if (!fs::is_fifo(path_info)) {
3183 fs::rename(temppath, path);
3200 fs::create_directories(
m_path);
3204 LogInfo(
"Failed to clean up temporary UTXO database at %s, please remove it manually.",
3215 const fs::path& path,
3216 const fs::path& tmppath,
3217 const bool in_memory)
3224 std::optional<TemporaryUTXODatabase> temp_db_cleaner;
3226 temp_db_cleaner.emplace(temp_db_path);
3228 LogInfo(
"Using in-memory database for UTXO-set rollback (this may require significant RAM).");
3233 .
path = temp_db_path,
3235 .memory_only = in_memory,
3241 std::unique_ptr<CCoinsViewDB> temp_db = std::make_unique<CCoinsViewDB>(
3242 std::move(db_params),
3247 LogInfo(
"Copying current UTXO set to temporary database.");
3250 std::unique_ptr<CCoinsViewCursor> cursor;
3259 size_t coins_count = 0;
3260 while (cursor->Valid()) {
3261 node.rpc_interruption_point();
3265 if (cursor->GetKey(key) && cursor->GetValue(coin)) {
3266 temp_cache.
AddCoin(key, std::move(coin),
false);
3270 if (coins_count % 10'000'000 == 0) {
3271 LogInfo(
"Copying UTXO set: %uM coins copied.", coins_count / 1'000'000);
3275 if (coins_count % 100'000 == 0) {
3283 LogInfo(
"UTXO set copy complete: %u coins total", coins_count);
3286 LogInfo(
"Rolling back from height %d to %d", tip->nHeight, target->
nHeight);
3289 const size_t total_blocks{
static_cast<size_t>(block_index->nHeight - target->
nHeight)};
3291 rollback_cache.
SetBestBlock(block_index->GetBlockHash());
3292 size_t blocks_processed = 0;
3293 int last_progress{0};
3296 while (block_index->nHeight > target->
nHeight) {
3297 node.rpc_interruption_point();
3300 if (!
node.chainman->m_blockman.ReadBlock(block, *block_index)) {
3302 strprintf(
"Failed to read block at height %d", block_index->nHeight));
3305 WITH_LOCK(
::cs_main, res = chainstate.DisconnectBlock(block, block_index, rollback_cache));
3308 strprintf(
"Failed to roll back block at height %d", block_index->nHeight));
3312 int progress{
static_cast<int>(blocks_processed * 100 / total_blocks)};
3313 if (progress >= last_progress + 5) {
3314 LogInfo(
"Rolled back %d%% of blocks.", progress);
3315 last_progress = progress;
3316 rollback_cache.
Flush();
3319 block_index = block_index->pprev;
3323 rollback_cache.
Flush();
3325 LogInfo(
"Rollback complete. Computing UTXO statistics for created txoutset dump.");
3326 std::optional<CCoinsStats> maybe_stats =
GetUTXOStats(temp_db.get(),
3328 CoinStatsHashType::HASH_SERIALIZED,
3329 node.rpc_interruption_point);
3335 std::unique_ptr<CCoinsViewCursor> pcursor{temp_db->Cursor()};
3340 LogInfo(
"Writing snapshot to disk.");
3348 node.rpc_interruption_point);
3354 const std::function<
void()>& interruption_point)
3356 std::unique_ptr<CCoinsViewCursor> pcursor;
3357 std::optional<CCoinsStats> maybe_stats;
3395 const fs::path& path,
3396 const fs::path& temppath,
3397 const std::function<
void()>& interruption_point)
3410 unsigned int iter{0};
3411 size_t written_coins_count{0};
3412 std::vector<std::pair<uint32_t, Coin>> coins;
3421 auto write_coins_to_file = [&](
AutoFile& afile,
const Txid& last_hash,
const std::vector<std::pair<uint32_t, Coin>>& coins,
size_t& written_coins_count) {
3424 for (
const auto& [n, coin] : coins) {
3427 ++written_coins_count;
3432 last_hash = key.
hash;
3433 while (pcursor->
Valid()) {
3434 if (iter % 5000 == 0) interruption_point();
3437 if (key.
hash != last_hash) {
3438 write_coins_to_file(afile, last_hash, coins, written_coins_count);
3439 last_hash = key.
hash;
3442 coins.emplace_back(key.
n, coin);
3447 if (!coins.empty()) {
3448 write_coins_to_file(afile, last_hash, coins, written_coins_count);
3453 if (afile.
fclose() != 0) {
3454 throw std::ios_base::failure(
3459 result.
pushKV(
"coins_written", written_coins_count);
3462 result.
pushKV(
"path", path.utf8string());
3472 const fs::path& path,
3473 const fs::path& tmppath)
3483 node.rpc_interruption_point);
3490 "Load the serialized UTXO set from a file.\n"
3491 "Once this snapshot is loaded, its contents will be "
3492 "deserialized into a second chainstate data structure, which is then used to sync to "
3493 "the network's tip. "
3494 "Meanwhile, the original chainstate will complete the initial block download process in "
3495 "the background, eventually validating up to the block that the snapshot is based upon.\n\n"
3497 "The result is a usable bitcoind instance that is current with the network tip in a "
3498 "matter of minutes rather than hours. UTXO snapshot are typically obtained from "
3499 "third-party sources (HTTP, torrent, etc.) which is reasonable since their "
3500 "contents are always checked by hash.\n\n"
3502 "You can find more information on this process in the `assumeutxo` design "
3503 "document (<https://github.com/bitcoin/bitcoin/blob/master/doc/design/assumeutxo.md>).",
3508 "path to the snapshot file. If relative, will be prefixed by datadir."},
3530 if (afile.IsNull()) {
3533 "Couldn't open file " + path.utf8string() +
" for reading.");
3539 }
catch (
const std::ios_base::failure& e) {
3544 if (!activation_result) {
3558 result.
pushKV(
"coins_loaded", metadata.m_coins_count);
3559 result.
pushKV(
"tip_hash", snapshot_index.GetBlockHash().ToString());
3560 result.
pushKV(
"base_height", snapshot_index.nHeight);
3574 {
RPCResult::Type::STR_HEX,
"snapshot_blockhash",
true,
"the base block of the snapshot this chainstate is based on, if any"},
3577 {
RPCResult::Type::BOOL,
"validated",
"whether the chainstate is fully validated. True if all blocks in the chainstate were validated, false if the chain is based on a snapshot and the snapshot has not yet been validated."},
3584 "Return information about chainstates.\n",
3606 if (!
cs.m_chain.Tip()) {
3618 data.pushKV(
"coins_db_cache_bytes",
cs.m_coinsdb_cache_size_bytes);
3619 data.pushKV(
"coins_tip_cache_bytes",
cs.m_coinstip_cache_size_bytes);
3620 if (
cs.m_from_snapshot_blockhash) {
3621 data.pushKV(
"snapshot_blockhash",
cs.m_from_snapshot_blockhash->ToString());
3627 obj.
pushKV(
"headers", chainman.m_best_header ? chainman.m_best_header->nHeight : -1);
3630 obj_chainstates.push_back(make_chain_data(*
cs));
3633 obj.
pushKV(
"chainstates", std::move(obj_chainstates));
3674 for (
const auto& c : commands) {
3675 t.appendCommand(c.name, &c);
static constexpr CAmount MAX_MONEY
No amount larger than this (in satoshi) is valid.
bool MoneyRange(const CAmount &nValue)
int64_t CAmount
Amount in satoshis (Can be negative)
fs::path AbsPathForConfigVal(const ArgsManager &args, const fs::path &path, bool net_specific=true)
Most paths passed as configuration arguments are treated as relative to the datadir if they are not a...
static std::vector< RPCResult > GetBlockFields(RPCResult tx_result, std::optional< std::string > elision_msg=std::nullopt)
static const auto scan_result_abort
static std::atomic< bool > g_scan_in_progress
static bool SetHasKeys(const std::set< T > &set)
static T CalculateTruncatedMedian(std::vector< T > &scores)
static int ComputeNextBlockAndDepth(const CBlockIndex &tip, const CBlockIndex &blockindex, const CBlockIndex *&next)
CoinStatsHashType ParseHashType(std::string_view hash_type_input)
static RPCMethod getdifficulty()
static CBlockUndo GetUndoChecked(BlockManager &blockman, const CBlockIndex &blockindex)
static std::vector< std::byte > GetRawBlockChecked(BlockManager &blockman, const CBlockIndex &blockindex)
static RPCMethod getblockheader()
std::tuple< std::unique_ptr< CCoinsViewCursor >, CCoinsStats, const CBlockIndex * > PrepareUTXOSnapshot(Chainstate &chainstate, const std::function< void()> &interruption_point={}) EXCLUSIVE_LOCKS_REQUIRED(UniValue WriteUTXOSnapshot(Chainstate &chainstate, CCoinsViewCursor *pcursor, CCoinsStats *maybe_stats, const CBlockIndex *tip, AutoFile &&afile, const fs::path &path, const fs::path &temppath, const std::function< void()> &interruption_point={})
static RPCMethod reconsiderblock()
static RPCMethod getblockfilter()
static RPCMethod pruneblockchain()
static RPCMethod scanblocks()
std::tuple< std::unique_ptr< CCoinsViewCursor >, CCoinsStats, const CBlockIndex * > PrepareUTXOSnapshot(Chainstate &chainstate, const std::function< void()> &interruption_point)
static std::atomic< int > g_scanfilter_progress_height
static const auto output_descriptor_obj
static RPCMethod getbestblockhash()
void CheckBlockDataAvailability(BlockManager &blockman, const CBlockIndex &blockindex, bool check_for_undo)
static std::atomic< bool > g_scanfilter_should_abort_scan
static std::atomic< int > g_scanfilter_progress
RAII object to prevent concurrency issue when scanning blockfilters.
static RPCMethod preciousblock()
static RPCMethod getdescriptoractivity()
static void SoftForkDescPushBack(const CBlockIndex *blockindex, UniValue &softforks, const ChainstateManager &chainman, Consensus::BuriedDeployment dep)
UniValue coinbaseTxToJSON(const CTransaction &coinbase_tx)
Serialize coinbase transaction metadata.
static constexpr size_t PER_UTXO_OVERHEAD
static RPCMethod getblockcount()
double GetDifficulty(const CBlockIndex &blockindex)
Get the difficulty of the net wrt to the given block index.
static const auto scan_objects_arg_desc
static RPCMethod scantxoutset()
static RPCMethod waitforblock()
static bool CheckBlockFilterMatches(BlockManager &blockman, const CBlockIndex &blockindex, const GCSFilter::ElementSet &needles)
void InvalidateBlock(ChainstateManager &chainman, const uint256 block_hash)
static RPCMethod getblockfrompeer()
static RPCMethod invalidateblock()
UniValue CreateUTXOSnapshot(node::NodeContext &node, Chainstate &chainstate, AutoFile &&afile, const fs::path &path, const fs::path &tmppath)
Helper to create UTXO snapshots given a chainstate and a file handle.
static RPCMethod waitfornewblock()
static std::atomic< int > g_scan_progress
RAII object to prevent concurrency issue when scanning the txout set.
static CBlock GetBlockChecked(BlockManager &blockman, const CBlockIndex &blockindex)
std::optional< int > GetPruneHeight(const BlockManager &blockman, const CChain &chain)
Return height of highest block that has been pruned, or std::nullopt if no blocks have been pruned.
UniValue blockToJSON(BlockManager &blockman, const CBlock &block, const CBlockIndex &tip, const CBlockIndex &blockindex, TxVerbosity verbosity, const uint256 pow_limit)
Block description to JSON.
static RPCMethod getchainstates()
static RPCMethod verifychain()
static RPCMethod gettxout()
static RPCMethod syncwithvalidationinterfacequeue()
static RPCMethod getblockhash()
void ReconsiderBlock(ChainstateManager &chainman, uint256 block_hash)
static RPCMethod getchaintips()
UniValue blockheaderToJSON(const CBlockIndex &tip, const CBlockIndex &blockindex, const uint256 pow_limit)
Block header to JSON.
static RPCMethod getchaintxstats()
static const auto scan_result_status_some
static RPCMethod getblockstats()
void RegisterBlockchainRPCCommands(CRPCTable &t)
void CalculatePercentilesByWeight(CAmount result[NUM_GETBLOCKSTATS_PERCENTILES], std::vector< std::pair< CAmount, int64_t > > &scores, int64_t total_weight)
Used by getblockstats to get feerates at different percentiles by weight
static std::optional< kernel::CCoinsStats > GetUTXOStats(CCoinsView *view, node::BlockManager &blockman, kernel::CoinStatsHashType hash_type, const std::function< void()> &interruption_point={}, const CBlockIndex *pindex=nullptr, bool index_requested=true)
Calculate statistics about the unspent transaction output set.
static std::atomic< bool > g_should_abort_scan
const std::vector< RPCResult > RPCHelpForChainstate
static const auto scan_action_arg_desc
static const CBlockIndex * ParseHashOrHeight(const UniValue ¶m, ChainstateManager &chainman)
RPCMethod getblockchaininfo()
static RPCMethod waitforblockheight()
static RPCMethod dumptxoutset()
Serialize the UTXO set to a file for loading elsewhere.
RPCMethod getdeploymentinfo()
static RPCMethod loadtxoutset()
static std::atomic< bool > g_scanfilter_in_progress
UniValue CreateRolledBackUTXOSnapshot(NodeContext &node, Chainstate &chainstate, const CBlockIndex *target, AutoFile &&afile, const fs::path &path, const fs::path &tmppath, bool in_memory)
static RPCMethod getblock()
static const auto scan_result_status_none
static RPCMethod gettxoutsetinfo()
static constexpr int NUM_GETBLOCKSTATS_PERCENTILES
const std::string & BlockFilterTypeName(BlockFilterType filter_type)
Get the human-readable name for a filter type.
bool BlockFilterTypeByName(std::string_view name, BlockFilterType &filter_type)
Find a filter type by its human-readable name.
BlockFilterIndex * GetBlockFilterIndex(BlockFilterType filter_type)
Get a block filter index by type.
@ BLOCK_VALID_SCRIPTS
Scripts & signatures ok.
@ BLOCK_VALID_TREE
All parent headers found, difficulty matches, timestamp >= median previous.
@ BLOCK_HAVE_UNDO
undo data available in rev*.dat
@ BLOCK_HAVE_DATA
full block available in blk*.dat
@ BLOCK_FAILED_VALID
stage after last reached validness failed
static constexpr int64_t TIMESTAMP_WINDOW
Timestamp window used as a grace period by code that compares external timestamps (such as timestamps...
#define LIST_CHAIN_NAMES
List of possible chain / network names
#define CHECK_NONFATAL(condition)
Identity function.
fs::path GetDataDirNet() const EXCLUSIVE_LOCKS_REQUIRED(!cs_args)
Get data directory path with appended network identifier.
Non-refcounted RAII wrapper for FILE*.
int64_t size()
Return the size of the file.
Complete block filter struct as defined in BIP 157.
const std::vector< unsigned char > & GetEncodedFilter() const LIFETIMEBOUND
BlockFilterIndex is used to store and retrieve block filters, hashes, and headers for a range of bloc...
bool LookupFilterRange(int start_height, const CBlockIndex *stop_index, std::vector< BlockFilter > &filters_out) const
Get a range of filters between two heights on a chain.
bool LookupFilter(const CBlockIndex *block_index, BlockFilter &filter_out) const
Get a single filter by block.
bool LookupFilterHeader(const CBlockIndex *block_index, uint256 &header_out) EXCLUSIVE_LOCKS_REQUIRED(!m_cs_headers_cache)
Get a single filter header by block.
BlockFiltersScanReserver()=default
~BlockFiltersScanReserver()
std::vector< CTransactionRef > vtx
The block chain is a tree shaped structure starting with the genesis block at the root,...
bool IsValid(enum BlockStatus nUpTo) const EXCLUSIVE_LOCKS_REQUIRED(
Check whether this block index entry is valid up to the passed validity level.
CBlockIndex * pprev
pointer to the index of the predecessor of this block
uint64_t m_chain_tx_count
(memory only) Number of transactions in the chain up to and including this block.
CBlockHeader GetBlockHeader() const
arith_uint256 nChainWork
(memory only) Total amount of work (expected number of hashes) in the chain up to and including this ...
uint256 GetBlockHash() const
int64_t GetBlockTime() const
int64_t GetMedianTimePast() const
unsigned int nTx
Number of transactions in this block.
int32_t nVersion
block header
CBlockIndex * GetAncestor(int height)
Efficiently find an ancestor of this block.
int nHeight
height of the entry in the chain. The genesis block has height 0
FlatFilePos GetBlockPos() const EXCLUSIVE_LOCKS_REQUIRED(
Undo information for a CBlock.
std::vector< CTxUndo > vtxundo
An in-memory indexed chain of blocks.
bool Contains(const CBlockIndex &index) const
Efficiently check whether a block is present in this chain.
CBlockIndex * Tip() const
Returns the index entry for the tip of this chain, or nullptr if none.
const CBlockIndex * FindFork(const CBlockIndex &index) const
Find the last common block between this chain and a block index entry.
CBlockIndex * Genesis() const
Returns the index entry for the genesis block of this chain, or nullptr if none.
CBlockIndex * FindEarliestAtLeast(int64_t nTime, int height) const
Find the earliest block with timestamp equal or greater than the given time and height equal or great...
int Height() const
Return the maximal height in the chain.
std::string GetChainTypeString() const
Return the chain type string.
const MessageStartChars & MessageStart() const
const Consensus::Params & GetConsensus() const
uint64_t PruneAfterHeight() const
ChainType GetChainType() const
Return the chain type.
CCoinsView that adds a memory cache for transactions to another CCoinsView.
void AddCoin(const COutPoint &outpoint, Coin &&coin, bool possible_overwrite)
Add a coin.
void Flush(bool reallocate_cache=true)
Push the modifications applied to this cache to its base and wipe local state.
void SetBestBlock(const uint256 &block_hash)
uint256 GetBestBlock() const override
Retrieve the block hash whose state this CCoinsView currently represents.
std::optional< Coin > GetCoin(const COutPoint &outpoint) const override
Retrieve the Coin (unspent transaction output) for a given outpoint.
Cursor for iterating over CoinsView state.
virtual bool Valid() const =0
virtual bool GetKey(COutPoint &key) const =0
virtual bool GetValue(Coin &coin) const =0
std::unique_ptr< CCoinsViewCursor > Cursor() const override
Get a cursor to iterate over the whole state. Implementations may return nullptr.
Pure abstract view on the open txout dataset.
virtual uint256 GetBestBlock() const =0
Retrieve the block hash whose state this CCoinsView currently represents.
CCoinsView that brings transactions from a mempool into view.
std::optional< Coin > GetCoin(const COutPoint &outpoint) const override
GetCoin, returning whether it exists and is not spent.
An outpoint - a combination of a transaction hash and an index n into its vout.
Serialized script, used inside transaction inputs and outputs.
The basic transaction that is broadcasted on the network and contained in blocks.
const std::vector< CTxIn > vin
An input of a transaction.
CTxMemPoolEntry stores data about the corresponding transaction, as well as data about all in-mempool...
CTxMemPool stores valid-according-to-the-current-best-chain transactions that may be included in the ...
CTransactionRef get(const Txid &hash) const
std::vector< CTxMemPoolEntryRef > entryAll() const EXCLUSIVE_LOCKS_REQUIRED(cs)
bool isSpent(const COutPoint &outpoint) const
An output of a transaction.
Undo information for a CTransaction.
RAII wrapper for VerifyDB: Verify consistency of the block and coin databases.
VerifyDBResult VerifyDB(Chainstate &chainstate, const Consensus::Params &consensus_params, CCoinsView &coinsview, int nCheckLevel, int nCheckDepth) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Chainstate stores and provides an API to update our local knowledge of the current best chain.
CChain m_chain
The current chain of blockheaders we consult and build on.
CCoinsViewCache & CoinsTip() EXCLUSIVE_LOCKS_REQUIRED(
bool PreciousBlock(BlockValidationState &state, CBlockIndex *pindex) LOCKS_EXCLUDED(bool InvalidateBlock(BlockValidationState &state, CBlockIndex *pindex) LOCKS_EXCLUDED(void SetBlockFailureFlags(CBlockIndex *pindex) EXCLUSIVE_LOCKS_REQUIRED(voi ResetBlockFailureFlags)(CBlockIndex *pindex) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Mark a block as precious and reorganize.
CCoinsViewDB & CoinsDB() EXCLUSIVE_LOCKS_REQUIRED(
ChainstateManager & m_chainman
The chainstate manager that owns this chainstate.
node::BlockManager & m_blockman
Reference to a BlockManager instance which itself is shared across all Chainstate instances.
void ForceFlushStateToDisk(bool wipe_cache=true)
Flush all changes to disk.
Interface for managing multiple Chainstate objects, where each chainstate is associated with chainsta...
Chainstate * HistoricalChainstate() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
Return historical chainstate targeting a specific block, if any.
node::BlockMap & BlockIndex() EXCLUSIVE_LOCKS_REQUIRED(
double GetBackgroundVerificationProgress(const CBlockIndex &pindex) const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
Guess background verification progress in case assume-utxo was used (as a fraction between 0....
double GuessVerificationProgress(const CBlockIndex *pindex) const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
Guess verification progress (as a fraction between 0.0=genesis and 1.0=current tip).
bool IsInitialBlockDownload() const noexcept
Check whether we are doing an initial block download (synchronizing from disk or network)
kernel::Notifications & GetNotifications() const
RecursiveMutex & GetMutex() const LOCK_RETURNED(
Alias for cs_main.
CBlockIndex * ActiveTip() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
Chainstate & ActiveChainstate() const
Alternatives to CurrentChainstate() used by older code to query latest chainstate information without...
SnapshotCompletionResult MaybeValidateSnapshot(Chainstate &validated_cs, Chainstate &unvalidated_cs) EXCLUSIVE_LOCKS_REQUIRED(Chainstate & CurrentChainstate() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
Try to validate an assumeutxo snapshot by using a validated historical chainstate targeted at the sna...
VersionBitsCache m_versionbitscache
Track versionbit status.
const CChainParams & GetParams() const
const Consensus::Params & GetConsensus() const
Chainstate &InitializeChainstate(CTxMemPool *mempool) EXCLUSIVE_LOCKS_REQUIRED(util::Result< CBlockIndex * ActivateSnapshot)(AutoFile &coins_file, const node::SnapshotMetadata &metadata, bool in_memory)
Instantiate a new chainstate.
CChain & ActiveChain() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
node::BlockManager m_blockman
A single BlockManager instance is shared across each constructed chainstate to avoid duplicating bloc...
CTxOut out
unspent transaction output
uint32_t nHeight
at which height this containing transaction was included in the active block chain
CoinsViewScanReserver()=default
Double ended buffer combining vector and stream-like interfaces.
std::unordered_set< Element, ByteVectorHash > ElementSet
virtual util::Expected< void, std::string > FetchBlock(NodeId peer_id, const CBlockIndex &block_index)=0
Attempt to manually fetch block from a given peer.
auto Arg(std::string_view key) const
Helper to get a required or default-valued request argument.
Minimal stream for reading from an existing byte array by std::span.
RAII class that registers a prune lock in its constructor to prevent block data from being pruned,...
BlockManager & m_blockman
static constexpr const char * LOCK_NAME
TemporaryPruneLock(BlockManager &blockman, int height)
RAII class that creates a temporary database directory in its constructor and removes it in its destr...
TemporaryUTXODatabase(const fs::path &path)
void push_back(UniValue val)
const std::string & get_str() const
const std::vector< UniValue > & getValues() const
const UniValue & get_array() const
void reserve(size_t new_cap)
void pushKV(std::string key, UniValue val)
void push_backV(const std::vector< UniValue > &vec)
std::string ToString() const
BIP9Info Info(const CBlockIndex &block_index, const Consensus::Params ¶ms, Consensus::DeploymentPos id) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
256-bit unsigned big integer.
std::string ToString() const
std::string GetHex() const
uint64_t GetLow64() const
std::string GetHex() const
Hex encoding of the number (with the most significant digits first).
Interface giving clients (RPC, Stratum v2 Template Provider in the future) ability to create block te...
virtual std::optional< BlockRef > waitTipChanged(uint256 current_tip, MillisecondsDouble timeout=MillisecondsDouble::max())=0
Waits for the connected tip to change.
virtual std::optional< BlockRef > getTip()=0
Returns the hash and height for the tip of this chain.
Maintains a tree of blocks (stored in m_block_index) which is consulted to determine where the most-w...
CBlockIndex * LookupBlockIndex(const uint256 &hash) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
bool ReadBlockUndo(CBlockUndo &blockundo, const CBlockIndex &index) const
CBlockFileInfo *GetBlockFileInfo(size_t n) EXCLUSIVE_LOCKS_REQUIRED(bool WriteBlockUndo(const CBlockUndo &blockundo, BlockValidationState &state, CBlockIndex &block) EXCLUSIVE_LOCKS_REQUIRED(FlatFilePos WriteBlock(const CBlock &block, int nHeight) EXCLUSIVE_LOCKS_REQUIRED(void UpdateBlockInfo(const CBlock &block, unsigned int nHeight, const FlatFilePos &pos) EXCLUSIVE_LOCKS_REQUIRED(bool IsPruneMode() const
Get block file info entry for one block file.
uint64_t GetPruneTarget() const
Attempt to stay below this number of bytes of block files.
ReadRawBlockResult ReadRawBlock(const FlatFilePos &pos, std::optional< std::pair< size_t, size_t > > block_part=std::nullopt) const
bool ReadBlock(CBlock &block, const FlatFilePos &pos, const std::optional< uint256 > &expected_hash) const
Functions for disk access for blocks.
constexpr const std::byte * begin() const
std::string GetHex() const
static transaction_identifier FromUint256(const uint256 &id)
std::unique_ptr< CoinStatsIndex > g_coin_stats_index
The global UTXO set hash object.
static int64_t GetBlockWeight(const CBlock &block)
static int32_t GetTransactionWeight(const CTransaction &tx)
static const unsigned int MAX_BLOCK_SERIALIZED_SIZE
The maximum allowed size for a serialized block, in bytes (only for buffer size limits)
static const int WITNESS_SCALE_FACTOR
void ScriptToUniv(const CScript &script, UniValue &out, bool include_hex, bool include_address, const SigningProvider *provider)
void TxToUniv(const CTransaction &tx, const uint256 &block_hash, UniValue &entry, bool include_hex, const CTxUndo *txundo, TxVerbosity verbosity, std::function< bool(const CTxOut &)> is_change_func)
UniValue ValueFromAmount(const CAmount amount)
TxVerbosity
Verbose level for block's transaction.
@ SHOW_DETAILS_AND_PREVOUT
The same as previous option with information about prevouts if available.
@ SHOW_TXID
Only TXID for each block's transaction.
@ SHOW_DETAILS
Include TXID, inputs, outputs, and other common block's transaction information.
RecursiveMutex cs_main
Mutex to guard access to validation specific variables, such as reading or changing the chainstate.
bool DestroyDB(const std::string &path_str)
std::string DeploymentName(Consensus::BuriedDeployment dep)
bool DeploymentActiveAfter(const CBlockIndex *pindexPrev, const Consensus::Params ¶ms, Consensus::BuriedDeployment dep, VersionBitsCache &versionbitscache)
Determine if a deployment is active for the next block.
bool DeploymentEnabled(const Consensus::Params ¶ms, Consensus::BuriedDeployment dep)
Determine if a deployment is enabled (can ever be active)
const std::string CURRENCY_UNIT
static path u8path(std::string_view utf8_str)
static bool exists(const path &p)
static std::string PathToString(const path &path)
Convert path object to a byte string.
#define T(expected, seed, data)
std::string HexStr(const std::span< const uint8_t > s)
Convert a span of bytes to a lower-case hexadecimal string.
std::vector< std::string > GetScriptFlagNames(script_verify_flags flags)
#define LogDebug(category,...)
BuriedDeployment
A buried deployment is one where the height of the activation has been hardcoded into the client impl...
FILE * fopen(const fs::path &p, const char *mode)
fs::path AbsPathJoin(const fs::path &base, const fs::path &path)
Helper function for joining two paths.
static std::optional< CCoinsStats > ComputeUTXOStats(T hash_obj, CCoinsView *view, node::BlockManager &blockman, const std::function< void()> &interruption_point)
Calculate statistics about the unspent transaction output set.
UniValue GetWarningsForRpc(const Warnings &warnings, bool use_deprecated)
RPC helper function that wraps warnings.GetMessages().
bilingual_str ErrorString(const Result< T > &result)
std::string MakeUnorderedList(const std::vector< std::string > &items)
Create an unordered multi-line list of items.
static constexpr TransactionSerParams TX_NO_WITNESS
static constexpr TransactionSerParams TX_WITH_WITNESS
std::shared_ptr< const CTransaction > CTransactionRef
std::vector< RPCResult > TxDoc(const TxDocOptions &opts)
Explain the UniValue "decoded" transaction object, may include extra fields if processed by wallet.
@ WithSummary
first field carries elision_summary as "...", rest skipped
@ Silent
all top-level fields skipped silently (no "..." line)
UniValue JSONRPCError(int code, const std::string &message)
@ RPC_MISC_ERROR
General application defined errors.
@ RPC_INVALID_PARAMETER
Invalid, missing or duplicate parameter.
@ RPC_DATABASE_ERROR
Database error.
@ RPC_DESERIALIZATION_ERROR
Error parsing or validating structure in raw format.
@ RPC_INVALID_ADDRESS_OR_KEY
Invalid address or key.
std::vector< CScript > EvalDescriptorStringOrObject(const UniValue &scanobject, FlatSigningProvider &provider, const bool expand_priv)
Evaluate a descriptor given as a string, or as a {"desc":...,"range":...} object, with default range ...
std::string HelpExampleCli(const std::string &methodname, const std::string &args)
uint256 GetTarget(const CBlockIndex &blockindex, const uint256 pow_limit)
std::string HelpExampleRpc(const std::string &methodname, const std::string &args)
const std::string UNIX_EPOCH_TIME
String used to describe UNIX epoch time in documentation, factored out to a constant for consistency.
int ParseVerbosity(const UniValue &arg, int default_verbosity, bool allow_bool)
Parses verbosity from provided UniValue.
uint256 ParseHashV(const UniValue &v, std::string_view name)
Utilities: convert hex-encoded Values (throws error if not hex).
std::vector< RPCResult > ScriptPubKeyDoc()
std::unique_ptr< Descriptor > InferDescriptor(const CScript &script, const SigningProvider &provider)
Find a descriptor for the specified script, using information from provider where possible.
void WriteCompactSize(SizeComputer &os, uint64_t nSize)
uint64_t GetSerializeSize(const T &t)
bool IsDeprecatedRPCEnabled(const std::string &method)
ChainstateManager & EnsureAnyChainman(const std::any &context)
NodeContext & EnsureAnyNodeContext(const std::any &context)
CTxMemPool & EnsureMemPool(const NodeContext &node)
PeerManager & EnsurePeerman(const NodeContext &node)
ChainstateManager & EnsureChainman(const NodeContext &node)
ArgsManager & EnsureArgsman(const NodeContext &node)
interfaces::Mining & EnsureMining(const NodeContext &node)
ArgsManager & EnsureAnyArgsman(const std::any &context)
unsigned char * UCharCast(char *c)
Detailed status of an enabled BIP9 deployment.
User-controlled performance and debug options.
Comparison function for sorting the getchaintips heads.
bool operator()(const CBlockIndex *a, const CBlockIndex *b) const
std::vector< uint8_t > signet_challenge
uint256 powLimit
Proof of work parameters.
int DeploymentHeight(BuriedDeployment dep) const
std::array< BIP9Deployment, MAX_VERSION_BITS_DEPLOYMENTS > vDeployments
int64_t nPowTargetSpacing
User-controlled performance and debug options.
Application-specific storage settings.
fs::path path
Location in the filesystem where leveldb data will be stored.
@ RANGE
Special type that is a NUM or [NUM,NUM].
@ STR_HEX
Special type that is a STR with only hex chars.
@ OBJ_NAMED_PARAMS
Special type that behaves almost exactly like OBJ, defining an options object with a list of pre-defi...
std::string DefaultHint
Hint for default value.
@ OMITTED
Optional argument for which the default value is omitted from help text for one of two reasons:
UniValue Default
Default constant value.
std::string oneline_description
Should be empty unless it is supposed to override the auto-generated summary line.
@ NUM_TIME
Special numeric to denote unix epoch time.
@ ARR_FIXED
Special array that has a fixed number of entries.
@ 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.
HelpElision print_elision
Hash/height pair to help track and identify blocks.
arith_uint256 total_prevout_spent_amount
Total cumulative amount of prevouts spent up to and including this block.
std::optional< CAmount > total_amount
The total amount, or nullopt if an overflow occurred calculating it.
CAmount total_unspendables_scripts
Total cumulative amount of outputs sent to unspendable scripts (OP_RETURN for example) up to and incl...
uint64_t coins_count
The number of coins contained.
arith_uint256 total_coinbase_amount
Total cumulative amount of coinbase outputs up to and including this block.
uint64_t nTransactionOutputs
bool index_used
Signals if the coinstatsindex was used to retrieve the statistics.
CAmount total_unspendables_bip30
The two unspendable coinbase outputs total amount caused by BIP30.
CAmount total_unspendables_genesis_block
The unspendable coinbase amount from the genesis block.
arith_uint256 total_new_outputs_ex_coinbase_amount
Total cumulative amount of outputs created up to and including this block.
CAmount total_unspendables_unclaimed_rewards
Total cumulative amount of coins lost due to unclaimed miner rewards up to and including this block.
NodeContext struct containing references to chain state and connection state.
#define AssertLockNotHeld(cs)
#define WITH_LOCK(cs, code)
Run code while locking a mutex.
std::string SysErrorString(int err)
Return system error string from errno value.
FuzzedDataProvider provider
#define EXCLUSIVE_LOCKS_REQUIRED(...)
#define LOG_TIME_SECONDS(end_msg)
consteval auto _(util::TranslatedLiteral str)
static const uint32_t MEMPOOL_HEIGHT
Fake height value used in Coin to signify they are only in the memory pool (since 0....
const UniValue NullUniValue
std::chrono::duration< double, std::chrono::milliseconds::period > MillisecondsDouble
script_verify_flags GetBlockScriptFlags(const CBlockIndex &block_index, const ChainstateManager &chainman)
CAmount GetBlockSubsidy(int nHeight, const Consensus::Params &consensusParams)
const std::vector< std::string > CHECKLEVEL_DOC
Documentation for argument 'checklevel'.
void PruneBlockFilesManual(Chainstate &active_chainstate, int nManualPruneHeight)
Prune block files up to a given height.
bool IsBIP30Repeat(const CBlockIndex &block_index)
Identifies blocks that overwrote an existing coinbase output in the UTXO set (see BIP30)
static constexpr int DEFAULT_CHECKLEVEL
static const unsigned int MIN_BLOCKS_TO_KEEP
Block files containing a block-height within MIN_BLOCKS_TO_KEEP of ActiveChain().Tip() will not be pr...
@ VALIDATED
Every block in the chain has been validated.
static const signed int DEFAULT_CHECKBLOCKS