Skip to content

A C++ meta-logging layer that keeps the log-source API stable and allows to replace the logging backend (workhorse).

License

Notifications You must be signed in to change notification settings

ClausKlein/cxx.simplelog

 
 

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

88 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

cxx.simplelog: A Meta-Logging Layer for C++

Requires:Modern C++ (C++11 or newer)
License:MIT
Library Kind:Header-only (currently; backends may have libs)

cxx.simplelog provides a meta-logging layer for C++. It does not provide a logging library (or framework). But it uses excellent existing C++ logging libraries (like spdlog) as logging backend instead.

In addition, it provides a stable API as logging-frontend to log-users / log-sources that makes it easier to replace one logging backend with a another one. Therefore, there is no need to adapt the many places where the logging statements are used (in many libraries).

This approach makes it also easier to integrate and support new logging backends in the future.

Core ideas:

  • Provide common set of preprocessor macros for logging (and tracing)
  • Log engine/subsystem is hidden and can be replaced w/ another implementation
  • Log setup can be specific w/ concrete logging subsystem

PROCESSING MODEL:

 0..N LOG-SOURCE(s) => LOGGING-SUBSYSTEM => 0..M LOG-SINKS
|<-  Stable API   ->|<- Logging backend specific part (or sink) ->|

* Log-source(s) generate log records by using the logging subsystem
* Log-source(s) may perform log-level based filtering (to avoid records)
* The logging subsystem delegates log-record output to log-sinks
* The logging subsystem has normally at least one log-sink (configured)
* A log-sink may perform additional filtering (by log-level, etc.)
* A log-source may use zero or more log-sink(s) (normally: 1 or more)
* A log-source 1 may use other log-sink(s) than log-source 2

DOMAIN SPECIFIC ASPECTS:

  • log-user: log statements are used in many places
  • log-user: Does not care about setup of logging subsystem and threshold levels
  • log-admin: Setup of logging subsystem/backend
  • log-admin: Setup of log levels, log-sinks, formatters, ...
  • log-admin: Done only in main/main.cpp context (decision: Per program)

DESIGN INTENTIONS:

  • Keep log-user code clean (independent of the concrete logging subsystem)
  • log-user: Recompile of library w/ different logging subsystem is acceptable
  • Support multiple logging subsystems/backends (if possible)
  • Avoid vendor-lockin (of concrete logging subsystem)

NEEDS:

  • Support for multiple categories/loggers per program

Note

REASON

  • configuration-point for threshold, context for functionality,
  • reduce or avoid unneeded log-records at log-source point (as earliest possible filter).

DESIGN VARIATION POINT: Approach to placeholders in log-records

  • printf-like/var-args, iostream-like, format-like (fmt), ...
  • A compact, safe and efficient syntax is needed (iostream-style is not compact; printf-style is compact, but not safe)
  • Many new C++ logging frameworks use the fmt library (due to its compact syntax and efficiency)
  • The fmt library is part of STD-C++20

DESIGN DECISIONS:

  • Use fmt library for placeholders by default (reason: see above)
  • Use explicit log-level configuration with wildcards (instead of: log-level inheritance)
  • Avoid log-level enums/types at log-points (logging-sources) (should be hidden by macro to make it easier to replace implementation)
  • Use short-log-macro names if possible and enabled
  • Use default logging-module (logger) in most cases (if possible). REASON: Makes logging statement shorter/simpler (at logging source)

SUPPORTED LOGGING BACKENDS:

  • spdlog (sinks: console, file, rotating-file, sd_journal, syslog, ...)
  • null (as blackhole ;-)

CANDIDATES FOR LOGGING BACKENDS:

  • sd_journal (systemd journal service; supported by spdlog as sink)
  • syslog (supported by spdlog as sink)
  • ...

SEE ALSO:

RELATED: Overview C++ logging libraries

OPEN ISSUE: spdlog

  • Read logging configuration from config-file

