OSCAR-code/oscar/logger.cpp
Guy Scharf b338d7aa3d Check for ability to write to data directory at startup
Show critical error message and exit OSCAR if unable to write to data directory
    Improve qWarning message in logger.cpp
    Present warning dialog if logger cannot write to disk
2020-08-16 12:46:36 -07:00

285 lines
7.7 KiB
C++

/* OSCAR Logger module implementation
*
* Copyright (c) 2020 The OSCAR Team
* Copyright (c) 2011-2018 Mark Watkins <mark@jedimark.net>
*
* 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. */
#include "logger.h"
#include "SleepLib/preferences.h"
#include "version.h"
#include <QDir>
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);
if (logger && logger->isRunning()) {
logger->append(msg);
} else {
fprintf(stderr, "%s\n", msg.toLocal8Bit().constData());
}
if (type == QtFatalMsg) {
abort();
}
}
static QMutex s_LoggerRunning;
void initializeLogger()
{
s_LoggerRunning.lock(); // lock until the thread starts running
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.
if (b) {
qDebug() << "Started logging thread";
} else {
qWarning() << "Logging thread did not start correctly";
}
}
void LogThread::connectionReady()
{
strlock.lock();
connected = true;
logTrigger.wakeAll();
strlock.unlock();
qDebug() << "Logging UI initialized";
}
bool LogThread::logToFile()
{
if (m_logStream) {
qWarning().noquote() << "Already logging to" << m_logFile->fileName();
return true;
}
QString debugLog = GetLogDir() + "/debug.txt";
rotateLogs(debugLog, 4); // 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);
}
logTrigger.wakeAll();
strlock.unlock();
if (m_logStream) {
qDebug().noquote() << "Logging to" << debugLog;
} else {
qWarning().noquote() << "Could not open" << debugLog << "error code" << m_logFile->error() << m_logFile->errorString();
return false;
}
return true;
}
LogThread::~LogThread()
{
QMutexLocker lock(&strlock);
Q_ASSERT(running == false);
if (m_logStream) {
delete m_logStream;
m_logStream = nullptr;
Q_ASSERT(m_logFile);
delete m_logFile;
m_logFile = nullptr;
}
}
QString LogThread::logFileName()
{
if (!m_logFile) {
return "";
}
return m_logFile->fileName();
}
void shutdownLogger()
{
if (logger) {
logger->quit();
// The thread is automatically destroyed when its run() method exits.
otherThreadPool->waitForDone(-1); // wait until that happens
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);
appendClean(tmp);
}
void LogThread::appendClean(QString msg)
{
fprintf(stderr, "%s\n", msg.toLocal8Bit().constData());
strlock.lock();
buffer.append(msg);
logTrigger.wakeAll();
strlock.unlock();
}
void LogThread::quit() {
qDebug() << "Shutting down logging thread";
qInstallMessageHandler(0); // Remove our logger.
strlock.lock();
running = false; // Force the thread to exit after its next iteration.
logTrigger.wakeAll(); // Trigger the final flush.
strlock.unlock(); // Release the lock so that the thread can complete.
}
void LogThread::run()
{
QMutexLocker lock(&strlock);
running = true;
s_LoggerRunning.unlock(); // unlock as soon as the thread begins to run
do {
logTrigger.wait(&strlock); // releases strlock while it waits
while (connected && m_logFile && !buffer.isEmpty()) {
QString msg = buffer.takeFirst();
if (m_logStream) {
*m_logStream << msg << endl;
}
emit outputLog(msg);
}
} while (running);
// strlock will be released when lock goes out of scope
}
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();
}
}
}
}