OSCAR-code/sleepyhead/SleepLib/profiles.cpp

1813 lines
44 KiB
C++
Raw Normal View History

2014-08-17 12:56:05 +00:00
/* SleepLib Profiles Implementation
*
* Copyright (c) 2011-2014 Mark Watkins <jedimark@users.sourceforge.net>
*
* 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. */
2011-06-26 08:30:44 +00:00
#include <QString>
#include <QDateTime>
#include <QDir>
#include <QMessageBox>
2011-07-01 10:10:44 +00:00
#include <QDebug>
#include <QProcess>
#include <QByteArray>
#include <QHostInfo>
#include <algorithm>
#include <cmath>
2011-06-26 08:30:44 +00:00
#include "preferences.h"
#include "profiles.h"
#include "machine.h"
#include "machine_common.h"
2011-06-26 08:30:44 +00:00
#include "machine_loader.h"
#include <QApplication>
#include "mainwindow.h"
extern MainWindow *mainwin;
2011-06-26 08:30:44 +00:00
Preferences *p_pref;
Preferences *p_layout;
Profile *p_profile;
2011-06-26 08:30:44 +00:00
Profile::Profile(QString path)
: is_first_day(true),
m_opened(false),
m_machopened(false)
{
p_name = STR_GEN_Profile;
if (path.isEmpty()) {
p_path = GetAppRoot();
} 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;
2011-06-26 08:30:44 +00:00
machlist.clear();
doctor = nullptr;
user = nullptr;
cpap = nullptr;
oxi = nullptr;
appearance = nullptr;
session = nullptr;
general = nullptr;
2011-06-26 08:30:44 +00:00
}
Profile::~Profile()
{
QString lockfile=p_path+"/lockfile";
QFile file(lockfile);
file.remove();
if (m_opened) {
delete user;
delete doctor;
delete cpap;
delete oxi;
delete appearance;
delete session;
delete general;
2014-05-06 18:03:13 +00:00
for (QHash<MachineID, Machine *>::iterator it = machlist.begin(); it != machlist.end(); it++) {
delete it.value();
}
m_opened=false;
2011-06-26 08:30:44 +00:00
}
for (QMap<QDate, Day *>::iterator d = daylist.begin(); d != daylist.end(); d++) {
delete d.value();
}
2011-06-26 08:30:44 +00:00
}
bool Profile::Save(QString filename)
{
if (m_opened) {
return Preferences::Save(filename) && p_profile->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;
}
bool Profile::Open(QString filename)
{
p_profile = this;
if (filename.isEmpty()) {
filename=p_filename;
}
if (m_opened) {
qDebug() << "Profile" << filename << "all ready open";
return true;
}
bool b = Preferences::Open(filename);
m_opened=true;
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);
return b;
}
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";
bool Profile::OpenMachines()
{
if (m_machopened)
return true;
if (!m_opened) {
Open();
}
QFile lockfile(p_path+"lockfile");
lockfile.open(QFile::WriteOnly);
QByteArray ba;
ba.append(QHostInfo::localHostName());
lockfile.write(ba);
lockfile.close();
QString filename = p_path+"machines.xml";
QFile file(filename);
if (!file.open(QFile::ReadOnly)) {
qWarning() << "Could not open" << QDir::toNativeSeparators(filename);
return false;
}
QDomDocument doc("machines.xml");
if (!doc.setContent(&file)) {
qWarning() << "Invalid XML Content in" << QDir::toNativeSeparators(filename);
return false;
}
file.close();
QDomElement root = doc.firstChild().toElement();
if (root.tagName().toLower() != "machines") {
//qDebug() << "No Machines Tag in Profiles.xml";
return false;
}
QDomElement elem = root.firstChildElement();
while (!elem.isNull()) {
QString pKey = elem.tagName();
if (pKey.toLower() != "machine") {
qWarning() << "Profile::ExtraLoad() 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<QString, QString> 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 == "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;
m = MachineLoader::CreateMachine(info, m_id);
//m->setId(m_id);
if (m) m->properties = prop;
elem = elem.nextSiblingElement();
}
m_machopened = true;
return true;
}
bool Profile::StoreMachines()
{
QDomDocument doc("Machines");
QDomElement elem = ExtraSave(doc);
doc.appendChild(elem);
QDomElement mach = doc.createElement("machines");
for (QHash<MachineID, Machine *>::iterator i = machlist.begin(); i != machlist.end(); i++) {
QDomElement me = doc.createElement("machine");
Machine *m = i.value();
me.setAttribute("id", (int)m->id());
me.setAttribute("type", (int)m->type());
me.setAttribute("class", m->loaderName());
QDomElement pe = doc.createElement("properties");
me.appendChild(pe);
for (QHash<QString, QString>::iterator j = i.value()->properties.begin(); j != i.value()->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);
mach.appendChild(me);
}
doc.appendChild(mach);
QString filename = p_path+"machines.xml";
QFile file(filename);
if (!file.open(QFile::WriteOnly)) {
return false;
}
file.write(doc.toByteArray());
return true;
}
2014-07-03 02:31:50 +00:00
#if defined(Q_OS_WIN)
2014-07-03 02:45:38 +00:00
class Environment
2014-07-03 02:31:50 +00:00
{
2014-07-03 02:45:38 +00:00
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(';');
2014-07-03 02:31:50 +00:00
}
2014-07-03 02:45:38 +00:00
QString Environment::searchInDirectory(const QStringList & execs, QString directory)
2014-07-03 02:31:50 +00:00
{
const QChar slash = QLatin1Char('/');
if (directory.isEmpty())
return QString();
if (!directory.endsWith(slash))
directory += slash;
foreach (const QString & exec, execs) {
QFileInfo fi(directory + exec);
if (fi.exists() && fi.isFile() && fi.isExecutable())
return fi.absoluteFilePath();
}
return QString();
}
2014-07-03 02:45:38 +00:00
QString Environment::searchInPath(const QString &executable, const QStringList & additionalDirs)
2014-07-03 02:31:50 +00:00
{
if (executable.isEmpty()) return QString();
QString exec = QDir::cleanPath(executable);
QFileInfo fi(exec);
QStringList execs(exec);
if (fi.suffix().isEmpty()) {
2014-07-03 02:45:38 +00:00
QStringList extensions = env.value(QLatin1String("PATHEXT")).split(QLatin1Char(';'));
2014-07-03 02:31:50 +00:00
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<QString> 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();
foreach (const QString &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)
2014-10-08 16:51:09 +00:00
void showInGraphicalShell(const QString & /*pathIn*/)
{
2014-10-08 16:51:09 +00:00
//QWidget * parent = NULL;
// Mac, Windows support folder or file.
#if defined(Q_OS_WIN)
2014-07-03 02:45:38 +00:00
Environment env;
const QString explorer = env.searchInPath(QLatin1String("explorer.exe"));
if (explorer.isEmpty()) {
QMessageBox::warning(parent,
2014-07-03 02:45:38 +00:00
QObject::tr("Launching Windows Explorer failed"),
QObject::tr("Could not find explorer.exe in path to launch Windows Explorer."));
return;
}
QString param;
2014-07-03 02:45:38 +00:00
//if (!QFileInfo(pathIn).isDir())
param = QLatin1String("/select,");
param += QDir::toNativeSeparators(pathIn);
QProcess::startDetached(explorer, QStringList(param));
#elif defined(Q_OS_MAC)
2014-10-08 16:51:09 +00:00
// 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
// 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 = "<font size=+1>"+QObject::tr("SleepyHead (%1) needs to upgrade its database for %2 %3 %4").
arg(FullVersionString).
arg(m->brand()).arg(m->model()).arg(m->serial())
+ "</font><br/><br/>";
bool backups = false;
if (p_profile->session->backupCardData()) {
QString bpath = m->getBackupPath();
int cnt = dirCount(bpath);
if (cnt > 0) backups = true;
}
2014-05-13 01:28:41 +00:00
if (backups) {
msg = msg + QObject::tr("<b>SleepyHead maintains a backup of your devices data card that it uses for this purpose.</b>")+ "<br/><br/>";
msg = msg + QObject::tr("<i>Your old machine data should be regenerated provided this backup feature has not been disabled in preferences during a previous data import.</i>") + "<br/><br/>";
backups = true;
} else {
msg = msg + "<font size=+1>"+STR_MessageBox_Warning+":</font> "+QObject::tr("SleepyHead does not yet have any automatic card backups stored for this device.") + "<br/><br/>";
msg = msg + QObject::tr("This means you will need to import this machine data again afterwards from your own backups or data card.") + "<br/><br/>";
}
msg += "<font size=+1>"+QObject::tr("Important:")+"</font> "+QObject::tr("Once you upgrade, you <font size=+1>can not</font> use this profile with the previous version anymore.")+"<br/><br/>"+
QObject::tr("If you are concerned, click No to exit, and backup your profile manually, before starting SleepyHead again.")+ "<br/><br/>";
msg = msg + "<font size=+1>"+QObject::tr("Are you ready to upgrade, so you can run the new version of SleepyHead?")+"</font>";
QMessageBox * question = new QMessageBox(QMessageBox::Warning, QObject::tr("Machine 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 SleepyHead can't start.")+"\n\n"+
QObject::tr("The machine 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);
}
2014-05-13 01:28:41 +00:00
// Note: I deliberately haven't added a Profile help for this
if (backups) {
mainwin->importCPAP(ImportPath(m->getBackupPath(), lookupLoader(m)), QObject::tr("Rebuilding from %1 Backup").arg(m->brand()));
} 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 SleepyHead 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("SleepyHead will now start the import wizard so you can reinstall your %1 data.").arg(m->brand())
,QMessageBox::Ok, QMessageBox::Ok);
mainwin->startImportDialog();
}
2014-07-11 12:09:38 +00:00
p_profile->Save();
delete question;
} else {
delete question;
QMessageBox::information(nullptr, STR_MessageBox_Information,
QObject::tr("SleepyHead 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 Sleepyhead and complete the upgrade process.")
, QMessageBox::Ok, QMessageBox::Ok);
showInGraphicalShell(Get(p_preferences[STR_GEN_DataFolder].toString()));
QApplication::exit(-1);
}
return;
}
2011-06-26 08:30:44 +00:00
void Profile::LoadMachineData()
{
if (!m_machopened) OpenMachines();
QHash<MachineID, QMap<QDate, QHash<ChannelID, EventDataType> > > cache;
for (QHash<MachineID, Machine *>::iterator i = machlist.begin(); i != machlist.end(); i++) {
Machine *m = i.value();
MachineLoader *loader = lookupLoader(m);
2011-06-26 08:30:44 +00:00
if (loader) {
if (m->version() < loader->Version()) {
DataFormatError(m);
} else {
try {
m->Load();
} catch (OldDBVersion e) {
Q_UNUSED(e)
DataFormatError(m);
2011-06-26 08:30:44 +00:00
}
}
2011-06-26 08:30:44 +00:00
} else {
m->Load();
}
}
}
2011-06-26 08:30:44 +00:00
/**
* @brief Upgrade Machine XML section from old "profile.xml"
2011-06-26 08:30:44 +00:00
* @param root
*/
void Profile::ExtraLoad(QDomElement &root)
2011-06-26 08:30:44 +00:00
{
if (root.tagName().toLower() != "machines") {
// Good!
return;
2011-06-26 08:30:44 +00:00
}
// Save this sucker
QDomDocument doc("Machines");
doc.appendChild(root);
QFile file(p_path+"/machines.xml");
// Don't do anything if machines.xml already exists.. the user ran the old version!
if (file.exists()) return;
file.open(QFile::WriteOnly);
file.write(doc.toByteArray());
file.close();
2011-06-26 08:30:44 +00:00
}
void Profile::AddMachine(Machine *m)
{
if (!m) {
qWarning() << "Empty Machine in Profile::AddMachine()";
return;
}
machlist[m->id()] = m;
}
void Profile::DelMachine(Machine *m)
{
if (!m) {
qWarning() << "Empty Machine in Profile::AddMachine()";
return;
}
m->loader()->removeMachine(m);
machlist.erase(machlist.find(m->id()));
}
2011-06-26 08:30:44 +00:00
Day *Profile::addDay(QDate date)
{
QMap<QDate, Day *>::iterator dit = daylist.find(date);
if (dit == daylist.end()) {
dit = daylist.insert(date, new Day());
2012-01-08 03:29:22 +00:00
}
Day * day = dit.value();
2014-09-17 02:34:50 +00:00
day->setDate(date);
2011-06-26 08:30:44 +00:00
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;
2011-06-26 08:30:44 +00:00
}
return day;
2011-06-26 08:30:44 +00:00
}
// Get Day record if data available for date and machine 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 machine match, find at least one enabled Session.
for (int i = 0; i < day->size(); ++i) {
Session * sess = (*day)[i];
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 machine match, find at least one enabled Session.
for (int i = 0; i < day->size(); ++i) {
Session * sess = (*day)[i];
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)
2011-06-26 08:30:44 +00:00
{
QMap<QDate, Day *>::iterator 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)
{
QMap<QDate, Day *>::iterator 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;
2011-06-26 08:30:44 +00:00
}
2011-07-15 13:30:41 +00:00
int Profile::Import(QString path)
2011-06-26 08:30:44 +00:00
{
int c = 0;
2011-07-01 10:10:44 +00:00
qDebug() << "Importing " << path;
path = path.replace("\\", "/");
if (path.endsWith("/")) {
2013-09-11 12:57:06 +00:00
path.chop(1);
}
2014-05-25 07:07:08 +00:00
QList<MachineLoader *>loaders = GetLoaders(MT_CPAP);
Q_FOREACH(MachineLoader * loader, loaders) {
2014-07-11 12:09:38 +00:00
if (c += loader->Open(path)) {
2013-09-11 12:57:06 +00:00
break;
}
2011-06-26 08:30:44 +00:00
}
2011-07-15 13:30:41 +00:00
return c;
2011-06-26 08:30:44 +00:00
}
MachineLoader *GetLoader(QString name)
2011-06-26 08:30:44 +00:00
{
QList<MachineLoader *>loaders = GetLoaders();
Q_FOREACH(MachineLoader * loader, loaders) {
if (loader->loaderName() == name) {
return loader;
2011-06-26 08:30:44 +00:00
}
}
return nullptr;
2011-06-26 08:30:44 +00:00
}
// Returns a QVector containing all machine objects regisered of type t
QList<Machine *> Profile::GetMachines(MachineType t)
2011-06-26 08:30:44 +00:00
{
QList<Machine *> vec;
QHash<MachineID, Machine *>::iterator i;
QHash<MachineID, Machine *>::iterator machlist_end=machlist.end();
2011-06-26 08:30:44 +00:00
for (i = machlist.begin(); i != machlist_end; i++) {
if (!i.value()) {
qWarning() << "Profile::GetMachines() i->second == nullptr";
continue;
}
MachineType mt = i.value()->type();
if ((t == MT_UNKNOWN) || (mt == t)) {
vec.push_back(i.value());
2011-06-26 08:30:44 +00:00
}
}
2011-06-26 08:30:44 +00:00
return vec;
}
Machine *Profile::GetMachine(MachineType t)
2011-06-26 08:30:44 +00:00
{
QList<Machine *>vec = GetMachines(t);
if (vec.size() == 0) {
return nullptr;
}
2011-06-26 08:30:44 +00:00
return vec[0];
}
//bool Profile::trashMachine(Machine * mach)
//{
// QMap<QDate, QList<Day *> >::iterator it_end = daylist.end();
// QMap<QDate, QList<Day *> >::iterator it;
// QList<QDate> datelist;
// QList<Day *> 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)
{
QMap<QDate, Day *>::iterator it;
QMap<QDate, Day *>::iterator it_end = daylist.end();
// Find the key...
for (it = daylist.begin(); it != it_end; ++it) {
if (it.value() == day) {
daylist.erase(it);
return true;
}
}
return false;
}
//Profile *profile=nullptr;
2011-06-26 08:30:44 +00:00
QString SHA1(QString pass)
{
return pass;
}
namespace Profiles {
2011-06-26 08:30:44 +00:00
QMap<QString, Profile *> profiles;
2011-06-26 08:30:44 +00:00
void Done()
{
PREF.Save();
LAYOUT.Save();
2014-05-26 19:29:01 +00:00
p_profile->Save();
delete p_profile;
2011-06-26 08:30:44 +00:00
profiles.clear();
delete p_pref;
delete p_layout;
DestroyLoaders();
2011-06-26 08:30:44 +00:00
}
Profile *Get(QString name)
{
if (profiles.find(name) != profiles.end()) {
2011-06-26 08:30:44 +00:00
return profiles[name];
}
2011-06-26 08:30:44 +00:00
return nullptr;
2011-06-26 08:30:44 +00:00
}
2011-10-01 12:54:20 +00:00
Profile *Create(QString name)
2011-06-26 08:30:44 +00:00
{
QString path = PREF.Get("{home}/Profiles/") + name;
2011-06-26 08:30:44 +00:00
QDir dir(path);
if (!dir.exists(path)) {
2013-09-11 12:57:06 +00:00
dir.mkpath(path);
}
2011-06-26 08:30:44 +00:00
//path+="/"+name;
p_profile = new Profile(path);
p_profile->Open();
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("}"));
2011-06-26 08:30:44 +00:00
2014-07-11 12:09:38 +00:00
Machine *m = new Machine(0);
m->setType(MT_JOURNAL);
MachineInfo info(MT_JOURNAL, 0, STR_MACH_Journal, "SleepyHead", STR_MACH_Journal, QString(), m->hexid(), QString(), QDateTime::currentDateTime(), 0);
m->setInfo(info);
p_profile->AddMachine(m);
2011-06-26 08:30:44 +00:00
p_profile->Save();
2011-06-26 08:30:44 +00:00
return p_profile;
2011-06-26 08:30:44 +00:00
}
Profile *Get()
{
2011-06-27 06:26:29 +00:00
// username lookup
//getUserName()
return profiles[getUserName()];;
2011-06-26 08:30:44 +00:00
}
void saveProfileList()
{
QString filename = PREF.Get("{home}/profiles.xml");
QDomDocument doc("profiles");
QDomElement root = doc.createElement("profiles");
doc.appendChild(root);
QMap<QString, Profile *>::iterator it;
for (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);
file.open(QFile::WriteOnly);
file.write(doc.toByteArray());
file.close();
}
2011-06-26 08:30:44 +00:00
/**
* @brief Scan Profile directory loading user profiles
*/
void Scan()
{
QString path = PREF.Get("{home}/Profiles");
2011-06-26 08:30:44 +00:00
QDir dir(path);
if (!dir.exists(path)) {
return;
}
2011-06-26 08:30:44 +00:00
if (!dir.isReadable()) {
2011-07-01 10:10:44 +00:00
qWarning() << "Can't open " << path;
2011-06-26 08:30:44 +00:00
return;
}
dir.setFilter(QDir::Dirs | QDir::NoDotAndDotDot);
//dir.setSorting(QDir::Name);
QFileInfoList list = dir.entryInfoList();
2011-06-26 08:30:44 +00:00
2011-07-17 07:03:26 +00:00
// Iterate through subdirectories and load profiles..
for (int i = 0; i < list.size(); i++) {
QFileInfo fi = list.at(i);
QString npath = fi.canonicalFilePath();
Profile *prof = new Profile(npath);
//prof->Open();
profiles[fi.fileName()] = prof;
2011-06-26 08:30:44 +00:00
}
// Update profiles.xml for mobile version
saveProfileList();
2011-06-26 08:30:44 +00:00
}
} // namespace Profiles
2011-06-26 08:30:44 +00:00
// Returns a list of all days records matching machine type between start and end date
QList<Day *> Profile::getDays(MachineType mt, QDate start, QDate end)
{
QList<Day *> list;
if (!start.isValid()) {
return list;
}
if (!end.isValid()) {
return list;
}
QDate date = start;
if (date.isNull()) {
return list;
}
QMap<QDate, Day *>::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;
}
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()) {
2013-09-11 12:57:06 +00:00
return 0;
}
2013-09-11 12:57:06 +00:00
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 {
2014-09-30 08:40:12 +00:00
Day *day = FindGoodDay(date, mt);
if (day) {
if (day->hours(mt) > compliance) { days++; }
}
date = date.addDays(1);
} while (date <= end);
return days;
}
2011-12-22 13:22:40 +00:00
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()) {
2013-09-11 12:57:06 +00:00
return 0;
}
double val = 0;
2011-12-22 13:22:40 +00:00
do {
Day *day = GetGoodDay(date, mt);
2011-12-22 13:22:40 +00:00
if (day) {
val += day->count(code);
2011-12-22 13:22:40 +00:00
}
date = date.addDays(1);
} while (date <= end);
2011-12-22 13:22:40 +00:00
return val;
}
2011-12-22 13:22:40 +00:00
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;
2011-12-22 13:22:40 +00:00
do {
Day *day = GetGoodDay(date, mt);
2011-12-22 13:22:40 +00:00
if (day) {
val += day->sum(code);
2011-12-22 13:22:40 +00:00
}
date = date.addDays(1);
} while (date <= end);
2011-12-22 13:22:40 +00:00
return val;
}
2011-12-22 13:22:40 +00:00
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()) {
2013-09-11 12:57:06 +00:00
return 0;
}
double val = 0;
2011-12-22 13:22:40 +00:00
do {
Day *day = GetGoodDay(date, mt);
2011-12-22 13:22:40 +00:00
if (day) {
val += day->hours();
2011-12-22 13:22:40 +00:00
}
date = date.addDays(1);
} while (date <= end);
2011-12-22 13:22:40 +00:00
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;
}
2011-12-22 13:22:40 +00:00
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()) {
2013-09-11 12:57:06 +00:00
return 0;
}
double val = 0;
int cnt = 0;
2011-12-22 13:22:40 +00:00
do {
Day *day = GetGoodDay(date, mt);
2011-12-22 13:22:40 +00:00
if (day) {
if (!day->summaryOnly() || day->hasData(code, ST_AVG)) {
val += day->sum(code);
cnt++;
}
2011-12-22 13:22:40 +00:00
}
date = date.addDays(1);
} while (date <= end);
if (!cnt) {
2013-09-11 12:57:06 +00:00
return 0;
}
return val / float(cnt);
2011-12-22 13:22:40 +00:00
}
2011-12-22 13:22:40 +00:00
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()) {
2013-09-11 12:57:06 +00:00
return 0;
}
double val = 0, tmp, tmph, hours = 0;
2011-12-22 13:22:40 +00:00
do {
Day *day = GetGoodDay(date, mt);
2011-12-22 13:22:40 +00:00
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) {
2013-09-11 12:57:06 +00:00
return 0;
}
val = val / hours;
2011-12-22 13:22:40 +00:00
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()) {
2013-09-11 12:57:06 +00:00
return 0;
}
2013-09-11 12:57:06 +00:00
2014-07-02 13:58:59 +00:00
bool first = true;
2014-07-02 13:58:59 +00:00
double min = 0, tmp;
do {
Day *day = GetGoodDay(date, mt);
if (day) {
if (!day->summaryOnly() || day->hasData(code, ST_MIN)) {
tmp = day->Min(code);
2014-07-02 13:58:59 +00:00
if (first || (min > tmp)) {
min = tmp;
2014-07-02 13:58:59 +00:00
first = false;
}
}
}
date = date.addDays(1);
} while (date <= end);
2014-07-02 13:58:59 +00:00
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;
2014-07-02 13:58:59 +00:00
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);
2014-07-02 13:58:59 +00:00
if (first || (max < tmp)) {
max = tmp;
2014-07-02 13:58:59 +00:00
first = false;
}
}
}
date = date.addDays(1);
} while (date <= end);
2014-07-02 13:58:59 +00:00
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()) {
2013-09-11 12:57:06 +00:00
return 0;
}
2014-07-02 13:58:59 +00:00
bool first = true;
double min = 0, tmp;
do {
Day *day = GetGoodDay(date, mt);
if (day) {
tmp = day->settings_min(code);
2014-07-02 13:58:59 +00:00
if (first || (min > tmp)) {
min = tmp;
2014-07-02 13:58:59 +00:00
first = false;
}
}
date = date.addDays(1);
} while (date <= end);
2014-07-02 13:58:59 +00:00
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()) {
2013-09-11 12:57:06 +00:00
return 0;
}
2014-07-02 13:58:59 +00:00
bool first = true;
double max = 0, tmp;
do {
Day *day = GetGoodDay(date, mt);
if (day) {
tmp = day->settings_max(code);
2014-07-02 13:58:59 +00:00
if (first || (max < tmp)) {
max = tmp;
2014-07-02 13:58:59 +00:00
first = false;
}
}
date = date.addDays(1);
} while (date <= end);
2014-07-02 13:58:59 +00:00
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)
2011-12-22 13:22:40 +00:00
{
if (!start.isValid()) {
start = LastGoodDay(mt);
}
if (!end.isValid()) {
end = LastGoodDay(mt);
}
QDate date = start;
2011-12-22 13:22:40 +00:00
if (date.isNull()) {
2013-09-11 12:57:06 +00:00
return 0;
}
2011-12-22 13:22:40 +00:00
QMap<EventDataType, qint64> wmap;
QMap<EventDataType, qint64>::iterator wmi;
QHash<ChannelID, QHash<EventStoreType, EventStoreType> >::iterator vsi;
QHash<ChannelID, QHash<EventStoreType, quint32> >::iterator tsi;
EventDataType gain;
//bool setgain=false;
EventDataType value;
int weight;
qint64 SN = 0;
bool timeweight;
bool summaryOnly = false;
2011-12-22 13:22:40 +00:00
do {
Day *day = GetGoodDay(date, mt);
if (day) {
if (day->summaryOnly()) {
summaryOnly = true;
break;
}
for (int i = 0; i < day->size(); i++) {
for (QList<Session *>::iterator s = day->begin(); s != day->end(); s++) {
if (!(*s)->enabled()) {
continue;
}
Session *sess = *s;
gain = sess->m_gain[code];
if (!gain) { gain = 1; }
2011-12-28 12:03:48 +00:00
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<EventStoreType, EventStoreType> &vsum = vsi.value();
QHash<EventStoreType, quint32> &tsum = tsi.value();
if (timeweight) {
for (QHash<EventStoreType, quint32>::iterator k = tsum.begin(); k != tsum.end(); 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 (QHash<EventStoreType, EventStoreType>::iterator k = vsum.begin(); k != vsum.end(); 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<ValueCount> 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
qSort(valcnt);
//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;
2011-12-22 13:22:40 +00:00
}
// Lookup first day record of the specified machine type, or return the first day overall if MT_UNKNOWN
2011-12-22 13:22:40 +00:00
QDate Profile::FirstDay(MachineType mt)
{
if ((mt == MT_UNKNOWN) || (!m_last.isValid()) || (!m_first.isValid())) {
2011-12-22 13:22:40 +00:00
return m_first;
}
QDate d = m_first;
2011-12-22 13:22:40 +00:00
do {
if (FindDay(d, mt) != nullptr) {
2013-09-11 12:57:06 +00:00
return d;
}
d = d.addDays(1);
} while (d <= m_last);
2011-12-22 13:22:40 +00:00
return m_last;
}
// Lookup last day record of the specified machine type, or return the first day overall if MT_UNKNOWN
2011-12-22 13:22:40 +00:00
QDate Profile::LastDay(MachineType mt)
{
if ((mt == MT_UNKNOWN) || (!m_last.isValid()) || (!m_first.isValid())) {
2011-12-22 13:22:40 +00:00
return m_last;
}
QDate d = m_last;
2011-12-22 13:22:40 +00:00
do {
if (FindDay(d, mt) != nullptr) {
2013-09-11 12:57:06 +00:00
return d;
}
d = d.addDays(-1);
} while (d >= m_first);
2011-12-22 13:22:40 +00:00
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) {
2013-09-11 12:57:06 +00:00
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())) {
2013-09-11 12:57:06 +00:00
return QDate();
}
do {
if (FindGoodDay(d, mt) != nullptr) {
2013-09-11 12:57:06 +00:00
return d;
}
d = d.addDays(-1);
} while (d >= f);
return f;
}
bool Profile::channelAvailable(ChannelID code)
{
QHash<MachineID, Machine *>::iterator it;
QHash<MachineID, Machine *>::iterator machlist_end=machlist.end();
for (it = machlist.begin(); it != machlist_end; it++) {
Machine * mach = it.value();
if (mach->hasChannel(code))
return true;
}
return false;
}
bool Profile::hasChannel(ChannelID code)
{
QDate d = LastDay();
QDate f = FirstDay();
if (!(d.isValid() && f.isValid())) {
2013-09-11 12:57:06 +00:00
return false;
}
QMap<QDate, Day *>::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;
}