PRS1 Oximetery attachment fixes

This commit is contained in:
Mark Watkins 2014-09-30 15:25:11 +10:00
parent be6e66bdd7
commit de695e153d
5 changed files with 113 additions and 82 deletions

View File

@ -130,7 +130,6 @@ PRS1Loader::PRS1Loader()
m_pixmaps["System One (60 Series)"] = QPixmap(PRS1_60_ICON); m_pixmaps["System One (60 Series)"] = QPixmap(PRS1_60_ICON);
//genCRCTable(); // find what I did with this.. //genCRCTable(); // find what I did with this..
m_buffer = nullptr;
m_type = MT_CPAP; m_type = MT_CPAP;
} }
@ -176,8 +175,15 @@ QString PRS1Loader::checkDir(const QString & path)
QFile lastfile(newpath+"/last.txt"); QFile lastfile(newpath+"/last.txt");
bool exists = true;
if (!lastfile.exists()) {
lastfile.setFileName(newpath+"/LAST.TXT");
if (!lastfile.exists())
exists = false;
}
QString machpath; QString machpath;
if (lastfile.exists()) { if (exists) {
if (!lastfile.open(QIODevice::ReadOnly)) { if (!lastfile.open(QIODevice::ReadOnly)) {
qDebug() << "PRS1Loader: last.txt exists but I couldn't open it!"; qDebug() << "PRS1Loader: last.txt exists but I couldn't open it!";
} else { } else {
@ -264,7 +270,7 @@ void parseModel(MachineInfo & info, QString modelnum)
} }
} }
bool PRS1Loader::PeekProperties(MachineInfo & info, QString filename) bool PRS1Loader::PeekProperties(MachineInfo & info, QString filename, Machine * mach)
{ {
QFile f(filename); QFile f(filename);
if (!f.open(QFile::ReadOnly)) { if (!f.open(QFile::ReadOnly)) {
@ -275,10 +281,21 @@ bool PRS1Loader::PeekProperties(MachineInfo & info, QString filename)
QString line = in.readLine(); QString line = in.readLine();
QStringList pair = line.split("="); QStringList pair = line.split("=");
if (pair[0].compare("ModelNumber", Qt::CaseInsensitive) == 0) { bool skip = false;
if (pair[0].contains("ModelNumber", Qt::CaseInsensitive)) {
QString modelnum = pair[1]; QString modelnum = pair[1];
parseModel(info, modelnum); parseModel(info, modelnum);
skip = true;
} }
if (pair[0].contains("SerialNumber", Qt::CaseInsensitive)) {
info.serial = pair[1];
skip = true;
}
if (!mach || skip) continue;
mach->properties[pair[0]] = pair[1];
} while (!in.atEnd()); } while (!in.atEnd());
return true; return true;
} }
@ -292,6 +309,7 @@ MachineInfo PRS1Loader::PeekInfo(const QString & path)
MachineInfo info = newInfo(); MachineInfo info = newInfo();
info.serial = newpath.section("/", -1); info.serial = newpath.section("/", -1);
PeekProperties(info, newpath+"/properties.txt"); PeekProperties(info, newpath+"/properties.txt");
return info; return info;
} }
@ -320,8 +338,8 @@ int PRS1Loader::Open(QString path)
dir.setSorting(QDir::Name); dir.setSorting(QDir::Name);
QFileInfoList flist = dir.entryInfoList(); QFileInfoList flist = dir.entryInfoList();
QList<QString> SerialNumbers; QStringList SerialNumbers;
QList<QString>::iterator sn; QStringList::iterator sn;
for (int i = 0; i < flist.size(); i++) { for (int i = 0; i < flist.size(); i++) {
QFileInfo fi = flist.at(i); QFileInfo fi = flist.at(i);
@ -351,40 +369,23 @@ int PRS1Loader::Open(QString path)
if (SerialNumbers.empty()) { return -1; } if (SerialNumbers.empty()) { return -1; }
m_buffer = new unsigned char [max_load_buffer_size]; //allocate once and reuse.
Machine *m;
int c = 0; int c = 0;
for (sn = SerialNumbers.begin(); sn != SerialNumbers.end(); sn++) { for (sn = SerialNumbers.begin(); sn != SerialNumbers.end(); sn++) {
if (*sn == last) { if ((*sn)[0].isLetter()) {
c += OpenMachine(newpath + "/" + *sn);
} }
MachineInfo info = newInfo();
info.serial = *sn;
m = CreateMachine(info);
try {
if (m) {
c += OpenMachine(m, newpath + "/" + info.serial);
} }
} catch (OneTypePerDay e) { for (sn = SerialNumbers.begin(); sn != SerialNumbers.end(); sn++) {
Q_UNUSED(e) if (!(*sn)[0].isLetter()) {
p_profile->DelMachine(m); c += OpenMachine(newpath + "/" + *sn);
PRS1List.erase(PRS1List.find(info.serial));
QMessageBox::warning(nullptr,
QObject::tr("Import Error"),
QObject::tr("This Machine Record cannot be imported in this profile.\nThe Day records overlap with already existing content."),
QMessageBox::Ok);
delete m;
} }
} }
delete [] m_buffer;
return c; return c;
} }
bool PRS1Loader::ParseProperties(Machine *m, QString filename) /*bool PRS1Loader::ParseProperties(Machine *m, QString filename)
{ {
QFile f(filename); QFile f(filename);
@ -441,9 +442,9 @@ bool PRS1Loader::ParseProperties(Machine *m, QString filename)
f.close(); f.close();
return true; return true;
} }*/
int PRS1Loader::OpenMachine(Machine *m, QString path) int PRS1Loader::OpenMachine(QString path)
{ {
Q_ASSERT(p_profile != nullptr); Q_ASSERT(p_profile != nullptr);
@ -466,6 +467,8 @@ int PRS1Loader::OpenMachine(Machine *m, QString path)
int sessionid_base = 10; int sessionid_base = 10;
QString propertyfile;
for (int i = 0; i < flist.size(); i++) { for (int i = 0; i < flist.size(); i++) {
QFileInfo fi = flist.at(i); QFileInfo fi = flist.at(i);
filename = fi.fileName(); filename = fi.fileName();
@ -479,21 +482,18 @@ int PRS1Loader::OpenMachine(Machine *m, QString path)
// Reminder: I have been given some info about these. should check it over. // Reminder: I have been given some info about these. should check it over.
} }
} else if (filename.compare("properties.txt",Qt::CaseInsensitive) == 0) { } else if (filename.compare("properties.txt",Qt::CaseInsensitive) == 0) {
ParseProperties(m, fi.canonicalFilePath()); propertyfile = fi.canonicalFilePath();
} else if (filename.compare("PROP.TXT",Qt::CaseInsensitive) == 0) { } else if (filename.compare("PROP.TXT",Qt::CaseInsensitive) == 0) {
sessionid_base = 16; sessionid_base = 16;
ParseProperties(m, fi.canonicalFilePath()); propertyfile = fi.canonicalFilePath();
} }
} }
QString backupPath = m->getBackupPath() + path.section("/", -2); MachineInfo info = newInfo();
// Have a peek first to get the serial number.
PeekProperties(info, propertyfile);
if (QDir::cleanPath(path).compare(QDir::cleanPath(backupPath)) != 0) { QString modelstr = info.modelnumber;
copyPath(path, backupPath);
}
QString modelstr = m->modelnumber();
if (modelstr.endsWith("P")) if (modelstr.endsWith("P"))
modelstr.chop(1); modelstr.chop(1);
@ -501,20 +501,54 @@ int PRS1Loader::OpenMachine(Machine *m, QString path)
bool ok; bool ok;
int model = modelstr.toInt(&ok); int model = modelstr.toInt(&ok);
if (ok) { if (ok) {
int series = ((model / 10) % 10);
int type = (model / 100);
// Assumption is made here all PRS1 machines less than 450P are not data capable.. this could be wrong one day. // Assumption is made here all PRS1 machines less than 450P are not data capable.. this could be wrong one day.
if ((model < 450) && p_profile->cpap->brickWarning()) { if ((type < 4) && p_profile->cpap->brickWarning()) {
QApplication::processEvents(); QApplication::processEvents();
QMessageBox::information(QApplication::activeWindow(), QMessageBox::information(QApplication::activeWindow(),
QObject::tr("Non Data Capable Machine"), QObject::tr("Non Data Capable Machine"),
QString(QObject::tr("Your Philips Respironics CPAP machine (Model %1) is unfortunately not a data capable model.")+"\n\n"+ QString(QObject::tr("Your Philips Respironics CPAP machine (Model %1) is unfortunately not a data capable model.")+"\n\n"+
QObject::tr("I'm sorry to report that SleepyHead can only track hours of use and very basic settings for this machine.")). QObject::tr("I'm sorry to report that SleepyHead can only track hours of use and very basic settings for this machine.")).
arg(m->modelnumber()),QMessageBox::Ok); arg(info.modelnumber),QMessageBox::Ok);
p_profile->cpap->setBrickWarning(false); p_profile->cpap->setBrickWarning(false);
} }
} else {
// model number didn't parse.. Meh... // A bit of protection against future annoyances..
if (((series != 5) && (series != 6)) || (type >= 10)) {
QMessageBox::information(QApplication::activeWindow(),
QObject::tr("Machine Unsupported"),
QObject::tr("Sorry, your Philips Respironics CPAP machine (Model %1) is not supported yet.").arg(info.modelnumber) +"\n\n"+
QObject::tr("JediMark needs a .zip copy of this machines' SD card and matching Encore .pdf reports to make it work with SleepyHead.")
,QMessageBox::Ok);
return -1;
} }
} else {
// model number didn't parse.. Meh... Silently ignore it
// QMessageBox::information(QApplication::activeWindow(),
// QObject::tr("Machine Unsupported"),
// QObject::tr("SleepyHead could not parse the model number, this machine can not be imported..") +"\n\n"+
// QObject::tr("JediMark needs a .zip copy of this machines' SD card and matching Encore .pdf reports to make it work with SleepyHead.")
// ,QMessageBox::Ok);
return -1;
}
// Which is needed to get the right machine record..
Machine *m = CreateMachine(info);
// This time supply the machine object so it can populate machine properties..
PeekProperties(m->info, propertyfile, m);
QString backupPath = m->getBackupPath() + path.section("/", -2);
if (QDir::cleanPath(path).compare(QDir::cleanPath(backupPath)) != 0) {
copyPath(path, backupPath);
}
SessionID sid; SessionID sid;
long ext; long ext;
@ -565,6 +599,8 @@ int PRS1Loader::OpenMachine(Machine *m, QString path)
if (it != sesstasks.end()) { if (it != sesstasks.end()) {
task = it.value(); task = it.value();
} else { } else {
// Should probably check if session already imported has this data missing..
// Create the group if we see it first.. // Create the group if we see it first..
task = new PRS1Import(this, sid, m); task = new PRS1Import(this, sid, m);
sesstasks[sid] = task; sesstasks[sid] = task;
@ -597,15 +633,20 @@ int PRS1Loader::OpenMachine(Machine *m, QString path)
} }
} }
sid = chunk->sessionid; SessionID chunk_sid = chunk->sessionid;
if (m->SessionExists(sid)) {
delete chunk;
continue;
}
task = nullptr; task = nullptr;
QHash<SessionID, PRS1Import *>::iterator it = sesstasks.find(sid); QHash<SessionID, PRS1Import *>::iterator it = sesstasks.find(chunk_sid);
if (it != sesstasks.end()) { if (it != sesstasks.end()) {
task = it.value(); task = it.value();
} else { } else {
task = new PRS1Import(this, sid, m); task = new PRS1Import(this, chunk_sid, m);
sesstasks[sid] = task; sesstasks[chunk_sid] = task;
// save a loop an que this now // save a loop an que this now
queTask(task); queTask(task);
} }

