Crow/include/crow/logging.h

164 lines
4.2 KiB
C
Raw Normal View History

2014-05-20 16:17:56 +00:00
#pragma once
#include "crow/settings.h"
2014-07-08 09:26:55 +00:00
#include <cstdio>
#include <cstdlib>
#include <ctime>
#include <iostream>
#include <sstream>
#include <string>
namespace crow
2014-05-20 16:17:56 +00:00
{
2014-08-07 16:14:27 +00:00
enum class LogLevel
{
#ifndef ERROR
2021-06-02 15:05:52 +00:00
#ifndef DEBUG
DEBUG = 0,
2014-08-07 16:14:27 +00:00
INFO,
WARNING,
ERROR,
CRITICAL,
2021-06-02 15:05:52 +00:00
#endif
#endif
Debug = 0,
Info,
Warning,
Error,
Critical,
2014-08-07 16:14:27 +00:00
};
class ILogHandler
{
public:
virtual void log(std::string message, LogLevel level) = 0;
2014-08-07 16:14:27 +00:00
};
class CerrLogHandler : public ILogHandler
{
public:
void log(std::string message, LogLevel level) override
{
std::string prefix;
switch (level)
{
case LogLevel::Debug:
prefix = "DEBUG ";
break;
case LogLevel::Info:
prefix = "INFO ";
break;
case LogLevel::Warning:
prefix = "WARNING ";
break;
case LogLevel::Error:
prefix = "ERROR ";
break;
case LogLevel::Critical:
prefix = "CRITICAL";
break;
2014-08-07 16:14:27 +00:00
}
std::cerr << "(" << timestamp() << ") [" << prefix << "] " << message << std::endl;
}
2014-08-07 16:14:27 +00:00
private:
static std::string timestamp()
{
char date[32];
time_t t = time(0);
2015-02-20 02:07:23 +00:00
tm my_tm;
2015-02-20 02:07:23 +00:00
#if defined(_MSC_VER) || defined(__MINGW32__)
#ifdef CROW_USE_LOCALTIMEZONE
localtime_s(&my_tm, &t);
#else
gmtime_s(&my_tm, &t);
#endif
#else
#ifdef CROW_USE_LOCALTIMEZONE
localtime_r(&t, &my_tm);
2015-02-20 02:07:23 +00:00
#else
gmtime_r(&t, &my_tm);
#endif
2015-02-20 02:07:23 +00:00
#endif
size_t sz = strftime(date, sizeof(date), "%Y-%m-%d %H:%M:%S", &my_tm);
return std::string(date, date + sz);
}
};
2014-08-07 16:14:27 +00:00
class logger
{
public:
logger(LogLevel level):
level_(level)
{}
~logger()
{
#ifdef CROW_ENABLE_LOGGING
if (level_ >= get_current_log_level())
{
get_handler_ref()->log(stringstream_.str(), level_);
2014-08-07 16:14:27 +00:00
}
#endif
}
2014-08-07 16:14:27 +00:00
//
template<typename T>
logger& operator<<(T const& value)
{
#ifdef CROW_ENABLE_LOGGING
if (level_ >= get_current_log_level())
2014-08-07 16:14:27 +00:00
{
stringstream_ << value;
2014-08-07 16:14:27 +00:00
}
#endif
return *this;
}
//
static void setLogLevel(LogLevel level) { get_log_level_ref() = level; }
static void setHandler(ILogHandler* handler) { get_handler_ref() = handler; }
static LogLevel get_current_log_level() { return get_log_level_ref(); }
private:
//
static LogLevel& get_log_level_ref()
{
static LogLevel current_level = static_cast<LogLevel>(CROW_LOG_LEVEL);
return current_level;
}
static ILogHandler*& get_handler_ref()
{
static CerrLogHandler default_handler;
static ILogHandler* current_handler = &default_handler;
return current_handler;
}
//
std::ostringstream stringstream_;
LogLevel level_;
2014-08-07 16:14:27 +00:00
};
} // namespace crow
#define CROW_LOG_CRITICAL \
if (crow::logger::get_current_log_level() <= crow::LogLevel::Critical) \
crow::logger(crow::LogLevel::Critical)
#define CROW_LOG_ERROR \
if (crow::logger::get_current_log_level() <= crow::LogLevel::Error) \
crow::logger(crow::LogLevel::Error)
#define CROW_LOG_WARNING \
if (crow::logger::get_current_log_level() <= crow::LogLevel::Warning) \
crow::logger(crow::LogLevel::Warning)
#define CROW_LOG_INFO \
if (crow::logger::get_current_log_level() <= crow::LogLevel::Info) \
crow::logger(crow::LogLevel::Info)
#define CROW_LOG_DEBUG \
if (crow::logger::get_current_log_level() <= crow::LogLevel::Debug) \
crow::logger(crow::LogLevel::Debug)