LCOV - code coverage report
Current view: top level - src/rpc - txoutproof.cpp (source / functions) Coverage Total Hit
Test: total_coverage.info Lines: 97.2 % 107 104
Test Date: 2025-10-25 05:06:34 Functions: 100.0 % 5 5
Branches: 54.6 % 282 154

             Branch data     Line data    Source code
       1                 :             : // Copyright (c) 2010 Satoshi Nakamoto
       2                 :             : // Copyright (c) 2009-2022 The Bitcoin Core developers
       3                 :             : // Distributed under the MIT software license, see the accompanying
       4                 :             : // file COPYING or http://www.opensource.org/licenses/mit-license.php.
       5                 :             : 
       6                 :             : #include <chain.h>
       7                 :             : #include <chainparams.h>
       8                 :             : #include <coins.h>
       9                 :             : #include <index/txindex.h>
      10                 :             : #include <merkleblock.h>
      11                 :             : #include <node/blockstorage.h>
      12                 :             : #include <primitives/transaction.h>
      13                 :             : #include <rpc/blockchain.h>
      14                 :             : #include <rpc/server.h>
      15                 :             : #include <rpc/server_util.h>
      16                 :             : #include <rpc/util.h>
      17                 :             : #include <univalue.h>
      18                 :             : #include <util/strencodings.h>
      19                 :             : #include <validation.h>
      20                 :             : 
      21                 :             : using node::GetTransaction;
      22                 :             : 
      23                 :        2339 : static RPCHelpMan gettxoutproof()
      24                 :             : {
      25                 :        2339 :     return RPCHelpMan{
      26                 :        2339 :         "gettxoutproof",
      27         [ +  - ]:        4678 :         "Returns a hex-encoded proof that \"txid\" was included in a block.\n"
      28                 :             :         "\nNOTE: By default this function only works sometimes. This is when there is an\n"
      29                 :             :         "unspent output in the utxo for this transaction. To make it always work,\n"
      30                 :             :         "you need to maintain a transaction index, using the -txindex command line option or\n"
      31                 :             :         "specify the block in which the transaction is included manually (by blockhash).\n",
      32                 :             :         {
      33   [ +  -  +  - ]:        4678 :             {"txids", RPCArg::Type::ARR, RPCArg::Optional::NO, "The txids to filter",
      34                 :             :                 {
      35   [ +  -  +  - ]:        4678 :                     {"txid", RPCArg::Type::STR_HEX, RPCArg::Optional::OMITTED, "A transaction hash"},
      36                 :             :                 },
      37                 :             :             },
      38   [ +  -  +  - ]:        4678 :             {"blockhash", RPCArg::Type::STR_HEX, RPCArg::Optional::OMITTED, "If specified, looks for txid in the block with this hash"},
      39                 :             :         },
      40         [ +  - ]:        4678 :         RPCResult{
      41   [ +  -  +  - ]:        4678 :             RPCResult::Type::STR, "data", "A string that is a serialized, hex-encoded data for the proof."
      42         [ +  - ]:        4678 :         },
      43   [ +  -  +  - ]:        7017 :         RPCExamples{""},
      44                 :        2339 :         [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
      45                 :             :         {
      46         [ +  - ]:          23 :             std::set<Txid> setTxids;
      47   [ +  -  +  -  :          23 :             UniValue txids = request.params[0].get_array();
                   +  - ]
      48   [ -  +  +  + ]:          23 :             if (txids.empty()) {
      49   [ +  -  +  - ]:           2 :                 throw JSONRPCError(RPC_INVALID_PARAMETER, "Parameter 'txids' cannot be empty");
      50                 :             :             }
      51   [ -  +  +  + ]:          48 :             for (unsigned int idx = 0; idx < txids.size(); idx++) {
      52   [ +  -  +  +  :          29 :                 auto ret{setTxids.insert(Txid::FromUint256(ParseHashV(txids[idx], "txid")))};
                   +  - ]
      53         [ +  + ]:          27 :                 if (!ret.second) {
      54   [ +  -  +  -  :           2 :                     throw JSONRPCError(RPC_INVALID_PARAMETER, std::string("Invalid parameter, duplicated txid: ") + txids[idx].get_str());
          +  -  +  -  +  
                      - ]
      55                 :             :                 }
      56                 :             :             }
      57                 :             : 
      58                 :          19 :             const CBlockIndex* pblockindex = nullptr;
      59                 :          19 :             uint256 hashBlock;
      60         [ +  - ]:          19 :             ChainstateManager& chainman = EnsureAnyChainman(request.context);
      61   [ +  -  +  + ]:          19 :             if (!request.params[1].isNull()) {
      62         [ +  - ]:           6 :                 LOCK(cs_main);
      63   [ +  -  +  + ]:           6 :                 hashBlock = ParseHashV(request.params[1], "blockhash");
      64         [ +  - ]:           4 :                 pblockindex = chainman.m_blockman.LookupBlockIndex(hashBlock);
      65         [ +  + ]:           4 :                 if (!pblockindex) {
      66   [ +  -  +  - ]:           2 :                     throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Block not found");
      67                 :             :                 }
      68                 :           6 :             } else {
      69         [ +  - ]:          13 :                 LOCK(cs_main);
      70         [ +  - ]:          13 :                 Chainstate& active_chainstate = chainman.ActiveChainstate();
      71                 :             : 
      72                 :             :                 // Loop through txids and try to find which block they're in. Exit loop once a block is found.
      73         [ +  + ]:          19 :                 for (const auto& tx : setTxids) {
      74   [ +  -  +  - ]:          16 :                     const Coin& coin{AccessByTxid(active_chainstate.CoinsTip(), tx)};
      75         [ +  + ]:          16 :                     if (!coin.IsSpent()) {
      76   [ -  +  +  - ]:          23 :                         pblockindex = active_chainstate.m_chain[coin.nHeight];
      77                 :             :                         break;
      78                 :             :                     }
      79                 :             :                 }
      80                 :          13 :             }
      81                 :             : 
      82                 :             : 
      83                 :             :             // Allow txindex to catch up if we need to query it and before we acquire cs_main.
      84   [ +  +  +  + ]:          16 :             if (g_txindex && !pblockindex) {
      85         [ +  - ]:           1 :                 g_txindex->BlockUntilSyncedToCurrentChain();
      86                 :             :             }
      87                 :             : 
      88         [ +  + ]:          16 :             if (pblockindex == nullptr) {
      89         [ +  - ]:           3 :                 const CTransactionRef tx = GetTransaction(/*block_index=*/nullptr, /*mempool=*/nullptr, *setTxids.begin(), hashBlock, chainman.m_blockman);
      90   [ +  +  +  - ]:           4 :                 if (!tx || hashBlock.IsNull()) {
      91   [ +  -  +  - ]:           4 :                     throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Transaction not yet in block");
      92                 :             :                 }
      93                 :             : 
      94         [ +  - ]:           1 :                 LOCK(cs_main);
      95         [ +  - ]:           1 :                 pblockindex = chainman.m_blockman.LookupBlockIndex(hashBlock);
      96         [ -  + ]:           1 :                 if (!pblockindex) {
      97   [ #  #  #  # ]:           0 :                     throw JSONRPCError(RPC_INTERNAL_ERROR, "Transaction index corrupt");
      98                 :             :                 }
      99         [ +  - ]:           4 :             }
     100                 :             : 
     101                 :          14 :             {
     102         [ +  - ]:          14 :                 LOCK(cs_main);
     103         [ +  + ]:          14 :                 CheckBlockDataAvailability(chainman.m_blockman, *pblockindex, /*check_for_undo=*/false);
     104                 :           1 :             }
     105                 :          13 :             CBlock block;
     106   [ +  -  -  + ]:          13 :             if (!chainman.m_blockman.ReadBlock(block, *pblockindex)) {
     107   [ #  #  #  # ]:           0 :                 throw JSONRPCError(RPC_INTERNAL_ERROR, "Can't read block from disk");
     108                 :             :             }
     109                 :             : 
     110                 :          13 :             unsigned int ntxFound = 0;
     111         [ +  + ]:          49 :             for (const auto& tx : block.vtx) {
     112         [ +  + ]:          36 :                 if (setTxids.count(tx->GetHash())) {
     113                 :          18 :                     ntxFound++;
     114                 :             :                 }
     115                 :             :             }
     116         [ +  + ]:          13 :             if (ntxFound != setTxids.size()) {
     117   [ +  -  +  - ]:           2 :                 throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Not all transactions found in specified or retrieved block");
     118                 :             :             }
     119                 :             : 
     120                 :          12 :             DataStream ssMB{};
     121         [ +  - ]:          12 :             CMerkleBlock mb(block, setTxids);
     122         [ +  - ]:          12 :             ssMB << mb;
     123   [ -  +  +  - ]:          12 :             std::string strHex = HexStr(ssMB);
     124         [ +  - ]:          12 :             return strHex;
     125                 :          23 :         },
     126   [ +  -  +  -  :       21051 :     };
          +  -  +  +  +  
             +  -  -  -  
                      - ]
     127   [ +  -  +  -  :       14034 : }
             +  -  -  - ]
     128                 :             : 
     129                 :        2329 : static RPCHelpMan verifytxoutproof()
     130                 :             : {
     131                 :        2329 :     return RPCHelpMan{
     132                 :        2329 :         "verifytxoutproof",
     133         [ +  - ]:        4658 :         "Verifies that a proof points to a transaction in a block, returning the transaction it commits to\n"
     134                 :             :         "and throwing an RPC error if the block is not in our best chain\n",
     135                 :             :         {
     136   [ +  -  +  - ]:        4658 :             {"proof", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "The hex-encoded proof generated by gettxoutproof"},
     137                 :             :         },
     138         [ +  - ]:        4658 :         RPCResult{
     139   [ +  -  +  - ]:        4658 :             RPCResult::Type::ARR, "", "",
     140                 :             :             {
     141   [ +  -  +  - ]:        4658 :                 {RPCResult::Type::STR_HEX, "txid", "The txid(s) which the proof commits to, or empty array if the proof cannot be validated."},
     142                 :             :             }
     143   [ +  -  +  -  :        6987 :         },
             +  +  -  - ]
     144   [ +  -  +  - ]:        6987 :         RPCExamples{""},
     145                 :        2329 :         [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
     146                 :             :         {
     147   [ -  +  +  - ]:          13 :             DataStream ssMB{ParseHexV(request.params[0], "proof")};
     148         [ +  - ]:          13 :             CMerkleBlock merkleBlock;
     149         [ +  - ]:          13 :             ssMB >> merkleBlock;
     150                 :             : 
     151                 :          13 :             UniValue res(UniValue::VARR);
     152                 :             : 
     153                 :          13 :             std::vector<Txid> vMatch;
     154                 :          13 :             std::vector<unsigned int> vIndex;
     155   [ +  -  +  - ]:          13 :             if (merkleBlock.txn.ExtractMatches(vMatch, vIndex) != merkleBlock.header.hashMerkleRoot)
     156                 :             :                 return res;
     157                 :             : 
     158         [ +  - ]:          13 :             ChainstateManager& chainman = EnsureAnyChainman(request.context);
     159         [ +  - ]:          13 :             LOCK(cs_main);
     160                 :             : 
     161   [ +  -  +  - ]:          13 :             const CBlockIndex* pindex = chainman.m_blockman.LookupBlockIndex(merkleBlock.header.GetHash());
     162   [ +  -  +  -  :          26 :             if (!pindex || !chainman.ActiveChain().Contains(pindex) || pindex->nTx == 0) {
                   -  + ]
     163   [ #  #  #  # ]:           0 :                 throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Block not found in chain");
     164                 :             :             }
     165                 :             : 
     166                 :             :             // Check if proof is valid, only add results if so
     167         [ +  + ]:          13 :             if (pindex->nTx == merkleBlock.txn.GetNumTransactions()) {
     168         [ +  + ]:          29 :                 for (const auto& txid : vMatch) {
     169   [ +  -  +  -  :          18 :                     res.push_back(txid.GetHex());
                   +  - ]
     170                 :             :                 }
     171                 :             :             }
     172                 :             : 
     173         [ +  - ]:          13 :             return res;
     174                 :          26 :         },
     175   [ +  -  +  -  :       11645 :     };
             +  +  -  - ]
     176   [ +  -  +  - ]:        6987 : }
     177                 :             : 
     178                 :        1285 : void RegisterTxoutProofRPCCommands(CRPCTable& t)
     179                 :             : {
     180                 :        1285 :     static const CRPCCommand commands[]{
     181         [ +  - ]:        2306 :         {"blockchain", &gettxoutproof},
     182         [ +  - ]:        2306 :         {"blockchain", &verifytxoutproof},
     183   [ +  +  +  -  :        3591 :     };
          +  -  +  -  -  
                      - ]
     184         [ +  + ]:        3855 :     for (const auto& c : commands) {
     185                 :        2570 :         t.appendCommand(c.name, &c);
     186                 :             :     }
     187                 :        1285 : }
        

Generated by: LCOV version 2.0-1