7#include <blockfilter.h>
9#include <chainparams.h>
65#include <bitcoin-build-config.h>
73#include <boost/signals2/signal.hpp>
95#ifdef ENABLE_EXTERNAL_SIGNER
96class ExternalSignerImpl :
public interfaces::ExternalSigner
99 ExternalSignerImpl(::ExternalSigner signer) : m_signer(std::move(signer)) {}
100 std::string getName()
override {
return m_signer.m_name; }
101 ::ExternalSigner m_signer;
105class NodeImpl :
public Node
108 explicit NodeImpl(NodeContext& context) { setContext(&context); }
111 bilingual_str getWarnings()
override {
return Join(
Assert(m_context->warnings)->GetMessages(),
Untranslated(
"<hr />")); }
112 int getExitStatus()
override {
return Assert(m_context)->exit_status.load(); }
114 bool baseInitialize()
override
119 m_context->warnings = std::make_unique<node::Warnings>();
120 m_context->kernel = std::make_unique<kernel::Context>();
121 m_context->ecc_context = std::make_unique<ECC_Context>();
129 bool appInitMain(interfaces::BlockAndHeaderTipInfo* tip_info)
override
131 if (
AppInitMain(*m_context, tip_info))
return true;
133 m_context->exit_status.store(EXIT_FAILURE);
136 void appShutdown()
override
140 void startShutdown()
override
142 NodeContext& ctx{*
Assert(m_context)};
144 LogError(
"Failed to send shutdown signal\n");
149 bool isSettingIgnored(
const std::string&
name)
override
151 bool ignored =
false;
154 ignored = !options->empty();
181 void resetSettings()
override
189 void mapPort(
bool enable)
override {
StartMapPort(enable); }
190 bool getProxy(
Network net, Proxy& proxy_info)
override {
return GetProxy(net, proxy_info); }
193 return m_context->connman ? m_context->connman->GetNodeCount(
flags) : 0;
195 bool getNodesStats(NodesStats& stats)
override
199 if (m_context->connman) {
200 std::vector<CNodeStats> stats_temp;
201 m_context->connman->GetNodeStats(stats_temp);
203 stats.reserve(stats_temp.size());
204 for (
auto& node_stats_temp : stats_temp) {
205 stats.emplace_back(std::move(node_stats_temp),
false, CNodeStateStats());
209 if (m_context->peerman) {
212 for (
auto& node_stats : stats) {
213 std::get<1>(node_stats) =
214 m_context->peerman->GetNodeStateStats(std::get<0>(node_stats).nodeid, std::get<2>(node_stats));
222 bool getBanned(
banmap_t& banmap)
override
224 if (m_context->banman) {
225 m_context->banman->GetBanned(banmap);
230 bool ban(
const CNetAddr& net_addr, int64_t ban_time_offset)
override
232 if (m_context->banman) {
233 m_context->banman->Ban(net_addr, ban_time_offset);
238 bool unban(
const CSubNet&
ip)
override
240 if (m_context->banman) {
241 m_context->banman->Unban(
ip);
246 bool disconnectByAddress(
const CNetAddr& net_addr)
override
248 if (m_context->connman) {
249 return m_context->connman->DisconnectNode(net_addr);
253 bool disconnectById(
NodeId id)
override
255 if (m_context->connman) {
256 return m_context->connman->DisconnectNode(
id);
260 std::vector<std::unique_ptr<interfaces::ExternalSigner>> listExternalSigners()
override
262#ifdef ENABLE_EXTERNAL_SIGNER
263 std::vector<ExternalSigner> signers = {};
267 std::vector<std::unique_ptr<interfaces::ExternalSigner>> result;
268 result.reserve(signers.size());
269 for (
auto& signer : signers) {
270 result.emplace_back(std::make_unique<ExternalSignerImpl>(std::move(signer)));
283 int64_t getTotalBytesRecv()
override {
return m_context->connman ? m_context->connman->GetTotalBytesRecv() : 0; }
284 int64_t getTotalBytesSent()
override {
return m_context->connman ? m_context->connman->GetTotalBytesSent() : 0; }
285 size_t getMempoolSize()
override {
return m_context->mempool ? m_context->mempool->size() : 0; }
286 size_t getMempoolDynamicUsage()
override {
return m_context->mempool ? m_context->mempool->DynamicMemoryUsage() : 0; }
287 size_t getMempoolMaxUsage()
override {
return m_context->mempool ? m_context->mempool->m_opts.max_size_bytes : 0; }
288 bool getHeaderTip(
int& height, int64_t& block_time)
override
291 auto best_header = chainman().m_best_header;
293 height = best_header->nHeight;
294 block_time = best_header->GetBlockTime();
299 std::map<CNetAddr, LocalServiceInfo> getNetLocalAddresses()
override
301 if (m_context->connman)
302 return m_context->connman->getNetLocalAddresses();
306 int getNumBlocks()
override
309 return chainman().ActiveChain().Height();
311 uint256 getBestBlockHash()
override
314 return tip ? tip->
GetBlockHash() : chainman().GetParams().GenesisBlock().GetHash();
316 int64_t getLastBlockTime()
override
319 if (chainman().ActiveChain().Tip()) {
320 return chainman().ActiveChain().Tip()->GetBlockTime();
322 return chainman().GetParams().GenesisBlock().GetBlockTime();
324 double getVerificationProgress()
override
326 LOCK(chainman().GetMutex());
327 return chainman().GuessVerificationProgress(chainman().ActiveTip());
329 bool isInitialBlockDownload()
override
331 return chainman().IsInitialBlockDownload();
333 bool isLoadingBlocks()
override {
return chainman().m_blockman.LoadingBlocks(); }
334 void setNetworkActive(
bool active)
override
336 if (m_context->connman) {
337 m_context->connman->SetNetworkActive(active);
340 bool getNetworkActive()
override {
return m_context->connman && m_context->connman->GetNetworkActive(); }
341 CFeeRate getDustRelayFee()
override
344 return m_context->mempool->m_opts.dust_relay_feerate;
346 UniValue executeRpc(
const std::string&
command,
const UniValue& params,
const std::string& uri)
override
353 return ::tableRPC.execute(req);
355 std::vector<std::string> listRpcCommands()
override { return ::tableRPC.listCommands(); }
356 std::optional<Coin> getUnspentOutput(
const COutPoint& output)
override
359 return chainman().ActiveChainstate().CoinsTip().GetCoin(output);
370 WalletLoader& walletLoader()
override
372 return *
Assert(m_context->wallet_loader);
374 std::unique_ptr<Handler> handleInitMessage(InitMessageFn fn)
override
378 std::unique_ptr<Handler> handleMessageBox(MessageBoxFn fn)
override
382 std::unique_ptr<Handler> handleQuestion(QuestionFn fn)
override
386 std::unique_ptr<Handler> handleShowProgress(ShowProgressFn fn)
override
390 std::unique_ptr<Handler> handleInitWallet(InitWalletFn fn)
override
394 std::unique_ptr<Handler> handleNotifyNumConnectionsChanged(NotifyNumConnectionsChangedFn fn)
override
398 std::unique_ptr<Handler> handleNotifyNetworkActiveChanged(NotifyNetworkActiveChangedFn fn)
override
402 std::unique_ptr<Handler> handleNotifyAlertChanged(NotifyAlertChangedFn fn)
override
406 std::unique_ptr<Handler> handleBannedListChanged(BannedListChangedFn fn)
override
410 std::unique_ptr<Handler> handleNotifyBlockTip(NotifyBlockTipFn fn)
override
416 std::unique_ptr<Handler> handleNotifyHeaderTip(NotifyHeaderTipFn fn)
override
420 fn(sync_state, BlockTip{(int)height, timestamp, uint256{}}, presync);
423 NodeContext* context()
override {
return m_context; }
424 void setContext(NodeContext* context)
override
429 ChainstateManager& chainman() {
return *
Assert(m_context->chainman); }
430 NodeContext* m_context{
nullptr};
436 if (!index)
return false;
437 if (block.m_hash) *block.m_hash = index->
GetBlockHash();
438 if (block.m_height) *block.m_height = index->
nHeight;
439 if (block.m_time) *block.m_time = index->
GetBlockTime();
442 if (block.m_in_active_chain) *block.m_in_active_chain = active[index->nHeight] == index;
443 if (block.m_locator) { *block.m_locator =
GetLocator(index); }
444 if (block.m_next_block) FillBlock(active[index->nHeight] == index ? active[index->nHeight + 1] :
nullptr, *block.m_next_block, lock, active, blockman);
447 if (!blockman.ReadBlock(*block.m_data, *index)) block.m_data->SetNull();
453class NotificationsProxy :
public CValidationInterface
456 explicit NotificationsProxy(std::shared_ptr<Chain::Notifications> notifications)
457 : m_notifications(std::move(notifications)) {}
458 virtual ~NotificationsProxy() =
default;
459 void TransactionAddedToMempool(
const NewMempoolTransactionInfo& tx, uint64_t mempool_sequence)
override
461 m_notifications->transactionAddedToMempool(tx.
info.
m_tx);
465 m_notifications->transactionRemovedFromMempool(tx, reason);
467 void BlockConnected(
const ChainstateRole& role,
const std::shared_ptr<const CBlock>& block,
const CBlockIndex* index)
override
471 void BlockDisconnected(
const std::shared_ptr<const CBlock>& block,
const CBlockIndex* index)
override
475 void UpdatedBlockTip(
const CBlockIndex* index,
const CBlockIndex* fork_index,
bool is_ibd)
override
477 m_notifications->updatedBlockTip();
479 void ChainStateFlushed(
const ChainstateRole& role,
const CBlockLocator& locator)
override
481 m_notifications->chainStateFlushed(role, locator);
483 std::shared_ptr<Chain::Notifications> m_notifications;
486class NotificationsHandlerImpl :
public Handler
489 explicit NotificationsHandlerImpl(ValidationSignals& signals, std::shared_ptr<Chain::Notifications> notifications)
490 : m_signals{signals}, m_proxy{std::make_shared<NotificationsProxy>(std::move(notifications))}
494 ~NotificationsHandlerImpl()
override { disconnect(); }
495 void disconnect()
override
498 m_signals.UnregisterSharedValidationInterface(m_proxy);
502 ValidationSignals& m_signals;
503 std::shared_ptr<NotificationsProxy> m_proxy;
506class RpcHandlerImpl :
public Handler
509 explicit RpcHandlerImpl(
const CRPCCommand&
command) : m_command(
command), m_wrapped_command(&
command)
511 m_command.actor = [
this](
const JSONRPCRequest& request, UniValue& result,
bool last_handler) {
512 if (!m_wrapped_command)
return false;
514 return m_wrapped_command->actor(request, result, last_handler);
515 }
catch (
const UniValue& e) {
520 const UniValue& code = e[
"code"];
531 void disconnect() final
533 if (m_wrapped_command) {
534 m_wrapped_command =
nullptr;
539 ~RpcHandlerImpl()
override { disconnect(); }
541 CRPCCommand m_command;
542 const CRPCCommand* m_wrapped_command;
545class ChainImpl :
public Chain
548 explicit ChainImpl(NodeContext& node) : m_node(node) {}
549 std::optional<int> getHeight()
override
552 return height >= 0 ? std::optional{height} : std::nullopt;
554 uint256 getBlockHash(
int height)
override
557 return Assert(chainman().ActiveChain()[height])->GetBlockHash();
559 bool haveBlockOnDisk(
int height)
override
562 const CBlockIndex* block{chainman().ActiveChain()[height]};
565 std::optional<int> findLocatorFork(
const CBlockLocator& locator)
override
568 if (
const CBlockIndex* fork = chainman().ActiveChainstate().FindForkInGlobalIndex(locator)) {
569 return fork->nHeight;
580 if (!block_filter_index)
return std::nullopt;
583 const CBlockIndex* index{
WITH_LOCK(
::cs_main,
return chainman().m_blockman.LookupBlockIndex(block_hash))};
584 if (index ==
nullptr || !block_filter_index->
LookupFilter(index, filter))
return std::nullopt;
587 bool findBlock(
const uint256& hash,
const FoundBlock& block)
override
590 return FillBlock(chainman().m_blockman.LookupBlockIndex(hash), block, lock, chainman().ActiveChain(), chainman().m_blockman);
592 bool findFirstBlockWithTimeAndHeight(int64_t min_time,
int min_height,
const FoundBlock& block)
override
595 const CChain& active = chainman().ActiveChain();
596 return FillBlock(active.
FindEarliestAtLeast(min_time, min_height), block, lock, active, chainman().m_blockman);
598 bool findAncestorByHeight(
const uint256& block_hash,
int ancestor_height,
const FoundBlock& ancestor_out)
override
601 const CChain& active = chainman().ActiveChain();
602 if (
const CBlockIndex* block = chainman().m_blockman.LookupBlockIndex(block_hash)) {
603 if (
const CBlockIndex* ancestor = block->GetAncestor(ancestor_height)) {
604 return FillBlock(ancestor, ancestor_out, lock, active, chainman().m_blockman);
607 return FillBlock(
nullptr, ancestor_out, lock, active, chainman().m_blockman);
609 bool findAncestorByHash(
const uint256& block_hash,
const uint256& ancestor_hash,
const FoundBlock& ancestor_out)
override
612 const CBlockIndex* block = chainman().m_blockman.LookupBlockIndex(block_hash);
613 const CBlockIndex* ancestor = chainman().m_blockman.LookupBlockIndex(ancestor_hash);
614 if (block && ancestor && block->
GetAncestor(ancestor->
nHeight) != ancestor) ancestor =
nullptr;
615 return FillBlock(ancestor, ancestor_out, lock, chainman().ActiveChain(), chainman().m_blockman);
617 bool findCommonAncestor(
const uint256& block_hash1,
const uint256& block_hash2,
const FoundBlock& ancestor_out,
const FoundBlock& block1_out,
const FoundBlock& block2_out)
override
620 const CChain& active = chainman().ActiveChain();
621 const CBlockIndex* block1 = chainman().m_blockman.LookupBlockIndex(block_hash1);
622 const CBlockIndex* block2 = chainman().m_blockman.LookupBlockIndex(block_hash2);
623 const CBlockIndex* ancestor = block1 && block2 ?
LastCommonAncestor(block1, block2) : nullptr;
627 return int{FillBlock(ancestor, ancestor_out, lock, active, chainman().m_blockman)} &
628 int{FillBlock(block1, block1_out, lock, active, chainman().m_blockman)} &
629 int{FillBlock(block2, block2_out, lock, active, chainman().m_blockman)};
631 void findCoins(std::map<COutPoint, Coin>& coins)
override {
return FindCoins(m_node, coins); }
632 double guessVerificationProgress(
const uint256& block_hash)
override
634 LOCK(chainman().GetMutex());
635 return chainman().GuessVerificationProgress(chainman().m_blockman.LookupBlockIndex(block_hash));
637 bool hasBlocks(
const uint256& block_hash,
int min_height, std::optional<int> max_height)
override
647 if (
const CBlockIndex* block = chainman().m_blockman.LookupBlockIndex(block_hash)) {
648 if (max_height && block->
nHeight >= *max_height) block = block->
GetAncestor(*max_height);
651 if (block->
nHeight <= min_height || !block->
pprev)
return true;
659 LOCK(m_node.mempool->cs);
662 bool isInMempool(
const Txid& txid)
override
664 if (!m_node.mempool)
return false;
665 return m_node.mempool->exists(txid);
667 bool hasDescendantsInMempool(
const Txid& txid)
override
669 if (!m_node.mempool)
return false;
670 return m_node.mempool->HasDescendants(txid);
675 std::string& err_string)
override
683 void getTransactionAncestry(
const Txid& txid,
size_t& ancestors,
size_t& cluster_count,
size_t* ancestorsize,
CAmount* ancestorfees)
override
685 ancestors = cluster_count = 0;
686 if (!m_node.mempool)
return;
687 m_node.mempool->GetTransactionAncestry(txid, ancestors, cluster_count, ancestorsize, ancestorfees);
690 std::map<COutPoint, CAmount> calculateIndividualBumpFees(
const std::vector<COutPoint>& outpoints,
const CFeeRate& target_feerate)
override
692 if (!m_node.mempool) {
693 std::map<COutPoint, CAmount> bump_fees;
694 for (
const auto& outpoint : outpoints) {
695 bump_fees.emplace(outpoint, 0);
699 return MiniMiner(*m_node.mempool, outpoints).CalculateBumpFees(target_feerate);
702 std::optional<CAmount> calculateCombinedBumpFee(
const std::vector<COutPoint>& outpoints,
const CFeeRate& target_feerate)
override
704 if (!m_node.mempool) {
707 return MiniMiner(*m_node.mempool, outpoints).CalculateTotalBumpFees(target_feerate);
709 void getPackageLimits(
unsigned int& limit_ancestor_count,
unsigned int& limit_descendant_count)
override
713 const CTxMemPool::Limits& limits{m_node.mempool ? m_node.mempool->m_opts.limits : default_limits};
718 util::Result<void> checkChainLimits(
const CTransactionRef& tx)
override
720 if (!m_node.mempool)
return {};
721 if (!m_node.mempool->CheckPolicyLimits(tx)) {
722 return util::Error{
Untranslated(
"too many unconfirmed transactions in cluster")};
726 CFeeRate estimateSmartFee(
int num_blocks,
bool conservative, FeeCalculation* calc)
override
728 if (!m_node.fee_estimator)
return {};
729 return m_node.fee_estimator->estimateSmartFee(num_blocks, calc, conservative);
731 unsigned int estimateMaxBlocks()
override
733 if (!m_node.fee_estimator)
return 0;
736 CFeeRate mempoolMinFee()
override
738 if (!m_node.mempool)
return {};
739 return m_node.mempool->GetMinFee();
741 CFeeRate relayMinFee()
override
744 return m_node.mempool->m_opts.min_relay_feerate;
746 CFeeRate relayIncrementalFee()
override
749 return m_node.mempool->m_opts.incremental_relay_feerate;
751 CFeeRate relayDustFee()
override
754 return m_node.mempool->m_opts.dust_relay_feerate;
756 bool havePruned()
override
759 return chainman().m_blockman.m_have_pruned;
761 std::optional<int> getPruneHeight()
override
763 LOCK(chainman().GetMutex());
764 return GetPruneHeight(chainman().m_blockman, chainman().ActiveChain());
766 bool isReadyToBroadcast()
override {
return !chainman().m_blockman.LoadingBlocks() && !isInitialBlockDownload(); }
767 bool isInitialBlockDownload()
override
769 return chainman().IsInitialBlockDownload();
772 void initMessage(
const std::string& message)
override {
::uiInterface.InitMessage(message); }
773 void initWarning(
const bilingual_str& message)
override {
InitWarning(message); }
774 void initError(
const bilingual_str& message)
override {
InitError(message); }
775 void showProgress(
const std::string& title,
int progress,
bool resume_possible)
override
777 ::uiInterface.ShowProgress(title, progress, resume_possible);
779 std::unique_ptr<Handler> handleNotifications(std::shared_ptr<Notifications> notifications)
override
781 return std::make_unique<NotificationsHandlerImpl>(validation_signals(), std::move(notifications));
783 void waitForNotificationsIfTipChanged(
const uint256& old_tip)
override
785 if (!old_tip.
IsNull() && old_tip ==
WITH_LOCK(
::cs_main,
return chainman().ActiveChain().Tip()->GetBlockHash()))
return;
786 validation_signals().SyncWithValidationInterfaceQueue();
788 void waitForNotifications()
override
790 validation_signals().SyncWithValidationInterfaceQueue();
792 std::unique_ptr<Handler> handleRpc(
const CRPCCommand&
command)
override
794 return std::make_unique<RpcHandlerImpl>(
command);
801 std::vector<common::SettingsValue> getSettingsList(
const std::string&
name)
override
815 bool updateRwSetting(
const std::string&
name,
818 std::optional<interfaces::SettingsAction> action;
821 action = update_settings_func(*value);
825 action = update_settings_func(new_value);
829 if (!action)
return false;
836 settings = std::move(value);
842 return overwriteRwSetting(
name, {}, action);
844 void requestMempoolTransactions(Notifications& notifications)
override
846 if (!m_node.mempool)
return;
848 for (
const CTxMemPoolEntry& entry : m_node.mempool->entryAll()) {
849 notifications.transactionAddedToMempool(entry.GetSharedTx());
852 bool hasAssumedValidChain()
override
855 return bool{chainman().CurrentChainstate().m_from_snapshot_blockhash};
858 NodeContext* context()
override {
return &m_node; }
859 ArgsManager&
args() {
return *
Assert(m_node.args); }
860 ChainstateManager& chainman() {
return *
Assert(m_node.chainman); }
861 ValidationSignals& validation_signals() {
return *
Assert(m_node.validation_signals); }
865class BlockTemplateImpl :
public BlockTemplate
869 std::unique_ptr<CBlockTemplate> block_template,
870 NodeContext& node) : m_assemble_options(std::move(assemble_options)),
871 m_block_template(std::move(block_template)),
877 CBlockHeader getBlockHeader()
override
879 return m_block_template->block;
882 CBlock getBlock()
override
884 return m_block_template->block;
887 std::vector<CAmount> getTxFees()
override
889 return m_block_template->vTxFees;
892 std::vector<int64_t> getTxSigops()
override
894 return m_block_template->vTxSigOpsCost;
897 CoinbaseTx getCoinbaseTx()
override
899 return m_block_template->m_coinbase_tx;
902 std::vector<uint256> getCoinbaseMerklePath()
override
907 bool submitSolution(uint32_t version, uint32_t timestamp, uint32_t
nonce,
CTransactionRef coinbase)
override
910 return chainman().ProcessNewBlock(std::make_shared<const CBlock>(m_block_template->block),
true,
true,
nullptr);
913 std::unique_ptr<BlockTemplate> waitNext(BlockWaitOptions options)
override
915 auto new_template =
WaitAndCreateNewBlock(chainman(), notifications(), m_node.mempool.get(), m_block_template, options, m_assemble_options, m_interrupt_wait);
916 if (new_template)
return std::make_unique<BlockTemplateImpl>(m_assemble_options, std::move(new_template), m_node);
920 void interruptWait()
override
927 const std::unique_ptr<CBlockTemplate> m_block_template;
929 bool m_interrupt_wait{
false};
930 ChainstateManager& chainman() {
return *
Assert(m_node.chainman); }
931 KernelNotifications& notifications() {
return *
Assert(m_node.notifications); }
935class MinerImpl :
public Mining
938 explicit MinerImpl(NodeContext& node) : m_node(node) {}
940 bool isTestChain()
override
942 return chainman().GetParams().IsTestChain();
945 bool isInitialBlockDownload()
override
947 return chainman().IsInitialBlockDownload();
950 std::optional<BlockRef>
getTip()
override
952 return GetTip(chainman());
955 std::optional<BlockRef> waitTipChanged(uint256 current_tip,
MillisecondsDouble timeout)
override
957 return WaitTipChanged(chainman(), notifications(), current_tip, timeout, m_interrupt_mining);
960 std::unique_ptr<BlockTemplate> createNewBlock(
const BlockCreateOptions& options,
bool cooldown)
override
967 throw std::runtime_error(
strprintf(
"block_reserved_weight (%zu) must be at least %u weight units",
968 *options.block_reserved_weight,
973 std::optional<BlockRef> maybe_tip{waitTipChanged(
uint256::ZERO, MillisecondsDouble::max())};
975 if (!maybe_tip)
return {};
983 while (chainman().IsInitialBlockDownload()) {
985 if (!maybe_tip || chainman().m_interrupt ||
WITH_LOCK(notifications().m_tip_block_mutex,
return m_interrupt_mining))
return {};
994 return std::make_unique<BlockTemplateImpl>(assemble_options, BlockAssembler{chainman().ActiveChainstate(), context()->mempool.get(), assemble_options}.CreateNewBlock(), m_node);
997 void interrupt()
override
1002 bool checkBlock(
const CBlock& block,
const node::BlockCheckOptions& options, std::string& reason, std::string& debug)
override
1004 LOCK(chainman().GetMutex());
1011 NodeContext* context()
override {
return &m_node; }
1012 ChainstateManager& chainman() {
return *
Assert(m_node.chainman); }
1013 KernelNotifications& notifications() {
return *
Assert(m_node.notifications); }
1015 bool m_interrupt_mining{
false};
1016 NodeContext& m_node;
1031 return kernel_notifications.m_state.chainstate_loaded || interrupt;
1033 if (interrupt)
return nullptr;
1035 return std::make_unique<node::MinerImpl>(context);
int64_t CAmount
Amount in satoshis (Can be negative).
auto Join(const C &container, const S &separator, UnaryOp unary_op)
Join all container items.
std::optional< int > GetPruneHeight(const BlockManager &blockman, const CChain &chain)
Return height of highest block that has been pruned, or std::nullopt if no blocks have been pruned.
BlockFilterIndex * GetBlockFilterIndex(BlockFilterType filter_type)
Get a block filter index by type.
CBlockLocator GetLocator(const CBlockIndex *index)
Get a locator for a block index entry.
const CBlockIndex * LastCommonAncestor(const CBlockIndex *pa, const CBlockIndex *pb)
Find the last common ancestor two blocks have.
@ BLOCK_HAVE_DATA
full block available in blk*.dat
const CChainParams & Params()
Return the currently selected parameters.
#define Assert(val)
Identity function.
common::SettingsValue GetSetting(const std::string &arg) const
Get setting value.
void LockSettings(Fn &&fn)
Access settings with lock held.
bool WriteSettingsFile(std::vector< std::string > *errors=nullptr, bool backup=false) const
Write settings file or backup settings file.
std::vector< common::SettingsValue > GetSettingsList(const std::string &arg) const
Get list of setting values.
common::SettingsValue GetPersistentSetting(const std::string &name) const
Get current setting from config file or read/write settings file, ignoring nonpersistent command line...
std::string GetArg(const std::string &strArg, const std::string &strDefault) const
Return string argument or default value.
CategoryMask GetCategoryMask() const
const GCSFilter & GetFilter() const LIFETIMEBOUND
bool LookupFilter(const CBlockIndex *block_index, BlockFilter &filter_out) const
Get a single filter by block.
CBlockIndex * pprev
pointer to the index of the predecessor of this block
uint256 GetBlockHash() const
int64_t GetBlockTime() const
int64_t GetMedianTimePast() const
int64_t GetBlockTimeMax() const
CBlockIndex * GetAncestor(int height)
Efficiently find an ancestor of this block.
int nHeight
height of the entry in the chain. The genesis block has height 0
CBlockIndex * FindEarliestAtLeast(int64_t nTime, int height) const
Find the earliest block with timestamp equal or greater than the given time and height equal or great...
bool removeCommand(const std::string &name, const CRPCCommand *pcmd)
void appendCommand(const std::string &name, const CRPCCommand *pcmd)
Appends a CRPCCommand to the dispatch table.
kernel::MemPoolLimits Limits
static bool Enumerate(const std::string &command, std::vector< ExternalSigner > &signers, const std::string &chain)
std::unordered_set< Element, ByteVectorHash > ElementSet
bool MatchAny(const ElementSet &elements) const
Checks if any of the given elements may be in the set.
void RegisterSharedValidationInterface(std::shared_ptr< CValidationInterface > callbacks)
Register subscriber.
std::string GetRejectReason() const
std::string GetDebugMessage() const
constexpr bool IsNull() const
Block template interface.
Top-level interface for a bitcoin node (bitcoind process).
Generate a new block, without valid proof-of-work.
Maintains a tree of blocks (stored in m_block_index) which is consulted to determine where the most-w...
static const uint256 ZERO
Helper class that manages an interrupt flag, and allows a thread or signal to interrupt another threa...
std::vector< uint256 > TransactionMerklePath(const CBlock &block, uint32_t position)
Compute merkle path to the specified transaction.
RecursiveMutex cs_main
Mutex to guard access to validation specific variables, such as reading or changing the chainstate.
static CService ip(uint32_t i)
void InitLogging(const ArgsManager &args)
Initialize global loggers.
bool AppInitBasicSetup(const ArgsManager &args, std::atomic< int > &exit_status)
Initialize bitcoin core: Basic context setup.
bool ShutdownRequested(node::NodeContext &node)
Return whether node shutdown was requested.
bool AppInitParameterInteraction(const ArgsManager &args)
Initialization: parameter interaction.
bool AppInitInterfaces(NodeContext &node)
Initialize node and wallet interface pointers.
void InitParameterInteraction(ArgsManager &args)
Parameter interaction: change current parameters depending on various rules.
bool AppInitMain(NodeContext &node, interfaces::BlockAndHeaderTipInfo *tip_info)
Bitcoin core main initialization.
bool AppInitLockDirectories()
Lock bitcoin core critical directories.
bool AppInitSanityChecks(const kernel::Context &kernel)
Initialization sanity checks.
CClientUIInterface uiInterface
void InitWarning(const bilingual_str &str)
Show warning message.
bool InitError(const bilingual_str &str)
Show error message.
BCLog::Logger & LogInstance()
void StartMapPort(bool enable)
MemPoolRemovalReason
Reason why a transaction was removed from the mempool, this is passed to the notification signal.
auto FindKey(Map &&map, Key &&key) -> decltype(&map.at(key))
Map lookup helper.
std::unique_ptr< Node > MakeNode(node::NodeContext &context)
Return implementation of Node interface.
std::unique_ptr< Handler > MakeSignalHandler(boost::signals2::connection connection)
Return handler wrapping a boost signal connection.
std::unique_ptr< Mining > MakeMining(node::NodeContext &node, bool wait_loaded=true)
std::function< std::optional< interfaces::SettingsAction >(common::SettingsValue &)> SettingsUpdate
std::unique_ptr< Chain > MakeChain(node::NodeContext &node)
Return implementation of Chain interface.
interfaces::BlockInfo MakeBlockInfo(const CBlockIndex *index, const CBlock *data)
Return data from block index.
void FindCoins(const NodeContext &node, std::map< COutPoint, Coin > &coins)
Look up unspent output information.
std::optional< BlockRef > WaitTipChanged(ChainstateManager &chainman, KernelNotifications &kernel_notifications, const uint256 ¤t_tip, MillisecondsDouble &timeout, bool &interrupt)
TxBroadcast
How to broadcast a local transaction.
@ MEMPOOL_AND_BROADCAST_TO_ALL
Add the transaction to the mempool and broadcast to all peers for which tx relay is enabled.
util::Result< void > ApplyArgsManOptions(const ArgsManager &args, BlockManager::Options &opts)
std::unique_ptr< CBlockTemplate > WaitAndCreateNewBlock(ChainstateManager &chainman, KernelNotifications &kernel_notifications, CTxMemPool *mempool, const std::unique_ptr< CBlockTemplate > &block_template, const BlockWaitOptions &options, const BlockAssembler::Options &assemble_options, bool &interrupt_wait)
Return a new block template when fees rise to a certain threshold or after a new tip; return nullopt ...
TransactionError BroadcastTransaction(NodeContext &node, const CTransactionRef tx, std::string &err_string, const CAmount &max_tx_fee, TxBroadcast broadcast_method, bool wait_callback)
Submit a transaction to the mempool and (optionally) relay it to all P2P peers.
void InterruptWait(KernelNotifications &kernel_notifications, bool &interrupt_wait)
void AddMerkleRootAndCoinbase(CBlock &block, CTransactionRef coinbase, uint32_t version, uint32_t timestamp, uint32_t nonce)
bool CooldownIfHeadersAhead(ChainstateManager &chainman, KernelNotifications &kernel_notifications, const BlockRef &last_tip, bool &interrupt_mining)
Wait while the best known header extends the current chain tip AND at least one block is being added ...
std::optional< BlockRef > GetTip(ChainstateManager &chainman)
auto Join(const C &container, const S &separator, UnaryOp unary_op)
Join all container items.
std::map< CSubNet, CBanEntry > banmap_t
bool GetProxy(enum Network net, Proxy &proxyInfoOut)
std::unique_ptr< Handler > MakeSignalHandler(boost::signals2::connection connection)
Return handler wrapping a boost signal connection.
RBFTransactionState IsRBFOptInEmptyMempool(const CTransaction &tx)
RBFTransactionState IsRBFOptIn(const CTransaction &tx, const CTxMemPool &pool)
Determine whether an unconfirmed transaction is signaling opt-in to RBF according to BIP 125 This inv...
RBFTransactionState
The rbf state of unconfirmed transactions.
static constexpr unsigned int DEFAULT_INCREMENTAL_RELAY_FEE
Default for -incrementalrelayfee, which sets the minimum feerate increase for mempool limiting or rep...
static constexpr unsigned int DUST_RELAY_TX_FEE
Min feerate for defining dust.
static constexpr unsigned int MINIMUM_BLOCK_RESERVED_WEIGHT
This accounts for the block header, var_int encoding of the transaction count and a minimally viable ...
static constexpr unsigned int DEFAULT_MIN_RELAY_TX_FEE
Default for -minrelaytxfee, minimum relay fee for transactions.
std::shared_ptr< const CTransaction > CTransactionRef
@ RPC_WALLET_NOT_FOUND
Invalid wallet specified.
bool IsDeprecatedRPCEnabled(const std::string &method)
std::function< bool()> shutdown_request
Function to request a shutdown.
const CTransactionRef m_tx
std::map< std::string, SettingsValue > forced_settings
Map of setting name to forced setting value.
std::map< std::string, SettingsValue > rw_settings
Map of setting name to read-write file setting value.
std::map< std::string, std::vector< SettingsValue > > command_line_options
Map of setting name to list of command line values.
Hash/height pair to help track and identify blocks.
Block tip (could be a header or not, depends on the subscribed signal).
Information about chainstate that notifications are sent from.
int64_t descendant_count
The maximum allowed number of transactions in a package including the entry and its descendants.
int64_t ancestor_count
The maximum allowed number of transactions in a package including the entry and its ancestors.
bool check_pow
Set false to omit the proof-of-work check.
bool check_merkle_root
Set false to omit the merkle root check.
Template containing all coinbase transaction fields that are set by our miner code.
std::unique_ptr< KernelNotifications > notifications
Issues blocking calls about sync status, errors and warnings.
util::SignalInterrupt * shutdown_signal
Interrupt object used to track whether node shutdown was requested.
#define WAIT_LOCK(cs, name)
#define REVERSE_LOCK(g, cs)
#define WITH_LOCK(cs, code)
#define TRY_LOCK(cs, name)
#define EXCLUSIVE_LOCKS_REQUIRED(...)
transaction_identifier< false > Txid
Txid commits to all transaction fields except the witness.
bilingual_str Untranslated(std::string original)
Mark a bilingual_str as untranslated.
std::chrono::duration< double, std::chrono::milliseconds::period > MillisecondsDouble
BlockValidationState TestBlockValidity(Chainstate &chainstate, const CBlock &block, const bool check_pow, const bool check_merkle_root)
Verify a block, including transactions.
SynchronizationState
Current sync state passed to tip changed callbacks.
TipBlock getTip(const CChainParams ¶ms, const node::NodeContext &context)