View File

@ -153,7 +153,7 @@ class PRS1Loader : public CPAPLoader
virtual ~PRS1Loader(); virtual ~PRS1Loader();
QString checkDir(const QString & path); QString checkDir(const QString & path);
bool PeekProperties(MachineInfo & info, QString path); bool PeekProperties(MachineInfo & info, QString path, Machine * mach = nullptr);
//! \brief Detect if the given path contains a valid Folder structure //! \brief Detect if the given path contains a valid Folder structure
@ -170,9 +170,6 @@ class PRS1Loader : public CPAPLoader
//! \brief Return the loaderName, in this case "PRS1" //! \brief Return the loaderName, in this case "PRS1"
virtual const QString &loaderName() { return prs1_class_name; } virtual const QString &loaderName() { return prs1_class_name; }
// //! \brief Create a new PRS1 machine record, indexed by Serial number.
//Machine *CreateMachine(QString serial);
QList<PRS1DataChunk *> ParseFile(QString path); QList<PRS1DataChunk *> ParseFile(QString path);
//! \brief Register this Module to the list of Loaders, so it knows to search for PRS1 data. //! \brief Register this Module to the list of Loaders, so it knows to search for PRS1 data.
@ -200,20 +197,14 @@ class PRS1Loader : public CPAPLoader
QHash<QString, Machine *> PRS1List; QHash<QString, Machine *> PRS1List;
//! \brief Opens the SD folder structure for this machine, scans for data files and imports any new sessions //! \brief Opens the SD folder structure for this machine, scans for data files and imports any new sessions
int OpenMachine(Machine *m, QString path); int OpenMachine(QString path);
//! \brief Parses "properties.txt" file containing machine information // //! \brief Parses "properties.txt" file containing machine information
bool ParseProperties(Machine *m, QString filename); // bool ParseProperties(Machine *m, QString filename);
//bool OpenSummary(Session *session,QString filename);
//bool OpenEvents(Session *session,QString filename);
//! \brief Parse a .005 waveform file, extracting Flow Rate waveform (and Mask Pressure data if available) //! \brief Parse a .005 waveform file, extracting Flow Rate waveform (and Mask Pressure data if available)
bool OpenWaveforms(SessionID sid, QString filename); bool OpenWaveforms(SessionID sid, QString filename);
// //! \brief ParseWaveform chunk.. Currently unused, as the old one works fine.
//bool ParseWaveform(qint32 sequence, quint32 timestamp, unsigned char *data, quint16 size, quint16 duration, quint16 num_signals, quint16 interleave, quint8 sample_format);
//! \brief Parse a data chunk from the .000 (brick) and .001 (summary) files. //! \brief Parse a data chunk from the .000 (brick) and .001 (summary) files.
bool ParseSummary(Machine *mach, qint32 sequence, quint32 timestamp, unsigned char *data, bool ParseSummary(Machine *mach, qint32 sequence, quint32 timestamp, unsigned char *data,
quint16 size, int family, int familyVersion); quint16 size, int family, int familyVersion);
@ -223,10 +214,6 @@ class PRS1Loader : public CPAPLoader
//! \brief Open a PRS1 data file, and break into data chunks, delivering them to the correct parser. //! \brief Open a PRS1 data file, and break into data chunks, delivering them to the correct parser.
bool OpenFile(Machine *mach, QString filename); bool OpenFile(Machine *mach, QString filename);
//bool Parse002(Session *session,unsigned char *buffer,int size,qint64 timestamp,long fpos);
//bool Parse002ASV(Session *session,unsigned char *buffer,int size,qint64 timestamp,long fpos);
unsigned char *m_buffer;
QHash<SessionID, Session *> extra_session; QHash<SessionID, Session *> extra_session;
//! \brief PRS1 Data files can store multiple sessions, so store them in this list for later processing. //! \brief PRS1 Data files can store multiple sessions, so store them in this list for later processing.

