LCOV - code coverage report
Current view: top level - src/init - common.cpp (source / functions) Coverage Total Hit
Test: total_coverage.info Lines: 92.7 % 82 76
Test Date: 2025-01-19 05:08:01 Functions: 100.0 % 7 7
Branches: 50.4 % 254 128

             Branch data     Line data    Source code
       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                 :        1679 : void AddLoggingArgs(ArgsManager& argsman)
      28                 :             : {
      29   [ +  -  +  - ]:        3358 :     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   [ +  -  +  - ]:        3358 :     argsman.AddArg("-debug=<category>", "Output debug and trace logging (default: -nodebug, supplying <category> is optional). "
      31   [ +  -  +  - ]:        5037 :         "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                 :        1679 :         ArgsManager::ALLOW_ANY, OptionsCategory::DEBUG_TEST);
      33   [ +  -  +  - ]:        3358 :     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   [ +  -  +  - ]:        3358 :     argsman.AddArg("-logips", strprintf("Include IP addresses in debug output (default: %u)", DEFAULT_LOGIPS), ArgsManager::ALLOW_ANY, OptionsCategory::DEBUG_TEST);
      35   [ +  -  +  -  :        3358 :     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   [ +  -  +  - ]:        3358 :     argsman.AddArg("-logtimestamps", strprintf("Prepend debug output with timestamp (default: %u)", DEFAULT_LOGTIMESTAMPS), ArgsManager::ALLOW_ANY, OptionsCategory::DEBUG_TEST);
      37   [ +  -  +  - ]:        3358 :     argsman.AddArg("-logthreadnames", strprintf("Prepend debug output with name of the originating thread (default: %u)", DEFAULT_LOGTHREADNAMES), ArgsManager::ALLOW_ANY, OptionsCategory::DEBUG_TEST);
      38   [ +  -  +  - ]:        3358 :     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   [ +  -  +  - ]:        3358 :     argsman.AddArg("-logtimemicros", strprintf("Add microsecond precision to debug timestamps (default: %u)", DEFAULT_LOGTIMEMICROS), ArgsManager::ALLOW_ANY | ArgsManager::DEBUG_ONLY, OptionsCategory::DEBUG_TEST);
      40   [ +  -  +  - ]:        3358 :     argsman.AddArg("-loglevelalways", strprintf("Always prepend a category and level (default: %u)", DEFAULT_LOGLEVELALWAYS), ArgsManager::ALLOW_ANY, OptionsCategory::DEBUG_TEST);
      41   [ +  -  +  - ]:        3358 :     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   [ +  -  +  - ]:        3358 :     argsman.AddArg("-shrinkdebugfile", "Shrink debug.log file on client startup (default: 1 when no -debug)", ArgsManager::ALLOW_ANY, OptionsCategory::DEBUG_TEST);
      43                 :        1679 : }
      44                 :             : 
      45                 :        1643 : void SetLoggingOptions(const ArgsManager& args)
      46                 :             : {
      47   [ +  -  +  - ]:        1643 :     LogInstance().m_print_to_file = !args.IsArgNegated("-debuglogfile");
      48   [ +  -  +  -  :        4929 :     LogInstance().m_file_path = AbsPathForConfigVal(args, args.GetPathArg("-debuglogfile", DEFAULT_DEBUGLOGFILE));
             +  -  +  - ]
      49   [ +  -  +  -  :        1643 :     LogInstance().m_print_to_console = args.GetBoolArg("-printtoconsole", !args.GetBoolArg("-daemon", false));
             +  -  +  - ]
      50   [ +  -  +  - ]:        1643 :     LogInstance().m_log_timestamps = args.GetBoolArg("-logtimestamps", DEFAULT_LOGTIMESTAMPS);
      51   [ +  -  +  - ]:        1643 :     LogInstance().m_log_time_micros = args.GetBoolArg("-logtimemicros", DEFAULT_LOGTIMEMICROS);
      52   [ +  -  +  - ]:        1643 :     LogInstance().m_log_threadnames = args.GetBoolArg("-logthreadnames", DEFAULT_LOGTHREADNAMES);
      53   [ +  -  +  - ]:        1643 :     LogInstance().m_log_sourcelocations = args.GetBoolArg("-logsourcelocations", DEFAULT_LOGSOURCELOCATIONS);
      54   [ +  -  +  - ]:        1643 :     LogInstance().m_always_print_category_level = args.GetBoolArg("-loglevelalways", DEFAULT_LOGLEVELALWAYS);
      55                 :             : 
      56         [ +  - ]:        1643 :     fLogIPs = args.GetBoolArg("-logips", DEFAULT_LOGIPS);
      57                 :        1643 : }
      58                 :             : 
      59                 :        1635 : util::Result<void> SetLoggingLevel(const ArgsManager& args)
      60                 :             : {
      61   [ +  -  +  - ]:        1635 :     if (args.IsArgSet("-loglevel")) {
      62   [ +  -  +  + ]:        3272 :         for (const std::string& level_str : args.GetArgs("-loglevel")) {
      63         [ +  + ]:        1640 :             if (level_str.find_first_of(':', 3) == std::string::npos) {
      64                 :             :                 // user passed a global log level, i.e. -loglevel=<level>
      65   [ +  -  +  -  :        1635 :                 if (!LogInstance().SetLogLevel(level_str)) {
                   +  + ]
      66   [ +  -  +  -  :           3 :                     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         [ +  - ]:           5 :                 const auto& toks = SplitString(level_str, ':');
      71   [ +  +  +  -  :           5 :                 if (!(toks.size() == 2 && LogInstance().SetCategoryLogLevel(toks[0], toks[1]))) {
             +  -  +  + ]
      72   [ +  -  +  -  :           6 :                     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                 :           5 :             }
      75                 :        1635 :         }
      76                 :             :     }
      77                 :        1632 :     return {};
      78                 :             : }
      79                 :             : 
      80                 :        1634 : util::Result<void> SetLoggingCategories(const ArgsManager& args)
      81                 :             : {
      82   [ +  -  +  - ]:        1634 :     if (args.IsArgSet("-debug")) {
      83                 :             :         // Special-case: if -debug=0/-nodebug is set, turn off debugging messages
      84         [ +  - ]:        1634 :         const std::vector<std::string> categories = args.GetArgs("-debug");
      85                 :             : 
      86   [ +  -  +  - ]:        1634 :         if (std::none_of(categories.begin(), categories.end(),
      87   [ +  -  -  + ]:        1635 :             [](std::string cat){return cat == "0" || cat == "none";})) {
      88         [ +  + ]:        3268 :             for (const auto& cat : categories) {
      89   [ +  -  +  -  :        1635 :                 if (!LogInstance().EnableCategory(cat)) {
                   +  + ]
      90         [ +  - ]:           2 :                     return util::Error{strprintf(_("Unsupported logging category %s=%s."), "-debug", cat)};
      91                 :             :                 }
      92                 :             :             }
      93                 :             :         }
      94                 :        1634 :     }
      95                 :             : 
      96                 :             :     // Now remove the logging categories which were explicitly excluded
      97   [ +  -  +  + ]:        5926 :     for (const std::string& cat : args.GetArgs("-debugexclude")) {
      98   [ +  -  +  -  :        4294 :         if (!LogInstance().DisableCategory(cat)) {
                   +  + ]
      99         [ +  - ]:           2 :             return util::Error{strprintf(_("Unsupported logging category %s=%s."), "-debugexclude", cat)};
     100                 :             :         }
     101                 :        1633 :     }
     102                 :        1632 :     return {};
     103                 :             : }
     104                 :             : 
     105                 :        1015 : bool StartLogging(const ArgsManager& args)
     106                 :             : {
     107         [ +  + ]:        1015 :     if (LogInstance().m_print_to_file) {
     108   [ +  -  -  + ]:        1014 :         if (args.GetBoolArg("-shrinkdebugfile", LogInstance().DefaultShrinkDebugFile())) {
     109                 :             :             // Do this first since it both loads a bunch of debug.log into memory,
     110                 :             :             // and because this needs to happen before any other debug.log printing
     111                 :           0 :             LogInstance().ShrinkDebugFile();
     112                 :             :         }
     113                 :             :     }
     114         [ +  + ]:        1015 :     if (!LogInstance().StartLogging()) {
     115   [ +  -  +  - ]:           4 :             return InitError(Untranslated(strprintf("Could not open debug log file %s",
     116         [ +  - ]:           4 :                 fs::PathToString(LogInstance().m_file_path))));
     117                 :             :     }
     118                 :             : 
     119         [ -  + ]:        1013 :     if (!LogInstance().m_log_timestamps)
     120         [ #  # ]:           0 :         LogPrintf("Startup time: %s\n", FormatISO8601DateTime(GetTime()));
     121   [ +  -  +  - ]:        3039 :     LogPrintf("Default data directory %s\n", fs::PathToString(GetDefaultDataDir()));
     122   [ +  -  +  - ]:        3039 :     LogPrintf("Using data directory %s\n", fs::PathToString(gArgs.GetDataDirNet()));
     123                 :             : 
     124                 :             :     // Only log conf file usage message if conf file actually exists.
     125                 :        1013 :     fs::path config_file_path = args.GetConfigFilePath();
     126   [ +  -  +  -  :        1013 :     if (args.IsArgNegated("-conf")) {
                   +  + ]
     127         [ +  - ]:           2 :         LogInfo("Config file: <disabled>");
     128   [ +  -  -  + ]:        1011 :     } else if (fs::is_directory(config_file_path)) {
     129   [ #  #  #  # ]:           0 :         LogWarning("Config file: %s (is directory, not file)", fs::PathToString(config_file_path));
     130   [ +  -  +  - ]:        1011 :     } else if (fs::exists(config_file_path)) {
     131   [ +  -  +  - ]:        3033 :         LogPrintf("Config file: %s\n", fs::PathToString(config_file_path));
     132   [ #  #  #  #  :           0 :     } else if (args.IsArgSet("-conf")) {
                   #  # ]
     133   [ #  #  #  #  :           0 :         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   [ #  #  #  # ]:           0 :         LogPrintf("Config file: %s (not found, skipping)\n", fs::PathToString(config_file_path));
     137                 :             :     }
     138                 :             : 
     139                 :             :     // Log the config arguments to debug.log
     140         [ +  - ]:        1013 :     args.LogArgs();
     141                 :             : 
     142                 :        1013 :     return true;
     143                 :        1013 : }
     144                 :             : 
     145                 :        1643 : void LogPackageVersion()
     146                 :             : {
     147                 :        1643 :     std::string version_string = FormatFullVersion();
     148                 :             : #ifdef DEBUG
     149                 :             :     version_string += " (debug build)";
     150                 :             : #else
     151         [ +  - ]:        1643 :     version_string += " (release build)";
     152                 :             : #endif
     153         [ +  - ]:        1643 :     LogPrintf(CLIENT_NAME " version %s\n", version_string);
     154                 :        1643 : }
     155                 :             : } // namespace init
        

Generated by: LCOV version 2.0-1