mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-06 11:10:44 +00:00
Reworked logger to better handle multithreading
This commit is contained in:
parent
3d3e3fa5c8
commit
482542c34d
@ -25,6 +25,8 @@
|
|||||||
#include <QSettings>
|
#include <QSettings>
|
||||||
#include <QFileDialog>
|
#include <QFileDialog>
|
||||||
#include <QSysInfo>
|
#include <QSysInfo>
|
||||||
|
#include <QThreadPool>
|
||||||
|
|
||||||
|
|
||||||
#include "SleepLib/schema.h"
|
#include "SleepLib/schema.h"
|
||||||
#include "mainwindow.h"
|
#include "mainwindow.h"
|
||||||
@ -50,18 +52,19 @@
|
|||||||
|
|
||||||
MainWindow *mainwin = nullptr;
|
MainWindow *mainwin = nullptr;
|
||||||
|
|
||||||
|
QMutex mutex;
|
||||||
|
|
||||||
#if QT_VERSION < QT_VERSION_CHECK(5,0,0)
|
#if QT_VERSION < QT_VERSION_CHECK(5,0,0)
|
||||||
void MyOutputHandler(QtMsgType type, const char *msgtxt)
|
void MyOutputHandler(QtMsgType type, const char *msgtxt)
|
||||||
{
|
{
|
||||||
|
|
||||||
#else
|
#else
|
||||||
void MyOutputHandler(QtMsgType type, const QMessageLogContext &context, const QString &msgtxt)
|
void MyOutputHandler(QtMsgType type, const QMessageLogContext &context, const QString &msgtxt)
|
||||||
{
|
{
|
||||||
Q_UNUSED(context)
|
Q_UNUSED(context)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
if (!mainwin) {
|
if (!logger) {
|
||||||
// qInstallMessageHandler(0);
|
|
||||||
|
|
||||||
#if QT_VERSION >= QT_VERSION_CHECK(5,0,0)
|
#if QT_VERSION >= QT_VERSION_CHECK(5,0,0)
|
||||||
fprintf(stderr, "Pre/Post: %s\n", msgtxt.toLocal8Bit().constData());
|
fprintf(stderr, "Pre/Post: %s\n", msgtxt.toLocal8Bit().constData());
|
||||||
#else
|
#else
|
||||||
@ -96,13 +99,16 @@ void MyOutputHandler(QtMsgType type, const QMessageLogContext &context, const QS
|
|||||||
#else
|
#else
|
||||||
msg = typestr + msgtxt;
|
msg = typestr + msgtxt;
|
||||||
#endif
|
#endif
|
||||||
mainwin->Log(msg);
|
|
||||||
|
if (logger && logger->isRunning()) logger->append(msg);
|
||||||
|
else {
|
||||||
|
fprintf(stderr, msg.toLocal8Bit().data());
|
||||||
|
}
|
||||||
|
|
||||||
if (type == QtFatalMsg) {
|
if (type == QtFatalMsg) {
|
||||||
abort();
|
abort();
|
||||||
}
|
}
|
||||||
|
|
||||||
//loglock.unlock();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void initialize()
|
void initialize()
|
||||||
@ -173,6 +179,23 @@ int main(int argc, char *argv[])
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
logger = new LogThread();
|
||||||
|
QThreadPool * threadpool = QThreadPool::globalInstance();
|
||||||
|
bool b = threadpool->tryStart(logger);
|
||||||
|
if (b) {
|
||||||
|
qWarning() << "Started logging thread";
|
||||||
|
} else {
|
||||||
|
qWarning() << "Logging thread did not start correctly";
|
||||||
|
}
|
||||||
|
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(5,0,0)
|
||||||
|
qInstallMessageHandler(MyOutputHandler);
|
||||||
|
#else
|
||||||
|
qInstallMsgHandler(MyOutputHandler);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
// Language Selection
|
// Language Selection
|
||||||
////////////////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
@ -430,18 +453,12 @@ retry_directory:
|
|||||||
|
|
||||||
qDebug() << "Selected Font" << QApplication::font().family();
|
qDebug() << "Selected Font" << QApplication::font().family();
|
||||||
|
|
||||||
#if QT_VERSION >= QT_VERSION_CHECK(5,0,0)
|
|
||||||
qInstallMessageHandler(MyOutputHandler);
|
|
||||||
#else
|
|
||||||
qInstallMsgHandler(MyOutputHandler);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// Must be initialized AFTER profile creation
|
// Must be initialized AFTER profile creation
|
||||||
MainWindow w;
|
MainWindow w;
|
||||||
|
|
||||||
mainwin = &w;
|
mainwin = &w;
|
||||||
|
|
||||||
if (check_updates) { mainwin->CheckForUpdates(); }
|
// if (check_updates) { mainwin->CheckForUpdates(); }
|
||||||
|
|
||||||
w.show();
|
w.show();
|
||||||
|
|
||||||
|
@ -33,6 +33,8 @@
|
|||||||
#include <QTranslator>
|
#include <QTranslator>
|
||||||
#include <QPushButton>
|
#include <QPushButton>
|
||||||
#include <QCalendarWidget>
|
#include <QCalendarWidget>
|
||||||
|
#include <QThreadPool>
|
||||||
|
|
||||||
#include "common_gui.h"
|
#include "common_gui.h"
|
||||||
|
|
||||||
#include <cmath>
|
#include <cmath>
|
||||||
@ -114,31 +116,47 @@ QString getGraphicsEngine()
|
|||||||
return gfxEngine;
|
return gfxEngine;
|
||||||
}
|
}
|
||||||
|
|
||||||
void MainWindow::Log(QString s)
|
LogThread * logger = nullptr;
|
||||||
|
|
||||||
|
void LogThread::append(QString msg)
|
||||||
{
|
{
|
||||||
|
QString tmp = QString("%1: %2").arg(logtime.elapsed(), 5, 10, QChar('0')).arg(msg);
|
||||||
if (!strlock.tryLock()) {
|
//QStringList appears not to be threadsafe
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// strlock.lock();
|
|
||||||
QString tmp = QString("%1: %2").arg(logtime.elapsed(), 5, 10, QChar('0')).arg(s);
|
|
||||||
|
|
||||||
logbuffer.append(tmp); //QStringList appears not to be threadsafe
|
|
||||||
strlock.unlock();
|
|
||||||
|
|
||||||
strlock.lock();
|
strlock.lock();
|
||||||
|
buffer.append(tmp);
|
||||||
// only do this in the main thread?
|
|
||||||
for (int i = 0; i < logbuffer.size(); i++) {
|
|
||||||
ui->logText->appendPlainText(logbuffer[i]);
|
|
||||||
fprintf(stderr, "%s\n", logbuffer[i].toLocal8Bit().constData());
|
|
||||||
}
|
|
||||||
|
|
||||||
logbuffer.clear();
|
|
||||||
strlock.unlock();
|
strlock.unlock();
|
||||||
|
}
|
||||||
|
|
||||||
//loglock.unlock();
|
void LogThread::quit() {
|
||||||
|
qDebug() << "Shutting down logging thread";
|
||||||
|
running = false;
|
||||||
|
strlock.lock();
|
||||||
|
while (!buffer.isEmpty()) {
|
||||||
|
QString msg = buffer.takeFirst();
|
||||||
|
fprintf(stderr, "%s\n", msg.toLocal8Bit().constData());
|
||||||
|
}
|
||||||
|
strlock.unlock();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void LogThread::run()
|
||||||
|
{
|
||||||
|
running = true;
|
||||||
|
do {
|
||||||
|
strlock.lock();
|
||||||
|
while (!buffer.isEmpty()) {
|
||||||
|
QString msg = buffer.takeFirst();
|
||||||
|
emit outputLog(msg);
|
||||||
|
fprintf(stderr, "%s\n", msg.toLocal8Bit().constData());
|
||||||
|
}
|
||||||
|
strlock.unlock();
|
||||||
|
QThread::msleep(1000);
|
||||||
|
} while (running);
|
||||||
|
}
|
||||||
|
|
||||||
|
void MainWindow::logMessage(QString msg)
|
||||||
|
{
|
||||||
|
ui->logText->appendPlainText(msg);
|
||||||
}
|
}
|
||||||
|
|
||||||
MainWindow::MainWindow(QWidget *parent) :
|
MainWindow::MainWindow(QWidget *parent) :
|
||||||
@ -147,9 +165,12 @@ MainWindow::MainWindow(QWidget *parent) :
|
|||||||
{
|
{
|
||||||
Q_ASSERT(p_profile != nullptr);
|
Q_ASSERT(p_profile != nullptr);
|
||||||
|
|
||||||
logtime.start();
|
|
||||||
ui->setupUi(this);
|
ui->setupUi(this);
|
||||||
|
|
||||||
|
if (logger) {
|
||||||
|
connect(logger, SIGNAL(outputLog(QString)), this, SLOT(logMessage(QString)));
|
||||||
|
}
|
||||||
|
|
||||||
QString version = VersionString;
|
QString version = VersionString;
|
||||||
|
|
||||||
#ifdef TEST_BUILD
|
#ifdef TEST_BUILD
|
||||||
@ -164,6 +185,14 @@ MainWindow::MainWindow(QWidget *parent) :
|
|||||||
|
|
||||||
|
|
||||||
this->setWindowTitle(STR_TR_SleepyHead + QString(" v%1 (" + tr("Profile") + ": %2)").arg(version).arg(PREF[STR_GEN_Profile].toString()));
|
this->setWindowTitle(STR_TR_SleepyHead + QString(" v%1 (" + tr("Profile") + ": %2)").arg(version).arg(PREF[STR_GEN_Profile].toString()));
|
||||||
|
|
||||||
|
qDebug() << STR_TR_SleepyHeadVersion.toLocal8Bit().data() << "built with Qt" << QT_VERSION_STR <<
|
||||||
|
"on" << __DATE__ << __TIME__;
|
||||||
|
|
||||||
|
#ifdef BROKEN_OPENGL_BUILD
|
||||||
|
qDebug() << "This build has been created especially for computers with older graphics hardware.\n";
|
||||||
|
#endif
|
||||||
|
|
||||||
//ui->tabWidget->setCurrentIndex(1);
|
//ui->tabWidget->setCurrentIndex(1);
|
||||||
|
|
||||||
#ifdef Q_OS_MAC
|
#ifdef Q_OS_MAC
|
||||||
@ -315,27 +344,28 @@ MainWindow::MainWindow(QWidget *parent) :
|
|||||||
|
|
||||||
void MainWindow::closeEvent(QCloseEvent * event)
|
void MainWindow::closeEvent(QCloseEvent * event)
|
||||||
{
|
{
|
||||||
|
// Shutdown and Save the current User profile
|
||||||
|
Profiles::Done();
|
||||||
|
|
||||||
if (daily) {
|
if (daily) {
|
||||||
daily->close();
|
//daily->close();
|
||||||
daily->deleteLater();
|
daily->deleteLater();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (overview) {
|
if (overview) {
|
||||||
overview->close();
|
//overview->close();
|
||||||
overview->deleteLater();
|
overview->deleteLater();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (oximetry) {
|
if (oximetry) {
|
||||||
oximetry->close();
|
//oximetry->close();
|
||||||
oximetry->deleteLater();
|
oximetry->deleteLater();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Shutdown and Save the current User profile
|
|
||||||
Profiles::Done();
|
|
||||||
|
|
||||||
// Save current window position
|
// Save current window position
|
||||||
QSettings settings(getDeveloperName(), getAppName());
|
QSettings settings(getDeveloperName(), getAppName());
|
||||||
settings.setValue("MainWindow/geometry", saveGeometry());
|
settings.setValue("MainWindow/geometry", saveGeometry());
|
||||||
|
|
||||||
QMainWindow::closeEvent(event);
|
QMainWindow::closeEvent(event);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -349,6 +379,12 @@ MainWindow::~MainWindow()
|
|||||||
// Trash anything allocated by the Graph objects
|
// Trash anything allocated by the Graph objects
|
||||||
DestroyGraphGlobals();
|
DestroyGraphGlobals();
|
||||||
|
|
||||||
|
logger->quit();
|
||||||
|
disconnect(logger, SIGNAL(outputLog(QString)), this, SLOT(logMessage(QString)));
|
||||||
|
|
||||||
|
QThreadPool::globalInstance()->waitForDone(-1);
|
||||||
|
logger = nullptr;
|
||||||
|
|
||||||
mainwin = nullptr;
|
mainwin = nullptr;
|
||||||
delete ui;
|
delete ui;
|
||||||
}
|
}
|
||||||
@ -400,13 +436,6 @@ void MainWindow::PopulatePurgeMenu()
|
|||||||
|
|
||||||
void MainWindow::Startup()
|
void MainWindow::Startup()
|
||||||
{
|
{
|
||||||
qDebug() << STR_TR_SleepyHeadVersion.toLocal8Bit().data() << "built with Qt" << QT_VERSION_STR <<
|
|
||||||
"on" << __DATE__ << __TIME__;
|
|
||||||
|
|
||||||
#ifdef BROKEN_OPENGL_BUILD
|
|
||||||
qDebug() << "This build has been created especially for computers with older graphics hardware.\n";
|
|
||||||
#endif
|
|
||||||
|
|
||||||
qstatus->setText(tr("Loading Data"));
|
qstatus->setText(tr("Loading Data"));
|
||||||
qprogress->show();
|
qprogress->show();
|
||||||
//qstatusbar->showMessage(tr("Loading Data"),0);
|
//qstatusbar->showMessage(tr("Loading Data"),0);
|
||||||
@ -1219,11 +1248,14 @@ void MainWindow::on_actionDebug_toggled(bool checked)
|
|||||||
{
|
{
|
||||||
PROFILE.general->setShowDebug(checked);
|
PROFILE.general->setShowDebug(checked);
|
||||||
|
|
||||||
|
logger->strlock.lock();
|
||||||
if (checked) {
|
if (checked) {
|
||||||
ui->logText->show();
|
ui->logText->show();
|
||||||
} else {
|
} else {
|
||||||
ui->logText->hide();
|
ui->logText->hide();
|
||||||
}
|
}
|
||||||
|
// QApplication::processEvents();
|
||||||
|
logger->strlock.unlock();
|
||||||
}
|
}
|
||||||
|
|
||||||
void MainWindow::on_action_Reset_Graph_Layout_triggered()
|
void MainWindow::on_action_Reset_Graph_Layout_triggered()
|
||||||
|
@ -17,6 +17,8 @@
|
|||||||
#include <QNetworkAccessManager>
|
#include <QNetworkAccessManager>
|
||||||
#include <QNetworkReply>
|
#include <QNetworkReply>
|
||||||
#include <QSystemTrayIcon>
|
#include <QSystemTrayIcon>
|
||||||
|
#include <QRunnable>
|
||||||
|
#include <QPlainTextEdit>
|
||||||
|
|
||||||
#include "version.h"
|
#include "version.h"
|
||||||
#include "daily.h"
|
#include "daily.h"
|
||||||
@ -65,6 +67,30 @@ class Daily;
|
|||||||
class Report;
|
class Report;
|
||||||
class Overview;
|
class Overview;
|
||||||
|
|
||||||
|
class LogThread:public QObject, public QRunnable
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
public:
|
||||||
|
explicit LogThread() { running = false; logtime.start(); }
|
||||||
|
virtual ~LogThread() {}
|
||||||
|
|
||||||
|
void run();
|
||||||
|
void append(QString msg);
|
||||||
|
bool isRunning() { return running; }
|
||||||
|
|
||||||
|
void quit();
|
||||||
|
|
||||||
|
QStringList buffer;
|
||||||
|
QMutex strlock;
|
||||||
|
signals:
|
||||||
|
void outputLog(QString);
|
||||||
|
protected:
|
||||||
|
volatile bool running;
|
||||||
|
QTime logtime;
|
||||||
|
};
|
||||||
|
extern LogThread * logger;
|
||||||
|
|
||||||
|
|
||||||
/*! \class MainWindow
|
/*! \class MainWindow
|
||||||
\author Mark Watkins
|
\author Mark Watkins
|
||||||
\brief The Main Application window for SleepyHead
|
\brief The Main Application window for SleepyHead
|
||||||
@ -78,9 +104,6 @@ class MainWindow : public QMainWindow
|
|||||||
explicit MainWindow(QWidget *parent = 0);
|
explicit MainWindow(QWidget *parent = 0);
|
||||||
~MainWindow();
|
~MainWindow();
|
||||||
|
|
||||||
//! \brief Log message s to Application Debug log
|
|
||||||
void Log(QString s);
|
|
||||||
|
|
||||||
//! \brief Update the list of Favourites (Bookmarks) in the right sidebar.
|
//! \brief Update the list of Favourites (Bookmarks) in the right sidebar.
|
||||||
void updateFavourites();
|
void updateFavourites();
|
||||||
|
|
||||||
@ -314,6 +337,8 @@ class MainWindow : public QMainWindow
|
|||||||
|
|
||||||
void on_actionPurgeCurrentDaysOximetry_triggered();
|
void on_actionPurgeCurrentDaysOximetry_triggered();
|
||||||
|
|
||||||
|
void logMessage(QString msg);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
int importCPAP(const QString &path, const QString &message);
|
int importCPAP(const QString &path, const QString &message);
|
||||||
void importCPAPBackups();
|
void importCPAPBackups();
|
||||||
@ -329,8 +354,6 @@ private:
|
|||||||
Oximetry *oximetry;
|
Oximetry *oximetry;
|
||||||
bool first_load;
|
bool first_load;
|
||||||
PreferencesDialog *prefdialog;
|
PreferencesDialog *prefdialog;
|
||||||
QMutex loglock, strlock;
|
|
||||||
QStringList logbuffer;
|
|
||||||
QTime logtime;
|
QTime logtime;
|
||||||
QSystemTrayIcon *systray;
|
QSystemTrayIcon *systray;
|
||||||
QMenu *systraymenu;
|
QMenu *systraymenu;
|
||||||
@ -343,8 +366,6 @@ private:
|
|||||||
|
|
||||||
//! \brief Destroy ALL the CPAP data for the selected machine
|
//! \brief Destroy ALL the CPAP data for the selected machine
|
||||||
void purgeMachine(Machine *);
|
void purgeMachine(Machine *);
|
||||||
|
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // MAINWINDOW_H
|
#endif // MAINWINDOW_H
|
||||||
|
Loading…
Reference in New Issue
Block a user