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

209 lines
4.6 KiB
C
Raw Normal View History

2020-05-05 12:27:56 +00:00
/*
2024-02-09 02:45:57 +00:00
* Copyright (c) 2013-2024, The PurpleI2P Project
2020-05-05 12:27:56 +00:00
*
* 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>
2020-05-05 15:13:59 +00:00
#include <functional>
2020-05-05 12:27:56 +00:00
#include "Queue.h"
#ifndef _WIN32
#include <syslog.h>
#endif
enum LogLevel
{
eLogNone = 0,
2023-03-30 21:31:12 +00:00
eLogCritical,
2020-05-05 12:27:56 +00:00
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;
2020-05-05 12:27:56 +00:00
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
2020-05-05 12:27:56 +00:00
*
* This function internally caches the result for last provided value
*/
const char * TimeAsString(std::time_t ts);
public:
Log ();
~Log ();
2024-02-09 02:45:57 +00:00
LogType GetLogType () const { return m_Destination; };
LogLevel GetLogLevel () const { return m_MinLevel; };
2020-05-05 12:27:56 +00:00
void Start ();
void Stop ();
/**
* @brief Sets minimal allowed level for log messages
* @param level String with wanted minimal msg level
2020-05-05 12:27:56 +00:00
*/
void SetLogLevel (const std::string& level);
2020-05-05 12:27:56 +00:00
/**
* @brief Sets log destination to logfile
* @param path Path to logfile
2020-05-05 12:27:56 +00:00
*/
void SendTo (const std::string &path);
/**
* @brief Sets log destination to given output stream
* @param os Output stream
2020-05-05 12:27:56 +00:00
*/
void SendTo (std::shared_ptr<std::ostream> os);
/**
* @brief Sets format for timestamps in log
* @param format String with timestamp format
2020-05-05 12:27:56 +00:00
*/
void SetTimeFormat (std::string format) { m_TimeFormat = format; };
#ifndef _WIN32
/**
* @brief Sets log destination to syslog
* @param name Wanted program name
2020-05-05 12:27:56 +00:00
* @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
2020-05-05 12:27:56 +00:00
*/
void Append(std::shared_ptr<i2p::log::LogMsg> &);
/** @brief Reopen log file */
2020-05-05 12:27:56 +00:00
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 */
2020-05-05 12:27:56 +00:00
std::thread::id tid; /**< id of thread that generated message */
2021-03-07 20:55:55 +00:00
LogMsg (LogLevel lvl, std::time_t ts, std::string&& txt): timestamp(ts), text(std::move(txt)), level(lvl) {}
2020-05-05 12:27:56 +00:00
};
Log & Logger();
2020-05-05 15:13:59 +00:00
typedef std::function<void (const std::string&)> ThrowFunction;
2020-05-05 15:13:59 +00:00
ThrowFunction GetThrowFunction ();
void SetThrowFunction (ThrowFunction f);
2020-05-05 12:27:56 +00:00
} // log
} // i2p
2020-05-05 12:27:56 +00:00
2024-02-09 02:45:57 +00:00
inline bool CheckLogLevel (LogLevel level) noexcept
{
return level <= i2p::log::Logger().GetLogLevel ();
}
2020-05-05 12:27:56 +00:00
/** 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);
}
/**
* @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
{
2024-02-09 02:45:57 +00:00
if (!CheckLogLevel (level)) return;
2020-05-05 12:27:56 +00:00
// fold message to single string
2021-03-07 20:55:55 +00:00
std::stringstream ss;
2020-05-05 12:27:56 +00:00
(LogPrint (ss, std::forward<TArgs>(args)), ...);
2021-03-07 20:55:55 +00:00
auto msg = std::make_shared<i2p::log::LogMsg>(level, std::time(nullptr), std::move(ss).str());
2020-05-05 12:27:56 +00:00
msg->tid = std::this_thread::get_id();
2024-02-09 02:45:57 +00:00
i2p::log::Logger().Append(msg);
2020-05-05 12:27:56 +00:00
}
/**
* @brief Throw fatal error message with the list of arguments
2020-05-05 12:27:56 +00:00
* @param args Array of message parts
*/
template<typename... TArgs>
2020-05-05 13:35:41 +00:00
void ThrowFatal (TArgs&&... args) noexcept
2020-05-05 12:27:56 +00:00
{
2020-05-05 15:13:59 +00:00
auto f = i2p::log::GetThrowFunction ();
if (!f) return;
2020-05-05 12:27:56 +00:00
// fold message to single string
std::stringstream ss("");
(LogPrint (ss, std::forward<TArgs>(args)), ...);
2020-05-05 15:13:59 +00:00
f (ss.str ());
2020-05-05 12:27:56 +00:00
}
#endif // LOG_H__