Bitcoin Core Fuzz Coverage Report

Coverage Report

Created: 2026-06-01 16:00

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/home/zip/work/bitcoin/src/rpc/blockchain.cpp
Line
Count
Source
1
// Copyright (c) 2010 Satoshi Nakamoto
2
// Copyright (c) 2009-present The Bitcoin Core developers
3
// Distributed under the MIT software license, see the accompanying
4
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
5
6
#include <rpc/blockchain.h>
7
8
#include <blockfilter.h>
9
#include <chain.h>
10
#include <chainparams.h>
11
#include <chainparamsbase.h>
12
#include <clientversion.h>
13
#include <coins.h>
14
#include <common/args.h>
15
#include <consensus/amount.h>
16
#include <consensus/params.h>
17
#include <consensus/validation.h>
18
#include <core_io.h>
19
#include <deploymentinfo.h>
20
#include <deploymentstatus.h>
21
#include <flatfile.h>
22
#include <hash.h>
23
#include <index/blockfilterindex.h>
24
#include <index/coinstatsindex.h>
25
#include <interfaces/mining.h>
26
#include <kernel/coinstats.h>
27
#include <logging/timer.h>
28
#include <net.h>
29
#include <net_processing.h>
30
#include <node/blockstorage.h>
31
#include <node/context.h>
32
#include <node/transaction.h>
33
#include <node/utxo_snapshot.h>
34
#include <node/warnings.h>
35
#include <primitives/transaction.h>
36
#include <rpc/server.h>
37
#include <rpc/server_util.h>
38
#include <rpc/util.h>
39
#include <script/descriptor.h>
40
#include <serialize.h>
41
#include <streams.h>
42
#include <sync.h>
43
#include <tinyformat.h>
44
#include <txdb.h>
45
#include <txmempool.h>
46
#include <undo.h>
47
#include <univalue.h>
48
#include <util/check.h>
49
#include <util/fs.h>
50
#include <util/strencodings.h>
51
#include <util/syserror.h>
52
#include <util/translation.h>
53
#include <validation.h>
54
#include <validationinterface.h>
55
#include <versionbits.h>
56
57
#include <cstdint>
58
59
#include <condition_variable>
60
#include <iterator>
61
#include <memory>
62
#include <mutex>
63
#include <optional>
64
#include <string>
65
#include <string_view>
66
#include <vector>
67
68
using kernel::CCoinsStats;
69
using kernel::CoinStatsHashType;
70
71
using interfaces::BlockRef;
72
using interfaces::Mining;
73
using node::BlockManager;
74
using node::NodeContext;
75
using node::SnapshotMetadata;
76
using util::MakeUnorderedList;
77
78
std::tuple<std::unique_ptr<CCoinsViewCursor>, CCoinsStats, const CBlockIndex*>
79
PrepareUTXOSnapshot(
80
    Chainstate& chainstate,
81
    const std::function<void()>& interruption_point = {})
82
    EXCLUSIVE_LOCKS_REQUIRED(::cs_main);
83
84
UniValue WriteUTXOSnapshot(
85
    Chainstate& chainstate,
86
    CCoinsViewCursor* pcursor,
87
    CCoinsStats* maybe_stats,
88
    const CBlockIndex* tip,
89
    AutoFile&& afile,
90
    const fs::path& path,
91
    const fs::path& temppath,
92
    const std::function<void()>& interruption_point = {});
93
94
UniValue CreateRolledBackUTXOSnapshot(
95
    NodeContext& node,
96
    Chainstate& chainstate,
97
    const CBlockIndex* target,
98
    AutoFile&& afile,
99
    const fs::path& path,
100
    const fs::path& tmppath,
101
    bool in_memory);
102
103
/* Calculate the difficulty for a given block index.
104
 */