View File

@ -398,10 +398,10 @@ bool Machine::unlinkSession(Session * sess)
b=true; b=true;
if (!d->searchMachine(mt)) { if (!d->searchMachine(mt)) {
d->machines.remove(mt); d->machines.remove(mt);
day.remove(dates[i]);
} }
if (d->size() == 0) { if (d->size() == 0) {
day.remove(dates[i]);
p_profile->unlinkDay(d); p_profile->unlinkDay(d);
} }
} }
@ -437,7 +437,7 @@ bool Machine::Purge(int secret)
QFile rxcache(p_profile->Get("{" + STR_GEN_DataFolder + "}/RXChanges.cache" )); QFile rxcache(p_profile->Get("{" + STR_GEN_DataFolder + "}/RXChanges.cache" ));
rxcache.remove(); rxcache.remove();
QFile sumfile(getDataPath()+"Summaries.xml"); QFile sumfile(getDataPath()+"Summaries.xml.gz");
sumfile.remove(); sumfile.remove();
// Create a copy of the list so the hash can be manipulated // Create a copy of the list so the hash can be manipulated

View File

@ -231,8 +231,9 @@ class Machine
QString getPixmapPath(); QString getPixmapPath();
QPixmap & getPixmap(); QPixmap & getPixmap();
protected:
MachineInfo info; MachineInfo info;
protected:
QDate firstday, lastday; QDate firstday, lastday;
SessionID highest_sessionid; SessionID highest_sessionid;
MachineID m_id; MachineID m_id;

