/* SleepLib Profiles Implementation * * Copyright (c) 2019-2024 The OSCAR Team * Copyright (c) 2011-2018 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 source code * for more details. */ #define TEST_MACROS_ENABLEDoff #include #include #include #include #include #include #include #include #include #include #include #include #include #include "preferences.h" #include "profiles.h" #include "machine.h" #include "machine_common.h" #include "machine_loader.h" #include "mainwindow.h" #include "translation.h" #include "version.h" extern MainWindow *mainwin; Preferences *p_pref; Preferences *p_layout; Profile *p_profile; Profile::Profile(QString path, bool open) : is_first_day(true), m_opened(false) { p_name = STR_GEN_Profile; if (path.isEmpty()) { p_path = GetAppData(); } else { p_path = path; } (*this)[STR_GEN_DataFolder] = p_path; path = path.replace("\\", "/"); if (!p_path.endsWith("/")) { p_path += "/"; } p_filename = p_path + p_name + STR_ext_XML; m_machlist.clear(); if (open) { Open(p_filename); } Set(STR_GEN_DataFolder, QString("{home}/Profiles/{UserName}")); // Reset import warnings when running a new version of OSCAR init(STR_PREF_VersionString, getVersion().toString()); Version prefVersion = Version((*this)[STR_PREF_VersionString].toString()); if (prefVersion != getVersion()) { qDebug() << " Resetting import warnings: version" << prefVersion << "to" << getVersion(); Set(STR_PREF_VersionString, getVersion().toString()); this->Erase(STR_IS_WarnOnUntestedMachine); this->Erase(STR_IS_WarnOnUnexpectedData); } doctor = new DoctorInfo(this); user = new UserInfo(this); cpap = new CPAPSettings(this); oxi = new OxiSettings(this); appearance = new AppearanceSettings(this); session = new SessionSettings(this); general = new UserSettings(this); if (open) { OpenMachines(); m_opened=true; } } Profile::~Profile() { if (m_opened) { removeLock(); } // delete device objects... for (auto & mach : m_machlist) { delete mach; } for (auto & day : daylist) { delete day; } delete user; delete doctor; delete cpap; delete oxi; delete appearance; delete session; delete general; } bool Profile::Save(QString filename) { if (m_opened) { return Preferences::Save(filename) && StoreMachines(); } else return false; } bool Profile::removeLock() { QString filename=p_path+"/lockfile"; QFile file(filename); return file.remove(); } QString Profile::checkLock() { QString filename=p_path+"/lockfile"; QFile file(filename); if (!file.exists()) return QString(); file.open(QFile::ReadOnly); QString lockhost = file.readLine(1024).trimmed(); return lockhost; } // Properties for machines.xml: const QString STR_PROP_Brand = "brand"; const QString STR_PROP_Model = "model"; const QString STR_PROP_Series = "series"; const QString STR_PROP_ModelNumber = "modelnumber"; const QString STR_PROP_SubModel = "submodel"; const QString STR_PROP_Serial = "serial"; const QString STR_PROP_DataVersion = "dataversion"; const QString STR_PROP_LastImported = "lastimported"; const QString STR_PROP_PurgeDate = "purgedate"; void Profile::addLock() { QFile lockfile(p_path+"lockfile"); lockfile.open(QFile::WriteOnly); QByteArray ba; ba.append(QHostInfo::localHostName().toUtf8()); lockfile.write(ba); lockfile.close(); } bool Profile::OpenMachines() { if (m_machlist.size() > 0) { qCritical() << "Skipping redundant call to Profile::OpenMachines"; return true; } QString filename = p_path+"machines.xml"; QFile file(filename); if (!file.open(QFile::ReadOnly)) { qWarning() << "Could not open" << filename.toLocal8Bit().data() << "for reading, error code" << file.error() << file.errorString(); return false; } // qDebug() << "OpenMachines opened" << filename.toLocal8Bit().data(); QDomDocument doc("machines.xml"); if (!doc.setContent(&file)) { qWarning() << "Invalid XML Content in" << filename.toLocal8Bit().data(); return false; } file.close(); QDomElement root = doc.firstChild().toElement(); if (root.tagName().toLower() != "machines") { qWarning() << "No Machines Tag in machines.xml"; return false; } QDomElement elem = root.firstChildElement(); while (!elem.isNull()) { QString pKey = elem.tagName(); if (pKey.toLower() != "machine") { qWarning() << "Profile::OpenMachines() pKey!=\"machine\""; elem = elem.nextSiblingElement(); continue; } int m_id; bool ok; m_id = elem.attribute("id", "").toInt(&ok); int mt; mt = elem.attribute("type", "").toInt(&ok); MachineType m_type = (MachineType)mt; QString m_class = elem.attribute("class", ""); MachineInfo info; info.type = m_type; info.loadername = m_class; QHash prop; QDomElement e = elem.firstChildElement(); for (; !e.isNull(); e = e.nextSiblingElement()) { QString pKey = e.tagName(); QString key = pKey.toLower(); if (key == STR_PROP_Brand) { info.brand = e.text(); } else if (key == STR_PROP_Model) { info.model = e.text(); } else if (key == STR_PROP_ModelNumber) { info.modelnumber = e.text(); } else if (key == STR_PROP_Serial) { info.serial = e.text(); } else if (key == STR_PROP_Series) { info.series = e.text(); } else if (key == STR_PROP_DataVersion) { info.version = e.text().toInt(); } else if (key == STR_PROP_LastImported) { info.lastimported = QDateTime::fromString(e.text(), Qt::ISODate); } else if (key == STR_PROP_PurgeDate) { info.purgeDate = QDate::fromString(e.text(), Qt::ISODate); } else if (key == "properties") { QDomElement pe = e.firstChildElement(); for (; !pe.isNull(); pe = pe.nextSiblingElement()) { prop[pe.tagName()] = pe.text(); } } else { // skip any old rubbish if ((key == "backuppath") || (key == "path") || (key == "submodel")) continue; prop[pKey] = e.text(); } } Machine *m = nullptr; // Create device needs a profile passed to it.. m = CreateMachine(info, m_id); if (m) m->info.properties = prop; elem = elem.nextSiblingElement(); } return true; } bool Profile::StoreMachines() { QDomDocument doc("Machines"); QDomElement mach = doc.createElement("machines"); for (int i=0; iid()); me.setAttribute("type", (int)m->type()); me.setAttribute("class", m->loaderName()); QDomElement pe = doc.createElement("properties"); me.appendChild(pe); for (QHash::iterator j = m->info.properties.begin(); j != m->info.properties.end(); j++) { QDomElement pp = doc.createElement(j.key()); pp.appendChild(doc.createTextNode(j.value())); pe.appendChild(pp); } QDomElement mp = doc.createElement(STR_PROP_Brand); mp.appendChild(doc.createTextNode(m->brand())); me.appendChild(mp); mp = doc.createElement(STR_PROP_Model); mp.appendChild(doc.createTextNode(m->model())); me.appendChild(mp); mp = doc.createElement(STR_PROP_ModelNumber); mp.appendChild(doc.createTextNode(m->modelnumber())); me.appendChild(mp); mp = doc.createElement(STR_PROP_Serial); mp.appendChild(doc.createTextNode(m->serial())); me.appendChild(mp); mp = doc.createElement(STR_PROP_Series); mp.appendChild(doc.createTextNode(m->series())); me.appendChild(mp); mp = doc.createElement(STR_PROP_DataVersion); mp.appendChild(doc.createTextNode(QString::number(m->version()))); me.appendChild(mp); mp = doc.createElement(STR_PROP_LastImported); mp.appendChild(doc.createTextNode(m->lastImported().toString(Qt::ISODate))); me.appendChild(mp); mp = doc.createElement(STR_PROP_PurgeDate); mp.appendChild(doc.createTextNode(m->purgeDate().toString(Qt::ISODate))); me.appendChild(mp); mach.appendChild(me); } doc.appendChild(mach); QString filename = p_path+"machines.xml"; QFile file(filename); if (!file.open(QFile::WriteOnly)) { qWarning() << "Could not open" << filename << "for writing, error code" << file.error() << file.errorString(); return false; } file.write(doc.toByteArray()); return true; } qint64 Profile::diskSpaceSummaries() { qint64 size = 0; for (auto & mach : m_machlist) { size += mach->diskSpaceSummaries(); } return size; } qint64 Profile::diskSpaceEvents() { qint64 size = 0; for (auto & mach : m_machlist) { size += mach->diskSpaceEvents(); } return size; } qint64 Profile::diskSpaceBackups() { qint64 size = 0; for (auto & mach : m_machlist) { size += mach->diskSpaceBackups(); } return size; } qint64 Profile::diskSpace() { return (diskSpaceSummaries()+diskSpaceEvents()+diskSpaceBackups()); } void Profile::forceResmedPrefs() { session->setBackupCardData(true); session->setDaySplitTime(QTime(12,0,0)); session->setIgnoreShortSessions(0); session->setCombineCloseSessions(0); session->setLockSummarySessions(true); general->setPrefCalcPercentile(95.0); // 95% general->setPrefCalcMiddle(0); // Median (50%) general->setPrefCalcMax(1); // 99.9th percentile max } #if defined(Q_OS_WIN) class Environment { public: Environment(); QStringList path(); QString searchInDirectory(const QStringList & execs, QString directory); QString searchInPath(const QString &executable, const QStringList & additionalDirs = QStringList()); QProcessEnvironment env; }; Environment::Environment() { env = QProcessEnvironment::systemEnvironment(); } QStringList Environment::path() { return env.value(QLatin1String("PATH"), "").split(';'); } QString Environment::searchInDirectory(const QStringList & execs, QString directory) { const QChar slash = QLatin1Char('/'); if (directory.isEmpty()) return QString(); if (!directory.endsWith(slash)) directory += slash; for (auto & exec : execs) { QFileInfo fi(directory + exec); if (fi.exists() && fi.isFile() && fi.isExecutable()) return fi.absoluteFilePath(); } return QString(); } QString Environment::searchInPath(const QString &executable, const QStringList & additionalDirs) { if (executable.isEmpty()) return QString(); QString exec = QDir::cleanPath(executable); QFileInfo fi(exec); QStringList execs(exec); if (fi.suffix().isEmpty()) { QStringList extensions = env.value(QLatin1String("PATHEXT")).split(QLatin1Char(';')); foreach (const QString &ext, extensions) { QString tmp = executable + ext.toLower(); if (fi.isAbsolute()) { if (QFile::exists(tmp)) return tmp; } else { execs << tmp; } } } if (fi.isAbsolute()) return exec; QSet alreadyChecked; foreach (const QString &dir, additionalDirs) { if (alreadyChecked.contains(dir)) continue; alreadyChecked.insert(dir); QString tmp = searchInDirectory(execs, dir); if (!tmp.isEmpty()) return tmp; } if (executable.indexOf(QLatin1Char('/')) != -1) return QString(); for (auto & p : path()) { if (alreadyChecked.contains(p)) continue; alreadyChecked.insert(p); QString tmp = searchInDirectory(execs, QDir::fromNativeSeparators(p)); if (!tmp.isEmpty()) return tmp; } return QString(); } #endif // Borrowed from QtCreator (http://stackoverflow.com/questions/3490336/how-to-reveal-in-finder-or-show-in-explorer-with-qt) void showInGraphicalShell(const QString & pathIn) { // Mac, Windows support folder or file. #if defined(Q_OS_WIN) QWidget * parent = NULL; Environment env; const QString explorer = env.searchInPath(QLatin1String("explorer.exe")); if (explorer.isEmpty()) { QMessageBox::warning(parent, QObject::tr("Launching Windows Explorer failed"), QObject::tr("Could not find explorer.exe in path to launch Windows Explorer.")); return; } QString param; //if (!QFileInfo(pathIn).isDir()) param = QLatin1String("/select,"); param += QDir::toNativeSeparators(pathIn); QProcess::startDetached(explorer, QStringList(param)); #elif defined(Q_OS_MAC) // Q_UNUSED(parent) QStringList scriptArgs; scriptArgs << QLatin1String("-e") << QString::fromLatin1("tell application \"Finder\" to reveal POSIX file \"%1\"") .arg(pathIn); QProcess::execute(QLatin1String("/usr/bin/osascript"), scriptArgs); scriptArgs.clear(); scriptArgs << QLatin1String("-e") << QLatin1String("tell application \"Finder\" to activate"); QProcess::execute("/usr/bin/osascript", scriptArgs); #else Q_UNUSED(pathIn); // we cannot select a file here, because no file browser really supports it... /* const QFileInfo fileInfo(pathIn); const QString folder = fileInfo.absoluteFilePath(); const QString app = Utils::UnixUtils::fileBrowser(Core::ICore::instance()->settings()); QProcess browserProc; const QString browserArgs = Utils::UnixUtils::substituteFileBrowserParameters(app, folder); if (debug) qDebug() << browserArgs; bool success = browserProc.startDetached(browserArgs); const QString error = QString::fromLocal8Bit(browserProc.readAllStandardError()); success = success && error.isEmpty(); if (!success) { QMessageBox::warning(NULL,STR_MessageBox_Error, "Could not find the file browser for your system, you will have to find your profile directory yourself."+"\n\n"+error, QMessageBox::Ok); // showGraphicalShellError(parent, app, error); }*/ #endif } int dirCount(QString path) { QDir dir(path); QStringList list = dir.entryList(QDir::NoDotAndDotDot | QDir::Files | QDir::Dirs); return list.size(); } void Profile::DataFormatError(Machine *m) { QString msg; msg = ""+QObject::tr("OSCAR %1 needs to upgrade its database for %2 %3 %4"). arg(getVersion().displayString()). arg(m->brand()).arg(m->model()).arg(m->serial()) + "

