// ***************************************************************************** // * This file is part of the FreeFileSync project. It is distributed under * // * GNU General Public License: https://www.gnu.org/licenses/gpl-3.0 * // * Copyright (C) Zenju (zenju AT freefilesync DOT org) - All Rights Reserved * // ***************************************************************************** #ifndef ERROR_LOG_H_8917590832147915 #define ERROR_LOG_H_8917590832147915 #include #include #include #include #include "time.h" #include "i18n.h" #include "zstring.h" namespace zen { enum MessageType { MSG_TYPE_INFO = 0x1, MSG_TYPE_WARNING = 0x2, MSG_TYPE_ERROR = 0x4, MSG_TYPE_FATAL_ERROR = 0x8, }; struct LogEntry { time_t time = 0; MessageType type = MSG_TYPE_FATAL_ERROR; Zstringw message; //std::wstring may employ small string optimization: we cannot accept bloating the "ErrorLog::entries" memory block below (think 1 million items) }; std::wstring formatMessage(const LogEntry& entry); class ErrorLog { public: void logMsg(const std::wstring& msg, MessageType type); int getItemCount(int typeFilter = MSG_TYPE_INFO | MSG_TYPE_WARNING | MSG_TYPE_ERROR | MSG_TYPE_FATAL_ERROR) const; //subset of std::vector<> interface: using const_iterator = std::vector::const_iterator; const_iterator begin() const { return entries_.begin(); } const_iterator end () const { return entries_.end (); } bool empty() const { return entries_.empty(); } private: std::vector entries_; }; //######################## implementation ########################## inline void ErrorLog::logMsg(const std::wstring& msg, MessageType type) { entries_.push_back({ std::time(nullptr), type, copyStringTo(msg) }); } inline int ErrorLog::getItemCount(int typeFilter) const { return static_cast(std::count_if(entries_.begin(), entries_.end(), [&](const LogEntry& e) { return e.type & typeFilter; })); } namespace { std::wstring formatMessageImpl(const LogEntry& entry) { auto getTypeName = [&] { switch (entry.type) { case MSG_TYPE_INFO: return _("Info"); case MSG_TYPE_WARNING: return _("Warning"); case MSG_TYPE_ERROR: return _("Error"); case MSG_TYPE_FATAL_ERROR: return _("Serious Error"); } assert(false); return std::wstring(); }; std::wstring msgFmt = L"[" + formatTime(FORMAT_TIME, getLocalTime(entry.time)) + L"] " + getTypeName() + L": "; const size_t prefixLen = msgFmt.size(); //considers UTF-16 only! for (auto it = entry.message.begin(); it != entry.message.end(); ) if (*it == L'\n') { msgFmt += L'\n'; msgFmt.append(prefixLen, L' '); do //skip duplicate newlines { ++it; } while (it != entry.message.end() && *it == L'\n'); } else msgFmt += *it++; return msgFmt; } } inline std::wstring formatMessage(const LogEntry& entry) { return formatMessageImpl(entry); } } #endif //ERROR_LOG_H_8917590832147915