This commit is contained in:
Ryan Ofsky 2025-04-29 12:02:58 +02:00 committed by GitHub
commit bd666bfa1e
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
12 changed files with 234 additions and 10 deletions

View file

@ -675,6 +675,7 @@ message(" bitcoin-util ........................ ${BUILD_UTIL}")
message(" bitcoin-wallet ...................... ${BUILD_WALLET_TOOL}") message(" bitcoin-wallet ...................... ${BUILD_WALLET_TOOL}")
message(" bitcoin-chainstate (experimental) ... ${BUILD_UTIL_CHAINSTATE}") message(" bitcoin-chainstate (experimental) ... ${BUILD_UTIL_CHAINSTATE}")
message(" libbitcoinkernel (experimental) ..... ${BUILD_KERNEL_LIB}") message(" libbitcoinkernel (experimental) ..... ${BUILD_KERNEL_LIB}")
message(" bitcoin-mine (experimental) ......... ${bitcoin_daemon_status}")
message("Optional features:") message("Optional features:")
message(" wallet support ...................... ${ENABLE_WALLET}") message(" wallet support ...................... ${ENABLE_WALLET}")
if(ENABLE_WALLET) if(ENABLE_WALLET)

View file

@ -44,12 +44,6 @@ declare -A SUPPRESS
# init.cpp file currently calls Berkeley DB sanity check function on startup, so # init.cpp file currently calls Berkeley DB sanity check function on startup, so
# there is an undocumented dependency of the node library on the wallet library. # there is an undocumented dependency of the node library on the wallet library.
SUPPRESS["init.cpp.o bdb.cpp.o _ZN6wallet27BerkeleyDatabaseSanityCheckEv"]=1 SUPPRESS["init.cpp.o bdb.cpp.o _ZN6wallet27BerkeleyDatabaseSanityCheckEv"]=1
# init/common.cpp file calls InitError and InitWarning from interface_ui which
# is currently part of the node library. interface_ui should just be part of the
# common library instead, and is moved in
# https://github.com/bitcoin/bitcoin/issues/10102
SUPPRESS["common.cpp.o interface_ui.cpp.o _Z11InitWarningRK13bilingual_str"]=1
SUPPRESS["common.cpp.o interface_ui.cpp.o _Z9InitErrorRK13bilingual_str"]=1
usage() { usage() {
echo "Usage: $(basename "${BASH_SOURCE[0]}") [BUILD_DIR]" echo "Usage: $(basename "${BASH_SOURCE[0]}") [BUILD_DIR]"

View file

@ -11,6 +11,7 @@ import argparse
BINARIES = [ BINARIES = [
'bin/bitcoind', 'bin/bitcoind',
'bin/bitcoin-cli', 'bin/bitcoin-cli',
'bin/bitcoin-mine',
'bin/bitcoin-tx', 'bin/bitcoin-tx',
'bin/bitcoin-wallet', 'bin/bitcoin-wallet',
'bin/bitcoin-util', 'bin/bitcoin-util',

View file

@ -165,6 +165,7 @@ add_library(bitcoin_common STATIC EXCLUDE_FROM_ALL
net_types.cpp net_types.cpp
netaddress.cpp netaddress.cpp
netbase.cpp netbase.cpp
node/interface_ui.cpp
outputtype.cpp outputtype.cpp
policy/feerate.cpp policy/feerate.cpp
policy/policy.cpp policy/policy.cpp
@ -262,7 +263,6 @@ add_library(bitcoin_node STATIC EXCLUDE_FROM_ALL
node/context.cpp node/context.cpp
node/database_args.cpp node/database_args.cpp
node/eviction.cpp node/eviction.cpp
node/interface_ui.cpp
node/interfaces.cpp node/interfaces.cpp
node/kernel_notifications.cpp node/kernel_notifications.cpp
node/mempool_args.cpp node/mempool_args.cpp
@ -358,6 +358,17 @@ if(ENABLE_IPC AND BUILD_DAEMON)
$<TARGET_NAME_IF_EXISTS:bitcoin_wallet> $<TARGET_NAME_IF_EXISTS:bitcoin_wallet>
) )
install_binary_component(bitcoin-node) install_binary_component(bitcoin-node)
add_executable(bitcoin-mine
bitcoin-mine.cpp
init/bitcoin-mine.cpp
)
target_link_libraries(bitcoin-mine
core_interface
bitcoin_common
bitcoin_ipc
)
install_binary_component(bitcoin-mine)
endif() endif()
if(ENABLE_IPC AND BUILD_TESTS) if(ENABLE_IPC AND BUILD_TESTS)

124
src/bitcoin-mine.cpp Normal file
View file

@ -0,0 +1,124 @@
// Copyright (c) 2025 The Bitcoin Core developers
// Distributed under the MIT software license, see the accompanying
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
#include <bitcoin-build-config.h> // IWYU pragma: keep
#include <chainparams.h>
#include <chainparamsbase.h>
#include <clientversion.h>
#include <common/args.h>
#include <common/system.h>
#include <compat/compat.h>
#include <init/common.h>
#include <interfaces/init.h>
#include <interfaces/ipc.h>
#include <logging.h>
#include <tinyformat.h>
#include <util/translation.h>
static const char* const HELP_USAGE{R"(
bitcoin-mine is a test program for interacting with bitcoin-node via IPC.
Usage:
bitcoin-mine [options]
)"};
static const char* HELP_EXAMPLES{R"(
Examples:
# Start separate bitcoin-node that bitcoin-mine can connect to.
bitcoin-node -regtest -ipcbind=unix
# Connect to bitcoin-node and print tip block hash.
bitcoin-mine -regtest
# Run with debug output.
bitcoin-mine -regtest -debug
)"};
const TranslateFn G_TRANSLATION_FUN{nullptr};
static void AddArgs(ArgsManager& args)
{
SetupHelpOptions(args);
SetupChainParamsBaseOptions(args);
args.AddArg("-version", "Print version and exit", ArgsManager::ALLOW_ANY, OptionsCategory::OPTIONS);
args.AddArg("-datadir=<dir>", "Specify data directory", ArgsManager::ALLOW_ANY, OptionsCategory::OPTIONS);
args.AddArg("-ipcconnect=<address>", "Connect to bitcoin-node process in the background to perform online operations. Valid <address> values are 'unix' to connect to the default socket, 'unix:<socket path>' to connect to a socket at a nonstandard path. Default value: unix", ArgsManager::ALLOW_ANY, OptionsCategory::IPC);
init::AddLoggingArgs(args);
}
MAIN_FUNCTION
{
ArgsManager& args = gArgs;
AddArgs(args);
std::string error_message;
if (!args.ParseParameters(argc, argv, error_message)) {
tfm::format(std::cerr, "Error parsing command line arguments: %s\n", error_message);
return EXIT_FAILURE;
}
if (!args.ReadConfigFiles(error_message, true)) {
tfm::format(std::cerr, "Error reading config files: %s\n", error_message);
return EXIT_FAILURE;
}
if (HelpRequested(args) || args.IsArgSet("-version")) {
std::string output{strprintf("%s bitcoin-mine version", CLIENT_NAME) + " " + FormatFullVersion() + "\n"};
if (args.IsArgSet("-version")) {
output += FormatParagraph(LicenseInfo());
} else {
output += HELP_USAGE;
output += args.GetHelpMessage();
output += HELP_EXAMPLES;
}
tfm::format(std::cout, "%s", output);
return EXIT_SUCCESS;
}
if (!CheckDataDirOption(args)) {
tfm::format(std::cerr, "Error: Specified data directory \"%s\" does not exist.\n", args.GetArg("-datadir", ""));
return EXIT_FAILURE;
}
SelectParams(args.GetChainType());
// Set logging options but override -printtoconsole default to depend on -debug rather than -daemon
init::SetLoggingOptions(args);
if (auto result{init::SetLoggingCategories(args)}; !result) {
tfm::format(std::cerr, "Error: %s\n", util::ErrorString(result).original);
return EXIT_FAILURE;
}
if (auto result{init::SetLoggingLevel(args)}; !result) {
tfm::format(std::cerr, "Error: %s\n", util::ErrorString(result).original);
return EXIT_FAILURE;
}
LogInstance().m_print_to_console = args.GetBoolArg("-printtoconsole", LogInstance().GetCategoryMask());
if (!init::StartLogging(args)) {
tfm::format(std::cerr, "Error: StartLogging failed\n");
return EXIT_FAILURE;
}
// Connect to existing bitcoin-node process or spawn new one.
std::unique_ptr<interfaces::Init> mine_init{interfaces::MakeMineInit(argc, argv)};
assert(mine_init);
std::unique_ptr<interfaces::Init> node_init;
try {
std::string address{args.GetArg("-ipcconnect", "unix")};
node_init = mine_init->ipc()->connectAddress(address);
} catch (const std::exception& exception) {
tfm::format(std::cerr, "Error: %s\n", exception.what());
tfm::format(std::cerr, "Probably bitcoin-node is not running or not listening on a unix socket. Can be started with:\n\n");
tfm::format(std::cerr, " bitcoin-node -chain=%s -ipcbind=unix\n", args.GetChainTypeString());
return EXIT_FAILURE;
}
assert(node_init);
tfm::format(std::cout, "Connected to bitcoin-node\n");
std::unique_ptr<interfaces::Mining> mining{node_init->makeMining()};
assert(mining);
auto tip{mining->getTip()};
if (tip) {
tfm::format(std::cout, "Tip hash is %s.\n", tip->hash.ToString());
} else {
tfm::format(std::cout, "Tip hash is null.\n");
}
return EXIT_SUCCESS;
}

29
src/init/bitcoin-mine.cpp Normal file
View file

@ -0,0 +1,29 @@
// Copyright (c) 2025 The Bitcoin Core developers
// Distributed under the MIT software license, see the accompanying
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
#include <interfaces/init.h>
#include <interfaces/ipc.h>
namespace init {
namespace {
const char* EXE_NAME = "bitcoin-mine";
class BitcoinMineInit : public interfaces::Init
{
public:
BitcoinMineInit(const char* arg0) : m_ipc(interfaces::MakeIpc(EXE_NAME, arg0, *this))
{
}
interfaces::Ipc* ipc() override { return m_ipc.get(); }
std::unique_ptr<interfaces::Ipc> m_ipc;
};
} // namespace
} // namespace init
namespace interfaces {
std::unique_ptr<Init> MakeMineInit(int argc, char* argv[])
{
return std::make_unique<init::BitcoinMineInit>(argc > 0 ? argv[0] : "");
}
} // namespace interfaces

View file

@ -53,6 +53,9 @@ std::unique_ptr<Init> MakeWalletInit(int argc, char* argv[], int& exit_status);
//! Return implementation of Init interface for the gui process. //! Return implementation of Init interface for the gui process.
std::unique_ptr<Init> MakeGuiInit(int argc, char* argv[]); std::unique_ptr<Init> MakeGuiInit(int argc, char* argv[]);
//! Return implementation of Init interface for the bitcoin-mine process.
std::unique_ptr<Init> MakeMineInit(int argc, char* argv[]);
} // namespace interfaces } // namespace interfaces
#endif // BITCOIN_INTERFACES_INIT_H #endif // BITCOIN_INTERFACES_INIT_H

View file

@ -26,6 +26,7 @@ function(create_test_config)
set_configure_variable(WITH_ZMQ ENABLE_ZMQ) set_configure_variable(WITH_ZMQ ENABLE_ZMQ)
set_configure_variable(ENABLE_EXTERNAL_SIGNER ENABLE_EXTERNAL_SIGNER) set_configure_variable(ENABLE_EXTERNAL_SIGNER ENABLE_EXTERNAL_SIGNER)
set_configure_variable(WITH_USDT ENABLE_USDT_TRACEPOINTS) set_configure_variable(WITH_USDT ENABLE_USDT_TRACEPOINTS)
set_configure_variable(WITH_MULTIPROCESS WITH_MULTIPROCESS)
configure_file(config.ini.in config.ini USE_SOURCE_PERMISSIONS @ONLY) configure_file(config.ini.in config.ini USE_SOURCE_PERMISSIONS @ONLY)
endfunction() endfunction()

View file

@ -26,3 +26,4 @@ RPCAUTH=@abs_top_srcdir@/share/rpcauth/rpcauth.py
@ENABLE_ZMQ_TRUE@ENABLE_ZMQ=true @ENABLE_ZMQ_TRUE@ENABLE_ZMQ=true
@ENABLE_EXTERNAL_SIGNER_TRUE@ENABLE_EXTERNAL_SIGNER=true @ENABLE_EXTERNAL_SIGNER_TRUE@ENABLE_EXTERNAL_SIGNER=true
@ENABLE_USDT_TRACEPOINTS_TRUE@ENABLE_USDT_TRACEPOINTS=true @ENABLE_USDT_TRACEPOINTS_TRUE@ENABLE_USDT_TRACEPOINTS=true
@WITH_MULTIPROCESS_TRUE@WITH_MULTIPROCESS=true

View file

@ -0,0 +1,46 @@
#!/usr/bin/env python3
# Copyright (c) 2025 The Bitcoin Core developers
# Distributed under the MIT software license, see the accompanying
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
"""Test bitcoin-mine"""
from test_framework.test_framework import BitcoinTestFramework
from test_framework.util import (
assert_equal,
)
import subprocess
import tempfile
class TestBitcoinMine(BitcoinTestFramework):
def set_test_params(self):
self.setup_clean_chain = True
self.num_nodes = 1
def skip_test_if_missing_module(self):
self.skip_if_no_multiprocess()
def setup_nodes(self):
# Always run multiprocess binaries
self.binary_paths.bitcoind = self.binary_paths.bitcoin_node
# Work around default CI path exceeding maximum socket path length.
# On Linux sun_path is 108 bytes, on macOS it's only 104. Includes
# null terminator.
socket_path = self.options.tmpdir + "/node0/regtest/node.sock"
if len(socket_path.encode('utf-8')) < 104:
self.extra_args = [["-ipcbind=unix"]]
self.mine_args = []
else:
sock_path = tempfile.mktemp()
self.extra_args = [[f"-ipcbind=unix:{sock_path}"]]
self.mine_args = [f"-ipcconnect=unix:{sock_path}"]
super().setup_nodes()
def run_test(self):
args = [self.binary_paths.bitcoin_mine, f"-datadir={self.nodes[0].datadir_path}"] + self.mine_args
result = subprocess.run(args, stdout=subprocess.PIPE, stderr=subprocess.STDOUT, text=True, check=True)
assert_equal(result.stdout, "Connected to bitcoin-node\nTip hash is 0f9188f13cb7b2c71f2a335e3a4fc328bf5beb436012afca590b1a11466e2206.\n")
if __name__ == '__main__':
TestBitcoinMine(__file__).main()

View file

@ -285,13 +285,16 @@ class BitcoinTestFramework(metaclass=BitcoinTestMetaClass):
"bitcoin-chainstate": ("bitcoinchainstate", "BITCOINCHAINSTATE"), "bitcoin-chainstate": ("bitcoinchainstate", "BITCOINCHAINSTATE"),
"bitcoin-wallet": ("bitcoinwallet", "BITCOINWALLET"), "bitcoin-wallet": ("bitcoinwallet", "BITCOINWALLET"),
} }
for binary, [attribute_name, env_variable_name] in binaries.items(): def binary_path(binary):
default_filename = os.path.join( return os.path.join(
self.config["environment"]["BUILDDIR"], self.config["environment"]["BUILDDIR"],
"bin", "bin",
binary + self.config["environment"]["EXEEXT"], binary + self.config["environment"]["EXEEXT"],
) )
setattr(paths, attribute_name, os.getenv(env_variable_name, default=default_filename)) for binary, [attribute_name, env_variable_name] in binaries.items():
setattr(paths, attribute_name, os.getenv(env_variable_name) or binary_path(binary))
paths.bitcoin_mine = binary_path("bitcoin-mine")
paths.bitcoin_node = binary_path("bitcoin-node")
return paths return paths
def get_binaries(self, bin_dir=None): def get_binaries(self, bin_dir=None):
@ -1007,6 +1010,11 @@ class BitcoinTestFramework(metaclass=BitcoinTestMetaClass):
if not self.is_cli_compiled(): if not self.is_cli_compiled():
raise SkipTest("bitcoin-cli has not been compiled.") raise SkipTest("bitcoin-cli has not been compiled.")
def skip_if_no_multiprocess(self):
"""Skip the running test if multiprocess binaries are not compiled."""
if not self.is_multiprocess_compiled():
raise SkipTest("multiprocess binaries have not been compiled.")
def skip_if_no_previous_releases(self): def skip_if_no_previous_releases(self):
"""Skip the running test if previous releases are not available.""" """Skip the running test if previous releases are not available."""
if not self.has_previous_releases(): if not self.has_previous_releases():
@ -1061,5 +1069,9 @@ class BitcoinTestFramework(metaclass=BitcoinTestMetaClass):
"""Checks whether the wallet module was compiled with BDB support.""" """Checks whether the wallet module was compiled with BDB support."""
return self.config["components"].getboolean("USE_BDB") return self.config["components"].getboolean("USE_BDB")
def is_multiprocess_compiled(self):
"""Checks whether multiprocess binaries are compiled."""
return self.config["components"].getboolean("WITH_MULTIPROCESS")
def has_blockfile(self, node, filenum: str): def has_blockfile(self, node, filenum: str):
return (node.blocks_path/ f"blk{filenum}.dat").is_file() return (node.blocks_path/ f"blk{filenum}.dat").is_file()

View file

@ -355,6 +355,7 @@ BASE_SCRIPTS = [
'rpc_help.py', 'rpc_help.py',
'p2p_handshake.py', 'p2p_handshake.py',
'p2p_handshake.py --v2transport', 'p2p_handshake.py --v2transport',
'interface_ipc_mining.py',
'feature_dirsymlinks.py', 'feature_dirsymlinks.py',
'feature_help.py', 'feature_help.py',
'feature_shutdown.py', 'feature_shutdown.py',