mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-05 10:40:42 +00:00
Add ImportContext to begin reducing loader dependencies on Profile.
For now only the PRS1Loader makes use of the ImportContext.
This commit is contained in:
parent
332fc4294d
commit
de40161e8c
96
oscar/SleepLib/importcontext.cpp
Normal file
96
oscar/SleepLib/importcontext.cpp
Normal file
@ -0,0 +1,96 @@
|
||||
/* SleepLib Import Context Implementation
|
||||
*
|
||||
* Copyright (c) 2021 The OSCAR Team
|
||||
*
|
||||
* 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 <QApplication>
|
||||
#include <QMessageBox>
|
||||
|
||||
#include "SleepLib/importcontext.h"
|
||||
|
||||
|
||||
ImportContext::~ImportContext()
|
||||
{
|
||||
FlushUnexpectedMessages();
|
||||
}
|
||||
|
||||
void ImportContext::LogUnexpectedMessage(const QString & message)
|
||||
{
|
||||
m_mutex.lock();
|
||||
m_unexpectedMessages += message;
|
||||
m_mutex.unlock();
|
||||
}
|
||||
|
||||
void ImportContext::FlushUnexpectedMessages()
|
||||
{
|
||||
if (m_unexpectedMessages.count() > 0 && m_machine) {
|
||||
// Compare this to the list of messages previously seen for this machine
|
||||
// and only alert if there are new ones.
|
||||
QSet<QString> newMessages = m_unexpectedMessages - m_machine->previouslySeenUnexpectedData();
|
||||
if (newMessages.count() > 0) {
|
||||
emit importEncounteredUnexpectedData(m_machine->getInfo(), newMessages);
|
||||
m_machine->previouslySeenUnexpectedData() += newMessages;
|
||||
}
|
||||
}
|
||||
m_unexpectedMessages.clear();
|
||||
}
|
||||
|
||||
|
||||
ProfileImportContext::ProfileImportContext(Profile* profile)
|
||||
: m_profile(profile)
|
||||
{
|
||||
Q_ASSERT(m_profile);
|
||||
}
|
||||
|
||||
bool ProfileImportContext::ShouldIgnoreOldSessions()
|
||||
{
|
||||
return m_profile->session->ignoreOlderSessions();
|
||||
}
|
||||
|
||||
QDateTime ProfileImportContext::IgnoreSessionsOlderThan()
|
||||
{
|
||||
return m_profile->session->ignoreOlderSessionsDate();
|
||||
}
|
||||
|
||||
Machine* ProfileImportContext::CreateMachineFromInfo(const MachineInfo & info)
|
||||
{
|
||||
m_machineInfo = info;
|
||||
m_machine = m_profile->CreateMachine(m_machineInfo);
|
||||
return m_machine;
|
||||
}
|
||||
|
||||
|
||||
// MARK: -
|
||||
|
||||
ImportUI::ImportUI(Profile* profile)
|
||||
: m_profile(profile)
|
||||
{
|
||||
Q_ASSERT(m_profile);
|
||||
}
|
||||
|
||||
void ImportUI::onUnexpectedData(const MachineInfo & info, QSet<QString> & /*newMessages*/)
|
||||
{
|
||||
QMessageBox::information(QApplication::activeWindow(),
|
||||
QObject::tr("Untested Data"),
|
||||
QObject::tr("Your %1 %2 (%3) generated data that OSCAR has never seen before.").arg(info.brand).arg(info.model).arg(info.modelnumber) +"\n\n"+
|
||||
QObject::tr("The imported data may not be entirely accurate, so the developers would like a .zip copy of this machine's SD card and matching official .pdf reports to make sure OSCAR is handling the data correctly.")
|
||||
,QMessageBox::Ok);
|
||||
}
|
||||
|
||||
void ImportUI::onDeviceReportsUsageOnly(const MachineInfo & /*info*/)
|
||||
{
|
||||
// TODO
|
||||
}
|
||||
|
||||
void ImportUI::onDeviceIsUntested(const MachineInfo & /*info*/)
|
||||
{
|
||||
// TODO
|
||||
}
|
||||
|
||||
void ImportUI::onDeviceIsUnsupported(const MachineInfo & /*info*/)
|
||||
{
|
||||
// TODO
|
||||
}
|
88
oscar/SleepLib/importcontext.h
Normal file
88
oscar/SleepLib/importcontext.h
Normal file
@ -0,0 +1,88 @@
|
||||
/* SleepLib Import Context Header
|
||||
*
|
||||
* Copyright (c) 2021 The OSCAR Team
|
||||
*
|
||||
* 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. */
|
||||
|
||||
#ifndef IMPORTCONTEXT_H
|
||||
#define IMPORTCONTEXT_H
|
||||
|
||||
#include "SleepLib/machine_loader.h"
|
||||
|
||||
|
||||
class ImportContext : public QObject
|
||||
{
|
||||
Q_OBJECT
|
||||
public:
|
||||
ImportContext() {}
|
||||
virtual ~ImportContext();
|
||||
|
||||
// Loaders will call this directly. It manages the machine's stored set of previously seen messages
|
||||
// and will emit an importEncounteredUnexpectedData signal in its dtor if any are new.
|
||||
void LogUnexpectedMessage(const QString & message);
|
||||
|
||||
signals:
|
||||
void importEncounteredUnexpectedData(const MachineInfo & info, QSet<QString> & newMessages);
|
||||
|
||||
public:
|
||||
virtual bool ShouldIgnoreOldSessions() { return false; }
|
||||
virtual QDateTime IgnoreSessionsOlderThan() { return QDateTime(); }
|
||||
|
||||
// TODO: Isolate the Machine object from the loader rather than returning it.
|
||||
virtual Machine* CreateMachineFromInfo(const MachineInfo & info) = 0;
|
||||
|
||||
protected:
|
||||
void FlushUnexpectedMessages();
|
||||
|
||||
QMutex m_mutex;
|
||||
QSet<QString> m_unexpectedMessages;
|
||||
MachineInfo m_machineInfo;
|
||||
Machine* m_machine;
|
||||
};
|
||||
|
||||
|
||||
// ProfileImportContext isolates the loader from Profile and its storage.
|
||||
class Profile;
|
||||
class ProfileImportContext : public ImportContext
|
||||
{
|
||||
Q_OBJECT
|
||||
public:
|
||||
ProfileImportContext(Profile* profile);
|
||||
virtual ~ProfileImportContext() {};
|
||||
|
||||
virtual bool ShouldIgnoreOldSessions();
|
||||
virtual QDateTime IgnoreSessionsOlderThan();
|
||||
|
||||
virtual Machine* CreateMachineFromInfo(const MachineInfo & info);
|
||||
|
||||
protected:
|
||||
Profile* m_profile;
|
||||
};
|
||||
|
||||
|
||||
// TODO: Add a TestImportContext that writes the data to YAML for regression tests.
|
||||
|
||||
|
||||
// TODO: Once all loaders support context and UI, move this into the main application
|
||||
// and refactor its import UI logic into this class.
|
||||
class ImportUI : public QObject
|
||||
{
|
||||
Q_OBJECT
|
||||
public:
|
||||
ImportUI(Profile* profile);
|
||||
virtual ~ImportUI() {}
|
||||
|
||||
public slots:
|
||||
void onUnexpectedData(const MachineInfo & machine, QSet<QString> & newMessages);
|
||||
void onDeviceReportsUsageOnly(const MachineInfo & info);
|
||||
void onDeviceIsUntested(const MachineInfo & info);
|
||||
void onDeviceIsUnsupported(const MachineInfo & info);
|
||||
|
||||
protected:
|
||||
Profile* m_profile;
|
||||
};
|
||||
|
||||
|
||||
#endif // IMPORTCONTEXT_H
|
@ -18,6 +18,7 @@
|
||||
#include <cmath>
|
||||
|
||||
#include "SleepLib/schema.h"
|
||||
#include "SleepLib/importcontext.h"
|
||||
#include "prs1_loader.h"
|
||||
#include "prs1_parser.h"
|
||||
#include "SleepLib/session.h"
|
||||
@ -53,9 +54,7 @@ QString ts(qint64 msecs)
|
||||
// TODO: See the LogUnexpectedMessage TODO about generalizing this for other loaders.
|
||||
void PRS1Loader::LogUnexpectedMessage(const QString & message)
|
||||
{
|
||||
m_importMutex.lock();
|
||||
m_unexpectedMessages += message;
|
||||
m_importMutex.unlock();
|
||||
m_ctx->LogUnexpectedMessage(message);
|
||||
}
|
||||
|
||||
|
||||
@ -459,6 +458,7 @@ PRS1Loader::PRS1Loader()
|
||||
#endif
|
||||
|
||||
m_type = MT_CPAP;
|
||||
m_ctx = nullptr;
|
||||
}
|
||||
|
||||
PRS1Loader::~PRS1Loader()
|
||||
@ -737,9 +737,25 @@ int PRS1Loader::Open(const QString & selectedPath)
|
||||
}
|
||||
|
||||
// Import each machine, from oldest to newest.
|
||||
// TODO: Loaders should return the set of machines during detection, so that Open() will
|
||||
// open a unique device, instead of surprising the user.
|
||||
int c = 0;
|
||||
for (auto & machinePath : machines) {
|
||||
#if 1
|
||||
// TODO: Move this to the main application once all loaders support contexts and UI signals.
|
||||
if (p_profile == nullptr) {
|
||||
qWarning() << "PRS1Loader::Open() called without a valid p_profile object present";
|
||||
return 0;
|
||||
}
|
||||
ImportUI ui(p_profile);
|
||||
ImportContext* ctx = new ProfileImportContext(p_profile);
|
||||
SetContext(ctx);
|
||||
connect(ctx, &ImportContext::importEncounteredUnexpectedData, &ui, &ImportUI::onUnexpectedData);
|
||||
#endif
|
||||
c += OpenMachine(machinePath);
|
||||
#if 1
|
||||
delete ctx;
|
||||
#endif
|
||||
}
|
||||
return c;
|
||||
}
|
||||
@ -747,10 +763,7 @@ int PRS1Loader::Open(const QString & selectedPath)
|
||||
|
||||
int PRS1Loader::OpenMachine(const QString & path)
|
||||
{
|
||||
if (p_profile == nullptr) {
|
||||
qWarning() << "PRS1Loader::OpenMachine() called without a valid p_profile object present";
|
||||
return 0;
|
||||
}
|
||||
Q_ASSERT(m_ctx);
|
||||
|
||||
qDebug() << "Opening PRS1 " << path;
|
||||
QDir dir(path);
|
||||
@ -802,22 +815,6 @@ int PRS1Loader::OpenMachine(const QString & path)
|
||||
|
||||
finishAddingSessions();
|
||||
|
||||
if (m_unexpectedMessages.count() > 0 && p_profile->session->warnOnUnexpectedData()) {
|
||||
// Compare this to the list of messages previously seen for this machine
|
||||
// and only alert if there are new ones.
|
||||
QSet<QString> newMessages = m_unexpectedMessages - m->previouslySeenUnexpectedData();
|
||||
if (newMessages.count() > 0) {
|
||||
// TODO: Rework the importer call structure so that this can become an
|
||||
// emit statement to the appropriate import job.
|
||||
QMessageBox::information(QApplication::activeWindow(),
|
||||
QObject::tr("Untested Data"),
|
||||
QObject::tr("Your Philips Respironics %1 (%2) generated data that OSCAR has never seen before.").arg(m->getInfo().model).arg(m->getInfo().modelnumber) +"\n\n"+
|
||||
QObject::tr("The imported data may not be entirely accurate, so the developers would like a .zip copy of this machine's SD card and matching Encore .pdf reports to make sure OSCAR is handling the data correctly.")
|
||||
,QMessageBox::Ok);
|
||||
m->previouslySeenUnexpectedData() += newMessages;
|
||||
}
|
||||
}
|
||||
|
||||
return m->unsupported() ? -1 : tasks;
|
||||
}
|
||||
|
||||
@ -907,7 +904,7 @@ Machine* PRS1Loader::CreateMachineFromProperties(QString propertyfile)
|
||||
}
|
||||
|
||||
// Which is needed to get the right machine record..
|
||||
Machine *m = p_profile->CreateMachine(info);
|
||||
Machine *m = m_ctx->CreateMachineFromInfo(info);
|
||||
|
||||
// This time supply the machine object so it can populate machine properties..
|
||||
PeekProperties(m->info, propertyfile, m);
|
||||
@ -975,7 +972,6 @@ void PRS1Loader::ScanFiles(const QStringList & paths, int sessionid_base, Machin
|
||||
|
||||
sesstasks.clear();
|
||||
new_sessions.clear(); // this hash is used by OpenFile
|
||||
m_unexpectedMessages.clear();
|
||||
|
||||
|
||||
PRS1Import * task = nullptr;
|
||||
@ -983,8 +979,8 @@ void PRS1Loader::ScanFiles(const QStringList & paths, int sessionid_base, Machin
|
||||
|
||||
QDateTime datetime;
|
||||
|
||||
qint64 ignoreBefore = p_profile->session->ignoreOlderSessionsDate().toMSecsSinceEpoch()/1000;
|
||||
bool ignoreOldSessions = p_profile->session->ignoreOlderSessions();
|
||||
qint64 ignoreBefore = m_ctx->IgnoreSessionsOlderThan().toMSecsSinceEpoch()/1000;
|
||||
bool ignoreOldSessions = m_ctx->ShouldIgnoreOldSessions();
|
||||
QSet<SessionID> skipped;
|
||||
|
||||
// for each p0/p1/p2/etc... folder
|
||||
@ -2333,8 +2329,8 @@ QList<PRS1DataChunk *> PRS1Import::CoalesceWaveformChunks(QList<PRS1DataChunk *>
|
||||
// This won't be perfect, since any coalesced chunks starting after midnight of the threshhold
|
||||
// date will also be imported, but those should be relatively few, and tolerable imprecision.
|
||||
QList<PRS1DataChunk *> coalescedAndFiltered;
|
||||
qint64 ignoreBefore = p_profile->session->ignoreOlderSessionsDate().toMSecsSinceEpoch()/1000;
|
||||
bool ignoreOldSessions = p_profile->session->ignoreOlderSessions();
|
||||
qint64 ignoreBefore = loader->context()->IgnoreSessionsOlderThan().toMSecsSinceEpoch()/1000;
|
||||
bool ignoreOldSessions = loader->context()->ShouldIgnoreOldSessions();
|
||||
|
||||
for (auto & chunk : coalesced) {
|
||||
if (ignoreOldSessions && chunk->timestamp < ignoreBefore) {
|
||||
|
@ -57,6 +57,7 @@ struct PRS1Waveform {
|
||||
quint8 sample_format;
|
||||
};
|
||||
|
||||
class ImportContext;
|
||||
|
||||
class PRS1DataChunk;
|
||||
class PRS1ParsedEvent;
|
||||
@ -179,6 +180,9 @@ class PRS1Loader : public CPAPLoader
|
||||
PRS1Loader();
|
||||
virtual ~PRS1Loader();
|
||||
|
||||
void SetContext(ImportContext* ctx) { m_ctx = ctx; }
|
||||
inline ImportContext* context() { return m_ctx; }
|
||||
|
||||
//! \brief Peek into PROP.TXT or properties.txt at given path, and return it as a normalized key/value hash
|
||||
bool PeekProperties(const QString & filename, QHash<QString,QString> & props);
|
||||
|
||||
@ -231,7 +235,13 @@ class PRS1Loader : public CPAPLoader
|
||||
|
||||
QHash<SessionID, PRS1Import*> sesstasks;
|
||||
|
||||
signals:
|
||||
void deviceReportsUsageOnly(MachineInfo & info);
|
||||
void deviceIsUntested(MachineInfo & info);
|
||||
void deviceIsUnsupported(MachineInfo & info);
|
||||
|
||||
protected:
|
||||
ImportContext* m_ctx;
|
||||
QString last;
|
||||
QHash<QString, Machine *> PRS1List;
|
||||
|
||||
@ -269,13 +279,6 @@ class PRS1Loader : public CPAPLoader
|
||||
//! \brief PRS1 Data files can store multiple sessions, so store them in this list for later processing.
|
||||
QHash<SessionID, Session *> new_sessions;
|
||||
|
||||
// TODO: This really belongs in a generic location that all loaders can use.
|
||||
// But that will require retooling the overall call structure so that there's
|
||||
// a top-level import job that's managing a specific import. Right now it's
|
||||
// essentially managed by the importCPAP method rather than an object instance
|
||||
// with state.
|
||||
QMutex m_importMutex;
|
||||
QSet<QString> m_unexpectedMessages;
|
||||
public:
|
||||
void LogUnexpectedMessage(const QString & message);
|
||||
};
|
||||
|
@ -288,6 +288,7 @@ SOURCES += \
|
||||
SleepLib/event.cpp \
|
||||
SleepLib/machine.cpp \
|
||||
SleepLib/machine_loader.cpp \
|
||||
SleepLib/importcontext.cpp \
|
||||
SleepLib/preferences.cpp \
|
||||
SleepLib/profiles.cpp \
|
||||
SleepLib/schema.cpp \
|
||||
@ -375,6 +376,7 @@ HEADERS += \
|
||||
SleepLib/machine.h \
|
||||
SleepLib/machine_common.h \
|
||||
SleepLib/machine_loader.h \
|
||||
SleepLib/importcontext.h \
|
||||
SleepLib/preferences.h \
|
||||
SleepLib/profiles.h \
|
||||
SleepLib/schema.h \
|
||||
|
@ -11,6 +11,7 @@
|
||||
|
||||
#include "../SleepLib/loader_plugins/prs1_loader.h"
|
||||
#include "../SleepLib/loader_plugins/prs1_parser.h"
|
||||
#include "../SleepLib/importcontext.h"
|
||||
|
||||
#define TESTDATA_PATH "./testdata/"
|
||||
|
||||
@ -80,6 +81,13 @@ void parseAndEmitSessionYaml(const QString & path)
|
||||
{
|
||||
qDebug() << path;
|
||||
|
||||
ImportContext* ctx = new ProfileImportContext(p_profile);
|
||||
s_loader->SetContext(ctx);
|
||||
|
||||
ctx->connect(ctx, &ImportContext::importEncounteredUnexpectedData, [=]() {
|
||||
qWarning() << "*** Found unexpected data";
|
||||
});
|
||||
|
||||
// This mirrors the functional bits of PRS1Loader::OpenMachine.
|
||||
// TODO: Refactor PRS1Loader so that the tests can use the same
|
||||
// underlying logic as OpenMachine rather than duplicating it here.
|
||||
@ -114,9 +122,8 @@ void parseAndEmitSessionYaml(const QString & path)
|
||||
delete session;
|
||||
delete task;
|
||||
}
|
||||
if (s_loader->m_unexpectedMessages.count() > 0) {
|
||||
qWarning() << "*** Found unexpected data";
|
||||
}
|
||||
|
||||
delete ctx;
|
||||
}
|
||||
|
||||
void PRS1Tests::testSessionsToYaml()
|
||||
@ -263,6 +270,9 @@ void parseAndEmitChunkYaml(const QString & path)
|
||||
bool FV = false; // set this to true to emit family/familyVersion for this path
|
||||
qDebug() << path;
|
||||
|
||||
ImportContext* ctx = new ProfileImportContext(p_profile);
|
||||
s_loader->SetContext(ctx);
|
||||
|
||||
QHash<QString,QSet<quint64>> written;
|
||||
QStringList paths;
|
||||
QString propertyfile;
|
||||
@ -371,6 +381,8 @@ void parseAndEmitChunkYaml(const QString & path)
|
||||
}
|
||||
}
|
||||
|
||||
delete ctx;
|
||||
|
||||
p_profile->removeMachine(m);
|
||||
delete m;
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user