2020-01-29 20:47:11 +00:00
|
|
|
/* SleepLib ZEO Loader Implementation
|
2014-04-09 21:01:57 +00:00
|
|
|
*
|
2020-01-29 20:47:11 +00:00
|
|
|
* Copyright (c) 2020 The OSCAR Team
|
2018-03-28 07:10:52 +00:00
|
|
|
* Copyright (c) 2011-2018 Mark Watkins <mark@jedimark.net>
|
2014-04-09 21:01:57 +00:00
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU General Public
|
2018-06-04 20:48:38 +00:00
|
|
|
* License. See the file COPYING in the main directory of the source code
|
|
|
|
* for more details. */
|
2011-06-26 08:30:44 +00:00
|
|
|
|
|
|
|
//********************************************************************************************
|
2011-12-19 08:17:19 +00:00
|
|
|
// IMPORTANT!!!
|
2011-06-26 08:30:44 +00:00
|
|
|
//********************************************************************************************
|
|
|
|
// Please INCREMENT the zeo_data_version in zel_loader.h when making changes to this loader
|
|
|
|
// that change loader behaviour or modify channels.
|
|
|
|
//********************************************************************************************
|
|
|
|
|
2011-11-27 06:25:27 +00:00
|
|
|
#include <QDir>
|
2012-01-11 13:55:46 +00:00
|
|
|
#include <QTextStream>
|
2011-06-26 08:30:44 +00:00
|
|
|
#include "zeo_loader.h"
|
|
|
|
#include "SleepLib/machine.h"
|
2020-01-29 22:05:03 +00:00
|
|
|
#include "csv.h"
|
2011-06-26 08:30:44 +00:00
|
|
|
|
|
|
|
ZEOLoader::ZEOLoader()
|
|
|
|
{
|
2014-05-25 07:07:08 +00:00
|
|
|
m_type = MT_SLEEPSTAGE;
|
2011-06-26 08:30:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ZEOLoader::~ZEOLoader()
|
|
|
|
{
|
2020-01-29 22:05:03 +00:00
|
|
|
closeCSV();
|
2011-06-26 08:30:44 +00:00
|
|
|
}
|
2014-04-23 13:19:56 +00:00
|
|
|
|
2018-04-27 04:29:03 +00:00
|
|
|
int ZEOLoader::Open(const QString & dirpath)
|
2011-06-26 08:30:44 +00:00
|
|
|
{
|
2011-11-27 06:25:27 +00:00
|
|
|
QString newpath;
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
QString dirtag = "zeo";
|
2012-01-11 13:55:46 +00:00
|
|
|
|
|
|
|
// Could Scan the ZEO folder for a list of CSVs
|
|
|
|
|
2018-04-27 04:29:03 +00:00
|
|
|
QString path(dirpath);
|
2014-04-17 05:58:57 +00:00
|
|
|
path = path.replace("\\", "/");
|
2013-09-14 23:32:14 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
if (path.toLower().endsWith("/" + dirtag)) {
|
2011-11-27 06:25:27 +00:00
|
|
|
return 0;
|
|
|
|
//newpath=path;
|
|
|
|
} else {
|
2014-04-17 05:58:57 +00:00
|
|
|
newpath = path + "/" + dirtag.toUpper();
|
2011-11-27 06:25:27 +00:00
|
|
|
}
|
|
|
|
|
2013-09-14 23:32:14 +00:00
|
|
|
//QString filename;
|
2011-11-27 06:25:27 +00:00
|
|
|
|
2011-06-26 08:30:44 +00:00
|
|
|
// ZEO folder structure detection stuff here.
|
|
|
|
|
2011-07-15 13:30:41 +00:00
|
|
|
return 0; // number of machines affected
|
2011-06-26 08:30:44 +00:00
|
|
|
}
|
2014-04-23 13:19:56 +00:00
|
|
|
|
2012-01-11 13:55:46 +00:00
|
|
|
/*15233: "Sleep Date"
|
|
|
|
15234: "ZQ"
|
|
|
|
15236: "Total Z"
|
|
|
|
15237: "Time to Z"
|
|
|
|
15237: "Time in Wake"
|
|
|
|
15238: "Time in REM"
|
|
|
|
15238: "Time in Light"
|
|
|
|
15241: "Time in Deep"
|
|
|
|
15242: "Awakenings"
|
|
|
|
15245: "Start of Night"
|
|
|
|
15246: "End of Night"
|
|
|
|
15246: "Rise Time"
|
|
|
|
15247: "Alarm Reason"
|
|
|
|
15247: "Snooze Time"
|
|
|
|
15254: "Wake Tone"
|
|
|
|
15259: "Wake Window"
|
|
|
|
15259: "Alarm Type"
|
|
|
|
15260: "First Alarm Ring"
|
|
|
|
15261: "Last Alarm Ring"
|
|
|
|
15261: "First Snooze Time"
|
|
|
|
15265: "Last Snooze Time"
|
|
|
|
15266: "Set Alarm Time"
|
|
|
|
15266: "Morning Feel"
|
|
|
|
15267: "Sleep Graph"
|
|
|
|
15267: "Detailed Sleep Graph"
|
|
|
|
15268: "Firmware Version" */
|
|
|
|
|
2018-04-27 04:29:03 +00:00
|
|
|
int ZEOLoader::OpenFile(const QString & filename)
|
2012-01-11 13:55:46 +00:00
|
|
|
{
|
2020-01-29 20:47:11 +00:00
|
|
|
if (!openCSV(filename)) {
|
2020-01-29 21:21:42 +00:00
|
|
|
return -1;
|
2020-01-29 20:47:11 +00:00
|
|
|
}
|
2020-01-29 21:21:42 +00:00
|
|
|
int count = 0;
|
2020-01-29 20:47:11 +00:00
|
|
|
Session* sess;
|
|
|
|
while ((sess = readNextSession()) != nullptr) {
|
|
|
|
sess->SetChanged(true);
|
|
|
|
mach->AddSession(sess);
|
2020-01-29 21:21:42 +00:00
|
|
|
count++;
|
2020-01-29 20:47:11 +00:00
|
|
|
}
|
|
|
|
mach->Save();
|
2020-01-29 22:05:03 +00:00
|
|
|
closeCSV();
|
2020-01-29 21:21:42 +00:00
|
|
|
return count;
|
2020-01-29 20:47:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ZEOLoader::openCSV(const QString & filename)
|
|
|
|
{
|
|
|
|
file.setFileName(filename);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2012-01-11 13:55:46 +00:00
|
|
|
if (filename.toLower().endsWith(".csv")) {
|
|
|
|
if (!file.open(QFile::ReadOnly)) {
|
|
|
|
qDebug() << "Couldn't open zeo file" << filename;
|
2020-01-29 21:21:42 +00:00
|
|
|
return false;
|
2012-01-11 13:55:46 +00:00
|
|
|
}
|
|
|
|
} else {// if (filename.toLower().endsWith(".dat")) {
|
|
|
|
|
2020-01-29 21:21:42 +00:00
|
|
|
return false;
|
2012-01-11 13:55:46 +00:00
|
|
|
// not supported.
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2020-01-29 22:05:03 +00:00
|
|
|
QStringList header;
|
|
|
|
csv = new CSVReader(file);
|
|
|
|
bool ok = csv->readRow(header);
|
|
|
|
if (!ok) {
|
|
|
|
qWarning() << "no header row";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
csv->setFieldNames(header);
|
2020-01-29 20:47:11 +00:00
|
|
|
|
|
|
|
MachineInfo info = newInfo();
|
|
|
|
mach = p_profile->CreateMachine(info);
|
|
|
|
|
2020-01-29 22:05:03 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ZEOLoader::closeCSV()
|
|
|
|
{
|
|
|
|
if (csv != nullptr) {
|
|
|
|
delete csv;
|
|
|
|
csv = nullptr;
|
|
|
|
}
|
|
|
|
if (file.isOpen()) {
|
|
|
|
file.close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// int idxTotalZ = header.indexOf("Total Z");
|
2020-01-29 20:47:11 +00:00
|
|
|
// int idxAlarmReason = header.indexOf("Alarm Reason");
|
|
|
|
// int idxSnoozeTime = header.indexOf("Snooze Time");
|
|
|
|
// int idxWakeTone = header.indexOf("Wake Tone");
|
|
|
|
// int idxWakeWindow = header.indexOf("Wake Window");
|
|
|
|
// int idxAlarmType = header.indexOf("Alarm Type");
|
|
|
|
|
|
|
|
Session* ZEOLoader::readNextSession()
|
|
|
|
{
|
2020-01-29 22:05:03 +00:00
|
|
|
if (csv == nullptr) {
|
|
|
|
qWarning() << "no CSV open!";
|
|
|
|
return nullptr;
|
|
|
|
}
|
2020-01-29 20:47:11 +00:00
|
|
|
Session* sess = nullptr;
|
2012-01-11 13:55:46 +00:00
|
|
|
QDateTime start_of_night, end_of_night, rise_time;
|
|
|
|
SessionID sid;
|
|
|
|
|
2013-01-17 18:26:11 +00:00
|
|
|
//const qint64 WindowSize=30000;
|
2014-04-17 05:58:57 +00:00
|
|
|
qint64 st, tt;
|
2012-01-11 13:55:46 +00:00
|
|
|
int stage;
|
|
|
|
|
2014-10-08 16:51:09 +00:00
|
|
|
int ZQ, TimeToZ, TimeInWake, TimeInREM, TimeInLight, TimeInDeep, Awakenings;
|
|
|
|
|
|
|
|
//int AlarmReason, SnoozeTime, WakeTone, WakeWindow, AlarmType, TotalZ;
|
|
|
|
int MorningFeel;
|
2012-01-13 01:16:46 +00:00
|
|
|
QString FirmwareVersion, MyZeoVersion;
|
2012-01-11 13:55:46 +00:00
|
|
|
|
|
|
|
QDateTime FirstAlarmRing, LastAlarmRing, FirstSnoozeTime, LastSnoozeTime, SetAlarmTime;
|
|
|
|
|
|
|
|
QStringList SG, DSG;
|
|
|
|
|
|
|
|
bool ok;
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2020-01-29 22:05:03 +00:00
|
|
|
QHash<QString,QString> row;
|
|
|
|
while (csv->readRow(row)) {
|
2020-01-30 17:20:50 +00:00
|
|
|
invalid_fields = false;
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2020-01-30 17:20:50 +00:00
|
|
|
start_of_night = readDateTime(row["Start of Night"]);
|
|
|
|
if (start_of_night.isValid()) {
|
|
|
|
sid = start_of_night.toTime_t();
|
|
|
|
if (mach->SessionExists(sid)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ZQ = readInt(row["ZQ"]);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2014-10-08 16:51:09 +00:00
|
|
|
// TotalZ = linecomp[idxTotalZ].toInt(&ok);
|
|
|
|
// if (!ok) { dodgy = true; }
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2020-01-30 17:20:50 +00:00
|
|
|
TimeToZ = readInt(row["Time to Z"]);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2020-01-30 17:20:50 +00:00
|
|
|
TimeInWake = readInt(row["Time in Wake"]);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2020-01-30 17:20:50 +00:00
|
|
|
TimeInREM = readInt(row["Time in REM"]);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2020-01-30 17:20:50 +00:00
|
|
|
TimeInLight = readInt(row["Time in Light"]);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2020-01-30 17:20:50 +00:00
|
|
|
TimeInDeep = readInt(row["Time in Deep"]);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2020-01-30 17:20:50 +00:00
|
|
|
Awakenings = readInt(row["Awakenings"]);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2020-01-29 22:05:03 +00:00
|
|
|
end_of_night = readDateTime(row["End of Night"]);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2020-01-29 22:05:03 +00:00
|
|
|
rise_time = readDateTime(row["Rise Time"]);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2014-10-08 16:51:09 +00:00
|
|
|
// AlarmReason = linecomp[idxAlarmReason].toInt(&ok);
|
|
|
|
// SnoozeTime = linecomp[idxSnoozeTime].toInt(&ok);
|
|
|
|
// WakeTone = linecomp[idxWakeTone].toInt(&ok);
|
|
|
|
// WakeWindow = linecomp[idxWakeWindow].toInt(&ok);
|
|
|
|
// AlarmType = linecomp[idxAlarmType].toInt(&ok);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2020-01-30 17:20:50 +00:00
|
|
|
FirstAlarmRing = readDateTime(row["First Alarm Ring"], false);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2020-01-30 17:20:50 +00:00
|
|
|
LastAlarmRing = readDateTime(row["Last Alarm Ring"], false);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2020-01-30 17:20:50 +00:00
|
|
|
FirstSnoozeTime = readDateTime(row["First Snooze Time"], false);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2020-01-30 17:20:50 +00:00
|
|
|
LastSnoozeTime = readDateTime(row["Last Snooze Time"], false);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2020-01-30 17:20:50 +00:00
|
|
|
SetAlarmTime = readDateTime(row["Set Alarm Time"], false);
|
2012-05-19 07:03:49 +00:00
|
|
|
|
2020-01-30 17:20:50 +00:00
|
|
|
MorningFeel = readInt(row["Morning Feel"], false);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2020-01-29 22:05:03 +00:00
|
|
|
FirmwareVersion = row["Firmware Version"];
|
2012-01-11 13:55:46 +00:00
|
|
|
|
2020-01-29 22:05:03 +00:00
|
|
|
MyZeoVersion = row["My ZEO Version"];
|
2012-01-11 13:55:46 +00:00
|
|
|
|
2020-01-30 17:20:50 +00:00
|
|
|
if (invalid_fields) {
|
2012-01-11 13:55:46 +00:00
|
|
|
continue;
|
2014-04-17 05:58:57 +00:00
|
|
|
}
|
|
|
|
|
2020-01-29 22:05:03 +00:00
|
|
|
SG = row["Sleep Graph"].split(" ");
|
|
|
|
DSG = row["Detailed Sleep Graph"].split(" ");
|
2012-01-11 13:55:46 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
if (DSG.size() == 0) {
|
2012-01-11 13:55:46 +00:00
|
|
|
continue;
|
2014-04-17 05:58:57 +00:00
|
|
|
}
|
|
|
|
|
2020-01-29 20:47:11 +00:00
|
|
|
sess = new Session(mach, sid);
|
|
|
|
break;
|
2020-01-29 22:05:03 +00:00
|
|
|
};
|
2020-01-29 20:47:11 +00:00
|
|
|
|
|
|
|
if (sess) {
|
2020-01-29 22:05:03 +00:00
|
|
|
const int WindowSize = 30 * 1000;
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
sess->settings[ZEO_Awakenings] = Awakenings;
|
|
|
|
sess->settings[ZEO_MorningFeel] = MorningFeel;
|
|
|
|
sess->settings[ZEO_TimeToZ] = TimeToZ;
|
|
|
|
sess->settings[ZEO_ZQ] = ZQ;
|
|
|
|
sess->settings[ZEO_TimeInWake] = TimeInWake;
|
|
|
|
sess->settings[ZEO_TimeInREM] = TimeInREM;
|
|
|
|
sess->settings[ZEO_TimeInLight] = TimeInLight;
|
|
|
|
sess->settings[ZEO_TimeInDeep] = TimeInDeep;
|
|
|
|
|
|
|
|
st = qint64(start_of_night.toTime_t()) * 1000L;
|
2012-01-11 13:55:46 +00:00
|
|
|
sess->really_set_first(st);
|
2014-04-17 05:58:57 +00:00
|
|
|
tt = st;
|
|
|
|
EventList *sleepstage = sess->AddEventList(ZEO_SleepStage, EVL_Event, 1, 0, 0, 4);
|
|
|
|
|
|
|
|
for (int i = 0; i < DSG.size(); i++) {
|
|
|
|
stage = DSG[i].toInt(&ok);
|
2012-01-11 13:55:46 +00:00
|
|
|
if (ok) {
|
2014-04-17 05:58:57 +00:00
|
|
|
sleepstage->AddEvent(tt, stage);
|
2012-01-11 13:55:46 +00:00
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
tt += WindowSize;
|
2012-01-11 13:55:46 +00:00
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2012-01-11 13:55:46 +00:00
|
|
|
sess->really_set_last(tt);
|
2020-01-29 21:21:42 +00:00
|
|
|
//int size = DSG.size();
|
|
|
|
//qDebug() << linecomp[0] << start_of_night << end_of_night << rise_time << size << "30 second chunks";
|
2020-01-29 20:47:11 +00:00
|
|
|
}
|
2012-01-11 13:55:46 +00:00
|
|
|
|
2020-01-29 20:47:11 +00:00
|
|
|
return sess;
|
2012-01-11 13:55:46 +00:00
|
|
|
}
|
|
|
|
|
2020-01-30 17:20:50 +00:00
|
|
|
QDateTime ZEOLoader::readDateTime(const QString & text, bool required)
|
2020-01-29 21:21:42 +00:00
|
|
|
{
|
|
|
|
QDateTime dt = QDateTime::fromString(text, "MM/dd/yyyy HH:mm");
|
2020-01-30 17:20:50 +00:00
|
|
|
if (required || !text.isEmpty()) {
|
|
|
|
if (!dt.isValid()) {
|
|
|
|
dt = QDateTime::fromString(text, "yyyy-MM-dd HH:mm:ss");
|
|
|
|
if (!dt.isValid()) {
|
|
|
|
invalid_fields = true;
|
|
|
|
}
|
|
|
|
}
|
2020-01-29 21:21:42 +00:00
|
|
|
}
|
|
|
|
return dt;
|
|
|
|
}
|
|
|
|
|
2020-01-30 17:20:50 +00:00
|
|
|
int ZEOLoader::readInt(const QString & text, bool required)
|
|
|
|
{
|
|
|
|
bool ok;
|
|
|
|
int value = text.toInt(&ok);
|
|
|
|
if (!ok) {
|
|
|
|
if (required) {
|
|
|
|
invalid_fields = true;
|
|
|
|
} else {
|
|
|
|
value = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
static bool zeo_initialized = false;
|
2011-06-26 08:30:44 +00:00
|
|
|
|
|
|
|
void ZEOLoader::Register()
|
|
|
|
{
|
2014-04-17 05:58:57 +00:00
|
|
|
if (zeo_initialized) { return; }
|
|
|
|
|
2011-06-26 08:30:44 +00:00
|
|
|
qDebug("Registering ZEOLoader");
|
|
|
|
RegisterLoader(new ZEOLoader());
|
|
|
|
//InitModelMap();
|
2014-04-17 05:58:57 +00:00
|
|
|
zeo_initialized = true;
|
2011-06-26 08:30:44 +00:00
|
|
|
}
|
|
|
|
|