8#include <chainparams.h>
40#include <validation.h>
44#if defined(HAVE_CONFIG_H)
45#include <config/bitcoin-config.h>
50#include <boost/signals2/signal.hpp>
68 class NodeImpl :
public Node {
73 explicit NodeImpl(NodeContext *context) { setContext(context); }
75 void initParameterInteraction()
override {
79 int getExitStatus()
override {
82 bool baseInitialize(
Config &config)
override {
90 m_context->kernel = std::make_unique<kernel::Context>();
107 if (
AppInitMain(config, rpcServer, httpRPCRequestProcessor,
112 m_context->exit_status.store(EXIT_FAILURE);
115 void appShutdown()
override {
119 void startShutdown()
override {
129 bool isPersistentSettingIgnored(
const std::string &
name)
override {
130 bool ignored =
false;
134 ignored = !options->empty();
140 getPersistentSetting(
const std::string &
name)
override {
143 void updateRwSetting(
const std::string &
name,
147 settings.rw_settings.erase(name);
149 settings.rw_settings[name] = value;
154 void forceSetting(
const std::string &
name,
158 settings.forced_settings.erase(name);
160 settings.forced_settings[name] = value;
164 void resetSettings()
override {
171 void mapPort(
bool use_upnp,
bool use_natpmp)
override {
174 bool getProxy(
Network net,
Proxy &proxy_info)
override {
181 bool getNodesStats(NodesStats &stats)
override {
185 std::vector<CNodeStats> stats_temp;
186 m_context->connman->GetNodeStats(stats_temp);
188 stats.reserve(stats_temp.size());
189 for (
auto &node_stats_temp : stats_temp) {
190 stats.emplace_back(std::move(node_stats_temp),
false,
198 for (
auto &node_stats : stats) {
199 std::get<1>(node_stats) =
201 std::get<0>(node_stats).nodeid,
202 std::get<2>(node_stats));
210 bool getBanned(
banmap_t &banmap)
override {
217 bool ban(
const CNetAddr &net_addr, int64_t ban_time_offset)
override {
219 m_context->banman->Ban(net_addr, ban_time_offset);
224 bool unban(
const CSubNet &ip)
override {
231 bool disconnectByAddress(
const CNetAddr &net_addr)
override {
233 return m_context->connman->DisconnectNode(net_addr);
237 bool disconnectById(
NodeId id)
override {
239 return m_context->connman->DisconnectNode(
id);
243 int64_t getTotalBytesRecv()
override {
247 int64_t getTotalBytesSent()
override {
251 size_t getMempoolSize()
override {
254 size_t getMempoolDynamicUsage()
override {
258 bool getHeaderTip(
int &height, int64_t &block_time)
override {
260 auto best_header = chainman().m_best_header;
262 height = best_header->nHeight;
263 block_time = best_header->GetBlockTime();
268 int getNumBlocks()
override {
270 return chainman().ActiveChain().Height();
276 : chainman().GetParams().GenesisBlock().GetHash();
278 int64_t getLastBlockTime()
override {
280 if (chainman().ActiveChain().Tip()) {
281 return chainman().ActiveChain().Tip()->GetBlockTime();
284 return chainman().GetParams().GenesisBlock().GetBlockTime();
286 double getVerificationProgress()
override {
290 tip = chainman().ActiveChain().Tip();
295 bool isInitialBlockDownload()
override {
296 return chainman().IsInitialBlockDownload();
298 bool isLoadingBlocks()
override {
299 return chainman().m_blockman.LoadingBlocks();
301 void setNetworkActive(
bool active)
override {
303 m_context->connman->SetNetworkActive(active);
306 bool getNetworkActive()
override {
309 CFeeRate getDustRelayFee()
override {
313 return m_context->mempool->m_dust_relay_feerate;
315 UniValue executeRpc(
const Config &config,
const std::string &command,
317 const std::string &uri)
override {
325 std::vector<std::string> listRpcCommands()
override {
334 bool getUnspentOutput(
const COutPoint &output,
Coin &coin)
override {
336 return chainman().ActiveChainstate().CoinsTip().GetCoin(output,
342 std::unique_ptr<Handler> handleInitMessage(InitMessageFn fn)
override {
345 std::unique_ptr<Handler> handleMessageBox(MessageBoxFn fn)
override {
348 std::unique_ptr<Handler> handleQuestion(QuestionFn fn)
override {
351 std::unique_ptr<Handler>
352 handleShowProgress(ShowProgressFn fn)
override {
355 std::unique_ptr<Handler> handleNotifyNumConnectionsChanged(
356 NotifyNumConnectionsChangedFn fn)
override {
360 std::unique_ptr<Handler> handleNotifyNetworkActiveChanged(
361 NotifyNetworkActiveChangedFn fn)
override {
365 std::unique_ptr<Handler>
366 handleNotifyAlertChanged(NotifyAlertChangedFn fn)
override {
369 std::unique_ptr<Handler>
370 handleBannedListChanged(BannedListChangedFn fn)
override {
373 std::unique_ptr<Handler>
374 handleNotifyBlockTip(NotifyBlockTipFn fn)
override {
384 std::unique_ptr<Handler>
385 handleNotifyHeaderTip(NotifyHeaderTipFn fn)
override {
389 int64_t timestamp,
bool presync) {
394 NodeContext *context()
override {
return m_context; }
395 void setContext(NodeContext *context)
override {
m_context = context; }
401 const BlockManager &blockman) {
427 FillBlock(active[index->
nHeight] == index
434 if (!blockman.ReadBlock(*block.
m_data, *index)) {
443 explicit NotificationsProxy(
444 std::shared_ptr<Chain::Notifications> notifications)
446 virtual ~NotificationsProxy() =
default;
448 std::shared_ptr<
const std::vector<Coin>>,
449 uint64_t mempool_sequence)
override {
454 uint64_t mempool_sequence)
override {
459 const std::shared_ptr<const CBlock> &block,
463 void BlockDisconnected(
const std::shared_ptr<const CBlock> &block,
469 bool is_ibd)
override {
479 class NotificationsHandlerImpl :
public Handler {
481 explicit NotificationsHandlerImpl(
482 std::shared_ptr<Chain::Notifications> notifications)
483 :
m_proxy(std::make_shared<NotificationsProxy>(
484 std::move(notifications))) {
487 ~NotificationsHandlerImpl()
override { disconnect(); }
488 void disconnect()
override {
497 class RpcHandlerImpl :
public Handler {
499 explicit RpcHandlerImpl(
const CRPCCommand &command)
503 UniValue &result,
bool last_handler) {
528 void disconnect() final {
535 ~RpcHandlerImpl()
override { disconnect(); }
541 class ChainImpl :
public Chain {
548 std::optional<int> getHeight()
override {
551 int height = active.
Height();
557 BlockHash getBlockHash(
int height)
override {
564 bool haveBlockOnDisk(
int height)
override {
568 return block && (block->nStatus.hasData() != 0) && block->
nTx > 0;
583 return fork->nHeight;
591 return FillBlock(
m_node.chainman->m_blockman.LookupBlockIndex(hash),
592 block, lock, active, chainman().m_blockman);
594 bool findFirstBlockWithTimeAndHeight(int64_t min_time,
int min_height,
599 block, lock, active, chainman().m_blockman);
601 bool findAncestorByHeight(
const BlockHash &block_hash,
607 m_node.chainman->m_blockman.LookupBlockIndex(block_hash)) {
609 block->GetAncestor(ancestor_height)) {
610 return FillBlock(ancestor, ancestor_out, lock, active,
611 chainman().m_blockman);
614 return FillBlock(
nullptr, ancestor_out, lock, active,
615 chainman().m_blockman);
617 bool findAncestorByHash(
const BlockHash &block_hash,
623 m_node.chainman->m_blockman.LookupBlockIndex(block_hash);
625 m_node.chainman->m_blockman.LookupBlockIndex(ancestor_hash);
626 if (block && ancestor &&
630 return FillBlock(ancestor, ancestor_out, lock, active,
631 chainman().m_blockman);
633 bool findCommonAncestor(
const BlockHash &block_hash1,
641 m_node.chainman->m_blockman.LookupBlockIndex(block_hash1);
643 m_node.chainman->m_blockman.LookupBlockIndex(block_hash2);
649 return int{FillBlock(ancestor, ancestor_out, lock, active,
650 chainman().m_blockman)} &
651 int{FillBlock(block1, block1_out, lock, active,
652 chainman().m_blockman)} &
653 int{FillBlock(block2, block2_out, lock, active,
654 chainman().m_blockman)};
656 void findCoins(std::map<COutPoint, Coin> &coins)
override {
659 double guessVerificationProgress(
const BlockHash &block_hash)
override {
662 chainman().GetParams().TxData(),
663 chainman().m_blockman.LookupBlockIndex(block_hash));
665 bool hasBlocks(
const BlockHash &block_hash,
int min_height,
666 std::optional<int> max_height)
override {
676 chainman().m_blockman.LookupBlockIndex(block_hash)) {
677 if (max_height && block->
nHeight >= *max_height) {
680 for (; block->nStatus.hasData(); block = block->
pprev) {
689 bool broadcastTransaction(
const Config &config,
691 const Amount &max_tx_fee,
bool relay,
692 std::string &err_string)
override {
703 CFeeRate estimateFee()
const override {
707 return m_node.mempool->estimateFee();
713 return m_node.mempool->m_min_relay_feerate;
719 return m_node.mempool->m_dust_relay_feerate;
721 bool havePruned()
override {
723 return m_node.chainman->m_blockman.m_have_pruned;
725 bool isReadyToBroadcast()
override {
726 return !chainman().m_blockman.LoadingBlocks() &&
727 !isInitialBlockDownload();
729 std::optional<int> getPruneHeight()
override {
730 LOCK(chainman().GetMutex());
732 chainman().ActiveChain());
734 bool isInitialBlockDownload()
override {
735 return chainman().IsInitialBlockDownload();
738 void initMessage(
const std::string &message)
override {
747 void showProgress(
const std::string &title,
int progress,
748 bool resume_possible)
override {
749 ::uiInterface.ShowProgress(title, progress, resume_possible);
751 std::unique_ptr<Handler> handleNotifications(
752 std::shared_ptr<Notifications> notifications)
override {
753 return std::make_unique<NotificationsHandlerImpl>(
754 std::move(notifications));
757 waitForNotificationsIfTipChanged(
const BlockHash &old_tip)
override {
768 std::unique_ptr<Handler>
770 return std::make_unique<RpcHandlerImpl>(command);
772 bool rpcEnableDeprecated(
const std::string &method)
override {
775 void rpcRunLater(
const std::string &
name, std::function<
void()> fn,
776 int64_t seconds)
override {
782 std::vector<util::SettingsValue>
783 getSettingsList(
const std::string &
name)
override {
796 bool updateRwSetting(
const std::string &
name,
798 bool write)
override {
801 settings.rw_settings.erase(name);
803 settings.rw_settings[name] = value;
808 void requestMempoolTransactions(Notifications ¬ifications)
override {
814 notifications.transactionAddedToMempool(entry->GetSharedTx(),
818 bool hasAssumedValidChain()
override {
822 NodeContext *context()
override {
return &
m_node; }
831 return std::make_unique<node::NodeImpl>(context);
835 return std::make_unique<node::ChainImpl>(
node, params);
std::optional< int > GetPruneHeight(const BlockManager &blockman, const CChain &chain)
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.
const CChainParams & Params()
Return the currently selected parameters.
#define Assert(val)
Identity function.
util::SettingsValue GetPersistentSetting(const std::string &name) const
Get current setting from config file or read/write settings file, ignoring nonpersistent command line...
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< util::SettingsValue > GetSettingsList(const std::string &arg) const
Get list of setting values.
util::SettingsValue GetSetting(const std::string &arg) const
Get setting value.
bool GetBoolArg(const std::string &strArg, bool fDefault) const
Return boolean argument or default value.
The block chain is a tree shaped structure starting with the genesis block at the root,...
CBlockIndex * pprev
pointer to the index of the predecessor of this block
int64_t GetBlockTime() const
int64_t GetMedianTimePast() const
int64_t GetBlockTimeMax() const
unsigned int nTx
Number of transactions in this block.
CBlockIndex * GetAncestor(int height)
Efficiently find an ancestor of this block.
BlockHash GetBlockHash() const
int nHeight
height of the entry in the chain. The genesis block has height 0
An in-memory indexed chain of blocks.
CBlockIndex * Tip() const
Returns the index entry for the tip of this chain, or nullptr if none.
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...
int Height() const
Return the maximal height in the chain.
CBlockLocator GetLocator() const
Return a CBlockLocator that refers to the tip of this chain.
CChainParams defines various tweakable parameters of a given instance of the Bitcoin system.
Fee rate in satoshis per kilobyte: Amount / kB.
bool removeCommand(const std::string &name, const CRPCCommand *pcmd)
std::vector< std::string > listCommands() const
Returns a list of registered commands.
UniValue execute(const Config &config, const JSONRPCRequest &request) const
Execute a method.
void appendCommand(const std::string &name, const CRPCCommand *pcmd)
Appends a CRPCCommand to the dispatch table.
Implement this to subscribe to events generated in validation.
Chainstate stores and provides an API to update our local knowledge of the current best chain.
void ClearBlockIndexCandidates() EXCLUSIVE_LOCKS_REQUIRED(const CBlockIndex * FindForkInGlobalIndex(const CBlockLocator &locator) const EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Find the last common block of this chain and a locator.
Provides an interface for creating and interacting with one or two chainstates: an IBD chainstate gen...
Class for registering and managing all RPC calls.
Wrapper around std::unique_lock style lock for Mutex.
Interface giving clients (wallet processes, maybe other analysis tools in the future) ability to acce...
Helper for findBlock to selectively return pieces of block data.
const FoundBlock * m_next_block
CBlockLocator * m_locator
Generic interface for managing an event handler or callback function registered with another interfac...
Top-level interface for a bitcoin node (bitcoind process).
Wallet chain client that in addition to having chain client methods for starting up,...
RecursiveMutex cs_main
Mutex to guard access to validation specific variables, such as reading or changing the chainstate.
void Interrupt(NodeContext &node)
Interrupt threads.
void InitLogging(const ArgsManager &args)
Initialize global loggers.
bool AppInitLockDataDirectory()
Lock bitcoin data directory.
void Shutdown(NodeContext &node)
bool AppInitBasicSetup(const ArgsManager &args, std::atomic< int > &exit_status)
Initialize bitcoin: Basic context setup.
bool AppInitMain(Config &config, RPCServer &rpcServer, HTTPRPCRequestProcessor &httpRPCRequestProcessor, NodeContext &node, interfaces::BlockAndHeaderTipInfo *tip_info)
Bitcoin main initialization.
bool AppInitInterfaces(NodeContext &node)
Initialize node and wallet interface pointers.
void InitParameterInteraction(ArgsManager &args)
Parameter interaction: change current parameters depending on various rules.
bool AppInitParameterInteraction(Config &config, const ArgsManager &args)
Initialization: parameter interaction.
bool AppInitSanityChecks(const kernel::Context &kernel)
Initialization sanity checks.
ChainstateRole
This enum describes the various roles a specific Chainstate instance can take.
void StartMapPort(bool use_upnp, bool use_natpmp)
std::unique_ptr< Handler > MakeHandler(boost::signals2::connection connection)
Return handler wrapping a boost signal connection.
std::unique_ptr< Node > MakeNode(node::NodeContext *context)
Return implementation of Node interface.
std::unique_ptr< Chain > MakeChain(node::NodeContext &node, const CChainParams ¶ms)
Return implementation of Chain interface.
TransactionError BroadcastTransaction(const NodeContext &node, const CTransactionRef tx, std::string &err_string, const Amount max_tx_fee, bool relay, bool wait_callback)
Submit a transaction to the mempool and (optionally) relay it to all P2P peers.
void FindCoins(const NodeContext &node, std::map< COutPoint, Coin > &coins)
Look up unspent output information.
Implement std::hash so RCUPtr can be used as a key for maps or sets.
auto FindKey(Map &&map, Key &&key) -> decltype(&map.at(key))
Map lookup helper.
std::map< CSubNet, CBanEntry > banmap_t
bool GetProxy(enum Network net, Proxy &proxyInfoOut)
std::shared_ptr< Chain::Notifications > m_notifications
const CChainParams & m_params
const CRPCCommand * m_wrapped_command
std::shared_ptr< NotificationsProxy > m_proxy
is a home for public enum and struct type definitions that are used by internally by node code,...
static constexpr Amount DUST_RELAY_TX_FEE(1000 *SATOSHI)
Min feerate for defining dust.
static constexpr Amount DEFAULT_MIN_RELAY_TX_FEE_PER_KB(1000 *SATOSHI)
Default for -minrelaytxfee, minimum relay fee for transactions.
std::shared_ptr< const CTransaction > CTransactionRef
@ RPC_WALLET_NOT_FOUND
Invalid wallet specified.
void RPCSetTimerInterfaceIfUnset(RPCTimerInterface *iface)
Set the factory function for timer, but only, if unset.
bool IsDeprecatedRPCEnabled(const ArgsManager &args, const std::string &method)
void RPCUnsetTimerInterface(RPCTimerInterface *iface)
Unset factory function for timers.
void RPCRunLater(const std::string &name, std::function< void()> func, int64_t nSeconds)
Run func nSeconds from now.
bool ShutdownRequested()
Returns true if a shutdown is requested, false otherwise.
void StartShutdown()
Request shutdown of the application.
A BlockHash is a unqiue identifier for a block.
Describes a place in the block chain to another node such that if the other node doesn't have the sam...
Block and header tip information.
Block tip (could be a header or not, depends on the subscribed signal).
NodeContext struct containing references to chain state and connection state.
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.
#define WAIT_LOCK(cs, name)
#define WITH_LOCK(cs, code)
Run code while locking a mutex.
#define TRY_LOCK(cs, name)
MemPoolRemovalReason
Reason why a transaction was removed from the mempool, this is passed to the notification signal.
CClientUIInterface uiInterface
void InitWarning(const bilingual_str &str)
Show warning message.
bool InitError(const bilingual_str &str)
Show error message.
double GuessVerificationProgress(const ChainTxData &data, const CBlockIndex *pindex)
Guess how far we are in the verification process at the given block index require cs_main if pindex h...
SynchronizationState
Current sync state passed to tip changed callbacks.
void UnregisterSharedValidationInterface(std::shared_ptr< CValidationInterface > callbacks)
Unregister subscriber.
void SyncWithValidationInterfaceQueue()
This is a synonym for the following, which asserts certain locks are not held: std::promise<void> pro...
void RegisterSharedValidationInterface(std::shared_ptr< CValidationInterface > callbacks)
Register subscriber.
bilingual_str GetWarnings(bool verbose)
Format a string that describes several potential problems detected by the core.