OSCAR-code/oscar/logger.cpp

262 lines
7.1 KiB
C++
Raw Normal View History

/* OSCAR Logger module implementation
2014-08-17 12:56:05 +00:00
*
* Copyright (c) 2020 The OSCAR Team
* Copyright (c) 2011-2018 Mark Watkins <mark@jedimark.net>
2014-08-17 12:56:05 +00:00
*
* This file is subject to the terms and conditions of the GNU General Public
* License. See the file COPYING in the main directory of the source code
* for more details. */
2014-08-17 12:56:05 +00:00
2014-06-20 07:05:40 +00:00
#include "logger.h"
#include "SleepLib/preferences.h"
#include "version.h"
#include <QDir>
2014-06-20 07:05:40 +00:00
QThreadPool * otherThreadPool = NULL;
void MyOutputHandler(QtMsgType type, const QMessageLogContext &context, const QString &msgtxt)
{
Q_UNUSED(context)
if (!logger) {
fprintf(stderr, "Pre/Post: %s\n", msgtxt.toLocal8Bit().constData());
return;
}
QString msg, typestr;
switch (type) {
case QtWarningMsg:
typestr = QString("Warning: ");
break;
case QtFatalMsg:
typestr = QString("Fatal: ");
break;
case QtCriticalMsg:
typestr = QString("Critical: ");
break;
default:
typestr = QString("Debug: ");
break;
}
msg = typestr + msgtxt; //+QString(" (%1:%2, %3)").arg(context.file).arg(context.line).arg(context.function);
2014-06-20 07:05:40 +00:00
if (logger && logger->isRunning()) {
logger->append(msg);
}
fprintf(stderr, "%s\n", msg.toLocal8Bit().constData());
2014-06-20 07:05:40 +00:00
if (type == QtFatalMsg) {
abort();
}
}
static QMutex s_LoggerRunning;
2014-06-20 07:05:40 +00:00
void initializeLogger()
{
s_LoggerRunning.lock(); // lock until the thread starts running
2014-06-20 07:05:40 +00:00
logger = new LogThread();
otherThreadPool = new QThreadPool();
bool b = otherThreadPool->tryStart(logger);
if (b) {
s_LoggerRunning.lock(); // wait until the thread begins running
s_LoggerRunning.unlock(); // we no longer need the lock
}
qInstallMessageHandler(MyOutputHandler); // NOTE: comment this line out when debugging a crash, otherwise the deferred output will mislead you.
2014-06-20 07:05:40 +00:00
if (b) {
qDebug() << "Started logging thread";
2014-06-20 07:05:40 +00:00
} else {
qWarning() << "Logging thread did not start correctly";
}
}
2014-06-20 07:05:40 +00:00
void LogThread::connectionReady()
{
strlock.lock();
connected = true;
strlock.unlock();
qDebug() << "Logging UI initialized";
2014-06-20 07:05:40 +00:00
}
void LogThread::logToFile()
{
QString debugLog = GetLogDir() + "/debug.txt";
rotateLogs(debugLog); // keep a limited set of previous logs
strlock.lock();
m_logFile = new QFile(debugLog);
Q_ASSERT(m_logFile);
if (m_logFile->open(QFile::ReadWrite | QFile::Text)) {
m_logStream = new QTextStream(m_logFile);
}
strlock.unlock();
if (m_logStream) {
qDebug().noquote() << "Logging to" << debugLog;
} else {
qWarning().noquote() << "Unable to open" << debugLog;
}
}
2014-06-20 07:05:40 +00:00
void shutdownLogger()
{
if (logger) {
logger->quit();
otherThreadPool->waitForDone(-1);
logger = NULL;
}
delete otherThreadPool;
}
LogThread * logger = NULL;
void LogThread::append(QString msg)
{
QString tmp = QString("%1: %2").arg(logtime.elapsed(), 5, 10, QChar('0')).arg(msg);
//QStringList appears not to be threadsafe
strlock.lock();
buffer.append(tmp);
strlock.unlock();
}
void LogThread::appendClean(QString msg)
{
strlock.lock();
buffer.append(msg);
strlock.unlock();
}
2014-06-20 07:05:40 +00:00
void LogThread::quit() {
qDebug() << "Shutting down logging thread";
running = false;
strlock.lock();
while (!buffer.isEmpty()) {
QString msg = buffer.takeFirst();
if (m_logStream) {
*m_logStream << msg << endl;
}
}
if (m_logStream) {
delete m_logStream;
m_logStream = nullptr;
Q_ASSERT(m_logFile);
delete m_logFile;
m_logFile = nullptr;
2014-06-20 07:05:40 +00:00
}
strlock.unlock();
}
void LogThread::run()
{
running = true;
s_LoggerRunning.unlock(); // unlock as soon as the thread begins to run
2014-06-20 07:05:40 +00:00
do {
strlock.lock();
2014-06-22 15:14:46 +00:00
//int r = receivers(SIGNAL(outputLog(QString())));
// Wait to flush the buffer until the UI is connected and the log file has been opened.
while (connected && m_logFile && !buffer.isEmpty()) {
2014-06-20 07:05:40 +00:00
QString msg = buffer.takeFirst();
if (m_logStream) {
*m_logStream << msg << endl;
}
emit outputLog(msg);
2014-06-20 07:05:40 +00:00
}
strlock.unlock();
QThread::msleep(1000);
} while (running);
}
QString GetLogDir()
{
static const QString LOG_DIR_NAME = "logs";
Q_ASSERT(!GetAppData().isEmpty()); // If GetLogDir gets called before GetAppData() is valid, this would point at root.
QDir oscarData(GetAppData());
Q_ASSERT(oscarData.exists());
if (!oscarData.exists(LOG_DIR_NAME)) {
oscarData.mkdir(LOG_DIR_NAME);
}
QDir logDir(oscarData.canonicalPath() + "/" + LOG_DIR_NAME);
if (!logDir.exists()) {
qWarning() << "Unable to create" << logDir.absolutePath() << "reverting to" << oscarData.canonicalPath();
logDir = oscarData;
}
Q_ASSERT(logDir.exists());
return logDir.canonicalPath();
}
void rotateLogs(const QString & filePath, int maxPrevious)
{
if (maxPrevious < 0) {
if (getVersion().IsReleaseVersion()) {
maxPrevious = 1;
} else {
// keep more in testing builds
maxPrevious = 4;
}
}
// Build the list of rotated logs for this filePath.
QFileInfo info(filePath);
QString path = QDir(info.absolutePath()).canonicalPath();
QString base = info.baseName();
QString ext = info.completeSuffix();
if (!ext.isEmpty()) {
ext = "." + ext;
}
if (path.isEmpty()) {
qWarning() << "Skipping log rotation, directory does not exist:" << info.absoluteFilePath();
return;
}
QStringList logs;
logs.append(filePath);
for (int i = 0; i < maxPrevious; i++) {
logs.append(QString("%1/%2.%3%4").arg(path).arg(base).arg(i).arg(ext));
}
// Remove the expired log.
QFileInfo expired(logs[maxPrevious]);
if (expired.exists()) {
if (expired.isDir()) {
QDir dir(expired.canonicalFilePath());
//qDebug() << "Removing expired log directory" << dir.canonicalPath();
if (!dir.removeRecursively()) {
qWarning() << "Unable to delete expired log directory" << dir.canonicalPath();
}
} else {
QFile file(expired.canonicalFilePath());
//qDebug() << "Removing expired log file" << file.fileName();
if (!file.remove()) {
qWarning() << "Unable to delete expired log file" << file.fileName();
}
}
}
// Rotate the remaining logs.
for (int i = maxPrevious; i > 0; i--) {
QFileInfo from(logs[i-1]);
QFileInfo to(logs[i]);
if (from.exists()) {
if (to.exists()) {
qWarning() << "Unable to rotate log:" << to.absoluteFilePath() << "exists";
continue;
}
//qDebug() << "Renaming" << from.absoluteFilePath() << "to" << to.absoluteFilePath();
if (!QFile::rename(from.absoluteFilePath(), to.absoluteFilePath())) {
qWarning() << "Unable to rename" << from.absoluteFilePath() << "to" << to.absoluteFilePath();
}
}
}
}