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