mirror of https://github.com/PurpleI2P/i2pd.git
orignal
5 years ago
1 changed files with 236 additions and 236 deletions
@ -1,236 +1,236 @@
@@ -1,236 +1,236 @@
|
||||
/*
|
||||
* Copyright (c) 2013-2016, The PurpleI2P Project |
||||
* |
||||
* This file is part of Purple i2pd project and licensed under BSD3 |
||||
* |
||||
* See full license text in LICENSE file at top of project tree |
||||
*/ |
||||
|
||||
#ifndef LOG_H__ |
||||
#define LOG_H__ |
||||
|
||||
#include <ctime> |
||||
#include <string> |
||||
#include <iostream> |
||||
#include <fstream> |
||||
#include <sstream> |
||||
#include <chrono> |
||||
#include <memory> |
||||
#include <thread> |
||||
#include "Queue.h" |
||||
|
||||
#ifndef _WIN32 |
||||
#include <syslog.h> |
||||
#endif |
||||
|
||||
#ifdef WIN32_APP |
||||
#include <windows.h> // TODO: move away to win32app |
||||
#endif |
||||
|
||||
enum LogLevel |
||||
{ |
||||
eLogNone = 0, |
||||
eLogError, |
||||
eLogWarning, |
||||
eLogInfo, |
||||
eLogDebug, |
||||
eNumLogLevels |
||||
}; |
||||
|
||||
enum LogType { |
||||
eLogStdout = 0, |
||||
eLogStream, |
||||
eLogFile, |
||||
#ifndef _WIN32 |
||||
eLogSyslog, |
||||
#endif |
||||
}; |
||||
|
||||
namespace i2p { |
||||
namespace log { |
||||
|
||||
struct LogMsg; /* forward declaration */ |
||||
|
||||
class Log |
||||
{ |
||||
private: |
||||
|
||||
enum LogType m_Destination; |
||||
enum LogLevel m_MinLevel; |
||||
std::shared_ptr<std::ostream> m_LogStream; |
||||
std::string m_Logfile; |
||||
std::time_t m_LastTimestamp; |
||||
char m_LastDateTime[64]; |
||||
i2p::util::Queue<std::shared_ptr<LogMsg> > m_Queue; |
||||
bool m_HasColors; |
||||
std::string m_TimeFormat; |
||||
volatile bool m_IsRunning; |
||||
std::thread * m_Thread; |
||||
|
||||
private: |
||||
|
||||
/** prevent making copies */ |
||||
Log (const Log &); |
||||
const Log& operator=(const Log&); |
||||
|
||||
void Run (); |
||||
void Process (std::shared_ptr<LogMsg> msg); |
||||
|
||||
/**
|
||||
* @brief Makes formatted string from unix timestamp |
||||
* @param ts Second since epoch |
||||
* |
||||
* This function internally caches the result for last provided value |
||||
*/ |
||||
const char * TimeAsString(std::time_t ts); |
||||
|
||||
public: |
||||
|
||||
Log (); |
||||
~Log (); |
||||
|
||||
LogType GetLogType () { return m_Destination; }; |
||||
LogLevel GetLogLevel () { return m_MinLevel; }; |
||||
|
||||
void Start (); |
||||
void Stop (); |
||||
|
||||
/**
|
||||
* @brief Sets minimal allowed level for log messages |
||||
* @param level String with wanted minimal msg level |
||||
*/ |
||||
void SetLogLevel (const std::string& level); |
||||
|
||||
/**
|
||||
* @brief Sets log destination to logfile |
||||
* @param path Path to logfile |
||||
*/ |
||||
void SendTo (const std::string &path); |
||||
|
||||
/**
|
||||
* @brief Sets log destination to given output stream |
||||
* @param os Output stream |
||||
*/ |
||||
void SendTo (std::shared_ptr<std::ostream> os); |
||||
|
||||
/**
|
||||
* @brief Sets format for timestamps in log |
||||
* @param format String with timestamp format |
||||
*/ |
||||
void SetTimeFormat (std::string format) { m_TimeFormat = format; }; |
||||
|
||||
#ifndef _WIN32 |
||||
/**
|
||||
* @brief Sets log destination to syslog |
||||
* @param name Wanted program name |
||||
* @param facility Wanted log category |
||||
*/ |
||||
void SendTo (const char *name, int facility); |
||||
#endif |
||||
|
||||
/**
|
||||
* @brief Format log message and write to output stream/syslog |
||||
* @param msg Pointer to processed message |
||||
*/ |
||||
void Append(std::shared_ptr<i2p::log::LogMsg> &); |
||||
|
||||
/** @brief Reopen log file */ |
||||
void Reopen(); |
||||
}; |
||||
|
||||
/**
|
||||
* @struct LogMsg |
||||
* @brief Log message container |
||||
* |
||||
* We creating it somewhere with LogPrint(), |
||||
* then put in MsgQueue for later processing. |
||||
*/ |
||||
struct LogMsg { |
||||
std::time_t timestamp; |
||||
std::string text; /**< message text as single string */ |
||||
LogLevel level; /**< message level */ |
||||
std::thread::id tid; /**< id of thread that generated message */ |
||||
|
||||
LogMsg (LogLevel lvl, std::time_t ts, const std::string & txt): timestamp(ts), text(txt), level(lvl) {}; |
||||
}; |
||||
|
||||
Log & Logger(); |
||||
} // log
|
||||
} |
||||
|
||||
/** internal usage only -- folding args array to single string */ |
||||
template<typename TValue> |
||||
void LogPrint (std::stringstream& s, TValue&& arg) noexcept |
||||
{ |
||||
s << std::forward<TValue>(arg); |
||||
} |
||||
|
||||
#if (__cplusplus < 201703L) // below C++ 17
|
||||
/** internal usage only -- folding args array to single string */ |
||||
template<typename TValue, typename... TArgs> |
||||
void LogPrint (std::stringstream& s, TValue&& arg, TArgs&&... args) noexcept |
||||
{ |
||||
LogPrint (s, std::forward<TValue>(arg)); |
||||
LogPrint (s, std::forward<TArgs>(args)...); |
||||
} |
||||
#endif |
||||
|
||||
/**
|
||||
* @brief Create log message and send it to queue |
||||
* @param level Message level (eLogError, eLogInfo, ...) |
||||
* @param args Array of message parts |
||||
*/ |
||||
template<typename... TArgs> |
||||
void LogPrint (LogLevel level, TArgs&&... args) noexcept |
||||
{ |
||||
i2p::log::Log &log = i2p::log::Logger(); |
||||
if (level > log.GetLogLevel ()) |
||||
return; |
||||
|
||||
// fold message to single string
|
||||
std::stringstream ss(""); |
||||
|
||||
#if (__cplusplus >= 201703L) // C++ 17 or higher
|
||||
(LogPrint (ss, std::forward<TArgs>(args)), ...); |
||||
#else |
||||
LogPrint (ss, std::forward<TArgs>(args)...); |
||||
#endif |
||||
|
||||
auto msg = std::make_shared<i2p::log::LogMsg>(level, std::time(nullptr), ss.str()); |
||||
msg->tid = std::this_thread::get_id(); |
||||
log.Append(msg); |
||||
} |
||||
|
||||
#ifdef WIN32_APP |
||||
/**
|
||||
* @brief Show message box for user with message in it |
||||
* @param level Message level (eLogError, eLogInfo, ...) |
||||
* @param args Array of message parts |
||||
*/ |
||||
template<typename... TArgs> |
||||
void ShowMessageBox (LogLevel level, TArgs&&... args) noexcept |
||||
{ |
||||
// fold message to single string
|
||||
std::stringstream ss(""); |
||||
|
||||
#if (__cplusplus >= 201703L) // C++ 17 or higher
|
||||
(LogPrint (ss, std::forward<TArgs>(args)), ...); |
||||
#else |
||||
LogPrint (ss, std::forward<TArgs>(args)...); |
||||
#endif |
||||
|
||||
UINT uType; |
||||
switch (level) |
||||
{ |
||||
case eLogError : |
||||
case eLogWarning : |
||||
uType = MB_ICONWARNING; |
||||
break; |
||||
default : |
||||
uType = MB_ICONINFORMATION; |
||||
} |
||||
MessageBox(0, TEXT(ss.str ().c_str ()), TEXT("i2pd"), uType | MB_TASKMODAL | MB_OK ); |
||||
} |
||||
#endif // WIN32_APP
|
||||
|
||||
#endif // LOG_H__
|
||||
/*
|
||||
* Copyright (c) 2013-2016, The PurpleI2P Project |
||||
* |
||||
* This file is part of Purple i2pd project and licensed under BSD3 |
||||
* |
||||
* See full license text in LICENSE file at top of project tree |
||||
*/ |
||||
|
||||
#ifndef LOG_H__ |
||||
#define LOG_H__ |
||||
|
||||
#include <ctime> |
||||
#include <string> |
||||
#include <iostream> |
||||
#include <fstream> |
||||
#include <sstream> |
||||
#include <chrono> |
||||
#include <memory> |
||||
#include <thread> |
||||
#include "Queue.h" |
||||
|
||||
#ifndef _WIN32 |
||||
#include <syslog.h> |
||||
#endif |
||||
|
||||
#ifdef WIN32_APP |
||||
#include <windows.h> // TODO: move away to win32app |
||||
#endif |
||||
|
||||
enum LogLevel |
||||
{ |
||||
eLogNone = 0, |
||||
eLogError, |
||||
eLogWarning, |
||||
eLogInfo, |
||||
eLogDebug, |
||||
eNumLogLevels |
||||
}; |
||||
|
||||
enum LogType { |
||||
eLogStdout = 0, |
||||
eLogStream, |
||||
eLogFile, |
||||
#ifndef _WIN32 |
||||
eLogSyslog, |
||||
#endif |
||||
}; |
||||
|
||||
namespace i2p { |
||||
namespace log { |
||||
|
||||
struct LogMsg; /* forward declaration */ |
||||
|
||||
class Log |
||||
{ |
||||
private: |
||||
|
||||
enum LogType m_Destination; |
||||
enum LogLevel m_MinLevel; |
||||
std::shared_ptr<std::ostream> m_LogStream; |
||||
std::string m_Logfile; |
||||
std::time_t m_LastTimestamp; |
||||
char m_LastDateTime[64]; |
||||
i2p::util::Queue<std::shared_ptr<LogMsg> > m_Queue; |
||||
bool m_HasColors; |
||||
std::string m_TimeFormat; |
||||
volatile bool m_IsRunning; |
||||
std::thread * m_Thread; |
||||
|
||||
private: |
||||
|
||||
/** prevent making copies */ |
||||
Log (const Log &); |
||||
const Log& operator=(const Log&); |
||||
|
||||
void Run (); |
||||
void Process (std::shared_ptr<LogMsg> msg); |
||||
|
||||
/**
|
||||
* @brief Makes formatted string from unix timestamp |
||||
* @param ts Second since epoch |
||||
* |
||||
* This function internally caches the result for last provided value |
||||
*/ |
||||
const char * TimeAsString(std::time_t ts); |
||||
|
||||
public: |
||||
|
||||
Log (); |
||||
~Log (); |
||||
|
||||
LogType GetLogType () { return m_Destination; }; |
||||
LogLevel GetLogLevel () { return m_MinLevel; }; |
||||
|
||||
void Start (); |
||||
void Stop (); |
||||
|
||||
/**
|
||||
* @brief Sets minimal allowed level for log messages |
||||
* @param level String with wanted minimal msg level |
||||
*/ |
||||
void SetLogLevel (const std::string& level); |
||||
|
||||
/**
|
||||
* @brief Sets log destination to logfile |
||||
* @param path Path to logfile |
||||
*/ |
||||
void SendTo (const std::string &path); |
||||
|
||||
/**
|
||||
* @brief Sets log destination to given output stream |
||||
* @param os Output stream |
||||
*/ |
||||
void SendTo (std::shared_ptr<std::ostream> os); |
||||
|
||||
/**
|
||||
* @brief Sets format for timestamps in log |
||||
* @param format String with timestamp format |
||||
*/ |
||||
void SetTimeFormat (std::string format) { m_TimeFormat = format; }; |
||||
|
||||
#ifndef _WIN32 |
||||
/**
|
||||
* @brief Sets log destination to syslog |
||||
* @param name Wanted program name |
||||
* @param facility Wanted log category |
||||
*/ |
||||
void SendTo (const char *name, int facility); |
||||
#endif |
||||
|
||||
/**
|
||||
* @brief Format log message and write to output stream/syslog |
||||
* @param msg Pointer to processed message |
||||
*/ |
||||
void Append(std::shared_ptr<i2p::log::LogMsg> &); |
||||
|
||||
/** @brief Reopen log file */ |
||||
void Reopen(); |
||||
}; |
||||
|
||||
/**
|
||||
* @struct LogMsg |
||||
* @brief Log message container |
||||
* |
||||
* We creating it somewhere with LogPrint(), |
||||
* then put in MsgQueue for later processing. |
||||
*/ |
||||
struct LogMsg { |
||||
std::time_t timestamp; |
||||
std::string text; /**< message text as single string */ |
||||
LogLevel level; /**< message level */ |
||||
std::thread::id tid; /**< id of thread that generated message */ |
||||
|
||||
LogMsg (LogLevel lvl, std::time_t ts, const std::string & txt): timestamp(ts), text(txt), level(lvl) {}; |
||||
}; |
||||
|
||||
Log & Logger(); |
||||
} // log
|
||||
} |
||||
|
||||
/** internal usage only -- folding args array to single string */ |
||||
template<typename TValue> |
||||
void LogPrint (std::stringstream& s, TValue&& arg) noexcept |
||||
{ |
||||
s << std::forward<TValue>(arg); |
||||
} |
||||
|
||||
#if (__cplusplus < 201703L) // below C++ 17
|
||||
/** internal usage only -- folding args array to single string */ |
||||
template<typename TValue, typename... TArgs> |
||||
void LogPrint (std::stringstream& s, TValue&& arg, TArgs&&... args) noexcept |
||||
{ |
||||
LogPrint (s, std::forward<TValue>(arg)); |
||||
LogPrint (s, std::forward<TArgs>(args)...); |
||||
} |
||||
#endif |
||||
|
||||
/**
|
||||
* @brief Create log message and send it to queue |
||||
* @param level Message level (eLogError, eLogInfo, ...) |
||||
* @param args Array of message parts |
||||
*/ |
||||
template<typename... TArgs> |
||||
void LogPrint (LogLevel level, TArgs&&... args) noexcept |
||||
{ |
||||
i2p::log::Log &log = i2p::log::Logger(); |
||||
if (level > log.GetLogLevel ()) |
||||
return; |
||||
|
||||
// fold message to single string
|
||||
std::stringstream ss(""); |
||||
|
||||
#if (__cplusplus >= 201703L) // C++ 17 or higher
|
||||
(LogPrint (ss, std::forward<TArgs>(args)), ...); |
||||
#else |
||||
LogPrint (ss, std::forward<TArgs>(args)...); |
||||
#endif |
||||
|
||||
auto msg = std::make_shared<i2p::log::LogMsg>(level, std::time(nullptr), ss.str()); |
||||
msg->tid = std::this_thread::get_id(); |
||||
log.Append(msg); |
||||
} |
||||
|
||||
#ifdef WIN32_APP |
||||
/**
|
||||
* @brief Show message box for user with message in it |
||||
* @param level Message level (eLogError, eLogInfo, ...) |
||||
* @param args Array of message parts |
||||
*/ |
||||
template<typename... TArgs> |
||||
void ShowMessageBox (LogLevel level, TArgs&&... args) noexcept |
||||
{ |
||||
// fold message to single string
|
||||
std::stringstream ss(""); |
||||
|
||||
#if (__cplusplus >= 201703L) // C++ 17 or higher
|
||||
(LogPrint (ss, std::forward<TArgs>(args)), ...); |
||||
#else |
||||
LogPrint (ss, std::forward<TArgs>(args)...); |
||||
#endif |
||||
|
||||
UINT uType; |
||||
switch (level) |
||||
{ |
||||
case eLogError : |
||||
case eLogWarning : |
||||
uType = MB_ICONWARNING; |
||||
break; |
||||
default : |
||||
uType = MB_ICONINFORMATION; |
||||
} |
||||
MessageBox(0, TEXT(ss.str ().c_str ()), TEXT("i2pd"), uType | MB_TASKMODAL | MB_OK ); |
||||
} |
||||
#endif // WIN32_APP
|
||||
|
||||
#endif // LOG_H__
|
||||
|
Loading…
Reference in new issue