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