View File

@ -683,6 +683,8 @@ void Daily::UpdateEventsTree(QTreeWidget *tree,Day *day)
for (QHash<ChannelID,QTreeWidgetItem *>::iterator m=mcroot.begin();m!=mcroot.end();m++) { for (QHash<ChannelID,QTreeWidgetItem *>::iterator m=mcroot.begin();m!=mcroot.end();m++) {
tree->insertTopLevelItem(cnt++,m.value()); tree->insertTopLevelItem(cnt++,m.value());
} }
if (day->hasMachine(MT_CPAP) || day->hasMachine(MT_OXIMETER) || day->hasMachine(MT_POSITION)) {
QTreeWidgetItem * start = new QTreeWidgetItem(QStringList(tr("Session Start Times"))); QTreeWidgetItem * start = new QTreeWidgetItem(QStringList(tr("Session Start Times")));
QTreeWidgetItem * end = new QTreeWidgetItem(QStringList(tr("Session End Times"))); QTreeWidgetItem * end = new QTreeWidgetItem(QStringList(tr("Session End Times")));
tree->insertTopLevelItem(cnt++ , start); tree->insertTopLevelItem(cnt++ , start);
@ -700,7 +702,7 @@ void Daily::UpdateEventsTree(QTreeWidget *tree,Day *day)
item->setData(0,Qt::UserRole, (*s)->last()); item->setData(0,Qt::UserRole, (*s)->last());
end->addChild(item); end->addChild(item);
} }
}
//tree->insertTopLevelItem(cnt++,new QTreeWidgetItem(QStringList("[Total Events ("+QString::number(total_events)+")]"))); //tree->insertTopLevelItem(cnt++,new QTreeWidgetItem(QStringList("[Total Events ("+QString::number(total_events)+")]")));
tree->sortByColumn(0,Qt::AscendingOrder); tree->sortByColumn(0,Qt::AscendingOrder);
//tree->expandAll(); //tree->expandAll();