mirror of
https://github.com/TrinityCore/TrinityCore.git
synced 2026-01-16 15:40:45 +01:00
501 lines
13 KiB
C++
501 lines
13 KiB
C++
/*
|
|
* Copyright (C) 2008-2013 TrinityCore <http://www.trinitycore.org/>
|
|
* Copyright (C) 2005-2008 MaNGOS <http://getmangos.com/>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License as published by the
|
|
* Free Software Foundation; either version 2 of the License, or (at your
|
|
* option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
* more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "Log.h"
|
|
#include "Common.h"
|
|
#include "Config.h"
|
|
#include "Util.h"
|
|
#include "AppenderConsole.h"
|
|
#include "AppenderFile.h"
|
|
#include "AppenderDB.h"
|
|
#include "LogOperation.h"
|
|
|
|
#include <cstdarg>
|
|
#include <cstdio>
|
|
#include <sstream>
|
|
|
|
Log::Log() : worker(NULL)
|
|
{
|
|
m_logsTimestamp = "_" + GetTimestampStr();
|
|
LoadFromConfig();
|
|
}
|
|
|
|
Log::~Log()
|
|
{
|
|
Close();
|
|
}
|
|
|
|
uint8 Log::NextAppenderId()
|
|
{
|
|
return AppenderId++;
|
|
}
|
|
|
|
int32 GetConfigIntDefault(std::string base, const char* name, int32 value)
|
|
{
|
|
base.append(name);
|
|
return ConfigMgr::GetIntDefault(base.c_str(), value);
|
|
}
|
|
|
|
std::string GetConfigStringDefault(std::string base, const char* name, const char* value)
|
|
{
|
|
base.append(name);
|
|
return ConfigMgr::GetStringDefault(base.c_str(), value);
|
|
}
|
|
|
|
// Returns default logger if the requested logger is not found
|
|
Logger* Log::GetLoggerByType(LogFilterType filter)
|
|
{
|
|
LoggerMap::iterator it = loggers.begin();
|
|
while (it != loggers.end() && it->second.getType() != filter)
|
|
++it;
|
|
|
|
return it == loggers.end() ? &(loggers[0]) : &(it->second);
|
|
}
|
|
|
|
Appender* Log::GetAppenderByName(std::string const& name)
|
|
{
|
|
AppenderMap::iterator it = appenders.begin();
|
|
while (it != appenders.end() && it->second && it->second->getName() != name)
|
|
++it;
|
|
|
|
return it == appenders.end() ? NULL : it->second;
|
|
}
|
|
|
|
void Log::CreateAppenderFromConfig(const char* name)
|
|
{
|
|
if (!name || *name == '\0')
|
|
return;
|
|
|
|
// Format=type, level, flags, optional1, optional2
|
|
// if type = File. optional1 = file and option2 = mode
|
|
// if type = Console. optional1 = Color
|
|
std::string options = "Appender.";
|
|
options.append(name);
|
|
options = ConfigMgr::GetStringDefault(options.c_str(), "");
|
|
Tokenizer tokens(options, ',');
|
|
Tokenizer::const_iterator iter = tokens.begin();
|
|
uint8 size = tokens.size();
|
|
|
|
if (size < 2)
|
|
{
|
|
fprintf(stderr, "Log::CreateAppenderFromConfig: Wrong configuration for appender %s. Config line: %s\n", name, options.c_str());
|
|
return;
|
|
}
|
|
|
|
AppenderFlags flags = APPENDER_FLAGS_NONE;
|
|
AppenderType type = AppenderType(atoi(*iter));
|
|
LogLevel level = LogLevel(atoi(*(++iter)));
|
|
if (level > LOG_LEVEL_FATAL)
|
|
{
|
|
fprintf(stderr, "Log::CreateAppenderFromConfig: Wrong Log Level %d for appender %s\n", level, name);
|
|
return;
|
|
}
|
|
|
|
if (size > 2)
|
|
flags = AppenderFlags(atoi(*(++iter)));
|
|
|
|
switch (type)
|
|
{
|
|
case APPENDER_CONSOLE:
|
|
{
|
|
AppenderConsole* appender = new AppenderConsole(NextAppenderId(), name, level, flags);
|
|
appenders[appender->getId()] = appender;
|
|
if (size > 3)
|
|
appender->InitColors(*(++iter));
|
|
//fprintf(stdout, "Log::CreateAppenderFromConfig: Created Appender %s (%u), Type CONSOLE, Mask %u\n", appender->getName().c_str(), appender->getId(), appender->getLogLevel()); // DEBUG - RemoveMe
|
|
break;
|
|
}
|
|
case APPENDER_FILE:
|
|
{
|
|
std::string filename;
|
|
std::string mode = "a";
|
|
|
|
if (size < 4)
|
|
{
|
|
fprintf(stderr, "Log::CreateAppenderFromConfig: Missing file name for appender %s\n", name);
|
|
return;
|
|
}
|
|
|
|
filename = *(++iter);
|
|
|
|
if (size > 4)
|
|
mode = *(++iter);
|
|
|
|
if (flags & APPENDER_FLAGS_USE_TIMESTAMP)
|
|
{
|
|
size_t dot_pos = filename.find_last_of(".");
|
|
if (dot_pos != filename.npos)
|
|
filename.insert(dot_pos, m_logsTimestamp);
|
|
else
|
|
filename += m_logsTimestamp;
|
|
}
|
|
|
|
uint64 maxFileSize = 0;
|
|
if (size > 5)
|
|
maxFileSize = atoi(*(++iter));
|
|
|
|
uint8 id = NextAppenderId();
|
|
appenders[id] = new AppenderFile(id, name, level, filename.c_str(), m_logsDir.c_str(), mode.c_str(), flags, maxFileSize);
|
|
//fprintf(stdout, "Log::CreateAppenderFromConfig: Created Appender %s (%u), Type FILE, Mask %u, File %s, Mode %s\n", name, id, level, filename.c_str(), mode.c_str()); // DEBUG - RemoveMe
|
|
break;
|
|
}
|
|
case APPENDER_DB:
|
|
{
|
|
uint8 id = NextAppenderId();
|
|
appenders[id] = new AppenderDB(id, name, level);
|
|
break;
|
|
}
|
|
default:
|
|
fprintf(stderr, "Log::CreateAppenderFromConfig: Unknown type %d for appender %s\n", type, name);
|
|
break;
|
|
}
|
|
}
|
|
|
|
void Log::CreateLoggerFromConfig(const char* name)
|
|
{
|
|
if (!name || *name == '\0')
|
|
return;
|
|
|
|
LogLevel level = LOG_LEVEL_DISABLED;
|
|
uint8 type = uint8(-1);
|
|
|
|
std::string options = "Logger.";
|
|
options.append(name);
|
|
options = ConfigMgr::GetStringDefault(options.c_str(), "");
|
|
|
|
if (options.empty())
|
|
{
|
|
fprintf(stderr, "Log::CreateLoggerFromConfig: Missing config option Logger.%s\n", name);
|
|
return;
|
|
}
|
|
|
|
Tokenizer tokens(options, ',');
|
|
Tokenizer::const_iterator iter = tokens.begin();
|
|
|
|
if (tokens.size() != 3)
|
|
{
|
|
fprintf(stderr, "Log::CreateLoggerFromConfig: Wrong config option Logger.%s=%s\n", name, options.c_str());
|
|
return;
|
|
}
|
|
|
|
type = uint8(atoi(*iter));
|
|
if (type > MaxLogFilter)
|
|
{
|
|
fprintf(stderr, "Log::CreateLoggerFromConfig: Wrong type %u for logger %s\n", type, name);
|
|
return;
|
|
}
|
|
|
|
Logger& logger = loggers[type];
|
|
if (!logger.getName().empty())
|
|
{
|
|
fprintf(stderr, "Error while configuring Logger %s. Already defined\n", name);
|
|
return;
|
|
}
|
|
|
|
++iter;
|
|
level = LogLevel(atoi(*iter));
|
|
if (level > LOG_LEVEL_FATAL)
|
|
{
|
|
fprintf(stderr, "Log::CreateLoggerFromConfig: Wrong Log Level %u for logger %s\n", type, name);
|
|
return;
|
|
}
|
|
|
|
logger.Create(name, LogFilterType(type), level);
|
|
//fprintf(stdout, "Log::CreateLoggerFromConfig: Created Logger %s, Type %u, mask %u\n", name, LogFilterType(type), level); // DEBUG - RemoveMe
|
|
|
|
++iter;
|
|
std::istringstream ss(*iter);
|
|
std::string str;
|
|
|
|
ss >> str;
|
|
while (ss)
|
|
{
|
|
if (Appender* appender = GetAppenderByName(str))
|
|
{
|
|
logger.addAppender(appender->getId(), appender);
|
|
//fprintf(stdout, "Log::CreateLoggerFromConfig: Added Appender %s to Logger %s\n", appender->getName().c_str(), name); // DEBUG - RemoveMe
|
|
}
|
|
else
|
|
fprintf(stderr, "Error while configuring Appender %s in Logger %s. Appender does not exist", str.c_str(), name);
|
|
ss >> str;
|
|
}
|
|
}
|
|
|
|
void Log::ReadAppendersFromConfig()
|
|
{
|
|
std::istringstream ss(ConfigMgr::GetStringDefault("Appenders", ""));
|
|
std::string name;
|
|
|
|
do
|
|
{
|
|
ss >> name;
|
|
CreateAppenderFromConfig(name.c_str());
|
|
name = "";
|
|
}
|
|
while (ss);
|
|
}
|
|
|
|
void Log::ReadLoggersFromConfig()
|
|
{
|
|
std::istringstream ss(ConfigMgr::GetStringDefault("Loggers", ""));
|
|
std::string name;
|
|
|
|
do
|
|
{
|
|
ss >> name;
|
|
CreateLoggerFromConfig(name.c_str());
|
|
name = "";
|
|
}
|
|
while (ss);
|
|
|
|
// root logger must exist. Marking as disabled as its not configured
|
|
if (loggers.find(LOG_FILTER_GENERAL) == loggers.end())
|
|
loggers[LOG_FILTER_GENERAL].Create("root", LOG_FILTER_GENERAL, LOG_LEVEL_DISABLED);
|
|
}
|
|
|
|
void Log::vlog(LogFilterType filter, LogLevel level, char const* str, va_list argptr)
|
|
{
|
|
char text[MAX_QUERY_LEN];
|
|
vsnprintf(text, MAX_QUERY_LEN, str, argptr);
|
|
write(new LogMessage(level, filter, text));
|
|
}
|
|
|
|
void Log::write(LogMessage* msg)
|
|
{
|
|
if (loggers.empty())
|
|
return;
|
|
|
|
msg->text.append("\n");
|
|
Logger* logger = GetLoggerByType(msg->type);
|
|
|
|
if (worker)
|
|
worker->enqueue(new LogOperation(logger, msg));
|
|
else
|
|
logger->write(*msg);
|
|
}
|
|
|
|
std::string Log::GetTimestampStr()
|
|
{
|
|
time_t t = time(NULL);
|
|
tm* aTm = localtime(&t);
|
|
// YYYY year
|
|
// MM month (2 digits 01-12)
|
|
// DD day (2 digits 01-31)
|
|
// HH hour (2 digits 00-23)
|
|
// MM minutes (2 digits 00-59)
|
|
// SS seconds (2 digits 00-59)
|
|
char buf[20];
|
|
snprintf(buf, 20, "%04d-%02d-%02d_%02d-%02d-%02d", aTm->tm_year+1900, aTm->tm_mon+1, aTm->tm_mday, aTm->tm_hour, aTm->tm_min, aTm->tm_sec);
|
|
return std::string(buf);
|
|
}
|
|
|
|
bool Log::SetLogLevel(std::string const& name, const char* newLevelc, bool isLogger /* = true */)
|
|
{
|
|
LogLevel newLevel = LogLevel(atoi(newLevelc));
|
|
if (newLevel < 0)
|
|
return false;
|
|
|
|
if (isLogger)
|
|
{
|
|
LoggerMap::iterator it = loggers.begin();
|
|
while (it != loggers.end() && it->second.getName() != name)
|
|
++it;
|
|
|
|
if (it == loggers.end())
|
|
return false;
|
|
|
|
it->second.setLogLevel(newLevel);
|
|
}
|
|
else
|
|
{
|
|
Appender* appender = GetAppenderByName(name);
|
|
if (!appender)
|
|
return false;
|
|
|
|
appender->setLogLevel(newLevel);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool Log::ShouldLog(LogFilterType type, LogLevel level) const
|
|
{
|
|
LoggerMap::const_iterator it = loggers.find(uint8(type));
|
|
if (it != loggers.end())
|
|
{
|
|
LogLevel loggerLevel = it->second.getLogLevel();
|
|
return loggerLevel && loggerLevel <= level;
|
|
}
|
|
|
|
if (type != LOG_FILTER_GENERAL)
|
|
return ShouldLog(LOG_FILTER_GENERAL, level);
|
|
|
|
return false;
|
|
}
|
|
|
|
void Log::outTrace(LogFilterType filter, const char * str, ...)
|
|
{
|
|
if (!str || !ShouldLog(filter, LOG_LEVEL_TRACE))
|
|
return;
|
|
|
|
va_list ap;
|
|
va_start(ap, str);
|
|
|
|
vlog(filter, LOG_LEVEL_TRACE, str, ap);
|
|
|
|
va_end(ap);
|
|
}
|
|
|
|
void Log::outDebug(LogFilterType filter, const char * str, ...)
|
|
{
|
|
if (!str || !ShouldLog(filter, LOG_LEVEL_DEBUG))
|
|
return;
|
|
|
|
va_list ap;
|
|
va_start(ap, str);
|
|
|
|
vlog(filter, LOG_LEVEL_DEBUG, str, ap);
|
|
|
|
va_end(ap);
|
|
}
|
|
|
|
void Log::outInfo(LogFilterType filter, const char * str, ...)
|
|
{
|
|
if (!str || !ShouldLog(filter, LOG_LEVEL_INFO))
|
|
return;
|
|
|
|
va_list ap;
|
|
va_start(ap, str);
|
|
|
|
vlog(filter, LOG_LEVEL_INFO, str, ap);
|
|
|
|
va_end(ap);
|
|
}
|
|
|
|
void Log::outWarn(LogFilterType filter, const char * str, ...)
|
|
{
|
|
if (!str || !ShouldLog(filter, LOG_LEVEL_WARN))
|
|
return;
|
|
|
|
va_list ap;
|
|
va_start(ap, str);
|
|
|
|
vlog(filter, LOG_LEVEL_WARN, str, ap);
|
|
|
|
va_end(ap);
|
|
}
|
|
|
|
void Log::outError(LogFilterType filter, const char * str, ...)
|
|
{
|
|
if (!str || !ShouldLog(filter, LOG_LEVEL_ERROR))
|
|
return;
|
|
|
|
va_list ap;
|
|
va_start(ap, str);
|
|
|
|
vlog(filter, LOG_LEVEL_ERROR, str, ap);
|
|
|
|
va_end(ap);
|
|
}
|
|
|
|
void Log::outFatal(LogFilterType filter, const char * str, ...)
|
|
{
|
|
if (!str || !ShouldLog(filter, LOG_LEVEL_FATAL))
|
|
return;
|
|
|
|
va_list ap;
|
|
va_start(ap, str);
|
|
|
|
vlog(filter, LOG_LEVEL_FATAL, str, ap);
|
|
|
|
va_end(ap);
|
|
}
|
|
|
|
void Log::outCharDump(char const* str, uint32 accountId, uint32 guid, char const* name)
|
|
{
|
|
if (!str || !ShouldLog(LOG_FILTER_PLAYER_DUMP, LOG_LEVEL_INFO))
|
|
return;
|
|
|
|
std::ostringstream ss;
|
|
ss << "== START DUMP == (account: " << accountId << " guid: " << guid << " name: " << name
|
|
<< ")\n" << str << "\n== END DUMP ==\n";
|
|
|
|
LogMessage* msg = new LogMessage(LOG_LEVEL_INFO, LOG_FILTER_PLAYER_DUMP, ss.str());
|
|
std::ostringstream param;
|
|
param << guid << '_' << name;
|
|
|
|
msg->param1 = param.str();
|
|
|
|
write(msg);
|
|
}
|
|
|
|
void Log::outCommand(uint32 account, const char * str, ...)
|
|
{
|
|
if (!str || !ShouldLog(LOG_FILTER_GMCOMMAND, LOG_LEVEL_INFO))
|
|
return;
|
|
|
|
va_list ap;
|
|
va_start(ap, str);
|
|
char text[MAX_QUERY_LEN];
|
|
vsnprintf(text, MAX_QUERY_LEN, str, ap);
|
|
va_end(ap);
|
|
|
|
LogMessage* msg = new LogMessage(LOG_LEVEL_INFO, LOG_FILTER_GMCOMMAND, text);
|
|
|
|
std::ostringstream ss;
|
|
ss << account;
|
|
msg->param1 = ss.str();
|
|
|
|
write(msg);
|
|
}
|
|
|
|
void Log::SetRealmId(uint32 id)
|
|
{
|
|
for (AppenderMap::iterator it = appenders.begin(); it != appenders.end(); ++it)
|
|
if (it->second && it->second->getType() == APPENDER_DB)
|
|
((AppenderDB *)it->second)->setRealmId(id);
|
|
}
|
|
|
|
void Log::Close()
|
|
{
|
|
delete worker;
|
|
worker = NULL;
|
|
loggers.clear();
|
|
for (AppenderMap::iterator it = appenders.begin(); it != appenders.end(); ++it)
|
|
{
|
|
delete it->second;
|
|
it->second = NULL;
|
|
}
|
|
appenders.clear();
|
|
}
|
|
|
|
void Log::LoadFromConfig()
|
|
{
|
|
Close();
|
|
|
|
if (ConfigMgr::GetBoolDefault("Log.Async.Enable", false))
|
|
worker = new LogWorker();
|
|
|
|
AppenderId = 0;
|
|
m_logsDir = ConfigMgr::GetStringDefault("LogsDir", "");
|
|
if (!m_logsDir.empty())
|
|
if ((m_logsDir.at(m_logsDir.length() - 1) != '/') && (m_logsDir.at(m_logsDir.length() - 1) != '\\'))
|
|
m_logsDir.push_back('/');
|
|
ReadAppendersFromConfig();
|
|
ReadLoggersFromConfig();
|
|
}
|