/*
* synergy -- mouse and keyboard sharing utility
* Copyright (C) 2024 Symless Ltd.
*
* This package is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* found in the file LICENSE that should have accompanied this file.
*
* This package 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 .
*/
#include "Logger.h"
#include "string_utils.h"
#include
#include
#include
#include
#if defined(Q_OS_WIN)
#include
#endif
namespace synergy::gui {
const auto kForceDebugMessages = QStringList{
"Synergy", // TEST
"No functional TLS backend was found",
"No TLS backend is available",
"QSslSocket::connectToHostEncrypted: TLS initialization failed",
"Retrying to obtain clipboard.",
"Unable to obtain clipboard."};
Logger Logger::s_instance;
QString fileLine(const QMessageLogContext &context) {
if (!context.file) {
return "";
}
return QString("%1:%2").arg(context.file).arg(context.line);
}
QString printLine(
FILE *out, const QString &type, const QString &message,
const QString &fileLine = "") {
auto datetime = QDateTime::currentDateTime().toString("yyyy-MM-ddTHH:mm:ss");
auto logLine = QString("[%1] %2: %3").arg(datetime).arg(type).arg(message);
QTextStream stream(&logLine);
if (!fileLine.isEmpty()) {
stream << Qt::endl << "\t" + fileLine;
}
QString logLineReturn = logLine;
QTextStream streamReturn(&logLineReturn);
streamReturn << Qt::endl;
auto logLineReturn_c = qPrintable(logLineReturn);
#if defined(Q_OS_WIN)
// Debug output is viewable using either VS Code, Visual Studio, DebugView, or
// DbgView++ (only one can be used at once). It's important to send output to
// the debug output API, because it's difficult to view stdout and stderr from
// a Windows GUI app.
OutputDebugStringA(logLineReturn_c);
#else
fprintf(out, "%s", logLineReturn_c);
fflush(out);
#endif
return logLine;
}
void Logger::loadEnvVars() {
const auto debugEnvVar = qEnvironmentVariable("SYNERGY_GUI_DEBUG");
if (!debugEnvVar.isEmpty()) {
m_debug = strToTrue(debugEnvVar);
}
const auto verboseEnvVar = qEnvironmentVariable("SYNERGY_GUI_VERBOSE");
if (!verboseEnvVar.isEmpty()) {
m_verbose = strToTrue(verboseEnvVar);
}
}
void Logger::logVerbose(const QString &message) const {
if (m_verbose) {
printLine(stdout, "VERBOSE", message);
}
}
void Logger::handleMessage(
const QtMsgType type, const QMessageLogContext &context,
const QString &message) {
auto mutatedType = type;
if (kForceDebugMessages.contains(message)) {
mutatedType = QtDebugMsg;
}
QString typeString;
auto out = stdout;
switch (mutatedType) {
case QtDebugMsg:
typeString = "DEBUG";
if (!m_debug) {
return;
}
break;
case QtInfoMsg:
typeString = "INFO";
break;
case QtWarningMsg:
typeString = "WARNING";
out = stderr;
break;
case QtCriticalMsg:
typeString = "CRITICAL";
out = stderr;
break;
case QtFatalMsg:
typeString = "FATAL";
out = stderr;
break;
}
const auto logLine = printLine(out, typeString, message, fileLine(context));
emit newLine(logLine);
}
} // namespace synergy::gui