Support for 'Wellue O2Ring S' code author:P. Fisher

This commit is contained in:
LoudSnorer 2024-08-18 11:32:22 -04:00
parent d6b7c05ed2
commit 9b041cf22d
2 changed files with 116 additions and 21 deletions

View File

@ -127,8 +127,35 @@ Session* ViatomLoader::ParseFile(const QString & filename, bool *existing)
return nullptr;
}
ViatomFile v(file);
if (v.ParseHeader() == false) {
// Read in Viatom database version number
QByteArray data = file.read(2);
if (data.size() < 2) {
qDebug() << filename << "too short for a Viatom data file";
return nullptr;
}
const unsigned char* header = (const unsigned char*) data.constData();
int sig = header[0] | (header[1] << 8);
std::unique_ptr<ViatomFile> v = nullptr;
switch (sig) {
case 0x0003:
case 0x0005:
v = std::unique_ptr<ViatomFile>(new ViatomFile(file));
break;
case 0x0301:
v = std::unique_ptr<O2RingS>(new O2RingS(file));
break;
default:
qDebug() << filename << "Unrecognized DB version number in Viatom data file" << sig;
return nullptr;
}
if (!file.seek(0)) {
qDebug() << filename << "unable to seek to begining of file";
return nullptr;
}
// Parse header specific to database version number
if (v->ParseHeader() == false) {
return nullptr;
}
@ -144,7 +171,7 @@ Session* ViatomLoader::ParseFile(const QString & filename, bool *existing)
}
Machine *mach = p_profile->CreateMachine(info);
if (mach->SessionExists(v.sessionid())) {
if (mach->SessionExists(v->sessionid())) {
// Skip already imported session
//qDebug() << filename << "session already exists, skipping" << v.sessionid();
if (existing) {
@ -154,12 +181,12 @@ Session* ViatomLoader::ParseFile(const QString & filename, bool *existing)
return nullptr;
}
qint64 time_ms = v.timestamp();
m_session = new Session(mach, v.sessionid());
qint64 time_ms = v->timestamp();
m_session = new Session(mach, v->sessionid());
m_session->set_first(time_ms);
QList<ViatomFile::Record> records = v.ReadData();
m_step = v.duration() / records.size() * 1000L;
QList<ViatomFile::Record> records = v->ReadData();
m_step = v->duration() / records.size() * 1000L;
// Import data
for (auto & rec : records) {
@ -267,6 +294,20 @@ ViatomFile::ViatomFile(QFile & file) : m_file(file)
{
}
QDateTime ViatomFile::getFilenameTimestamp()
{
QString date_string = QFileInfo(m_file).fileName().section("_", -1); // Strip any SleepU_ etc. prefix.
int lastPoint = date_string.lastIndexOf("."); // Added to strip off any filename extension
date_string = date_string.left(lastPoint);
QString format_string = "yyyyMMddHHmmss";
if (date_string.contains(":")) {
format_string = "yyyy-MM-dd HH:mm:ss";
}
return QDateTime::fromString(date_string, format_string);
}
bool ViatomFile::ParseHeader()
{
static const int HEADER_SIZE = 40;
@ -312,17 +353,7 @@ bool ViatomFile::ParseHeader()
// starting timestamp). Technically these should probably be square charts, but
// the code currently forces them to be non-square.
QDateTime data_timestamp = QDateTime(QDate(year, month, day), QTime(hour, min, sec));
QString date_string = QFileInfo(m_file).fileName().section("_", -1); // Strip any SleepU_ etc. prefix.
int lastPoint = date_string.lastIndexOf("."); // Added to strip off any filename extension
date_string = date_string.left(lastPoint);
QString format_string = "yyyyMMddHHmmss";
if (date_string.contains(":")) {
format_string = "yyyy-MM-dd HH:mm:ss";
}
QDateTime filename_timestamp = QDateTime::fromString(date_string, format_string);
QDateTime filename_timestamp = getFilenameTimestamp();
if (filename_timestamp.isValid()) {
if (filename_timestamp != data_timestamp) {
// TODO: Once there's a better/easier way to adjust session times within OSCAR, we can remove the below.
@ -462,3 +493,57 @@ QList<ViatomFile::Record> ViatomFile::ReadData()
return records;
}
O2RingS::O2RingS(QFile & file) : ViatomFile(file)
{
}
bool O2RingS::ParseHeader()
{
// For the O2Ring S, the header only contains the signature
// Additional metadata is stored at the end of the file
// The record count is stored 36 bytes prior to EOF
int record_count_loc = m_file.size() - 36;
if (record_count_loc < 0 || !m_file.seek(record_count_loc)) {
qDebug() << m_file.fileName() << "error locating Viatom record count";
return false;
}
// read record count as a 2-byte little endian value
// max number of records in a O2Ring S file is 36000
QDataStream in(m_file.read(2));
in.setByteOrder(QDataStream::LittleEndian);
quint16 record_count;
in >> record_count;
m_sig = 0x0301;
m_record_count = m_duration = record_count;
m_timestamp = getFilenameTimestamp().toMSecsSinceEpoch();
m_sessionid = m_timestamp / 1000L;
m_resolution = 1000;
// advance past the header
return m_file.seek(10);
}
QList<ViatomFile::Record> O2RingS::ReadData()
{
QList<ViatomFile::Record> records;
// Read all Pulse, SPO2 and Motion data
// 0xFF for spo2 or hr indicates an interruption in measurement
// Vibration data is likely stored in a variable length block following the
// fixed-width pulse/SPO2/motion data. Zero out for now since OSCAR doesn't
// use this data.
QDataStream in(m_file.readAll());
do {
ViatomFile::Record rec;
in >> rec.spo2 >> rec.hr >> rec.motion;
rec.oximetry_invalid = (rec.spo2 == 0xFF || rec.hr == 0xFF) ? 0xFF : 0;
rec.vibration = 0;
records.append(rec);
} while (records.size() < m_record_count);
// Confirm that we have a 1s sample rate
CHECK_VALUE(duration() / records.size(), 1);
return records;
}

View File

@ -71,13 +71,14 @@ public:
unsigned char vibration;
};
ViatomFile(QFile & file);
~ViatomFile() = default;
virtual ~ViatomFile() = default;
bool ParseHeader();
QList<Record> ReadData();
virtual bool ParseHeader();
virtual QList<Record> ReadData();
SessionID sessionid() const { return m_sessionid; }
quint64 timestamp() const { return m_timestamp; }
int duration() const { return m_duration; }
QDateTime getFilenameTimestamp();
protected:
static const int RECORD_SIZE = 5;
@ -90,4 +91,13 @@ protected:
SessionID m_sessionid;
};
class O2RingS : public ViatomFile
{
public:
O2RingS(QFile & file);
~O2RingS() = default;
bool ParseHeader();
QList<Record> ReadData();
};
#endif // VIATOMLOADER_H