mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-05 18:50:44 +00:00
Something's not quite right about the hypnogram timestamps, since there are more than would fit within the start/stop times.
259 lines
7.7 KiB
C++
259 lines
7.7 KiB
C++
/* SleepLib Dreem Loader Implementation
|
|
*
|
|
* Copyright (c) 2020 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. */
|
|
|
|
//********************************************************************************************
|
|
// IMPORTANT!!!
|
|
//********************************************************************************************
|
|
// Please INCREMENT the dreem_data_version in dreem_loader.h when making changes to this loader
|
|
// that change loader behaviour or modify channels.
|
|
//********************************************************************************************
|
|
|
|
#include <QDir>
|
|
#include <QTextStream>
|
|
#include <QApplication>
|
|
#include <QMessageBox>
|
|
#include "dreem_loader.h"
|
|
#include "SleepLib/machine.h"
|
|
#include "csv.h"
|
|
|
|
static QSet<QString> s_unexpectedMessages;
|
|
|
|
bool
|
|
DreemLoader::Detect(const QString & path)
|
|
{
|
|
// This is only used for CPAP machines, when detecting CPAP cards.
|
|
qDebug() << "DreemLoader::Detect(" << path << ")";
|
|
return false;
|
|
}
|
|
|
|
int
|
|
DreemLoader::Open(const QString & dirpath)
|
|
{
|
|
qDebug() << "DreemLoader::Open(" << dirpath << ")";
|
|
// Dreem currently crams everything into a single file like Zeo did.
|
|
// See OpenFile.
|
|
return false;
|
|
}
|
|
|
|
int DreemLoader::OpenFile(const QString & filename)
|
|
{
|
|
if (!openCSV(filename)) {
|
|
closeCSV();
|
|
return -1;
|
|
}
|
|
int count = 0;
|
|
Session* sess;
|
|
while ((sess = readNextSession()) != nullptr) {
|
|
sess->SetChanged(true);
|
|
mach->AddSession(sess);
|
|
count++;
|
|
}
|
|
mach->Save();
|
|
closeCSV();
|
|
return count;
|
|
}
|
|
|
|
bool DreemLoader::openCSV(const QString & filename)
|
|
{
|
|
file.setFileName(filename);
|
|
|
|
if (filename.toLower().endsWith(".csv")) {
|
|
if (!file.open(QFile::ReadOnly)) {
|
|
qDebug() << "Couldn't open Dreem file" << filename;
|
|
return false;
|
|
}
|
|
} else {
|
|
return false;
|
|
}
|
|
|
|
QStringList header;
|
|
csv = new CSVReader(file, ";", "#");
|
|
bool ok = csv->readRow(header);
|
|
if (!ok) {
|
|
qWarning() << "no header row";
|
|
return false;
|
|
}
|
|
csv->setFieldNames(header);
|
|
|
|
MachineInfo info = newInfo();
|
|
mach = p_profile->CreateMachine(info);
|
|
|
|
return true;
|
|
}
|
|
|
|
void DreemLoader::closeCSV()
|
|
{
|
|
if (csv != nullptr) {
|
|
delete csv;
|
|
csv = nullptr;
|
|
}
|
|
if (file.isOpen()) {
|
|
file.close();
|
|
}
|
|
}
|
|
|
|
|
|
|
|
const QStringList s_sleepStageLabels = { "WAKE", "REM", "Light", "Deep" };
|
|
|
|
Session* DreemLoader::readNextSession()
|
|
{
|
|
if (csv == nullptr) {
|
|
qWarning() << "no CSV open!";
|
|
return nullptr;
|
|
}
|
|
static QHash<const QString,int> s_sleepStages;
|
|
for (int i = 0; i < s_sleepStageLabels.size(); i++) {
|
|
const QString & label = s_sleepStageLabels[i];
|
|
s_sleepStages[label] = i+1; // match ZEO sleep stages for now
|
|
// TODO: generalize sleep stage integers between Dreem and Zeo
|
|
}
|
|
|
|
Session* sess = nullptr;
|
|
|
|
QDateTime start_time, stop_time;
|
|
int sleep_onset, sleep_duration;
|
|
int light_sleep_duration, deep_sleep_duration, rem_duration, awakened_duration;
|
|
int awakenings, position_changes, average_hr, average_rr;
|
|
float sleep_efficiency;
|
|
QStringList hypnogram;
|
|
|
|
QHash<QString,QString> row;
|
|
while (csv->readRow(row)) {
|
|
SessionID sid;
|
|
invalid_fields = false;
|
|
|
|
start_time = readDateTime(row["Start Time"]);
|
|
if (start_time.isValid()) {
|
|
sid = start_time.toTime_t();
|
|
if (mach->SessionExists(sid)) {
|
|
continue;
|
|
}
|
|
}
|
|
|
|
// "Type" always seems to be "night"
|
|
stop_time = readDateTime(row["Stop Time"]);
|
|
sleep_onset = readDuration(row["Sleep Onset Duration"]);
|
|
sleep_duration = readDuration(row["Sleep Duration"]);
|
|
light_sleep_duration = readDuration(row["Light Sleep Duration"]);
|
|
deep_sleep_duration = readDuration(row["Deep Sleep Duration"]);
|
|
rem_duration = readDuration(row["REM Duration"]);
|
|
awakened_duration = readDuration(row["Wake After Sleep Onset Duration"]);
|
|
awakenings = readInt(row["Number of awakenings"]);
|
|
position_changes = readInt(row["Position Changes"]);
|
|
average_hr = readInt(row["Mean Heart Rate"]);
|
|
average_rr = readInt(row["Mean Respiration CPM"]);
|
|
// "Number of Stimulations" is 0 for US models
|
|
sleep_efficiency = readInt(row["Sleep efficiency"]) / 100.0;
|
|
|
|
if (invalid_fields) {
|
|
continue;
|
|
}
|
|
|
|
QString h = row["Hypnogram"]; // with "[" at the beginning and "]" at the end
|
|
hypnogram = h.mid(1, h.length()-2).split(",");
|
|
if (hypnogram.size() == 0) {
|
|
continue;
|
|
}
|
|
|
|
sess = new Session(mach, sid);
|
|
break;
|
|
};
|
|
|
|
if (sess) {
|
|
const quint64 step = 30 * 1000;
|
|
|
|
// TODO: rename Zeo channels to be generic
|
|
sess->settings[ZEO_Awakenings] = awakenings;
|
|
sess->settings[ZEO_TimeToZ] = sleep_onset / 60; // TODO: convert durations to seconds and update Zeo loader accordingly, also below
|
|
sess->settings[ZEO_ZQ] = int(sleep_efficiency * 100.0); // TODO: ZQ may be better expressed as a percent?
|
|
sess->settings[ZEO_TimeInWake] = awakened_duration / 60;
|
|
sess->settings[ZEO_TimeInREM] = rem_duration / 60;
|
|
sess->settings[ZEO_TimeInLight] = light_sleep_duration / 60;
|
|
sess->settings[ZEO_TimeInDeep] = deep_sleep_duration / 60;
|
|
sess->settings[OXI_Pulse] = average_hr;
|
|
sess->settings[CPAP_RespRate] = average_rr;
|
|
// Dreem also provides:
|
|
// total sleep duration
|
|
// # position changes
|
|
|
|
qint64 st = qint64(start_time.toTime_t()) * 1000L;
|
|
qint64 last = qint64(stop_time.toTime_t()) * 1000L;
|
|
sess->really_set_first(st);
|
|
|
|
// It appears that the first sample occurs at start time and
|
|
// the second sample occurs at the next 30-second boundary.
|
|
//
|
|
// TODO: About half the time there are still too many samples?
|
|
qint64 tt = st;
|
|
qint64 second_sample_tt = ((tt + step - 1L) / step) * step;
|
|
|
|
EventList *sleepstage = sess->AddEventList(ZEO_SleepStage, EVL_Event, 1, 0, 0, 4);
|
|
|
|
for (int i = 0; i < hypnogram.size(); i++) {
|
|
auto & label = hypnogram.at(i);
|
|
if (s_sleepStages.contains(label)) {
|
|
int stage = s_sleepStages[label];
|
|
|
|
// It appears that the last sample occurs at the stop time.
|
|
if (tt > last) {
|
|
if (i != hypnogram.size() - 1) {
|
|
qWarning() << sess->session() << "more hypnogram samples than time" << tt << last;
|
|
}
|
|
tt = last;
|
|
}
|
|
|
|
sleepstage->AddEvent(tt, stage);
|
|
}
|
|
if (i == 0) {
|
|
tt = second_sample_tt;
|
|
} else {
|
|
tt += step;
|
|
}
|
|
}
|
|
|
|
sess->really_set_last(last);
|
|
}
|
|
|
|
return sess;
|
|
}
|
|
|
|
QDateTime DreemLoader::readDateTime(const QString & text)
|
|
{
|
|
QDateTime dt = QDateTime::fromString(text, Qt::ISODate);
|
|
if (!dt.isValid()) invalid_fields = true;
|
|
return dt;
|
|
}
|
|
|
|
int DreemLoader::readDuration(const QString & text)
|
|
{
|
|
QTime t = QTime::fromString(text, "H:mm:ss");
|
|
if (!t.isValid()) invalid_fields = true;
|
|
return t.msecsSinceStartOfDay() / 1000L;
|
|
}
|
|
|
|
int DreemLoader::readInt(const QString & text)
|
|
{
|
|
bool ok;
|
|
int value = text.toInt(&ok);
|
|
if (!ok) invalid_fields = true;
|
|
return value;
|
|
}
|
|
|
|
static bool dreem_initialized = false;
|
|
|
|
void DreemLoader::Register()
|
|
{
|
|
if (dreem_initialized) { return; }
|
|
|
|
qDebug("Registering DreemLoader");
|
|
RegisterLoader(new DreemLoader());
|
|
dreem_initialized = true;
|
|
}
|
|
|