Logging consists of targets and their associated formatter. An log target is connected with various severity levels in a many-to-many configuration as registered with the record message sender (obtained by reference from the logger during initialization).
For use in an application, the logger class is the central piece that provides convenient methods for logging. See the inherit logger class example with its test driver for an real-world example on using citrus-logging in an application.
In this example we are using file and stream targets. Any message with severity error or higher will be logged to both myapp-system.txt and stderr. It also demonstrate using ordinal substitution and conversion specifiers for log message formatting.
#include <iostream>
#include <citrus/logging.hpp>
using namespace Citrus::Logging;
int main(int argc, const char **argv)
{
Logger logger;
TargetFile target1("/var/log/myapp-debug.txt");
TargetFile target2("/var/log/myapp-system.txt", RecordFormat<FormatJson>::Object());
TargetStream target3(std::cerr, RecordFormat<FormatPrefix>::Object());
logger.GetSender().Register(&target1, Level::Debug);
logger.GetSender().Register(&target2, Level::Information, Level::Emergent);
logger.GetSender().Register(&target3, Level::Error, Level::Emergent);
logger.SetThreshold(Level::Debug);
logger.SetIdentity("my-logger");
logger.Debug("Hello world!");
logger.Debug("Hello world from %1 on year %2", {"logger", 2019});
std::string message("Logging is simple!");
logger.Alert("Alert message (%s)", {message});
logger.Critical("Critical message (%s)", {message});
logger.Debug("Debug message (%s)", {message});
logger.Emergent("Emergent message (%s)", {message});
logger.Error("Error message (%s)", {message});
logger.Information("Information message (%s)", {message});
logger.Notice("Alert message (%s)", {message});
logger(Level::Notice, "Notice message logged by operator");
logger(Level::Notice, "Notice message logged by operator (%s)", {message});
return 0;
}
Have a look inside the example directory for simple programs that demonstrate each class in the citrus-logging library.
The list of targets consist of File, HTTP, SMTP, I/O Stream, Memory and UNIX syslog. Each target class has a default formatter that can be replaced by passing a second argument. In this example, the first http target will format messages as JSON, while the second will format as XML.
TargetHttp target1("http://localhost/http.php");
TargetHttp target2("http://localhost/http.php", RecordFormat<FormatXml>::Object());
Some targets provides utility methods. For example, the HTTP and SMTP targets provides wrapper for passing options to libcurl:
TargetSmtp target("smtp://smtp.unix.qnet", message, formatter);
target.SetOption(CURLOPT_VERBOSE, 1);
The memory and buffer targets are pseudo-targets without a real target on their own. Instead they both acts as buffers for logged records, but in a slightly different way.
The first pseudo-target is the memory target that provides in RAM storage of logged records, with an std::function (i.e. lambda or functional object) being called on overflow. It's also possible to use this class without an overflow handler.
TargetMemory target([](const MemoryStrategy * strategy) {
WriteRecord(strategy->GetLatest()); // Overflow
});
The second pseudo-target is buffer that wraps any other target class. This example adds buffering of stderr causing it every fifth records to be flushed to the standard error stream:
TargetStream stream(std::cerr);
TargetBuffer buffer(stream, 5);
One use case for these pseudo-targets could be to implement state-ful logging having them to check whether the log target is online and otherwise cache records.
The list of formatters consist of numerous classes, i.e. JSON, SQL, CSV, XML and most target classes uses sensible default formatters.
If these don't suite your taste, you can either create your own custom format classes or use the swiss-army knife classes String (format specifier) or User (lambda callback):
FormatString format1("[%3][%2][%6] %1 <%5>: %4\n")
FormatUser format2([](const Record & record) -> std::string {
return record.GetDateTime().GetString(DateTime::Format::ISO8601) + ": " + record.GetMessage() + "\n";
});
A decent C++ compiler supporting version 11 should be sufficient. Some parts uses version 17 with code emulation for non-conforming compilers (i.e. lack of std::map<K,V>::merge()).
For building the HTTP and SMTP logger you need to have libcurl installed with development headers and relevant protocols enabled.
A standard installation into /usr/local:
bash$> ./autogen.sh
bash$> ./configure
bash$> make && make install
Pass --prefix=path to configure or use environment variable DESTDIR for installation to non-standard locations.
See the examples directory for complete code ready to be run.
This library is released under LGPL-3.0 and can be linked and used with commercial applications.
Please see the AUTHORS file for contact information. Feel free to submit bug reports and suggestions.