/* SleepLib Machine Class Implementation * * Copyright (c) 2011-2014 Mark Watkins * * 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 Linux * distribution for more details. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "mainwindow.h" #include "progressdialog.h" #include #include "machine.h" #include "profiles.h" #include #include "SleepLib/schema.h" #include "SleepLib/day.h" extern QProgressBar *qprogress; ////////////////////////////////////////////////////////////////////////////////////////// // Machine Base-Class implmementation ////////////////////////////////////////////////////////////////////////////////////////// Machine::Machine(MachineID id) { day.clear(); highest_sessionid = 0; if (!id) { srand(time(nullptr)); MachineID temp; do { temp = rand(); } while (p_profile->machlist.find(temp) != p_profile->machlist.end()); m_id = temp; } else { m_id = id; } m_loader = nullptr; // qDebug() << "Create Machine: " << hex << m_id; //%lx",m_id); m_type = MT_UNKNOWN; firstsession = true; } Machine::~Machine() { saveSessionInfo(); qDebug() << "Destroy Machine" << info.loadername << hex << m_id; } Session *Machine::SessionExists(SessionID session) { if (sessionlist.find(session) != sessionlist.end()) { return sessionlist[session]; } else { return nullptr; } } const quint16 sessinfo_version = 1; bool Machine::saveSessionInfo() { QFile file(getDataPath() + "Sessions.info"); file.open(QFile::WriteOnly); QDataStream out(&file); out.setByteOrder(QDataStream::LittleEndian); out.setVersion(QDataStream::Qt_5_0); out << magic; out << filetype_sessenabled; out << sessinfo_version; out << m_availableChannels; QHash::iterator s; out << (int)sessionlist.size(); for (s = sessionlist.begin(); s != sessionlist.end(); ++s) { Session * sess = s.value(); out << (quint32) sess->session(); out << (bool)(sess->enabled()); } return true; } bool Machine::loadSessionInfo() { QHash::iterator s; QFile file(getDataPath() + "Sessions.info"); if (!file.open(QFile::ReadOnly)) { for (s = sessionlist.begin(); s!= sessionlist.end(); ++s) { Session * sess = s.value(); QHash::iterator it = sess->settings.find(SESSION_ENABLED); bool b = true; if (it != sess->settings.end()) { b = it.value().toBool(); } sess->setEnabled(b); // Extract from session settings and save.. } saveSessionInfo(); return true; } QDataStream in(&file); in.setByteOrder(QDataStream::LittleEndian); in.setVersion(QDataStream::Qt_5_0); quint32 mag32; in >> mag32; quint16 ft16, version; in >> ft16; in >> version; if (version >= 1) { in >> m_availableChannels; } int size; in >> size; quint32 sid; bool b; for (int i=0; i< size; ++i) { in >> sid; in >> b; s = sessionlist.find(sid); if (s != sessionlist.end()) { Session * sess = s.value(); sess->setEnabled(b); } } return true; } // Find date this session belongs in QDate Machine::pickDate(qint64 first) { QTime split_time = p_profile->session->daySplitTime(); int combine_sessions = p_profile->session->combineCloseSessions(); QDateTime d2 = QDateTime::fromTime_t(first / 1000); QDate date = d2.date(); QTime time = d2.time(); int closest_session = 0; if (time < split_time) { date = date.addDays(-1); } else if (combine_sessions > 0) { QMap::iterator dit = day.find(date.addDays(-1)); // Check Day Before if (dit != day.end()) { QDateTime lt = QDateTime::fromTime_t(dit.value()->last() / 1000L); closest_session = lt.secsTo(d2) / 60; if (closest_session < combine_sessions) { date = date.addDays(-1); } } } return date; } bool Machine::AddSession(Session *s) { Q_ASSERT(s != nullptr); Q_ASSERT(p_profile); Q_ASSERT(p_profile->isOpen()); if (p_profile->session->ignoreOlderSessions()) { qint64 ignorebefore = p_profile->session->ignoreOlderSessionsDate().toMSecsSinceEpoch(); if (s->last() < ignorebefore) { skipped_sessions++; return false; } } if (s->session() > highest_sessionid) { highest_sessionid = s->session(); } QTime split_time; int combine_sessions; bool locksessions = p_profile->session->lockSummarySessions(); if (locksessions) { split_time = s->summaryOnly() ? QTime(12,0,0) : p_profile->session->daySplitTime(); combine_sessions = s->summaryOnly() ? 0 : p_profile->session->combineCloseSessions(); } else { split_time = p_profile->session->daySplitTime(); combine_sessions = p_profile->session->combineCloseSessions(); } int ignore_sessions = p_profile->session->ignoreShortSessions(); int session_length = s->last() - s->first(); session_length /= 60000; sessionlist[s->session()] = s; // To make sure it get's saved later even if it's not wanted. //int drift=p_profile->cpap->clockDrift(); QDateTime d2 = QDateTime::fromTime_t(s->first() / 1000); QDate date = d2.date(); QTime time = d2.time(); QMap::iterator dit, nextday; bool combine_next_day = false; int closest_session = 0; // Multithreaded import screws this up. :( if (time < split_time) { date = date.addDays(-1); } else if (combine_sessions > 0) { dit = day.find(date.addDays(-1)); // Check Day Before if (dit != day.end()) { QDateTime lt = QDateTime::fromTime_t(dit.value()->last() / 1000); closest_session = lt.secsTo(d2) / 60; if (closest_session < combine_sessions) { date = date.addDays(-1); } else { if ((split_time < time) && (split_time.secsTo(time) < 2)) { if (s->machine()->loaderName() == STR_MACH_ResMed) { date = date.addDays(-1); } } } } else { nextday = day.find(date.addDays(1)); // Check Day Afterwards if (nextday != day.end()) { QDateTime lt = QDateTime::fromTime_t(nextday.value()->first() / 1000); closest_session = d2.secsTo(lt) / 60; if (closest_session < combine_sessions) { // add todays here. pull all tomorrows records to this date. combine_next_day = true; } } } } if (session_length < ignore_sessions) { // keep the session to save importing it again, but don't add it to the day record this time return true; } if (!firstsession) { if (firstday > date) { firstday = date; } if (lastday < date) { lastday = date; } } else { firstday = lastday = date; firstsession = false; } Day *dd = nullptr; dit = day.find(date); if (dit == day.end()) { dit = day.insert(date, p_profile->addDay(date)); } dd = dit.value(); dd->addSession(s); if (combine_next_day) { for (QList::iterator i = nextday.value()->begin(); i != nextday.value()->end(); i++) { // i may need to do something here if (locksessions && (*i)->summaryOnly()) continue; // can't move summary only sessions.. unlinkSession(*i); // Add it back sessionlist[(*i)->session()] = *i; dd->addSession(*i); } // QMap >::iterator nd = p_profile->daylist.find(date.addDays(1)); // if (nd != p_profile->daylist.end()) { // p_profile->unlinkDay(nd.key(), nd.value()); // } // QList::iterator iend = nd.value().end(); // for (QList::iterator i = nd.value()->begin(); i != iend; ++i) { // if (*i == nextday.value()) { // nd.value().erase(i); // } // } // day.erase(nextday); } return true; } bool Machine::unlinkDay(Day * d) { return day.remove(day.key(d)) > 0; } bool Machine::unlinkSession(Session * sess) { // Remove the object from the machine object's session list bool b=sessionlist.remove(sess->session()); QList dates; QList days; QMap::iterator it; Day * d; // Doing this in case of accidental double linkages for (it = day.begin(); it != day.end(); ++it) { d = it.value(); if (it.value()->sessions.contains(sess)) { days.push_back(d); dates.push_back(it.key()); } } for (int i=0; i < days.size(); ++i) { d = days.at(i); if (d->sessions.removeAll(sess)) { b=true; if (d->size() == 0) { day.remove(dates[i]); p_profile->unlinkDay(d); } } } return b; } // This functions purpose is murder and mayhem... It deletes all of a machines data. bool Machine::Purge(int secret) { // Boring api key to stop this function getting called by accident :) if (secret != 3478216) { return false; } QString path = getDataPath(); QDir dir(path); if (!dir.exists()) { // It doesn't exist anyway. return true; } if (!dir.isReadable()) { return false; } qDebug() << "Purging" << info.loadername << info.serial << dir.absoluteFilePath(path); // Remove any imported file list QFile impfile(getDataPath()+"/imported_files.csv"); impfile.remove(); QFile rxcache(p_profile->Get("{" + STR_GEN_DataFolder + "}/RXChanges.cache" )); rxcache.remove(); QFile sumfile(getDataPath()+"Summaries.xml"); sumfile.remove(); // Create a copy of the list so the hash can be manipulated QList sessions = sessionlist.values(); // Clean up any loaded sessions from memory first.. bool success = true; for (int i=0; i < sessions.size(); ++i) { Session * sess = sessions[i]; if (!sess->Destroy()) { qDebug() << "Could not destroy "+ info.loadername +" ("+info.serial+") session" << sess->session(); success = false; } else { // sessionlist.remove(sess->session()); } delete sess; } // Remove EVERYTHING under Events folder.. QString eventspath = getEventsPath(); QDir evdir(eventspath); evdir.removeRecursively(); QString summariespath = getSummariesPath(); QDir sumdir(summariespath); sumdir.removeRecursively(); // Clean up any straggling files (like from short sessions not being loaded...) dir.setFilter(QDir::Files | QDir::Hidden | QDir::NoSymLinks); dir.setSorting(QDir::Name); QFileInfoList list = dir.entryInfoList(); int could_not_kill = 0; int size = list.size(); for (int i = 0; i < size; ++i) { QFileInfo fi = list.at(i); QString fullpath = fi.canonicalFilePath(); QString ext_s = fullpath.section('.', -1); bool ok; ext_s.toInt(&ok, 10); if (ok) { qDebug() << "Deleting " << QDir::toNativeSeparators(fullpath); if (!dir.remove(fullpath)) { qDebug() << "Could not purge file" << fullpath; success=false; could_not_kill++; } } else { qDebug() << "Didn't bother deleting cruft file" << fullpath; // cruft file.. } } if (could_not_kill > 0) { qWarning() << "Could not purge path" << could_not_kill << "files in " << path; return false; } return true; } void Machine::setLoaderName(QString value) { info.loadername = value; m_loader = GetLoader(value); } void Machine::setInfo(MachineInfo inf) { info = inf; m_loader = GetLoader(inf.loadername); } //const quint32 channel_version=1; const QString Machine::getDataPath() { return p_profile->Get("{" + STR_GEN_DataFolder + "}/" + info.loadername + "_" + (info.serial.isEmpty() ? hexid() : info.serial)) + "/"; } const QString Machine::getSummariesPath() { return getDataPath() + "Summaries/"; } const QString Machine::getEventsPath() { return getDataPath() + "Events/"; } const QString Machine::getBackupPath() { return p_profile->Get("{" + STR_GEN_DataFolder + "}/" + info.loadername + "_" + (info.serial.isEmpty() ? hexid() : info.serial) + "/Backup/"); } bool Machine::Load() { QString path = getDataPath(); QDir dir(path); qDebug() << "Loading Database" << QDir::toNativeSeparators(path); if (!dir.exists() || !dir.isReadable()) { return false; } ProgressDialog * popup = new ProgressDialog(nullptr); QPixmap image(getCPAPPixmap(info.loadername)); if (!image.isNull()) { image = image.scaled(64,64); popup->setPixmap(image); } popup->setMessage(QObject::tr("Loading %1 data...").arg(info.brand)); popup->show(); QProgressBar * progress = popup->progress; if (!LoadSummary()) { // No XML index file, so assume upgrading, or it simply just got screwed up or deleted... QTime time; time.start(); dir.setFilter(QDir::Files | QDir::Hidden | QDir::NoSymLinks); /////////////////////////////////////////////////////////////////////// // First move any old files to correct locations /////////////////////////////////////////////////////////////////////// QString summarypath = getSummariesPath(); QString eventpath = getEventsPath(); if (!dir.exists(summarypath)) dir.mkpath(summarypath); QStringList filters; filters << "*.000"; dir.setNameFilters(filters); QStringList filelist = dir.entryList(); int size = filelist.size(); if (progress) { progress->setMinimum(0); progress->setMaximum(0); progress->setValue(0); QApplication::processEvents(); } for (int i=0; i < size; i++) { QString filename = filelist.at(i); QFile::copy(path+filename, summarypath+filename); QFile::remove(path+filename); } // Copy old Event files to folder filters.clear(); filters << "*.001"; dir.setNameFilters(filters); filelist = dir.entryList(); size = filelist.size(); if (size > 0) { if (!dir.exists(eventpath)) dir.mkpath(eventpath); for (int i=0; i< filelist.size(); i++) { if ((i % 50) == 0) { // This is slow.. :-/ if (progress) { progress->setValue((float(i) / float(size) * 100.0)); } QApplication::processEvents(); } QString filename = filelist.at(i); QFile::copy(path+filename, eventpath+filename); QFile::remove(path+filename); } } /////////////////////////////////////////////////////////////////////// // Now read summary files from correct location and load them /////////////////////////////////////////////////////////////////////// dir.setPath(summarypath); filters.clear(); filters << "*.000"; dir.setNameFilters(filters); filelist = dir.entryList(); size = filelist.size(); if (progress) { progress->setMinimum(0); progress->setMaximum(size); progress->setValue(0); QApplication::processEvents(); } QString sesstr; SessionID sessid; bool ok; for (int i=0; i < size; i++) { if ((i % 50) == 0) { // This is slow.. :-/ if (progress) { progress->setValue(i); } QApplication::processEvents(); } QString filename = filelist.at(i); sesstr = filename.section(".", 0, -2); sessid = sesstr.toLong(&ok, 16); if (!ok) { continue; } Session *sess = new Session(this, sessid); // Forced to load it, because know nothing about this session.. if (sess->LoadSummary()) { AddSession(sess); } else { qWarning() << "Error loading summary file" << filename; delete sess; } } SaveSummary(); qDebug() << "Loaded" << info.model << "data in" << time.elapsed() << "ms"; if (progress) { progress->setValue(size); } } else { if (progress) { progress->setValue(100); } } loadSessionInfo(); QApplication::processEvents(); popup->hide(); delete popup; return true; } bool Machine::SaveSession(Session *sess) { QString path = getDataPath(); if (sess->IsChanged()) { sess->Store(path); } return true; } void Machine::queSaveList(Session * sess) { if (!m_save_threads_running) { // Threads aren't being used.. so run the actual immediately... int i = (float(m_donetasks) / float(m_totaltasks) * 100.0); qprogress->setValue(i); QApplication::processEvents(); sess->UpdateSummaries(); sess->Store(getDataPath()); if (!p_profile->session->cacheSessions()) { sess->TrashEvents(); } } else { listMutex.lock(); m_savelist.append(sess); listMutex.unlock(); } } Session *Machine::popSaveList() { Session *sess = nullptr; listMutex.lock(); if (!m_savelist.isEmpty()) { sess = m_savelist.at(0); m_savelist.pop_front(); m_donetasks++; } listMutex.unlock(); return sess; } // Call any time queing starts void Machine::StartSaveThreads() { m_savelist.clear(); if (!p_profile->session->multithreading()) return; QString path = getDataPath(); int threads = QThread::idealThreadCount(); savelistSem = new QSemaphore(threads); savelistSem->acquire(threads); m_save_threads_running = true; m_donetasks=0; m_totaltasks=0; for (int i = 0; i < threads; i++) { SaveThread * thr = new SaveThread(this, path); QObject::connect(thr, SIGNAL(UpdateProgress(int)), qprogress, SLOT(setValue(int))); thread.push_back(thr); thread[i]->start(); } } // Call when all queing is completed void Machine::FinishSaveThreads() { if (!m_save_threads_running) return; m_save_threads_running = false; // Wait for all tasks to finish while (!savelistSem->tryAcquire(thread.size(), 250)) { if (qprogress) { QApplication::processEvents(); } } for (int i = 0; i < thread.size(); ++i) { while (thread[i]->isRunning()) { SaveThread::msleep(250); QApplication::processEvents(); } QObject::disconnect(thread[i], SIGNAL(UpdateProgress(int)), qprogress, SLOT(setValue(int))); delete thread[i]; } delete savelistSem; } void SaveThread::run() { bool running = true; while (running) { Session *sess = machine->popSaveList(); if (sess) { if (machine->m_donetasks % 10 == 0) { int i = (float(machine->m_donetasks) / float(machine->m_totaltasks) * 100.0); emit UpdateProgress(i); } sess->UpdateSummaries(); machine->saveMutex.lock(); sess->Store(path); machine->saveMutex.unlock(); sess->TrashEvents(); } else { if (!machine->m_save_threads_running) { break; // done } else { yieldCurrentThread(); // go do something else for a while } } } machine->savelistSem->release(1); } class SaveTask:public ImportTask { public: SaveTask(Session * s, Machine * m): sess(s), mach(m) {} virtual ~SaveTask() {} virtual void run(); protected: Session * sess; Machine * mach; }; void SaveTask::run() { sess->UpdateSummaries(); mach->saveMutex.lock(); sess->Store(mach->getDataPath()); mach->saveMutex.unlock(); sess->TrashEvents(); } void Machine::queTask(ImportTask * task) { if (0) { //p_profile->session->multithreading()) { m_tasklist.push_back(task); return; } task->run(); return; } void Machine::runTasks() { if (0) { //!p_profile->session->multithreading()) { Q_ASSERT(m_tasklist.isEmpty()); return; } QThreadPool * threadpool = QThreadPool::globalInstance(); int m_totaltasks=m_tasklist.size(); int m_currenttask=0; while (!m_tasklist.isEmpty()) { if (threadpool->tryStart(m_tasklist.at(0))) { m_tasklist.pop_front(); float f = float(m_currenttask) / float(m_totaltasks) * 100.0; qprogress->setValue(f); m_currenttask++; } QApplication::processEvents(); } QThreadPool::globalInstance()->waitForDone(-1); } bool Machine::hasModifiedSessions() { QHash::iterator s; for (s = sessionlist.begin(); s != sessionlist.end(); s++) { if (s.value()->IsChanged()) { return true; } } return false; } const QString summaryFileName = "Summaries.xml"; bool Machine::LoadSummary(bool everything) { QTime time; time.start(); qDebug() << "Loading Summaries"; QString filename = getDataPath() + summaryFileName; QDomDocument doc; QFile file(filename); qDebug() << "Opening " << filename; if (!file.open(QIODevice::ReadOnly)) { qWarning() << "Could not open" << filename; return false; } QByteArray data=file.readAll(); QByteArray uncompressed = qUncompress(data); QString errorMsg; int errorLine; if (!doc.setContent(uncompressed, false, &errorMsg, &errorLine)) { qWarning() << "Invalid XML Content in" << filename; qWarning() << "Error line" << errorLine << ":" << errorMsg; return false; } file.close(); QDomElement root = doc.documentElement(); QDomNode node; bool s_ok; QDomNodeList sessionlist = root.childNodes(); int size = sessionlist.size(); for (int s=0; s < size; ++s) { node = sessionlist.at(s); QDomElement e = node.toElement(); SessionID sessid = e.attribute("id", "0").toLong(&s_ok); qint64 first = e.attribute("first", "0").toLongLong(); qint64 last = e.attribute("last", "0").toLongLong(); bool enabled = e.attribute("enabled", "1").toInt() == 1; bool events = e.attribute("events", "1").toInt() == 1; if (s_ok) { Session * sess = new Session(this, sessid); sess->really_set_first(first); sess->really_set_last(last); sess->setEnabled(enabled); sess->setSummaryOnly(!events); if (!AddSession(sess)) delete sess; // sess->LoadSummary(); } } qDebug() << "Loaded" << info.series << info.model << "data in" << time.elapsed() << "ms"; return true; } const int summaryxml_version=0; bool Machine::SaveSummary() { qDebug() << "Saving Summaries"; QString filename = getDataPath() + summaryFileName; QDomDocument doc("SleepyHeadSessionIndex"); QDomElement root = doc.createElement("sessions"); root.setAttribute("version", summaryxml_version); root.setAttribute("profile", p_profile->user->userName()); root.setAttribute("count", sessionlist.size()); root.setAttribute("loader", info.loadername); root.setAttribute("serial", info.serial); doc.appendChild(root); if (!QDir().exists(getSummariesPath())) QDir().mkpath(getSummariesPath()); QHash::iterator s; QHash::iterator sess_end = sessionlist.end(); for (s = sessionlist.begin(); s != sess_end; ++s) { QDomElement el = doc.createElement("session"); Session * sess = s.value(); el.setAttribute("id", (quint32)sess->session()); el.setAttribute("first", sess->realFirst()); el.setAttribute("last", sess->realLast()); el.setAttribute("enabled", sess->enabled() ? "1" : "0"); el.setAttribute("events", sess->summaryOnly() ? "0" : "1"); root.appendChild(el); if (sess->IsChanged()) sess->StoreSummary(); } QFile file(filename); file.open(QIODevice::WriteOnly); QByteArray ba = qCompress(doc.toByteArray()); file.write(ba); file.close(); return true; } bool Machine::Save() { //int size; int cnt = 0; QString path = getDataPath(); QDir dir(path); if (!dir.exists()) { dir.mkdir(path); } QHash::iterator s; m_savelist.clear(); // store any event summaries.. for (s = sessionlist.begin(); s != sessionlist.end(); s++) { cnt++; if ((*s)->IsChanged()) { queTask(new SaveTask(*s, this)); } } runTasks(); return true; } void Machine::updateChannels(Session * sess) { int size = sess->m_availableChannels.size(); for (int i=0; i < size; ++i) { ChannelID code = sess->m_availableChannels.at(i); m_availableChannels[code] = true; } } QList Machine::availableChannels(quint32 chantype) { QList list; QHash::iterator end = m_availableChannels.end(); QHash::iterator it; for (it = m_availableChannels.begin(); it != end; ++it) { ChannelID code = it.key(); const schema::Channel & chan = schema::channel[code]; if (chan.type() & chantype) { list.push_back(code); } } return list; } ////////////////////////////////////////////////////////////////////////////////////////// // CPAP implmementation ////////////////////////////////////////////////////////////////////////////////////////// CPAP::CPAP(MachineID id): Machine(id) { m_type = MT_CPAP; } CPAP::~CPAP() { } ////////////////////////////////////////////////////////////////////////////////////////// // Oximeter Class implmementation ////////////////////////////////////////////////////////////////////////////////////////// Oximeter::Oximeter(MachineID id): Machine(id) { m_type = MT_OXIMETER; } Oximeter::~Oximeter() { } ////////////////////////////////////////////////////////////////////////////////////////// // SleepStage Class implmementation ////////////////////////////////////////////////////////////////////////////////////////// SleepStage::SleepStage(MachineID id): Machine(id) { m_type = MT_SLEEPSTAGE; } SleepStage::~SleepStage() { } ////////////////////////////////////////////////////////////////////////////////////////// // PositionSensor Class implmementation ////////////////////////////////////////////////////////////////////////////////////////// PositionSensor::PositionSensor(MachineID id): Machine(id) { m_type = MT_POSITION; } PositionSensor::~PositionSensor() { }