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