105
double GetDifficulty(const CBlockIndex& blockindex)
106
0
{
107
0
    int nShift = (blockindex.nBits >> 24) & 0xff;
108
0
    double dDiff =
109
0
        (double)0x0000ffff / (double)(blockindex.nBits & 0x00ffffff);
110
111
0
    while (nShift < 29)
112
0
    {
113
0
        dDiff *= 256.0;
114
0
        nShift++;
115
0
    }
116
0
    while (nShift > 29)
117
0
    {
118
0
        dDiff /= 256.0;
119
0
        nShift--;
120
0
    }
121
122
0
    return dDiff;
123
0
}
124
125
static int ComputeNextBlockAndDepth(const CBlockIndex& tip, const CBlockIndex& blockindex, const CBlockIndex*& next)
126
0
{
127
0
    next = tip.GetAncestor(blockindex.nHeight + 1);
128
0
    if (next && next->pprev == &blockindex) {
129
0
        return tip.nHeight - blockindex.nHeight + 1;
130
0
    }
131
0
    next = nullptr;
132
0
    return &blockindex == &tip ? 1 : -1;
133
0
}
134
135
static const CBlockIndex* ParseHashOrHeight(const UniValue& param, ChainstateManager& chainman)
136
0
{
137
0
    LOCK(::cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
138
0
    CChain& active_chain = chainman.ActiveChain();
139
140
0
    if (param.isNum()) {
141
0
        const int height{param.getInt<int>()};
142
0
        if (height < 0) {
143
0
            throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Target block height %d is negative", height));
Line
Count
Source
1172
0
#define strprintf tfm::format
144
0
        }
145
0
        const int current_tip{active_chain.Height()};
146
0
        if (height > current_tip) {
147
0
            throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Target block height %d after current tip %d", height, current_tip));
Line
Count
Source
1172
0
#define strprintf tfm::format
148
0
        }
149
150
0
        return active_chain[height];
151
0
    } else {
152
0
        const uint256 hash{ParseHashV(param, "hash_or_height")};
153
0
        const CBlockIndex* pindex = chainman.m_blockman.LookupBlockIndex(hash);
154
155
0
        if (!pindex) {
156
0
            throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Block not found");
157
0
        }
158
159
0
        return pindex;
160
0
    }
161
0
}
162
163
UniValue blockheaderToJSON(const CBlockIndex& tip, const CBlockIndex& blockindex, const uint256 pow_limit)
164
0
{
165
    // Serialize passed information without accessing chain state of the active chain!
166
0
    AssertLockNotHeld(cs_main); // For performance reasons
Line
Count
Source
149
0
#define AssertLockNotHeld(cs) AssertLockNotHeldInline(#cs, __FILE__, __LINE__, &cs)
167
168
0
    UniValue result(UniValue::VOBJ);
169
0
    result.pushKV("hash", blockindex.GetBlockHash().GetHex());
170
0
    const CBlockIndex* pnext;
171
0
    int confirmations = ComputeNextBlockAndDepth(tip, blockindex, pnext);
172
0
    result.pushKV("confirmations", confirmations);
173
0
    result.pushKV("height", blockindex.nHeight);
174
0
    result.pushKV("version", blockindex.nVersion);
175
0
    result.pushKV("versionHex", strprintf("%08x", blockindex.nVersion));
Line
Count
Source
1172
0
#define strprintf tfm::format
176
0
    result.pushKV("merkleroot", blockindex.hashMerkleRoot.GetHex());
177
0
    result.pushKV("time", blockindex.nTime);
178
0
    result.pushKV("mediantime", blockindex.GetMedianTimePast());
179
0
    result.pushKV("nonce", blockindex.nNonce);
180
0
    result.pushKV("bits", strprintf("%08x", blockindex.nBits));
Line
Count
Source
1172
0
#define strprintf tfm::format
181
0
    result.pushKV("target", GetTarget(blockindex, pow_limit).GetHex());
182
0
    result.pushKV("difficulty", GetDifficulty(blockindex));
183
0
    result.pushKV("chainwork", blockindex.nChainWork.GetHex());
184
0
    result.pushKV("nTx", blockindex.nTx);
185
186
0
    if (blockindex.pprev)
187
0
        result.pushKV("previousblockhash", blockindex.pprev->GetBlockHash().GetHex());
188
0
    if (pnext)
189
0
        result.pushKV("nextblockhash", pnext->GetBlockHash().GetHex());
190
0
    return result;
191
0
}
192
193
/** Serialize coinbase transaction metadata */
194
UniValue coinbaseTxToJSON(const CTransaction& coinbase_tx)
195
0
{
196
0
    CHECK_NONFATAL(!coinbase_tx.vin.empty());
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
197
0
    const CTxIn& vin_0{coinbase_tx.vin[0]};
198
0
    UniValue coinbase_tx_obj(UniValue::VOBJ);
199
0
    coinbase_tx_obj.pushKV("version", coinbase_tx.version);
200
0
    coinbase_tx_obj.pushKV("locktime", coinbase_tx.nLockTime);
201
0
    coinbase_tx_obj.pushKV("sequence", vin_0.nSequence);
202
0
    coinbase_tx_obj.pushKV("coinbase", HexStr(vin_0.scriptSig));
203
0
    const auto& witness_stack{vin_0.scriptWitness.stack};
204
0
    if (!witness_stack.empty()) {
205
0
        CHECK_NONFATAL(witness_stack.size() == 1);
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
206
0
        coinbase_tx_obj.pushKV("witness", HexStr(witness_stack[0]));
207
0
    }
208
0
    return coinbase_tx_obj;
209
0
}
210
211
UniValue blockToJSON(BlockManager& blockman, const CBlock& block, const CBlockIndex& tip, const CBlockIndex& blockindex, TxVerbosity verbosity, const uint256 pow_limit)
212
0
{
213
0
    UniValue result = blockheaderToJSON(tip, blockindex, pow_limit);
214
215
0
    result.pushKV("strippedsize", ::GetSerializeSize(TX_NO_WITNESS(block)));
216
0
    result.pushKV("size", ::GetSerializeSize(TX_WITH_WITNESS(block)));
217
0
    result.pushKV("weight", ::GetBlockWeight(block));
218
219
0
    CHECK_NONFATAL(!block.vtx.empty());
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
220
0
    result.pushKV("coinbase_tx", coinbaseTxToJSON(*block.vtx[0]));
221
222
0
    UniValue txs(UniValue::VARR);
223
0
    txs.reserve(block.vtx.size());
224
225
0
    switch (verbosity) {
226
0
        case TxVerbosity::SHOW_TXID:
227
0
            for (const CTransactionRef& tx : block.vtx) {
228
0
                txs.push_back(tx->GetHash().GetHex());
229
0
            }
230
0
            break;
231
232
0
        case TxVerbosity::SHOW_DETAILS:
233
0
        case TxVerbosity::SHOW_DETAILS_AND_PREVOUT:
234
0
            CBlockUndo blockUndo;
235
0
            const bool is_not_pruned{WITH_LOCK(::cs_main, return !blockman.IsBlockPruned(blockindex))};
Line
Count
Source
299
0
#define WITH_LOCK(cs, code) (MaybeCheckNotHeld(cs), [&]() -> decltype(auto) { LOCK(cs); code; }())
236
0
            bool have_undo{is_not_pruned && WITH_LOCK(::cs_main, return blockindex.nStatus & BLOCK_HAVE_UNDO)};
Line
Count
Source
299
0
#define WITH_LOCK(cs, code) (MaybeCheckNotHeld(cs), [&]() -> decltype(auto) { LOCK(cs); code; }())
237
0
            if (have_undo && !blockman.ReadBlockUndo(blockUndo, blockindex)) {
238
0
                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.");
239
0
            }
240
0
            for (size_t i = 0; i < block.vtx.size(); ++i) {
241
0
                const CTransactionRef& tx = block.vtx.at(i);
242
                // coinbase transaction (i.e. i == 0) doesn't have undo data
243
0
                const CTxUndo* txundo = (have_undo && i > 0) ? &blockUndo.vtxundo.at(i - 1) : nullptr;
244
0
                UniValue objTx(UniValue::VOBJ);
245
0
                TxToUniv(*tx, /*block_hash=*/uint256(), /*entry=*/objTx, /*include_hex=*/true, txundo, verbosity);
246
0
                txs.push_back(std::move(objTx));
247
0
            }
248
0
            break;
249
0
    }
250
251
0
    result.pushKV("tx", std::move(txs));
252
253
0
    return result;
254
0
}
255
256
static RPCMethod getblockcount()
257
0
{
258
0
    return RPCMethod{
259
0
        "getblockcount",
260
0
        "Returns the height of the most-work fully-validated chain.\n"
261
0
                "The genesis block has height 0.\n",
262
0
                {},
263
0
                RPCResult{
264
0
                    RPCResult::Type::NUM, "", "The current block count"},
265
0
                RPCExamples{
266
0
                    HelpExampleCli("getblockcount", "")
267
0
            + HelpExampleRpc("getblockcount", "")
268
0
                },
269
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
270
0
{
271
0
    ChainstateManager& chainman = EnsureAnyChainman(request.context);
272
0
    LOCK(cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
273
0
    return chainman.ActiveChain().Height();
274
0
},
275
0
    };
276
0
}
277
278
static RPCMethod getbestblockhash()
279
0
{
280
0
    return RPCMethod{
281
0
        "getbestblockhash",
282
0
        "Returns the hash of the best (tip) block in the most-work fully-validated chain.\n",
283
0
                {},
284
0
                RPCResult{
285
0
                    RPCResult::Type::STR_HEX, "", "the block hash, hex-encoded"},
286
0
                RPCExamples{
287
0
                    HelpExampleCli("getbestblockhash", "")
288
0
            + HelpExampleRpc("getbestblockhash", "")
289
0
                },
290
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
291
0
{
292
0
    ChainstateManager& chainman = EnsureAnyChainman(request.context);
293
0
    LOCK(cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
294
0
    return chainman.ActiveChain().Tip()->GetBlockHash().GetHex();
295
0
},
296
0
    };
297
0
}
298
299
static RPCMethod waitfornewblock()
300
0
{
301
0
    return RPCMethod{
302
0
        "waitfornewblock",
303
0
        "Waits for any new block and returns useful info about it.\n"
304
0
                "\nReturns the current block on timeout or exit.\n"
305
0
                "\nMake sure to use no RPC timeout (bitcoin-cli -rpcclienttimeout=0)",
306
0
                {
307
0
                    {"timeout", RPCArg::Type::NUM, RPCArg::Default{0}, "Time in milliseconds to wait for a response. 0 indicates no timeout."},
308
0
                    {"current_tip", RPCArg::Type::STR_HEX, RPCArg::Optional::OMITTED, "Method waits for the chain tip to differ from this."},
309
0
                },
310
0
                RPCResult{
311
0
                    RPCResult::Type::OBJ, "", "",
312
0
                    {
313
0
                        {RPCResult::Type::STR_HEX, "hash", "The blockhash"},
314
0
                        {RPCResult::Type::NUM, "height", "Block height"},
315
0
                    }},
316
0
                RPCExamples{
317
0
                    HelpExampleCli("waitfornewblock", "1000")
318
0
            + HelpExampleRpc("waitfornewblock", "1000")
319
0
                },
320
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
321
0
{
322
0
    int timeout = 0;
323
0
    if (!request.params[0].isNull())
324
0
        timeout = request.params[0].getInt<int>();
325
0
    if (timeout < 0) throw JSONRPCError(RPC_MISC_ERROR, "Negative timeout");
326
327
0
    NodeContext& node = EnsureAnyNodeContext(request.context);
328
0
    Mining& miner = EnsureMining(node);
329
330
    // If the caller provided a current_tip value, pass it to waitTipChanged().
331
    //
332
    // If the caller did not provide a current tip hash, call getTip() to get
333
    // one and wait for the tip to be different from this value. This mode is
334
    // less reliable because if the tip changed between waitfornewblock calls,
335
    // it will need to change a second time before this call returns.
336
0
    BlockRef current_block{CHECK_NONFATAL(miner.getTip()).value()};
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
337
338
0
    uint256 tip_hash{request.params[1].isNull()
339
0
        ? current_block.hash
340
0
        : ParseHashV(request.params[1], "current_tip")};
341
342
    // If the user provided an invalid current_tip then this call immediately
343
    // returns the current tip.
344
0
    std::optional<BlockRef> block = timeout ? miner.waitTipChanged(tip_hash, std::chrono::milliseconds(timeout)) :
345
0
                                              miner.waitTipChanged(tip_hash);
346
347
    // Return current block upon shutdown
348
0
    if (block) current_block = *block;
349
350
0
    UniValue ret(UniValue::VOBJ);
351
0
    ret.pushKV("hash", current_block.hash.GetHex());
352
0
    ret.pushKV("height", current_block.height);
353
0
    return ret;
354
0
},
355
0
    };
356
0
}
357
358
static RPCMethod waitforblock()
359
0
{
360
0
    return RPCMethod{
361
0
        "waitforblock",
362
0
        "Waits for a specific new block and returns useful info about it.\n"
363
0
                "\nReturns the current block on timeout or exit.\n"
364
0
                "\nMake sure to use no RPC timeout (bitcoin-cli -rpcclienttimeout=0)",
365
0
                {
366
0
                    {"blockhash", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "Block hash to wait for."},
367
0
                    {"timeout", RPCArg::Type::NUM, RPCArg::Default{0}, "Time in milliseconds to wait for a response. 0 indicates no timeout."},
368
0
                },
369
0
                RPCResult{
370
0
                    RPCResult::Type::OBJ, "", "",
371
0
                    {
372
0
                        {RPCResult::Type::STR_HEX, "hash", "The blockhash"},
373
0
                        {RPCResult::Type::NUM, "height", "Block height"},
374
0
                    }},
375
0
                RPCExamples{
376
0
                    HelpExampleCli("waitforblock", "\"0000000000079f8ef3d2c688c244eb7a4570b24c9ed7b4a8c619eb02596f8862\" 1000")
377
0
            + HelpExampleRpc("waitforblock", "\"0000000000079f8ef3d2c688c244eb7a4570b24c9ed7b4a8c619eb02596f8862\", 1000")
378
0
                },
379
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
380
0
{
381
0
    int timeout = 0;
382
383
0
    uint256 hash(ParseHashV(request.params[0], "blockhash"));
384
385
0
    if (!request.params[1].isNull())
386
0
        timeout = request.params[1].getInt<int>();
387
0
    if (timeout < 0) throw JSONRPCError(RPC_MISC_ERROR, "Negative timeout");
388
389
0
    NodeContext& node = EnsureAnyNodeContext(request.context);
390
0
    Mining& miner = EnsureMining(node);
391
392
    // Abort if RPC came out of warmup too early
393
0
    BlockRef current_block{CHECK_NONFATAL(miner.getTip()).value()};
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
394
395
0
    const auto deadline{std::chrono::steady_clock::now() + 1ms * timeout};
396
0
    while (current_block.hash != hash) {
397
0
        std::optional<BlockRef> block;
398
0
        if (timeout) {
399
0
            auto now{std::chrono::steady_clock::now()};
400
0
            if (now >= deadline) break;
401
0
            const MillisecondsDouble remaining{deadline - now};
402
0
            block = miner.waitTipChanged(current_block.hash, remaining);
403
0
        } else {
404
0
            block = miner.waitTipChanged(current_block.hash);
405
0
        }
406
        // Return current block upon shutdown
407
0
        if (!block) break;
408
0
        current_block = *block;
409
0
    }
410
411
0
    UniValue ret(UniValue::VOBJ);
412
0
    ret.pushKV("hash", current_block.hash.GetHex());
413
0
    ret.pushKV("height", current_block.height);
414
0
    return ret;
415
0
},
416
0
    };
417
0
}
418
419
static RPCMethod waitforblockheight()
420
0
{
421
0
    return RPCMethod{
422
0
        "waitforblockheight",
423
0
        "Waits for (at least) block height and returns the height and hash\n"
424
0
                "of the current tip.\n"
425
0
                "\nReturns the current block on timeout or exit.\n"
426
0
                "\nMake sure to use no RPC timeout (bitcoin-cli -rpcclienttimeout=0)",
427
0
                {
428
0
                    {"height", RPCArg::Type::NUM, RPCArg::Optional::NO, "Block height to wait for."},
429
0
                    {"timeout", RPCArg::Type::NUM, RPCArg::Default{0}, "Time in milliseconds to wait for a response. 0 indicates no timeout."},
430
0
                },
431
0
                RPCResult{
432
0
                    RPCResult::Type::OBJ, "", "",
433
0
                    {
434
0
                        {RPCResult::Type::STR_HEX, "hash", "The blockhash"},
435
0
                        {RPCResult::Type::NUM, "height", "Block height"},
436
0
                    }},
437
0
                RPCExamples{
438
0
                    HelpExampleCli("waitforblockheight", "100 1000")
439
0
            + HelpExampleRpc("waitforblockheight", "100, 1000")
440
0
                },
441
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
442
0
{
443
0
    int timeout = 0;
444
445
0
    int height = request.params[0].getInt<int>();
446
447
0
    if (!request.params[1].isNull())
448
0
        timeout = request.params[1].getInt<int>();
449
0
    if (timeout < 0) throw JSONRPCError(RPC_MISC_ERROR, "Negative timeout");
450
451
0
    NodeContext& node = EnsureAnyNodeContext(request.context);
452
0
    Mining& miner = EnsureMining(node);
453
454
    // Abort if RPC came out of warmup too early
455
0
    BlockRef current_block{CHECK_NONFATAL(miner.getTip()).value()};
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
456
457
0
    const auto deadline{std::chrono::steady_clock::now() + 1ms * timeout};
458
459
0
    while (current_block.height < height) {
460
0
        std::optional<BlockRef> block;
461
0
        if (timeout) {
462
0
            auto now{std::chrono::steady_clock::now()};
463
0
            if (now >= deadline) break;
464
0
            const MillisecondsDouble remaining{deadline - now};
465
0
            block = miner.waitTipChanged(current_block.hash, remaining);
466
0
        } else {
467
0
            block = miner.waitTipChanged(current_block.hash);
468
0
        }
469
        // Return current block on shutdown
470
0
        if (!block) break;
471
0
        current_block = *block;
472
0
    }
473
474
0
    UniValue ret(UniValue::VOBJ);
475
0
    ret.pushKV("hash", current_block.hash.GetHex());
476
0
    ret.pushKV("height", current_block.height);
477
0
    return ret;
478
0
},
479
0
    };
480
0
}
481
482
static RPCMethod syncwithvalidationinterfacequeue()
483
0
{
484
0
    return RPCMethod{
485
0
        "syncwithvalidationinterfacequeue",
486
0
        "Waits for the validation interface queue to catch up on everything that was there when we entered this function.\n",
487
0
                {},
488
0
                RPCResult{RPCResult::Type::NONE, "", ""},
489
0
                RPCExamples{
490
0
                    HelpExampleCli("syncwithvalidationinterfacequeue","")
491
0
            + HelpExampleRpc("syncwithvalidationinterfacequeue","")
492
0
                },
493
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
494
0
{
495
0
    NodeContext& node = EnsureAnyNodeContext(request.context);
496
0
    CHECK_NONFATAL(node.validation_signals)->SyncWithValidationInterfaceQueue();
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
497
0
    return UniValue::VNULL;
498
0
},
499
0
    };
500
0
}
501
502
static RPCMethod getdifficulty()
503
0
{
504
0
    return RPCMethod{
505
0
        "getdifficulty",
506
0
        "Returns the proof-of-work difficulty as a multiple of the minimum difficulty.\n",
507
0
                {},
508
0
                RPCResult{
509
0
                    RPCResult::Type::NUM, "", "the proof-of-work difficulty as a multiple of the minimum difficulty."},
510
0
                RPCExamples{
511
0
                    HelpExampleCli("getdifficulty", "")
512
0
            + HelpExampleRpc("getdifficulty", "")
513
0
                },
514
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
515
0
{
516
0
    ChainstateManager& chainman = EnsureAnyChainman(request.context);
517
0
    LOCK(cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
518
0
    return GetDifficulty(*CHECK_NONFATAL(chainman.ActiveChain().Tip()));
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
519
0
},
520
0
    };
521
0
}
522
523
static RPCMethod getblockfrompeer()
524
0
{
525
0
    return RPCMethod{
526
0
        "getblockfrompeer",
527
0
        "Attempt to fetch block from a given peer.\n\n"
528
0
        "We must have the header for this block, e.g. using submitheader.\n"
529
0
        "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"
530
0
        "Subsequent calls for the same block may cause the response from the previous peer to be ignored.\n"
531
0
        "Peers generally ignore requests for a stale block that they never fully verified, or one that is more than a month old.\n"
532
0
        "When a peer does not respond with a block, we will disconnect.\n"
533
0
        "Note: The block could be re-pruned as soon as it is received.\n\n"
534
0
        "Returns an empty JSON object if the request was successfully scheduled.",
535
0
        {
536
0
            {"blockhash", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "The block hash to try to fetch"},
537
0
            {"peer_id", RPCArg::Type::NUM, RPCArg::Optional::NO, "The peer to fetch it from (see getpeerinfo for peer IDs)"},
538
0
        },
539
0
        RPCResult{RPCResult::Type::OBJ, "", /*optional=*/false, "", {}},
540
0
        RPCExamples{
541
0
            HelpExampleCli("getblockfrompeer", "\"00000000c937983704a73af28acdec37b049d214adbda81d7e2a3dd146f6ed09\" 0")
542
0
            + HelpExampleRpc("getblockfrompeer", "\"00000000c937983704a73af28acdec37b049d214adbda81d7e2a3dd146f6ed09\" 0")
543
0
        },
544
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
545
0
{
546
0
    const NodeContext& node = EnsureAnyNodeContext(request.context);
547
0
    ChainstateManager& chainman = EnsureChainman(node);
548
0
    PeerManager& peerman = EnsurePeerman(node);
549
550
0
    const uint256& block_hash{ParseHashV(request.params[0], "blockhash")};
551
0
    const NodeId peer_id{request.params[1].getInt<int64_t>()};
552
553
0
    const CBlockIndex* const index = WITH_LOCK(cs_main, return chainman.m_blockman.LookupBlockIndex(block_hash););
Line
Count
Source
299
0
#define WITH_LOCK(cs, code) (MaybeCheckNotHeld(cs), [&]() -> decltype(auto) { LOCK(cs); code; }())
554
555
0
    if (!index) {
556
0
        throw JSONRPCError(RPC_MISC_ERROR, "Block header missing");
557
0
    }
558
559
    // Fetching blocks before the node has syncing past their height can prevent block files from
560
    // being pruned, so we avoid it if the node is in prune mode.
561
0
    if (chainman.m_blockman.IsPruneMode() && index->nHeight > WITH_LOCK(chainman.GetMutex(), return chainman.ActiveTip()->nHeight)) {
Line
Count
Source
299
0
#define WITH_LOCK(cs, code) (MaybeCheckNotHeld(cs), [&]() -> decltype(auto) { LOCK(cs); code; }())
562
0
        throw JSONRPCError(RPC_MISC_ERROR, "In prune mode, only blocks that the node has already synced previously can be fetched from a peer");
563
0
    }
564
565
0
    const bool block_has_data = WITH_LOCK(::cs_main, return index->nStatus & BLOCK_HAVE_DATA);
Line
Count
Source
299
0
#define WITH_LOCK(cs, code) (MaybeCheckNotHeld(cs), [&]() -> decltype(auto) { LOCK(cs); code; }())
566
0
    if (block_has_data) {
567
0
        throw JSONRPCError(RPC_MISC_ERROR, "Block already downloaded");
568
0
    }
569
570
0
    if (const auto res{peerman.FetchBlock(peer_id, *index)}; !res) {
571
0
        throw JSONRPCError(RPC_MISC_ERROR, res.error());
572
0
    }
573
0
    return UniValue::VOBJ;
574
0
},
575
0
    };
576
0
}
577
578
static RPCMethod getblockhash()
579
0
{
580
0
    return RPCMethod{
581
0
        "getblockhash",
582
0
        "Returns hash of block in best-block-chain at height provided.\n",
583
0
                {
584
0
                    {"height", RPCArg::Type::NUM, RPCArg::Optional::NO, "The height index"},
585
0
                },
586
0
                RPCResult{
587
0
                    RPCResult::Type::STR_HEX, "", "The block hash"},
588
0
                RPCExamples{
589
0
                    HelpExampleCli("getblockhash", "1000")
590
0
            + HelpExampleRpc("getblockhash", "1000")
591
0
                },
592
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
593
0
{
594
0
    ChainstateManager& chainman = EnsureAnyChainman(request.context);
595
0
    LOCK(cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
596
0
    const CChain& active_chain = chainman.ActiveChain();
597
598
0
    int nHeight = request.params[0].getInt<int>();
599
0
    if (nHeight < 0 || nHeight > active_chain.Height())
600
0
        throw JSONRPCError(RPC_INVALID_PARAMETER, "Block height out of range");
601
602
0
    const CBlockIndex* pblockindex = active_chain[nHeight];
603
0
    return pblockindex->GetBlockHash().GetHex();
604
0
},
605
0
    };
606
0
}
607
608
static RPCMethod getblockheader()
609
0
{
610
0
    return RPCMethod{
611
0
        "getblockheader",
612
0
        "If verbose is false, returns a string that is serialized, hex-encoded data for blockheader 'hash'.\n"
613
0
                "If verbose is true, returns an Object with information about blockheader <hash>.\n",
614
0
                {
615
0
                    {"blockhash", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "The block hash"},
616
0
                    {"verbose", RPCArg::Type::BOOL, RPCArg::Default{true}, "true for a json object, false for the hex-encoded data"},
617
0
                },
618
0
                {
619
0
                    RPCResult{"for verbose = true",
620
0
                        RPCResult::Type::OBJ, "", "",
621
0
                        {
622
0
                            {RPCResult::Type::STR_HEX, "hash", "the block hash (same as provided)"},
623
0
                            {RPCResult::Type::NUM, "confirmations", "The number of confirmations, or -1 if the block is not on the main chain"},
624
0
                            {RPCResult::Type::NUM, "height", "The block height or index"},
625
0
                            {RPCResult::Type::NUM, "version", "The block version"},
626
0
                            {RPCResult::Type::STR_HEX, "versionHex", "The block version formatted in hexadecimal"},
627
0
                            {RPCResult::Type::STR_HEX, "merkleroot", "The merkle root"},
628
0
                            {RPCResult::Type::NUM_TIME, "time", "The block time expressed in " + UNIX_EPOCH_TIME},
629
0
                            {RPCResult::Type::NUM_TIME, "mediantime", "The median block time expressed in " + UNIX_EPOCH_TIME},
630
0
                            {RPCResult::Type::NUM, "nonce", "The nonce"},
631
0
                            {RPCResult::Type::STR_HEX, "bits", "nBits: compact representation of the block difficulty target"},
632
0
                            {RPCResult::Type::STR_HEX, "target", "The difficulty target"},
633
0
                            {RPCResult::Type::NUM, "difficulty", "The difficulty"},
634
0
                            {RPCResult::Type::STR_HEX, "chainwork", "Expected number of hashes required to produce the current chain"},
635
0
                            {RPCResult::Type::NUM, "nTx", "The number of transactions in the block"},
636
0
                            {RPCResult::Type::STR_HEX, "previousblockhash", /*optional=*/true, "The hash of the previous block (if available)"},
637
0
                            {RPCResult::Type::STR_HEX, "nextblockhash", /*optional=*/true, "The hash of the next block (if available)"},
638
0
                        }},
639
0
                    RPCResult{"for verbose=false",
640
0
                        RPCResult::Type::STR_HEX, "", "A string that is serialized, hex-encoded data for block 'hash'"},
641
0
                },
642
0
                RPCExamples{
643
0
                    HelpExampleCli("getblockheader", "\"00000000c937983704a73af28acdec37b049d214adbda81d7e2a3dd146f6ed09\"")
644
0
            + HelpExampleRpc("getblockheader", "\"00000000c937983704a73af28acdec37b049d214adbda81d7e2a3dd146f6ed09\"")
645
0
                },
646
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
647
0
{
648
0
    uint256 hash(ParseHashV(request.params[0], "hash"));
649
650
0
    bool fVerbose = true;
651
0
    if (!request.params[1].isNull())
652
0
        fVerbose = request.params[1].get_bool();
653
654
0
    const CBlockIndex* pblockindex;
655
0
    const CBlockIndex* tip;
656
0
    ChainstateManager& chainman = EnsureAnyChainman(request.context);
657
0
    {
658
0
        LOCK(cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
659
0
        pblockindex = chainman.m_blockman.LookupBlockIndex(hash);
660
0
        tip = chainman.ActiveChain().Tip();
661
0
    }
662
663
0
    if (!pblockindex) {
664
0
        throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Block not found");
665
0
    }
666
667
0
    if (!fVerbose)
668
0
    {
669
0
        DataStream ssBlock{};
670
0
        ssBlock << pblockindex->GetBlockHeader();
671
0
        std::string strHex = HexStr(ssBlock);
672
0
        return strHex;
673
0
    }
674
675
0
    return blockheaderToJSON(*tip, *pblockindex, chainman.GetConsensus().powLimit);
676
0
},
677
0
    };
678
0
}
679
680
void CheckBlockDataAvailability(BlockManager& blockman, const CBlockIndex& blockindex, bool check_for_undo)
681
0
{
682
0
    AssertLockHeld(cs_main);
Line
Count
Source
144
0
#define AssertLockHeld(cs) AssertLockHeldInternal(#cs, __FILE__, __LINE__, &cs)
683
0
    uint32_t flag = check_for_undo ? BLOCK_HAVE_UNDO : BLOCK_HAVE_DATA;
684
0
    if (!(blockindex.nStatus & flag)) {
685
0
        if (blockman.IsBlockPruned(blockindex)) {
686
0
            throw JSONRPCError(RPC_MISC_ERROR, strprintf("%s not available (pruned data)", check_for_undo ? "Undo data" : "Block"));
Line
Count
Source
1172
0
#define strprintf tfm::format
687
0
        }
688
0
        if (check_for_undo) {
689
0
            throw JSONRPCError(RPC_MISC_ERROR, "Undo data not available");
690
0
        }
691
0
        throw JSONRPCError(RPC_MISC_ERROR, "Block not available (not fully downloaded)");
692
0
    }
693
0
}
694
695
static CBlock GetBlockChecked(BlockManager& blockman, const CBlockIndex& blockindex)
696
0
{
697
0
    CBlock block;
698
0
    {
699
0
        LOCK(cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
700
0
        CheckBlockDataAvailability(blockman, blockindex, /*check_for_undo=*/false);
701
0
    }
702
703
0
    if (!blockman.ReadBlock(block, blockindex)) {
704
        // Block not found on disk. This shouldn't normally happen unless the block was
705
        // pruned right after we released the lock above.
706
0
        throw JSONRPCError(RPC_MISC_ERROR, "Block not found on disk");
707
0
    }
708
709
0
    return block;
710
0
}
711
712
static std::vector<std::byte> GetRawBlockChecked(BlockManager& blockman, const CBlockIndex& blockindex)
713
0
{
714
0
    FlatFilePos pos{};
715
0
    {
716
0
        LOCK(cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
717
0
        CheckBlockDataAvailability(blockman, blockindex, /*check_for_undo=*/false);
718
0
        pos = blockindex.GetBlockPos();
719
0
    }
720
721
0
    if (auto data{blockman.ReadRawBlock(pos)}) return std::move(*data);
722
    // Block not found on disk. This shouldn't normally happen unless the block was
723
    // pruned right after we released the lock above.
724
0
    throw JSONRPCError(RPC_MISC_ERROR, "Block not found on disk");
725
0
}
726
727
static CBlockUndo GetUndoChecked(BlockManager& blockman, const CBlockIndex& blockindex)
728
0
{
729
0
    CBlockUndo blockUndo;
730
731
    // The Genesis block does not have undo data
732
0
    if (blockindex.nHeight == 0) return blockUndo;
733
734
0
    {
735
0
        LOCK(cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
736
0
        CheckBlockDataAvailability(blockman, blockindex, /*check_for_undo=*/true);
737
0
    }
738
739
0
    if (!blockman.ReadBlockUndo(blockUndo, blockindex)) {
740
0
        throw JSONRPCError(RPC_MISC_ERROR, "Can't read undo data from disk");
741
0
    }
742
743
0
    return blockUndo;
744
0
}
745
746
const RPCResult& GetBlockVin()
747
0
{
748
0
    static const RPCResult getblock_vin{
749
0
        RPCResult::Type::ARR, "vin", "",
750
0
        {
751
0
            {RPCResult::Type::OBJ, "", "",
752
0
            {
753
0
                {RPCResult::Type::ELISION, "", "The same output as verbosity = 2"},
754
0
                {RPCResult::Type::OBJ, "prevout", "(Only if undo information is available)",
755
0
                {
756
0
                    {RPCResult::Type::BOOL, "generated", "Coinbase or not"},
757
0
                    {RPCResult::Type::NUM, "height", "The height of the prevout"},
758
0
                    {RPCResult::Type::STR_AMOUNT, "value", "The value in " + CURRENCY_UNIT},
759
0
                    {RPCResult::Type::OBJ, "scriptPubKey", "",
760
0
                    {
761
0
                        {RPCResult::Type::STR, "asm", "Disassembly of the output script"},
762
0
                        {RPCResult::Type::STR, "desc", "Inferred descriptor for the output"},
763
0
                        {RPCResult::Type::STR_HEX, "hex", "The raw output script bytes, hex-encoded"},
764
0
                        {RPCResult::Type::STR, "address", /*optional=*/true, "The Bitcoin address (only if a well-defined address exists)"},
765
0
                        {RPCResult::Type::STR, "type", "The type (one of: " + GetAllOutputTypes() + ")"},
766
0
                    }},
767
0
                }},
768
0
            }},
769
0
        }
770
0
    };
771
0
    return getblock_vin;
772
0
}
773
774
static RPCMethod getblock()
775
0
{
776
0
    return RPCMethod{
777
0
        "getblock",
778
0
        "If verbosity is 0, returns a string that is serialized, hex-encoded data for block 'hash'.\n"
779
0
                "If verbosity is 1, returns an Object with information about block <hash>.\n"
780
0
                "If verbosity is 2, returns an Object with information about block <hash> and information about each transaction.\n"
781
0
                "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",
782
0
                {
783
0
                    {"blockhash", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "The block hash"},
784
0
                    {"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",
785
0
                     RPCArgOptions{.skip_type_check = true}},
786
0
                },
787
0
                {
788
0
                    RPCResult{"for verbosity = 0",
789
0
                RPCResult::Type::STR_HEX, "", "A string that is serialized, hex-encoded data for block 'hash'"},
790
0
                    RPCResult{"for verbosity = 1",
791
0
                RPCResult::Type::OBJ, "", "",
792
0
                {
793
0
                    {RPCResult::Type::STR_HEX, "hash", "the block hash (same as provided)"},
794
0
                    {RPCResult::Type::NUM, "confirmations", "The number of confirmations, or -1 if the block is not on the main chain"},
795
0
                    {RPCResult::Type::NUM, "size", "The block size"},
796
0
                    {RPCResult::Type::NUM, "strippedsize", "The block size excluding witness data"},
797
0
                    {RPCResult::Type::NUM, "weight", "The block weight as defined in BIP 141"},
798
0
                    {RPCResult::Type::OBJ, "coinbase_tx", "Coinbase transaction metadata",
799
0
                    {
800
0
                        {RPCResult::Type::NUM, "version", "The coinbase transaction version"},
801
0
                        {RPCResult::Type::NUM, "locktime", "The coinbase transaction's locktime (nLockTime)"},
802
0
                        {RPCResult::Type::NUM, "sequence", "The coinbase input's sequence number (nSequence)"},
803
0
                        {RPCResult::Type::STR_HEX, "coinbase", "The coinbase input's script"},
804
0
                        {RPCResult::Type::STR_HEX, "witness", /*optional=*/true, "The coinbase input's first (and only) witness stack element, if present"},
805
0
                    }},
806
0
                    {RPCResult::Type::NUM, "height", "The block height or index"},
807
0
                    {RPCResult::Type::NUM, "version", "The block version"},
808
0
                    {RPCResult::Type::STR_HEX, "versionHex", "The block version formatted in hexadecimal"},
809
0
                    {RPCResult::Type::STR_HEX, "merkleroot", "The merkle root"},
810
0
                    {RPCResult::Type::ARR, "tx", "The transaction ids",
811
0
                        {{RPCResult::Type::STR_HEX, "", "The transaction id"}}},
812
0
                    {RPCResult::Type::NUM_TIME, "time",       "The block time expressed in " + UNIX_EPOCH_TIME},
813
0
                    {RPCResult::Type::NUM_TIME, "mediantime", "The median block time expressed in " + UNIX_EPOCH_TIME},
814
0
                    {RPCResult::Type::NUM, "nonce", "The nonce"},
815
0
                    {RPCResult::Type::STR_HEX, "bits", "nBits: compact representation of the block difficulty target"},
816
0
                    {RPCResult::Type::STR_HEX, "target", "The difficulty target"},
817
0
                    {RPCResult::Type::NUM, "difficulty", "The difficulty"},
818
0
                    {RPCResult::Type::STR_HEX, "chainwork", "Expected number of hashes required to produce the chain up to this block (in hex)"},
819
0
                    {RPCResult::Type::NUM, "nTx", "The number of transactions in the block"},
820
0
                    {RPCResult::Type::STR_HEX, "previousblockhash", /*optional=*/true, "The hash of the previous block (if available)"},
821
0
                    {RPCResult::Type::STR_HEX, "nextblockhash", /*optional=*/true, "The hash of the next block (if available)"},
822
0
                }},
823
0
                    RPCResult{"for verbosity = 2",
824
0
                RPCResult::Type::OBJ, "", "",
825
0
                {
826
0
                    {RPCResult::Type::ELISION, "", "Same output as verbosity = 1"},
827
0
                    {RPCResult::Type::ARR, "tx", "",
828
0
                    {
829
0
                        {RPCResult::Type::OBJ, "", "",
830
0
                        {
831
0
                            {RPCResult::Type::ELISION, "", "The transactions in the format of the getrawtransaction RPC. Different from verbosity = 1 \"tx\" result"},
832
0
                            {RPCResult::Type::NUM, "fee", /*optional=*/true, "The transaction fee in " + CURRENCY_UNIT + ", omitted if block undo data is not available"},
833
0
                        }},
834
0
                    }},
835
0
                }},
836
0
                    RPCResult{"for verbosity = 3",
837
0
                RPCResult::Type::OBJ, "", "",
838
0
                {
839
0
                    {RPCResult::Type::ELISION, "", "Same output as verbosity = 2"},
840
0
                    {RPCResult::Type::ARR, "tx", "",
841
0
                    {
842
0
                        {RPCResult::Type::OBJ, "", "",
843
0
                        {
844
0
                            GetBlockVin(),
845
0
                        }},
846
0
                    }},
847
0
                }},
848
0
        },
849
0
                RPCExamples{
850
0
                    HelpExampleCli("getblock", "\"00000000c937983704a73af28acdec37b049d214adbda81d7e2a3dd146f6ed09\"")
851
0
            + HelpExampleRpc("getblock", "\"00000000c937983704a73af28acdec37b049d214adbda81d7e2a3dd146f6ed09\"")
852
0
                },
853
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
854
0
{
855
0
    uint256 hash(ParseHashV(request.params[0], "blockhash"));
856
857
0
    int verbosity{ParseVerbosity(request.params[1], /*default_verbosity=*/1, /*allow_bool=*/true)};
858
859
0
    const CBlockIndex* pblockindex;
860
0
    const CBlockIndex* tip;
861
0
    ChainstateManager& chainman = EnsureAnyChainman(request.context);
862
0
    {
863
0
        LOCK(cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
864
0
        pblockindex = chainman.m_blockman.LookupBlockIndex(hash);
865
0
        tip = chainman.ActiveChain().Tip();
866
867
0
        if (!pblockindex) {
868
0
            throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Block not found");
869
0
        }
870
0
    }
871
872
0
    const std::vector<std::byte> block_data{GetRawBlockChecked(chainman.m_blockman, *pblockindex)};
873
874
0
    if (verbosity <= 0) {
875
0
        return HexStr(block_data);
876
0
    }
877
878
0
    CBlock block{};
879
0
    SpanReader{block_data} >> TX_WITH_WITNESS(block);
880
881
0
    TxVerbosity tx_verbosity;
882
0
    if (verbosity == 1) {
883
0
        tx_verbosity = TxVerbosity::SHOW_TXID;
884
0
    } else if (verbosity == 2) {
885
0
        tx_verbosity = TxVerbosity::SHOW_DETAILS;
886
0
    } else {
887
0
        tx_verbosity = TxVerbosity::SHOW_DETAILS_AND_PREVOUT;
888
0
    }
889
890
0
    return blockToJSON(chainman.m_blockman, block, *tip, *pblockindex, tx_verbosity, chainman.GetConsensus().powLimit);
891
0
},
892
0
    };
893
0
}
894
895
//! Return height of highest block that has been pruned, or std::nullopt if no blocks have been pruned
896
0
std::optional<int> GetPruneHeight(const BlockManager& blockman, const CChain& chain) {
897
0
    AssertLockHeld(::cs_main);
Line
Count
Source
144
0
#define AssertLockHeld(cs) AssertLockHeldInternal(#cs, __FILE__, __LINE__, &cs)
898
899
    // Search for the last block missing block data or undo data. Don't let the
900
    // search consider the genesis block, because the genesis block does not
901
    // have undo data, but should not be considered pruned.
902
0
    const CBlockIndex* first_block{chain[1]};
903
0
    const CBlockIndex* chain_tip{chain.Tip()};
904
905
    // If there are no blocks after the genesis block, or no blocks at all, nothing is pruned.
906
0
    if (!first_block || !chain_tip) return std::nullopt;
907
908
    // If the chain tip is pruned, everything is pruned.
909
0
    if ((chain_tip->nStatus & BLOCK_HAVE_MASK) != BLOCK_HAVE_MASK) return chain_tip->nHeight;
910
911
0
    const auto& first_unpruned{blockman.GetFirstBlock(*chain_tip, /*status_mask=*/BLOCK_HAVE_MASK, first_block)};
912
0
    if (&first_unpruned == first_block) {
913
        // All blocks between first_block and chain_tip have data, so nothing is pruned.
914
0
        return std::nullopt;
915
0
    }
916
917
    // Block before the first unpruned block is the last pruned block.
918
0
    return CHECK_NONFATAL(first_unpruned.pprev)->nHeight;
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
919
0
}
920
921
static RPCMethod pruneblockchain()
922
0
{
923
0
    return RPCMethod{"pruneblockchain",
924
0
                "Attempts to delete block and undo data up to a specified height or timestamp, if eligible for pruning.\n"
925
0
                "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",
926
0
                {
927
0
                    {"height", RPCArg::Type::NUM, RPCArg::Optional::NO, "The block height to prune up to. May be set to a discrete height, or to a " + UNIX_EPOCH_TIME + "\n"
928
0
            "                  to prune blocks whose block time is at least 2 hours older than the provided timestamp."},
929
0
                },
930
0
                RPCResult{
931
0
                    RPCResult::Type::NUM, "", "Height of the last block pruned"},
932
0
                RPCExamples{
933
0
                    HelpExampleCli("pruneblockchain", "1000")
934
0
            + HelpExampleRpc("pruneblockchain", "1000")
935
0
                },
936
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
937
0
{
938
0
    ChainstateManager& chainman = EnsureAnyChainman(request.context);
939
0
    if (!chainman.m_blockman.IsPruneMode()) {
940
0
        throw JSONRPCError(RPC_MISC_ERROR, "Cannot prune blocks because node is not in prune mode.");
941
0
    }
942
943
0
    LOCK(cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
944
0
    Chainstate& active_chainstate = chainman.ActiveChainstate();
945
0
    CChain& active_chain = active_chainstate.m_chain;
946
947
0
    int heightParam = request.params[0].getInt<int>();
948
0
    if (heightParam < 0) {
949
0
        throw JSONRPCError(RPC_INVALID_PARAMETER, "Negative block height.");
950
0
    }
951
952
    // Height value more than a billion is too high to be a block height, and
953
    // too low to be a block time (corresponds to timestamp from Sep 2001).
954
0
    if (heightParam > 1000000000) {
955
        // Add a 2 hour buffer to include blocks which might have had old timestamps
956
0
        const CBlockIndex* pindex = active_chain.FindEarliestAtLeast(heightParam - TIMESTAMP_WINDOW, 0);
957
0
        if (!pindex) {
958
0
            throw JSONRPCError(RPC_INVALID_PARAMETER, "Could not find block with at least the specified timestamp.");
959
0
        }
960
0
        heightParam = pindex->nHeight;
961
0
    }
962
963
0
    unsigned int height = (unsigned int) heightParam;
964
0
    unsigned int chainHeight = (unsigned int) active_chain.Height();
965
0
    if (chainHeight < chainman.GetParams().PruneAfterHeight()) {
966
0
        throw JSONRPCError(RPC_MISC_ERROR, "Blockchain is too short for pruning.");
967
0
    } else if (height > chainHeight) {
968
0
        throw JSONRPCError(RPC_INVALID_PARAMETER, "Blockchain is shorter than the attempted prune height.");
969
0
    } else if (height > chainHeight - MIN_BLOCKS_TO_KEEP) {
970
0
        LogDebug(BCLog::RPC, "Attempt to prune blocks close to the tip.  Retaining the minimum number of blocks.\n");
Line
Count
Source
143
0
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, util::log::ShouldDebugLog, util::log::Level::Debug, __VA_ARGS__)
Line
Count
Source
136
0
    do {                                                                                      \
137
0
        if (shouldlog(category)) {                                                            \
138
0
            detail_LogWithSrcLoc((category), (level), util::log::NO_RATE_LIMIT, __VA_ARGS__); \
Line
Count
Source
119
0
#define detail_LogWithSrcLoc(category, level, ...) util::log::LogPrintFormatInternal(SourceLocation{__func__}, category, level, __VA_ARGS__)
139
0
        }                                                                                     \
140
0
    } while (0)
971
0
        height = chainHeight - MIN_BLOCKS_TO_KEEP;
972
0
    }
973
974
0
    PruneBlockFilesManual(active_chainstate, height);
975
0
    return GetPruneHeight(chainman.m_blockman, active_chain).value_or(-1);
976
0
},
977
0
    };
978
0
}
979
980
CoinStatsHashType ParseHashType(std::string_view hash_type_input)
981
0
{
982
0
    if (hash_type_input == "hash_serialized_3") {
983
0
        return CoinStatsHashType::HASH_SERIALIZED;
984
0
    } else if (hash_type_input == "muhash") {
985
0
        return CoinStatsHashType::MUHASH;
986
0
    } else if (hash_type_input == "none") {
987
0
        return CoinStatsHashType::NONE;
988
0
    } else {
989
0
        throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("'%s' is not a valid hash_type", hash_type_input));
Line
Count
Source
1172
0
#define strprintf tfm::format
990
0
    }
991
0
}
992
993
/**
994
 * Calculate statistics about the unspent transaction output set
995
 *
996
 * @param[in] index_requested Signals if the coinstatsindex should be used (when available).
997
 */
998
static std::optional<kernel::CCoinsStats> GetUTXOStats(CCoinsView* view, node::BlockManager& blockman,
999
                                                       kernel::CoinStatsHashType hash_type,
1000
                                                       const std::function<void()>& interruption_point = {},
1001
                                                       const CBlockIndex* pindex = nullptr,
1002
                                                       bool index_requested = true)
1003
0
{
1004
    // Use CoinStatsIndex if it is requested and available and a hash_type of Muhash or None was requested
1005
0
    if ((hash_type == kernel::CoinStatsHashType::MUHASH || hash_type == kernel::CoinStatsHashType::NONE) && g_coin_stats_index && index_requested) {
1006
0
        if (pindex) {
1007
0
            return g_coin_stats_index->LookUpStats(*pindex);
1008
0
        } else {
1009
0
            CBlockIndex& block_index = *CHECK_NONFATAL(WITH_LOCK(::cs_main, return blockman.LookupBlockIndex(view->GetBestBlock())));
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
1010
0
            return g_coin_stats_index->LookUpStats(block_index);
1011
0
        }
1012
0
    }
1013
1014
    // If the coinstats index isn't requested or is otherwise not usable, the
1015
    // pindex should either be null or equal to the view's best block. This is
1016
    // because without the coinstats index we can only get coinstats about the
1017
    // best block.
1018
0
    CHECK_NONFATAL(!pindex || pindex->GetBlockHash() == view->GetBestBlock());
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
1019
1020
0
    return kernel::ComputeUTXOStats(hash_type, view, blockman, interruption_point);
1021
0
}
1022
1023
static RPCMethod gettxoutsetinfo()
1024
0
{
1025
0
    return RPCMethod{
1026
0
        "gettxoutsetinfo",
1027
0
        "Returns statistics about the unspent transaction output set.\n"
1028
0
                "Note this call may take some time if you are not using coinstatsindex.\n",
1029
0
                {
1030
0
                    {"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
0
                    {"hash_or_height", RPCArg::Type::NUM, RPCArg::DefaultHint{"the current best block"}, "The block hash or height of the target height (only available with coinstatsindex).",
1032
0
                     RPCArgOptions{
1033
0
                         .skip_type_check = true,
1034
0
                         .type_str = {"", "string or numeric"},
1035
0
                     }},
1036
0
                    {"use_index", RPCArg::Type::BOOL, RPCArg::Default{true}, "Use coinstatsindex, if available."},
1037
0
                },
1038
0
                RPCResult{
1039
0
                    RPCResult::Type::OBJ, "", "",
1040
0
                    {
1041
0
                        {RPCResult::Type::NUM, "height", "The block height (index) of the returned statistics"},
1042
0
                        {RPCResult::Type::STR_HEX, "bestblock", "The hash of the block at which these statistics are calculated"},
1043
0
                        {RPCResult::Type::NUM, "txouts", "The number of unspent transaction outputs"},
1044
0
                        {RPCResult::Type::NUM, "bogosize", "Database-independent, meaningless metric indicating the UTXO set size"},
1045
0
                        {RPCResult::Type::STR_HEX, "hash_serialized_3", /*optional=*/true, "The serialized hash (only present if 'hash_serialized_3' hash_type is chosen)"},
1046
0
                        {RPCResult::Type::STR_HEX, "muhash", /*optional=*/true, "The serialized hash (only present if 'muhash' hash_type is chosen)"},
1047
0
                        {RPCResult::Type::NUM, "transactions", /*optional=*/true, "The number of transactions with unspent outputs (not available when coinstatsindex is used)"},
1048
0
                        {RPCResult::Type::NUM, "disk_size", /*optional=*/true, "The estimated size of the chainstate on disk (not available when coinstatsindex is used)"},
1049
0
                        {RPCResult::Type::STR_AMOUNT, "total_amount", "The total amount of coins in the UTXO set"},
1050
0
                        {RPCResult::Type::STR_AMOUNT, "total_unspendable_amount", /*optional=*/true, "The total amount of coins permanently excluded from the UTXO set (only available if coinstatsindex is used)"},
1051
0
                        {RPCResult::Type::OBJ, "block_info", /*optional=*/true, "Info on amounts in the block at this block height (only available if coinstatsindex is used)",
1052
0
                        {
1053
0
                            {RPCResult::Type::STR_AMOUNT, "prevout_spent", "Total amount of all prevouts spent in this block"},
1054
0
                            {RPCResult::Type::STR_AMOUNT, "coinbase", "Coinbase subsidy amount of this block"},
1055
0
                            {RPCResult::Type::STR_AMOUNT, "new_outputs_ex_coinbase", "Total amount of new outputs created by this block"},
1056
0
                            {RPCResult::Type::STR_AMOUNT, "unspendable", "Total amount of unspendable outputs created in this block"},
1057
0
                            {RPCResult::Type::OBJ, "unspendables", "Detailed view of the unspendable categories",
1058
0
                            {
1059
0
                                {RPCResult::Type::STR_AMOUNT, "genesis_block", "The unspendable amount of the Genesis block subsidy"},
1060
0
                                {RPCResult::Type::STR_AMOUNT, "bip30", "Transactions overridden by duplicates (no longer possible with BIP30)"},
1061
0
                                {RPCResult::Type::STR_AMOUNT, "scripts", "Amounts sent to scripts that are unspendable (for example OP_RETURN outputs)"},
1062
0
                                {RPCResult::Type::STR_AMOUNT, "unclaimed_rewards", "Fee rewards that miners did not claim in their coinbase transaction"},
1063
0
                            }}
1064
0
                        }},
1065
0
                    }},
1066
0
                RPCExamples{
1067
0
                    HelpExampleCli("gettxoutsetinfo", "") +
1068
0
                    HelpExampleCli("gettxoutsetinfo", R"("none")") +
1069
0
                    HelpExampleCli("gettxoutsetinfo", R"("none" 1000)") +
1070
0
                    HelpExampleCli("gettxoutsetinfo", R"("none" '"00000000c937983704a73af28acdec37b049d214adbda81d7e2a3dd146f6ed09"')") +
1071
0
                    HelpExampleCli("-named gettxoutsetinfo", R"(hash_type='muhash' use_index='false')") +
1072
0
                    HelpExampleRpc("gettxoutsetinfo", "") +
1073
0
                    HelpExampleRpc("gettxoutsetinfo", R"("none")") +
1074
0
                    HelpExampleRpc("gettxoutsetinfo", R"("none", 1000)") +
1075
0
                    HelpExampleRpc("gettxoutsetinfo", R"("none", "00000000c937983704a73af28acdec37b049d214adbda81d7e2a3dd146f6ed09")")
1076
0
                },
1077
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
1078
0
{
1079
0
    UniValue ret(UniValue::VOBJ);
1080
1081
0
    const CoinStatsHashType hash_type{ParseHashType(self.Arg<std::string_view>("hash_type"))};
1082
0
    bool index_requested = request.params[2].isNull() || request.params[2].get_bool();
1083
1084
0
    NodeContext& node = EnsureAnyNodeContext(request.context);
1085
0
    ChainstateManager& chainman = EnsureChainman(node);
1086
0
    Chainstate& active_chainstate = chainman.ActiveChainstate();
1087
0
    active_chainstate.ForceFlushStateToDisk(/*wipe_cache=*/false);
1088
1089
0
    CCoinsView* coins_view;
1090
0
    BlockManager* blockman;
1091
0
    {
1092
0
        LOCK(::cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
1093
0
        coins_view = &active_chainstate.CoinsDB();
1094
0
        blockman = &active_chainstate.m_blockman;
1095
0
    }
1096
1097
0
    const CBlockIndex* pindex{nullptr};
1098
0
    if (!request.params[1].isNull()) {
1099
0
        if (!g_coin_stats_index) {
1100
0
            throw JSONRPCError(RPC_INVALID_PARAMETER, "Querying specific block heights requires coinstatsindex");
1101
0
        }
1102
1103
0
        if (hash_type == CoinStatsHashType::HASH_SERIALIZED) {
1104
0
            throw JSONRPCError(RPC_INVALID_PARAMETER, "hash_serialized_3 hash type cannot be queried for a specific block");
1105
0
        }
1106
1107
0
        if (!index_requested) {
1108
0
            throw JSONRPCError(RPC_INVALID_PARAMETER, "Cannot set use_index to false when querying for a specific block");
1109
0
        }
1110
0
        pindex = ParseHashOrHeight(request.params[1], chainman);
1111
0
    }
1112
1113
0
    if (index_requested && g_coin_stats_index) {
1114
0
        if (!g_coin_stats_index->BlockUntilSyncedToCurrentChain()) {
1115
0
            const IndexSummary summary{g_coin_stats_index->GetSummary()};
1116
1117
            // If a specific block was requested and the index has already synced past that height, we can return the
1118
            // data already even though the index is not fully synced yet.
1119
0
            if (pindex && pindex->nHeight > summary.best_block_height) {
1120
0
                throw JSONRPCError(RPC_INTERNAL_ERROR, strprintf("Unable to get data because coinstatsindex is still syncing. Current height: %d", summary.best_block_height));
Line
Count
Source
1172
0
#define strprintf tfm::format
1121
0
            }
1122
0
        }
1123
0
    }
1124
1125
0
    const std::optional<CCoinsStats> maybe_stats = GetUTXOStats(coins_view, *blockman, hash_type, node.rpc_interruption_point, pindex, index_requested);
1126
0
    if (maybe_stats.has_value()) {
1127
0
        const CCoinsStats& stats = maybe_stats.value();
1128
0
        ret.pushKV("height", stats.nHeight);
1129
0
        ret.pushKV("bestblock", stats.hashBlock.GetHex());
1130
0
        ret.pushKV("txouts", stats.nTransactionOutputs);
1131
0
        ret.pushKV("bogosize", stats.nBogoSize);
1132
0
        if (hash_type == CoinStatsHashType::HASH_SERIALIZED) {
1133
0
            ret.pushKV("hash_serialized_3", stats.hashSerialized.GetHex());
1134
0
        }
1135
0
        if (hash_type == CoinStatsHashType::MUHASH) {
1136
0
            ret.pushKV("muhash", stats.hashSerialized.GetHex());
1137
0
        }
1138
0
        CHECK_NONFATAL(stats.total_amount.has_value());
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
1139
0
        ret.pushKV("total_amount", ValueFromAmount(stats.total_amount.value()));
1140
0
        if (!stats.index_used) {
1141
0
            ret.pushKV("transactions", stats.nTransactions);
1142
0
            ret.pushKV("disk_size", stats.nDiskSize);
1143
0
        } else {
1144
0
            CCoinsStats prev_stats{};
1145
0
            if (stats.nHeight > 0) {
1146
0
                const CBlockIndex& block_index = *CHECK_NONFATAL(WITH_LOCK(::cs_main, return blockman->LookupBlockIndex(stats.hashBlock)));
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
1147
0
                const std::optional<CCoinsStats> maybe_prev_stats = GetUTXOStats(coins_view, *blockman, hash_type, node.rpc_interruption_point, block_index.pprev, index_requested);
1148
0
                if (!maybe_prev_stats) {
1149
0
                    throw JSONRPCError(RPC_INTERNAL_ERROR, "Unable to read UTXO set");
1150
0
                }
1151
0
                prev_stats = maybe_prev_stats.value();
1152
0
            }
1153
1154
0
            CAmount block_total_unspendable_amount = stats.total_unspendables_genesis_block +
1155
0
                                                     stats.total_unspendables_bip30 +
1156
0
                                                     stats.total_unspendables_scripts +
1157
0
                                                     stats.total_unspendables_unclaimed_rewards;
1158
0
            CAmount prev_block_total_unspendable_amount = prev_stats.total_unspendables_genesis_block +
1159
0
                                                          prev_stats.total_unspendables_bip30 +
1160
0
                                                          prev_stats.total_unspendables_scripts +
1161
0
                                                          prev_stats.total_unspendables_unclaimed_rewards;
1162
1163
0
            ret.pushKV("total_unspendable_amount", ValueFromAmount(block_total_unspendable_amount));
1164
1165
0
            UniValue block_info(UniValue::VOBJ);
1166
            // These per-block values should fit uint64 under normal circumstances
1167
0
            arith_uint256 diff_prevout = stats.total_prevout_spent_amount - prev_stats.total_prevout_spent_amount;
1168
0
            arith_uint256 diff_coinbase = stats.total_coinbase_amount - prev_stats.total_coinbase_amount;
1169
0
            arith_uint256 diff_outputs = stats.total_new_outputs_ex_coinbase_amount - prev_stats.total_new_outputs_ex_coinbase_amount;
1170
0
            CAmount prevout_amount = static_cast<CAmount>(diff_prevout.GetLow64());
1171
0
            CAmount coinbase_amount = static_cast<CAmount>(diff_coinbase.GetLow64());
1172
0
            CAmount outputs_amount = static_cast<CAmount>(diff_outputs.GetLow64());
1173
0
            block_info.pushKV("prevout_spent", ValueFromAmount(prevout_amount));
1174
0
            block_info.pushKV("coinbase", ValueFromAmount(coinbase_amount));
1175
0
            block_info.pushKV("new_outputs_ex_coinbase", ValueFromAmount(outputs_amount));
1176
0
            block_info.pushKV("unspendable", ValueFromAmount(block_total_unspendable_amount - prev_block_total_unspendable_amount));
1177
1178
0
            UniValue unspendables(UniValue::VOBJ);
1179
0
            unspendables.pushKV("genesis_block", ValueFromAmount(stats.total_unspendables_genesis_block - prev_stats.total_unspendables_genesis_block));
1180
0
            unspendables.pushKV("bip30", ValueFromAmount(stats.total_unspendables_bip30 - prev_stats.total_unspendables_bip30));
1181
0
            unspendables.pushKV("scripts", ValueFromAmount(stats.total_unspendables_scripts - prev_stats.total_unspendables_scripts));
1182
0
            unspendables.pushKV("unclaimed_rewards", ValueFromAmount(stats.total_unspendables_unclaimed_rewards - prev_stats.total_unspendables_unclaimed_rewards));
1183
0
            block_info.pushKV("unspendables", std::move(unspendables));
1184
1185
0
            ret.pushKV("block_info", std::move(block_info));
1186
0
        }
1187
0
    } else {
1188
0
        throw JSONRPCError(RPC_INTERNAL_ERROR, "Unable to read UTXO set");
1189
0
    }
1190
0
    return ret;
1191
0
},
1192
0
    };
1193
0
}
1194
1195
static RPCMethod gettxout()
1196
0
{
1197
0
    return RPCMethod{
1198
0
        "gettxout",
1199
0
        "Returns details about an unspent transaction output.\n",
1200
0
        {
1201
0
            {"txid", RPCArg::Type::STR, RPCArg::Optional::NO, "The transaction id"},
1202
0
            {"n", RPCArg::Type::NUM, RPCArg::Optional::NO, "vout number"},
1203
0
            {"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."},
1204
0
        },
1205
0
        {
1206
0
            RPCResult{"If the UTXO was not found", RPCResult::Type::NONE, "", ""},
1207
0
            RPCResult{"Otherwise", RPCResult::Type::OBJ, "", "", {
1208
0
                {RPCResult::Type::STR_HEX, "bestblock", "The hash of the block at the tip of the chain"},
1209
0
                {RPCResult::Type::NUM, "confirmations", "The number of confirmations"},
1210
0
                {RPCResult::Type::STR_AMOUNT, "value", "The transaction value in " + CURRENCY_UNIT},
1211
0
                {RPCResult::Type::OBJ, "scriptPubKey", "", {
1212
0
                    {RPCResult::Type::STR, "asm", "Disassembly of the output script"},
1213
0
                    {RPCResult::Type::STR, "desc", "Inferred descriptor for the output"},
1214
0
                    {RPCResult::Type::STR_HEX, "hex", "The raw output script bytes, hex-encoded"},
1215
0
                    {RPCResult::Type::STR, "type", "The type, eg pubkeyhash"},
1216
0
                    {RPCResult::Type::STR, "address", /*optional=*/true, "The Bitcoin address (only if a well-defined address exists)"},
1217
0
                }},
1218
0
                {RPCResult::Type::BOOL, "coinbase", "Coinbase or not"},
1219
0
            }},
1220
0
        },
1221
0
        RPCExamples{
1222
0
            "\nGet unspent transactions\n"
1223
0
            + HelpExampleCli("listunspent", "") +
1224
0
            "\nView the details\n"
1225
0
            + HelpExampleCli("gettxout", "\"txid\" 1") +
1226
0
            "\nAs a JSON-RPC call\n"
1227
0
            + HelpExampleRpc("gettxout", "\"txid\", 1")
1228
0
                },
1229
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
1230
0
{
1231
0
    NodeContext& node = EnsureAnyNodeContext(request.context);
1232
0
    ChainstateManager& chainman = EnsureChainman(node);
1233
0
    LOCK(cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
1234
1235
0
    UniValue ret(UniValue::VOBJ);
1236
1237
0
    auto hash{Txid::FromUint256(ParseHashV(request.params[0], "txid"))};
1238
0
    COutPoint out{hash, request.params[1].getInt<uint32_t>()};
1239
0
    bool fMempool = true;
1240
0
    if (!request.params[2].isNull())
1241
0
        fMempool = request.params[2].get_bool();
1242
1243
0
    Chainstate& active_chainstate = chainman.ActiveChainstate();
1244
0
    CCoinsViewCache* coins_view = &active_chainstate.CoinsTip();
1245
1246
0
    std::optional<Coin> coin;
1247
0
    if (fMempool) {
1248
0
        const CTxMemPool& mempool = EnsureMemPool(node);
1249
0
        LOCK(mempool.cs);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
1250
0
        CCoinsViewMemPool view(coins_view, mempool);
1251
0
        if (!mempool.isSpent(out)) coin = view.GetCoin(out);
1252
0
    } else {
1253
0
        coin = coins_view->GetCoin(out);
1254
0
    }
1255
0
    if (!coin) return UniValue::VNULL;
1256
1257
0
    const CBlockIndex* pindex = active_chainstate.m_blockman.LookupBlockIndex(coins_view->GetBestBlock());
1258
0
    ret.pushKV("bestblock", pindex->GetBlockHash().GetHex());
1259
0
    if (coin->nHeight == MEMPOOL_HEIGHT) {
1260
0
        ret.pushKV("confirmations", 0);
1261
0
    } else {
1262
0
        ret.pushKV("confirmations", pindex->nHeight - coin->nHeight + 1);
1263
0
    }
1264
0
    ret.pushKV("value", ValueFromAmount(coin->out.nValue));
1265
0
    UniValue o(UniValue::VOBJ);
1266
0
    ScriptToUniv(coin->out.scriptPubKey, /*out=*/o, /*include_hex=*/true, /*include_address=*/true);
1267
0
    ret.pushKV("scriptPubKey", std::move(o));
1268
0
    ret.pushKV("coinbase", coin->IsCoinBase());
1269
1270
0
    return ret;
1271
0
},
1272
0
    };
1273
0
}
1274
1275
static RPCMethod verifychain()
1276
0
{
1277
0
    return RPCMethod{
1278
0
        "verifychain",
1279
0
        "Verifies blockchain database.\n",
1280
0
                {
1281
0
                    {"checklevel", RPCArg::Type::NUM, RPCArg::DefaultHint{strprintf("%d, range=0-4", DEFAULT_CHECKLEVEL)},
Line
Count
Source
1172
0
#define strprintf tfm::format
1282
0
                        strprintf("How thorough the block verification is:\n%s", MakeUnorderedList(CHECKLEVEL_DOC))},
Line
Count
Source
1172
0
#define strprintf tfm::format
1283
0
                    {"nblocks", RPCArg::Type::NUM, RPCArg::DefaultHint{strprintf("%d, 0=all", DEFAULT_CHECKBLOCKS)}, "The number of blocks to check."},
Line
Count
Source
1172
0
#define strprintf tfm::format
1284
0
                },
1285
0
                RPCResult{
1286
0
                    RPCResult::Type::BOOL, "", "Verification finished successfully. If false, check debug log for reason."},
1287
0
                RPCExamples{
1288
0
                    HelpExampleCli("verifychain", "")
1289
0
            + HelpExampleRpc("verifychain", "")
1290
0
                },
1291
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
1292
0
{
1293
0
    const int check_level{request.params[0].isNull() ? DEFAULT_CHECKLEVEL : request.params[0].getInt<int>()};
1294
0
    const int check_depth{request.params[1].isNull() ? DEFAULT_CHECKBLOCKS : request.params[1].getInt<int>()};
1295
1296
0
    ChainstateManager& chainman = EnsureAnyChainman(request.context);
1297
0
    LOCK(cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
1298
1299
0
    Chainstate& active_chainstate = chainman.ActiveChainstate();
1300
0
    return CVerifyDB(chainman.GetNotifications()).VerifyDB(
1301
0
               active_chainstate, chainman.GetParams().GetConsensus(), active_chainstate.CoinsTip(), check_level, check_depth) == VerifyDBResult::SUCCESS;
1302
0
},
1303
0
    };
1304
0
}
1305
1306
static void SoftForkDescPushBack(const CBlockIndex* blockindex, UniValue& softforks, const ChainstateManager& chainman, Consensus::BuriedDeployment dep)
1307
0
{
1308
    // For buried deployments.
1309
1310
0
    if (!DeploymentEnabled(chainman, dep)) return;
1311
1312
0
    UniValue rv(UniValue::VOBJ);
1313
0
    rv.pushKV("type", "buried");
1314
    // getdeploymentinfo reports the softfork as active from when the chain height is
1315
    // one below the activation height
1316
0
    rv.pushKV("active", DeploymentActiveAfter(blockindex, chainman, dep));
1317
0
    rv.pushKV("height", chainman.GetConsensus().DeploymentHeight(dep));
1318
0
    softforks.pushKV(DeploymentName(dep), std::move(rv));
1319
0
}
1320
1321
static void SoftForkDescPushBack(const CBlockIndex* blockindex, UniValue& softforks, const ChainstateManager& chainman, Consensus::DeploymentPos id)
1322
0
{
1323
    // For BIP9 deployments.
1324
0
    if (!DeploymentEnabled(chainman, id)) return;
1325
0
    if (blockindex == nullptr) return;
1326
1327
0
    UniValue bip9(UniValue::VOBJ);
1328
0
    BIP9Info info{chainman.m_versionbitscache.Info(*blockindex, chainman.GetConsensus(), id)};
1329
0
    const auto& depparams{chainman.GetConsensus().vDeployments[id]};
1330
1331
    // BIP9 parameters
1332
0
    if (info.stats.has_value()) {
1333
0
        bip9.pushKV("bit", depparams.bit);
1334
0
    }
1335
0
    bip9.pushKV("start_time", depparams.nStartTime);
1336
0
    bip9.pushKV("timeout", depparams.nTimeout);
1337
0
    bip9.pushKV("min_activation_height", depparams.min_activation_height);
1338
1339
    // BIP9 status
1340
0
    bip9.pushKV("status", info.current_state);
1341
0
    bip9.pushKV("since", info.since);
1342
0
    bip9.pushKV("status_next", info.next_state);
1343
1344
    // BIP9 signalling status, if applicable
1345
0
    if (info.stats.has_value()) {
1346
0
        UniValue statsUV(UniValue::VOBJ);
1347
0
        statsUV.pushKV("period", info.stats->period);
1348
0
        statsUV.pushKV("elapsed", info.stats->elapsed);
1349
0
        statsUV.pushKV("count", info.stats->count);
1350
0
        if (info.stats->threshold > 0 || info.stats->possible) {
1351
0
            statsUV.pushKV("threshold", info.stats->threshold);
1352
0
            statsUV.pushKV("possible", info.stats->possible);
1353
0
        }
1354
0
        bip9.pushKV("statistics", std::move(statsUV));
1355
1356
0
        std::string sig;
1357
0
        sig.reserve(info.signalling_blocks.size());
1358
0
        for (const bool s : info.signalling_blocks) {
1359
0
            sig.push_back(s ? '#' : '-');
1360
0
        }
1361
0
        bip9.pushKV("signalling", sig);
1362
0
    }
1363
1364
0
    UniValue rv(UniValue::VOBJ);
1365
0
    rv.pushKV("type", "bip9");
1366
0
    bool is_active = false;
1367
0
    if (info.active_since.has_value()) {
1368
0
        rv.pushKV("height", *info.active_since);
1369
0
        is_active = (*info.active_since <= blockindex->nHeight + 1);
1370
0
    }
1371
0
    rv.pushKV("active", is_active);
1372
0
    rv.pushKV("bip9", bip9);
1373
0
    softforks.pushKV(DeploymentName(id), std::move(rv));
1374
0
}
1375
1376
// used by rest.cpp:rest_chaininfo, so cannot be static
1377
RPCMethod getblockchaininfo()
1378
0
{
1379
0
    return RPCMethod{"getblockchaininfo",
1380
0
        "Returns an object containing various state info regarding blockchain processing.\n",
1381
0
        {},
1382
0
        RPCResult{
1383
0
            RPCResult::Type::OBJ, "", "",
1384
0
            {
1385
0
                {RPCResult::Type::STR, "chain", "current network name (" LIST_CHAIN_NAMES ")"},
1386
0
                {RPCResult::Type::NUM, "blocks", "the height of the most-work fully-validated chain. The genesis block has height 0"},
1387
0
                {RPCResult::Type::NUM, "headers", "the current number of headers we have validated"},
1388
0
                {RPCResult::Type::STR, "bestblockhash", "the hash of the currently best block"},
1389
0
                {RPCResult::Type::STR_HEX, "bits", "nBits: compact representation of the block difficulty target"},
1390
0
                {RPCResult::Type::STR_HEX, "target", "the difficulty target"},
1391
0
                {RPCResult::Type::NUM, "difficulty", "the current difficulty"},
1392
0
                {RPCResult::Type::NUM_TIME, "time", "the block time expressed in " + UNIX_EPOCH_TIME},
1393
0
                {RPCResult::Type::NUM_TIME, "mediantime", "the median block time expressed in " + UNIX_EPOCH_TIME},
1394
0
                {RPCResult::Type::NUM, "verificationprogress", "estimate of verification progress [0..1]"},
1395
0
                {RPCResult::Type::BOOL, "initialblockdownload", "(debug information) estimate of whether this node is in Initial Block Download mode"},
1396
0
                {RPCResult::Type::OBJ, "backgroundvalidation", /*optional=*/true, "state info regarding background validation process",
1397
0
                {
1398
0
                    {RPCResult::Type::NUM, "snapshotheight", "the height of the snapshot block. Background validation verifies the chain from genesis up to this height"},
1399
0
                    {RPCResult::Type::NUM, "blocks", "the height of the most-work background fully-validated chain. The genesis block has height 0"},
1400
0
                    {RPCResult::Type::STR, "bestblockhash", "the hash of the currently best block validated in the background"},
1401
0
                    {RPCResult::Type::NUM_TIME, "mediantime", "the median block time expressed in " + UNIX_EPOCH_TIME},
1402
0
                    {RPCResult::Type::NUM, "verificationprogress", "estimate of background verification progress [0..1]"},
1403
0
                    {RPCResult::Type::STR_HEX, "chainwork", "total amount of work in background validated chain, in hexadecimal"},
1404
0
                }},
1405
0
                {RPCResult::Type::STR_HEX, "chainwork", "total amount of work in active chain, in hexadecimal"},
1406
0
                {RPCResult::Type::NUM, "size_on_disk", "the estimated size of the block and undo files on disk"},
1407
0
                {RPCResult::Type::BOOL, "pruned", "if the blocks are subject to pruning"},
1408
0
                {RPCResult::Type::NUM, "pruneheight", /*optional=*/true, "the first block unpruned, all previous blocks were pruned (only present if pruning is enabled)"},
1409
0
                {RPCResult::Type::BOOL, "automatic_pruning", /*optional=*/true, "whether automatic pruning is enabled (only present if pruning is enabled)"},
1410
0
                {RPCResult::Type::NUM, "prune_target_size", /*optional=*/true, "the target size used by pruning (only present if automatic pruning is enabled)"},
1411
0
                {RPCResult::Type::STR_HEX, "signet_challenge", /*optional=*/true, "the block challenge (aka. block script), in hexadecimal (only present if the current network is a signet)"},
1412
0
                (IsDeprecatedRPCEnabled("warnings") ?
1413
0
                    RPCResult{RPCResult::Type::STR, "warnings", "any network and blockchain warnings (DEPRECATED)"} :
1414
0
                    RPCResult{RPCResult::Type::ARR, "warnings", "any network and blockchain warnings (run with `-deprecatedrpc=warnings` to return the latest warning as a single string)",
1415
0
                    {
1416
0
                        {RPCResult::Type::STR, "", "warning"},
1417
0
                    }
1418
0
                    }
1419
0
                ),
1420
0
            }},
1421
0
        RPCExamples{
1422
0
            HelpExampleCli("getblockchaininfo", "")
1423
0
            + HelpExampleRpc("getblockchaininfo", "")
1424
0
        },
1425
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
1426
0
{
1427
0
    ChainstateManager& chainman = EnsureAnyChainman(request.context);
1428
0
    LOCK(cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
1429
0
    Chainstate& active_chainstate = chainman.ActiveChainstate();
1430
1431
0
    const CBlockIndex& tip{*CHECK_NONFATAL(active_chainstate.m_chain.Tip())};
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
1432
0
    const int height{tip.nHeight};
1433
0
    UniValue obj(UniValue::VOBJ);
1434
0
    obj.pushKV("chain", chainman.GetParams().GetChainTypeString());
1435
0
    obj.pushKV("blocks", height);
1436
0
    obj.pushKV("headers", chainman.m_best_header ? chainman.m_best_header->nHeight : -1);
1437
0
    obj.pushKV("bestblockhash", tip.GetBlockHash().GetHex());
1438
0
    obj.pushKV("bits", strprintf("%08x", tip.nBits));
Line
Count
Source
1172
0
#define strprintf tfm::format
1439
0
    obj.pushKV("target", GetTarget(tip, chainman.GetConsensus().powLimit).GetHex());
1440
0
    obj.pushKV("difficulty", GetDifficulty(tip));
1441
0
    obj.pushKV("time", tip.GetBlockTime());
1442
0
    obj.pushKV("mediantime", tip.GetMedianTimePast());
1443
0
    obj.pushKV("verificationprogress", chainman.GuessVerificationProgress(&tip));
1444
0
    obj.pushKV("initialblockdownload", chainman.IsInitialBlockDownload());
1445
0
    auto historical_blocks{chainman.GetHistoricalBlockRange()};
1446
0
    if (historical_blocks) {
1447
0
        UniValue background_validation(UniValue::VOBJ);
1448
0
        const CBlockIndex& btip{*CHECK_NONFATAL(historical_blocks->first)};
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
1449
0
        const CBlockIndex& btarget{*CHECK_NONFATAL(historical_blocks->second)};
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
1450
0
        background_validation.pushKV("snapshotheight", btarget.nHeight);
1451
0
        background_validation.pushKV("blocks", btip.nHeight);
1452
0
        background_validation.pushKV("bestblockhash", btip.GetBlockHash().GetHex());
1453
0
        background_validation.pushKV("mediantime", btip.GetMedianTimePast());
1454
0
        background_validation.pushKV("chainwork", btip.nChainWork.GetHex());
1455
0
        background_validation.pushKV("verificationprogress", chainman.GetBackgroundVerificationProgress(btip));
1456
0
        obj.pushKV("backgroundvalidation", std::move(background_validation));
1457
0
    }
1458
0
    obj.pushKV("chainwork", tip.nChainWork.GetHex());
1459
0
    obj.pushKV("size_on_disk", chainman.m_blockman.CalculateCurrentUsage());
1460
0
    obj.pushKV("pruned", chainman.m_blockman.IsPruneMode());
1461
0
    if (chainman.m_blockman.IsPruneMode()) {
1462
0
        const auto prune_height{GetPruneHeight(chainman.m_blockman, active_chainstate.m_chain)};
1463
0
        obj.pushKV("pruneheight", prune_height ? prune_height.value() + 1 : 0);
1464
1465
0
        const bool automatic_pruning{chainman.m_blockman.GetPruneTarget() != BlockManager::PRUNE_TARGET_MANUAL};
1466
0
        obj.pushKV("automatic_pruning",  automatic_pruning);
1467
0
        if (automatic_pruning) {
1468
0
            obj.pushKV("prune_target_size", chainman.m_blockman.GetPruneTarget());
1469
0
        }
1470
0
    }
1471
0
    if (chainman.GetParams().GetChainType() == ChainType::SIGNET) {
1472
0
        const std::vector<uint8_t>& signet_challenge =
1473
0
            chainman.GetParams().GetConsensus().signet_challenge;
1474
0
        obj.pushKV("signet_challenge", HexStr(signet_challenge));
1475
0
    }
1476
1477
0
    NodeContext& node = EnsureAnyNodeContext(request.context);
1478
0
    obj.pushKV("warnings", node::GetWarningsForRpc(*CHECK_NONFATAL(node.warnings), IsDeprecatedRPCEnabled("warnings")));
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
1479
0
    return obj;
1480
0
},
1481
0
    };
1482
0
}
1483
1484
namespace {
1485
const std::vector<RPCResult> RPCHelpForDeployment{
1486
    {RPCResult::Type::STR, "type", "one of \"buried\", \"bip9\""},
1487
    {RPCResult::Type::NUM, "height", /*optional=*/true, "height of the first block which the rules are or will be enforced (only for \"buried\" type, or \"bip9\" type with \"active\" status)"},
1488
    {RPCResult::Type::BOOL, "active", "true if the rules are enforced for the mempool and the next block"},
1489
    {RPCResult::Type::OBJ, "bip9", /*optional=*/true, "status of bip9 softforks (only for \"bip9\" type)",
1490
    {
1491
        {RPCResult::Type::NUM, "bit", /*optional=*/true, "the bit (0-28) in the block version field used to signal this softfork (only for \"started\" and \"locked_in\" status)"},
1492
        {RPCResult::Type::NUM_TIME, "start_time", "the minimum median time past of a block at which the bit gains its meaning"},
1493
        {RPCResult::Type::NUM_TIME, "timeout", "the median time past of a block at which the deployment is considered failed if not yet locked in"},
1494
        {RPCResult::Type::NUM, "min_activation_height", "minimum height of blocks for which the rules may be enforced"},
1495
        {RPCResult::Type::STR, "status", "status of deployment at specified block (one of \"defined\", \"started\", \"locked_in\", \"active\", \"failed\")"},
1496
        {RPCResult::Type::NUM, "since", "height of the first block to which the status applies"},
1497
        {RPCResult::Type::STR, "status_next", "status of deployment at the next block"},
1498
        {RPCResult::Type::OBJ, "statistics", /*optional=*/true, "numeric statistics about signalling for a softfork (only for \"started\" and \"locked_in\" status)",
1499
        {
1500
            {RPCResult::Type::NUM, "period", "the length in blocks of the signalling period"},
1501
            {RPCResult::Type::NUM, "threshold", /*optional=*/true, "the number of blocks with the version bit set required to activate the feature (only for \"started\" status)"},
1502
            {RPCResult::Type::NUM, "elapsed", "the number of blocks elapsed since the beginning of the current period"},
1503
            {RPCResult::Type::NUM, "count", "the number of blocks with the version bit set in the current period"},
1504
            {RPCResult::Type::BOOL, "possible", /*optional=*/true, "returns false if there are not enough blocks left in this period to pass activation threshold (only for \"started\" status)"},
1505
        }},
1506
        {RPCResult::Type::STR, "signalling", /*optional=*/true, "indicates blocks that signalled with a # and blocks that did not with a -"},
1507
    }},
1508
};
1509
1510
UniValue DeploymentInfo(const CBlockIndex* blockindex, const ChainstateManager& chainman)
1511
0
{
1512
0
    UniValue softforks(UniValue::VOBJ);
1513
0
    SoftForkDescPushBack(blockindex, softforks, chainman, Consensus::DEPLOYMENT_HEIGHTINCB);
1514
0
    SoftForkDescPushBack(blockindex, softforks, chainman, Consensus::DEPLOYMENT_DERSIG);
1515
0
    SoftForkDescPushBack(blockindex, softforks, chainman, Consensus::DEPLOYMENT_CLTV);
1516
0
    SoftForkDescPushBack(blockindex, softforks, chainman, Consensus::DEPLOYMENT_CSV);
1517
0
    SoftForkDescPushBack(blockindex, softforks, chainman, Consensus::DEPLOYMENT_SEGWIT);
1518
0
    SoftForkDescPushBack(blockindex, softforks, chainman, Consensus::DEPLOYMENT_TESTDUMMY);
1519
0
    return softforks;
1520
0
}
1521
} // anon namespace
1522
1523
RPCMethod getdeploymentinfo()
1524
0
{
1525
0
    return RPCMethod{"getdeploymentinfo",
1526
0
        "Returns an object containing various state info regarding deployments of consensus changes.\n"
1527
0
        "Consensus changes for which the new rules are enforced from genesis are not listed in \"deployments\".",
1528
0
        {
1529
0
            {"blockhash", RPCArg::Type::STR_HEX, RPCArg::Default{"hash of current chain tip"}, "The block hash at which to query deployment state"},
1530
0
        },
1531
0
        RPCResult{
1532
0
            RPCResult::Type::OBJ, "", "", {
1533
0
                {RPCResult::Type::STR, "hash", "requested block hash (or tip)"},
1534
0
                {RPCResult::Type::NUM, "height", "requested block height (or tip)"},
1535
0
                {RPCResult::Type::ARR, "script_flags", "script verify flags for the block", {
1536
0
                    {RPCResult::Type::STR, "flag", "a script verify flag"},
1537
0
                }},
1538
0
                {RPCResult::Type::OBJ_DYN, "deployments", "", {
1539
0
                    {RPCResult::Type::OBJ, "xxxx", "name of the deployment", RPCHelpForDeployment}
1540
0
                }},
1541
0
            }
1542
0
        },
1543
0
        RPCExamples{ HelpExampleCli("getdeploymentinfo", "") + HelpExampleRpc("getdeploymentinfo", "") },
1544
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
1545
0
        {
1546
0
            const ChainstateManager& chainman = EnsureAnyChainman(request.context);
1547
0
            LOCK(cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
1548
0
            const Chainstate& active_chainstate = chainman.ActiveChainstate();
1549
1550
0
            const CBlockIndex* blockindex;
1551
0
            if (request.params[0].isNull()) {
1552
0
                blockindex = CHECK_NONFATAL(active_chainstate.m_chain.Tip());
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
1553
0
            } else {
1554
0
                const uint256 hash(ParseHashV(request.params[0], "blockhash"));
1555
0
                blockindex = chainman.m_blockman.LookupBlockIndex(hash);
1556
0
                if (!blockindex) {
1557
0
                    throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Block not found");
1558
0
                }
1559
0
            }
1560
1561
0
            UniValue deploymentinfo(UniValue::VOBJ);
1562
0
            deploymentinfo.pushKV("hash", blockindex->GetBlockHash().ToString());
1563
0
            deploymentinfo.pushKV("height", blockindex->nHeight);
1564
0
            {
1565
0
                const auto flagnames = GetScriptFlagNames(GetBlockScriptFlags(*blockindex, chainman));
1566
0
                UniValue uv_flagnames(UniValue::VARR);
1567
0
                uv_flagnames.push_backV(flagnames.begin(), flagnames.end());
1568
0
                deploymentinfo.pushKV("script_flags", uv_flagnames);
1569
0
            }
1570
0
            deploymentinfo.pushKV("deployments", DeploymentInfo(blockindex, chainman));
1571
0
            return deploymentinfo;
1572
0
        },
1573
0
    };
1574
0
}
1575
1576
/** Comparison function for sorting the getchaintips heads.  */
1577
struct CompareBlocksByHeight
1578
{
1579
    bool operator()(const CBlockIndex* a, const CBlockIndex* b) const
1580
0
    {
1581
        /* Make sure that unequal blocks with the same height do not compare
1582
           equal. Use the pointers themselves to make a distinction. */
1583
1584
0
        if (a->nHeight != b->nHeight)
1585
0
          return (a->nHeight > b->nHeight);
1586
1587
0
        return a < b;
1588
0
    }
1589
};
1590
1591
static RPCMethod getchaintips()
1592
0
{
1593
0
    return RPCMethod{"getchaintips",
1594
0
                "Return information about all known tips in the block tree,"
1595
0
                " including the main chain as well as orphaned branches.\n",
1596
0
                {},
1597
0
                RPCResult{
1598
0
                    RPCResult::Type::ARR, "", "",
1599
0
                    {{RPCResult::Type::OBJ, "", "",
1600
0
                        {
1601
0
                            {RPCResult::Type::NUM, "height", "height of the chain tip"},
1602
0
                            {RPCResult::Type::STR_HEX, "hash", "block hash of the tip"},
1603
0
                            {RPCResult::Type::NUM, "branchlen", "zero for main chain, otherwise length of branch connecting the tip to the main chain"},
1604
0
                            {RPCResult::Type::STR, "status", "status of the chain, \"active\" for the main chain\n"
1605
0
            "Possible values for status:\n"
1606
0
            "1.  \"invalid\"               This branch contains at least one invalid block\n"
1607
0
            "2.  \"headers-only\"          Not all blocks for this branch are available, but the headers are valid\n"
1608
0
            "3.  \"valid-headers\"         All blocks are available for this branch, but they were never fully validated\n"
1609
0
            "4.  \"valid-fork\"            This branch is not part of the active chain, but is fully validated\n"
1610
0
            "5.  \"active\"                This is the tip of the active main chain, which is certainly valid"},
1611
0
                        }}}},
1612
0
                RPCExamples{
1613
0
                    HelpExampleCli("getchaintips", "")
1614
0
            + HelpExampleRpc("getchaintips", "")
1615
0
                },
1616
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
1617
0
{
1618
0
    ChainstateManager& chainman = EnsureAnyChainman(request.context);
1619
0
    LOCK(cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
1620
0
    CChain& active_chain = chainman.ActiveChain();
1621
1622
    /*
1623
     * Idea: The set of chain tips is the active chain tip, plus orphan blocks which do not have another orphan building off of them.
1624
     * Algorithm:
1625
     *  - Make one pass through BlockIndex(), picking out the orphan blocks, and also storing a set of the orphan block's pprev pointers.
1626
     *  - Iterate through the orphan blocks. If the block isn't pointed to by another orphan, it is a chain tip.
1627
     *  - Add the active chain tip
1628
     */
1629
0
    std::set<const CBlockIndex*, CompareBlocksByHeight> setTips;
1630
0
    std::set<const CBlockIndex*> setOrphans;
1631
0
    std::set<const CBlockIndex*> setPrevs;
1632
1633
0
    for (const auto& [_, block_index] : chainman.BlockIndex()) {
1634
0
        if (!active_chain.Contains(block_index)) {
1635
0
            setOrphans.insert(&block_index);
1636
0
            setPrevs.insert(block_index.pprev);
1637
0
        }
1638
0
    }
1639
1640
0
    for (std::set<const CBlockIndex*>::iterator it = setOrphans.begin(); it != setOrphans.end(); ++it) {
1641
0
        if (setPrevs.erase(*it) == 0) {
1642
0
            setTips.insert(*it);
1643
0
        }
1644
0
    }
1645
1646
    // Always report the currently active tip.
1647
0
    setTips.insert(active_chain.Tip());
1648
1649
    /* Construct the output array.  */
1650
0
    UniValue res(UniValue::VARR);
1651
0
    for (const CBlockIndex* block : setTips) {
1652
0
        CHECK_NONFATAL(block);
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
1653
0
        UniValue obj(UniValue::VOBJ);
1654
0
        obj.pushKV("height", block->nHeight);
1655
0
        obj.pushKV("hash", block->phashBlock->GetHex());
1656
1657
0
        const int branchLen = block->nHeight - active_chain.FindFork(*block)->nHeight;
1658
0
        obj.pushKV("branchlen", branchLen);
1659
1660
0
        std::string status;
1661
0
        if (active_chain.Contains(*block)) {
1662
            // This block is part of the currently active chain.
1663
0
            status = "active";
1664
0
        } else if (block->nStatus & BLOCK_FAILED_VALID) {
1665
            // This block or one of its ancestors is invalid.
1666
0
            status = "invalid";
1667
0
        } else if (!block->HaveNumChainTxs()) {
1668
            // This block cannot be connected because full block data for it or one of its parents is missing.
1669
0
            status = "headers-only";
1670
0
        } else if (block->IsValid(BLOCK_VALID_SCRIPTS)) {
1671
            // This block is fully validated, but no longer part of the active chain. It was probably the active block once, but was reorganized.
1672
0
            status = "valid-fork";
1673
0
        } else if (block->IsValid(BLOCK_VALID_TREE)) {
1674
            // The headers for this block are valid, but it has not been validated. It was probably never part of the most-work chain.
1675
0
            status = "valid-headers";
1676
0
        } else {
1677
            // No clue.
1678
0
            status = "unknown";
1679
0
        }
1680
0
        obj.pushKV("status", status);
1681
1682
0
        res.push_back(std::move(obj));
1683
0
    }
1684
1685
0
    return res;
1686
0
},
1687
0
    };
1688
0
}
1689
1690
static RPCMethod preciousblock()
1691
0
{
1692
0
    return RPCMethod{
1693
0
        "preciousblock",
1694
0
        "Treats a block as if it were received before others with the same work.\n"
1695
0
                "\nA later preciousblock call can override the effect of an earlier one.\n"
1696
0
                "\nThe effects of preciousblock are not retained across restarts.\n",
1697
0
                {
1698
0
                    {"blockhash", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "the hash of the block to mark as precious"},
1699
0
                },
1700
0
                RPCResult{RPCResult::Type::NONE, "", ""},
1701
0
                RPCExamples{
1702
0
                    HelpExampleCli("preciousblock", "\"blockhash\"")
1703
0
            + HelpExampleRpc("preciousblock", "\"blockhash\"")
1704
0
                },
1705
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
1706
0
{
1707
0
    uint256 hash(ParseHashV(request.params[0], "blockhash"));
1708
0
    CBlockIndex* pblockindex;
1709
1710
0
    ChainstateManager& chainman = EnsureAnyChainman(request.context);
1711
0
    {
1712
0
        LOCK(cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
1713
0
        pblockindex = chainman.m_blockman.LookupBlockIndex(hash);
1714
0
        if (!pblockindex) {
1715
0
            throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Block not found");
1716
0
        }
1717
0
    }
1718
1719
0
    BlockValidationState state;
1720
0
    chainman.ActiveChainstate().PreciousBlock(state, pblockindex);
1721
1722
0
    if (!state.IsValid()) {
1723
0
        throw JSONRPCError(RPC_DATABASE_ERROR, state.ToString());
1724
0
    }
1725
1726
0
    return UniValue::VNULL;
1727
0
},
1728
0
    };
1729
0
}
1730
1731
0
void InvalidateBlock(ChainstateManager& chainman, const uint256 block_hash) {
1732
0
    BlockValidationState state;
1733
0
    CBlockIndex* pblockindex;
1734
0
    {
1735
0
        LOCK(chainman.GetMutex());
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
1736
0
        pblockindex = chainman.m_blockman.LookupBlockIndex(block_hash);
1737
0
        if (!pblockindex) {
1738
0
            throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Block not found");
1739
0
        }
1740
0
    }
1741
0
    chainman.ActiveChainstate().InvalidateBlock(state, pblockindex);
1742
1743
0
    if (state.IsValid()) {
1744
0
        chainman.ActiveChainstate().ActivateBestChain(state);
1745
0
    }
1746
1747
0
    if (!state.IsValid()) {
1748
0
        throw JSONRPCError(RPC_DATABASE_ERROR, state.ToString());
1749
0
    }
1750
0
}
1751
1752
static RPCMethod invalidateblock()
1753
0
{
1754
0
    return RPCMethod{
1755
0
        "invalidateblock",
1756
0
        "Permanently marks a block as invalid, as if it violated a consensus rule.\n",
1757
0
                {
1758
0
                    {"blockhash", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "the hash of the block to mark as invalid"},
1759
0
                },
1760
0
                RPCResult{RPCResult::Type::NONE, "", ""},
1761
0
                RPCExamples{
1762
0
                    HelpExampleCli("invalidateblock", "\"blockhash\"")
1763
0
            + HelpExampleRpc("invalidateblock", "\"blockhash\"")
1764
0
                },
1765
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
1766
0
{
1767
0
    ChainstateManager& chainman = EnsureAnyChainman(request.context);
1768
0
    uint256 hash(ParseHashV(request.params[0], "blockhash"));
1769
1770
0
    InvalidateBlock(chainman, hash);
1771
1772
0
    return UniValue::VNULL;
1773
0
},
1774
0
    };
1775
0
}
1776
1777
0
void ReconsiderBlock(ChainstateManager& chainman, uint256 block_hash) {
1778
0
    {
1779
0
        LOCK(chainman.GetMutex());
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
1780
0
        CBlockIndex* pblockindex = chainman.m_blockman.LookupBlockIndex(block_hash);
1781
0
        if (!pblockindex) {
1782
0
            throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Block not found");
1783
0
        }
1784
1785
0
        chainman.ActiveChainstate().ResetBlockFailureFlags(pblockindex);
1786
0
        chainman.RecalculateBestHeader();
1787
0
    }
1788
1789
0
    BlockValidationState state;
1790
0
    chainman.ActiveChainstate().ActivateBestChain(state);
1791
1792
0
    if (!state.IsValid()) {
1793
0
        throw JSONRPCError(RPC_DATABASE_ERROR, state.ToString());
1794
0
    }
1795
0
}
1796
1797
static RPCMethod reconsiderblock()
1798
0
{
1799
0
    return RPCMethod{
1800
0
        "reconsiderblock",
1801
0
        "Removes invalidity status of a block, its ancestors and its descendants, reconsider them for activation.\n"
1802
0
                "This can be used to undo the effects of invalidateblock.\n",
1803
0
                {
1804
0
                    {"blockhash", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "the hash of the block to reconsider"},
1805
0
                },
1806
0
                RPCResult{RPCResult::Type::NONE, "", ""},
1807
0
                RPCExamples{
1808
0
                    HelpExampleCli("reconsiderblock", "\"blockhash\"")
1809
0
            + HelpExampleRpc("reconsiderblock", "\"blockhash\"")
1810
0
                },
1811
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
1812
0
{
1813
0
    ChainstateManager& chainman = EnsureAnyChainman(request.context);
1814
0
    uint256 hash(ParseHashV(request.params[0], "blockhash"));
1815
1816
0
    ReconsiderBlock(chainman, hash);
1817
1818
0
    return UniValue::VNULL;
1819
0
},
1820
0
    };
1821
0
}
1822
1823
static RPCMethod getchaintxstats()
1824
0
{
1825
0
    return RPCMethod{
1826
0
        "getchaintxstats",
1827
0
        "Compute statistics about the total number and rate of transactions in the chain.\n",
1828
0
                {
1829
0
                    {"nblocks", RPCArg::Type::NUM, RPCArg::DefaultHint{"one month"}, "Size of the window in number of blocks"},
1830
0
                    {"blockhash", RPCArg::Type::STR_HEX, RPCArg::DefaultHint{"chain tip"}, "The hash of the block that ends the window."},
1831
0
                },
1832
0
                RPCResult{
1833
0
                    RPCResult::Type::OBJ, "", "",
1834
0
                    {
1835
0
                        {RPCResult::Type::NUM_TIME, "time", "The timestamp for the final block in the window, expressed in " + UNIX_EPOCH_TIME},
1836
0
                        {RPCResult::Type::NUM, "txcount", /*optional=*/true,
1837
0
                         "The total number of transactions in the chain up to that point, if known. "
1838
0
                         "It may be unknown when using assumeutxo."},
1839
0
                        {RPCResult::Type::STR_HEX, "window_final_block_hash", "The hash of the final block in the window"},
1840
0
                        {RPCResult::Type::NUM, "window_final_block_height", "The height of the final block in the window."},
1841
0
                        {RPCResult::Type::NUM, "window_block_count", "Size of the window in number of blocks"},
1842
0
                        {RPCResult::Type::NUM, "window_interval", /*optional=*/true, "The elapsed time in the window in seconds. Only returned if \"window_block_count\" is > 0"},
1843
0
                        {RPCResult::Type::NUM, "window_tx_count", /*optional=*/true,
1844
0
                         "The number of transactions in the window. "
1845
0
                         "Only returned if \"window_block_count\" is > 0 and if txcount exists for the start and end of the window."},
1846
0
                        {RPCResult::Type::NUM, "txrate", /*optional=*/true,
1847
0
                         "The average rate of transactions per second in the window. "
1848
0
                         "Only returned if \"window_interval\" is > 0 and if window_tx_count exists."},
1849
0
                    }},
1850
0
                RPCExamples{
1851
0
                    HelpExampleCli("getchaintxstats", "")
1852
0
            + HelpExampleRpc("getchaintxstats", "2016")
1853
0
                },
1854
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
1855
0
{
1856
0
    ChainstateManager& chainman = EnsureAnyChainman(request.context);
1857
0
    const CBlockIndex* pindex;
1858
0
    int blockcount = 30 * 24 * 60 * 60 / chainman.GetParams().GetConsensus().nPowTargetSpacing; // By default: 1 month
1859
1860
0
    if (request.params[1].isNull()) {
1861
0
        LOCK(cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
1862
0
        pindex = chainman.ActiveChain().Tip();
1863
0
    } else {
1864
0
        uint256 hash(ParseHashV(request.params[1], "blockhash"));
1865
0
        LOCK(cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
1866
0
        pindex = chainman.m_blockman.LookupBlockIndex(hash);
1867
0
        if (!pindex) {
1868
0
            throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Block not found");
1869
0
        }
1870
0
        if (!chainman.ActiveChain().Contains(*pindex)) {
1871
0
            throw JSONRPCError(RPC_INVALID_PARAMETER, "Block is not in main chain");
1872
0
        }
1873
0
    }
1874
1875
0
    CHECK_NONFATAL(pindex != nullptr);
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
1876
1877
0
    if (request.params[0].isNull()) {
1878
0
        blockcount = std::max(0, std::min(blockcount, pindex->nHeight - 1));
1879
0
    } else {
1880
0
        blockcount = request.params[0].getInt<int>();
1881
1882
0
        if (blockcount < 0 || (blockcount > 0 && blockcount >= pindex->nHeight)) {
1883
0
            throw JSONRPCError(RPC_INVALID_PARAMETER, "Invalid block count: should be between 0 and the block's height - 1");
1884
0
        }
1885
0
    }
1886
1887
0
    const CBlockIndex& past_block{*CHECK_NONFATAL(pindex->GetAncestor(pindex->nHeight - blockcount))};
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
1888
0
    const int64_t nTimeDiff{pindex->GetMedianTimePast() - past_block.GetMedianTimePast()};
1889
1890
0
    UniValue ret(UniValue::VOBJ);
1891
0
    ret.pushKV("time", pindex->nTime);
1892
0
    if (pindex->m_chain_tx_count) {
1893
0
        ret.pushKV("txcount", pindex->m_chain_tx_count);
1894
0
    }
1895
0
    ret.pushKV("window_final_block_hash", pindex->GetBlockHash().GetHex());
1896
0
    ret.pushKV("window_final_block_height", pindex->nHeight);
1897
0
    ret.pushKV("window_block_count", blockcount);
1898
0
    if (blockcount > 0) {
1899
0
        ret.pushKV("window_interval", nTimeDiff);
1900
0
        if (pindex->m_chain_tx_count != 0 && past_block.m_chain_tx_count != 0) {
1901
0
            const auto window_tx_count = pindex->m_chain_tx_count - past_block.m_chain_tx_count;
1902
0
            ret.pushKV("window_tx_count", window_tx_count);
1903
0
            if (nTimeDiff > 0) {
1904
0
                ret.pushKV("txrate", double(window_tx_count) / nTimeDiff);
1905
0
            }
1906
0
        }
1907
0
    }
1908
1909
0
    return ret;
1910
0
},
1911
0
    };
1912
0
}
1913
1914
template<typename T>
1915
static T CalculateTruncatedMedian(std::vector<T>& scores)
1916
0
{
1917
0
    size_t size = scores.size();
1918
0
    if (size == 0) {
1919
0
        return 0;
1920
0
    }
1921
1922
0
    std::sort(scores.begin(), scores.end());
1923
0
    if (size % 2 == 0) {
1924
0
        return (scores[size / 2 - 1] + scores[size / 2]) / 2;
1925
0
    } else {
1926
0
        return scores[size / 2];
1927
0
    }
1928
0
}
1929
1930
void CalculatePercentilesByWeight(CAmount result[NUM_GETBLOCKSTATS_PERCENTILES], std::vector<std::pair<CAmount, int64_t>>& scores, int64_t total_weight)
1931
0
{
1932
0
    if (scores.empty()) {
1933
0
        return;
1934
0
    }
1935
1936
0
    std::sort(scores.begin(), scores.end());
1937
1938
    // 10th, 25th, 50th, 75th, and 90th percentile weight units.
1939
0
    const double weights[NUM_GETBLOCKSTATS_PERCENTILES] = {
1940
0
        total_weight / 10.0, total_weight / 4.0, total_weight / 2.0, (total_weight * 3.0) / 4.0, (total_weight * 9.0) / 10.0
1941
0
    };
1942
1943
0
    int64_t next_percentile_index = 0;
1944
0
    int64_t cumulative_weight = 0;
1945
0
    for (const auto& element : scores) {
1946
0
        cumulative_weight += element.second;
1947
0
        while (next_percentile_index < NUM_GETBLOCKSTATS_PERCENTILES && cumulative_weight >= weights[next_percentile_index]) {
1948
0
            result[next_percentile_index] = element.first;
1949
0
            ++next_percentile_index;
1950
0
        }
1951
0
    }
1952
1953
    // Fill any remaining percentiles with the last value.
1954
0
    for (int64_t i = next_percentile_index; i < NUM_GETBLOCKSTATS_PERCENTILES; i++) {
1955
0
        result[i] = scores.back().first;
1956
0
    }
1957
0
}
1958
1959
template<typename T>
1960
0
static inline bool SetHasKeys(const std::set<T>& set) {return false;}
1961
template<typename T, typename Tk, typename... Args>
1962
static inline bool SetHasKeys(const std::set<T>& set, const Tk& key, const Args&... args)
1963
0
{
1964
0
    return (set.contains(key)) || SetHasKeys(set, args...);
1965
0
}
Unexecuted instantiation: blockchain.cpp:bool SetHasKeys<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, char [14], char [21], char [14], char [21], char [9], char [7], char [11], char [7], char [7], char [11], char [11]>(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, char const (&) [14], char const (&) [21], char const (&) [14], char const (&) [21], char const (&) [9], char const (&) [7], char const (&) [11], char const (&) [7], char const (&) [7], char const (&) [11], char const (&) [11])
Unexecuted instantiation: blockchain.cpp:bool SetHasKeys<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, char [21], char [14], char [21], char [9], char [7], char [11], char [7], char [7], char [11], char [11]>(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, char const (&) [21], char const (&) [14], char const (&) [21], char const (&) [9], char const (&) [7], char const (&) [11], char const (&) [7], char const (&) [7], char const (&) [11], char const (&) [11])
Unexecuted instantiation: blockchain.cpp:bool SetHasKeys<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, char [14], char [21], char [9], char [7], char [11], char [7], char [7], char [11], char [11]>(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, char const (&) [14], char const (&) [21], char const (&) [9], char const (&) [7], char const (&) [11], char const (&) [7], char const (&) [7], char const (&) [11], char const (&) [11])
Unexecuted instantiation: blockchain.cpp:bool SetHasKeys<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, char [21], char [9], char [7], char [11], char [7], char [7], char [11], char [11]>(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, char const (&) [21], char const (&) [9], char const (&) [7], char const (&) [11], char const (&) [7], char const (&) [7], char const (&) [11], char const (&) [11])
Unexecuted instantiation: blockchain.cpp:bool SetHasKeys<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, char [9], char [7], char [11], char [7], char [7], char [11], char [11]>(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, char const (&) [9], char const (&) [7], char const (&) [11], char const (&) [7], char const (&) [7], char const (&) [11], char const (&) [11])
Unexecuted instantiation: blockchain.cpp:bool SetHasKeys<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, char [7], char [11], char [7], char [7], char [11], char [11]>(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, char const (&) [7], char const (&) [11], char const (&) [7], char const (&) [7], char const (&) [11], char const (&) [11])
Unexecuted instantiation: blockchain.cpp:bool SetHasKeys<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, char [11], char [7], char [7], char [11], char [11]>(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, char const (&) [11], char const (&) [7], char const (&) [7], char const (&) [11], char const (&) [11])
Unexecuted instantiation: blockchain.cpp:bool SetHasKeys<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, char [7], char [7], char [11], char [11]>(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, char const (&) [7], char const (&) [7], char const (&) [11], char const (&) [11])
Unexecuted instantiation: blockchain.cpp:bool SetHasKeys<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, char [7], char [11], char [11]>(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, char const (&) [7], char const (&) [11], char const (&) [11])
Unexecuted instantiation: blockchain.cpp:bool SetHasKeys<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, char [11], char [11]>(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, char const (&) [11], char const (&) [11])
Unexecuted instantiation: blockchain.cpp:bool SetHasKeys<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, char [11]>(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, char const (&) [11])
Unexecuted instantiation: blockchain.cpp:bool SetHasKeys<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, char [11], char [10], char [10], char [10], char [13]>(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, char const (&) [11], char const (&) [10], char const (&) [10], char const (&) [10], char const (&) [13])
Unexecuted instantiation: blockchain.cpp:bool SetHasKeys<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, char [10], char [10], char [10], char [13]>(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, char const (&) [10], char const (&) [10], char const (&) [10], char const (&) [13])
Unexecuted instantiation: blockchain.cpp:bool SetHasKeys<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, char [10], char [10], char [13]>(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, char const (&) [10], char const (&) [10], char const (&) [13])
Unexecuted instantiation: blockchain.cpp:bool SetHasKeys<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, char [10], char [13]>(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, char const (&) [10], char const (&) [13])
Unexecuted instantiation: blockchain.cpp:bool SetHasKeys<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, char [13]>(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, char const (&) [13])
Unexecuted instantiation: blockchain.cpp:bool SetHasKeys<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, char [13], char [11], char [15], char [11], char [20], char [11], char [11]>(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, char const (&) [13], char const (&) [11], char const (&) [15], char const (&) [11], char const (&) [20], char const (&) [11], char const (&) [11])
Unexecuted instantiation: blockchain.cpp:bool SetHasKeys<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, char [11], char [15], char [11], char [20], char [11], char [11]>(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, char const (&) [11], char const (&) [15], char const (&) [11], char const (&) [20], char const (&) [11], char const (&) [11])
Unexecuted instantiation: blockchain.cpp:bool SetHasKeys<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, char [15], char [11], char [20], char [11], char [11]>(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, char const (&) [15], char const (&) [11], char const (&) [20], char const (&) [11], char const (&) [11])
Unexecuted instantiation: blockchain.cpp:bool SetHasKeys<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, char [11], char [20], char [11], char [11]>(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, char const (&) [11], char const (&) [20], char const (&) [11], char const (&) [11])
Unexecuted instantiation: blockchain.cpp:bool SetHasKeys<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, char [20], char [11], char [11]>(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, char const (&) [20], char const (&) [11], char const (&) [11])
Unexecuted instantiation: blockchain.cpp:bool SetHasKeys<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, char [6], char [13], char [15]>(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, char const (&) [6], char const (&) [13], char const (&) [15])
Unexecuted instantiation: blockchain.cpp:bool SetHasKeys<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, char [13], char [15]>(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, char const (&) [13], char const (&) [15])
Unexecuted instantiation: blockchain.cpp:bool SetHasKeys<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, char [15]>(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&, char const (&) [15])
1966
1967
// outpoint (needed for the utxo index) + nHeight|fCoinBase
1968
static constexpr size_t PER_UTXO_OVERHEAD = sizeof(COutPoint) + sizeof(uint32_t);
1969
1970
static RPCMethod getblockstats()
1971
0
{
1972
0
    return RPCMethod{
1973
0
        "getblockstats",
1974
0
        "Compute per block statistics for a given window. All amounts are in satoshis.\n"
1975
0
                "It won't work for some heights with pruning.\n",
1976
0
                {
1977
0
                    {"hash_or_height", RPCArg::Type::NUM, RPCArg::Optional::NO, "The block hash or height of the target block",
1978
0
                     RPCArgOptions{
1979
0
                         .skip_type_check = true,
1980
0
                         .type_str = {"", "string or numeric"},
1981
0
                     }},
1982
0
                    {"stats", RPCArg::Type::ARR, RPCArg::DefaultHint{"all values"}, "Values to plot (see result below)",
1983
0
                        {
1984
0
                            {"height", RPCArg::Type::STR, RPCArg::Optional::OMITTED, "Selected statistic"},
1985
0
                            {"time", RPCArg::Type::STR, RPCArg::Optional::OMITTED, "Selected statistic"},
1986
0
                        },
1987
0
                        RPCArgOptions{.oneline_description="stats"}},
1988
0
                },
1989
0
                RPCResult{
1990
0
            RPCResult::Type::OBJ, "", "",
1991
0
            {
1992
0
                {RPCResult::Type::NUM, "avgfee", /*optional=*/true, "Average fee in the block"},
1993
0
                {RPCResult::Type::NUM, "avgfeerate", /*optional=*/true, "Average feerate (in satoshis per virtual byte)"},
1994
0
                {RPCResult::Type::NUM, "avgtxsize", /*optional=*/true, "Average transaction size"},
1995
0
                {RPCResult::Type::STR_HEX, "blockhash", /*optional=*/true, "The block hash (to check for potential reorgs)"},
1996
0
                {RPCResult::Type::ARR_FIXED, "feerate_percentiles", /*optional=*/true, "Feerates at the 10th, 25th, 50th, 75th, and 90th percentile weight unit (in satoshis per virtual byte)",
1997
0
                {
1998
0
                    {RPCResult::Type::NUM, "10th_percentile_feerate", "The 10th percentile feerate"},
1999
0
                    {RPCResult::Type::NUM, "25th_percentile_feerate", "The 25th percentile feerate"},
2000
0
                    {RPCResult::Type::NUM, "50th_percentile_feerate", "The 50th percentile feerate"},
2001
0
                    {RPCResult::Type::NUM, "75th_percentile_feerate", "The 75th percentile feerate"},
2002
0
                    {RPCResult::Type::NUM, "90th_percentile_feerate", "The 90th percentile feerate"},
2003
0
                }},
2004
0
                {RPCResult::Type::NUM, "height", /*optional=*/true, "The height of the block"},
2005
0
                {RPCResult::Type::NUM, "ins", /*optional=*/true, "The number of inputs (excluding coinbase)"},
2006
0
                {RPCResult::Type::NUM, "maxfee", /*optional=*/true, "Maximum fee in the block"},
2007
0
                {RPCResult::Type::NUM, "maxfeerate", /*optional=*/true, "Maximum feerate (in satoshis per virtual byte)"},
2008
0
                {RPCResult::Type::NUM, "maxtxsize", /*optional=*/true, "Maximum transaction size"},
2009
0
                {RPCResult::Type::NUM, "medianfee", /*optional=*/true, "Truncated median fee in the block"},
2010
0
                {RPCResult::Type::NUM, "mediantime", /*optional=*/true, "The block median time past"},
2011
0
                {RPCResult::Type::NUM, "mediantxsize", /*optional=*/true, "Truncated median transaction size"},
2012
0
                {RPCResult::Type::NUM, "minfee", /*optional=*/true, "Minimum fee in the block"},
2013
0
                {RPCResult::Type::NUM, "minfeerate", /*optional=*/true, "Minimum feerate (in satoshis per virtual byte)"},
2014
0
                {RPCResult::Type::NUM, "mintxsize", /*optional=*/true, "Minimum transaction size"},
2015
0
                {RPCResult::Type::NUM, "outs", /*optional=*/true, "The number of outputs"},
2016
0
                {RPCResult::Type::NUM, "subsidy", /*optional=*/true, "The block subsidy"},
2017
0
                {RPCResult::Type::NUM, "swtotal_size", /*optional=*/true, "Total size of all segwit transactions"},
2018
0
                {RPCResult::Type::NUM, "swtotal_weight", /*optional=*/true, "Total weight of all segwit transactions"},
2019
0
                {RPCResult::Type::NUM, "swtxs", /*optional=*/true, "The number of segwit transactions"},
2020
0
                {RPCResult::Type::NUM, "time", /*optional=*/true, "The block time"},
2021
0
                {RPCResult::Type::NUM, "total_out", /*optional=*/true, "Total amount in all outputs (excluding coinbase and thus reward [ie subsidy + totalfee])"},
2022
0
                {RPCResult::Type::NUM, "total_size", /*optional=*/true, "Total size of all non-coinbase transactions"},
2023
0
                {RPCResult::Type::NUM, "total_weight", /*optional=*/true, "Total weight of all non-coinbase transactions"},
2024
0
                {RPCResult::Type::NUM, "totalfee", /*optional=*/true, "The fee total"},
2025
0
                {RPCResult::Type::NUM, "txs", /*optional=*/true, "The number of transactions (including coinbase)"},
2026
0
                {RPCResult::Type::NUM, "utxo_increase", /*optional=*/true, "The increase/decrease in the number of unspent outputs (not discounting op_return and similar)"},
2027
0
                {RPCResult::Type::NUM, "utxo_size_inc", /*optional=*/true, "The increase/decrease in size for the utxo index (not discounting op_return and similar)"},
2028
0
                {RPCResult::Type::NUM, "utxo_increase_actual", /*optional=*/true, "The increase/decrease in the number of unspent outputs, not counting unspendables"},
2029
0
                {RPCResult::Type::NUM, "utxo_size_inc_actual", /*optional=*/true, "The increase/decrease in size for the utxo index, not counting unspendables"},
2030
0
            }},
2031
0
                RPCExamples{
2032
0
                    HelpExampleCli("getblockstats", R"('"00000000c937983704a73af28acdec37b049d214adbda81d7e2a3dd146f6ed09"' '["minfeerate","avgfeerate"]')") +
2033
0
                    HelpExampleCli("getblockstats", R"(1000 '["minfeerate","avgfeerate"]')") +
2034
0
                    HelpExampleRpc("getblockstats", R"("00000000c937983704a73af28acdec37b049d214adbda81d7e2a3dd146f6ed09", ["minfeerate","avgfeerate"])") +
2035
0
                    HelpExampleRpc("getblockstats", R"(1000, ["minfeerate","avgfeerate"])")
2036
0
                },
2037
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
2038
0
{
2039
0
    ChainstateManager& chainman = EnsureAnyChainman(request.context);
2040
0
    const CBlockIndex& pindex{*CHECK_NONFATAL(ParseHashOrHeight(request.params[0], chainman))};
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
2041
2042
0
    std::set<std::string> stats;
2043
0
    if (!request.params[1].isNull()) {
2044
0
        const UniValue stats_univalue = request.params[1].get_array();
2045
0
        for (unsigned int i = 0; i < stats_univalue.size(); i++) {
2046
0
            const std::string stat = stats_univalue[i].get_str();
2047
0
            stats.insert(stat);
2048
0
        }
2049
0
    }
2050
2051
0
    const CBlock& block = GetBlockChecked(chainman.m_blockman, pindex);
2052
0
    const CBlockUndo& blockUndo = GetUndoChecked(chainman.m_blockman, pindex);
2053
2054
0
    const bool do_all = stats.size() == 0; // Calculate everything if nothing selected (default)
2055
0
    const bool do_mediantxsize = do_all || stats.contains("mediantxsize");
2056
0
    const bool do_medianfee = do_all || stats.contains("medianfee");
2057
0
    const bool do_feerate_percentiles = do_all || stats.contains("feerate_percentiles");
2058
0
    const bool loop_inputs = do_all || do_medianfee || do_feerate_percentiles ||
2059
0
        SetHasKeys(stats, "utxo_increase", "utxo_increase_actual", "utxo_size_inc", "utxo_size_inc_actual", "totalfee", "avgfee", "avgfeerate", "minfee", "maxfee", "minfeerate", "maxfeerate");
2060
0
    const bool loop_outputs = do_all || loop_inputs || stats.contains("total_out");
2061
0
    const bool do_calculate_size = do_mediantxsize ||
2062
0
        SetHasKeys(stats, "total_size", "avgtxsize", "mintxsize", "maxtxsize", "swtotal_size");
2063
0
    const bool do_calculate_weight = do_all || SetHasKeys(stats, "total_weight", "avgfeerate", "swtotal_weight", "avgfeerate", "feerate_percentiles", "minfeerate", "maxfeerate");
2064
0
    const bool do_calculate_sw = do_all || SetHasKeys(stats, "swtxs", "swtotal_size", "swtotal_weight");
2065
2066
0
    CAmount maxfee = 0;
2067
0
    CAmount maxfeerate = 0;
2068
0
    CAmount minfee = MAX_MONEY;
2069
0
    CAmount minfeerate = MAX_MONEY;
2070
0
    CAmount total_out = 0;
2071
0
    CAmount totalfee = 0;
2072
0
    int64_t inputs = 0;
2073
0
    int64_t maxtxsize = 0;
2074
0
    int64_t mintxsize = MAX_BLOCK_SERIALIZED_SIZE;
2075
0
    int64_t outputs = 0;
2076
0
    int64_t swtotal_size = 0;
2077
0
    int64_t swtotal_weight = 0;
2078
0
    int64_t swtxs = 0;
2079
0
    int64_t total_size = 0;
2080
0
    int64_t total_weight = 0;
2081
0
    int64_t utxos = 0;
2082
0
    int64_t utxo_size_inc = 0;
2083
0
    int64_t utxo_size_inc_actual = 0;
2084
0
    std::vector<CAmount> fee_array;
2085
0
    std::vector<std::pair<CAmount, int64_t>> feerate_array;
2086
0
    std::vector<int64_t> txsize_array;
2087
2088
0
    for (size_t i = 0; i < block.vtx.size(); ++i) {
2089
0
        const auto& tx = block.vtx.at(i);
2090
0
        outputs += tx->vout.size();
2091
2092
0
        CAmount tx_total_out = 0;
2093
0
        if (loop_outputs) {
2094
0
            for (const CTxOut& out : tx->vout) {
2095
0
                tx_total_out += out.nValue;
2096
2097
0
                uint64_t out_size{GetSerializeSize(out) + PER_UTXO_OVERHEAD};
2098
0
                utxo_size_inc += out_size;
2099
2100
                // The Genesis block and the repeated BIP30 block coinbases don't change the UTXO
2101
                // set counts, so they have to be excluded from the statistics
2102
0
                if (pindex.nHeight == 0 || (IsBIP30Repeat(pindex) && tx->IsCoinBase())) continue;
2103
                // Skip unspendable outputs since they are not included in the UTXO set
2104
0
                if (out.scriptPubKey.IsUnspendable()) continue;
2105
2106
0
                ++utxos;
2107
0
                utxo_size_inc_actual += out_size;
2108
0
            }
2109
0
        }
2110
2111
0
        if (tx->IsCoinBase()) {
2112
0
            continue;
2113
0
        }
2114
2115
0
        inputs += tx->vin.size(); // Don't count coinbase's fake input
2116
0
        total_out += tx_total_out; // Don't count coinbase reward
2117
2118
0
        int64_t tx_size = 0;
2119
0
        if (do_calculate_size) {
2120
2121
0
            tx_size = tx->ComputeTotalSize();
2122
0
            if (do_mediantxsize) {
2123
0
                txsize_array.push_back(tx_size);
2124
0
            }
2125
0
            maxtxsize = std::max(maxtxsize, tx_size);
2126
0
            mintxsize = std::min(mintxsize, tx_size);
2127
0
            total_size += tx_size;
2128
0
        }
2129
2130
0
        int64_t weight = 0;
2131
0
        if (do_calculate_weight) {
2132
0
            weight = GetTransactionWeight(*tx);
2133
0
            total_weight += weight;
2134
0
        }
2135
2136
0
        if (do_calculate_sw && tx->HasWitness()) {
2137
0
            ++swtxs;
2138
0
            swtotal_size += tx_size;
2139
0
            swtotal_weight += weight;
2140
0
        }
2141
2142
0
        if (loop_inputs) {
2143
0
            CAmount tx_total_in = 0;
2144
0
            const auto& txundo = blockUndo.vtxundo.at(i - 1);
2145
0
            for (const Coin& coin: txundo.vprevout) {
2146
0
                const CTxOut& prevoutput = coin.out;
2147
2148
0
                tx_total_in += prevoutput.nValue;
2149
0
                uint64_t prevout_size{GetSerializeSize(prevoutput) + PER_UTXO_OVERHEAD};
2150
0
                utxo_size_inc -= prevout_size;
2151
0
                utxo_size_inc_actual -= prevout_size;
2152
0
            }
2153
2154
0
            CAmount txfee = tx_total_in - tx_total_out;
2155
0
            CHECK_NONFATAL(MoneyRange(txfee));
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
2156
0
            if (do_medianfee) {
2157
0
                fee_array.push_back(txfee);
2158
0
            }
2159
0
            maxfee = std::max(maxfee, txfee);
2160
0
            minfee = std::min(minfee, txfee);
2161
0
            totalfee += txfee;
2162
2163
            // New feerate uses satoshis per virtual byte instead of per serialized byte
2164
0
            CAmount feerate = weight ? (txfee * WITNESS_SCALE_FACTOR) / weight : 0;
2165
0
            if (do_feerate_percentiles) {
2166
0
                feerate_array.emplace_back(feerate, weight);
2167
0
            }
2168
0
            maxfeerate = std::max(maxfeerate, feerate);
2169
0
            minfeerate = std::min(minfeerate, feerate);
2170
0
        }
2171
0
    }
2172
2173
0
    CAmount feerate_percentiles[NUM_GETBLOCKSTATS_PERCENTILES] = { 0 };
2174
0
    CalculatePercentilesByWeight(feerate_percentiles, feerate_array, total_weight);
2175
2176
0
    UniValue feerates_res(UniValue::VARR);
2177
0
    for (int64_t i = 0; i < NUM_GETBLOCKSTATS_PERCENTILES; i++) {
2178
0
        feerates_res.push_back(feerate_percentiles[i]);
2179
0
    }
2180
2181
0
    UniValue ret_all(UniValue::VOBJ);
2182
0
    ret_all.pushKV("avgfee", (block.vtx.size() > 1) ? totalfee / (block.vtx.size() - 1) : 0);
2183
0
    ret_all.pushKV("avgfeerate", total_weight ? (totalfee * WITNESS_SCALE_FACTOR) / total_weight : 0); // Unit: sat/vbyte
2184
0
    ret_all.pushKV("avgtxsize", (block.vtx.size() > 1) ? total_size / (block.vtx.size() - 1) : 0);
2185
0
    ret_all.pushKV("blockhash", pindex.GetBlockHash().GetHex());
2186
0
    ret_all.pushKV("feerate_percentiles", std::move(feerates_res));
2187
0
    ret_all.pushKV("height", pindex.nHeight);
2188
0
    ret_all.pushKV("ins", inputs);
2189
0
    ret_all.pushKV("maxfee", maxfee);
2190
0
    ret_all.pushKV("maxfeerate", maxfeerate);
2191
0
    ret_all.pushKV("maxtxsize", maxtxsize);
2192
0
    ret_all.pushKV("medianfee", CalculateTruncatedMedian(fee_array));
2193
0
    ret_all.pushKV("mediantime", pindex.GetMedianTimePast());
2194
0
    ret_all.pushKV("mediantxsize", CalculateTruncatedMedian(txsize_array));
2195
0
    ret_all.pushKV("minfee", (minfee == MAX_MONEY) ? 0 : minfee);
2196
0
    ret_all.pushKV("minfeerate", (minfeerate == MAX_MONEY) ? 0 : minfeerate);
2197
0
    ret_all.pushKV("mintxsize", mintxsize == MAX_BLOCK_SERIALIZED_SIZE ? 0 : mintxsize);
2198
0
    ret_all.pushKV("outs", outputs);
2199
0
    ret_all.pushKV("subsidy", GetBlockSubsidy(pindex.nHeight, chainman.GetParams().GetConsensus()));
2200
0
    ret_all.pushKV("swtotal_size", swtotal_size);
2201
0
    ret_all.pushKV("swtotal_weight", swtotal_weight);
2202
0
    ret_all.pushKV("swtxs", swtxs);
2203
0
    ret_all.pushKV("time", pindex.GetBlockTime());
2204
0
    ret_all.pushKV("total_out", total_out);
2205
0
    ret_all.pushKV("total_size", total_size);
2206
0
    ret_all.pushKV("total_weight", total_weight);
2207
0
    ret_all.pushKV("totalfee", totalfee);
2208
0
    ret_all.pushKV("txs", block.vtx.size());
2209
0
    ret_all.pushKV("utxo_increase", outputs - inputs);
2210
0
    ret_all.pushKV("utxo_size_inc", utxo_size_inc);
2211
0
    ret_all.pushKV("utxo_increase_actual", utxos - inputs);
2212
0
    ret_all.pushKV("utxo_size_inc_actual", utxo_size_inc_actual);
2213
2214
0
    if (do_all) {
2215
0
        return ret_all;
2216
0
    }
2217
2218
0
    UniValue ret(UniValue::VOBJ);
2219
0
    for (const std::string& stat : stats) {
2220
0
        const UniValue& value = ret_all[stat];
2221
0
        if (value.isNull()) {
2222
0
            throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Invalid selected statistic '%s'", stat));
Line
Count
Source
1172
0
#define strprintf tfm::format
2223
0
        }
2224
0
        ret.pushKV(stat, value);
2225
0
    }
2226
0
    return ret;
2227
0
},
2228
0
    };
2229
0
}
2230
2231
namespace {
2232
//! Search for a given set of pubkey scripts
2233
bool 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
0
{
2235
0
    scan_progress = 0;
2236
0
    count = 0;
2237
0
    while (cursor->Valid()) {
2238
0
        COutPoint key;
2239
0
        Coin coin;
2240
0
        if (!cursor->GetKey(key) || !cursor->GetValue(coin)) return false;
2241
0
        if (++count % 8192 == 0) {
2242
0
            interruption_point();
2243
0
            if (should_abort) {
2244
                // allow to abort the scan via the abort reference
2245
0
                return false;
2246
0
            }
2247
0
        }
2248
0
        if (count % 256 == 0) {
2249
            // update progress reference every 256 item
2250
0
            uint32_t high = 0x100 * *UCharCast(key.hash.begin()) + *(UCharCast(key.hash.begin()) + 1);
2251
0
            scan_progress = (int)(high * 100.0 / 65536.0 + 0.5);
2252
0
        }
2253
0
        if (needles.contains(coin.out.scriptPubKey)) {
2254
0
            out_results.emplace(key, coin);
2255
0
        }
2256
0
        cursor->Next();
2257
0
    }
2258
0
    scan_progress = 100;
2259
0
    return true;
2260
0
}
2261
} // namespace
2262
2263
/** RAII object to prevent concurrency issue when scanning the txout set */
2264
static std::atomic<int> g_scan_progress;
2265
static std::atomic<bool> g_scan_in_progress;
2266
static std::atomic<bool> g_should_abort_scan;
2267
class CoinsViewScanReserver
2268
{
2269
private:
2270
    bool m_could_reserve{false};
2271
public:
2272
0
    explicit CoinsViewScanReserver() = default;
2273
2274
0
    bool reserve() {
2275
0
        CHECK_NONFATAL(!m_could_reserve);
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
2276
0
        if (g_scan_in_progress.exchange(true)) {
2277
0
            return false;
2278
0
        }
2279
0
        CHECK_NONFATAL(g_scan_progress == 0);
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
2280
0
        m_could_reserve = true;
2281
0
        return true;
2282
0
    }
2283
2284
0
    ~CoinsViewScanReserver() {
2285
0
        if (m_could_reserve) {
2286
0
            g_scan_in_progress = false;
2287
0
            g_scan_progress = 0;
2288
0
        }
2289
0
    }
2290
};
2291
2292
static const auto scan_action_arg_desc = RPCArg{
2293
    "action", RPCArg::Type::STR, RPCArg::Optional::NO, "The action to execute\n"
2294
        "\"start\" for starting a scan\n"
2295
        "\"abort\" for aborting the current scan (returns true when abort was successful)\n"
2296
        "\"status\" for progress report (in %) of the current scan"
2297
};
2298
2299
static const auto output_descriptor_obj = RPCArg{
2300
    "", RPCArg::Type::OBJ, RPCArg::Optional::OMITTED, "An object with output descriptor and metadata",
2301
    {
2302
        {"desc", RPCArg::Type::STR, RPCArg::Optional::NO, "An output descriptor"},
2303
        {"range", RPCArg::Type::RANGE, RPCArg::Default{1000}, "The range of HD chain indexes to explore (either end or [begin,end])"},
2304
    }
2305
};
2306
2307
static const auto scan_objects_arg_desc = RPCArg{
2308
    "scanobjects", RPCArg::Type::ARR, RPCArg::Optional::OMITTED, "Array of scan objects. Required for \"start\" action\n"
2309
        "Every scan object is either a string descriptor or an object:",
2310
    {
2311
        {"descriptor", RPCArg::Type::STR, RPCArg::Optional::OMITTED, "An output descriptor"},
2312
        output_descriptor_obj,
2313
    },
2314
    RPCArgOptions{.oneline_description="[scanobjects,...]"},
2315
};
2316
2317
static const auto scan_result_abort = RPCResult{
2318
    "when action=='abort'", RPCResult::Type::BOOL, "success",
2319
    "True if scan will be aborted (not necessarily before this RPC returns), or false if there is no scan to abort"
2320
};
2321
static const auto scan_result_status_none = RPCResult{
2322
    "when action=='status' and no scan is in progress - possibly already completed", RPCResult::Type::NONE, "", ""
2323
};
2324
static const auto scan_result_status_some = RPCResult{
2325
    "when action=='status' and a scan is currently in progress", RPCResult::Type::OBJ, "", "",
2326
    {{RPCResult::Type::NUM, "progress", "Approximate percent complete"},}
2327
};
2328
2329
2330
static RPCMethod scantxoutset()
2331
0
{
2332
    // raw() descriptor corresponding to mainnet address 12cbQLTFMXRnSzktFkuoG3eHoMeFtpTu3S
2333
0
    const std::string EXAMPLE_DESCRIPTOR_RAW = "raw(76a91411b366edfc0a8b66feebae5c2e25a7b6a5d1cf3188ac)#fm24fxxy";
2334
2335
0
    return RPCMethod{
2336
0
        "scantxoutset",
2337
0
        "Scans the unspent transaction output set for entries that match certain output descriptors.\n"
2338
0
        "Examples of output descriptors are:\n"
2339
0
        "    addr(<address>)                      Outputs whose output script corresponds to the specified address (does not include P2PK)\n"
2340
0
        "    raw(<hex script>)                    Outputs whose output script equals the specified hex-encoded bytes\n"
2341
0
        "    combo(<pubkey>)                      P2PK, P2PKH, P2WPKH, and P2SH-P2WPKH outputs for the given pubkey\n"
2342
0
        "    pkh(<pubkey>)                        P2PKH outputs for the given pubkey\n"
2343
0
        "    sh(multi(<n>,<pubkey>,<pubkey>,...)) P2SH-multisig outputs for the given threshold and pubkeys\n"
2344
0
        "    tr(<pubkey>)                         P2TR\n"
2345
0
        "    tr(<pubkey>,{pk(<pubkey>)})          P2TR with single fallback pubkey in tapscript\n"
2346
0
        "    rawtr(<pubkey>)                      P2TR with the specified key as output key rather than inner\n"
2347
0
        "    wsh(and_v(v:pk(<pubkey>),after(2)))  P2WSH miniscript with mandatory pubkey and a timelock\n"
2348
0
        "\nIn the above, <pubkey> either refers to a fixed public key in hexadecimal notation, or to an xpub/xprv optionally followed by one\n"
2349
0
        "or more path elements separated by \"/\", and optionally ending in \"/*\" (unhardened), or \"/*'\" or \"/*h\" (hardened) to specify all\n"
2350
0
        "unhardened or hardened child keys.\n"
2351
0
        "In the latter case, a range needs to be specified by below if different from 1000.\n"
2352
0
        "For more information on output descriptors, see the documentation in the doc/descriptors.md file.\n",
2353
0
        {
2354
0
            scan_action_arg_desc,
2355
0
            scan_objects_arg_desc,
2356
0
        },
2357
0
        {
2358
0
            RPCResult{"when action=='start'; only returns after scan completes", RPCResult::Type::OBJ, "", "", {
2359
0
                {RPCResult::Type::BOOL, "success", "Whether the scan was completed"},
2360
0
                {RPCResult::Type::NUM, "txouts", "The number of unspent transaction outputs scanned"},
2361
0
                {RPCResult::Type::NUM, "height", "The block height at which the scan was done"},
2362
0
                {RPCResult::Type::STR_HEX, "bestblock", "The hash of the block at the tip of the chain"},
2363
0
                {RPCResult::Type::ARR, "unspents", "",
2364
0
                {
2365
0
                    {RPCResult::Type::OBJ, "", "",
2366
0
                    {
2367
0
                        {RPCResult::Type::STR_HEX, "txid", "The transaction id"},
2368
0
                        {RPCResult::Type::NUM, "vout", "The vout value"},
2369
0
                        {RPCResult::Type::STR_HEX, "scriptPubKey", "The output script"},
2370
0
                        {RPCResult::Type::STR, "desc", "A specialized descriptor for the matched output script"},
2371
0
                        {RPCResult::Type::STR_AMOUNT, "amount", "The total amount in " + CURRENCY_UNIT + " of the unspent output"},
2372
0
                        {RPCResult::Type::BOOL, "coinbase", "Whether this is a coinbase output"},
2373
0
                        {RPCResult::Type::NUM, "height", "Height of the unspent transaction output"},
2374
0
                        {RPCResult::Type::STR_HEX, "blockhash", "Blockhash of the unspent transaction output"},
2375
0
                        {RPCResult::Type::NUM, "confirmations", "Number of confirmations of the unspent transaction output when the scan was done"},
2376
0
                    }},
2377
0
                }},
2378
0
                {RPCResult::Type::STR_AMOUNT, "total_amount", "The total amount of all found unspent outputs in " + CURRENCY_UNIT},
2379
0
            }},
2380
0
            scan_result_abort,
2381
0
            scan_result_status_some,
2382
0
            scan_result_status_none,
2383
0
        },
2384
0
        RPCExamples{
2385
0
            HelpExampleCli("scantxoutset", "start \'[\"" + EXAMPLE_DESCRIPTOR_RAW + "\"]\'") +
2386
0
            HelpExampleCli("scantxoutset", "status") +
2387
0
            HelpExampleCli("scantxoutset", "abort") +
2388
0
            HelpExampleRpc("scantxoutset", "\"start\", [\"" + EXAMPLE_DESCRIPTOR_RAW + "\"]") +
2389
0
            HelpExampleRpc("scantxoutset", "\"status\"") +
2390
0
            HelpExampleRpc("scantxoutset", "\"abort\"")
2391
0
        },
2392
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
2393
0
{
2394
0
    UniValue result(UniValue::VOBJ);
2395
0
    const auto action{self.Arg<std::string_view>("action")};
2396
0
    if (action == "status") {
2397
0
        CoinsViewScanReserver reserver;
2398
0
        if (reserver.reserve()) {
2399
            // no scan in progress
2400
0
            return UniValue::VNULL;
2401
0
        }
2402
0
        result.pushKV("progress", g_scan_progress.load());
2403
0
        return result;
2404
0
    } else if (action == "abort") {
2405
0
        CoinsViewScanReserver reserver;
2406
0
        if (reserver.reserve()) {
2407
            // reserve was possible which means no scan was running
2408
0
            return false;
2409
0
        }
2410
        // set the abort flag
2411
0
        g_should_abort_scan = true;
2412
0
        return true;
2413
0
    } else if (action == "start") {
2414
0
        CoinsViewScanReserver reserver;
2415
0
        if (!reserver.reserve()) {
2416
0
            throw JSONRPCError(RPC_INVALID_PARAMETER, "Scan already in progress, use action \"abort\" or \"status\"");
2417
0
        }
2418
2419
0
        if (request.params.size() < 2) {
2420
0
            throw JSONRPCError(RPC_MISC_ERROR, "scanobjects argument is required for the start action");
2421
0
        }
2422
2423
0
        std::set<CScript> needles;
2424
0
        std::map<CScript, std::string> descriptors;
2425
0
        CAmount total_in = 0;
2426
2427
        // loop through the scan objects
2428
0
        for (const UniValue& scanobject : request.params[1].get_array().getValues()) {
2429
0
            FlatSigningProvider provider;
2430
0
            auto scripts = EvalDescriptorStringOrObject(scanobject, provider);
2431
0
            for (CScript& script : scripts) {
2432
0
                std::string inferred = InferDescriptor(script, provider)->ToString();
2433
0
                needles.emplace(script);
2434
0
                descriptors.emplace(std::move(script), std::move(inferred));
2435
0
            }
2436
0
        }
2437
2438
        // Scan the unspent transaction output set for inputs
2439
0
        UniValue unspents(UniValue::VARR);
2440
0
        std::vector<CTxOut> input_txos;
2441
0
        std::map<COutPoint, Coin> coins;
2442
0
        g_should_abort_scan = false;
2443
0
        int64_t count = 0;
2444
0
        std::unique_ptr<CCoinsViewCursor> pcursor;
2445
0
        const CBlockIndex* tip;
2446
0
        NodeContext& node = EnsureAnyNodeContext(request.context);
2447
0
        {
2448
0
            ChainstateManager& chainman = EnsureChainman(node);
2449
0
            LOCK(cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
2450
0
            Chainstate& active_chainstate = chainman.ActiveChainstate();
2451
0
            active_chainstate.ForceFlushStateToDisk(/*wipe_cache=*/false);
2452
0
            pcursor = CHECK_NONFATAL(active_chainstate.CoinsDB().Cursor());
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
2453
0
            tip = CHECK_NONFATAL(active_chainstate.m_chain.Tip());
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
2454
0
        }
2455
0
        bool res = FindScriptPubKey(g_scan_progress, g_should_abort_scan, count, pcursor.get(), needles, coins, node.rpc_interruption_point);
2456
0
        result.pushKV("success", res);
2457
0
        result.pushKV("txouts", count);
2458
0
        result.pushKV("height", tip->nHeight);
2459
0
        result.pushKV("bestblock", tip->GetBlockHash().GetHex());
2460
2461
0
        for (const auto& it : coins) {
2462
0
            const COutPoint& outpoint = it.first;
2463
0
            const Coin& coin = it.second;
2464
0
            const CTxOut& txo = coin.out;
2465
0
            const CBlockIndex& coinb_block{*CHECK_NONFATAL(tip->GetAncestor(coin.nHeight))};
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
2466
0
            input_txos.push_back(txo);
2467
0
            total_in += txo.nValue;
2468
2469
0
            UniValue unspent(UniValue::VOBJ);
2470
0
            unspent.pushKV("txid", outpoint.hash.GetHex());
2471
0
            unspent.pushKV("vout", outpoint.n);
2472
0
            unspent.pushKV("scriptPubKey", HexStr(txo.scriptPubKey));
2473
0
            unspent.pushKV("desc", descriptors[txo.scriptPubKey]);
2474
0
            unspent.pushKV("amount", ValueFromAmount(txo.nValue));
2475
0
            unspent.pushKV("coinbase", coin.IsCoinBase());
2476
0
            unspent.pushKV("height", coin.nHeight);
2477
0
            unspent.pushKV("blockhash", coinb_block.GetBlockHash().GetHex());
2478
0
            unspent.pushKV("confirmations", tip->nHeight - coin.nHeight + 1);
2479
2480
0
            unspents.push_back(std::move(unspent));
2481
0
        }
2482
0
        result.pushKV("unspents", std::move(unspents));
2483
0
        result.pushKV("total_amount", ValueFromAmount(total_in));
2484
0
    } else {
2485
0
        throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Invalid action '%s'", action));
Line
Count
Source
1172
0
#define strprintf tfm::format
2486
0
    }
2487
0
    return result;
2488
0
},
2489
0
    };
2490
0
}
2491
2492
/** RAII object to prevent concurrency issue when scanning blockfilters */
2493
static std::atomic<int> g_scanfilter_progress;
2494
static std::atomic<int> g_scanfilter_progress_height;
2495
static std::atomic<bool> g_scanfilter_in_progress;
2496
static std::atomic<bool> g_scanfilter_should_abort_scan;
2497
class BlockFiltersScanReserver
2498
{
2499
private:
2500
    bool m_could_reserve{false};
2501
public:
2502
0
    explicit BlockFiltersScanReserver() = default;
2503
2504
0
    bool reserve() {
2505
0
        CHECK_NONFATAL(!m_could_reserve);
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
2506
0
        if (g_scanfilter_in_progress.exchange(true)) {
2507
0
            return false;
2508
0
        }
2509
0
        m_could_reserve = true;
2510
0
        return true;
2511
0
    }
2512
2513
0
    ~BlockFiltersScanReserver() {
2514
0
        if (m_could_reserve) {
2515
0
            g_scanfilter_in_progress = false;
2516
0
        }
2517
0
    }
2518
};
2519
2520
static bool CheckBlockFilterMatches(BlockManager& blockman, const CBlockIndex& blockindex, const GCSFilter::ElementSet& needles)
2521
0
{
2522
0
    const CBlock block{GetBlockChecked(blockman, blockindex)};
2523
0
    const CBlockUndo block_undo{GetUndoChecked(blockman, blockindex)};
2524
2525
    // Check if any of the outputs match the scriptPubKey
2526
0
    for (const auto& tx : block.vtx) {
2527
0
        if (std::any_of(tx->vout.cbegin(), tx->vout.cend(), [&](const auto& txout) {
2528
0
                return needles.contains(std::vector<unsigned char>(txout.scriptPubKey.begin(), txout.scriptPubKey.end()));
2529
0
            })) {
2530
0
            return true;
2531
0
        }
2532
0
    }
2533
    // Check if any of the inputs match the scriptPubKey
2534
0
    for (const auto& txundo : block_undo.vtxundo) {
2535
0
        if (std::any_of(txundo.vprevout.cbegin(), txundo.vprevout.cend(), [&](const auto& coin) {
2536
0
                return needles.contains(std::vector<unsigned char>(coin.out.scriptPubKey.begin(), coin.out.scriptPubKey.end()));
2537
0
            })) {
2538
0
            return true;
2539
0
        }
2540
0
    }
2541
2542
0
    return false;
2543
0
}
2544
2545
static RPCMethod scanblocks()
2546
0
{
2547
0
    return RPCMethod{
2548
0
        "scanblocks",
2549
0
        "Return relevant blockhashes for given descriptors (requires blockfilterindex).\n"
2550
0
        "This call may take several minutes. Make sure to use no RPC timeout (bitcoin-cli -rpcclienttimeout=0)",
2551
0
        {
2552
0
            scan_action_arg_desc,
2553
0
            scan_objects_arg_desc,
2554
0
            RPCArg{"start_height", RPCArg::Type::NUM, RPCArg::Default{0}, "Height to start to scan from"},
2555
0
            RPCArg{"stop_height", RPCArg::Type::NUM, RPCArg::DefaultHint{"chain tip"}, "Height to stop to scan"},
2556
0
            RPCArg{"filtertype", RPCArg::Type::STR, RPCArg::Default{BlockFilterTypeName(BlockFilterType::BASIC)}, "The type name of the filter"},
2557
0
            RPCArg{"options", RPCArg::Type::OBJ_NAMED_PARAMS, RPCArg::Optional::OMITTED, "",
2558
0
                {
2559
0
                    {"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"},
2560
0
                },
2561
0
                RPCArgOptions{.oneline_description="options"}},
2562
0
        },
2563
0
        {
2564
0
            scan_result_status_none,
2565
0
            RPCResult{"When action=='start'; only returns after scan completes", RPCResult::Type::OBJ, "", "", {
2566
0
                {RPCResult::Type::NUM, "from_height", "The height we started the scan from"},
2567
0
                {RPCResult::Type::NUM, "to_height", "The height we ended the scan at"},
2568
0
                {RPCResult::Type::ARR, "relevant_blocks", "Blocks that may have matched a scanobject.", {
2569
0
                    {RPCResult::Type::STR_HEX, "blockhash", "A relevant blockhash"},
2570
0
                }},
2571
0
                {RPCResult::Type::BOOL, "completed", "true if the scan process was not aborted"}
2572
0
            }},
2573
0
            RPCResult{"when action=='status' and a scan is currently in progress", RPCResult::Type::OBJ, "", "", {
2574
0
                    {RPCResult::Type::NUM, "progress", "Approximate percent complete"},
2575
0
                    {RPCResult::Type::NUM, "current_height", "Height of the block currently being scanned"},
2576
0
                },
2577
0
            },
2578
0
            scan_result_abort,
2579
0
        },
2580
0
        RPCExamples{
2581
0
            HelpExampleCli("scanblocks", "start '[\"addr(bcrt1q4u4nsgk6ug0sqz7r3rj9tykjxrsl0yy4d0wwte)\"]' 300000") +
2582
0
            HelpExampleCli("scanblocks", "start '[\"addr(bcrt1q4u4nsgk6ug0sqz7r3rj9tykjxrsl0yy4d0wwte)\"]' 100 150 basic") +
2583
0
            HelpExampleCli("scanblocks", "status") +
2584
0
            HelpExampleRpc("scanblocks", "\"start\", [\"addr(bcrt1q4u4nsgk6ug0sqz7r3rj9tykjxrsl0yy4d0wwte)\"], 300000") +
2585
0
            HelpExampleRpc("scanblocks", "\"start\", [\"addr(bcrt1q4u4nsgk6ug0sqz7r3rj9tykjxrsl0yy4d0wwte)\"], 100, 150, \"basic\"") +
2586
0
            HelpExampleRpc("scanblocks", "\"status\"")
2587
0
        },
2588
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
2589
0
{
2590
0
    UniValue ret(UniValue::VOBJ);
2591
0
    auto action{self.Arg<std::string_view>("action")};
2592
0
    if (action == "status") {
2593
0
        BlockFiltersScanReserver reserver;
2594
0
        if (reserver.reserve()) {
2595
            // no scan in progress
2596
0
            return NullUniValue;
2597
0
        }
2598
0
        ret.pushKV("progress", g_scanfilter_progress.load());
2599
0
        ret.pushKV("current_height", g_scanfilter_progress_height.load());
2600
0
        return ret;
2601
0
    } else if (action == "abort") {
2602
0
        BlockFiltersScanReserver reserver;
2603
0
        if (reserver.reserve()) {
2604
            // reserve was possible which means no scan was running
2605
0
            return false;
2606
0
        }
2607
        // set the abort flag
2608
0
        g_scanfilter_should_abort_scan = true;
2609
0
        return true;
2610
0
    } else if (action == "start") {
2611
0
        BlockFiltersScanReserver reserver;
2612
0
        if (!reserver.reserve()) {
2613
0
            throw JSONRPCError(RPC_INVALID_PARAMETER, "Scan already in progress, use action \"abort\" or \"status\"");
2614
0
        }
2615
0
        auto filtertype_name{self.Arg<std::string_view>("filtertype")};
2616
2617
0
        BlockFilterType filtertype;
2618
0
        if (!BlockFilterTypeByName(filtertype_name, filtertype)) {
2619
0
            throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Unknown filtertype");
2620
0
        }
2621
2622
0
        UniValue options{request.params[5].isNull() ? UniValue::VOBJ : request.params[5]};
2623
0
        bool filter_false_positives{options.exists("filter_false_positives") ? options["filter_false_positives"].get_bool() : false};
2624
2625
0
        BlockFilterIndex* index = GetBlockFilterIndex(filtertype);
2626
0
        if (!index) {
2627
0
            throw JSONRPCError(RPC_MISC_ERROR, tfm::format("Index is not enabled for filtertype %s", filtertype_name));
2628
0
        }
2629
2630
0
        NodeContext& node = EnsureAnyNodeContext(request.context);
2631
0
        ChainstateManager& chainman = EnsureChainman(node);
2632
2633
        // set the start-height
2634
0
        const CBlockIndex* start_index = nullptr;
2635
0
        const CBlockIndex* stop_block = nullptr;
2636
0
        {
2637
0
            LOCK(cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
2638
0
            CChain& active_chain = chainman.ActiveChain();
2639
0
            start_index = active_chain.Genesis();
2640
0
            stop_block = active_chain.Tip(); // If no stop block is provided, stop at the chain tip.
2641
0
            if (!request.params[2].isNull()) {
2642
0
                start_index = active_chain[request.params[2].getInt<int>()];
2643
0
                if (!start_index) {
2644
0
                    throw JSONRPCError(RPC_MISC_ERROR, "Invalid start_height");
2645
0
                }
2646
0
            }
2647
0
            if (!request.params[3].isNull()) {
2648
0
                stop_block = active_chain[request.params[3].getInt<int>()];
2649
0
                if (!stop_block || stop_block->nHeight < start_index->nHeight) {
2650
0
                    throw JSONRPCError(RPC_MISC_ERROR, "Invalid stop_height");
2651
0
                }
2652
0
            }
2653
0
        }
2654
0
        CHECK_NONFATAL(start_index);
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
2655
0
        CHECK_NONFATAL(stop_block);
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
2656
2657
        // loop through the scan objects, add scripts to the needle_set
2658
0
        GCSFilter::ElementSet needle_set;
2659
0
        for (const UniValue& scanobject : request.params[1].get_array().getValues()) {
2660
0
            FlatSigningProvider provider;
2661
0
            std::vector<CScript> scripts = EvalDescriptorStringOrObject(scanobject, provider);
2662
0
            for (const CScript& script : scripts) {
2663
0
                needle_set.emplace(script.begin(), script.end());
2664
0
            }
2665
0
        }
2666
0
        UniValue blocks(UniValue::VARR);
2667
0
        const int amount_per_chunk = 10000;
2668
0
        std::vector<BlockFilter> filters;
2669
0
        int start_block_height = start_index->nHeight; // for progress reporting
2670
0
        const int total_blocks_to_process = stop_block->nHeight - start_block_height;
2671
2672
0
        g_scanfilter_should_abort_scan = false;
2673
0
        g_scanfilter_progress = 0;
2674
0
        g_scanfilter_progress_height = start_block_height;
2675
0
        bool completed = true;
2676
2677
0
        const CBlockIndex* end_range = nullptr;
2678
0
        do {
2679
0
            node.rpc_interruption_point(); // allow a clean shutdown
2680
0
            if (g_scanfilter_should_abort_scan) {
2681
0
                completed = false;
2682
0
                break;
2683
0
            }
2684
2685
            // split the lookup range in chunks if we are deeper than 'amount_per_chunk' blocks from the stopping block
2686
0
            int start_block = !end_range ? start_index->nHeight : start_index->nHeight + 1; // to not include the previous round 'end_range' block
2687
0
            end_range = (start_block + amount_per_chunk < stop_block->nHeight) ?
2688
0
                    WITH_LOCK(::cs_main, return chainman.ActiveChain()[start_block + amount_per_chunk]) :
Line
Count
Source
299
0
#define WITH_LOCK(cs, code) (MaybeCheckNotHeld(cs), [&]() -> decltype(auto) { LOCK(cs); code; }())
2689
0
                    stop_block;
2690
2691
0
            if (index->LookupFilterRange(start_block, end_range, filters)) {
2692
0
                for (const BlockFilter& filter : filters) {
2693
                    // compare the elements-set with each filter
2694
0
                    if (filter.GetFilter().MatchAny(needle_set)) {
2695
0
                        if (filter_false_positives) {
2696
                            // Double check the filter matches by scanning the block
2697
0
                            const CBlockIndex& blockindex = *CHECK_NONFATAL(WITH_LOCK(cs_main, return chainman.m_blockman.LookupBlockIndex(filter.GetBlockHash())));
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
2698
2699
0
                            if (!CheckBlockFilterMatches(chainman.m_blockman, blockindex, needle_set)) {
2700
0
                                continue;
2701
0
                            }
2702
0
                        }
2703
2704
0
                        blocks.push_back(filter.GetBlockHash().GetHex());
2705
0
                    }
2706
0
                }
2707
0
            }
2708
0
            start_index = end_range;
2709
2710
            // update progress
2711
0
            int blocks_processed = end_range->nHeight - start_block_height;
2712
0
            if (total_blocks_to_process > 0) { // avoid division by zero
2713
0
                g_scanfilter_progress = (int)(100.0 / total_blocks_to_process * blocks_processed);
2714
0
            } else {
2715
0
                g_scanfilter_progress = 100;
2716
0
            }
2717
0
            g_scanfilter_progress_height = end_range->nHeight;
2718
2719
        // Finish if we reached the stop block
2720
0
        } while (start_index != stop_block);
2721
2722
0
        ret.pushKV("from_height", start_block_height);
2723
0
        ret.pushKV("to_height", start_index->nHeight); // start_index is always the last scanned block here
2724
0
        ret.pushKV("relevant_blocks", std::move(blocks));
2725
0
        ret.pushKV("completed", completed);
2726
0
    } else {
2727
0
        throw JSONRPCError(RPC_INVALID_PARAMETER, tfm::format("Invalid action '%s'", action));
2728
0
    }
2729
0
    return ret;
2730
0
},
2731
0
    };
2732
0
}
2733
2734
static RPCMethod getdescriptoractivity()
2735
0
{
2736
0
    return RPCMethod{
2737
0
        "getdescriptoractivity",
2738
0
        "Get spend and receive activity associated with a set of descriptors for a set of blocks. "
2739
0
        "This command pairs well with the `relevant_blocks` output of `scanblocks()`.\n"
2740
0
        "This call may take several minutes. If you encounter timeouts, try specifying no RPC timeout (bitcoin-cli -rpcclienttimeout=0)",
2741
0
        {
2742
0
            RPCArg{"blockhashes", RPCArg::Type::ARR, RPCArg::Optional::NO, "The list of blockhashes to examine for activity. Order doesn't matter. Must be along main chain or an error is thrown.\n", {
2743
0
                {"blockhash", RPCArg::Type::STR_HEX, RPCArg::Optional::OMITTED, "A valid blockhash"},
2744
0
            }},
2745
0
            RPCArg{"scanobjects", RPCArg::Type::ARR, RPCArg::Optional::NO, "The list of descriptors (scan objects) to examine for activity. Every scan object is either a string descriptor or an object:",
2746
0
                {
2747
0
                    {"descriptor", RPCArg::Type::STR, RPCArg::Optional::OMITTED, "An output descriptor"},
2748
0
                    output_descriptor_obj,
2749
0
                },
2750
0
                RPCArgOptions{.oneline_description="[scanobjects,...]"},
2751
0
            },
2752
0
            {"include_mempool", RPCArg::Type::BOOL, RPCArg::Default{true}, "Whether to include unconfirmed activity"},
2753
0
        },
2754
0
        RPCResult{
2755
0
            RPCResult::Type::OBJ, "", "", {
2756
0
                {RPCResult::Type::ARR, "activity", "events", {
2757
0
                    {RPCResult::Type::OBJ, "", "", {
2758
0
                        {RPCResult::Type::STR, "type", "always 'spend'"},
2759
0
                        {RPCResult::Type::STR_AMOUNT, "amount", "The total amount in " + CURRENCY_UNIT + " of the spent output"},
2760
0
                        {RPCResult::Type::STR_HEX, "blockhash", /*optional=*/true, "The blockhash this spend appears in (omitted if unconfirmed)"},
2761
0
                        {RPCResult::Type::NUM, "height", /*optional=*/true, "Height of the spend (omitted if unconfirmed)"},
2762
0
                        {RPCResult::Type::STR_HEX, "spend_txid", "The txid of the spending transaction"},
2763
0
                        {RPCResult::Type::NUM, "spend_vin", "The input index of the spend"},
2764
0
                        {RPCResult::Type::STR_HEX, "prevout_txid", "The txid of the prevout"},
2765
0
                        {RPCResult::Type::NUM, "prevout_vout", "The vout of the prevout"},
2766
0
                        {RPCResult::Type::OBJ, "prevout_spk", "", ScriptPubKeyDoc()},
2767
0
                    }},
2768
0
                    {RPCResult::Type::OBJ, "", "", {
2769
0
                        {RPCResult::Type::STR, "type", "always 'receive'"},
2770
0
                        {RPCResult::Type::STR_AMOUNT, "amount", "The total amount in " + CURRENCY_UNIT + " of the new output"},
2771
0
                        {RPCResult::Type::STR_HEX, "blockhash", /*optional=*/true, "The block that this receive is in (omitted if unconfirmed)"},
2772
0
                        {RPCResult::Type::NUM, "height", /*optional=*/true, "The height of the receive (omitted if unconfirmed)"},
2773
0
                        {RPCResult::Type::STR_HEX, "txid", "The txid of the receiving transaction"},
2774
0
                        {RPCResult::Type::NUM, "vout", "The vout of the receiving output"},
2775
0
                        {RPCResult::Type::OBJ, "output_spk", "", ScriptPubKeyDoc()},
2776
0
                    }},
2777
                    // TODO is the skip_type_check avoidable with a heterogeneous ARR?
2778
0
                }, {.skip_type_check=true}, },
2779
0
            },
2780
0
        },
2781
0
        RPCExamples{
2782
0
            HelpExampleCli("getdescriptoractivity", "'[\"000000000000000000001347062c12fded7c528943c8ce133987e2e2f5a840ee\"]' '[\"addr(bc1qzl6nsgqzu89a66l50cvwapnkw5shh23zarqkw9)\"]'")
2783
0
        },
2784
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
2785
0
{
2786
0
    UniValue ret(UniValue::VOBJ);
2787
0
    UniValue activity(UniValue::VARR);
2788
0
    NodeContext& node = EnsureAnyNodeContext(request.context);
2789
0
    ChainstateManager& chainman = EnsureChainman(node);
2790
2791
0
    struct CompareByHeightAscending {
2792
0
        bool operator()(const CBlockIndex* a, const CBlockIndex* b) const {
2793
0
            return a->nHeight < b->nHeight;
2794
0
        }
2795
0
    };
2796
2797
0
    std::set<const CBlockIndex*, CompareByHeightAscending> blockindexes_sorted;
2798
2799
0
    {
2800
        // Validate all given blockhashes, and ensure blocks are along a single chain.
2801
0
        LOCK(::cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
2802
0
        for (const UniValue& blockhash : request.params[0].get_array().getValues()) {
2803
0
            uint256 bhash = ParseHashV(blockhash, "blockhash");
2804
0
            CBlockIndex* pindex = chainman.m_blockman.LookupBlockIndex(bhash);
2805
0
            if (!pindex) {
2806
0
                throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Block not found");
2807
0
            }
2808
0
            if (!chainman.ActiveChain().Contains(*pindex)) {
2809
0
                throw JSONRPCError(RPC_INVALID_PARAMETER, "Block is not in main chain");
2810
0
            }
2811
0
            blockindexes_sorted.insert(pindex);
2812
0
        }
2813
0
    }
2814
2815
0
    std::set<CScript> scripts_to_watch;
2816
2817
    // Determine scripts to watch.
2818
0
    for (const UniValue& scanobject : request.params[1].get_array().getValues()) {
2819
0
        FlatSigningProvider provider;
2820
0
        std::vector<CScript> scripts = EvalDescriptorStringOrObject(scanobject, provider);
2821
2822
0
        for (const CScript& script : scripts) {
2823
0
            scripts_to_watch.insert(script);
2824
0
        }
2825
0
    }
2826
2827
0
    const auto AddSpend = [&](
2828
0
            const CScript& spk,
2829
0
            const CAmount val,
2830
0
            const CTransactionRef& tx,
2831
0
            int vin,
2832
0
            const CTxIn& txin,
2833
0
            const CBlockIndex* index
2834
0
            ) {
2835
0
        UniValue event(UniValue::VOBJ);
2836
0
        UniValue spkUv(UniValue::VOBJ);
2837
0
        ScriptToUniv(spk, /*out=*/spkUv, /*include_hex=*/true, /*include_address=*/true);
2838
2839
0
        event.pushKV("type", "spend");
2840
0
        event.pushKV("amount", ValueFromAmount(val));
2841
0
        if (index) {
2842
0
            event.pushKV("blockhash", index->GetBlockHash().ToString());
2843
0
            event.pushKV("height", index->nHeight);
2844
0
        }
2845
0
        event.pushKV("spend_txid", tx->GetHash().ToString());
2846
0
        event.pushKV("spend_vin", vin);
2847
0
        event.pushKV("prevout_txid", txin.prevout.hash.ToString());
2848
0
        event.pushKV("prevout_vout", txin.prevout.n);
2849
0
        event.pushKV("prevout_spk", spkUv);
2850
2851
0
        return event;
2852
0
    };
2853
2854
0
    const auto AddReceive = [&](const CTxOut& txout, const CBlockIndex* index, int vout, const CTransactionRef& tx) {
2855
0
        UniValue event(UniValue::VOBJ);
2856
0
        UniValue spkUv(UniValue::VOBJ);
2857
0
        ScriptToUniv(txout.scriptPubKey, /*out=*/spkUv, /*include_hex=*/true, /*include_address=*/true);
2858
2859
0
        event.pushKV("type", "receive");
2860
0
        event.pushKV("amount", ValueFromAmount(txout.nValue));
2861
0
        if (index) {
2862
0
            event.pushKV("blockhash", index->GetBlockHash().ToString());
2863
0
            event.pushKV("height", index->nHeight);
2864
0
        }
2865
0
        event.pushKV("txid", tx->GetHash().ToString());
2866
0
        event.pushKV("vout", vout);
2867
0
        event.pushKV("output_spk", spkUv);
2868
2869
0
        return event;
2870
0
    };
2871
2872
0
    BlockManager* blockman;
2873
0
    Chainstate& active_chainstate = chainman.ActiveChainstate();
2874
0
    {
2875
0
        LOCK(::cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
2876
0
        blockman = CHECK_NONFATAL(&active_chainstate.m_blockman);
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
2877
0
    }
2878
2879
0
    for (const CBlockIndex* blockindex : blockindexes_sorted) {
2880
0
        const CBlock block{GetBlockChecked(chainman.m_blockman, *blockindex)};
2881
0
        const CBlockUndo block_undo{GetUndoChecked(*blockman, *blockindex)};
2882
2883
0
        for (size_t i = 0; i < block.vtx.size(); ++i) {
2884
0
            const auto& tx = block.vtx.at(i);
2885
2886
0
            if (!tx->IsCoinBase()) {
2887
                // skip coinbase; spends can't happen there.
2888
0
                const auto& txundo = block_undo.vtxundo.at(i - 1);
2889
2890
0
                for (size_t vin_idx = 0; vin_idx < tx->vin.size(); ++vin_idx) {
2891
0
                    const auto& coin = txundo.vprevout.at(vin_idx);
2892
0
                    const auto& txin = tx->vin.at(vin_idx);
2893
0
                    if (scripts_to_watch.contains(coin.out.scriptPubKey)) {
2894
0
                        activity.push_back(AddSpend(
2895
0
                                    coin.out.scriptPubKey, coin.out.nValue, tx, vin_idx, txin, blockindex));
2896
0
                    }
2897
0
                }
2898
0
            }
2899
2900
0
            for (size_t vout_idx = 0; vout_idx < tx->vout.size(); ++vout_idx) {
2901
0
                const auto& vout = tx->vout.at(vout_idx);
2902
0
                if (scripts_to_watch.contains(vout.scriptPubKey)) {
2903
0
                    activity.push_back(AddReceive(vout, blockindex, vout_idx, tx));
2904
0
                }
2905
0
            }
2906
0
        }
2907
0
    }
2908
2909
0
    bool search_mempool = true;
2910
0
    if (!request.params[2].isNull()) {
2911
0
        search_mempool = request.params[2].get_bool();
2912
0
    }
2913
2914
0
    if (search_mempool) {
2915
0
        const CTxMemPool& mempool = EnsureMemPool(node);
2916
0
        LOCK(::cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
2917
0
        LOCK(mempool.cs);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
2918
0
        const CCoinsViewCache& coins_view = &active_chainstate.CoinsTip();
2919
2920
0
        for (const CTxMemPoolEntry& e : mempool.entryAll()) {
2921
0
            const auto& tx = e.GetSharedTx();
2922
2923
0
            for (size_t vin_idx = 0; vin_idx < tx->vin.size(); ++vin_idx) {
2924
0
                CScript scriptPubKey;
2925
0
                CAmount value;
2926
0
                const auto& txin = tx->vin.at(vin_idx);
2927
0
                std::optional<Coin> coin = coins_view.GetCoin(txin.prevout);
2928
2929
                // Check if the previous output is in the chain
2930
0
                if (!coin) {
2931
                    // If not found in the chain, check the mempool. Likely, this is a
2932
                    // child transaction of another transaction in the mempool.
2933
0
                    CTransactionRef prev_tx = CHECK_NONFATAL(mempool.get(txin.prevout.hash));
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
2934
2935
0
                    if (txin.prevout.n >= prev_tx->vout.size()) {
2936
0
                        throw std::runtime_error("Invalid output index");
2937
0
                    }
2938
0
                    const CTxOut& out = prev_tx->vout[txin.prevout.n];
2939
0
                    scriptPubKey = out.scriptPubKey;
2940
0
                    value = out.nValue;
2941
0
                } else {
2942
                    // Coin found in the chain
2943
0
                    const CTxOut& out = coin->out;
2944
0
                    scriptPubKey = out.scriptPubKey;
2945
0
                    value = out.nValue;
2946
0
                }
2947
2948
0
                if (scripts_to_watch.contains(scriptPubKey)) {
2949
0
                    UniValue event(UniValue::VOBJ);
2950
0
                    activity.push_back(AddSpend(
2951
0
                                scriptPubKey, value, tx, vin_idx, txin, nullptr));
2952
0
                }
2953
0
            }
2954
2955
0
            for (size_t vout_idx = 0; vout_idx < tx->vout.size(); ++vout_idx) {
2956
0
                const auto& vout = tx->vout.at(vout_idx);
2957
0
                if (scripts_to_watch.contains(vout.scriptPubKey)) {
2958
0
                    activity.push_back(AddReceive(vout, nullptr, vout_idx, tx));
2959
0
                }
2960
0
            }
2961
0
        }
2962
0
    }
2963
2964
0
    ret.pushKV("activity", activity);
2965
0
    return ret;
2966
0
},
2967
0
    };
2968
0
}
2969
2970
static RPCMethod getblockfilter()
2971
0
{
2972
0
    return RPCMethod{
2973
0
        "getblockfilter",
2974
0
        "Retrieve a BIP 157 content filter for a particular block.\n",
2975
0
                {
2976
0
                    {"blockhash", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "The hash of the block"},
2977
0
                    {"filtertype", RPCArg::Type::STR, RPCArg::Default{BlockFilterTypeName(BlockFilterType::BASIC)}, "The type name of the filter"},
2978
0
                },
2979
0
                RPCResult{
2980
0
                    RPCResult::Type::OBJ, "", "",
2981
0
                    {
2982
0
                        {RPCResult::Type::STR_HEX, "filter", "the hex-encoded filter data"},
2983
0
                        {RPCResult::Type::STR_HEX, "header", "the hex-encoded filter header"},
2984
0
                    }},
2985
0
                RPCExamples{
2986
0
                    HelpExampleCli("getblockfilter", "\"00000000c937983704a73af28acdec37b049d214adbda81d7e2a3dd146f6ed09\" \"basic\"") +
2987
0
                    HelpExampleRpc("getblockfilter", "\"00000000c937983704a73af28acdec37b049d214adbda81d7e2a3dd146f6ed09\", \"basic\"")
2988
0
                },
2989
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
2990
0
{
2991
0
    uint256 block_hash = ParseHashV(request.params[0], "blockhash");
2992
0
    auto filtertype_name{self.Arg<std::string_view>("filtertype")};
2993
2994
0
    BlockFilterType filtertype;
2995
0
    if (!BlockFilterTypeByName(filtertype_name, filtertype)) {
2996
0
        throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Unknown filtertype");
2997
0
    }
2998
2999
0
    BlockFilterIndex* index = GetBlockFilterIndex(filtertype);
3000
0
    if (!index) {
3001
0
        throw JSONRPCError(RPC_MISC_ERROR, tfm::format("Index is not enabled for filtertype %s", filtertype_name));
3002
0
    }
3003
3004
0
    const CBlockIndex* block_index;
3005
0
    bool block_was_connected;
3006
0
    {
3007
0
        ChainstateManager& chainman = EnsureAnyChainman(request.context);
3008
0
        LOCK(cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
3009
0
        block_index = chainman.m_blockman.LookupBlockIndex(block_hash);
3010
0
        if (!block_index) {
3011
0
            throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Block not found");
3012
0
        }
3013
0
        block_was_connected = block_index->IsValid(BLOCK_VALID_SCRIPTS);
3014
0
    }
3015
3016
0
    bool index_ready = index->BlockUntilSyncedToCurrentChain();
3017
3018
0
    BlockFilter filter;
3019
0
    uint256 filter_header;
3020
0
    if (!index->LookupFilter(block_index, filter) ||
3021
0
        !index->LookupFilterHeader(block_index, filter_header)) {
3022
0
        int err_code;
3023
0
        std::string errmsg = "Filter not found.";
3024
3025
0
        if (!block_was_connected) {
3026
0
            err_code = RPC_INVALID_ADDRESS_OR_KEY;
3027
0
            errmsg += " Block was not connected to active chain.";
3028
0
        } else if (!index_ready) {
3029
0
            err_code = RPC_MISC_ERROR;
3030
0
            errmsg += " Block filters are still in the process of being indexed.";
3031
0
        } else {
3032
0
            err_code = RPC_INTERNAL_ERROR;
3033
0
            errmsg += " This error is unexpected and indicates index corruption.";
3034
0
        }
3035
3036
0
        throw JSONRPCError(err_code, errmsg);
3037
0
    }
3038
3039
0
    UniValue ret(UniValue::VOBJ);
3040
0
    ret.pushKV("filter", HexStr(filter.GetEncodedFilter()));
3041
0
    ret.pushKV("header", filter_header.GetHex());
3042
0
    return ret;
3043
0
},
3044
0
    };
3045
0
}
3046
3047
/**
3048
 * RAII class that registers a prune lock in its constructor to prevent
3049
 * block data from being pruned, and removes it in its destructor.
3050
 */
3051
class TemporaryPruneLock
3052
{
3053
    static constexpr const char* LOCK_NAME{"dumptxoutset-rollback"};
3054
    BlockManager& m_blockman;
3055
public:
3056
0
    TemporaryPruneLock(BlockManager& blockman, int height) : m_blockman(blockman)
3057
0
    {
3058
0
        LOCK(::cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
3059
0
        m_blockman.UpdatePruneLock(LOCK_NAME, {height});
3060
0
        LogDebug(BCLog::PRUNE, "dumptxoutset: registered prune lock at height %d", height);
Line
Count
Source
143
0
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, util::log::ShouldDebugLog, util::log::Level::Debug, __VA_ARGS__)
Line
Count
Source
136
0
    do {                                                                                      \
137
0
        if (shouldlog(category)) {                                                            \
138
0
            detail_LogWithSrcLoc((category), (level), util::log::NO_RATE_LIMIT, __VA_ARGS__); \
Line
Count
Source
119
0
#define detail_LogWithSrcLoc(category, level, ...) util::log::LogPrintFormatInternal(SourceLocation{__func__}, category, level, __VA_ARGS__)
139
0
        }                                                                                     \
140
0
    } while (0)
3061
0
    }
3062
    ~TemporaryPruneLock()
3063
0
    {
3064
0
        LOCK(::cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
3065
0
        m_blockman.DeletePruneLock(LOCK_NAME);
3066
0
        LogDebug(BCLog::PRUNE, "dumptxoutset: released prune lock");
Line
Count
Source
143
0
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, util::log::ShouldDebugLog, util::log::Level::Debug, __VA_ARGS__)
Line
Count
Source
136
0
    do {                                                                                      \
137
0
        if (shouldlog(category)) {                                                            \
138
0
            detail_LogWithSrcLoc((category), (level), util::log::NO_RATE_LIMIT, __VA_ARGS__); \
Line
Count
Source
119
0
#define detail_LogWithSrcLoc(category, level, ...) util::log::LogPrintFormatInternal(SourceLocation{__func__}, category, level, __VA_ARGS__)
139
0
        }                                                                                     \
140
0
    } while (0)
3067
0
    }
3068
};
3069
3070
/**
3071
 * Serialize the UTXO set to a file for loading elsewhere.
3072
 *
3073
 * @see SnapshotMetadata
3074
 */
3075
static RPCMethod dumptxoutset()
3076
0
{
3077
0
    return RPCMethod{
3078
0
        "dumptxoutset",
3079
0
        "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"
3080
0
        "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"
3081
0
        "For deep rollbacks, make sure to use no RPC timeout (bitcoin-cli -rpcclienttimeout=0) as it may take several minutes.",
3082
0
        {
3083
0
            {"path", RPCArg::Type::STR, RPCArg::Optional::NO, "Path to the output file. If relative, will be prefixed by datadir."},
3084
0
            {"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
0
            {"options", RPCArg::Type::OBJ_NAMED_PARAMS, RPCArg::Optional::OMITTED, "",
3086
0
                {
3087
0
                    {"rollback", RPCArg::Type::NUM, RPCArg::Optional::OMITTED,
3088
0
                        "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.",
3089
0
                    RPCArgOptions{.skip_type_check = true, .type_str = {"", "string or numeric"}}},
3090
0
                    {"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)."},
3091
0
                },
3092
0
            },
3093
0
        },
3094
0
        RPCResult{
3095
0
            RPCResult::Type::OBJ, "", "",
3096
0
                {
3097
0
                    {RPCResult::Type::NUM, "coins_written", "the number of coins written in the snapshot"},
3098
0
                    {RPCResult::Type::STR_HEX, "base_hash", "the hash of the base of the snapshot"},
3099
0
                    {RPCResult::Type::NUM, "base_height", "the height of the base of the snapshot"},
3100
0
                    {RPCResult::Type::STR, "path", "the absolute path that the snapshot was written to"},
3101
0
                    {RPCResult::Type::STR_HEX, "txoutset_hash", "the hash of the UTXO set contents"},
3102
0
                    {RPCResult::Type::NUM, "nchaintx", "the number of transactions in the chain up to and including the base block"},
3103
0
                }
3104
0
        },
3105
0
        RPCExamples{
3106
0
            HelpExampleCli("-rpcclienttimeout=0 dumptxoutset", "utxo.dat latest") +
3107
0
            HelpExampleCli("-rpcclienttimeout=0 dumptxoutset", "utxo.dat rollback") +
3108
0
            HelpExampleCli("-rpcclienttimeout=0 -named dumptxoutset", R"(utxo.dat rollback=853456)") +
3109
0
            HelpExampleCli("-rpcclienttimeout=0 -named dumptxoutset", R"(utxo.dat rollback=853456 in_memory=true)")
3110
0
        },
3111
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
3112
0
{
3113
0
    NodeContext& node = EnsureAnyNodeContext(request.context);
3114
0
    const CBlockIndex* tip{WITH_LOCK(::cs_main, return node.chainman->ActiveChain().Tip())};
Line
Count
Source
299
0
#define WITH_LOCK(cs, code) (MaybeCheckNotHeld(cs), [&]() -> decltype(auto) { LOCK(cs); code; }())
3115
0
    const CBlockIndex* target_index{nullptr};
3116
0
    const auto snapshot_type{self.Arg<std::string_view>("type")};
3117
0
    const UniValue options{request.params[2].isNull() ? UniValue::VOBJ : request.params[2]};
3118
0
    if (options.exists("rollback")) {
3119
0
        if (!snapshot_type.empty() && snapshot_type != "rollback") {
3120
0
            throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Invalid snapshot type \"%s\" specified with rollback option", snapshot_type));
Line
Count
Source
1172
0
#define strprintf tfm::format
3121
0
        }
3122
0
        target_index = ParseHashOrHeight(options["rollback"], *node.chainman);
3123
0
    } else if (snapshot_type == "rollback") {
3124
0
        auto snapshot_heights = node.chainman->GetParams().GetAvailableSnapshotHeights();
3125
0
        CHECK_NONFATAL(snapshot_heights.size() > 0);
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
3126
0
        auto max_height = std::max_element(snapshot_heights.begin(), snapshot_heights.end());
3127
0
        target_index = ParseHashOrHeight(*max_height, *node.chainman);
3128
0
    } else if (snapshot_type == "latest") {
3129
0
        target_index = tip;
3130
0
    } else {
3131
0
        throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Invalid snapshot type \"%s\" specified. Please specify \"rollback\" or \"latest\"", snapshot_type));
Line
Count
Source
1172
0
#define strprintf tfm::format
3132
0
    }
3133
3134
0
    const ArgsManager& args{EnsureAnyArgsman(request.context)};
3135
0
    const fs::path path = fsbridge::AbsPathJoin(args.GetDataDirNet(), fs::u8path(self.Arg<std::string_view>("path")));
3136
0
    const auto path_info{fs::status(path)};
3137
    // Write to a temporary path and then move into `path` on completion
3138
    // to avoid confusion due to an interruption. If a named pipe passed, write directly to it.
3139
0
    const fs::path temppath = fs::is_fifo(path_info) ? path : path + ".incomplete";
3140
3141
0
    if (fs::exists(path_info) && !fs::is_fifo(path_info)) {
3142
0
        throw JSONRPCError(
3143
0
            RPC_INVALID_PARAMETER,
3144
0
            path.utf8string() + " already exists. If you are sure this is what you want, "
3145
0
            "move it out of the way first");
3146
0
    }
3147
3148
0
    FILE* file{fsbridge::fopen(temppath, "wb")};
3149
0
    AutoFile afile{file};
3150
0
    if (afile.IsNull()) {
3151
0
        throw JSONRPCError(
3152
0
            RPC_INVALID_PARAMETER,
3153
0
            "Couldn't open file " + temppath.utf8string() + " for writing.");
3154
0
    }
3155
3156
0
    UniValue result;
3157
0
    Chainstate& chainstate{node.chainman->ActiveChainstate()};
3158
0
    if (target_index == tip) {
3159
        // Dump the txoutset of the current tip
3160
0
        result = CreateUTXOSnapshot(node, chainstate, std::move(afile), path, temppath);
3161
0
    } else {
3162
        // Check pruning constraints before attempting rollback and prevent
3163
        // pruning of the necessary blocks with a temporary prune lock
3164
0
        std::optional<TemporaryPruneLock> temp_prune_lock;
3165
0
        if (node.chainman->m_blockman.IsPruneMode()) {
3166
0
            LOCK(node.chainman->GetMutex());
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
3167
0
            const CBlockIndex* current_tip{node.chainman->ActiveChain().Tip()};
3168
0
            const CBlockIndex& first_block{node.chainman->m_blockman.GetFirstBlock(*current_tip, /*status_mask=*/BLOCK_HAVE_MASK)};
3169
0
            if (first_block.nHeight > target_index->nHeight) {
3170
0
                throw JSONRPCError(RPC_MISC_ERROR, "Could not roll back to requested height since necessary block data is already pruned.");
3171
0
            }
3172
0
            temp_prune_lock.emplace(node.chainman->m_blockman, target_index->nHeight);
3173
0
        }
3174
3175
0
        const bool in_memory{options.exists("in_memory") ? options["in_memory"].get_bool() : false};
3176
0
        result = CreateRolledBackUTXOSnapshot(node,
3177
0
                                              chainstate,
3178
0
                                              target_index,
3179
0
                                              std::move(afile),
3180
0
                                              path,
3181
0
                                              temppath,
3182
0
                                              in_memory);
3183
0
    }
3184
3185
0
    if (!fs::is_fifo(path_info)) {
3186
0
        fs::rename(temppath, path);
3187
0
    }
3188
3189
0
    return result;
3190
0
},
3191
0
    };
3192
0
}
3193
3194
/**
3195
 * RAII class that creates a temporary database directory in its constructor
3196
 * and removes it in its destructor.
3197
 */
3198
class TemporaryUTXODatabase
3199
{
3200
    fs::path m_path;
3201
public:
3202
0
    TemporaryUTXODatabase(const fs::path& path) : m_path(path) {
3203
0
        fs::create_directories(m_path);
3204
0
    }
3205
0
    ~TemporaryUTXODatabase() {
3206
0
        if (!DestroyDB(fs::PathToString(m_path))) {
3207
0
            LogInfo("Failed to clean up temporary UTXO database at %s, please remove it manually.",
Line
Count
Source
125
0
#define LogInfo(...) detail_LogWithSrcLoc(BCLog::LogFlags::ALL, util::log::Level::Info, __VA_ARGS__)
Line
Count
Source
119
0
#define detail_LogWithSrcLoc(category, level, ...) util::log::LogPrintFormatInternal(SourceLocation{__func__}, category, level, __VA_ARGS__)
3208
0
                    fs::PathToString(m_path));
3209
0
        }
3210
0
    }
3211
};
3212
3213
UniValue CreateRolledBackUTXOSnapshot(
3214
    NodeContext& node,
3215
    Chainstate& chainstate,
3216
    const CBlockIndex* target,
3217
    AutoFile&& afile,
3218
    const fs::path& path,
3219
    const fs::path& tmppath,
3220
    const bool in_memory)
3221
0
{
3222
    // Create a temporary leveldb to store the UTXO set that is being rolled back
3223
0
    std::string temp_db_name{strprintf("temp_utxo_%d", target->nHeight)};
Line
Count
Source
1172
0
#define strprintf tfm::format
3224
0
    fs::path temp_db_path{fsbridge::AbsPathJoin(tmppath.parent_path(), fs::u8path(temp_db_name))};
3225
3226
    // Only create the on-disk temp directory when not using in-memory mode
3227
0
    std::optional<TemporaryUTXODatabase> temp_db_cleaner;
3228
0
    if (!in_memory) {
3229
0
        temp_db_cleaner.emplace(temp_db_path);
3230
0
    } else {
3231
0
        LogInfo("Using in-memory database for UTXO-set rollback (this may require significant RAM).");
Line
Count
Source
125
0
#define LogInfo(...) detail_LogWithSrcLoc(BCLog::LogFlags::ALL, util::log::Level::Info, __VA_ARGS__)
Line
Count
Source
119
0
#define detail_LogWithSrcLoc(category, level, ...) util::log::LogPrintFormatInternal(SourceLocation{__func__}, category, level, __VA_ARGS__)
3232
0
    }
3233
3234
    // Create temporary database
3235
0
    DBParams db_params{
3236
0
        .path = temp_db_path,
3237
0
        .cache_bytes = 0,
3238
0
        .memory_only = in_memory,
3239
0
        .wipe_data = true,
3240
0
        .obfuscate = false,
3241
0
        .options = DBOptions{}
3242
0
    };
3243
3244
0
    std::unique_ptr<CCoinsViewDB> temp_db = std::make_unique<CCoinsViewDB>(
3245
0
        std::move(db_params),
3246
0
        CoinsViewOptions{}
3247
0
    );
3248
3249
0
    const CBlockIndex* tip = nullptr;
3250
0
    LogInfo("Copying current UTXO set to temporary database.");
Line
Count
Source
125
0
#define LogInfo(...) detail_LogWithSrcLoc(BCLog::LogFlags::ALL, util::log::Level::Info, __VA_ARGS__)
Line
Count
Source
119
0
#define detail_LogWithSrcLoc(category, level, ...) util::log::LogPrintFormatInternal(SourceLocation{__func__}, category, level, __VA_ARGS__)
3251
0
    {
3252
0
        CCoinsViewCache temp_cache(temp_db.get());
3253
0
        std::unique_ptr<CCoinsViewCursor> cursor;
3254
0
        {
3255
0
            LOCK(::cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
3256
0
            tip = chainstate.m_chain.Tip();
3257
0
            chainstate.ForceFlushStateToDisk(/*wipe_cache=*/false);
3258
0
            cursor = chainstate.CoinsDB().Cursor();
3259
0
        }
3260
0
        temp_cache.SetBestBlock(tip->GetBlockHash());
3261
3262
0
        size_t coins_count = 0;
3263
0
        while (cursor->Valid()) {
3264
0
            node.rpc_interruption_point();
3265
3266
0
            COutPoint key;
3267
0
            Coin coin;
3268
0
            if (cursor->GetKey(key) && cursor->GetValue(coin)) {
3269
0
                temp_cache.AddCoin(key, std::move(coin), false);
3270
0
                coins_count++;
3271
3272
                // Log every 10M coins (optimized for mainnet)
3273
0
                if (coins_count % 10'000'000 == 0) {
3274
0
                    LogInfo("Copying UTXO set: %uM coins copied.", coins_count / 1'000'000);
Line
Count
Source
125
0
#define LogInfo(...) detail_LogWithSrcLoc(BCLog::LogFlags::ALL, util::log::Level::Info, __VA_ARGS__)
Line
Count
Source
119
0
#define detail_LogWithSrcLoc(category, level, ...) util::log::LogPrintFormatInternal(SourceLocation{__func__}, category, level, __VA_ARGS__)
3275
0
                }
3276
3277
                // Flush periodically
3278
0
                if (coins_count % 100'000 == 0) {
3279
0
                    temp_cache.Flush();
3280
0
                }
3281
0
            }
3282
0
            cursor->Next();
3283
0
        }
3284
3285
0
        temp_cache.Flush();
3286
0
        LogInfo("UTXO set copy complete: %u coins total", coins_count);
Line
Count
Source
125
0
#define LogInfo(...) detail_LogWithSrcLoc(BCLog::LogFlags::ALL, util::log::Level::Info, __VA_ARGS__)
Line
Count
Source
119
0
#define detail_LogWithSrcLoc(category, level, ...) util::log::LogPrintFormatInternal(SourceLocation{__func__}, category, level, __VA_ARGS__)
3287
0
    }
3288
3289
0
    LogInfo("Rolling back from height %d to %d", tip->nHeight, target->nHeight);
Line
Count
Source
125
0
#define LogInfo(...) detail_LogWithSrcLoc(BCLog::LogFlags::ALL, util::log::Level::Info, __VA_ARGS__)
Line
Count
Source
119
0
#define detail_LogWithSrcLoc(category, level, ...) util::log::LogPrintFormatInternal(SourceLocation{__func__}, category, level, __VA_ARGS__)
3290
3291
0
    const CBlockIndex* block_index{tip};
3292
0
    const size_t total_blocks{static_cast<size_t>(block_index->nHeight - target->nHeight)};
3293
0
    CCoinsViewCache rollback_cache(temp_db.get());
3294
0
    rollback_cache.SetBestBlock(block_index->GetBlockHash());
3295
0
    size_t blocks_processed = 0;
3296
0
    int last_progress{0};
3297
0
    DisconnectResult res;
3298
3299
0
    while (block_index->nHeight > target->nHeight) {
3300
0
        node.rpc_interruption_point();
3301
3302
0
        CBlock block;
3303
0
        if (!node.chainman->m_blockman.ReadBlock(block, *block_index)) {
3304
0
            throw JSONRPCError(RPC_INTERNAL_ERROR,
3305
0
                strprintf("Failed to read block at height %d", block_index->nHeight));
Line
Count
Source
1172
0
#define strprintf tfm::format
3306
0
        }
3307
3308
0
        WITH_LOCK(::cs_main, res = chainstate.DisconnectBlock(block, block_index, rollback_cache));
Line
Count
Source
299
0
#define WITH_LOCK(cs, code) (MaybeCheckNotHeld(cs), [&]() -> decltype(auto) { LOCK(cs); code; }())
3309
0
        if (res == DISCONNECT_FAILED) {
3310
0
            throw JSONRPCError(RPC_INTERNAL_ERROR,
3311
0
                strprintf("Failed to roll back block at height %d", block_index->nHeight));
Line
Count
Source
1172
0
#define strprintf tfm::format
3312
0
        }
3313
3314
0
        blocks_processed++;
3315
0
        int progress{static_cast<int>(blocks_processed * 100 / total_blocks)};
3316
0
        if (progress >= last_progress + 5) {
3317
0
            LogInfo("Rolled back %d%% of blocks.", progress);
Line
Count
Source
125
0
#define LogInfo(...) detail_LogWithSrcLoc(BCLog::LogFlags::ALL, util::log::Level::Info, __VA_ARGS__)
Line
Count
Source
119
0
#define detail_LogWithSrcLoc(category, level, ...) util::log::LogPrintFormatInternal(SourceLocation{__func__}, category, level, __VA_ARGS__)
3318
0
            last_progress = progress;
3319
0
            rollback_cache.Flush();
3320
0
        }
3321
3322
0
        block_index = block_index->pprev;
3323
0
    }
3324
3325
0
    CHECK_NONFATAL(rollback_cache.GetBestBlock() == target->GetBlockHash());
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
3326
0
    rollback_cache.Flush();
3327
3328
0
    LogInfo("Rollback complete. Computing UTXO statistics for created txoutset dump.");
Line
Count
Source
125
0
#define LogInfo(...) detail_LogWithSrcLoc(BCLog::LogFlags::ALL, util::log::Level::Info, __VA_ARGS__)
Line
Count
Source
119
0
#define detail_LogWithSrcLoc(category, level, ...) util::log::LogPrintFormatInternal(SourceLocation{__func__}, category, level, __VA_ARGS__)
3329
0
    std::optional<CCoinsStats> maybe_stats = GetUTXOStats(temp_db.get(),
3330
0
                                                          chainstate.m_blockman,
3331
0
                                                          CoinStatsHashType::HASH_SERIALIZED,
3332
0
                                                          node.rpc_interruption_point);
3333
3334
0
    if (!maybe_stats) {
3335
0
        throw JSONRPCError(RPC_INTERNAL_ERROR, "Unable to compute UTXO statistics");
3336
0
    }
3337
3338
0
    std::unique_ptr<CCoinsViewCursor> pcursor{temp_db->Cursor()};
3339
0
    if (!pcursor) {
3340
0
        throw JSONRPCError(RPC_INTERNAL_ERROR, "Unable to create UTXO cursor");
3341
0
    }
3342
3343
0
    LogInfo("Writing snapshot to disk.");
Line
Count
Source
125
0
#define LogInfo(...) detail_LogWithSrcLoc(BCLog::LogFlags::ALL, util::log::Level::Info, __VA_ARGS__)
Line
Count
Source
119
0
#define detail_LogWithSrcLoc(category, level, ...) util::log::LogPrintFormatInternal(SourceLocation{__func__}, category, level, __VA_ARGS__)
3344
0
    return WriteUTXOSnapshot(chainstate,
3345
0
                             pcursor.get(),
3346
0
                             &(*maybe_stats),
3347
0
                             target,
3348
0
                             std::move(afile),
3349
0
                             path,
3350
0
                             tmppath,
3351
0
                             node.rpc_interruption_point);
3352
0
}
3353
3354
std::tuple<std::unique_ptr<CCoinsViewCursor>, CCoinsStats, const CBlockIndex*>
3355
PrepareUTXOSnapshot(
3356
    Chainstate& chainstate,
3357
    const std::function<void()>& interruption_point)
3358
0
{
3359
0
    std::unique_ptr<CCoinsViewCursor> pcursor;
3360
0
    std::optional<CCoinsStats> maybe_stats;
3361
0
    const CBlockIndex* tip;
3362
3363
0
    {
3364
        // We need to lock cs_main to ensure that the coinsdb isn't written to
3365
        // between (i) flushing coins cache to disk (coinsdb), (ii) getting stats
3366
        // based upon the coinsdb, and (iii) constructing a cursor to the
3367
        // coinsdb for use in WriteUTXOSnapshot.
3368
        //
3369
        // Cursors returned by leveldb iterate over snapshots, so the contents
3370
        // of the pcursor will not be affected by simultaneous writes during
3371
        // use below this block.
3372
        //
3373
        // See discussion here:
3374
        //   https://github.com/bitcoin/bitcoin/pull/15606#discussion_r274479369
3375
        //
3376
0
        AssertLockHeld(::cs_main);
Line
Count
Source
144
0
#define AssertLockHeld(cs) AssertLockHeldInternal(#cs, __FILE__, __LINE__, &cs)
3377
3378
0
        chainstate.ForceFlushStateToDisk(/*wipe_cache=*/false);
3379
3380
0
        maybe_stats = GetUTXOStats(&chainstate.CoinsDB(), chainstate.m_blockman, CoinStatsHashType::HASH_SERIALIZED, interruption_point);
3381
0
        if (!maybe_stats) {
3382
0
            throw JSONRPCError(RPC_INTERNAL_ERROR, "Unable to read UTXO set");
3383
0
        }
3384
3385
0
        pcursor = chainstate.CoinsDB().Cursor();
3386
0
        tip = CHECK_NONFATAL(chainstate.m_blockman.LookupBlockIndex(maybe_stats->hashBlock));
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
3387
0
    }
3388
3389
0
    return {std::move(pcursor), *CHECK_NONFATAL(maybe_stats), tip};
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
3390
0
}
3391
3392
UniValue WriteUTXOSnapshot(
3393
    Chainstate& chainstate,
3394
    CCoinsViewCursor* pcursor,
3395
    CCoinsStats* maybe_stats,
3396
    const CBlockIndex* tip,
3397
    AutoFile&& afile,
3398
    const fs::path& path,
3399
    const fs::path& temppath,
3400
    const std::function<void()>& interruption_point)
3401
0
{
3402
0
    LOG_TIME_SECONDS(strprintf("writing UTXO snapshot at height %s (%s) to file %s (via %s)",
Line
Count
Source
108
0
    BCLog::Timer<std::chrono::seconds> UNIQUE_NAME(logging_timer)(__func__, end_msg)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
3403
0
        tip->nHeight, tip->GetBlockHash().ToString(),
3404
0
        fs::PathToString(path), fs::PathToString(temppath)));
3405
3406
0
    SnapshotMetadata metadata{chainstate.m_chainman.GetParams().MessageStart(), tip->GetBlockHash(), maybe_stats->coins_count};
3407
3408
0
    afile << metadata;
3409
3410
0
    COutPoint key;
3411
0
    Txid last_hash;
3412
0
    Coin coin;
3413
0
    unsigned int iter{0};
3414
0
    size_t written_coins_count{0};
3415
0
    std::vector<std::pair<uint32_t, Coin>> coins;
3416
3417
    // To reduce space the serialization format of the snapshot avoids
3418
    // duplication of tx hashes. The code takes advantage of the guarantee by
3419
    // leveldb that keys are lexicographically sorted.
3420
    // In the coins vector we collect all coins that belong to a certain tx hash
3421
    // (key.hash) and when we have them all (key.hash != last_hash) we write
3422
    // them to file using the below lambda function.
3423
    // See also https://github.com/bitcoin/bitcoin/issues/25675
3424
0
    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) {
3425
0
        afile << last_hash;
3426
0
        WriteCompactSize(afile, coins.size());
3427
0
        for (const auto& [n, coin] : coins) {
3428
0
            WriteCompactSize(afile, n);
3429
0
            afile << coin;
3430
0
            ++written_coins_count;
3431
0
        }
3432
0
    };
3433
3434
0
    pcursor->GetKey(key);
3435
0
    last_hash = key.hash;
3436
0
    while (pcursor->Valid()) {
3437
0
        if (iter % 5000 == 0) interruption_point();
3438
0
        ++iter;
3439
0
        if (pcursor->GetKey(key) && pcursor->GetValue(coin)) {
3440
0
            if (key.hash != last_hash) {
3441
0
                write_coins_to_file(afile, last_hash, coins, written_coins_count);
3442
0
                last_hash = key.hash;
3443
0
                coins.clear();
3444
0
            }
3445
0
            coins.emplace_back(key.n, coin);
3446
0
        }
3447
0
        pcursor->Next();
3448
0
    }
3449
3450
0
    if (!coins.empty()) {
3451
0
        write_coins_to_file(afile, last_hash, coins, written_coins_count);
3452
0
    }
3453
3454
0
    CHECK_NONFATAL(written_coins_count == maybe_stats->coins_count);
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
3455
3456
0
    if (afile.fclose() != 0) {
3457
0
        throw std::ios_base::failure(
3458
0
            strprintf("Error closing %s: %s", fs::PathToString(temppath), SysErrorString(errno)));
Line
Count
Source
1172
0
#define strprintf tfm::format
3459
0
    }
3460
3461
0
    UniValue result(UniValue::VOBJ);
3462
0
    result.pushKV("coins_written", written_coins_count);
3463
0
    result.pushKV("base_hash", tip->GetBlockHash().ToString());
3464
0
    result.pushKV("base_height", tip->nHeight);
3465
0
    result.pushKV("path", path.utf8string());
3466
0
    result.pushKV("txoutset_hash", maybe_stats->hashSerialized.ToString());
3467
0
    result.pushKV("nchaintx", tip->m_chain_tx_count);
3468
0
    return result;
3469
0
}
3470
3471
UniValue CreateUTXOSnapshot(
3472
    node::NodeContext& node,
3473
    Chainstate& chainstate,
3474
    AutoFile&& afile,
3475
    const fs::path& path,
3476
    const fs::path& tmppath)
3477
0
{
3478
0
    auto [cursor, stats, tip]{WITH_LOCK(::cs_main, return PrepareUTXOSnapshot(chainstate, node.rpc_interruption_point))};
Line
Count
Source
299
0
#define WITH_LOCK(cs, code) (MaybeCheckNotHeld(cs), [&]() -> decltype(auto) { LOCK(cs); code; }())
3479
0
    return WriteUTXOSnapshot(chainstate,
3480
0
                             cursor.get(),
3481
0
                             &stats,
3482
0
                             tip,
3483
0
                             std::move(afile),
3484
0
                             path,
3485
0
                             tmppath,
3486
0
                             node.rpc_interruption_point);
3487
0
}
3488
3489
static RPCMethod loadtxoutset()
3490
0
{
3491
0
    return RPCMethod{
3492
0
        "loadtxoutset",
3493
0
        "Load the serialized UTXO set from a file.\n"
3494
0
        "Once this snapshot is loaded, its contents will be "
3495
0
        "deserialized into a second chainstate data structure, which is then used to sync to "
3496
0
        "the network's tip. "
3497
0
        "Meanwhile, the original chainstate will complete the initial block download process in "
3498
0
        "the background, eventually validating up to the block that the snapshot is based upon.\n\n"
3499
3500
0
        "The result is a usable bitcoind instance that is current with the network tip in a "
3501
0
        "matter of minutes rather than hours. UTXO snapshot are typically obtained from "
3502
0
        "third-party sources (HTTP, torrent, etc.) which is reasonable since their "
3503
0
        "contents are always checked by hash.\n\n"
3504
3505
0
        "You can find more information on this process in the `assumeutxo` design "
3506
0
        "document (<https://github.com/bitcoin/bitcoin/blob/master/doc/design/assumeutxo.md>).",
3507
0
        {
3508
0
            {"path",
3509
0
                RPCArg::Type::STR,
3510
0
                RPCArg::Optional::NO,
3511
0
                "path to the snapshot file. If relative, will be prefixed by datadir."},
3512
0
        },
3513
0
        RPCResult{
3514
0
            RPCResult::Type::OBJ, "", "",
3515
0
                {
3516
0
                    {RPCResult::Type::NUM, "coins_loaded", "the number of coins loaded from the snapshot"},
3517
0
                    {RPCResult::Type::STR_HEX, "tip_hash", "the hash of the base of the snapshot"},
3518
0
                    {RPCResult::Type::NUM, "base_height", "the height of the base of the snapshot"},
3519
0
                    {RPCResult::Type::STR, "path", "the absolute path that the snapshot was loaded from"},
3520
0
                }
3521
0
        },
3522
0
        RPCExamples{
3523
0
            HelpExampleCli("-rpcclienttimeout=0 loadtxoutset", "utxo.dat")
3524
0
        },
3525
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
3526
0
{
3527
0
    NodeContext& node = EnsureAnyNodeContext(request.context);
3528
0
    ChainstateManager& chainman = EnsureChainman(node);
3529
0
    const fs::path path{AbsPathForConfigVal(EnsureArgsman(node), fs::u8path(self.Arg<std::string_view>("path")))};
3530
3531
0
    FILE* file{fsbridge::fopen(path, "rb")};
3532
0
    AutoFile afile{file};
3533
0
    if (afile.IsNull()) {
3534
0
        throw JSONRPCError(
3535
0
            RPC_INVALID_PARAMETER,
3536
0
            "Couldn't open file " + path.utf8string() + " for reading.");
3537
0
    }
3538
3539
0
    SnapshotMetadata metadata{chainman.GetParams().MessageStart()};
3540
0
    try {
3541
0
        afile >> metadata;
3542
0
    } catch (const std::ios_base::failure& e) {
3543
0
        throw JSONRPCError(RPC_DESERIALIZATION_ERROR, strprintf("Unable to parse metadata: %s", e.what()));
Line
Count
Source
1172
0
#define strprintf tfm::format
3544
0
    }
3545
3546
0
    auto activation_result{chainman.ActivateSnapshot(afile, metadata, false)};
3547
0
    if (!activation_result) {
3548
0
        throw JSONRPCError(RPC_INTERNAL_ERROR, strprintf("Unable to load UTXO snapshot: %s. (%s)", util::ErrorString(activation_result).original, path.utf8string()));
Line
Count
Source
1172
0
#define strprintf tfm::format
3549
0
    }
3550
3551
    // Because we can't provide historical blocks during tip or background sync.
3552
    // Update local services to reflect we are a limited peer until we are fully sync.
3553
0
    node.connman->RemoveLocalServices(NODE_NETWORK);
3554
    // Setting the limited state is usually redundant because the node can always
3555
    // provide the last 288 blocks, but it doesn't hurt to set it.
3556
0
    node.connman->AddLocalServices(NODE_NETWORK_LIMITED);
3557
3558
0
    CBlockIndex& snapshot_index{*CHECK_NONFATAL(*activation_result)};
Line
Count
Source
113
0
    inline_check_non_fatal(condition, std::source_location::current(), #condition)
3559
3560
0
    UniValue result(UniValue::VOBJ);
3561
0
    result.pushKV("coins_loaded", metadata.m_coins_count);
3562
0
    result.pushKV("tip_hash", snapshot_index.GetBlockHash().ToString());
3563
0
    result.pushKV("base_height", snapshot_index.nHeight);
3564
0
    result.pushKV("path", fs::PathToString(path));
3565
0
    return result;
3566
0
},
3567
0
    };
3568
0
}
3569
3570
const std::vector<RPCResult> RPCHelpForChainstate{
3571
    {RPCResult::Type::NUM, "blocks", "number of blocks in this chainstate"},
3572
    {RPCResult::Type::STR_HEX, "bestblockhash", "blockhash of the tip"},
3573
    {RPCResult::Type::STR_HEX, "bits", "nBits: compact representation of the block difficulty target"},
3574
    {RPCResult::Type::STR_HEX, "target", "The difficulty target"},
3575
    {RPCResult::Type::NUM, "difficulty", "difficulty of the tip"},
3576
    {RPCResult::Type::NUM, "verificationprogress", "progress towards the network tip"},
3577
    {RPCResult::Type::STR_HEX, "snapshot_blockhash", /*optional=*/true, "the base block of the snapshot this chainstate is based on, if any"},
3578
    {RPCResult::Type::NUM, "coins_db_cache_bytes", "size of the coinsdb cache"},
3579
    {RPCResult::Type::NUM, "coins_tip_cache_bytes", "size of the coinstip cache"},
3580
    {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."},
3581
};
3582
3583
static RPCMethod getchainstates()
3584
0
{
3585
0
return RPCMethod{
3586
0
        "getchainstates",
3587
0
        "Return information about chainstates.\n",
3588
0
        {},
3589
0
        RPCResult{
3590
0
            RPCResult::Type::OBJ, "", "", {
3591
0
                {RPCResult::Type::NUM, "headers", "the number of headers seen so far"},
3592
0
                {RPCResult::Type::ARR, "chainstates", "list of the chainstates ordered by work, with the most-work (active) chainstate last", {{RPCResult::Type::OBJ, "", "", RPCHelpForChainstate},}},
3593
0
            }
3594
0
        },
3595
0
        RPCExamples{
3596
0
            HelpExampleCli("getchainstates", "")
3597
0
    + HelpExampleRpc("getchainstates", "")
3598
0
        },
3599
0
        [](const RPCMethod& self, const JSONRPCRequest& request) -> UniValue
3600
0
{
3601
0
    LOCK(cs_main);
Line
Count
Source
268
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
3602
0
    UniValue obj(UniValue::VOBJ);
3603
3604
0
    ChainstateManager& chainman = EnsureAnyChainman(request.context);
3605
3606
0
    auto make_chain_data = [&](const Chainstate& cs) EXCLUSIVE_LOCKS_REQUIRED(::cs_main) {
3607
0
        AssertLockHeld(::cs_main);
Line
Count
Source
144
0
#define AssertLockHeld(cs) AssertLockHeldInternal(#cs, __FILE__, __LINE__, &cs)
3608
0
        UniValue data(UniValue::VOBJ);
3609
0
        if (!cs.m_chain.Tip()) {
3610
0
            return data;
3611
0
        }
3612
0
        const CChain& chain = cs.m_chain;
3613
0
        const CBlockIndex* tip = chain.Tip();
3614
3615
0
        data.pushKV("blocks", chain.Height());
3616
0
        data.pushKV("bestblockhash",         tip->GetBlockHash().GetHex());
3617
0
        data.pushKV("bits", strprintf("%08x", tip->nBits));
Line
Count
Source
1172
0
#define strprintf tfm::format
3618
0
        data.pushKV("target", GetTarget(*tip, chainman.GetConsensus().powLimit).GetHex());
3619
0
        data.pushKV("difficulty", GetDifficulty(*tip));
3620
0
        data.pushKV("verificationprogress", chainman.GuessVerificationProgress(tip));
3621
0
        data.pushKV("coins_db_cache_bytes",  cs.m_coinsdb_cache_size_bytes);
3622
0
        data.pushKV("coins_tip_cache_bytes", cs.m_coinstip_cache_size_bytes);
3623
0
        if (cs.m_from_snapshot_blockhash) {
3624
0
            data.pushKV("snapshot_blockhash", cs.m_from_snapshot_blockhash->ToString());
3625
0
        }
3626
0
        data.pushKV("validated", cs.m_assumeutxo == Assumeutxo::VALIDATED);
3627
0
        return data;
3628
0
    };
3629
3630
0
    obj.pushKV("headers", chainman.m_best_header ? chainman.m_best_header->nHeight : -1);
3631
0
    UniValue obj_chainstates{UniValue::VARR};
3632
0
    if (const Chainstate * cs{chainman.HistoricalChainstate()}) {
3633
0
        obj_chainstates.push_back(make_chain_data(*cs));
3634
0
    }
3635
0
    obj_chainstates.push_back(make_chain_data(chainman.CurrentChainstate()));
3636
0
    obj.pushKV("chainstates", std::move(obj_chainstates));
3637
0
    return obj;
3638
0
}
3639
0
    };
3640
0
}
3641
3642
3643
void RegisterBlockchainRPCCommands(CRPCTable& t)
3644
0
{
3645
0
    static const CRPCCommand commands[]{
3646
0
        {"blockchain", &getblockchaininfo},
3647
0
        {"blockchain", &getchaintxstats},
3648
0
        {"blockchain", &getblockstats},
3649
0
        {"blockchain", &getbestblockhash},
3650
0
        {"blockchain", &getblockcount},
3651
0
        {"blockchain", &getblock},
3652
0
        {"blockchain", &getblockfrompeer},
3653
0
        {"blockchain", &getblockhash},
3654
0
        {"blockchain", &getblockheader},
3655
0
        {"blockchain", &getchaintips},
3656
0
        {"blockchain", &getdifficulty},
3657
0
        {"blockchain", &getdeploymentinfo},
3658
0
        {"blockchain", &gettxout},
3659
0
        {"blockchain", &gettxoutsetinfo},
3660
0
        {"blockchain", &pruneblockchain},
3661
0
        {"blockchain", &verifychain},
3662
0
        {"blockchain", &preciousblock},
3663
0
        {"blockchain", &scantxoutset},
3664
0
        {"blockchain", &scanblocks},
3665
0
        {"blockchain", &getdescriptoractivity},
3666
0
        {"blockchain", &getblockfilter},
3667
0
        {"blockchain", &dumptxoutset},
3668
0
        {"blockchain", &loadtxoutset},
3669
0
        {"blockchain", &getchainstates},
3670
0
        {"hidden", &invalidateblock},
3671
0
        {"hidden", &reconsiderblock},
3672
0
        {"blockchain", &waitfornewblock},
3673
0
        {"blockchain", &waitforblock},
3674
0
        {"blockchain", &waitforblockheight},
3675
0
        {"hidden", &syncwithvalidationinterfacequeue},
3676
0
    };
3677
0
    for (const auto& c : commands) {
3678
0
        t.appendCommand(c.name, &c);
3679
0
    }
3680
0
}