8#include <chainparams.h>
39#include <validation.h>
43#if defined(HAVE_CONFIG_H)
44#include <config/bitcoin-config.h>
49#include <boost/signals2/signal.hpp>
67 class NodeImpl :
public Node {
72 explicit NodeImpl(NodeContext *context) { setContext(context); }
74 void initParameterInteraction()
override {
78 int getExitStatus()
override {
81 bool baseInitialize(
Config &config)
override {
89 m_context->kernel = std::make_unique<kernel::Context>();
106 if (
AppInitMain(config, rpcServer, httpRPCRequestProcessor,
111 m_context->exit_status.store(EXIT_FAILURE);
114 void appShutdown()
override {
118 void startShutdown()
override {
128 bool isPersistentSettingIgnored(
const std::string &
name)
override {
129 bool ignored =
false;
133 ignored = !options->empty();
139 getPersistentSetting(
const std::string &
name)
override {
142 void updateRwSetting(
const std::string &
name,
146 settings.rw_settings.erase(name);
148 settings.rw_settings[name] = value;
153 void forceSetting(
const std::string &
name,
157 settings.forced_settings.erase(name);
159 settings.forced_settings[name] = value;
163 void resetSettings()
override {
170 void mapPort(
bool use_upnp,
bool use_natpmp)
override {
180 bool getNodesStats(NodesStats &stats)
override {
184 std::vector<CNodeStats> stats_temp;
185 m_context->connman->GetNodeStats(stats_temp);
187 stats.reserve(stats_temp.size());
188 for (
auto &node_stats_temp : stats_temp) {
189 stats.emplace_back(std::move(node_stats_temp),
false,
197 for (
auto &node_stats : stats) {
198 std::get<1>(node_stats) =
200 std::get<0>(node_stats).nodeid,
201 std::get<2>(node_stats));
209 bool getBanned(
banmap_t &banmap)
override {
216 bool ban(
const CNetAddr &net_addr, int64_t ban_time_offset)
override {
218 m_context->banman->Ban(net_addr, ban_time_offset);
223 bool unban(
const CSubNet &ip)
override {
230 bool disconnectByAddress(
const CNetAddr &net_addr)
override {
232 return m_context->connman->DisconnectNode(net_addr);
236 bool disconnectById(
NodeId id)
override {
238 return m_context->connman->DisconnectNode(
id);
242 int64_t getTotalBytesRecv()
override {
246 int64_t getTotalBytesSent()
override {
250 size_t getMempoolSize()
override {
253 size_t getMempoolDynamicUsage()
override {
257 bool getHeaderTip(
int &height, int64_t &block_time)
override {
259 auto best_header = chainman().m_best_header;
261 height = best_header->nHeight;
262 block_time = best_header->GetBlockTime();
267 int getNumBlocks()
override {
269 return chainman().ActiveChain().Height();
275 : chainman().GetParams().GenesisBlock().GetHash();
277 int64_t getLastBlockTime()
override {
279 if (chainman().ActiveChain().Tip()) {
280 return chainman().ActiveChain().Tip()->GetBlockTime();
283 return chainman().GetParams().GenesisBlock().GetBlockTime();
285 double getVerificationProgress()
override {
289 tip = chainman().ActiveChain().Tip();
294 bool isInitialBlockDownload()
override {
295 return chainman().IsInitialBlockDownload();
297 bool isLoadingBlocks()
override {
298 return chainman().m_blockman.LoadingBlocks();
300 void setNetworkActive(
bool active)
override {
302 m_context->connman->SetNetworkActive(active);
305 bool getNetworkActive()
override {
308 CFeeRate getDustRelayFee()
override {
312 return m_context->mempool->m_dust_relay_feerate;
314 UniValue executeRpc(
const Config &config,
const std::string &command,
316 const std::string &uri)
override {
324 std::vector<std::string> listRpcCommands()
override {
333 bool getUnspentOutput(
const COutPoint &output,
Coin &coin)
override {
335 return chainman().ActiveChainstate().CoinsTip().GetCoin(output,
341 std::unique_ptr<Handler> handleInitMessage(InitMessageFn fn)
override {
344 std::unique_ptr<Handler> handleMessageBox(MessageBoxFn fn)
override {
347 std::unique_ptr<Handler> handleQuestion(QuestionFn fn)
override {
350 std::unique_ptr<Handler>
351 handleShowProgress(ShowProgressFn fn)
override {
354 std::unique_ptr<Handler> handleNotifyNumConnectionsChanged(
355 NotifyNumConnectionsChangedFn fn)
override {
359 std::unique_ptr<Handler> handleNotifyNetworkActiveChanged(
360 NotifyNetworkActiveChangedFn fn)
override {
364 std::unique_ptr<Handler>
365 handleNotifyAlertChanged(NotifyAlertChangedFn fn)
override {
368 std::unique_ptr<Handler>
369 handleBannedListChanged(BannedListChangedFn fn)
override {
372 std::unique_ptr<Handler>
373 handleNotifyBlockTip(NotifyBlockTipFn fn)
override {
383 std::unique_ptr<Handler>
384 handleNotifyHeaderTip(NotifyHeaderTipFn fn)
override {
388 int64_t timestamp,
bool presync) {
393 NodeContext *context()
override {
return m_context; }
394 void setContext(NodeContext *context)
override {
m_context = context; }
400 const BlockManager &blockman) {
426 FillBlock(active[index->
nHeight] == index
433 if (!blockman.ReadBlockFromDisk(*block.
m_data, *index)) {
442 explicit NotificationsProxy(
443 std::shared_ptr<Chain::Notifications> notifications)
445 virtual ~NotificationsProxy() =
default;
447 std::shared_ptr<
const std::vector<Coin>>,
448 uint64_t mempool_sequence)
override {
453 uint64_t mempool_sequence)
override {
458 const std::shared_ptr<const CBlock> &block,
462 void BlockDisconnected(
const std::shared_ptr<const CBlock> &block,
468 bool is_ibd)
override {
478 class NotificationsHandlerImpl :
public Handler {
480 explicit NotificationsHandlerImpl(
481 std::shared_ptr<Chain::Notifications> notifications)
482 :
m_proxy(std::make_shared<NotificationsProxy>(
483 std::move(notifications))) {
486 ~NotificationsHandlerImpl()
override { disconnect(); }
487 void disconnect()
override {
496 class RpcHandlerImpl :
public Handler {
498 explicit RpcHandlerImpl(
const CRPCCommand &command)
502 UniValue &result,
bool last_handler) {
527 void disconnect() final {
534 ~RpcHandlerImpl()
override { disconnect(); }
540 class ChainImpl :
public Chain {
547 std::optional<int> getHeight()
override {
550 int height = active.
Height();
556 BlockHash getBlockHash(
int height)
override {
563 bool haveBlockOnDisk(
int height)
override {
567 return block && (block->nStatus.hasData() != 0) && block->
nTx > 0;
582 return fork->nHeight;
590 return FillBlock(
m_node.chainman->m_blockman.LookupBlockIndex(hash),
591 block, lock, active, chainman().m_blockman);
593 bool findFirstBlockWithTimeAndHeight(int64_t min_time,
int min_height,
598 block, lock, active, chainman().m_blockman);
600 bool findAncestorByHeight(
const BlockHash &block_hash,
606 m_node.chainman->m_blockman.LookupBlockIndex(block_hash)) {
608 block->GetAncestor(ancestor_height)) {
609 return FillBlock(ancestor, ancestor_out, lock, active,
610 chainman().m_blockman);
613 return FillBlock(
nullptr, ancestor_out, lock, active,
614 chainman().m_blockman);
616 bool findAncestorByHash(
const BlockHash &block_hash,
622 m_node.chainman->m_blockman.LookupBlockIndex(block_hash);
624 m_node.chainman->m_blockman.LookupBlockIndex(ancestor_hash);
625 if (block && ancestor &&
629 return FillBlock(ancestor, ancestor_out, lock, active,
630 chainman().m_blockman);
632 bool findCommonAncestor(
const BlockHash &block_hash1,
640 m_node.chainman->m_blockman.LookupBlockIndex(block_hash1);
642 m_node.chainman->m_blockman.LookupBlockIndex(block_hash2);
648 return int{FillBlock(ancestor, ancestor_out, lock, active,
649 chainman().m_blockman)} &
650 int{FillBlock(block1, block1_out, lock, active,
651 chainman().m_blockman)} &
652 int{FillBlock(block2, block2_out, lock, active,
653 chainman().m_blockman)};
655 void findCoins(std::map<COutPoint, Coin> &coins)
override {
658 double guessVerificationProgress(
const BlockHash &block_hash)
override {
661 chainman().GetParams().TxData(),
662 chainman().m_blockman.LookupBlockIndex(block_hash));
664 bool hasBlocks(
const BlockHash &block_hash,
int min_height,
665 std::optional<int> max_height)
override {
675 chainman().m_blockman.LookupBlockIndex(block_hash)) {
676 if (max_height && block->
nHeight >= *max_height) {
679 for (; block->nStatus.hasData(); block = block->
pprev) {
688 bool broadcastTransaction(
const Config &config,
690 const Amount &max_tx_fee,
bool relay,
691 std::string &err_string)
override {
702 CFeeRate estimateFee()
const override {
706 return m_node.mempool->estimateFee();
712 return m_node.mempool->m_min_relay_feerate;
718 return m_node.mempool->m_dust_relay_feerate;
720 bool havePruned()
override {
722 return m_node.chainman->m_blockman.m_have_pruned;
724 bool isReadyToBroadcast()
override {
725 return !chainman().m_blockman.LoadingBlocks() &&
726 !isInitialBlockDownload();
728 std::optional<int> getPruneHeight()
override {
729 LOCK(chainman().GetMutex());
731 chainman().ActiveChain());
733 bool isInitialBlockDownload()
override {
734 return chainman().IsInitialBlockDownload();
737 void initMessage(
const std::string &message)
override {
746 void showProgress(
const std::string &title,
int progress,
747 bool resume_possible)
override {
748 ::uiInterface.ShowProgress(title, progress, resume_possible);
750 std::unique_ptr<Handler> handleNotifications(
751 std::shared_ptr<Notifications> notifications)
override {
752 return std::make_unique<NotificationsHandlerImpl>(
753 std::move(notifications));
756 waitForNotificationsIfTipChanged(
const BlockHash &old_tip)
override {
767 std::unique_ptr<Handler>
769 return std::make_unique<RpcHandlerImpl>(command);
771 bool rpcEnableDeprecated(
const std::string &method)
override {
774 void rpcRunLater(
const std::string &
name, std::function<
void()> fn,
775 int64_t seconds)
override {
781 std::vector<util::SettingsValue>
782 getSettingsList(
const std::string &
name)
override {
795 bool updateRwSetting(
const std::string &
name,
797 bool write)
override {
800 settings.rw_settings.erase(name);
802 settings.rw_settings[name] = value;
807 void requestMempoolTransactions(Notifications ¬ifications)
override {
813 notifications.transactionAddedToMempool(entry->GetSharedTx(),
817 bool hasAssumedValidChain()
override {
821 NodeContext *context()
override {
return &
m_node; }
830 return std::make_unique<node::NodeImpl>(context);
834 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, proxyType &proxyInfoOut)
std::shared_ptr< Chain::Notifications > m_notifications
const CChainParams & m_params
const CRPCCommand * m_wrapped_command
std::shared_ptr< NotificationsProxy > m_proxy
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.