/bitcoin/src/init/common.cpp
Line | Count | Source |
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 | | |
24 | | using util::SplitString; |
25 | | |
26 | | namespace init { |
27 | | void AddLoggingArgs(ArgsManager& argsman) |
28 | 11.0k | { |
29 | 11.0k | 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 | 11.0k | argsman.AddArg("-debug=<category>", "Output debug and trace logging (default: -nodebug, supplying <category> is optional). " |
31 | 11.0k | "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.", |
32 | 11.0k | ArgsManager::ALLOW_ANY, OptionsCategory::DEBUG_TEST); |
33 | 11.0k | 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 | 11.0k | argsman.AddArg("-logips", strprintf("Include IP addresses in debug output (default: %u)", DEFAULT_LOGIPS), ArgsManager::ALLOW_ANY, OptionsCategory::DEBUG_TEST); |
35 | 11.0k | 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 | 11.0k | argsman.AddArg("-logtimestamps", strprintf("Prepend debug output with timestamp (default: %u)", DEFAULT_LOGTIMESTAMPS), ArgsManager::ALLOW_ANY, OptionsCategory::DEBUG_TEST); |
37 | 11.0k | argsman.AddArg("-logthreadnames", strprintf("Prepend debug output with name of the originating thread (default: %u)", DEFAULT_LOGTHREADNAMES), ArgsManager::ALLOW_ANY, OptionsCategory::DEBUG_TEST); |
38 | 11.0k | 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 | 11.0k | argsman.AddArg("-logtimemicros", strprintf("Add microsecond precision to debug timestamps (default: %u)", DEFAULT_LOGTIMEMICROS), ArgsManager::ALLOW_ANY | ArgsManager::DEBUG_ONLY, OptionsCategory::DEBUG_TEST); |
40 | 11.0k | argsman.AddArg("-loglevelalways", strprintf("Always prepend a category and level (default: %u)", DEFAULT_LOGLEVELALWAYS), ArgsManager::ALLOW_ANY, OptionsCategory::DEBUG_TEST); |
41 | 11.0k | 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); |
42 | 11.0k | argsman.AddArg("-shrinkdebugfile", "Shrink debug.log file on client startup (default: 1 when no -debug)", ArgsManager::ALLOW_ANY, OptionsCategory::DEBUG_TEST); |
43 | 11.0k | } |
44 | | |
45 | | void SetLoggingOptions(const ArgsManager& args) |
46 | 11.0k | { |
47 | 11.0k | LogInstance().m_print_to_file = !args.IsArgNegated("-debuglogfile"); |
48 | 11.0k | LogInstance().m_file_path = AbsPathForConfigVal(args, args.GetPathArg("-debuglogfile", DEFAULT_DEBUGLOGFILE)); |
49 | 11.0k | LogInstance().m_print_to_console = args.GetBoolArg("-printtoconsole", !args.GetBoolArg("-daemon", false)); |
50 | 11.0k | LogInstance().m_log_timestamps = args.GetBoolArg("-logtimestamps", DEFAULT_LOGTIMESTAMPS); |
51 | 11.0k | LogInstance().m_log_time_micros = args.GetBoolArg("-logtimemicros", DEFAULT_LOGTIMEMICROS); |
52 | 11.0k | LogInstance().m_log_threadnames = args.GetBoolArg("-logthreadnames", DEFAULT_LOGTHREADNAMES); |
53 | 11.0k | LogInstance().m_log_sourcelocations = args.GetBoolArg("-logsourcelocations", DEFAULT_LOGSOURCELOCATIONS); |
54 | 11.0k | LogInstance().m_always_print_category_level = args.GetBoolArg("-loglevelalways", DEFAULT_LOGLEVELALWAYS); |
55 | | |
56 | 11.0k | fLogIPs = args.GetBoolArg("-logips", DEFAULT_LOGIPS); |
57 | 11.0k | } |
58 | | |
59 | | util::Result<void> SetLoggingLevel(const ArgsManager& args) |
60 | 11.0k | { |
61 | 11.0k | for (const std::string& level_str : args.GetArgs("-loglevel")) { Branch (61:43): [True: 0, False: 11.0k]
|
62 | 0 | if (level_str.find_first_of(':', 3) == std::string::npos) { Branch (62:17): [True: 0, False: 0]
|
63 | | // user passed a global log level, i.e. -loglevel=<level> |
64 | 0 | if (!LogInstance().SetLogLevel(level_str)) { Branch (64:21): [True: 0, False: 0]
|
65 | 0 | return util::Error{strprintf(_("Unsupported global logging level %s=%s. Valid values: %s."), "-loglevel", level_str, LogInstance().LogLevelsString())}; |
66 | 0 | } |
67 | 0 | } else { |
68 | | // user passed a category-specific log level, i.e. -loglevel=<category>:<level> |
69 | 0 | const auto& toks = SplitString(level_str, ':'); |
70 | 0 | if (!(toks.size() == 2 && LogInstance().SetCategoryLogLevel(toks[0], toks[1]))) { Branch (70:23): [True: 0, False: 0]
Branch (70:43): [True: 0, False: 0]
|
71 | 0 | 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())}; |
72 | 0 | } |
73 | 0 | } |
74 | 0 | } |
75 | 11.0k | return {}; |
76 | 11.0k | } |
77 | | |
78 | | util::Result<void> SetLoggingCategories(const ArgsManager& args) |
79 | 11.0k | { |
80 | 11.0k | const std::vector<std::string> categories = args.GetArgs("-debug"); |
81 | | |
82 | | // Special-case: Disregard any debugging categories appearing before -debug=0/none |
83 | 11.0k | const auto last_negated = std::find_if(categories.rbegin(), categories.rend(), |
84 | 11.0k | [](const std::string& cat) { return cat == "0" || cat == "none"; }); Branch (84:84): [True: 0, False: 11.0k]
Branch (84:98): [True: 0, False: 11.0k]
|
85 | | |
86 | 11.0k | const auto categories_to_process = (last_negated == categories.rend()) ? categories : std::ranges::subrange(last_negated.base(), categories.end()); Branch (86:44): [True: 11.0k, False: 0]
|
87 | | |
88 | 11.0k | for (const auto& cat : categories_to_process) { Branch (88:30): [True: 11.0k, False: 11.0k]
|
89 | 11.0k | if (!LogInstance().EnableCategory(cat)) { Branch (89:17): [True: 0, False: 11.0k]
|
90 | 0 | return util::Error{strprintf(_("Unsupported logging category %s=%s."), "-debug", cat)}; |
91 | 0 | } |
92 | 11.0k | } |
93 | | |
94 | | // Now remove the logging categories which were explicitly excluded |
95 | 22.1k | for (const std::string& cat : args.GetArgs("-debugexclude")) { Branch (95:33): [True: 22.1k, False: 11.0k]
|
96 | 22.1k | if (!LogInstance().DisableCategory(cat)) { Branch (96:13): [True: 0, False: 22.1k]
|
97 | 0 | return util::Error{strprintf(_("Unsupported logging category %s=%s."), "-debugexclude", cat)}; |
98 | 0 | } |
99 | 22.1k | } |
100 | 11.0k | return {}; |
101 | 11.0k | } |
102 | | |
103 | | bool StartLogging(const ArgsManager& args) |
104 | 11.0k | { |
105 | 11.0k | if (LogInstance().m_print_to_file) { Branch (105:9): [True: 11.0k, False: 0]
|
106 | 11.0k | if (args.GetBoolArg("-shrinkdebugfile", LogInstance().DefaultShrinkDebugFile())) { Branch (106:13): [True: 0, False: 11.0k]
|
107 | | // Do this first since it both loads a bunch of debug.log into memory, |
108 | | // and because this needs to happen before any other debug.log printing |
109 | 0 | LogInstance().ShrinkDebugFile(); |
110 | 0 | } |
111 | 11.0k | } |
112 | 11.0k | if (!LogInstance().StartLogging()) { Branch (112:9): [True: 0, False: 11.0k]
|
113 | 0 | return InitError(Untranslated(strprintf("Could not open debug log file %s", |
114 | 0 | fs::PathToString(LogInstance().m_file_path)))); |
115 | 0 | } |
116 | | |
117 | 11.0k | if (!LogInstance().m_log_timestamps) Branch (117:9): [True: 0, False: 11.0k]
|
118 | 11.0k | LogPrintf("Startup time: %s\n", FormatISO8601DateTime(GetTime())); |
119 | 11.0k | LogPrintf("Default data directory %s\n", fs::PathToString(GetDefaultDataDir())); |
120 | 11.0k | LogPrintf("Using data directory %s\n", fs::PathToString(gArgs.GetDataDirNet())); |
121 | | |
122 | | // Only log conf file usage message if conf file actually exists. |
123 | 11.0k | fs::path config_file_path = args.GetConfigFilePath(); |
124 | 11.0k | if (args.IsArgNegated("-conf")) { Branch (124:9): [True: 0, False: 11.0k]
|
125 | 0 | LogInfo("Config file: <disabled>"); |
126 | 11.0k | } else if (fs::is_directory(config_file_path)) { Branch (126:16): [True: 0, False: 11.0k]
|
127 | 0 | LogWarning("Config file: %s (is directory, not file)", fs::PathToString(config_file_path)); |
128 | 11.0k | } else if (fs::exists(config_file_path)) { Branch (128:16): [True: 0, False: 11.0k]
|
129 | 0 | LogPrintf("Config file: %s\n", fs::PathToString(config_file_path)); |
130 | 11.0k | } else if (args.IsArgSet("-conf")) { Branch (130:16): [True: 0, False: 11.0k]
|
131 | 0 | InitWarning(strprintf(_("The specified config file %s does not exist"), fs::PathToString(config_file_path))); |
132 | 11.0k | } else { |
133 | | // Not categorizing as "Warning" because it's the default behavior |
134 | 11.0k | LogPrintf("Config file: %s (not found, skipping)\n", fs::PathToString(config_file_path)); |
135 | 11.0k | } |
136 | | |
137 | | // Log the config arguments to debug.log |
138 | 11.0k | args.LogArgs(); |
139 | | |
140 | 11.0k | return true; |
141 | 11.0k | } |
142 | | |
143 | | void LogPackageVersion() |
144 | 11.0k | { |
145 | 11.0k | std::string version_string = FormatFullVersion(); |
146 | | #ifdef DEBUG |
147 | | version_string += " (debug build)"; |
148 | | #else |
149 | 11.0k | version_string += " (release build)"; |
150 | 11.0k | #endif |
151 | 11.0k | LogPrintf(CLIENT_NAME " version %s\n", version_string); |
152 | 11.0k | } |
153 | | } // namespace init |