mirror of https://github.com/PurpleI2P/i2pd.git
I2P: End-to-End encrypted and anonymous Internet
https://i2pd.website/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
249 lines
6.0 KiB
249 lines
6.0 KiB
/* |
|
* Copyright (c) 2013-2020, 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 |
|
*/ |
|
|
|
#include "Log.h" |
|
#include "util.h" |
|
|
|
//for std::transform |
|
#include <algorithm> |
|
|
|
namespace i2p { |
|
namespace log { |
|
static Log logger; |
|
/** |
|
* @brief Maps our loglevel to their symbolic name |
|
*/ |
|
static const char * g_LogLevelStr[eNumLogLevels] = |
|
{ |
|
"none", // eLogNone |
|
"error", // eLogError |
|
"warn", // eLogWarn |
|
"info", // eLogInfo |
|
"debug" // eLogDebug |
|
}; |
|
|
|
/** |
|
* @brief Colorize log output -- array of terminal control sequences |
|
* @note Using ISO 6429 (ANSI) color sequences |
|
*/ |
|
#ifdef _WIN32 |
|
static const char *LogMsgColors[] = { "", "", "", "", "", "" }; |
|
#else /* UNIX */ |
|
static const char *LogMsgColors[] = { |
|
[eLogNone] = "\033[0m", /* reset */ |
|
[eLogError] = "\033[1;31m", /* red */ |
|
[eLogWarning] = "\033[1;33m", /* yellow */ |
|
[eLogInfo] = "\033[1;36m", /* cyan */ |
|
[eLogDebug] = "\033[1;34m", /* blue */ |
|
[eNumLogLevels] = "\033[0m", /* reset */ |
|
}; |
|
#endif |
|
|
|
#ifndef _WIN32 |
|
/** |
|
* @brief Maps our log levels to syslog one |
|
* @return syslog priority LOG_*, as defined in syslog.h |
|
*/ |
|
static inline int GetSyslogPrio (enum LogLevel l) { |
|
int priority = LOG_DEBUG; |
|
switch (l) { |
|
case eLogNone : priority = LOG_CRIT; break; |
|
case eLogError : priority = LOG_ERR; break; |
|
case eLogWarning : priority = LOG_WARNING; break; |
|
case eLogInfo : priority = LOG_INFO; break; |
|
case eLogDebug : priority = LOG_DEBUG; break; |
|
default : priority = LOG_DEBUG; break; |
|
} |
|
return priority; |
|
} |
|
#endif |
|
|
|
Log::Log(): |
|
m_Destination(eLogStdout), m_MinLevel(eLogInfo), |
|
m_LogStream (nullptr), m_Logfile(""), m_HasColors(true), m_TimeFormat("%H:%M:%S"), |
|
m_IsRunning (false), m_Thread (nullptr) |
|
{ |
|
} |
|
|
|
Log::~Log () |
|
{ |
|
delete m_Thread; |
|
} |
|
|
|
void Log::Start () |
|
{ |
|
if (!m_IsRunning) |
|
{ |
|
m_IsRunning = true; |
|
m_Thread = new std::thread (std::bind (&Log::Run, this)); |
|
} |
|
} |
|
|
|
void Log::Stop () |
|
{ |
|
switch (m_Destination) |
|
{ |
|
#ifndef _WIN32 |
|
case eLogSyslog : |
|
closelog(); |
|
break; |
|
#endif |
|
case eLogFile: |
|
case eLogStream: |
|
if (m_LogStream) m_LogStream->flush(); |
|
break; |
|
default: |
|
/* do nothing */ |
|
break; |
|
} |
|
m_IsRunning = false; |
|
m_Queue.WakeUp (); |
|
if (m_Thread) |
|
{ |
|
m_Thread->join (); |
|
delete m_Thread; |
|
m_Thread = nullptr; |
|
} |
|
} |
|
|
|
std::string str_tolower(std::string s) { |
|
std::transform(s.begin(), s.end(), s.begin(), |
|
// static_cast<int(*)(int)>(std::tolower) // wrong |
|
// [](int c){ return std::tolower(c); } // wrong |
|
// [](char c){ return std::tolower(c); } // wrong |
|
[](unsigned char c){ return std::tolower(c); } // correct |
|
); |
|
return s; |
|
} |
|
|
|
void Log::SetLogLevel (const std::string& level_) { |
|
std::string level=str_tolower(level_); |
|
if (level == "none") { m_MinLevel = eLogNone; } |
|
else if (level == "error") { m_MinLevel = eLogError; } |
|
else if (level == "warn") { m_MinLevel = eLogWarning; } |
|
else if (level == "info") { m_MinLevel = eLogInfo; } |
|
else if (level == "debug") { m_MinLevel = eLogDebug; } |
|
else { |
|
LogPrint(eLogError, "Log: unknown loglevel: ", level); |
|
return; |
|
} |
|
LogPrint(eLogInfo, "Log: min messages level set to ", level); |
|
} |
|
|
|
const char * Log::TimeAsString(std::time_t t) { |
|
if (t != m_LastTimestamp) { |
|
strftime(m_LastDateTime, sizeof(m_LastDateTime), m_TimeFormat.c_str(), localtime(&t)); |
|
m_LastTimestamp = t; |
|
} |
|
return m_LastDateTime; |
|
} |
|
|
|
/** |
|
* @note This function better to be run in separate thread due to disk i/o. |
|
* Unfortunately, with current startup process with late fork() this |
|
* will give us nothing but pain. Maybe later. See in NetDb as example. |
|
*/ |
|
void Log::Process(std::shared_ptr<LogMsg> msg) |
|
{ |
|
if (!msg) return; |
|
std::hash<std::thread::id> hasher; |
|
unsigned short short_tid; |
|
short_tid = (short) (hasher(msg->tid) % 1000); |
|
switch (m_Destination) { |
|
#ifndef _WIN32 |
|
case eLogSyslog: |
|
syslog(GetSyslogPrio(msg->level), "[%03u] %s", short_tid, msg->text.c_str()); |
|
break; |
|
#endif |
|
case eLogFile: |
|
case eLogStream: |
|
if (m_LogStream) |
|
*m_LogStream << TimeAsString(msg->timestamp) |
|
<< "@" << short_tid |
|
<< "/" << g_LogLevelStr[msg->level] |
|
<< " - " << msg->text << std::endl; |
|
break; |
|
case eLogStdout: |
|
default: |
|
std::cout << TimeAsString(msg->timestamp) |
|
<< "@" << short_tid |
|
<< "/" << LogMsgColors[msg->level] << g_LogLevelStr[msg->level] << LogMsgColors[eNumLogLevels] |
|
<< " - " << msg->text << std::endl; |
|
break; |
|
} // switch |
|
} |
|
|
|
void Log::Run () |
|
{ |
|
i2p::util::SetThreadName("Logging"); |
|
|
|
Reopen (); |
|
while (m_IsRunning) |
|
{ |
|
std::shared_ptr<LogMsg> msg; |
|
while ((msg = m_Queue.Get ())) |
|
Process (msg); |
|
if (m_LogStream) m_LogStream->flush(); |
|
if (m_IsRunning) |
|
m_Queue.Wait (); |
|
} |
|
} |
|
|
|
void Log::Append(std::shared_ptr<i2p::log::LogMsg> & msg) |
|
{ |
|
m_Queue.Put(msg); |
|
} |
|
|
|
void Log::SendTo (const std::string& path) |
|
{ |
|
if (m_LogStream) m_LogStream = nullptr; // close previous |
|
auto flags = std::ofstream::out | std::ofstream::app; |
|
auto os = std::make_shared<std::ofstream> (path, flags); |
|
if (os->is_open ()) |
|
{ |
|
m_HasColors = false; |
|
m_Logfile = path; |
|
m_Destination = eLogFile; |
|
m_LogStream = os; |
|
return; |
|
} |
|
LogPrint(eLogError, "Log: can't open file ", path); |
|
} |
|
|
|
void Log::SendTo (std::shared_ptr<std::ostream> os) { |
|
m_HasColors = false; |
|
m_Destination = eLogStream; |
|
m_LogStream = os; |
|
} |
|
|
|
#ifndef _WIN32 |
|
void Log::SendTo(const char *name, int facility) { |
|
if (m_MinLevel == eLogNone) return; |
|
m_HasColors = false; |
|
m_Destination = eLogSyslog; |
|
m_LogStream = nullptr; |
|
openlog(name, LOG_CONS | LOG_PID, facility); |
|
} |
|
#endif |
|
|
|
void Log::Reopen() { |
|
if (m_Destination == eLogFile) |
|
SendTo(m_Logfile); |
|
} |
|
|
|
Log & Logger() { |
|
return logger; |
|
} |
|
|
|
static ThrowFunction g_ThrowFunction; |
|
ThrowFunction GetThrowFunction () { return g_ThrowFunction; } |
|
void SetThrowFunction (ThrowFunction f) { g_ThrowFunction = f; } |
|
|
|
} // log |
|
} // i2p |
|
|
|
|