"; bool backups = false; if (p_profile->session->backupCardData()) { QString bpath = m->getBackupPath(); int cnt = dirCount(bpath); if (cnt > 0) backups = true; } if (backups) { msg = msg + QObject::tr("OSCAR maintains a backup of your devices data card that it uses for this purpose.")+ "

"; msg = msg + QObject::tr("Your old device data should be regenerated provided this backup feature has not been disabled in preferences during a previous data import.") + "

"; backups = true; } else { msg = msg + ""+STR_MessageBox_Warning+": "+QObject::tr("OSCAR does not yet have any automatic card backups stored for this device.") + "

"; msg = msg + QObject::tr("This means you will need to import this device data again afterwards from your own backups or data card.") + "

"; } msg += ""+QObject::tr("Important:")+" "+QObject::tr("Once you upgrade, you cannot use this profile with the previous version anymore.")+"

"+ QObject::tr("If you are concerned, click No to exit, and backup your profile manually, before starting OSCAR again.")+ "

"; msg = msg + ""+QObject::tr("Are you ready to upgrade, so you can run the new version of OSCAR?")+""; QMessageBox * question = new QMessageBox(QMessageBox::Warning, QObject::tr("Device Database Changes"), msg, QMessageBox::Yes | QMessageBox::No); question->setDefaultButton(QMessageBox::Yes); QFont font("Sans Serif", 11, QFont::Normal); question->setFont(font); if (question->exec() == QMessageBox::Yes) { if (!m->Purge(3478216)) { // Purge failed.. probably a permissions error.. let the user deal with it. QMessageBox::critical(nullptr, STR_MessageBox_Error, QObject::tr("Sorry, the purge operation failed, which means this version of OSCAR can't start.")+"\n\n"+ QObject::tr("The device data folder needs to be removed manually.")+"\n\n"+ QObject::tr("This folder currently resides at the following location:")+"\n\n"+ QDir::toNativeSeparators(Get(p_preferences[STR_GEN_DataFolder].toString())), QMessageBox::Ok); QApplication::exit(-1); } // Note: I deliberately haven't added a Profile help for this if (backups) { MachineLoader * loader = lookupLoader(m); int c = mainwin->importCPAP(ImportPath(m->getBackupPath(), loader), QObject::tr("Rebuilding from %1 Backup").arg(m->brand())); if (c >= 0) { // Make sure the updated version gets saved, even if there were no sessions to import. mainwin->finishCPAPImport(); } } else { if (!p_profile->session->backupCardData()) { // Automatic backups not available for Intellipap users yet, so don't taunt them.. if (m->loaderName() != STR_MACH_Intellipap) { if (QMessageBox::question(nullptr, STR_MessageBox_Question, QObject::tr("Would you like to switch on automatic backups, so next time a new version of OSCAR needs to do so, it can rebuild from these?"), QMessageBox::Yes | QMessageBox::No, QMessageBox::Yes)) { p_profile->session->setBackupCardData(true); } } } QMessageBox::information(nullptr, STR_MessageBox_Information, QObject::tr("OSCAR will now start the import wizard so you can reinstall your %1 data.").arg(m->brand()) ,QMessageBox::Ok, QMessageBox::Ok); mainwin->startImportDialog(); } p_profile->Save(); delete question; } else { delete question; QMessageBox::information(nullptr, STR_MessageBox_Information, QObject::tr("OSCAR will now exit, then (attempt to) launch your computers file manager so you can manually back your profile up:")+"\n\n"+ QDir::toNativeSeparators(Get(p_preferences[STR_GEN_DataFolder].toString()))+"\n\n"+ QObject::tr("Use your file manager to make a copy of your profile directory, then afterwards, restart OSCAR and complete the upgrade process.") , QMessageBox::Ok, QMessageBox::Ok); showInGraphicalShell(Get(p_preferences[STR_GEN_DataFolder].toString())); QApplication::exit(-1); } return; } void Profile::UnloadMachineData() { for (auto & mach : m_machlist) { mach->saveSessionInfo(); mach->sessionlist.clear(); mach->day.clear(); } for (auto & day : daylist) { delete day; } daylist.clear(); removeLock(); } void Profile::LoadMachineData(ProgressDialog *progress) { addLock(); for (auto & mach : m_machlist) { MachineLoader *loader = lookupLoader(mach); if (loader) { if (mach->version() < loader->Version()) { qDebug() << "LoadMachineData" << mach->loaderName() << "data format error, machine version" << mach->version() << "loader version" << loader->Version(); progress->hide(); DataFormatError(mach); progress->show(); } else { try { mach->Load(progress); } catch (OldDBVersion& e) { qDebug() << "LoadMachineData" << mach->loaderName() << "load failure, machine version" << mach->version() << "loader version" << loader->Version(); Q_UNUSED(e) progress->hide(); DataFormatError(mach); progress->show(); } } } else { mach->Load(progress); } } progress->setMessage("Loading Channel Information"); loadChannels(); } void Profile::removeMachine(Machine * mach) { if (m_machlist.removeAll(mach)) { QHash >::iterator mlit = MachineList.find(mach->loaderName()); if (mlit != MachineList.end()) { QHash::iterator mit = mlit.value().find(mach->serial()); if (mit != mlit.value().end()) { mlit.value().erase(mit); } } } } Machine * Profile::lookupMachine(QString serial, QString loadername) { auto mlit = MachineList.find(loadername); if (mlit != MachineList.end()) { auto mit = mlit.value().find(serial); if (mit != mlit.value().end()) { return mit.value(); } } return nullptr; } Machine * Profile::CreateMachine(MachineInfo info, MachineID id) { Machine *m = nullptr; auto mlit = MachineList.find(info.loadername); if (mlit != MachineList.end()) { auto mit = mlit.value().find(info.serial); if (mit != mlit.value().end()) { mit.value()->setInfo(info); // update info return mit.value(); } } // Before we create, find any lost folder to get the old ID if ((id == 0) && ((info.type == MT_OXIMETER) || (info.type == MT_JOURNAL) || (info.type == MT_POSITION)|| (info.type == MT_SLEEPSTAGE))) { QString dataPath = Get("{" + STR_GEN_DataFolder + "}/"); QDir dir(dataPath); QStringList namefilter(QString(info.loadername+"_*")); QStringList files = dir.entryList(namefilter, QDir::Dirs); if (files.size() > 0) { QString idstr = files[0].section("_",-1); bool ok; id = idstr.toInt(&ok, 16); } } switch (info.type) { case MT_CPAP: m = new CPAP(this, id); break; case MT_SLEEPSTAGE: m = new SleepStage(this, id); break; case MT_OXIMETER: m = new Oximeter(this, id); break; case MT_POSITION: m = new PositionSensor(this, id); break; case MT_JOURNAL: m = new Machine(this, id); m->setType(MT_JOURNAL); break; default: m = new Machine(this, id); break; } m->setInfo(info); // qDebug() << "Reading" << info.loadername << "Machine Record" << (info.serial.isEmpty() ? m->hexid() : info.serial); MachineList[info.loadername][info.serial] = m; AddMachine(m); return m; } void Profile::AddMachine(Machine *m) { if (!m) { qWarning() << "Empty Machine in Profile::AddMachine()"; return; } m_machlist.append(m); } void Profile::DelMachine(Machine *m) { if (!m) { qWarning() << "Empty Machine in Profile::AddMachine()"; return; } removeMachine(m); } Day *Profile::addDay(QDate date) { auto dit = daylist.find(date); if (dit == daylist.end()) { dit = daylist.insert(date, new Day()); } Day * day = dit.value(); day->setDate(date); if (is_first_day) { m_first = m_last = date; is_first_day = false; } if (m_first > date) { m_first = date; } if (m_last < date) { m_last = date; } return day; } // Get Day record if data available for date and device type, // and has enabled session data, else return nullptr Day *Profile::GetGoodDay(QDate date, MachineType type) { Day *day = GetDay(date, type); if (!day) return nullptr; // For a device match, find at least one enabled Session. for (auto & sess : day->sessions) { if (((type == MT_UNKNOWN) || (sess->type() == type)) && sess->enabled()) { day->OpenSummary(); return day; } } // No enabled Sessions were found. return nullptr; } Day *Profile::FindGoodDay(QDate date, MachineType type) { Day *day = FindDay(date, type); if (!day) return nullptr; // For a device match, find at least one enabled Session. for (auto & sess : day->sessions) { if (((type == MT_UNKNOWN) || (sess->type() == type)) && sess->enabled()) { return day; } } // No enabled Sessions were found. return nullptr; } Day *Profile::GetDay(QDate date, MachineType type) { auto di = daylist.find(date); if (di == daylist.end()) return nullptr; Day * day = di.value(); if (type == MT_UNKNOWN) { day->OpenSummary(); return day; // just want the day record } if (day->machines.contains(type)) { day->OpenSummary(); return day; } return nullptr; } Day *Profile::FindDay(QDate date, MachineType type) { auto di = daylist.find(date); if (di == daylist.end()) return nullptr; Day * day = di.value(); if (type == MT_UNKNOWN) { return day; // just want the day record } if (day->machines.contains(type)) { return day; } return nullptr; } MachineLoader *GetLoader(QString name) { QList loaders = GetLoaders(); for (auto & loader : loaders) { if (loader->loaderName() == name) { return loader; } } return nullptr; } // Returns a QVector containing all device objects regisered of type t QList Profile::GetMachines(MachineType t) { QList vec; for (auto & mach : m_machlist) { if (!mach) { qWarning() << "Profile::GetMachines() m == nullptr"; continue; } MachineType mt = mach->type(); if ((t == MT_UNKNOWN) || (mt == t)) { vec.push_back(mach); } } return vec; } Machine *Profile::GetMachine(MachineType t) { QListvec = GetMachines(t); if (vec.size() == 0) { return nullptr; } // Find most recently imported device int idx = 0; for (int i=1; i < vec.size(); i++) { if (vec[i]->lastImported() > vec[idx]->lastImported()) idx = i; } return vec[idx]; } //bool Profile::trashMachine(Machine * mach) //{ // QMap >::iterator it_end = daylist.end(); // QMap >::iterator it; // // QList datelist; // QList days; // // for (it = daylist.begin(); it != it_end; ++it) { // for (int i = 0; i< it.value().size(); ++i) { // Day * day = it.value().at(i); // if (day->machine() == mach) { // days.push_back(day); // datelist.push_back(it.key()); // } // } // } // // for (int i=0; i < datelist.size(); ++i) { // Day * day = days.at(i); // it = daylist.find(datelist.at(i)); // if (it != daylist.end()) { // it.value().removeAll(day); // if (it.value().size() == 0) { // daylist.erase(it); // } // } // mach->unlinkDay(days.at(i)); // } // //} bool Profile::unlinkDay(Day * day) { // Find the key... for (auto it = daylist.begin(), it_end = daylist.end(); it != it_end; ++it) { if (it.value() == day) { daylist.erase(it); return true; } } return false; } //Profile *profile=nullptr; QString SHA1(QString pass) { return pass; } namespace Profiles { QMap profiles; void Done() { p_pref->Save(); profiles.clear(); delete p_pref; delete AppSetting; DestroyLoaders(); } Profile *Get(QString name) { auto it = profiles.find(name); if (it != profiles.end()) { return it.value(); } return nullptr; } Profile *Create(QString name, const QString* in_path) { QString path; if (in_path == nullptr) { path = p_pref->Get("{home}/Profiles/") + name; } else { path = *in_path; } QDir dir(path); if (!dir.exists(path)) { dir.mkpath(path); } //path+="/"+name; p_profile = new Profile(path); profiles[name] = p_profile; p_profile->user->setUserName(name); //p_profile->Set("Realname",realname); //if (!password.isEmpty()) p_profile.user->setPassword(password); p_profile->Set(STR_GEN_DataFolder, QString("{home}/Profiles/{") + QString(STR_UI_UserName) + QString("}")); Machine *m = new Machine(p_profile, 0); m->setType(MT_JOURNAL); MachineInfo info(MT_JOURNAL, 0, STR_MACH_Journal, "OSCAR", STR_MACH_Journal, QString(), m->hexid(), QString(), QDateTime::currentDateTime(), 0); m->setInfo(info); p_profile->AddMachine(m); p_profile->Save(); return p_profile; } Profile *Get() { // username lookup //getUserName() return profiles[getUserName()];; } //profiles.xml is never read so it does not need to be saved. #if 0 void saveProfileList() { QString filename = p_pref->Get("{home}/profiles.xml"); QDomDocument doc("profiles"); QDomElement root = doc.createElement("profiles"); doc.appendChild(root); root.appendChild(doc.createComment("This file is created during Profile Scan for cloud access convenience, it's not used by Desktop version of OSCAR.")); for (auto it = profiles.begin(); it != profiles.end(); ++it) { QDomElement elem = doc.createElement("profile"); elem.setAttribute("name", it.key()); // Not technically nessesary.. elem.setAttribute("path", QString("{home}/Profiles/%1/Profile.xml").arg(it.key())); root.appendChild(elem); } QFile file(filename); if (!file.open(QFile::WriteOnly)) { qWarning() << "Could not open" << filename << "for writing, error code" << file.error() << file.errorString(); return; } file.write(doc.toByteArray()); file.close(); } #endif int CleanupProfile(Profile *prof) { // Migrate old per Profile settings that should have been put in program main preferences. QStringList migrateList; migrateList << STR_IS_Multithreading << STR_US_ShowPerformance << STR_US_ShowDebug << STR_US_ScrollDampening << STR_AS_CalendarVisible << STR_IS_CacheSessions << STR_AS_LineCursorMode << STR_AS_RightSidebarVisible << STR_AS_DailyPanelWidth << STR_US_ShowPerformance << STR_AS_GraphHeight << STR_AS_GraphSnapshots << STR_AS_AntiAliasing << STR_AS_LineThickness << STR_AS_UsePixmapCaching << STR_AS_SquareWave << STR_AS_RightPanelWidth << STR_US_TooltipTimeout << STR_AS_Animations << STR_AS_AllowYAxisScaling << STR_AS_GraphTooltips << STR_CS_UserEventPieChart << STR_AS_OverlayType #ifndef REMOVE_FITNESS << STR_AS_OverviewLinechartMode #endif ; int cnt = 0; for (auto & prf :migrateList) { if (prof->contains(prf)) { qDebug() << "Migrating profile preference" << prf; (*p_pref)[prf] = (*prof)[prf]; prof->Erase(prf); cnt++; } } if (cnt > 0) { qDebug() << "Migrated" << cnt << "preferences for profile" << (*prof)[STR_UI_UserName]; prof->Save(); } return cnt; } /** * @brief Scan Profile directory loading user profiles */ void Scan() { QString path = p_pref->Get("{home}/Profiles"); QDir dir(path); profiles.clear(); if (!dir.exists(path)) { return; } if (!dir.isReadable()) { qWarning() << "Can't open " << path; return; } dir.setFilter(QDir::Dirs | QDir::NoDotAndDotDot); //dir.setSorting(QDir::Name); QFileInfoList list = dir.entryInfoList(); int cleanup = 0; // Iterate through subdirectories and load profiles.. for (auto & fi : list) { QString npath = fi.canonicalFilePath(); QDir profilePath(npath); if (profilePath.isEmpty()) // skip any empty folders continue; Profile *prof = new Profile(npath); //prof->Open(); // validate user name in profile. QString dbname = prof->user->userName(); QString fname = QFileInfo(fi).fileName(); if (fname != dbname) { prof->user->setUserName(fname); QString message = QString("%1 %2 %3 %4").arg("Changing Profile Name").arg(dbname).arg("==>").arg(fname); qDebug() << message; } profiles[fi.fileName()] = prof; // Migrate any old settings cleanup += CleanupProfile(prof); } if (cleanup > 0) { qDebug() << "Saving preferences after migration"; p_pref->Save(); } // Update profiles.xml for mobile version // profiles.xml is never read so it does not need to be saved. // saveProfileList(); } } // namespace Profiles // Returns a list of all days records matching device type between start and end date QList Profile::getDays(MachineType mt, QDate start, QDate end) { QList list; if (!start.isValid()) { return list; } if (!end.isValid()) { return list; } QDate date = start; if (date.isNull()) { return list; } QMap::iterator it; do { it = daylist.find(date); if (it != daylist.end()) { Day *day = it.value(); if (mt != MT_UNKNOWN) { if (day->hasEnabledSessions(mt)) { list.push_back(day); } } else { if (day->hasEnabledSessions()) { list.push_back(day); } } } date = date.addDays(1); } while (date <= end); return list; } // Counts number of days in range with data for specified device type int Profile::countDays(MachineType mt, QDate start, QDate end) { if (!start.isValid()) { return 0; } if (!end.isValid()) { return 0; } QDate date = start; if (date.isNull()) { return 0; } int days = 0; do { Day *day = FindGoodDay(date, mt); if (day) { days++; } date = date.addDays(1); } while (date <= end); return days; } int Profile::countCompliantDays(MachineType mt, QDate start, QDate end) { EventDataType compliance = cpap->complianceHours(); if (!start.isValid()) { return 0; } if (!end.isValid()) { return 0; } QDate date = start; if (date.isNull()) { return 0; } int days = 0; do { Day *day = FindGoodDay(date, mt); if (day) { if (day->hours(mt) >= compliance) { days++; } // compliant days includes the value specified. } date = date.addDays(1); } while (date <= end); return days; } // Count number of events of type code in period EventDataType Profile::calcCount(ChannelID code, MachineType mt, QDate start, QDate end) { if (!start.isValid()) { start = LastGoodDay(mt); } if (!end.isValid()) { end = LastGoodDay(mt); } QDate date = start; if (date.isNull()) { return 0; } double val = 0; do { Day *day = GetGoodDay(date, mt); if (day) { val += day->count(code); } date = date.addDays(1); } while (date <= end); return val; } double Profile::calcSum(ChannelID code, MachineType mt, QDate start, QDate end) { if (!start.isValid()) { start = LastGoodDay(mt); } if (!end.isValid()) { end = LastGoodDay(mt); } QDate date = start; double val = 0; do { Day *day = GetGoodDay(date, mt); if (day) { val += day->sum(code); } date = date.addDays(1); } while (date <= end); return val; } EventDataType Profile::calcHours(MachineType mt, QDate start, QDate end) { if (!start.isValid()) { start = LastGoodDay(mt); } if (!end.isValid()) { end = LastGoodDay(mt); } QDate date = start; if (date.isNull()) { return 0; } double val = 0; do { Day *day = GetGoodDay(date, mt); if (day) { val += day->hours(mt); } date = date.addDays(1); } while (date <= end); return val; } EventDataType Profile::calcAboveThreshold(ChannelID code, EventDataType threshold, MachineType mt, QDate start, QDate end) { if (!start.isValid()) { start = LastGoodDay(mt); } if (!end.isValid()) { end = LastGoodDay(mt); } QDate date = start; if (date.isNull()) { return 0; } EventDataType val = 0; do { Day *day = GetGoodDay(date, mt); if (day) { val += day->timeAboveThreshold(code, threshold); } date = date.addDays(1); } while (date <= end); return val; } EventDataType Profile::calcBelowThreshold(ChannelID code, EventDataType threshold, MachineType mt, QDate start, QDate end) { if (!start.isValid()) { start = LastGoodDay(mt); } if (!end.isValid()) { end = LastGoodDay(mt); } QDate date = start; if (date.isNull()) { return 0; } EventDataType val = 0; do { Day *day = GetGoodDay(date, mt); if (day) { val += day->timeBelowThreshold(code, threshold); } date = date.addDays(1); } while (date <= end); return val; } Day * Profile::findSessionDay(Session * session) { for (auto it=p_profile->daylist.begin(),it_end = p_profile->daylist.end(); it != it_end; ++it) { Day *day = it.value(); for (auto & sess : day->sessions) { if (sess == session) { return day; } } } return nullptr; } EventDataType Profile::calcAvg(ChannelID code, MachineType mt, QDate start, QDate end) { if (!start.isValid()) { start = LastGoodDay(mt); } if (!end.isValid()) { end = LastGoodDay(mt); } QDate date = start; if (date.isNull()) { return 0; } double val = 0; int cnt = 0; do { Day *day = GetGoodDay(date, mt); if (day) { if (!day->summaryOnly() || day->hasData(code, ST_AVG)) { val += day->sum(code); cnt += day->count(code); } } date = date.addDays(1); } while (date <= end); if (!cnt) { return 0; } return val / float(cnt); } EventDataType Profile::calcWavg(ChannelID code, MachineType mt, QDate start, QDate end) { if (!start.isValid()) { start = LastGoodDay(mt); } if (!end.isValid()) { end = LastGoodDay(mt); } QDate date = start; if (date.isNull()) { return 0; } double val = 0, tmp, tmph, hours = 0; do { Day *day = GetGoodDay(date, mt); if (day) { if (!day->summaryOnly() || day->hasData(code, ST_WAVG)) { tmph = day->hours(); tmp = day->wavg(code); val += tmp * tmph; hours += tmph; } } date = date.addDays(1); } while (date <= end); if (!hours) { return 0; } val = val / hours; return val; } EventDataType Profile::calcMin(ChannelID code, MachineType mt, QDate start, QDate end) { if (!start.isValid()) { start = LastGoodDay(mt); } if (!end.isValid()) { end = LastGoodDay(mt); } QDate date = start; if (date.isNull()) { return 0; } bool first = true; double min = 0, tmp; do { Day *day = GetGoodDay(date, mt); if (day) { if (!day->summaryOnly() || day->hasData(code, ST_MIN)) { tmp = day->Min(code); if (first || (min > tmp)) { min = tmp; first = false; } } } date = date.addDays(1); } while (date <= end); if (first) { min = 0; } return min; } EventDataType Profile::calcMax(ChannelID code, MachineType mt, QDate start, QDate end) { if (!start.isValid()) { start = LastGoodDay(mt); } if (!end.isValid()) { end = LastGoodDay(mt); } QDate date = start; bool first = true; double max = 0, tmp; do { Day *day = GetGoodDay(date, mt); if (day) { if (!day->summaryOnly() || day->hasData(code, ST_MAX)) { tmp = day->Max(code); if (first || (max < tmp)) { max = tmp; first = false; } } } date = date.addDays(1); } while (date <= end); if (first) { max = 0; } return max; } EventDataType Profile::calcSettingsMin(ChannelID code, MachineType mt, QDate start, QDate end) { if (!start.isValid()) { start = LastGoodDay(mt); } if (!end.isValid()) { end = LastGoodDay(mt); } QDate date = start; if (date.isNull()) { return 0; } bool first = true; double min = 0, tmp; do { Day *day = GetGoodDay(date, mt); if (day) { tmp = day->settings_min(code); if (first || (min > tmp)) { min = tmp; first = false; } } date = date.addDays(1); } while (date <= end); if (first) { min = 0; } return min; } EventDataType Profile::calcSettingsMax(ChannelID code, MachineType mt, QDate start, QDate end) { if (!start.isValid()) { start = LastGoodDay(mt); } if (!end.isValid()) { end = LastGoodDay(mt); } QDate date = start; if (date.isNull()) { return 0; } bool first = true; double max = 0, tmp; do { Day *day = GetGoodDay(date, mt); if (day) { tmp = day->settings_max(code); if (first || (max < tmp)) { max = tmp; first = false; } } date = date.addDays(1); } while (date <= end); if (first) { max = 0; } return max; } struct CountSummary { CountSummary(EventStoreType v) : val(v), count(0), time(0) {} EventStoreType val; EventStoreType count; quint32 time; }; EventDataType Profile::calcPercentile(ChannelID code, EventDataType percent, MachineType mt, QDate start, QDate end) { if (!start.isValid()) { start = LastGoodDay(mt); } if (!end.isValid()) { end = LastGoodDay(mt); } QDate date = start; if (date.isNull()) { return 0; } QMap wmap; QMap::iterator wmi; QHash >::iterator vsi; QHash >::iterator tsi; EventDataType gain; //bool setgain=false; EventDataType value; int weight; qint64 SN = 0; bool timeweight; bool summaryOnly = true; do { Day *day = GetGoodDay(date, mt); if (day) { if (day->summaryOnly()) { date = date.addDays(1); continue; } summaryOnly = false; // why was this nested like this??? //for (int i = 0; i < day->size(); i++) { for (auto & sess : day->sessions) { if (!sess->enabled()) { continue; } gain = sess->m_gain[code]; if (!gain) { gain = 1; } vsi = sess->m_valuesummary.find(code); if (vsi == sess->m_valuesummary.end()) { continue; } tsi = sess->m_timesummary.find(code); timeweight = (tsi != sess->m_timesummary.end()); QHash &vsum = vsi.value(); QHash &tsum = tsi.value(); if (timeweight) { for (auto k=tsum.begin(), tsumend=tsum.end(); k != tsumend; k++) { weight = k.value(); value = EventDataType(k.key()) * gain; SN += weight; wmi = wmap.find(value); if (wmi == wmap.end()) { wmap[value] = weight; } else { wmi.value() += weight; } } } else { for (auto k=vsum.begin(), vsumend=vsum.end(); k!=vsumend; k++) { weight = k.value(); value = EventDataType(k.key()) * gain; SN += weight; wmi = wmap.find(value); if (wmi == wmap.end()) { wmap[value] = weight; } else { wmi.value() += weight; } } } } // } } date = date.addDays(1); } while (date <= end); if (summaryOnly) { // abort percentile calculation, there is not enough data return 0; } QVector valcnt; // Build sorted list of value/counts for (wmi = wmap.begin(); wmi != wmap.end(); wmi++) { ValueCount vc; vc.value = wmi.key(); vc.count = wmi.value(); vc.p = 0; valcnt.push_back(vc); } // sort by weight, then value std::sort(valcnt.begin(), valcnt.end()); //double SN=100.0/double(N); // 100% / overall sum double p = 100.0 * percent; double nth = double(SN) * percent; // index of the position in the unweighted set would be double nthi = floor(nth); qint64 sum1 = 0, sum2 = 0; qint64 w1, w2 = 0; double v1 = 0, v2 = 0; int N = valcnt.size(); int k = 0; for (k = 0; k < N; k++) { v1 = valcnt[k].value; w1 = valcnt[k].count; sum1 += w1; if (sum1 > nthi) { return v1; } if (sum1 == nthi) { break; // boundary condition } } if (k >= N) { return v1; } v2 = valcnt[k + 1].value; w2 = valcnt[k + 1].count; sum2 = sum1 + w2; // value lies between v1 and v2 double px = 100.0 / double(SN); // Percentile represented by one full value // calculate percentile ranks double p1 = px * (double(sum1) - (double(w1) / 2.0)); double p2 = px * (double(sum2) - (double(w2) / 2.0)); // calculate linear interpolation double v = v1 + ((p - p1) / (p2 - p1)) * (v2 - v1); // p1.....p.............p2 // 37 55 70 return v; } // Lookup first day record of the specified device type, or return the first day overall if MT_UNKNOWN QDate Profile::FirstDay(MachineType mt) { if ((mt == MT_UNKNOWN) || (!m_last.isValid()) || (!m_first.isValid())) { return m_first; } QDate d = m_first; do { if (FindDay(d, mt) != nullptr) { return d; } d = d.addDays(1); } while (d <= m_last); return m_last; } // Lookup last day record of the specified device type, or return the last day overall if MT_UNKNOWN QDate Profile::LastDay(MachineType mt) { if ((mt == MT_UNKNOWN) || (!m_last.isValid()) || (!m_first.isValid())) { return m_last; } QDate d = m_last; do { if (FindDay(d, mt) != nullptr) { return d; } d = d.addDays(-1); } while (d >= m_first); return m_first; } QDate Profile::FirstGoodDay(MachineType mt) { if (mt == MT_UNKNOWN) { return FirstDay(); } QDate d = FirstDay(mt); QDate l = LastDay(mt); // No data will return invalid date records if (!d.isValid() || !l.isValid()) { return QDate(); } do { if (FindGoodDay(d, mt) != nullptr) { return d; } d = d.addDays(1); } while (d <= l); return l; //m_last; } QDate Profile::LastGoodDay(MachineType mt) { if (mt == MT_UNKNOWN) { return FirstDay(); } QDate d = LastDay(mt); QDate f = FirstDay(mt); if (!(d.isValid() && f.isValid())) { return QDate(); } do { if (FindGoodDay(d, mt) != nullptr) { return d; } d = d.addDays(-1); } while (d >= f); return f; } bool Profile::channelAvailable(ChannelID code) { for (auto & mach : m_machlist) { if (mach->hasChannel(code)) return true; } return false; } bool Profile::hasChannel(ChannelID code) { QDate d = LastDay(); QDate f = FirstDay(); if (!(d.isValid() && f.isValid())) { return false; } QMap::iterator dit; bool found = false; do { dit = daylist.find(d); if (dit != daylist.end()) { Day *day = dit.value(); if (day->channelHasData(code)) { found = true; break; } } d = d.addDays(-1); } while (d >= f); return found; } const quint16 chandata_version = 1; void Profile::saveChannels() { // First save the XML version for Mobile versions // Profile/User/chanels.xml is not read so it does not need to be saved // schema::channel.Save(Get("{DataFolder}/") + "channels.xml"); QString filename = Get("{DataFolder}/") + "channels.dat"; QFile f(filename); qDebug() << "Saving Channel States"; f.open(QFile::WriteOnly); QDataStream out(&f); out.setVersion(QDataStream::Qt_4_6); out.setByteOrder(QDataStream::LittleEndian); out << (quint32)magic; out << (quint16)chandata_version; QSettings settings; (*p_profile)[STR_PREF_Language] = settings.value(LangSetting, "").toString(); quint16 size = schema::channel.channels.size(); out << size; for (auto it = schema::channel.channels.begin(),it_end = schema::channel.channels.end(); it != it_end; ++it) { schema::Channel * chan = it.value(); out << it.key(); out << chan->code(); out << chan->enabled(); out << chan->defaultColor(); out << chan->fullname(); out << chan->label(); out << chan->description(); out << chan->lowerThreshold(); out << chan->lowerThresholdColor(); out << chan->upperThreshold(); out << chan->upperThresholdColor(); out << chan->showInOverview(); } f.close(); } void Profile::loadChannels() { bool changing_language = false; QString filename = Get("{DataFolder}/") + "channels.dat"; QFile f(filename); if (!f.open(QFile::ReadOnly)) { return; } qDebug() << "Loading channel.dat States"; QDataStream in(&f); in.setVersion(QDataStream::Qt_4_6); in.setByteOrder(QDataStream::LittleEndian); quint32 mag; in >> mag; if (magic != mag) { qDebug() << "LoadChannels: Faulty data"; return; } quint16 version; in >> version; QSettings settings; QString language = Get(STR_PREF_Language); if (settings.value(LangSetting, "").toString() != language) { qDebug() << "Language change detected, resetting default channel names"; changing_language = true; } quint16 size; in >> size; QString name; ChannelID code; bool enabled; QColor color; EventDataType lowerThreshold; QColor lowerThresholdColor; EventDataType upperThreshold; QColor upperThresholdColor; QString fullname; QString label; QString description; bool showOverview = false; for (int i=0; i < size; i++) { in >> code; schema::Channel * chan = &schema::channel[code]; in >> name; if (chan->code() != name) { qDebug() << "Looking up channel" << name << "by name, as it's ChannedID must have changed"; chan = &schema::channel[name]; } in >> enabled; in >> color; in >> fullname; in >> label; in >> description; in >> lowerThreshold; in >> lowerThresholdColor; in >> upperThreshold; in >> upperThresholdColor; if (version >= 1) { in >> showOverview; } if (chan->isNull()) { qDebug() << "loadChannels has no idea about channel" << name; if (in.atEnd()) break; continue; } chan->setEnabled(enabled); chan->setDefaultColor(color); // Don't import channel descriptions if event renaming is turned off. (helps pick up new translations) if (changing_language) { // Nothing } else { chan->setFullname(fullname); chan->setLabel(label); chan->setDescription(description); } chan->setLowerThreshold(lowerThreshold); chan->setLowerThresholdColor(lowerThresholdColor); chan->setUpperThreshold(upperThreshold); chan->setUpperThresholdColor(upperThresholdColor); chan->setShowInOverview(showOverview); if (in.atEnd()) break; } f.close(); resetOxiChannelPref(); } void Profile::resetOxiChannelPref() { schema::channel[OXI_Pulse].setLowerThreshold(oxi->flagPulseBelow()); schema::channel[OXI_Pulse].setUpperThreshold(oxi->flagPulseAbove()); schema::channel[OXI_SPO2].setLowerThreshold(oxi->oxiDesaturationThreshold()); };