EXAMPLE

The following sections shows how the functionality can be used in C++:

  • Use case: LOG-SOURCE that generates log-records (generic, backend independent)
  • Use case: Setup of the logging subsystem (context: main.cpp; backend specific)
// -- FILE: my_library1/.../example_1.cpp
// EXAMPLE: Use LOG-MACROS with DEFAULT_MODULE in local scope

// -- INCLUDES:
#include <simplelog/LogMacros.hpp>

namespace {
    // -- USE DEFAULT LOGGING-MODULE (within a scope)
    SIMPLELOG_DEFINE_STATIC_DEFAULT_MODULE("example.1"); //< DEFAULT_MODULE_1
    SIMPLELOG_DEFINE_STATIC_MODULE(log1, "other");       //< OTHER_MODULE_2
}

void log_source_func1(void)
{
    // -- USE-SHORT-MACROS: SLOG_xxx() == SIMPLELOG_xxx()  (if possible)
    // USE: DEFAULT_MODULE_1: name="example.1"
    // OR: SIMPLELOG_DEFINE_STATIC_DEFAULT_MODULE("example.2");
    SLOG_FATAL("Message with zero args");
    SLOG_ERROR("Hello {0}", "Alice");
    SLOG_WARN( "Hello {0} and {1}", "Alice", std::string("Bob"));
    SLOG_INFO_IF(true, "OOPS: Hello {0}", "Alice"_s);

    // -- USE: OTHER_MODULE_2
    // MACROS: SLOGM_xxx(log, ...) == SIMPLELOGM_xxx(log, ...)
    SLOGM_INFO(log1, "Use another logger"); //< OTHER_MODULE_2
}
// -- FILE: main.cpp
// SETUP LOGGING HERE: With BACKEND=spdlog
#define SIMPLELOG_USE_BACKEND_SPDLOG 1
#include <simplelog/backend/spdlog/SetupUtil.hpp>   //< USE: assignSink(), ...
#include <simplelog/backend/spdlog/ModuleUtil.hpp>  //< USE: useOrCreateLogger()

void process_setupLogging(void)
{
    // -- HERE is the LOGGING-BACKEND-SPECIFIC part.
    auto console = spdlog::stdout_color_mt("console");
    auto theSink = console->sinks().front();

    // -- GLOBAL SETUP: Define log-sink(s), formatter pattern and DEFAULT log-level.
    // SEE: https://github.com/gabime/spdlog/wiki/3.-Custom-formatting#pattern-flags
    // PATTERN SCHEMA: <ISO_DATE>_<ISO_TIME>.<microseconds> <name>::<level>  <message>
    simplelog::backend_spdlog::assignSink(theSink);
    spdlog::set_pattern("%Y-%m-%d_%T.%f  %^%10n::%-7l%$  %v");
    spdlog::set_level(spdlog::level::warn);
    // SAME AS: simplelog::backend_spdlog::setLevel(spdlog::level::warn);

    // -- SPECIFIC SETUP: OVERRIDE CONFIG FOR SPECIFIC MODULES: log-level, ...
    using simplelog::backend_spdlog::useOrCreateLogger;
    auto log1 = useOrCreateLogger("foo.bar");
    auto log2 = useOrCreateLogger("example.1");
    log1->set_level(spdlog::level::info);
    log2->set_level(spdlog::level::debug);

    // -- MAYBE: Ensure that at least level=info is used (by all loggers).
    // simplelog::backend_spdlog::setMinLevel(spdlog::level::info);
}

int main(int argc, char** argv)
{
    process_setupLogging();
    ...     // Do something (and log-sources will use logging);
    return EXIT_SUCCESS;
}

DEPENDENCIES

About

A C++ meta-logging layer that keeps the log-source API stable and allows to replace the logging backend (workhorse).

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages

  • C++ 43.0%
  • Python 41.9%
  • CMake 14.2%
  • Other 0.9%