// Ryzom - MMORPG Framework // Copyright (C) 2010 Winch Gate Property Limited // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU Affero General Public License as // published by the Free Software Foundation, either version 3 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 Affero General Public License for more details. // // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . #ifndef LOG_STORAGE_H #define LOG_STORAGE_H #include "logger_service.h" /** Class to store logs in a 'compact' form on disk * The log stoage consiste of a series of 3 main chunks : * - the log description chunk that contains the description of each log type * - the log entry table, that contains an entry for each log with list of * index in each parameter table (see next chunk) * - the parameters tables. One table for each log name/type. */ class CLogStorage { public: /// Identity of a log parameter table struct TLogParamId { /// Name of the parameter std::string ParamName; /// Type of the parameter LGS::TSupportedParamType ParamType; /// Strict ordering to be used as map key bool operator < (const TLogParamId &other) const { if (ParamName < other.ParamName) return true; else if (ParamName == other.ParamName) return ParamType < other.ParamType; return false; } void serial(NLMISC::IStream &s) { s.serial(ParamName); s.serial(ParamType); } }; typedef std::vector TParamIndex; /// The log entries with parameter stored as table index struct TDiskLogEntry { /// Index in the log definition vector uint32 LogType; /// The date of the log uint32 LogDate; /// The size of the context stack uint32 ContextStack; /// The shard id that sent this log uint32 ShardId; /// Vector of index in each parameter table TParamIndex ParamIndex; /// Vector of vector of index in each parameter table for list param std::vector ListParamIndex; void serial(NLMISC::IStream &s) { s.serial(LogType); s.serial(LogDate); s.serial(ContextStack); s.serial(ShardId); s.serialCont(ParamIndex); uint32 nbList = (uint32)ListParamIndex.size(); s.serial(nbList); ListParamIndex.resize(nbList); for (uint i=0; i _DiskLogEntries; typedef std::vector TParamsTable; typedef std::map TParamsTables; /// The tables of all log params TParamsTables _ParamTables; /// The log definition TLogDefinitions _LogDefs; typedef std::map TLogDefLindex; // the log def index TLogDefLindex _LogDefIndex; /// The current size of the context stack uint32 _ContextStack; struct EInvalidLogName {}; /// Constructor used to store log file CLogStorage(const TLogDefinitions &logDefs) : _LogDefs(logDefs), _ContextStack(0) { // fill the log definition for (uint i=0; iSaveFilesDirectory.toString()+"/logs"; } const LGS::TLogDefinition &getLogDef(const std::string &logName) { TLogDefLindex::iterator logDefIt = _LogDefIndex.find(logName); if (logDefIt == _LogDefIndex.end()) throw EInvalidLogName(); nlassert(logDefIt->second < _LogDefs.size()); return _LogDefs[logDefIt->second]; } void loadLogs(const std::string &fileName) { // open the input file NLMISC::CIFile ifile(fileName); // and read the logs ifile.serial(*this); nldebug("Loaded %u logs from file %s", _DiskLogEntries.size(), fileName.c_str()); } /// Save the last minute of logs on disk void saveMinutely(const TLogInfos &logInfos) { uint32 now = NLMISC::CTime::getSecondsSince1970(); // A flag to skip saving if only context open/close are selected // (because they have date 0 or ~0 that do not match correctlry the // time test). bool atLeastOneSelected = false;; // go back to up to 1 minute ago TLogInfos::const_reverse_iterator it = logInfos.rbegin(); while (it != logInfos.rend() && (it->LogInfo.getTimeStamp() == 0 || it->LogInfo.getTimeStamp() == ~0 || it->LogInfo.getTimeStamp() > now-60)) { if (it->LogInfo.getTimeStamp() != 0 && it->LogInfo.getTimeStamp() != ~0) atLeastOneSelected = true; ++it; } if (atLeastOneSelected) { TLogInfos::const_iterator first(it.base()), last(logInfos.end()); // prepare the logs to save for (; first != last; ++first) { storeLog(*first); } } // write the container on disk saveLogfile("minutely_"); } // Save all logs in memory on disk void saveHourly(const TLogInfos &logInfos) { uint32 now = NLMISC::CTime::getSecondsSince1970(); // // go back to up to 1 minute ago // TLogInfos::const_reverse_iterator it = logInfos.rbegin(); // while (it != logInfos.rend() && // (it->LogInfo.getTimeStamp() == 0 // || it->LogInfo.getTimeStamp() == ~0 // || it->LogInfo.getTimeStamp() > now-60*60)) // ++it; TLogInfos::const_iterator first(logInfos.begin()), last(logInfos.end()); // prepare the logs to save for (; first != last; ++first) { storeLog(*first); } // write the container on disk saveLogfile("hourly_"); } void saveLogfile(const std::string &prefix) { if (_DiskLogEntries.empty()) // no log, do not store anything return; NLMISC::CSString fileName; // set the save directory and create it if needed fileName << getLogRoot() <<"/"; NLMISC::CFile::createDirectoryTree(fileName); // build the file name from the current date char dateStr[1024]; time_t now = time(NULL); struct tm *_tm = localtime(&now); strftime(dateStr, 1024, "%Y-%m-%d_%H-%M-%S", _tm); fileName << prefix << dateStr <<".binlog"; nldebug("Storing %u logs in file %s", _DiskLogEntries.size(), fileName.c_str()); // open the output file { NLMISC::COFile of(fileName+".tmp"); // and serial the logs of.serial(*this); } // rename the 'tmp" into finale output file NLMISC::CFile::moveFile(fileName.c_str(), (fileName+".tmp").c_str()); } void storeLog(const TLogEntry &logEntry) { // NB : not optimized at all, just a place holder before replacement with // PDR 2 storage system (witch allow storage of table) const LGS::TLogDefinition &ld = getLogDef(logEntry.LogInfo.getLogName()); // create a log entry TDiskLogEntry dle; // set the shard id dle.ShardId = logEntry.ShardId; // set the log date dle.LogDate = logEntry.LogInfo.getTimeStamp(); // pre decrement if close context if (dle.LogDate == ~0 && _ContextStack > 0) --_ContextStack; dle.ContextStack = _ContextStack; // set the type of the log dle.LogType = (uint32)_LogDefIndex[logEntry.LogInfo.getLogName()]; // post increment if open context if (dle.LogDate == 0) ++_ContextStack; // store each parameter for (uint i=0; i::const_iterator first(lpv.getParams().begin()), last(lpv.getParams().end()); for (; first != last; ++first) { uint32 index = (uint32)pt.size(); pt.push_back(*first); // store the index in the persistent log entry dle.ListParamIndex[i].push_back(index); } } // store the entry in the container _DiskLogEntries.push_back(dle); } void serial(NLMISC::IStream &s) { // serial the log definition s.serialCont(_LogDefs); // serial the log entryes s.serialCont(_DiskLogEntries); // serial the param tables uint32 nbTable = (uint32)_ParamTables.size(); if (s.isReading()) { s.serial(nbTable); for (uint i=0; ifirst); s.serialCont(first->second); } } } } void dumpLogs(NLMISC::CLog &log) { for (uint i=0; i<_DiskLogEntries.size(); ++i) { const TDiskLogEntry &dle = _DiskLogEntries[i]; // get the descriptor nlassert(dle.LogType < _LogDefs.size()); const LGS::TLogDefinition &ld = _LogDefs[dle.LogType]; nlassert(dle.ParamIndex.size() == ld.getParams().size()); // dump the log if (dle.LogDate == 0) log.display("Open Context : %s", ld.getLogName().c_str()); else if (dle.LogDate == ~0) log.display("Close Context : %s", ld.getLogName().c_str()); else log.display("%s : %s : %s ", formatDate(dle.LogDate).c_str(), ld.getLogName().c_str(), ld.getLogText().c_str()); for (uint j=0; j