Bitcoin Core 29.99.0
P2P Digital Currency
common.cpp
Go to the documentation of this file.
1// Copyright (c) 2021-2022 The Bitcoin Core developers
2// Distributed under the MIT software license, see the accompanying
3// file COPYING or http://www.opensource.org/licenses/mit-license.php.
4
5#include <bitcoin-build-config.h> // IWYU pragma: keep
6
7#include <clientversion.h>
8#include <common/args.h>
9#include <logging.h>
10#include <node/interface_ui.h>
11#include <tinyformat.h>
12#include <util/fs.h>
13#include <util/fs_helpers.h>
14#include <util/result.h>
15#include <util/string.h>
16#include <util/time.h>
17#include <util/translation.h>
18
19#include <algorithm>
20#include <filesystem>
21#include <string>
22#include <vector>
23
25
26namespace init {
28{
29 argsman.AddArg("-debuglogfile=<file>", strprintf("Specify location of debug log file (default: %s). Relative paths will be prefixed by a net-specific datadir location. Pass -nodebuglogfile to disable writing the log to a file.", DEFAULT_DEBUGLOGFILE), ArgsManager::ALLOW_ANY, OptionsCategory::OPTIONS);
30 argsman.AddArg("-debug=<category>", "Output debug and trace logging (default: -nodebug, supplying <category> is optional). "
31 "If <category> is not supplied or if <category> is 1 or \"all\", output all debug logging. If <category> is 0 or \"none\", any other categories are ignored. Other valid values for <category> are: " + LogInstance().LogCategoriesString() + ". This option can be specified multiple times to output multiple categories.",
33 argsman.AddArg("-debugexclude=<category>", "Exclude debug and trace logging for a category. Can be used in conjunction with -debug=1 to output debug and trace logging for all categories except the specified category. This option can be specified multiple times to exclude multiple categories. This takes priority over \"-debug\"", ArgsManager::ALLOW_ANY, OptionsCategory::DEBUG_TEST);
34 argsman.AddArg("-logips", strprintf("Include IP addresses in debug output (default: %u)", DEFAULT_LOGIPS), ArgsManager::ALLOW_ANY, OptionsCategory::DEBUG_TEST);
35 argsman.AddArg("-loglevel=<level>|<category>:<level>", strprintf("Set the global or per-category severity level for logging categories enabled with the -debug configuration option or the logging RPC. Possible values are %s (default=%s). The following levels are always logged: error, warning, info. If <category>:<level> is supplied, the setting will override the global one and may be specified multiple times to set multiple category-specific levels. <category> can be: %s.", LogInstance().LogLevelsString(), LogInstance().LogLevelToStr(BCLog::DEFAULT_LOG_LEVEL), LogInstance().LogCategoriesString()), ArgsManager::DISALLOW_NEGATION | ArgsManager::DISALLOW_ELISION | ArgsManager::DEBUG_ONLY, OptionsCategory::DEBUG_TEST);
36 argsman.AddArg("-logtimestamps", strprintf("Prepend debug output with timestamp (default: %u)", DEFAULT_LOGTIMESTAMPS), ArgsManager::ALLOW_ANY, OptionsCategory::DEBUG_TEST);
37 argsman.AddArg("-logthreadnames", strprintf("Prepend debug output with name of the originating thread (default: %u)", DEFAULT_LOGTHREADNAMES), ArgsManager::ALLOW_ANY, OptionsCategory::DEBUG_TEST);
38 argsman.AddArg("-logsourcelocations", strprintf("Prepend debug output with name of the originating source location (source file, line number and function name) (default: %u)", DEFAULT_LOGSOURCELOCATIONS), ArgsManager::ALLOW_ANY, OptionsCategory::DEBUG_TEST);
39 argsman.AddArg("-logtimemicros", strprintf("Add microsecond precision to debug timestamps (default: %u)", DEFAULT_LOGTIMEMICROS), ArgsManager::ALLOW_ANY | ArgsManager::DEBUG_ONLY, OptionsCategory::DEBUG_TEST);
40 argsman.AddArg("-loglevelalways", strprintf("Always prepend a category and level (default: %u)", DEFAULT_LOGLEVELALWAYS), ArgsManager::ALLOW_ANY, OptionsCategory::DEBUG_TEST);
41 argsman.AddArg("-logratelimit", strprintf("Apply rate limiting to unconditional logging to mitigate disk-filling attacks (default: %u)", BCLog::DEFAULT_LOGRATELIMIT), ArgsManager::ALLOW_ANY | ArgsManager::DEBUG_ONLY, OptionsCategory::DEBUG_TEST);
42 argsman.AddArg("-printtoconsole", "Send trace/debug info to console (default: 1 when no -daemon. To disable logging to file, set -nodebuglogfile)", ArgsManager::ALLOW_ANY, OptionsCategory::DEBUG_TEST);
43 argsman.AddArg("-shrinkdebugfile", "Shrink debug.log file on client startup (default: 1 when no -debug)", ArgsManager::ALLOW_ANY, OptionsCategory::DEBUG_TEST);
44}
45
47{
48 LogInstance().m_print_to_file = !args.IsArgNegated("-debuglogfile");
50 LogInstance().m_print_to_console = args.GetBoolArg("-printtoconsole", !args.GetBoolArg("-daemon", false));
56
58}
59
61{
62 for (const std::string& level_str : args.GetArgs("-loglevel")) {
63 if (level_str.find_first_of(':', 3) == std::string::npos) {
64 // user passed a global log level, i.e. -loglevel=<level>
65 if (!LogInstance().SetLogLevel(level_str)) {
66 return util::Error{strprintf(_("Unsupported global logging level %s=%s. Valid values: %s."), "-loglevel", level_str, LogInstance().LogLevelsString())};
67 }
68 } else {
69 // user passed a category-specific log level, i.e. -loglevel=<category>:<level>
70 const auto& toks = SplitString(level_str, ':');
71 if (!(toks.size() == 2 && LogInstance().SetCategoryLogLevel(toks[0], toks[1]))) {
72 return util::Error{strprintf(_("Unsupported category-specific logging level %1$s=%2$s. Expected %1$s=<category>:<loglevel>. Valid categories: %3$s. Valid loglevels: %4$s."), "-loglevel", level_str, LogInstance().LogCategoriesString(), LogInstance().LogLevelsString())};
73 }
74 }
75 }
76 return {};
77}
78
80{
81 const std::vector<std::string> categories = args.GetArgs("-debug");
82
83 // Special-case: Disregard any debugging categories appearing before -debug=0/none
84 const auto last_negated = std::find_if(categories.rbegin(), categories.rend(),
85 [](const std::string& cat) { return cat == "0" || cat == "none"; });
86
87 const auto categories_to_process = (last_negated == categories.rend()) ? categories : std::ranges::subrange(last_negated.base(), categories.end());
88
89 for (const auto& cat : categories_to_process) {
90 if (!LogInstance().EnableCategory(cat)) {
91 return util::Error{strprintf(_("Unsupported logging category %s=%s."), "-debug", cat)};
92 }
93 }
94
95 // Now remove the logging categories which were explicitly excluded
96 for (const std::string& cat : args.GetArgs("-debugexclude")) {
97 if (!LogInstance().DisableCategory(cat)) {
98 return util::Error{strprintf(_("Unsupported logging category %s=%s."), "-debugexclude", cat)};
99 }
100 }
101 return {};
102}
103
105{
106 if (LogInstance().m_print_to_file) {
107 if (args.GetBoolArg("-shrinkdebugfile", LogInstance().DefaultShrinkDebugFile())) {
108 // Do this first since it both loads a bunch of debug.log into memory,
109 // and because this needs to happen before any other debug.log printing
111 }
112 }
113 if (!LogInstance().StartLogging()) {
114 return InitError(Untranslated(strprintf("Could not open debug log file %s",
115 fs::PathToString(LogInstance().m_file_path))));
116 }
117
118 if (!LogInstance().m_log_timestamps) {
119 LogInfo("Startup time: %s", FormatISO8601DateTime(GetTime()));
120 }
121 LogInfo("Default data directory %s", fs::PathToString(GetDefaultDataDir()));
122 LogInfo("Using data directory %s", fs::PathToString(gArgs.GetDataDirNet()));
123
124 // Only log conf file usage message if conf file actually exists.
125 fs::path config_file_path = args.GetConfigFilePath();
126 if (args.IsArgNegated("-conf")) {
127 LogInfo("Config file: <disabled>");
128 } else if (fs::is_directory(config_file_path)) {
129 LogWarning("Config file: %s (is directory, not file)", fs::PathToString(config_file_path));
130 } else if (fs::exists(config_file_path)) {
131 LogInfo("Config file: %s", fs::PathToString(config_file_path));
132 } else if (args.IsArgSet("-conf")) {
133 InitWarning(strprintf(_("The specified config file %s does not exist"), fs::PathToString(config_file_path)));
134 } else {
135 // Not categorizing as "Warning" because it's the default behavior
136 LogInfo("Config file: %s (not found, skipping)", fs::PathToString(config_file_path));
137 }
138
139 // Log the config arguments to debug.log
140 args.LogArgs();
141
142 return true;
143}
144
146{
147 std::string version_string = FormatFullVersion();
148#ifdef DEBUG
149 version_string += " (debug build)";
150#else
151 version_string += " (release build)";
152#endif
153 LogInfo(CLIENT_NAME " version %s", version_string);
154}
155} // namespace init
fs::path GetDefaultDataDir()
Definition: args.cpp:732
ArgsManager gArgs
Definition: args.cpp:42
fs::path AbsPathForConfigVal(const ArgsManager &args, const fs::path &path, bool net_specific=true)
Most paths passed as configuration arguments are treated as relative to the datadir if they are not a...
Definition: config.cpp:226
ArgsManager & args
Definition: bitcoind.cpp:277
bool IsArgNegated(const std::string &strArg) const
Return true if the argument was originally passed as a negated option, i.e.
Definition: args.cpp:452
@ ALLOW_ANY
disable validation
Definition: args.h:106
@ DISALLOW_NEGATION
disallow -nofoo syntax
Definition: args.h:111
@ DISALLOW_ELISION
disallow -foo syntax that doesn't assign any value
Definition: args.h:112
@ DEBUG_ONLY
Definition: args.h:114
std::vector< std::string > GetArgs(const std::string &strArg) const
Return a vector of strings of the given argument.
Definition: args.cpp:362
fs::path GetDataDirNet() const
Get data directory path with appended network identifier.
Definition: args.h:234
bool IsArgSet(const std::string &strArg) const
Return true if the given argument has been manually set.
Definition: args.cpp:371
fs::path GetConfigFilePath() const
Return config file path (read-only)
Definition: args.cpp:770
void LogArgs() const
Log the config file options and the command line arguments, useful for troubleshooting.
Definition: args.cpp:865
bool GetBoolArg(const std::string &strArg, bool fDefault) const
Return boolean argument or default value.
Definition: args.cpp:507
void AddArg(const std::string &name, const std::string &help, unsigned int flags, const OptionsCategory &cat)
Add argument.
Definition: args.cpp:564
fs::path GetPathArg(std::string arg, const fs::path &default_value={}) const
Return path argument or default value.
Definition: args.cpp:272
bool m_always_print_category_level
Definition: logging.h:228
bool m_log_sourcelocations
Definition: logging.h:227
fs::path m_file_path
Definition: logging.h:230
bool m_log_time_micros
Definition: logging.h:225
bool m_log_threadnames
Definition: logging.h:226
void EnableCategory(LogFlags flag)
Definition: logging.cpp:123
bool m_log_timestamps
Definition: logging.h:224
void ShrinkDebugFile()
Definition: logging.cpp:512
bool m_print_to_file
Definition: logging.h:222
bool m_print_to_console
Definition: logging.h:221
std::string FormatFullVersion()
static bool exists(const path &p)
Definition: fs.h:89
static std::string PathToString(const path &path)
Convert path object to a byte string.
Definition: fs.h:151
void InitWarning(const bilingual_str &str)
Show warning message.
bool InitError(const bilingual_str &str)
Show error message.
BCLog::Logger & LogInstance()
Definition: logging.cpp:26
bool fLogIPs
Definition: logging.cpp:47
const char *const DEFAULT_DEBUGLOGFILE
Definition: logging.cpp:23
static const bool DEFAULT_LOGTIMESTAMPS
Definition: logging.h:31
#define LogWarning(...)
Definition: logging.h:357
static const bool DEFAULT_LOGIPS
Definition: logging.h:30
static const bool DEFAULT_LOGTHREADNAMES
Definition: logging.h:32
#define LogInfo(...)
Definition: logging.h:356
static const bool DEFAULT_LOGSOURCELOCATIONS
Definition: logging.h:33
static const bool DEFAULT_LOGTIMEMICROS
Definition: logging.h:29
static constexpr bool DEFAULT_LOGLEVELALWAYS
Definition: logging.h:34
constexpr bool DEFAULT_LOGRATELIMIT
Definition: logging.h:110
constexpr auto DEFAULT_LOG_LEVEL
Definition: logging.h:106
void AddLoggingArgs(ArgsManager &argsman)
Definition: common.cpp:27
util::Result< void > SetLoggingCategories(const ArgsManager &args)
Definition: common.cpp:79
bool StartLogging(const ArgsManager &args)
Definition: common.cpp:104
util::Result< void > SetLoggingLevel(const ArgsManager &args)
Definition: common.cpp:60
void SetLoggingOptions(const ArgsManager &args)
Definition: common.cpp:46
void LogPackageVersion()
Definition: common.cpp:145
std::vector< std::string > SplitString(std::string_view str, char sep)
Definition: string.h:148
#define strprintf
Format arguments and return the string or write to given std::ostream (see tinyformat::format doc for...
Definition: tinyformat.h:1172
consteval auto _(util::TranslatedLiteral str)
Definition: translation.h:79
bilingual_str Untranslated(std::string original)
Mark a bilingual_str as untranslated.
Definition: translation.h:82
int64_t GetTime()
DEPRECATED Use either ClockType::now() or Now<TimePointType>() if a cast is needed.
Definition: time.cpp:77
std::string FormatISO8601DateTime(int64_t nTime)
ISO 8601 formatting is preferred.
Definition: time.cpp:79