Files
deskflow/src/lib/base/Log.cpp
2025-05-22 20:46:41 +01:00

293 lines
6.7 KiB
C++

/*
* Deskflow -- mouse and keyboard sharing utility
* SPDX-FileCopyrightText: (C) 2012 - 2016 Symless Ltd.
* SPDX-FileCopyrightText: (C) 2002 Chris Schoeneman
* SPDX-License-Identifier: GPL-2.0-only WITH LicenseRef-OpenSSL-Exception
*/
#include "base/Log.h"
#include "arch/Arch.h"
#include "base/LogOutputters.h"
#include "common/Constants.h"
#include <cstdarg>
#include <cstdint>
#include <cstdio>
#include <cstring>
#include <ctime>
#include <iostream>
const int kPriorityPrefixLength = 3;
// names of priorities
static const char *g_priority[] = {"FATAL", "ERROR", "WARNING", "NOTE", "INFO", "DEBUG",
"DEBUG1", "DEBUG2", "DEBUG3", "DEBUG4", "DEBUG5"};
// number of priorities
static const int g_numPriority = 11;
// if NDEBUG (not debug) is not specified, i.e. you're building in debug,
// then set default log level to DEBUG, otherwise the max level is INFO.
//
// GOTCHA: if `-DCMAKE_BUILD_TYPE=Debug` isn't set when configuring cmake
// for visual studio, then NDEBUG will be set (even if your VS solution
// config is Debug).
#ifndef NDEBUG
static const int g_defaultMaxPriority = kDEBUG;
#else
static const int g_defaultMaxPriority = kINFO;
#endif
namespace {
ELevel getPriority(const char *&fmt)
{
if (strnlen(fmt, SIZE_MAX) < kPriorityPrefixLength) {
throw std::invalid_argument("invalid format string, too short");
}
if (fmt[0] != '%' || fmt[1] != 'z') {
throw std::invalid_argument("invalid format string, missing priority");
}
return static_cast<ELevel>(fmt[2] - '0');
}
void makeTimeString(std::vector<char> &buffer)
{
const int yearOffset = 1900;
const int monthOffset = 1;
time_t t;
time(&t);
struct tm tm;
#if WINAPI_MSWINDOWS
localtime_s(&tm, &t);
#else
localtime_r(&t, &tm);
#endif
snprintf(
buffer.data(), buffer.size(), "%04i-%02i-%02iT%02i:%02i:%02i", tm.tm_year + yearOffset, tm.tm_mon + monthOffset,
tm.tm_mday, tm.tm_hour, tm.tm_min, tm.tm_sec
);
}
std::vector<char> makeMessage(const char *filename, int lineNumber, const char *message, ELevel priority)
{
// base size includes null terminator, colon, space, etc.
const int baseSize = 10;
const int timeBufferSize = 50;
const int priorityMaxSize = 10;
std::vector<char> timeBuffer(timeBufferSize);
makeTimeString(timeBuffer);
size_t timestampLength = strnlen(timeBuffer.data(), timeBufferSize);
size_t priorityLength = strnlen(g_priority[priority], priorityMaxSize);
size_t messageLength = strnlen(message, SIZE_MAX);
size_t bufferSize = baseSize + timestampLength + priorityLength + messageLength;
const auto filenameSet = filename != nullptr && filename[0] != '\0';
if (filenameSet) {
size_t filenameLength = strnlen(filename, SIZE_MAX);
size_t lineNumberLength = snprintf(nullptr, 0, "%d", lineNumber);
bufferSize += filenameLength + lineNumberLength;
std::vector<char> buffer(bufferSize);
snprintf(
buffer.data(), bufferSize, "[%s] %s: %s\n\t%s:%d", timeBuffer.data(), g_priority[priority], message, filename,
lineNumber
);
return buffer;
} else {
std::vector<char> buffer(bufferSize);
snprintf(buffer.data(), bufferSize, "[%s] %s: %s", timeBuffer.data(), g_priority[priority], message);
return buffer;
}
}
} // namespace
//
// Log
//
Log *Log::s_log = nullptr;
Log::Log(bool singleton)
{
if (singleton) {
assert(s_log == nullptr);
}
// create mutex for multithread safe operation
m_mutex = ARCH->newMutex();
// other initalization
m_maxPriority = g_defaultMaxPriority;
insert(new ConsoleLogOutputter); // NOSONAR - Adopted by `Log`
if (singleton) {
s_log = this;
}
}
Log::Log(Log *src)
{
s_log = src;
}
Log::~Log()
{
// clean up
for (auto index = m_outputters.begin(); index != m_outputters.end(); ++index) {
delete *index;
}
for (auto index = m_alwaysOutputters.begin(); index != m_alwaysOutputters.end(); ++index) {
delete *index;
}
ARCH->closeMutex(m_mutex);
}
Log *Log::getInstance()
{
assert(s_log != nullptr);
return s_log;
}
const char *Log::getFilterName() const
{
return getFilterName(getFilter());
}
const char *Log::getFilterName(int level) const
{
if (level < 0) {
return "Message";
}
return g_priority[level];
}
void Log::print(const char *file, int line, const char *fmt, ...)
{
const int initBufferSize = 1024;
const int bufferResizeScale = 2;
ELevel priority = getPriority(fmt);
fmt += kPriorityPrefixLength;
if (priority > getFilter()) {
return;
}
std::vector<char> buffer(initBufferSize);
auto length = static_cast<int>(buffer.size());
while (true) {
va_list args;
va_start(args, fmt);
int n = vsnprintf(buffer.data(), length, fmt, args);
va_end(args);
if (n < 0 || n > length) {
length *= bufferResizeScale;
buffer.resize(length);
} else {
break;
}
}
if (priority == kPRINT) {
output(priority, buffer.data());
} else {
auto message = makeMessage(file, line, buffer.data(), priority);
output(priority, message.data());
}
}
void Log::insert(ILogOutputter *adoptedOutputter, bool alwaysAtHead)
{
assert(adoptedOutputter != nullptr);
ArchMutexLock lock(m_mutex);
if (alwaysAtHead) {
m_alwaysOutputters.push_front(adoptedOutputter);
} else {
m_outputters.push_front(adoptedOutputter);
}
adoptedOutputter->open(kAppName);
}
void Log::remove(ILogOutputter *outputter)
{
ArchMutexLock lock(m_mutex);
m_outputters.remove(outputter);
m_alwaysOutputters.remove(outputter);
}
void Log::pop_front(bool alwaysAtHead)
{
ArchMutexLock lock(m_mutex);
OutputterList *list = alwaysAtHead ? &m_alwaysOutputters : &m_outputters;
if (!list->empty()) {
delete list->front();
list->pop_front();
}
}
bool Log::setFilter(const char *maxPriority)
{
if (maxPriority != nullptr) {
for (int i = 0; i < g_numPriority; ++i) {
if (strcmp(maxPriority, g_priority[i]) == 0) {
setFilter(i);
return true;
}
}
return false;
}
return true;
}
void Log::setFilter(int maxPriority)
{
ArchMutexLock lock(m_mutex);
m_maxPriority = maxPriority;
}
int Log::getFilter() const
{
ArchMutexLock lock(m_mutex);
return m_maxPriority;
}
void Log::output(ELevel priority, const char *msg)
{
assert(priority >= -1 && priority < g_numPriority);
assert(msg != nullptr);
if (!msg)
return;
ArchMutexLock lock(m_mutex);
OutputterList::const_iterator i;
for (i = m_alwaysOutputters.begin(); i != m_alwaysOutputters.end(); ++i) {
// write to outputter
(*i)->write(priority, msg);
}
for (i = m_outputters.begin(); i != m_outputters.end(); ++i) {
// write to outputter and break out of loop if it returns false
if (!(*i)->write(priority, msg)) {
break;
}
}
}