1
0
mirror of https://github.com/PurpleI2P/i2pd.git synced 2025-01-19 21:59:58 +00:00
i2pd/Log.h

134 lines
2.3 KiB
C
Raw Normal View History

2013-12-10 08:00:13 -05:00
#ifndef LOG_H__
#define LOG_H__
2014-04-24 11:10:46 -04:00
#include <string>
2013-12-10 08:00:13 -05:00
#include <iostream>
#include <sstream>
2014-04-24 11:10:46 -04:00
#include <fstream>
2014-04-23 12:49:02 -04:00
#include <functional>
#include <chrono>
2013-12-10 08:00:13 -05:00
#include "Queue.h"
2014-10-28 16:36:17 -04:00
enum LogLevel
{
eLogError = 0,
eLogWarning,
eLogInfo,
eLogDebug,
2014-10-28 16:36:17 -04:00
eNumLogLevels
};
class Log;
2013-12-10 08:00:13 -05:00
struct LogMsg
{
std::stringstream s;
Log * log;
2014-10-28 16:36:17 -04:00
LogLevel level;
2013-12-10 08:00:13 -05:00
LogMsg (Log * l = nullptr, LogLevel lv = eLogInfo): log (l), level (lv) {};
void Process();
2013-12-10 08:00:13 -05:00
};
2014-04-23 12:49:02 -04:00
class Log: public i2p::util::MsgQueue<LogMsg>
{
public:
2014-04-23 12:49:02 -04:00
Log (): m_LogStream (nullptr) { SetOnEmpty (std::bind (&Log::Flush, this)); };
~Log () { delete m_LogStream; };
2014-04-24 11:10:46 -04:00
void SetLogFile (const std::string& fullFilePath);
void SetLogStream (std::ostream * logStream);
std::ostream * GetLogStream () const { return m_LogStream; };
const std::string& GetTimestamp ();
2014-04-23 12:49:02 -04:00
private:
2014-04-23 12:49:02 -04:00
void Flush ();
2014-04-24 11:10:46 -04:00
private:
std::ostream * m_LogStream;
std::string m_Timestamp;
#if !defined(__APPLE__)
2015-05-06 19:18:00 -04:00
#if (__GNUC__ == 4) && (__GNUC_MINOR__ <= 6) // gcc 4.6
std::chrono::monotonic_clock::time_point m_LastTimestampUpdate;
#else
std::chrono::steady_clock::time_point m_LastTimestampUpdate;
#endif
#else
std::chrono::steady_clock::time_point m_LastTimestampUpdate;
#endif
2014-04-23 12:49:02 -04:00
};
2014-07-02 13:48:45 -04:00
extern Log * g_Log;
inline void StartLog (const std::string& fullFilePath)
{
if (!g_Log)
{
2015-05-08 21:42:28 -04:00
auto log = new Log ();
2014-07-02 14:25:57 -04:00
if (fullFilePath.length () > 0)
2015-05-08 21:42:28 -04:00
log->SetLogFile (fullFilePath);
g_Log = log;
}
2014-07-02 13:48:45 -04:00
}
inline void StartLog (std::ostream * s)
{
if (!g_Log)
{
2015-05-08 21:42:28 -04:00
auto log = new Log ();
if (s)
2015-05-08 21:42:28 -04:00
log->SetLogStream (s);
g_Log = log;
}
}
2014-07-02 13:48:45 -04:00
inline void StopLog ()
{
if (g_Log)
{
2015-05-08 21:42:28 -04:00
auto log = g_Log;
2014-07-02 13:48:45 -04:00
g_Log = nullptr;
2015-05-08 21:42:28 -04:00
log->Stop ();
delete log;
}
2014-07-02 13:48:45 -04:00
}
2013-12-10 08:00:13 -05:00
template<typename TValue>
void LogPrint (std::stringstream& s, TValue arg)
2013-12-10 08:00:13 -05:00
{
s << arg;
2013-12-10 08:00:13 -05:00
}
2013-12-10 08:00:13 -05:00
template<typename TValue, typename... TArgs>
void LogPrint (std::stringstream& s, TValue arg, TArgs... args)
2013-12-10 08:00:13 -05:00
{
LogPrint (s, arg);
LogPrint (s, args...);
2013-12-10 08:00:13 -05:00
}
template<typename... TArgs>
void LogPrint (LogLevel level, TArgs... args)
2013-12-10 08:00:13 -05:00
{
LogMsg * msg = new LogMsg (g_Log, level);
2013-12-10 08:00:13 -05:00
LogPrint (msg->s, args...);
msg->s << std::endl;
if (g_Log)
2014-07-02 13:48:45 -04:00
g_Log->Put (msg);
else
{
msg->Process ();
delete msg;
}
2014-10-28 16:36:17 -04:00
}
template<typename... TArgs>
void LogPrint (TArgs... args)
2014-10-28 16:36:17 -04:00
{
LogPrint (eLogInfo, args...);
}
2013-12-10 08:00:13 -05:00
#endif