1
0
mirror of https://github.com/PurpleI2P/i2pd.git synced 2025-01-10 16:37:54 +00:00
i2pd/Log.h

208 lines
4.5 KiB
C
Raw Normal View History

2016-03-27 00:00:00 +00:00
/*
* 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
*/
2013-12-10 13:00:13 +00:00
#ifndef LOG_H__
#define LOG_H__
2016-03-27 00:00:00 +00:00
#include <ctime>
2014-04-24 15:10:46 +00:00
#include <string>
2013-12-10 13:00:13 +00:00
#include <iostream>
2014-04-24 15:10:46 +00:00
#include <fstream>
2016-03-27 00:00:00 +00:00
#include <sstream>
#include <chrono>
2016-02-04 17:36:58 +00:00
#include <memory>
2013-12-10 13:00:13 +00:00
#include "Queue.h"
2016-03-26 13:49:45 +00:00
#ifndef _WIN32
#include <syslog.h>
#endif
2014-10-28 20:36:17 +00:00
enum LogLevel
{
eLogError = 0,
eLogWarning,
eLogInfo,
eLogDebug,
eNumLogLevels
};
2016-03-27 00:00:00 +00:00
enum LogType {
eLogStdout = 0,
eLogStream,
eLogFile,
#ifndef _WIN32
eLogSyslog,
2016-03-26 13:40:19 +00:00
#endif
2014-04-23 16:49:02 +00:00
};
2016-07-28 13:49:43 +00:00
#ifdef _WIN32
const char LOG_COLOR_ERROR[] = "";
const char LOG_COLOR_WARNING[] = "";
const char LOG_COLOR_RESET[] = "";
#else
const char LOG_COLOR_ERROR[] = "\033[1;31m";
const char LOG_COLOR_WARNING[] = "\033[1;33m";
const char LOG_COLOR_RESET[] = "\033[0m";
#endif
2016-03-27 00:00:00 +00:00
namespace i2p {
namespace log {
2016-07-28 13:49:43 +00:00
2016-03-27 00:00:00 +00:00
struct LogMsg; /* forward declaration */
2014-07-02 17:48:45 +00:00
2016-03-27 00:00:00 +00:00
class Log
2014-07-02 17:48:45 +00:00
{
2016-03-27 00:00:00 +00:00
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;
volatile bool m_IsReady;
mutable std::mutex m_OutputLock;
private:
/** prevent making copies */
Log (const Log &);
const Log& operator=(const Log&);
/**
* @brief process stored messages in queue
*/
void Process ();
/**
* @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; };
/**
2016-05-31 00:00:00 +00:00
* @brief Sets minimal allowed level for log messages
2016-03-27 00:00:00 +00:00
* @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
*/
2016-05-31 00:00:00 +00:00
void SendTo (std::shared_ptr<std::ostream> os);
2016-03-27 00:00:00 +00:00
#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> &);
2016-03-28 14:00:00 +00:00
/** @brief Allow log output */
void Ready() { m_IsReady = true; }
2016-03-27 00:00:00 +00:00
/** @brief Flushes the output log stream */
void Flush();
/** @brief Reopen log file */
void Reopen();
};
/**
2016-05-31 00:00:00 +00:00
* @struct LogMsg
* @brief Log message container
2016-03-27 00:00:00 +00:00
*
* 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 */
2016-03-28 14:00:00 +00:00
std::thread::id tid; /**< id of thread that generated message */
2016-03-27 00:00:00 +00:00
LogMsg (LogLevel lvl, std::time_t ts, const std::string & txt): timestamp(ts), text(txt), level(lvl) {};
2016-03-27 00:00:00 +00:00
};
Log & Logger();
} // log
} // i2p
/** internal usage only -- folding args array to single string */
2013-12-10 13:00:13 +00:00
template<typename TValue>
2016-03-27 00:00:00 +00:00
void LogPrint (std::stringstream& s, TValue arg)
2013-12-10 13:00:13 +00:00
{
s << arg;
}
2016-03-27 00:00:00 +00:00
/** internal usage only -- folding args array to single string */
2013-12-10 13:00:13 +00:00
template<typename TValue, typename... TArgs>
2016-03-27 00:00:00 +00:00
void LogPrint (std::stringstream& s, TValue arg, TArgs... args)
2013-12-10 13:00:13 +00:00
{
LogPrint (s, arg);
LogPrint (s, args...);
}
2016-03-27 00:00:00 +00:00
/**
* @brief Create log message and send it to queue
* @param level Message level (eLogError, eLogInfo, ...)
* @param args Array of message parts
*/
2013-12-10 13:00:13 +00:00
template<typename... TArgs>
2016-03-27 00:00:00 +00:00
void LogPrint (LogLevel level, TArgs... args)
2013-12-10 13:00:13 +00:00
{
2016-03-27 00:00:00 +00:00
i2p::log::Log &log = i2p::log::Logger();
if (level > log.GetLogLevel ())
2015-12-28 00:00:00 +00:00
return;
2016-03-27 00:00:00 +00:00
// fold message to single string
std::stringstream ss("");
2016-07-28 13:49:43 +00:00
if(level == eLogError) // if log level is ERROR color log message red
ss << LOG_COLOR_ERROR;
else if (level == eLogWarning) // if log level is WARN color log message yellow
ss << LOG_COLOR_WARNING;
LogPrint (ss, args ...);
// reset color
ss << LOG_COLOR_RESET;
2016-03-27 00:00:00 +00:00
auto msg = std::make_shared<i2p::log::LogMsg>(level, std::time(nullptr), ss.str());
2016-03-28 14:00:00 +00:00
msg->tid = std::this_thread::get_id();
2016-03-27 00:00:00 +00:00
log.Append(msg);
2014-10-28 20:36:17 +00:00
}
2016-03-27 00:00:00 +00:00
#endif // LOG_H__