mirror of
https://github.com/binary-kitchen/doorlockd
synced 2025-01-08 17:22:13 +01:00
233 lines
5.9 KiB
C++
233 lines
5.9 KiB
C++
// (c) 2015 Ralf Ramsauer - Logger class from the sdfs project
|
|
|
|
#include <iostream>
|
|
#include <cstdarg>
|
|
#include <ctime>
|
|
#include <iomanip>
|
|
#include <sstream>
|
|
#include <map>
|
|
#include <mutex>
|
|
|
|
#ifdef _WIN32
|
|
#include <Windows.h>
|
|
#endif
|
|
|
|
#include "config.h"
|
|
#include "logger.h"
|
|
|
|
#ifdef USE_COLORIZED_LOGS
|
|
#ifdef _WIN32 // If Windows, Use Windows Terminal coloring
|
|
const static map<LogLevel, WORD> colorAttribute = {
|
|
{LogLevel::error, FOREGROUND_RED },
|
|
{LogLevel::warning, FOREGROUND_RED | FOREGROUND_GREEN },
|
|
{LogLevel::notice, FOREGROUND_BLUE },
|
|
{LogLevel::info, FOREGROUND_GREEN },
|
|
{LogLevel::debug, FOREGROUND_BLUE | FOREGROUND_RED },
|
|
{LogLevel::debug2, FOREGROUND_BLUE | FOREGROUND_RED },
|
|
};
|
|
#else // Use ANSI Escape sequences
|
|
const static std::map<LogLevel, std::string> prefix_ansicolor = {
|
|
{LogLevel::error, "\x1b[1m\x1b[31m" },
|
|
{LogLevel::warning, "\x1b[1m\x1b[33m" },
|
|
{LogLevel::notice, "\x1b[1m\x1b[34m" },
|
|
{LogLevel::info, "\x1b[1m\x1b[32m" },
|
|
{LogLevel::debug, "\x1b[1m\x1b[35m" },
|
|
{LogLevel::debug2, "\x1b[1m\x1b[36m" },
|
|
};
|
|
|
|
const static std::string suffix_ansireset = "\x1b[0m";
|
|
#endif
|
|
#endif
|
|
|
|
const static std::map<LogLevel, std::string> logLevel = {
|
|
{LogLevel::error, "ERROR " },
|
|
{LogLevel::warning, "WARNING" },
|
|
{LogLevel::notice, "NOTICE " },
|
|
{LogLevel::info, "INFO " },
|
|
{LogLevel::debug, "DEBUG " },
|
|
{LogLevel::debug2, "DEBUG2 " },
|
|
};
|
|
|
|
Logger::Logger(const LogLevel level) :
|
|
_level(level),
|
|
_ostreamMutex()
|
|
{
|
|
}
|
|
|
|
Logger::~Logger()
|
|
{
|
|
if (_logFile.is_open())
|
|
_logFile.close();
|
|
}
|
|
|
|
Logger &Logger::get()
|
|
{
|
|
static Logger l(DEFAULT_LOG_LEVEL);
|
|
return l;
|
|
}
|
|
|
|
void Logger::operator ()(const std::string &message,
|
|
const LogLevel level)
|
|
{
|
|
if(level > _level)
|
|
{
|
|
return;
|
|
}
|
|
|
|
std::ostringstream prefix;
|
|
|
|
auto t = std::time(nullptr);
|
|
auto tm = *std::localtime(&t);
|
|
|
|
#if defined(USE_COLORIZED_LOGS) && !defined(_WIN32)
|
|
prefix << prefix_ansicolor.at(level);
|
|
#endif
|
|
|
|
// GCC does not support put_time :-(
|
|
/*stringstream ss;
|
|
ss << std::put_time(&tm, "%d-%m-%Y %H-%M-%S");
|
|
prefix = "[" + ss.str() + "] ";*/
|
|
|
|
const size_t BUFFER_SIZE = 80;
|
|
char timeBuffer[BUFFER_SIZE];
|
|
std::strftime(timeBuffer, BUFFER_SIZE, "%Y-%m-%d %H:%M:%S", &tm);
|
|
prefix << "[" << timeBuffer << "] -- " << logLevel.at(level) << " :: ";
|
|
|
|
// Critical section
|
|
{
|
|
std::lock_guard<std::mutex> lock(_ostreamMutex);
|
|
|
|
#if defined(USE_COLORIZED_LOGS) && !defined(_WIN32)
|
|
if (_consoleActive) {
|
|
std::cerr << prefix.str() << message
|
|
<< suffix_ansireset << std::endl;
|
|
std::cerr.flush();
|
|
}
|
|
if (_logFileActive && _logFile.is_open()) {
|
|
_logFile << prefix.str() << message
|
|
<< suffix_ansireset << std::endl;
|
|
}
|
|
#elif defined(USE_COLORIZED_LOGS) && defined(_WIN32)
|
|
|
|
if (_consoleActive) {
|
|
// taken from GTEST
|
|
const HANDLE stdout_handle = GetStdHandle(STD_ERROR_HANDLE);
|
|
|
|
// Gets the current text color.
|
|
CONSOLE_SCREEN_BUFFER_INFO buffer_info;
|
|
GetConsoleScreenBufferInfo(stdout_handle, &buffer_info);
|
|
const WORD old_color_attrs = buffer_info.wAttributes;
|
|
|
|
// We need to flush the stream buffers into the console before each
|
|
// SetConsoleTextAttribute call lest it affect the text that is already
|
|
// printed but has not yet reached the console.
|
|
std::cerr.flush();
|
|
SetConsoleTextAttribute(stdout_handle,
|
|
colorAttribute.at(level) | FOREGROUND_INTENSITY);
|
|
|
|
std::cerr << prefix.str() << message << std::endl;
|
|
std::cerr.flush();
|
|
|
|
// Restores the text color.
|
|
SetConsoleTextAttribute(stdout_handle, old_color_attrs);
|
|
}
|
|
if (_logFileActive && _logFile.is_open()) {
|
|
_logFile << prefix.str() << message << std::endl;
|
|
}
|
|
|
|
#else
|
|
if (_consoleActive) {
|
|
std::cerr << prefix.str() << message << std::endl;
|
|
std::cerr.flush();
|
|
}
|
|
if (_logFileActive && _logFile.is_open()) {
|
|
_logFile << prefix.str() << message << std::endl;
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
|
|
void Logger::operator ()(const std::ostringstream &message, const LogLevel level)
|
|
{
|
|
(*this)(message.str(), level);
|
|
}
|
|
|
|
void Logger::operator ()(const LogLevel level, const char* format, ...)
|
|
{
|
|
if(level > _level)
|
|
{
|
|
return;
|
|
}
|
|
|
|
va_list argp;
|
|
char* message = nullptr;
|
|
|
|
// determine buffer length
|
|
va_start(argp, format);
|
|
int size = vsnprintf(nullptr, 0, format, argp) + 1;
|
|
va_end(argp);
|
|
|
|
if (size >= 0)
|
|
{
|
|
message = (char*)malloc(size);
|
|
if (message == nullptr)
|
|
{
|
|
(*this)("[LOGGER] CRITICAL: MEMORY ALLOCATION ERROR",
|
|
LogLevel::error);
|
|
}
|
|
|
|
va_start(argp,format);
|
|
vsnprintf(message, size, format, argp);
|
|
va_end(argp);
|
|
|
|
(*this)(std::string(message), level);
|
|
|
|
free(message);
|
|
} else {
|
|
(*this)("[LOGGER] CRITICAL: VSNPRINTF ERROR",
|
|
LogLevel::error);
|
|
}
|
|
}
|
|
|
|
void Logger::level(const LogLevel level)
|
|
{
|
|
_level = level;
|
|
}
|
|
|
|
LogLevel Logger::level() const
|
|
{
|
|
return _level;
|
|
}
|
|
|
|
bool Logger::console() const
|
|
{
|
|
return _consoleActive;
|
|
}
|
|
|
|
void Logger::console(const bool active)
|
|
{
|
|
std::lock_guard<std::mutex> lock(_ostreamMutex);
|
|
_consoleActive = active;
|
|
}
|
|
|
|
bool Logger::logFile() const
|
|
{
|
|
return _logFileActive;
|
|
}
|
|
|
|
void Logger::logFile(const bool active)
|
|
{
|
|
std::lock_guard<std::mutex> lock(_ostreamMutex);
|
|
_logFileActive = active;
|
|
}
|
|
|
|
void Logger::setLogFile(const std::string &logFile)
|
|
{
|
|
if (_logFile.is_open())
|
|
_logFile.close();
|
|
|
|
_logFile.open(logFile, std::ofstream::out | std::ofstream::app);
|
|
if (!_logFile.is_open())
|
|
throw std::runtime_error("Unable to open " + logFile);
|
|
}
|