Bitcoin Core  31.0.0
P2P Digital Currency
rest.cpp
Go to the documentation of this file.
1 // Copyright (c) 2009-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 <rest.h>
7 
8 #include <blockfilter.h>
9 #include <chain.h>
10 #include <chainparams.h>
11 #include <core_io.h>
12 #include <flatfile.h>
13 #include <httpserver.h>
14 #include <index/blockfilterindex.h>
15 #include <index/txindex.h>
16 #include <node/blockstorage.h>
17 #include <node/context.h>
18 #include <primitives/block.h>
19 #include <primitives/transaction.h>
20 #include <rpc/blockchain.h>
21 #include <rpc/mempool.h>
22 #include <rpc/protocol.h>
23 #include <rpc/server.h>
24 #include <rpc/server_util.h>
25 #include <streams.h>
26 #include <sync.h>
27 #include <txmempool.h>
28 #include <undo.h>
29 #include <util/any.h>
30 #include <util/check.h>
31 #include <util/strencodings.h>
32 #include <validation.h>
33 
34 #include <any>
35 #include <vector>
36 
37 #include <univalue.h>
38 
40 using node::NodeContext;
41 using util::SplitString;
42 
43 static const size_t MAX_GETUTXOS_OUTPOINTS = 15; //allow a max of 15 outpoints to be queried at once
44 static constexpr unsigned int MAX_REST_HEADERS_RESULTS = 2000;
45 
46 static const struct {
48  const char* name;
49 } rf_names[] = {
52  {RESTResponseFormat::HEX, "hex"},
53  {RESTResponseFormat::JSON, "json"},
54 };
55 
56 struct CCoin {
57  uint32_t nHeight;
59 
60  CCoin() : nHeight(0) {}
61  explicit CCoin(Coin&& in) : nHeight(in.nHeight), out(std::move(in.out)) {}
62 
64  {
65  uint32_t nTxVerDummy = 0;
66  READWRITE(nTxVerDummy, obj.nHeight, obj.out);
67  }
68 };
69 
70 static bool RESTERR(HTTPRequest* req, enum HTTPStatusCode status, std::string message)
71 {
72  req->WriteHeader("Content-Type", "text/plain");
73  req->WriteReply(status, message + "\r\n");
74  return false;
75 }
76 
84 static NodeContext* GetNodeContext(const std::any& context, HTTPRequest* req)
85 {
86  auto node_context = util::AnyPtr<NodeContext>(context);
87  if (!node_context) {
88  RESTERR(req, HTTP_INTERNAL_SERVER_ERROR, STR_INTERNAL_BUG("Node context not found!"));
89  return nullptr;
90  }
91  return node_context;
92 }
93 
101 static CTxMemPool* GetMemPool(const std::any& context, HTTPRequest* req)
102 {
103  auto node_context = util::AnyPtr<NodeContext>(context);
104  if (!node_context || !node_context->mempool) {
105  RESTERR(req, HTTP_NOT_FOUND, "Mempool disabled or instance not found");
106  return nullptr;
107  }
108  return node_context->mempool.get();
109 }
110 
118 static ChainstateManager* GetChainman(const std::any& context, HTTPRequest* req)
119 {
120  auto node_context = util::AnyPtr<NodeContext>(context);
121  if (!node_context || !node_context->chainman) {
122  RESTERR(req, HTTP_INTERNAL_SERVER_ERROR, STR_INTERNAL_BUG("Chainman disabled or instance not found!"));
123  return nullptr;
124  }
125  return node_context->chainman.get();
126 }
127 
128 RESTResponseFormat ParseDataFormat(std::string& param, const std::string& strReq)
129 {
130  // Remove query string (if any, separated with '?') as it should not interfere with
131  // parsing param and data format
132  param = strReq.substr(0, strReq.rfind('?'));
133  const std::string::size_type pos_format{param.rfind('.')};
134 
135  // No format string is found
136  if (pos_format == std::string::npos) {
138  }
139 
140  // Match format string to available formats
141  const std::string suffix(param, pos_format + 1);
142  for (const auto& rf_name : rf_names) {
143  if (suffix == rf_name.name) {
144  param.erase(pos_format);
145  return rf_name.rf;
146  }
147  }
148 
149  // If no suffix is found, return RESTResponseFormat::UNDEF and original string without query string
151 }
152 
153 static std::string AvailableDataFormatsString()
154 {
155  std::string formats;
156  for (const auto& rf_name : rf_names) {
157  if (strlen(rf_name.name) > 0) {
158  formats.append(".");
159  formats.append(rf_name.name);
160  formats.append(", ");
161  }
162  }
163 
164  if (formats.length() > 0)
165  return formats.substr(0, formats.length() - 2);
166 
167  return formats;
168 }
169 
170 static bool CheckWarmup(HTTPRequest* req)
171 {
172  std::string statusmessage;
173  if (RPCIsInWarmup(&statusmessage))
174  return RESTERR(req, HTTP_SERVICE_UNAVAILABLE, "Service temporarily unavailable: " + statusmessage);
175  return true;
176 }
177 
178 static bool rest_headers(const std::any& context,
179  HTTPRequest* req,
180  const std::string& uri_part)
181 {
182  if (!CheckWarmup(req))
183  return false;
184  std::string param;
185  const RESTResponseFormat rf = ParseDataFormat(param, uri_part);
186  std::vector<std::string> path = SplitString(param, '/');
187 
188  std::string raw_count;
189  std::string hashStr;
190  if (path.size() == 2) {
191  // deprecated path: /rest/headers/<count>/<hash>
192  hashStr = path[1];
193  raw_count = path[0];
194  } else if (path.size() == 1) {
195  // new path with query parameter: /rest/headers/<hash>?count=<count>
196  hashStr = path[0];
197  try {
198  raw_count = req->GetQueryParameter("count").value_or("5");
199  } catch (const std::runtime_error& e) {
200  return RESTERR(req, HTTP_BAD_REQUEST, e.what());
201  }
202  } else {
203  return RESTERR(req, HTTP_BAD_REQUEST, "Invalid URI format. Expected /rest/headers/<hash>.<ext>?count=<count>");
204  }
205 
206  const auto parsed_count{ToIntegral<size_t>(raw_count)};
207  if (!parsed_count.has_value() || *parsed_count < 1 || *parsed_count > MAX_REST_HEADERS_RESULTS) {
208  return RESTERR(req, HTTP_BAD_REQUEST, strprintf("Header count is invalid or out of acceptable range (1-%u): %s", MAX_REST_HEADERS_RESULTS, raw_count));
209  }
210 
211  auto hash{uint256::FromHex(hashStr)};
212  if (!hash) {
213  return RESTERR(req, HTTP_BAD_REQUEST, "Invalid hash: " + hashStr);
214  }
215 
216  const CBlockIndex* tip = nullptr;
217  std::vector<const CBlockIndex*> headers;
218  headers.reserve(*parsed_count);
219  ChainstateManager* maybe_chainman = GetChainman(context, req);
220  if (!maybe_chainman) return false;
221  ChainstateManager& chainman = *maybe_chainman;
222  {
223  LOCK(cs_main);
224  CChain& active_chain = chainman.ActiveChain();
225  tip = active_chain.Tip();
226  const CBlockIndex* pindex{chainman.m_blockman.LookupBlockIndex(*hash)};
227  while (pindex != nullptr && active_chain.Contains(pindex)) {
228  headers.push_back(pindex);
229  if (headers.size() == *parsed_count) {
230  break;
231  }
232  pindex = active_chain.Next(pindex);
233  }
234  }
235 
236  switch (rf) {
238  DataStream ssHeader{};
239  for (const CBlockIndex *pindex : headers) {
240  ssHeader << pindex->GetBlockHeader();
241  }
242 
243  req->WriteHeader("Content-Type", "application/octet-stream");
244  req->WriteReply(HTTP_OK, ssHeader);
245  return true;
246  }
247 
249  DataStream ssHeader{};
250  for (const CBlockIndex *pindex : headers) {
251  ssHeader << pindex->GetBlockHeader();
252  }
253 
254  std::string strHex = HexStr(ssHeader) + "\n";
255  req->WriteHeader("Content-Type", "text/plain");
256  req->WriteReply(HTTP_OK, strHex);
257  return true;
258  }
260  UniValue jsonHeaders(UniValue::VARR);
261  for (const CBlockIndex *pindex : headers) {
262  jsonHeaders.push_back(blockheaderToJSON(*tip, *pindex, chainman.GetConsensus().powLimit));
263  }
264  std::string strJSON = jsonHeaders.write() + "\n";
265  req->WriteHeader("Content-Type", "application/json");
266  req->WriteReply(HTTP_OK, strJSON);
267  return true;
268  }
269  default: {
270  return RESTERR(req, HTTP_NOT_FOUND, "output format not found (available: " + AvailableDataFormatsString() + ")");
271  }
272  }
273 }
274 
278 static void SerializeBlockUndo(DataStream& stream, const CBlockUndo& block_undo)
279 {
280  WriteCompactSize(stream, block_undo.vtxundo.size() + 1);
281  WriteCompactSize(stream, 0); // block_undo.vtxundo doesn't contain coinbase tx
282  for (const CTxUndo& tx_undo : block_undo.vtxundo) {
283  WriteCompactSize(stream, tx_undo.vprevout.size());
284  for (const Coin& coin : tx_undo.vprevout) {
285  coin.out.Serialize(stream);
286  }
287  }
288 }
289 
293 static void BlockUndoToJSON(const CBlockUndo& block_undo, UniValue& result)
294 {
295  result.push_back({UniValue::VARR}); // block_undo.vtxundo doesn't contain coinbase tx
296  for (const CTxUndo& tx_undo : block_undo.vtxundo) {
297  UniValue tx_prevouts(UniValue::VARR);
298  for (const Coin& coin : tx_undo.vprevout) {
299  UniValue prevout(UniValue::VOBJ);
300  prevout.pushKV("value", ValueFromAmount(coin.out.nValue));
301 
302  UniValue script_pub_key(UniValue::VOBJ);
303  ScriptToUniv(coin.out.scriptPubKey, /*out=*/script_pub_key, /*include_hex=*/true, /*include_address=*/true);
304  prevout.pushKV("scriptPubKey", std::move(script_pub_key));
305 
306  tx_prevouts.push_back(std::move(prevout));
307  }
308  result.push_back(std::move(tx_prevouts));
309  }
310 }
311 
312 static bool rest_spent_txouts(const std::any& context, HTTPRequest* req, const std::string& uri_part)
313 {
314  if (!CheckWarmup(req)) {
315  return false;
316  }
317  std::string param;
318  const RESTResponseFormat rf = ParseDataFormat(param, uri_part);
319  std::vector<std::string> path = SplitString(param, '/');
320 
321  std::string hashStr;
322  if (path.size() == 1) {
323  // path with query parameter: /rest/spenttxouts/<hash>
324  hashStr = path[0];
325  } else {
326  return RESTERR(req, HTTP_BAD_REQUEST, "Invalid URI format. Expected /rest/spenttxouts/<hash>.<ext>");
327  }
328 
329  auto hash{uint256::FromHex(hashStr)};
330  if (!hash) {
331  return RESTERR(req, HTTP_BAD_REQUEST, "Invalid hash: " + hashStr);
332  }
333 
334  ChainstateManager* chainman = GetChainman(context, req);
335  if (!chainman) {
336  return false;
337  }
338 
339  const CBlockIndex* pblockindex = WITH_LOCK(cs_main, return chainman->m_blockman.LookupBlockIndex(*hash));
340  if (!pblockindex) {
341  return RESTERR(req, HTTP_NOT_FOUND, hashStr + " not found");
342  }
343 
344  CBlockUndo block_undo;
345  if (pblockindex->nHeight > 0 && !chainman->m_blockman.ReadBlockUndo(block_undo, *pblockindex)) {
346  return RESTERR(req, HTTP_NOT_FOUND, hashStr + " undo not available");
347  }
348 
349  switch (rf) {
351  DataStream ssSpentResponse{};
352  SerializeBlockUndo(ssSpentResponse, block_undo);
353  req->WriteHeader("Content-Type", "application/octet-stream");
354  req->WriteReply(HTTP_OK, ssSpentResponse);
355  return true;
356  }
357 
359  DataStream ssSpentResponse{};
360  SerializeBlockUndo(ssSpentResponse, block_undo);
361  const std::string strHex{HexStr(ssSpentResponse) + "\n"};
362  req->WriteHeader("Content-Type", "text/plain");
363  req->WriteReply(HTTP_OK, strHex);
364  return true;
365  }
366 
369  BlockUndoToJSON(block_undo, result);
370  std::string strJSON = result.write() + "\n";
371  req->WriteHeader("Content-Type", "application/json");
372  req->WriteReply(HTTP_OK, strJSON);
373  return true;
374  }
375 
376  default: {
377  return RESTERR(req, HTTP_NOT_FOUND, "output format not found (available: " + AvailableDataFormatsString() + ")");
378  }
379  }
380 }
381 
388 static bool rest_block(const std::any& context,
389  HTTPRequest* req,
390  const std::string& uri_part,
391  std::optional<TxVerbosity> tx_verbosity,
392  std::optional<std::pair<size_t, size_t>> block_part = std::nullopt)
393 {
394  if (!CheckWarmup(req))
395  return false;
396  std::string hashStr;
397  const RESTResponseFormat rf = ParseDataFormat(hashStr, uri_part);
398 
399  auto hash{uint256::FromHex(hashStr)};
400  if (!hash) {
401  return RESTERR(req, HTTP_BAD_REQUEST, "Invalid hash: " + hashStr);
402  }
403 
404  FlatFilePos pos{};
405  const CBlockIndex* pblockindex = nullptr;
406  const CBlockIndex* tip = nullptr;
407  ChainstateManager* maybe_chainman = GetChainman(context, req);
408  if (!maybe_chainman) return false;
409  ChainstateManager& chainman = *maybe_chainman;
410  {
411  LOCK(cs_main);
412  tip = chainman.ActiveChain().Tip();
413  pblockindex = chainman.m_blockman.LookupBlockIndex(*hash);
414  if (!pblockindex) {
415  return RESTERR(req, HTTP_NOT_FOUND, hashStr + " not found");
416  }
417  if (!(pblockindex->nStatus & BLOCK_HAVE_DATA)) {
418  if (chainman.m_blockman.IsBlockPruned(*pblockindex)) {
419  return RESTERR(req, HTTP_NOT_FOUND, hashStr + " not available (pruned data)");
420  }
421  return RESTERR(req, HTTP_NOT_FOUND, hashStr + " not available (not fully downloaded)");
422  }
423  pos = pblockindex->GetBlockPos();
424  }
425 
426  const auto block_data{chainman.m_blockman.ReadRawBlock(pos, block_part)};
427  if (!block_data) {
428  switch (block_data.error()) {
429  case node::ReadRawError::IO: return RESTERR(req, HTTP_INTERNAL_SERVER_ERROR, "I/O error reading " + hashStr);
431  assert(block_part);
432  return RESTERR(req, HTTP_BAD_REQUEST, strprintf("Bad block part offset/size %d/%d for %s", block_part->first, block_part->second, hashStr));
433  } // no default case, so the compiler can warn about missing cases
434  assert(false);
435  }
436 
437  switch (rf) {
439  req->WriteHeader("Content-Type", "application/octet-stream");
440  req->WriteReply(HTTP_OK, *block_data);
441  return true;
442  }
443 
445  const std::string strHex{HexStr(*block_data) + "\n"};
446  req->WriteHeader("Content-Type", "text/plain");
447  req->WriteReply(HTTP_OK, strHex);
448  return true;
449  }
450 
452  if (tx_verbosity) {
453  CBlock block{};
454  SpanReader{*block_data} >> TX_WITH_WITNESS(block);
455  UniValue objBlock = blockToJSON(chainman.m_blockman, block, *tip, *pblockindex, *tx_verbosity, chainman.GetConsensus().powLimit);
456  std::string strJSON = objBlock.write() + "\n";
457  req->WriteHeader("Content-Type", "application/json");
458  req->WriteReply(HTTP_OK, strJSON);
459  return true;
460  }
461  return RESTERR(req, HTTP_BAD_REQUEST, "JSON output is not supported for this request type");
462  }
463 
464  default: {
465  return RESTERR(req, HTTP_NOT_FOUND, "output format not found (available: " + AvailableDataFormatsString() + ")");
466  }
467  }
468 }
469 
470 static bool rest_block_extended(const std::any& context, HTTPRequest* req, const std::string& uri_part)
471 {
472  return rest_block(context, req, uri_part, TxVerbosity::SHOW_DETAILS_AND_PREVOUT);
473 }
474 
475 static bool rest_block_notxdetails(const std::any& context, HTTPRequest* req, const std::string& uri_part)
476 {
477  return rest_block(context, req, uri_part, TxVerbosity::SHOW_TXID);
478 }
479 
480 static bool rest_block_part(const std::any& context, HTTPRequest* req, const std::string& uri_part)
481 {
482  try {
483  if (const auto opt_offset{ToIntegral<size_t>(req->GetQueryParameter("offset").value_or(""))}) {
484  if (const auto opt_size{ToIntegral<size_t>(req->GetQueryParameter("size").value_or(""))}) {
485  return rest_block(context, req, uri_part,
486  /*tx_verbosity=*/std::nullopt,
487  /*block_part=*/{{*opt_offset, *opt_size}});
488  } else {
489  return RESTERR(req, HTTP_BAD_REQUEST, "Block part size missing or invalid");
490  }
491  } else {
492  return RESTERR(req, HTTP_BAD_REQUEST, "Block part offset missing or invalid");
493  }
494  } catch (const std::runtime_error& e) {
495  return RESTERR(req, HTTP_BAD_REQUEST, e.what());
496  }
497 }
498 
499 static bool rest_filter_header(const std::any& context, HTTPRequest* req, const std::string& uri_part)
500 {
501  if (!CheckWarmup(req)) return false;
502 
503  std::string param;
504  const RESTResponseFormat rf = ParseDataFormat(param, uri_part);
505 
506  std::vector<std::string> uri_parts = SplitString(param, '/');
507  std::string raw_count;
508  std::string raw_blockhash;
509  if (uri_parts.size() == 3) {
510  // deprecated path: /rest/blockfilterheaders/<filtertype>/<count>/<blockhash>
511  raw_blockhash = uri_parts[2];
512  raw_count = uri_parts[1];
513  } else if (uri_parts.size() == 2) {
514  // new path with query parameter: /rest/blockfilterheaders/<filtertype>/<blockhash>?count=<count>
515  raw_blockhash = uri_parts[1];
516  try {
517  raw_count = req->GetQueryParameter("count").value_or("5");
518  } catch (const std::runtime_error& e) {
519  return RESTERR(req, HTTP_BAD_REQUEST, e.what());
520  }
521  } else {
522  return RESTERR(req, HTTP_BAD_REQUEST, "Invalid URI format. Expected /rest/blockfilterheaders/<filtertype>/<blockhash>.<ext>?count=<count>");
523  }
524 
525  const auto parsed_count{ToIntegral<size_t>(raw_count)};
526  if (!parsed_count.has_value() || *parsed_count < 1 || *parsed_count > MAX_REST_HEADERS_RESULTS) {
527  return RESTERR(req, HTTP_BAD_REQUEST, strprintf("Header count is invalid or out of acceptable range (1-%u): %s", MAX_REST_HEADERS_RESULTS, raw_count));
528  }
529 
530  auto block_hash{uint256::FromHex(raw_blockhash)};
531  if (!block_hash) {
532  return RESTERR(req, HTTP_BAD_REQUEST, "Invalid hash: " + raw_blockhash);
533  }
534 
535  BlockFilterType filtertype;
536  if (!BlockFilterTypeByName(uri_parts[0], filtertype)) {
537  return RESTERR(req, HTTP_BAD_REQUEST, "Unknown filtertype " + uri_parts[0]);
538  }
539 
540  BlockFilterIndex* index = GetBlockFilterIndex(filtertype);
541  if (!index) {
542  return RESTERR(req, HTTP_BAD_REQUEST, "Index is not enabled for filtertype " + uri_parts[0]);
543  }
544 
545  std::vector<const CBlockIndex*> headers;
546  headers.reserve(*parsed_count);
547  {
548  ChainstateManager* maybe_chainman = GetChainman(context, req);
549  if (!maybe_chainman) return false;
550  ChainstateManager& chainman = *maybe_chainman;
551  LOCK(cs_main);
552  CChain& active_chain = chainman.ActiveChain();
553  const CBlockIndex* pindex{chainman.m_blockman.LookupBlockIndex(*block_hash)};
554  while (pindex != nullptr && active_chain.Contains(pindex)) {
555  headers.push_back(pindex);
556  if (headers.size() == *parsed_count)
557  break;
558  pindex = active_chain.Next(pindex);
559  }
560  }
561 
562  bool index_ready = index->BlockUntilSyncedToCurrentChain();
563 
564  std::vector<uint256> filter_headers;
565  filter_headers.reserve(*parsed_count);
566  for (const CBlockIndex* pindex : headers) {
567  uint256 filter_header;
568  if (!index->LookupFilterHeader(pindex, filter_header)) {
569  std::string errmsg = "Filter not found.";
570 
571  if (!index_ready) {
572  errmsg += " Block filters are still in the process of being indexed.";
573  } else {
574  errmsg += " This error is unexpected and indicates index corruption.";
575  }
576 
577  return RESTERR(req, HTTP_NOT_FOUND, errmsg);
578  }
579  filter_headers.push_back(filter_header);
580  }
581 
582  switch (rf) {
584  DataStream ssHeader{};
585  for (const uint256& header : filter_headers) {
586  ssHeader << header;
587  }
588 
589  req->WriteHeader("Content-Type", "application/octet-stream");
590  req->WriteReply(HTTP_OK, ssHeader);
591  return true;
592  }
594  DataStream ssHeader{};
595  for (const uint256& header : filter_headers) {
596  ssHeader << header;
597  }
598 
599  std::string strHex = HexStr(ssHeader) + "\n";
600  req->WriteHeader("Content-Type", "text/plain");
601  req->WriteReply(HTTP_OK, strHex);
602  return true;
603  }
605  UniValue jsonHeaders(UniValue::VARR);
606  for (const uint256& header : filter_headers) {
607  jsonHeaders.push_back(header.GetHex());
608  }
609 
610  std::string strJSON = jsonHeaders.write() + "\n";
611  req->WriteHeader("Content-Type", "application/json");
612  req->WriteReply(HTTP_OK, strJSON);
613  return true;
614  }
615  default: {
616  return RESTERR(req, HTTP_NOT_FOUND, "output format not found (available: " + AvailableDataFormatsString() + ")");
617  }
618  }
619 }
620 
621 static bool rest_block_filter(const std::any& context, HTTPRequest* req, const std::string& uri_part)
622 {
623  if (!CheckWarmup(req)) return false;
624 
625  std::string param;
626  const RESTResponseFormat rf = ParseDataFormat(param, uri_part);
627 
628  // request is sent over URI scheme /rest/blockfilter/filtertype/blockhash
629  std::vector<std::string> uri_parts = SplitString(param, '/');
630  if (uri_parts.size() != 2) {
631  return RESTERR(req, HTTP_BAD_REQUEST, "Invalid URI format. Expected /rest/blockfilter/<filtertype>/<blockhash>");
632  }
633 
634  auto block_hash{uint256::FromHex(uri_parts[1])};
635  if (!block_hash) {
636  return RESTERR(req, HTTP_BAD_REQUEST, "Invalid hash: " + uri_parts[1]);
637  }
638 
639  BlockFilterType filtertype;
640  if (!BlockFilterTypeByName(uri_parts[0], filtertype)) {
641  return RESTERR(req, HTTP_BAD_REQUEST, "Unknown filtertype " + uri_parts[0]);
642  }
643 
644  BlockFilterIndex* index = GetBlockFilterIndex(filtertype);
645  if (!index) {
646  return RESTERR(req, HTTP_BAD_REQUEST, "Index is not enabled for filtertype " + uri_parts[0]);
647  }
648 
649  const CBlockIndex* block_index;
650  bool block_was_connected;
651  {
652  ChainstateManager* maybe_chainman = GetChainman(context, req);
653  if (!maybe_chainman) return false;
654  ChainstateManager& chainman = *maybe_chainman;
655  LOCK(cs_main);
656  block_index = chainman.m_blockman.LookupBlockIndex(*block_hash);
657  if (!block_index) {
658  return RESTERR(req, HTTP_NOT_FOUND, uri_parts[1] + " not found");
659  }
660  block_was_connected = block_index->IsValid(BLOCK_VALID_SCRIPTS);
661  }
662 
663  bool index_ready = index->BlockUntilSyncedToCurrentChain();
664 
665  BlockFilter filter;
666  if (!index->LookupFilter(block_index, filter)) {
667  std::string errmsg = "Filter not found.";
668 
669  if (!block_was_connected) {
670  errmsg += " Block was not connected to active chain.";
671  } else if (!index_ready) {
672  errmsg += " Block filters are still in the process of being indexed.";
673  } else {
674  errmsg += " This error is unexpected and indicates index corruption.";
675  }
676 
677  return RESTERR(req, HTTP_NOT_FOUND, errmsg);
678  }
679 
680  switch (rf) {
682  DataStream ssResp{};
683  ssResp << filter;
684 
685  req->WriteHeader("Content-Type", "application/octet-stream");
686  req->WriteReply(HTTP_OK, ssResp);
687  return true;
688  }
690  DataStream ssResp{};
691  ssResp << filter;
692 
693  std::string strHex = HexStr(ssResp) + "\n";
694  req->WriteHeader("Content-Type", "text/plain");
695  req->WriteReply(HTTP_OK, strHex);
696  return true;
697  }
700  ret.pushKV("filter", HexStr(filter.GetEncodedFilter()));
701  std::string strJSON = ret.write() + "\n";
702  req->WriteHeader("Content-Type", "application/json");
703  req->WriteReply(HTTP_OK, strJSON);
704  return true;
705  }
706  default: {
707  return RESTERR(req, HTTP_NOT_FOUND, "output format not found (available: " + AvailableDataFormatsString() + ")");
708  }
709  }
710 }
711 
712 // A bit of a hack - dependency on a function defined in rpc/blockchain.cpp
714 
715 static bool rest_chaininfo(const std::any& context, HTTPRequest* req, const std::string& uri_part)
716 {
717  if (!CheckWarmup(req))
718  return false;
719  std::string param;
720  const RESTResponseFormat rf = ParseDataFormat(param, uri_part);
721 
722  switch (rf) {
724  JSONRPCRequest jsonRequest;
725  jsonRequest.context = context;
726  jsonRequest.params = UniValue(UniValue::VARR);
727  UniValue chainInfoObject = getblockchaininfo().HandleRequest(jsonRequest);
728  std::string strJSON = chainInfoObject.write() + "\n";
729  req->WriteHeader("Content-Type", "application/json");
730  req->WriteReply(HTTP_OK, strJSON);
731  return true;
732  }
733  default: {
734  return RESTERR(req, HTTP_NOT_FOUND, "output format not found (available: json)");
735  }
736  }
737 }
738 
739 
741 
742 static bool rest_deploymentinfo(const std::any& context, HTTPRequest* req, const std::string& str_uri_part)
743 {
744  if (!CheckWarmup(req)) return false;
745 
746  std::string hash_str;
747  const RESTResponseFormat rf = ParseDataFormat(hash_str, str_uri_part);
748 
749  switch (rf) {
751  JSONRPCRequest jsonRequest;
752  jsonRequest.context = context;
753  jsonRequest.params = UniValue(UniValue::VARR);
754 
755  if (!hash_str.empty()) {
756  auto hash{uint256::FromHex(hash_str)};
757  if (!hash) {
758  return RESTERR(req, HTTP_BAD_REQUEST, "Invalid hash: " + hash_str);
759  }
760 
761  const ChainstateManager* chainman = GetChainman(context, req);
762  if (!chainman) return false;
763  if (!WITH_LOCK(::cs_main, return chainman->m_blockman.LookupBlockIndex(*hash))) {
764  return RESTERR(req, HTTP_BAD_REQUEST, "Block not found");
765  }
766 
767  jsonRequest.params.push_back(hash_str);
768  }
769 
770  req->WriteHeader("Content-Type", "application/json");
771  req->WriteReply(HTTP_OK, getdeploymentinfo().HandleRequest(jsonRequest).write() + "\n");
772  return true;
773  }
774  default: {
775  return RESTERR(req, HTTP_NOT_FOUND, "output format not found (available: json)");
776  }
777  }
778 
779 }
780 
781 static bool rest_mempool(const std::any& context, HTTPRequest* req, const std::string& str_uri_part)
782 {
783  if (!CheckWarmup(req))
784  return false;
785 
786  std::string param;
787  const RESTResponseFormat rf = ParseDataFormat(param, str_uri_part);
788  if (param != "contents" && param != "info") {
789  return RESTERR(req, HTTP_BAD_REQUEST, "Invalid URI format. Expected /rest/mempool/<info|contents>.json");
790  }
791 
792  const CTxMemPool* mempool = GetMemPool(context, req);
793  if (!mempool) return false;
794 
795  switch (rf) {
797  std::string str_json;
798  if (param == "contents") {
799  std::string raw_verbose;
800  try {
801  raw_verbose = req->GetQueryParameter("verbose").value_or("true");
802  } catch (const std::runtime_error& e) {
803  return RESTERR(req, HTTP_BAD_REQUEST, e.what());
804  }
805  if (raw_verbose != "true" && raw_verbose != "false") {
806  return RESTERR(req, HTTP_BAD_REQUEST, "The \"verbose\" query parameter must be either \"true\" or \"false\".");
807  }
808  std::string raw_mempool_sequence;
809  try {
810  raw_mempool_sequence = req->GetQueryParameter("mempool_sequence").value_or("false");
811  } catch (const std::runtime_error& e) {
812  return RESTERR(req, HTTP_BAD_REQUEST, e.what());
813  }
814  if (raw_mempool_sequence != "true" && raw_mempool_sequence != "false") {
815  return RESTERR(req, HTTP_BAD_REQUEST, "The \"mempool_sequence\" query parameter must be either \"true\" or \"false\".");
816  }
817  const bool verbose{raw_verbose == "true"};
818  const bool mempool_sequence{raw_mempool_sequence == "true"};
819  if (verbose && mempool_sequence) {
820  return RESTERR(req, HTTP_BAD_REQUEST, "Verbose results cannot contain mempool sequence values. (hint: set \"verbose=false\")");
821  }
822  str_json = MempoolToJSON(*mempool, verbose, mempool_sequence).write() + "\n";
823  } else {
824  str_json = MempoolInfoToJSON(*mempool).write() + "\n";
825  }
826 
827  req->WriteHeader("Content-Type", "application/json");
828  req->WriteReply(HTTP_OK, str_json);
829  return true;
830  }
831  default: {
832  return RESTERR(req, HTTP_NOT_FOUND, "output format not found (available: json)");
833  }
834  }
835 }
836 
837 static bool rest_tx(const std::any& context, HTTPRequest* req, const std::string& uri_part)
838 {
839  if (!CheckWarmup(req))
840  return false;
841  std::string hashStr;
842  const RESTResponseFormat rf = ParseDataFormat(hashStr, uri_part);
843 
844  auto hash{Txid::FromHex(hashStr)};
845  if (!hash) {
846  return RESTERR(req, HTTP_BAD_REQUEST, "Invalid hash: " + hashStr);
847  }
848 
849  if (g_txindex) {
850  g_txindex->BlockUntilSyncedToCurrentChain();
851  }
852 
853  const NodeContext* const node = GetNodeContext(context, req);
854  if (!node) return false;
855  uint256 hashBlock = uint256();
856  const CTransactionRef tx{GetTransaction(/*block_index=*/nullptr, node->mempool.get(), *hash, node->chainman->m_blockman, hashBlock)};
857  if (!tx) {
858  return RESTERR(req, HTTP_NOT_FOUND, hashStr + " not found");
859  }
860 
861  switch (rf) {
863  DataStream ssTx;
864  ssTx << TX_WITH_WITNESS(tx);
865 
866  req->WriteHeader("Content-Type", "application/octet-stream");
867  req->WriteReply(HTTP_OK, ssTx);
868  return true;
869  }
870 
872  DataStream ssTx;
873  ssTx << TX_WITH_WITNESS(tx);
874 
875  std::string strHex = HexStr(ssTx) + "\n";
876  req->WriteHeader("Content-Type", "text/plain");
877  req->WriteReply(HTTP_OK, strHex);
878  return true;
879  }
880 
882  UniValue objTx(UniValue::VOBJ);
883  TxToUniv(*tx, /*block_hash=*/hashBlock, /*entry=*/ objTx);
884  std::string strJSON = objTx.write() + "\n";
885  req->WriteHeader("Content-Type", "application/json");
886  req->WriteReply(HTTP_OK, strJSON);
887  return true;
888  }
889 
890  default: {
891  return RESTERR(req, HTTP_NOT_FOUND, "output format not found (available: " + AvailableDataFormatsString() + ")");
892  }
893  }
894 }
895 
896 static bool rest_getutxos(const std::any& context, HTTPRequest* req, const std::string& uri_part)
897 {
898  if (!CheckWarmup(req))
899  return false;
900  std::string param;
901  const RESTResponseFormat rf = ParseDataFormat(param, uri_part);
902 
903  std::vector<std::string> uriParts;
904  if (param.length() > 1)
905  {
906  std::string strUriParams = param.substr(1);
907  uriParts = SplitString(strUriParams, '/');
908  }
909 
910  // throw exception in case of an empty request
911  std::string strRequestMutable = req->ReadBody();
912  if (strRequestMutable.length() == 0 && uriParts.size() == 0)
913  return RESTERR(req, HTTP_BAD_REQUEST, "Error: empty request");
914 
915  bool fInputParsed = false;
916  bool fCheckMemPool = false;
917  std::vector<COutPoint> vOutPoints;
918 
919  // parse/deserialize input
920  // input-format = output-format, rest/getutxos/bin requires binary input, gives binary output, ...
921 
922  if (uriParts.size() > 0)
923  {
924  //inputs is sent over URI scheme (/rest/getutxos/checkmempool/txid1-n/txid2-n/...)
925  if (uriParts[0] == "checkmempool") fCheckMemPool = true;
926 
927  for (size_t i = (fCheckMemPool) ? 1 : 0; i < uriParts.size(); i++)
928  {
929  const auto txid_out{util::Split<std::string_view>(uriParts[i], '-')};
930  if (txid_out.size() != 2) {
931  return RESTERR(req, HTTP_BAD_REQUEST, "Parse error");
932  }
933  auto txid{Txid::FromHex(txid_out.at(0))};
934  auto output{ToIntegral<uint32_t>(txid_out.at(1))};
935 
936  if (!txid || !output) {
937  return RESTERR(req, HTTP_BAD_REQUEST, "Parse error");
938  }
939 
940  vOutPoints.emplace_back(*txid, *output);
941  }
942 
943  if (vOutPoints.size() > 0)
944  fInputParsed = true;
945  else
946  return RESTERR(req, HTTP_BAD_REQUEST, "Error: empty request");
947  }
948 
949  switch (rf) {
951  // convert hex to bin, continue then with bin part
952  std::vector<unsigned char> strRequestV = ParseHex(strRequestMutable);
953  strRequestMutable.assign(strRequestV.begin(), strRequestV.end());
954  [[fallthrough]];
955  }
956 
958  try {
959  //deserialize only if user sent a request
960  if (strRequestMutable.size() > 0)
961  {
962  if (fInputParsed) //don't allow sending input over URI and HTTP RAW DATA
963  return RESTERR(req, HTTP_BAD_REQUEST, "Combination of URI scheme inputs and raw post data is not allowed");
964 
965  DataStream oss{};
966  oss << strRequestMutable;
967  oss >> fCheckMemPool;
968  oss >> vOutPoints;
969  }
970  } catch (const std::ios_base::failure&) {
971  // abort in case of unreadable binary data
972  return RESTERR(req, HTTP_BAD_REQUEST, "Parse error");
973  }
974  break;
975  }
976 
978  if (!fInputParsed)
979  return RESTERR(req, HTTP_BAD_REQUEST, "Error: empty request");
980  break;
981  }
982  default: {
983  return RESTERR(req, HTTP_NOT_FOUND, "output format not found (available: " + AvailableDataFormatsString() + ")");
984  }
985  }
986 
987  // limit max outpoints
988  if (vOutPoints.size() > MAX_GETUTXOS_OUTPOINTS)
989  return RESTERR(req, HTTP_BAD_REQUEST, strprintf("Error: max outpoints exceeded (max: %d, tried: %d)", MAX_GETUTXOS_OUTPOINTS, vOutPoints.size()));
990 
991  // check spentness and form a bitmap (as well as a JSON capable human-readable string representation)
992  std::vector<unsigned char> bitmap;
993  std::vector<CCoin> outs;
994  std::string bitmapStringRepresentation;
995  std::vector<bool> hits;
996  bitmap.resize((vOutPoints.size() + 7) / 8);
997  ChainstateManager* maybe_chainman = GetChainman(context, req);
998  if (!maybe_chainman) return false;
999  ChainstateManager& chainman = *maybe_chainman;
1000  decltype(chainman.ActiveHeight()) active_height;
1001  uint256 active_hash;
1002  {
1003  auto process_utxos = [&vOutPoints, &outs, &hits, &active_height, &active_hash, &chainman](const CCoinsView& view, const CTxMemPool* mempool) EXCLUSIVE_LOCKS_REQUIRED(chainman.GetMutex()) {
1004  for (const COutPoint& vOutPoint : vOutPoints) {
1005  auto coin = !mempool || !mempool->isSpent(vOutPoint) ? view.GetCoin(vOutPoint) : std::nullopt;
1006  hits.push_back(coin.has_value());
1007  if (coin) outs.emplace_back(std::move(*coin));
1008  }
1009  active_height = chainman.ActiveHeight();
1010  active_hash = chainman.ActiveTip()->GetBlockHash();
1011  };
1012 
1013  if (fCheckMemPool) {
1014  const CTxMemPool* mempool = GetMemPool(context, req);
1015  if (!mempool) return false;
1016  // use db+mempool as cache backend in case user likes to query mempool
1017  LOCK2(cs_main, mempool->cs);
1018  CCoinsViewCache& viewChain = chainman.ActiveChainstate().CoinsTip();
1019  CCoinsViewMemPool viewMempool(&viewChain, *mempool);
1020  process_utxos(viewMempool, mempool);
1021  } else {
1022  LOCK(cs_main);
1023  process_utxos(chainman.ActiveChainstate().CoinsTip(), nullptr);
1024  }
1025 
1026  for (size_t i = 0; i < hits.size(); ++i) {
1027  const bool hit = hits[i];
1028  bitmapStringRepresentation.append(hit ? "1" : "0"); // form a binary string representation (human-readable for json output)
1029  bitmap[i / 8] |= ((uint8_t)hit) << (i % 8);
1030  }
1031  }
1032 
1033  switch (rf) {
1035  // serialize data
1036  // use exact same output as mentioned in Bip64
1037  DataStream ssGetUTXOResponse{};
1038  ssGetUTXOResponse << active_height << active_hash << bitmap << outs;
1039 
1040  req->WriteHeader("Content-Type", "application/octet-stream");
1041  req->WriteReply(HTTP_OK, ssGetUTXOResponse);
1042  return true;
1043  }
1044 
1045  case RESTResponseFormat::HEX: {
1046  DataStream ssGetUTXOResponse{};
1047  ssGetUTXOResponse << active_height << active_hash << bitmap << outs;
1048  std::string strHex = HexStr(ssGetUTXOResponse) + "\n";
1049 
1050  req->WriteHeader("Content-Type", "text/plain");
1051  req->WriteReply(HTTP_OK, strHex);
1052  return true;
1053  }
1054 
1055  case RESTResponseFormat::JSON: {
1056  UniValue objGetUTXOResponse(UniValue::VOBJ);
1057 
1058  // pack in some essentials
1059  // use more or less the same output as mentioned in Bip64
1060  objGetUTXOResponse.pushKV("chainHeight", active_height);
1061  objGetUTXOResponse.pushKV("chaintipHash", active_hash.GetHex());
1062  objGetUTXOResponse.pushKV("bitmap", bitmapStringRepresentation);
1063 
1064  UniValue utxos(UniValue::VARR);
1065  for (const CCoin& coin : outs) {
1066  UniValue utxo(UniValue::VOBJ);
1067  utxo.pushKV("height", coin.nHeight);
1068  utxo.pushKV("value", ValueFromAmount(coin.out.nValue));
1069 
1070  // include the script in a json output
1072  ScriptToUniv(coin.out.scriptPubKey, /*out=*/o, /*include_hex=*/true, /*include_address=*/true);
1073  utxo.pushKV("scriptPubKey", std::move(o));
1074  utxos.push_back(std::move(utxo));
1075  }
1076  objGetUTXOResponse.pushKV("utxos", std::move(utxos));
1077 
1078  // return json string
1079  std::string strJSON = objGetUTXOResponse.write() + "\n";
1080  req->WriteHeader("Content-Type", "application/json");
1081  req->WriteReply(HTTP_OK, strJSON);
1082  return true;
1083  }
1084  default: {
1085  return RESTERR(req, HTTP_NOT_FOUND, "output format not found (available: " + AvailableDataFormatsString() + ")");
1086  }
1087  }
1088 }
1089 
1090 static bool rest_blockhash_by_height(const std::any& context, HTTPRequest* req,
1091  const std::string& str_uri_part)
1092 {
1093  if (!CheckWarmup(req)) return false;
1094  std::string height_str;
1095  const RESTResponseFormat rf = ParseDataFormat(height_str, str_uri_part);
1096 
1097  const auto blockheight{ToIntegral<int32_t>(height_str)};
1098  if (!blockheight || *blockheight < 0) {
1099  return RESTERR(req, HTTP_BAD_REQUEST, "Invalid height: " + SanitizeString(height_str, SAFE_CHARS_URI));
1100  }
1101 
1102  CBlockIndex* pblockindex = nullptr;
1103  {
1104  ChainstateManager* maybe_chainman = GetChainman(context, req);
1105  if (!maybe_chainman) return false;
1106  ChainstateManager& chainman = *maybe_chainman;
1107  LOCK(cs_main);
1108  const CChain& active_chain = chainman.ActiveChain();
1109  if (*blockheight > active_chain.Height()) {
1110  return RESTERR(req, HTTP_NOT_FOUND, "Block height out of range");
1111  }
1112  pblockindex = active_chain[*blockheight];
1113  }
1114  switch (rf) {
1116  DataStream ss_blockhash{};
1117  ss_blockhash << pblockindex->GetBlockHash();
1118  req->WriteHeader("Content-Type", "application/octet-stream");
1119  req->WriteReply(HTTP_OK, ss_blockhash);
1120  return true;
1121  }
1122  case RESTResponseFormat::HEX: {
1123  req->WriteHeader("Content-Type", "text/plain");
1124  req->WriteReply(HTTP_OK, pblockindex->GetBlockHash().GetHex() + "\n");
1125  return true;
1126  }
1127  case RESTResponseFormat::JSON: {
1128  req->WriteHeader("Content-Type", "application/json");
1130  resp.pushKV("blockhash", pblockindex->GetBlockHash().GetHex());
1131  req->WriteReply(HTTP_OK, resp.write() + "\n");
1132  return true;
1133  }
1134  default: {
1135  return RESTERR(req, HTTP_NOT_FOUND, "output format not found (available: " + AvailableDataFormatsString() + ")");
1136  }
1137  }
1138 }
1139 
1140 static const struct {
1141  const char* prefix;
1142  bool (*handler)(const std::any& context, HTTPRequest* req, const std::string& strReq);
1143 } uri_prefixes[] = {
1144  {"/rest/tx/", rest_tx},
1145  {"/rest/block/notxdetails/", rest_block_notxdetails},
1146  {"/rest/block/", rest_block_extended},
1147  {"/rest/blockpart/", rest_block_part},
1148  {"/rest/blockfilter/", rest_block_filter},
1149  {"/rest/blockfilterheaders/", rest_filter_header},
1150  {"/rest/chaininfo", rest_chaininfo},
1151  {"/rest/mempool/", rest_mempool},
1152  {"/rest/headers/", rest_headers},
1153  {"/rest/getutxos", rest_getutxos},
1154  {"/rest/deploymentinfo/", rest_deploymentinfo},
1155  {"/rest/deploymentinfo", rest_deploymentinfo},
1156  {"/rest/blockhashbyheight/", rest_blockhash_by_height},
1157  {"/rest/spenttxouts/", rest_spent_txouts},
1158 };
1159 
1160 void StartREST(const std::any& context)
1161 {
1162  for (const auto& up : uri_prefixes) {
1163  auto handler = [context, up](HTTPRequest* req, const std::string& prefix) { return up.handler(context, req, prefix); };
1164  RegisterHTTPHandler(up.prefix, false, handler);
1165  }
1166 }
1167 
1169 {
1170 }
1171 
1172 void StopREST()
1173 {
1174  for (const auto& up : uri_prefixes) {
1175  UnregisterHTTPHandler(up.prefix, false);
1176  }
1177 }
std::shared_ptr< const CTransaction > CTransactionRef
Definition: transaction.h:403
uint32_t nHeight
Definition: rest.cpp:57
CCoinsViewCache & CoinsTip() EXCLUSIVE_LOCKS_REQUIRED(
Definition: validation.h:686
CAmount nValue
Definition: transaction.h:142
node::BlockManager m_blockman
A single BlockManager instance is shared across each constructed chainstate to avoid duplicating bloc...
Definition: validation.h:1038
static ChainstateManager * GetChainman(const std::any &context, HTTPRequest *req)
Get the node context chainstatemanager.
Definition: rest.cpp:118
bool LookupFilter(const CBlockIndex *block_index, BlockFilter &filter_out) const
Get a single filter by block.
Interval between compact filter checkpoints.
void push_back(UniValue val)
Definition: univalue.cpp:103
std::vector< Coin > vprevout
Definition: undo.h:56
UniValue MempoolToJSON(const CTxMemPool &pool, bool verbose, bool include_mempool_sequence)
Mempool to JSON.
Definition: mempool.cpp:571
int ret
static bool CheckWarmup(HTTPRequest *req)
Definition: rest.cpp:170
std::any context
Definition: request.h:62
std::vector< std::string > SplitString(std::string_view str, char sep)
Definition: string.h:149
The same as previous option with information about prevouts if available.
bool BlockFilterTypeByName(std::string_view name, BlockFilterType &filter_type)
Find a filter type by its human-readable name.
std::vector< Byte > ParseHex(std::string_view hex_str)
Like TryParseHex, but returns an empty vector on invalid input.
Definition: strencodings.h:69
assert(!tx.IsCoinBase())
void WriteReply(int nStatus, std::string_view reply="")
Write HTTP reply.
Definition: httpserver.h:141
CScript scriptPubKey
Definition: transaction.h:143
UniValue ValueFromAmount(const CAmount amount)
Definition: core_io.cpp:285
A UTXO entry.
Definition: coins.h:34
Definition: block.h:73
static bool rest_block_part(const std::any &context, HTTPRequest *req, const std::string &uri_part)
Definition: rest.cpp:480
static bool rest_tx(const std::any &context, HTTPRequest *req, const std::string &uri_part)
Definition: rest.cpp:837
UniValue blockToJSON(BlockManager &blockman, const CBlock &block, const CBlockIndex &tip, const CBlockIndex &blockindex, TxVerbosity verbosity, const uint256 pow_limit)
Block description to JSON.
Definition: blockchain.cpp:202
Interface for managing multiple Chainstate objects, where each chainstate is associated with chainsta...
Definition: validation.h:939
#define strprintf
Format arguments and return the string or write to given std::ostream (see tinyformat::format doc for...
Definition: tinyformat.h:1172
An in-memory indexed chain of blocks.
Definition: chain.h:379
RESTResponseFormat
Definition: rest.h:10
static std::string AvailableDataFormatsString()
Definition: rest.cpp:153
BlockFilterIndex * GetBlockFilterIndex(BlockFilterType filter_type)
Get a block filter index by type.
static bool rest_headers(const std::any &context, HTTPRequest *req, const std::string &uri_part)
Definition: rest.cpp:178
const char * prefix
Definition: rest.cpp:1141
CChain & ActiveChain() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
Definition: validation.h:1165
RESTResponseFormat ParseDataFormat(std::string &param, const std::string &strReq)
Parse a URI to get the data format and URI without data format and query string.
Definition: rest.cpp:128
int Height() const
Return the maximal height in the chain.
Definition: chain.h:425
CTxOut out
unspent transaction output
Definition: coins.h:38
HTTPStatusCode
HTTP status codes.
Definition: protocol.h:10
static bool rest_block_filter(const std::any &context, HTTPRequest *req, const std::string &uri_part)
Definition: rest.cpp:621
static bool rest_spent_txouts(const std::any &context, HTTPRequest *req, const std::string &uri_part)
Definition: rest.cpp:312
UniValue MempoolInfoToJSON(const CTxMemPool &pool)
Mempool information to JSON.
Definition: mempool.cpp:1033
static const struct @10 uri_prefixes[]
std::optional< std::string > GetQueryParameter(const std::string &key) const
Get the query parameter value from request uri for a specified key, or std::nullopt if the key is not...
Definition: httpserver.cpp:654
Definition: common.h:29
static bool rest_filter_header(const std::any &context, HTTPRequest *req, const std::string &uri_part)
Definition: rest.cpp:499
static bool rest_chaininfo(const std::any &context, HTTPRequest *req, const std::string &uri_part)
Definition: rest.cpp:715
void TxToUniv(const CTransaction &tx, const uint256 &block_hash, UniValue &entry, bool include_hex, const CTxUndo *txundo, TxVerbosity verbosity, std::function< bool(const CTxOut &)> is_change_func)
Definition: core_io.cpp:430
bool(* handler)(const std::any &context, HTTPRequest *req, const std::string &strReq)
Definition: rest.cpp:1142
void RegisterHTTPHandler(const std::string &prefix, bool exactMatch, const HTTPRequestHandler &handler)
Register handler for prefix.
Definition: httpserver.cpp:688
Minimal stream for reading from an existing byte array by std::span.
Definition: streams.h:82
SERIALIZE_METHODS(CCoin, obj)
Definition: rest.cpp:63
static bool rest_block_extended(const std::any &context, HTTPRequest *req, const std::string &uri_part)
Definition: rest.cpp:470
uint256 GetBlockHash() const
Definition: chain.h:198
static bool rest_mempool(const std::any &context, HTTPRequest *req, const std::string &str_uri_part)
Definition: rest.cpp:781
std::string SanitizeString(std::string_view str, int rule)
Remove unsafe chars.
BlockFilterType
Definition: blockfilter.h:93
NodeContext struct containing references to chain state and connection state.
Definition: context.h:56
#define LOCK2(cs1, cs2)
Definition: sync.h:259
uint256 powLimit
Proof of work parameters.
Definition: params.h:112
static bool RESTERR(HTTPRequest *req, enum HTTPStatusCode status, std::string message)
Definition: rest.cpp:70
RESTResponseFormat rf
Definition: rest.cpp:47
static void BlockUndoToJSON(const CBlockUndo &block_undo, UniValue &result)
Serialize spent outputs as a list of per-transaction CTxOut lists using JSON format.
Definition: rest.cpp:293
Scripts & signatures ok.
Definition: chain.h:69
Abstract view on the open txout dataset.
Definition: coins.h:307
UniValue params
Definition: request.h:57
UniValue blockheaderToJSON(const CBlockIndex &tip, const CBlockIndex &blockindex, const uint256 pow_limit)
Block header to JSON.
Definition: blockchain.cpp:154
#define LOCK(cs)
Definition: sync.h:258
const char * name
Definition: rest.cpp:48
Double ended buffer combining vector and stream-like interfaces.
Definition: streams.h:132
std::unique_ptr< TxIndex > g_txindex
The global transaction index, used in GetTransaction. May be null.
Definition: txindex.cpp:34
void WriteCompactSize(SizeComputer &os, uint64_t nSize)
Definition: serialize.h:1089
bool LookupFilterHeader(const CBlockIndex *block_index, uint256 &header_out) EXCLUSIVE_LOCKS_REQUIRED(!m_cs_headers_cache)
Get a single filter header by block.
Complete block filter struct as defined in BIP 157.
Definition: blockfilter.h:115
bool Contains(const CBlockIndex *pindex) const
Efficiently check whether a block is present in this chain.
Definition: chain.h:410
CBlockIndex * Next(const CBlockIndex *pindex) const
Find the successor of a block in this chain, or nullptr if the given index is not found or is the tip...
Definition: chain.h:416
static std::optional< uint256 > FromHex(std::string_view str)
Definition: uint256.h:197
void UnregisterHTTPHandler(const std::string &prefix, bool exactMatch)
Unregister handler for prefix.
Definition: httpserver.cpp:695
const Consensus::Params & GetConsensus() const
Definition: validation.h:1008
std::string write(unsigned int prettyIndent=0, unsigned int indentLevel=0) const
ReadRawBlockResult ReadRawBlock(const FlatFilePos &pos, std::optional< std::pair< size_t, size_t >> block_part=std::nullopt) const
CBlockIndex * ActiveTip() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
Definition: validation.h:1167
An output of a transaction.
Definition: transaction.h:139
RecursiveMutex & GetMutex() const LOCK_RETURNED(
Alias for cs_main.
Definition: validation.h:1032
An outpoint - a combination of a transaction hash and an index n into its vout.
Definition: transaction.h:28
CBlockIndex * LookupBlockIndex(const uint256 &hash) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
static const struct @9 rf_names[]
#define WITH_LOCK(cs, code)
Run code while locking a mutex.
Definition: sync.h:289
CCoin()
Definition: rest.cpp:60
CTxOut out
Definition: rest.cpp:58
CTransactionRef GetTransaction(const CBlockIndex *const block_index, const CTxMemPool *const mempool, const Txid &hash, const BlockManager &blockman, uint256 &hashBlock)
Return transaction with a given hash.
Definition: messages.h:21
void ScriptToUniv(const CScript &script, UniValue &out, bool include_hex, bool include_address, const SigningProvider *provider)
Definition: core_io.cpp:409
static bool rest_blockhash_by_height(const std::any &context, HTTPRequest *req, const std::string &str_uri_part)
Definition: rest.cpp:1090
static void SerializeBlockUndo(DataStream &stream, const CBlockUndo &block_undo)
Serialize spent outputs as a list of per-transaction CTxOut lists using binary format.
Definition: rest.cpp:278
static CTxMemPool * GetMemPool(const std::any &context, HTTPRequest *req)
Get the node context mempool.
Definition: rest.cpp:101
static const size_t MAX_GETUTXOS_OUTPOINTS
Definition: rest.cpp:43
256-bit opaque blob.
Definition: uint256.h:195
RPCHelpMan getblockchaininfo()
#define EXCLUSIVE_LOCKS_REQUIRED(...)
Definition: threadsafety.h:51
CTxMemPool stores valid-according-to-the-current-best-chain transactions that may be included in the ...
Definition: txmempool.h:186
auto result
Definition: common-types.h:74
void WriteHeader(const std::string &hdr, const std::string &value)
Write output header.
Definition: httpserver.cpp:571
Only TXID for each block&#39;s transaction.
RPCHelpMan getdeploymentinfo()
void StopREST()
Stop HTTP REST subsystem.
Definition: rest.cpp:1172
The block chain is a tree shaped structure starting with the genesis block at the root...
Definition: chain.h:93
void pushKV(std::string key, UniValue val)
Definition: univalue.cpp:125
Undo information for a CBlock.
Definition: undo.h:62
Undo information for a CTransaction.
Definition: undo.h:52
static NodeContext * GetNodeContext(const std::any &context, HTTPRequest *req)
Get the node context.
Definition: rest.cpp:84
CBlockIndex * Tip() const
Returns the index entry for the tip of this chain, or nullptr if none.
Definition: chain.h:396
std::string GetHex() const
Definition: uint256.cpp:11
int ActiveHeight() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
Definition: validation.h:1166
static bool rest_block_notxdetails(const std::any &context, HTTPRequest *req, const std::string &uri_part)
Definition: rest.cpp:475
bool RPCIsInWarmup(std::string *outStatus)
Definition: server.cpp:331
Definition: rest.cpp:56
static bool rest_getutxos(const std::any &context, HTTPRequest *req, const std::string &uri_part)
Definition: rest.cpp:896
std::string ReadBody()
Read request body.
Definition: httpserver.cpp:551
bool IsValid(enum BlockStatus nUpTo) const EXCLUSIVE_LOCKS_REQUIRED(
Check whether this block index entry is valid up to the passed validity level.
Definition: chain.h:250
#define STR_INTERNAL_BUG(msg)
Definition: check.h:96
In-flight HTTP request.
Definition: httpserver.h:70
int nHeight
height of the entry in the chain. The genesis block has height 0
Definition: chain.h:106
FlatFilePos GetBlockPos() const EXCLUSIVE_LOCKS_REQUIRED(
Definition: chain.h:163
const std::vector< unsigned char > & GetEncodedFilter() const LIFETIMEBOUND
Definition: blockfilter.h:139
Chainstate & ActiveChainstate() const
Alternatives to CurrentChainstate() used by older code to query latest chainstate information without...
void InterruptREST()
Interrupt RPC REST subsystem.
Definition: rest.cpp:1168
CCoinsView that adds a memory cache for transactions to another CCoinsView.
Definition: coins.h:367
full block available in blk*.dat
Definition: chain.h:75
CCoin(Coin &&in)
Definition: rest.cpp:61
Chars allowed in URIs (RFC 3986)
Definition: strencodings.h:36
UniValue HandleRequest(const JSONRPCRequest &request) const
Definition: util.cpp:635
#define READWRITE(...)
Definition: serialize.h:145
std::vector< CTxUndo > vtxundo
Definition: undo.h:65
RecursiveMutex cs_main
Mutex to guard access to validation specific variables, such as reading or changing the chainstate...
Definition: cs_main.cpp:8
CCoinsView that brings transactions from a mempool into view.
Definition: txmempool.h:752
static bool rest_deploymentinfo(const std::any &context, HTTPRequest *req, const std::string &str_uri_part)
Definition: rest.cpp:742
std::string HexStr(const std::span< const uint8_t > s)
Convert a span of bytes to a lower-case hexadecimal string.
Definition: hex_base.cpp:30
static std::optional< transaction_identifier > FromHex(std::string_view hex)
static constexpr unsigned int MAX_REST_HEADERS_RESULTS
Definition: rest.cpp:44
static bool rest_block(const std::any &context, HTTPRequest *req, const std::string &uri_part, std::optional< TxVerbosity > tx_verbosity, std::optional< std::pair< size_t, size_t >> block_part=std::nullopt)
This handler is used by multiple HTTP endpoints:
Definition: rest.cpp:388
RecursiveMutex cs
This mutex needs to be locked when accessing mapTx or other members that are guarded by it...
Definition: txmempool.h:260
static constexpr TransactionSerParams TX_WITH_WITNESS
Definition: transaction.h:180
bool ReadBlockUndo(CBlockUndo &blockundo, const CBlockIndex &index) const
void StartREST(const std::any &context)
Start HTTP REST subsystem.
Definition: rest.cpp:1160