OSCAR-code/SleepLib/machine.cpp

437 lines
12 KiB
C++
Raw Normal View History

2011-07-01 10:10:44 +00:00
/*
2011-06-26 08:30:44 +00:00
SleepLib Machine Class Implementation
Copyright (c)2011 Mark Watkins <jedimark@users.sourceforge.net>
License: GPL
2011-07-01 10:10:44 +00:00
*/
2011-06-26 08:30:44 +00:00
2011-07-30 00:36:31 +00:00
#include <QApplication>
2011-07-01 10:10:44 +00:00
#include <QDir>
#include <QProgressBar>
#include <QDebug>
#include <QString>
#include <QObject>
2011-06-26 08:30:44 +00:00
#include <tr1/random>
#include <sys/time.h>
2011-06-26 08:30:44 +00:00
#include "machine.h"
#include "profiles.h"
#include <algorithm>
2011-09-17 12:39:00 +00:00
#include "SleepLib/schema.h"
2011-06-26 08:30:44 +00:00
extern QProgressBar * qprogress;
2011-06-26 08:30:44 +00:00
qint64 timezoneOffset() {
static bool ok=false;
static qint64 _TZ_offset=0;
if (ok) return _TZ_offset;
QDateTime d1=QDateTime::currentDateTime();
QDateTime d2=d1;
d1.setTimeSpec(Qt::UTC);
_TZ_offset=d2.secsTo(d1);
_TZ_offset*=1000L;
return _TZ_offset;
}
2011-06-26 08:30:44 +00:00
//////////////////////////////////////////////////////////////////////////////////////////
// Machine Base-Class implmementation
//////////////////////////////////////////////////////////////////////////////////////////
Machine::Machine(Profile *p,MachineID id)
{
day.clear();
highest_sessionid=0;
profile=p;
if (!id) {
std::tr1::minstd_rand gen;
std::tr1::uniform_int<MachineID> unif(1, 0x7fffffff);
gen.seed((unsigned int) time(NULL));
MachineID temp;
do {
temp = unif(gen); //unif(gen) << 32 |
} while (profile->machlist.find(temp)!=profile->machlist.end());
m_id=temp;
} else m_id=id;
2011-10-01 12:54:20 +00:00
//qDebug() << "Create Machine: " << hex << m_id; //%lx",m_id);
2011-06-26 08:30:44 +00:00
m_type=MT_UNKNOWN;
firstsession=true;
}
Machine::~Machine()
{
2011-07-01 10:10:44 +00:00
qDebug() << "Destroy Machine";
for (QMap<QDate,Day *>::iterator d=day.begin();d!=day.end();d++) {
delete d.value();
2011-06-26 08:30:44 +00:00
}
}
Session *Machine::SessionExists(SessionID session)
{
if (sessionlist.find(session)!=sessionlist.end()) {
return sessionlist[session];
} else {
return NULL;
}
}
2011-06-26 08:30:44 +00:00
Day *Machine::AddSession(Session *s,Profile *p)
{
if (!s) {
qWarning() << "Empty Session in Machine::AddSession()";
return NULL;
}
if (!p) {
qWarning() << "Empty Profile in Machine::AddSession()";
return NULL;
}
2011-06-26 08:30:44 +00:00
if (s->session()>highest_sessionid)
highest_sessionid=s->session();
2011-08-05 07:52:32 +00:00
QTime split_time(12,0,0);
if (PROFILE.Exists("DaySplitTime")) {
split_time=PROFILE["DaySplitTime"].toTime();
2011-08-05 07:52:32 +00:00
}
int combine_sessions;
if (PROFILE.Exists("CombineCloserSessions")) {
combine_sessions=PROFILE["CombineCloserSessions"].toInt(); // In Minutes
2011-08-05 07:52:32 +00:00
} else combine_sessions=0;
2011-08-05 07:52:32 +00:00
int ignore_sessions;
if (PROFILE.Exists("IgnoreShorterSessions")) {
ignore_sessions=PROFILE["IgnoreShorterSessions"].toInt(); // In Minutes
2011-08-05 07:52:32 +00:00
} else ignore_sessions=0;
2011-08-05 07:52:32 +00:00
int session_length=s->last()-s->first();
session_length/=60000;
2011-08-05 07:52:32 +00:00
sessionlist[s->session()]=s; // To make sure it get's saved later even if it's not wanted.
2011-07-03 08:08:14 +00:00
2011-09-17 12:39:00 +00:00
QDateTime d2=QDateTime::fromTime_t(s->first()/1000);
2011-06-26 08:30:44 +00:00
2011-08-05 07:52:32 +00:00
QDate date=d2.date();
QTime time=d2.time();
2011-06-26 08:30:44 +00:00
2011-08-05 07:52:32 +00:00
QMap<QDate,Day *>::iterator dit,nextday;
2011-06-26 08:30:44 +00:00
2011-08-05 07:52:32 +00:00
bool combine_next_day=false;
int closest_session=0;
2011-07-03 03:38:55 +00:00
2011-08-05 07:52:32 +00:00
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()) {
2011-09-01 13:07:26 +00:00
QDateTime lt=QDateTime::fromTime_t(dit.value()->last()/1000);
2011-08-05 07:52:32 +00:00
closest_session=lt.secsTo(d2)/60;
if (closest_session<combine_sessions) {
date=date.addDays(-1);
}
} else {
nextday=day.find(date.addDays(1));// Check Day Afterwards
if (nextday!=day.end()) {
2011-09-01 13:07:26 +00:00
QDateTime lt=QDateTime::fromTime_t(nextday.value()->first()/1000);
2011-08-05 07:52:32 +00:00
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;
}
2011-07-03 08:08:14 +00:00
}
}
2011-06-26 08:30:44 +00:00
}
if (session_length<ignore_sessions) {
//if (!closest_session || (closest_session>=60))
return NULL;
}
2011-06-26 08:30:44 +00:00
if (!firstsession) {
if (firstday>date) firstday=date;
if (lastday<date) lastday=date;
} else {
firstday=lastday=date;
firstsession=false;
}
2011-07-03 08:08:14 +00:00
Day *dd=NULL;
dit=day.find(date);
if (dit==day.end()) {
//QString dstr=date.toString("yyyyMMdd");
//qDebug("Adding Profile Day %s",dstr.toAscii().data());
dd=new Day(this);
day[date]=dd;
2011-06-26 08:30:44 +00:00
// Add this Day record to profile
p->AddDay(date,dd,m_type);
} else {
dd=*dit;
2011-06-26 08:30:44 +00:00
}
dd->AddSession(s);
2011-06-26 08:30:44 +00:00
2011-08-05 07:52:32 +00:00
if (combine_next_day) {
for (QVector<Session *>::iterator i=nextday.value()->begin();i!=nextday.value()->end();i++) {
dd->AddSession(*i);
}
QMap<QDate,QList<Day *> >::iterator nd=p->daylist.find(date.addDays(1));
for (QList<Day *>::iterator i=nd->begin();i!=nd->end();i++) {
2011-08-05 07:52:32 +00:00
if (*i==nextday.value()) {
nd.value().erase(i);
}
}
day.erase(nextday);
}
return dd;
2011-06-26 08:30:44 +00:00
}
// This functions purpose is murder and mayhem... It deletes all of a machines data.
// Therefore this is the most dangerous function in this software..
bool Machine::Purge(int secret)
{
// Boring api key to stop this function getting called by accident :)
if (secret!=3478216) return false;
// It would be joyous if this function screwed up..
QString path=profile->Get("DataFolder")+"/"+hexid();
QDir dir(path);
if (!dir.exists()) // It doesn't exist anyway.
return true;
if (!dir.isReadable())
2011-06-26 08:30:44 +00:00
return false;
2011-07-01 10:10:44 +00:00
qDebug() << "Purging " << QDir::toNativeSeparators(path);
2011-06-26 08:30:44 +00:00
dir.setFilter(QDir::Files | QDir::Hidden | QDir::NoSymLinks);
dir.setSorting(QDir::Name);
QFileInfoList list=dir.entryInfoList();
int could_not_kill=0;
for (int i=0;i<list.size();++i) {
QFileInfo fi=list.at(i);
QString fullpath=fi.canonicalFilePath();
//int j=fullpath.lastIndexOf(".");
QString ext_s=fullpath.section('.',-1);//right(j);
2011-06-26 08:30:44 +00:00
bool ok;
2011-07-01 10:10:44 +00:00
ext_s.toInt(&ok,10);
2011-06-26 08:30:44 +00:00
if (ok) {
qDebug() << "Deleting " << fullpath;
2011-06-26 08:30:44 +00:00
dir.remove(fullpath);
} else could_not_kill++;
}
if (could_not_kill>0) {
// qWarning() << "Could not purge path\n" << path << "\n\n" << could_not_kill << " file(s) remain.. Suggest manually deleting this path\n";
// return false;
2011-06-26 08:30:44 +00:00
}
return true;
}
const quint32 channel_version=1;
2011-06-26 08:30:44 +00:00
bool Machine::Load()
{
QString path=profile->Get("DataFolder")+"/"+hexid();
2011-06-26 08:30:44 +00:00
QDir dir(path);
2011-07-01 10:10:44 +00:00
qDebug() << "Loading " << path;
2011-06-26 08:30:44 +00:00
if (!dir.exists() || !dir.isReadable())
return false;
dir.setFilter(QDir::Files | QDir::Hidden | QDir::NoSymLinks);
dir.setSorting(QDir::Name);
QFileInfoList list=dir.entryInfoList();
typedef QVector<QString> StringList;
QMap<SessionID,StringList> sessfiles;
QMap<SessionID,StringList>::iterator s;
2011-06-26 08:30:44 +00:00
QString fullpath,ext_s,sesstr;
int ext;
SessionID sessid;
bool ok;
for (int i=0;i<list.size();i++) {
QFileInfo fi=list.at(i);
fullpath=fi.canonicalFilePath();
ext_s=fi.fileName().section(".",-1);
ext=ext_s.toInt(&ok,10);
if (!ok) continue;
sesstr=fi.fileName().section(".",0,-2);
sessid=sesstr.toLong(&ok,16);
if (!ok) continue;
if (sessfiles[sessid].capacity()==0) sessfiles[sessid].resize(3);
sessfiles[sessid][ext]=fi.canonicalFilePath();
}
2011-06-27 06:26:29 +00:00
int size=sessfiles.size();
int cnt=0;
2011-06-26 08:30:44 +00:00
for (s=sessfiles.begin(); s!=sessfiles.end(); s++) {
2011-06-27 07:45:59 +00:00
cnt++;
if ((cnt % 10)==0)
if (qprogress) qprogress->setValue((float(cnt)/float(size)*100.0));
Session *sess=new Session(this,s.key());
2011-06-26 08:30:44 +00:00
if (sess->LoadSummary(s.value()[0])) {
sess->SetEventFile(s.value()[1]);
AddSession(sess,profile);
} else {
2011-07-03 11:49:47 +00:00
qWarning() << "Error unpacking summary data";
2011-06-26 08:30:44 +00:00
delete sess;
}
}
2011-06-27 07:45:59 +00:00
if (qprogress) qprogress->setValue(100);
2011-06-26 08:30:44 +00:00
return true;
}
bool Machine::SaveSession(Session *sess)
{
QString path=profile->Get("DataFolder")+"/"+hexid();
if (sess->IsChanged()) sess->Store(path);
2011-07-01 10:10:44 +00:00
return true;
2011-06-26 08:30:44 +00:00
}
2011-06-26 08:30:44 +00:00
bool Machine::Save()
{
//int size;
2011-06-26 08:30:44 +00:00
int cnt=0;
QString path=profile->Get("DataFolder")+"/"+hexid();
QDir dir(path);
if (!dir.exists()) {
dir.mkdir(path);
}
2011-08-05 07:52:32 +00:00
QHash<SessionID,Session *>::iterator s;
m_savelist.clear();
2011-08-05 07:52:32 +00:00
for (s=sessionlist.begin(); s!=sessionlist.end(); s++) {
cnt++;
if ((*s)->IsChanged()) {
m_savelist.push_back(*s);
}
}
savelistCnt=0;
savelistSize=m_savelist.size();
if (!PROFILE["EnableMultithreading"].toBool()) {
for (int i=0;i<savelistSize;i++) {
qprogress->setValue(0+(float(savelistCnt)/float(savelistSize)*100.0));
QApplication::processEvents();
Session *s=m_savelist.at(i);
s->UpdateSummaries();
s->Store(path);
s->TrashEvents();
savelistCnt++;
}
return true;
}
int threads=QThread::idealThreadCount();
savelistSem=new QSemaphore(threads);
savelistSem->acquire(threads);
QVector<SaveThread*>thread;
for (int i=0;i<threads;i++) {
thread.push_back(new SaveThread(this,path));
QObject::connect(thread[i],SIGNAL(UpdateProgress(int)),qprogress,SLOT(setValue(int)));
thread[i]->start();
2011-06-26 08:30:44 +00:00
}
while (!savelistSem->tryAcquire(threads,250)) {
if (qprogress) {
// qprogress->setValue(66.0+(float(savelistCnt)/float(savelistSize)*33.0));
QApplication::processEvents();
}
}
for (int i=0;i<threads;i++) {
while (thread[i]->isRunning()) {
SaveThread::msleep(250);
QApplication::processEvents();
}
delete thread[i];
}
delete savelistSem;
2011-06-26 08:30:44 +00:00
return true;
}
/*SaveThread::SaveThread(Machine *m,QString p)
{
machine=m;
path=p;
} */
void SaveThread::run()
{
while (Session *sess=machine->popSaveList()) {
int i=(float(machine->savelistCnt)/float(machine->savelistSize)*100.0);
emit UpdateProgress(i);
sess->UpdateSummaries();
sess->Store(path);
sess->TrashEvents();
}
machine->savelistSem->release(1);
}
Session *Machine::popSaveList()
{
Session *sess=NULL;
savelistMutex.lock();
if (m_savelist.size()>0) {
sess=m_savelist.at(0);
m_savelist.pop_front();
savelistCnt++;
}
savelistMutex.unlock();
return sess;
}
2011-06-26 08:30:44 +00:00
//////////////////////////////////////////////////////////////////////////////////////////
// CPAP implmementation
//////////////////////////////////////////////////////////////////////////////////////////
CPAP::CPAP(Profile *p,MachineID id):Machine(p,id)
{
m_type=MT_CPAP;
}
CPAP::~CPAP()
{
}
//////////////////////////////////////////////////////////////////////////////////////////
// Oximeter Class implmementation
//////////////////////////////////////////////////////////////////////////////////////////
Oximeter::Oximeter(Profile *p,MachineID id):Machine(p,id)
{
m_type=MT_OXIMETER;
}
Oximeter::~Oximeter()
{
}
//////////////////////////////////////////////////////////////////////////////////////////
// SleepStage Class implmementation
//////////////////////////////////////////////////////////////////////////////////////////
SleepStage::SleepStage(Profile *p,MachineID id):Machine(p,id)
{
m_type=MT_SLEEPSTAGE;
}
SleepStage::~SleepStage()
{
}