2016-03-27 01:00:00 +01: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 14:00:13 +01:00
|
|
|
|
2016-03-27 01:00:00 +01:00
|
|
|
#include "Log.h"
|
2014-04-23 00:10:21 +02:00
|
|
|
|
2016-03-27 01:00:00 +01:00
|
|
|
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
|
|
|
|
};
|
2016-03-26 14:40:19 +01:00
|
|
|
|
2016-03-26 14:49:45 +01:00
|
|
|
#ifndef _WIN32
|
2016-03-27 01:00:00 +01:00
|
|
|
/**
|
|
|
|
* @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 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;
|
|
|
|
}
|
2016-03-26 14:49:45 +01:00
|
|
|
#endif
|
2014-04-23 18:49:02 +02:00
|
|
|
|
2016-03-27 01:00:00 +01:00
|
|
|
Log::Log():
|
|
|
|
m_Destination(eLogStdout), m_MinLevel(eLogInfo),
|
|
|
|
m_LogStream (nullptr), m_Logfile(""), m_IsReady(false)
|
2015-05-06 17:24:35 +02:00
|
|
|
{
|
2016-03-27 01:00:00 +01:00
|
|
|
}
|
2014-04-23 18:49:02 +02:00
|
|
|
|
2016-03-27 01:00:00 +01:00
|
|
|
Log::~Log ()
|
2014-12-08 22:27:10 +01:00
|
|
|
{
|
2016-03-27 01:00:00 +01:00
|
|
|
switch (m_Destination) {
|
|
|
|
#ifndef _WIN32
|
|
|
|
case eLogSyslog :
|
|
|
|
closelog();
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
case eLogFile:
|
|
|
|
case eLogStream:
|
|
|
|
m_LogStream->flush();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
/* do nothing */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
Process();
|
|
|
|
}
|
2014-12-08 22:27:10 +01:00
|
|
|
|
2016-03-27 01:00:00 +01:00
|
|
|
void Log::SetLogLevel (const std::string& level) {
|
|
|
|
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), "%H:%M:%S", localtime(&t));
|
|
|
|
m_LastTimestamp = t;
|
|
|
|
}
|
|
|
|
return m_LastDateTime;
|
2016-02-04 19:53:38 +01:00
|
|
|
}
|
|
|
|
|
2016-03-27 01:00:00 +01:00
|
|
|
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
|
|
|
|
}
|
2016-02-04 19:53:38 +01:00
|
|
|
|
2016-03-27 01:00:00 +01:00
|
|
|
void Log::Append(std::shared_ptr<i2p::log::LogMsg> & msg) {
|
|
|
|
m_Queue.Put(msg);
|
|
|
|
if (!m_IsReady)
|
|
|
|
return;
|
|
|
|
Process();
|
|
|
|
}
|
2015-12-28 01:00:00 +01:00
|
|
|
|
2016-03-27 01:00:00 +01:00
|
|
|
void Log::SendTo (const std::string& path) {
|
|
|
|
auto flags = std::ofstream::out | std::ofstream::app;
|
|
|
|
auto os = std::make_shared<std::ofstream> (path, flags);
|
|
|
|
if (os->is_open ()) {
|
|
|
|
m_Logfile = path;
|
|
|
|
m_Destination = eLogFile;
|
|
|
|
m_LogStream = os;
|
|
|
|
m_IsReady = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
LogPrint(eLogError, "Log: can't open file ", path);
|
|
|
|
}
|
2016-03-26 14:40:19 +01:00
|
|
|
|
2016-03-27 01:00:00 +01:00
|
|
|
void Log::SendTo (std::shared_ptr<std::ostream> os) {
|
|
|
|
m_Destination = eLogStream;
|
|
|
|
m_IsReady = true;
|
|
|
|
m_LogStream = os;
|
|
|
|
}
|
2016-03-26 14:40:19 +01:00
|
|
|
|
2016-03-26 14:49:45 +01:00
|
|
|
#ifndef _WIN32
|
2016-03-27 01:00:00 +01:00
|
|
|
void Log::SendTo(const char *name, int facility) {
|
|
|
|
m_Destination = eLogSyslog;
|
|
|
|
m_LogStream = nullptr;
|
|
|
|
m_IsReady = true;
|
|
|
|
openlog(name, LOG_CONS | LOG_PID, facility);
|
|
|
|
}
|
2016-03-26 14:49:45 +01:00
|
|
|
#endif
|
2016-03-26 14:40:19 +01:00
|
|
|
|
2016-03-27 01:00:00 +01:00
|
|
|
void Log::Reopen() {
|
|
|
|
if (m_Destination == eLogFile)
|
|
|
|
SendTo(m_Logfile);
|
|
|
|
}
|
|
|
|
|
|
|
|
Log & Logger() {
|
|
|
|
return logger;
|
|
|
|
}
|
|
|
|
} // log
|
|
|
|
} // i2p
|