mirror of https://github.com/PurpleI2P/i2pd.git
hagen
9 years ago
8 changed files with 309 additions and 320 deletions
@ -1,133 +1,160 @@ |
|||||||
#include <boost/date_time/posix_time/posix_time.hpp> |
/*
|
||||||
|
* 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 |
||||||
|
*/ |
||||||
|
|
||||||
#include "Log.h" |
#include "Log.h" |
||||||
|
|
||||||
Log * g_Log = nullptr; |
namespace i2p { |
||||||
|
namespace log { |
||||||
|
Log logger; |
||||||
|
/**
|
||||||
|
* @enum Maps our loglevel to their symbolic name |
||||||
|
*/ |
||||||
|
static const char * g_LogLevelStr[eNumLogLevels] = |
||||||
|
{ |
||||||
|
"error", // eLogError
|
||||||
|
"warn", // eLogWarn
|
||||||
|
"info", // eLogInfo
|
||||||
|
"debug" // eLogDebug
|
||||||
|
}; |
||||||
|
|
||||||
static const char * g_LogLevelStr[eNumLogLevels] = |
|
||||||
{ |
|
||||||
"error", // eLogError
|
|
||||||
"warn", // eLogWarning
|
|
||||||
"info", // eLogInfo
|
|
||||||
"debug" // eLogDebug
|
|
||||||
}; |
|
||||||
#ifndef _WIN32 |
#ifndef _WIN32 |
||||||
/** convert LogLevel enum to syslog priority level */ |
/**
|
||||||
static int ToSyslogLevel(LogLevel lvl) |
* @brief Maps our log levels to syslog one |
||||||
{ |
* @return syslog priority LOG_*, as defined in syslog.h |
||||||
switch (lvl) { |
*/ |
||||||
case eLogError: |
static inline int GetSyslogPrio (enum LogLevel l) { |
||||||
return LOG_ERR; |
int priority = LOG_DEBUG; |
||||||
case eLogWarning: |
switch (l) { |
||||||
return LOG_WARNING; |
case eLogError : priority = LOG_ERR; break; |
||||||
case eLogInfo: |
case eLogWarning : priority = LOG_WARNING; break; |
||||||
return LOG_INFO; |
case eLogInfo : priority = LOG_INFO; break; |
||||||
case eLogDebug: |
case eLogDebug : priority = LOG_DEBUG; break; |
||||||
return LOG_DEBUG; |
default : priority = LOG_DEBUG; break; |
||||||
default: |
} |
||||||
// WTF? invalid log level?
|
return priority; |
||||||
return LOG_CRIT; |
} |
||||||
} |
|
||||||
} |
|
||||||
#endif |
#endif |
||||||
|
|
||||||
void LogMsg::Process() |
Log::Log(): |
||||||
{ |
m_Destination(eLogStdout), m_MinLevel(eLogInfo), |
||||||
#ifndef _WIN32 |
m_LogStream (nullptr), m_Logfile(""), m_IsReady(false) |
||||||
if (log && log->SyslogEnabled()) { |
{ |
||||||
// only log to syslog
|
} |
||||||
syslog(ToSyslogLevel(level), "%s", s.str().c_str()); |
|
||||||
return; |
|
||||||
} |
|
||||||
#endif |
|
||||||
auto stream = log ? log->GetLogStream () : nullptr; |
|
||||||
auto& output = stream ? *stream : std::cout; |
|
||||||
if (log) |
|
||||||
output << log->GetTimestamp (); |
|
||||||
else |
|
||||||
output << boost::posix_time::second_clock::local_time().time_of_day (); |
|
||||||
output << "/" << g_LogLevelStr[level] << " - "; |
|
||||||
output << s.str(); |
|
||||||
} |
|
||||||
|
|
||||||
const std::string& Log::GetTimestamp () |
Log::~Log () |
||||||
{ |
|
||||||
#if (__GNUC__ == 4) && (__GNUC_MINOR__ <= 6) && !defined(__clang__) |
|
||||||
auto ts = std::chrono::monotonic_clock::now (); |
|
||||||
#else |
|
||||||
auto ts = std::chrono::steady_clock::now (); |
|
||||||
#endif |
|
||||||
if (ts > m_LastTimestampUpdate + std::chrono::milliseconds (500)) // 0.5 second
|
|
||||||
{ |
{ |
||||||
m_LastTimestampUpdate = ts; |
switch (m_Destination) { |
||||||
m_Timestamp = boost::posix_time::to_simple_string (boost::posix_time::second_clock::local_time().time_of_day ()); |
#ifndef _WIN32 |
||||||
|
case eLogSyslog : |
||||||
|
closelog(); |
||||||
|
break; |
||||||
|
#endif |
||||||
|
case eLogFile: |
||||||
|
case eLogStream: |
||||||
|
m_LogStream->flush(); |
||||||
|
break; |
||||||
|
default: |
||||||
|
/* do nothing */ |
||||||
|
break; |
||||||
|
} |
||||||
|
Process(); |
||||||
} |
} |
||||||
return m_Timestamp; |
|
||||||
} |
|
||||||
|
|
||||||
void Log::Flush () |
void Log::SetLogLevel (const std::string& level) { |
||||||
{ |
if (level == "error") { m_MinLevel = eLogError; } |
||||||
if (m_LogStream) |
else if (level == "warn") { m_MinLevel = eLogWarning; } |
||||||
m_LogStream->flush(); |
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); |
||||||
|
} |
||||||
|
|
||||||
void Log::SetLogFile (const std::string& fullFilePath, bool truncate) |
const char * Log::TimeAsString(std::time_t t) { |
||||||
{ |
if (t != m_LastTimestamp) { |
||||||
m_FullFilePath = fullFilePath; |
strftime(m_LastDateTime, sizeof(m_LastDateTime), "%H:%M:%S", localtime(&t)); |
||||||
auto mode = std::ofstream::out | std::ofstream::binary; |
m_LastTimestamp = t; |
||||||
mode |= truncate ? std::ofstream::trunc : std::ofstream::app; |
} |
||||||
auto logFile = std::make_shared<std::ofstream> (fullFilePath, mode); |
return m_LastDateTime; |
||||||
if (logFile->is_open ()) |
|
||||||
{ |
|
||||||
SetLogStream (logFile); |
|
||||||
LogPrint(eLogInfo, "Log: will send messages to ", fullFilePath); |
|
||||||
} |
} |
||||||
} |
|
||||||
|
|
||||||
void Log::ReopenLogFile () |
/**
|
||||||
{ |
* @note This function better to be run in separate thread due to disk i/o. |
||||||
if (m_FullFilePath.length () > 0) |
* Unfortunately, with current startup process with late fork() this |
||||||
{ |
* will give us nothing but pain. Maybe later. See in NetDb as example. |
||||||
SetLogFile (m_FullFilePath, false); // don't truncate
|
*/ |
||||||
LogPrint(eLogInfo, "Log: file ", m_FullFilePath, " reopen"); |
void Log::Process() { |
||||||
|
std::unique_lock<std::mutex> l(m_OutputLock); |
||||||
|
while (1) { |
||||||
|
auto msg = m_Queue.GetNextWithTimeout (1); |
||||||
|
if (!msg) |
||||||
|
break; |
||||||
|
switch (m_Destination) { |
||||||
|
#ifndef _WIN32 |
||||||
|
case eLogSyslog: |
||||||
|
syslog(GetSyslogPrio(msg->level), "%s", msg->text.c_str()); |
||||||
|
break; |
||||||
|
#endif |
||||||
|
case eLogFile: |
||||||
|
case eLogStream: |
||||||
|
*m_LogStream << TimeAsString(msg->timestamp) << "/" << g_LogLevelStr[msg->level] << " - " << msg->text << std::endl; |
||||||
|
break; |
||||||
|
default: |
||||||
|
std::cout << TimeAsString(msg->timestamp) << "/" << g_LogLevelStr[msg->level] << " - " << msg->text << std::endl; |
||||||
|
break; |
||||||
|
} // switch
|
||||||
|
} // while
|
||||||
} |
} |
||||||
} |
|
||||||
|
|
||||||
|
void Log::Append(std::shared_ptr<i2p::log::LogMsg> & msg) { |
||||||
|
m_Queue.Put(msg); |
||||||
|
if (!m_IsReady) |
||||||
|
return; |
||||||
|
Process(); |
||||||
|
} |
||||||
|
|
||||||
void Log::SetLogLevel (const std::string& level) |
void Log::SendTo (const std::string& path) { |
||||||
{ |
auto flags = std::ofstream::out | std::ofstream::app; |
||||||
if (level == "error") { m_MinLevel = eLogError; } |
auto os = std::make_shared<std::ofstream> (path, flags); |
||||||
else if (level == "warn") { m_MinLevel = eLogWarning; } |
if (os->is_open ()) { |
||||||
else if (level == "info") { m_MinLevel = eLogInfo; } |
m_Logfile = path; |
||||||
else if (level == "debug") { m_MinLevel = eLogDebug; } |
m_Destination = eLogFile; |
||||||
else { |
m_LogStream = os; |
||||||
LogPrint(eLogError, "Log: Unknown loglevel: ", level); |
m_IsReady = true; |
||||||
return; |
return; |
||||||
} |
} |
||||||
LogPrint(eLogInfo, "Log: min msg level set to ", level); |
LogPrint(eLogError, "Log: can't open file ", path); |
||||||
} |
} |
||||||
|
|
||||||
void Log::SetLogStream (std::shared_ptr<std::ostream> logStream) |
void Log::SendTo (std::shared_ptr<std::ostream> os) { |
||||||
{ |
m_Destination = eLogStream; |
||||||
m_LogStream = logStream; |
m_IsReady = true; |
||||||
} |
m_LogStream = os; |
||||||
|
} |
||||||
|
|
||||||
void Log::StartSyslog(const std::string & ident, const int facility) |
|
||||||
{ |
|
||||||
#ifndef _WIN32 |
#ifndef _WIN32 |
||||||
m_Ident = ident; |
void Log::SendTo(const char *name, int facility) { |
||||||
openlog(m_Ident.c_str(), LOG_PID, facility); |
m_Destination = eLogSyslog; |
||||||
|
m_LogStream = nullptr; |
||||||
|
m_IsReady = true; |
||||||
|
openlog(name, LOG_CONS | LOG_PID, facility); |
||||||
|
} |
||||||
#endif |
#endif |
||||||
} |
|
||||||
|
|
||||||
void Log::StopSyslog() |
void Log::Reopen() { |
||||||
{ |
if (m_Destination == eLogFile) |
||||||
#ifndef _WIN32 |
SendTo(m_Logfile); |
||||||
closelog(); |
} |
||||||
m_Ident.clear(); |
|
||||||
#endif |
|
||||||
} |
|
||||||
|
|
||||||
bool Log::SyslogEnabled() |
Log & Logger() { |
||||||
{ |
return logger; |
||||||
return m_Ident.size() > 0; |
} |
||||||
} |
} // log
|
||||||
|
} // i2p
|
||||||
|
Loading…
Reference in new issue