OSCAR-code/oscar/SleepLib/session.h

465 lines
17 KiB
C
Raw Permalink Normal View History

/* SleepLib Session Header
*
2014-08-17 12:56:05 +00:00
* This stuff contains the session calculation smarts
*
* 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. */
2011-06-26 08:30:44 +00:00
#ifndef SESSION_H
#define SESSION_H
// this is added as DEFINES += SESSION_DEBUG in the Qmake line
// to see how the Resmed loader assigns files to sessions
// #define SESSION_DEBUG
#include <QDebug>
#include <QHash>
#include <QVector>
2011-06-26 08:30:44 +00:00
2011-07-27 09:21:53 +00:00
#include "SleepLib/machine.h"
2011-09-17 12:39:00 +00:00
#include "SleepLib/schema.h"
2011-07-27 09:21:53 +00:00
#include "SleepLib/event.h"
2011-09-17 12:39:00 +00:00
//class EventList;
2011-06-26 08:30:44 +00:00
class Machine;
2011-07-27 09:21:53 +00:00
enum SliceStatus {
UnknownStatus=0, EquipmentOff, MaskOn, MaskOff // is there an EquipmentOn?
};
class SessionSlice
{
public:
SessionSlice() {
start = end = 0;
status = UnknownStatus;
}
SessionSlice(const SessionSlice & copy) {
start = copy.start;
end = copy.end;
status = copy.status;
}
SessionSlice& operator=(const SessionSlice& other) = default;
SessionSlice(qint64 start, qint64 end, SliceStatus status):start(start), end(end), status(status) {}
qint64 start;
qint64 end;
SliceStatus status;
};
2011-12-18 13:20:01 +00:00
/*! \class Session
\brief Contains a single Sessions worth of device event/waveform information.
2011-12-18 13:20:01 +00:00
This class also contains all the primary database logic for SleepLib
*/
2011-06-26 08:30:44 +00:00
class Session
{
friend class Day;
friend class Machine;
public:
2011-12-18 13:20:01 +00:00
/*! \fn Session(Machine *,SessionID);
\brief Create a session object belonging to device, with supplied SessionID
2011-12-18 13:20:01 +00:00
If sessionID is 0, the next in sequence will be picked
*/
Session(Machine *, SessionID);
2011-06-26 08:30:44 +00:00
virtual ~Session();
//! \brief Checks whether the supplied time is within the bounds of this session (ms since epoch)
inline bool checkInside(qint64 time) {
return ((time >= s_first) && (time <= s_last));
}
2011-12-18 13:20:01 +00:00
//! \brief Stores the session in the directory supplied by path
2011-06-26 08:30:44 +00:00
bool Store(QString path);
2011-12-18 13:20:01 +00:00
//! \brief Writes the Sessions Summary Indexes to filename, in SleepLibs custom data format.
2014-09-04 14:59:54 +00:00
bool StoreSummary();
// //! \brief Save the Sessions Summary Indexes to the stream
// void StoreSummaryData(QDataStream & out) const;
2011-12-18 13:20:01 +00:00
//! \brief Writes the Sessions EventLists to filename, in SleepLibs custom data format.
2014-09-04 14:59:54 +00:00
bool StoreEvents();
2011-12-18 13:20:01 +00:00
2011-06-26 08:30:44 +00:00
//bool Load(QString path);
2011-12-18 13:20:01 +00:00
// //! \brief Loads the Sessions Summary Indexes from stream
// void LoadSummaryData(QDataStream & in);
2014-09-04 14:59:54 +00:00
2011-12-18 13:20:01 +00:00
//! \brief Loads the Sessions Summary Indexes from filename, from SleepLibs custom data format.
// debug option is set to false because file errors are normal when there is summary but no details data
bool LoadSummary(bool debug=false);
2011-12-18 13:20:01 +00:00
//! \brief Loads the Sessions EventLists from filename, from SleepLibs custom data format.
// debug option is set to false because file errors are normal when there is summary but no details data
bool LoadEvents(QString filename, bool debug=false);
2011-06-26 08:30:44 +00:00
2011-12-18 13:20:01 +00:00
//! \brief Loads the events for this session when requested (only the summaries are loaded at startup)
// debug option is set to false because file errors are normal when there is summary but no details data
bool OpenEvents(bool debug=false);
2011-12-18 13:20:01 +00:00
//! \brief Put the events away until needed again, freeing memory
2011-06-26 08:30:44 +00:00
void TrashEvents();
//! \brief Returns true if session contains an empty duration
inline bool isEmpty() { return (s_first == s_last); }
//! \brief Search for Event code happening at supplied time (ms since epoch)
EventDataType SearchValue(ChannelID code, qint64 time, bool square);
2011-12-18 13:20:01 +00:00
//! \brief Return the sessionID
inline const SessionID &session() {
2011-06-26 08:30:44 +00:00
return s_session;
}
2011-12-18 13:20:01 +00:00
2011-12-28 12:03:48 +00:00
//! \brief Returns whether or not session is being used.
2023-05-26 21:29:56 +00:00
bool enabled(bool realValues=false) const;
2011-12-28 12:03:48 +00:00
//! \brief Sets whether or not session is being used.
void setEnabled(bool b);
2011-12-28 12:03:48 +00:00
//! \brief Return the earliest time in session (in milliseconds since epoch)
2014-09-05 14:46:42 +00:00
inline qint64 realFirst() const { return s_first; }
//! \brief Return the latest time in session (in milliseconds since epoch)
2014-09-05 14:46:42 +00:00
inline qint64 realLast() const { return s_last; }
//! \brief Return the start of this sessions time range, adjusted for clock drift (in milliseconds since epoch)
qint64 first();
2011-12-18 13:20:01 +00:00
//! \brief Return the end of this sessions time range, adjusted for clock drift (in milliseconds since epoch)
qint64 last();
2011-12-18 13:20:01 +00:00
//! \brief Return the millisecond length of this session
qint64 length() {
qint64 duration=s_last - s_first;
return duration<0?0:duration;
// qint64 t;
// int size = m_slices.size();
// if (size == 0) {
// t = (s_last - s_first);
// } else {
// t = 0;
// for (int i=0; i<size; ++i) {
// const SessionSlice & slice = m_slices.at(i);
// if (slice.status == MaskOn) {
// t += slice.end - slice.start;
// }
// }
// }
// return t;
}
2011-12-18 13:20:01 +00:00
//! \brief Set this Sessions ID (Not does not update indexes)
2011-06-26 08:30:44 +00:00
void SetSessionID(SessionID s) {
s_session = s;
}
2011-12-18 13:20:01 +00:00
//! \brief Moves all of this session data by offset d milliseconds
void offsetSession(qint64 d);
2011-12-18 13:20:01 +00:00
//! \brief Just set the start of the timerange without comparing
void really_set_first(qint64 d) { s_first = d; }
2011-12-18 13:20:01 +00:00
//! \brief Just set the end of the timerange without comparing
void really_set_last(qint64 d) { s_last = d; }
2011-12-18 13:20:01 +00:00
//! \brief Set time to lower of 'd' and existing s_first
void set_first(qint64 d) {
if (!s_first) { s_first = d; }
else if (d < s_first) { s_first = d; }
}
//! \brief Set last time to higher of 'd' and existing s_last. Throw warning if 'd' less than s_first.
void set_last(qint64 d) {
if (d <= s_first) {
qWarning() << s_session << "Session::set_last() d<=s_first, d" << d << "s_first" << s_first;
return;
}
if (!s_last) { s_last = d; }
else if (s_last < d) { s_last = d; }
}
2011-06-26 08:30:44 +00:00
2011-12-18 13:20:01 +00:00
//! \brief Return Session Length in decimal hours
2011-07-03 02:43:50 +00:00
double hours() {
double t;
int size = m_slices.size();
if (size == 0) {
t = (s_last - s_first) / 3600000.0;
} else {
t = 0;
for (int i=0; i<size; ++i) {
const SessionSlice & slice = m_slices.at(i);
if (slice.status == MaskOn) {
t += slice.end - slice.start;
}
}
t = t / 3600000.0;
}
return t;
}
2011-06-26 08:30:44 +00:00
//! \brief Flag this Session as dirty, so device object can save it
2011-06-26 08:30:44 +00:00
void SetChanged(bool val) {
s_changed = val;
s_events_loaded = val; // dirty hack putting this here
}
2011-12-18 13:20:01 +00:00
//! \brief Return this Sessions dirty status
2011-06-26 08:30:44 +00:00
bool IsChanged() {
return s_changed;
}
2011-06-26 08:30:44 +00:00
//! \brief Return the unit type string used by events of supplied channel
QString dimension(ChannelID);
2011-12-18 13:20:01 +00:00
//! \brief Contains all the EventLists, indexed by ChannelID
QHash<ChannelID, QVector<EventList *> > eventlist;
2011-12-18 13:20:01 +00:00
//! \brief Sessions Settings List, contianing single settings for this session.
QHash<ChannelID, QVariant> settings;
2011-12-18 13:20:01 +00:00
// Session caches
QHash<ChannelID, EventDataType> m_cnt;
QHash<ChannelID, double> m_sum;
QHash<ChannelID, EventDataType> m_avg;
QHash<ChannelID, EventDataType> m_wavg;
QHash<ChannelID, EventDataType> m_min; // The actual minimum
QHash<ChannelID, EventDataType> m_max;
// This could go in channels, but different devices interpret it differently
// Under the new SleepyLib data Device model this can be done, but unfortunately not here..
QHash<ChannelID, EventDataType> m_physmin; // The physical minimum for graph display purposes
QHash<ChannelID, EventDataType> m_physmax; // The physical maximum
QHash<ChannelID, EventDataType> m_cph; // Counts per hour (eg AHI)
QHash<ChannelID, EventDataType> m_sph; // % indice (eg % night in CSR)
QHash<ChannelID, quint64> m_firstchan;
QHash<ChannelID, quint64> m_lastchan;
QHash<ChannelID, QHash<EventStoreType, EventStoreType> > m_valuesummary;
QHash<ChannelID, QHash<EventStoreType, quint32> > m_timesummary;
QHash<ChannelID, EventDataType> m_gain;
QHash<ChannelID, EventDataType> m_lowerThreshold;
QHash<ChannelID, EventDataType> m_timeBelowTheshold;
QHash<ChannelID, EventDataType> m_upperThreshold;
QHash<ChannelID, EventDataType> m_timeAboveTheshold;
QList<ChannelID> m_availableChannels;
QList<ChannelID> m_availableSettings;
QVector<SessionSlice> m_slices;
//! \brief Generates sum and time data for each distinct value in 'code' events..
void updateCountSummary(ChannelID code);
//! \brief Destroy any trace of event 'code', freeing any memory if loaded.
void destroyEvent(ChannelID code);
// UpdateSummaries may recalculate all these, but it may be faster setting upfront
void setCount(ChannelID id, EventDataType val) { m_cnt[id] = val; }
void setSum(ChannelID id, EventDataType val) { m_sum[id] = val; }
void setMin(ChannelID id, EventDataType val) { m_min[id] = val; }
void setMax(ChannelID id, EventDataType val) { m_max[id] = val; }
void setPhysMin(ChannelID id, EventDataType val) { m_physmin[id] = val; }
void setPhysMax(ChannelID id, EventDataType val) { m_physmax[id] = val; }
void updateMin(ChannelID id, EventDataType val) {
QHash<ChannelID, EventDataType>::iterator i = m_min.find(id);
if (i == m_min.end()) {
m_min[id] = val;
} else if (i.value() > val) {
i.value() = val;
}
}
void updateMax(ChannelID id, EventDataType val) {
QHash<ChannelID, EventDataType>::iterator i = m_max.find(id);
if (i == m_max.end()) {
m_max[id] = val;
} else if (i.value() < val) {
i.value() = val;
}
}
void setAvg(ChannelID id, EventDataType val) { m_avg[id] = val; }
void setWavg(ChannelID id, EventDataType val) { m_wavg[id] = val; }
// void setMedian(ChannelID id,EventDataType val) { m_med[id]=val; }
// void set90p(ChannelID id,EventDataType val) { m_90p[id]=val; }
// void set95p(ChannelID id,EventDataType val) { m_95p[id]=val; }
void setCph(ChannelID id, EventDataType val) { m_cph[id] = val; }
void setSph(ChannelID id, EventDataType val) { m_sph[id] = val; }
void setFirst(ChannelID id, qint64 val) { m_firstchan[id] = val; }
void setLast(ChannelID id, qint64 val) { m_lastchan[id] = val; }
EventDataType count(ChannelID id);
2011-12-18 13:20:01 +00:00
//! \brief Returns the Count of all events of type id between time range
EventDataType rangeCount(ChannelID id, qint64 first, qint64 last);
2011-12-18 13:20:01 +00:00
//! \brief Returns the Sum of all events of type id between time range
double rangeSum(ChannelID id, qint64 first, qint64 last);
2011-12-18 13:20:01 +00:00
//! \brief Returns the minimum of events of type id between time range
EventDataType rangeMin(ChannelID id, qint64 first, qint64 last);
2011-12-18 13:20:01 +00:00
//! \brief Returns the maximum of events of type id between time range
EventDataType rangeMax(ChannelID id, qint64 first, qint64 last);
//! \brief Returns the count of code events inside span flag event durations
EventDataType countInsideSpan(ChannelID span, ChannelID code);
2011-12-18 13:20:01 +00:00
//! \brief Returns (and caches) the Sum of all events of type id
double sum(ChannelID id);
2011-12-18 13:20:01 +00:00
//! \brief Returns (and caches) the Average of all events of type id
EventDataType avg(ChannelID id);
2011-12-18 13:20:01 +00:00
//! \brief Returns (and caches) the Time Weighted Average of all events of type id
EventDataType wavg(ChannelID i);
2011-12-18 13:20:01 +00:00
//! \brief Returns (and caches) the Minimum of all events of type id
2011-12-17 14:38:15 +00:00
EventDataType Min(ChannelID id);
2011-12-18 13:20:01 +00:00
//! \brief Returns (and caches) the Maximum of all events of type id
2011-12-17 14:38:15 +00:00
EventDataType Max(ChannelID id);
2011-12-18 13:20:01 +00:00
//! \brief Returns (and caches) the Minimum of all events of type id
EventDataType physMin(ChannelID id);
//! \brief Returns (and caches) the Maximum of all events of type id
EventDataType physMax(ChannelID id);
2011-12-18 13:20:01 +00:00
//! \brief Returns (and caches) the 90th Percentile of all events of type id
EventDataType p90(ChannelID id);
2011-12-18 13:20:01 +00:00
//! \brief Returns (and caches) the 95th Percentile of all events of type id
EventDataType p95(ChannelID id);
//! \brief Returns (and caches) the Median (50th Perc) of all events of type id
EventDataType median(ChannelID id);
2011-12-18 13:20:01 +00:00
//! \brief Returns (and caches) the Count-Per-Hour of all events of type id
EventDataType cph(ChannelID id);
2011-12-18 13:20:01 +00:00
//! \brief Returns (and caches) the Sum-Per-Hour of all events of type id
EventDataType sph(ChannelID id);
2011-12-18 13:20:01 +00:00
//! \brief Returns (without caching) the requested Percentile of all events of type id
EventDataType percentile(ChannelID id, EventDataType percentile);
//! \brief Returns the amount of time (in decimal minutes) the Channel spent above the threshold
EventDataType timeAboveThreshold(ChannelID id, EventDataType threshold);
//! \brief Returns the amount of time (in decimal minutes) the Channel spent below the threshold
EventDataType timeBelowThreshold(ChannelID id, EventDataType threshold);
//! \brief According to preferences..
EventDataType calcMiddle(ChannelID code);
EventDataType calcMax(ChannelID code);
EventDataType calcPercentile(ChannelID code);
2023-05-26 21:29:56 +00:00
2011-12-18 13:20:01 +00:00
//! \brief Returns true if the channel has events loaded, or a record of a count for when they are not
bool channelExists(ChannelID name);
2011-07-27 09:21:53 +00:00
//! \brief Returns true if the channel has event data available (must be loaded first)
bool channelDataExists(ChannelID id);
2011-07-27 09:21:53 +00:00
bool IsLoneSession() { return s_lonesession; }
void SetLoneSession(bool b) { s_lonesession = b; }
2011-12-18 13:20:01 +00:00
bool eventsLoaded() { return s_events_loaded; }
2011-12-18 13:20:01 +00:00
//! \brief Update this sessions first time if it's less than the current record
inline void updateFirst(qint64 v) { if (!s_first) { s_first = v; } else if (s_first > v) { s_first = v; } }
2011-12-18 13:20:01 +00:00
//! \brief Update this sessions latest time if it's more than the current record
inline void updateLast(qint64 v) { if (!s_last) { s_last = v; } else if (s_last < v) { s_last = v; } }
2011-12-18 13:20:01 +00:00
//! \brief Returns (and caches) the first time for Channel code
qint64 first(ChannelID code);
2011-12-18 13:20:01 +00:00
//! \brief Returns (and caches) the last time for Channel code
qint64 last(ChannelID code);
2011-12-18 13:20:01 +00:00
//! \brief Regenerates the Session Index Caches, and calls the fun calculation functions
void UpdateSummaries();
2011-12-18 13:20:01 +00:00
//! \brief Creates and returns a new EventList for the supplied Channel code
EventList *AddEventList(ChannelID code, EventListType et, EventDataType gain = 1.0,
EventDataType offset = 0.0, EventDataType min = 0.0, EventDataType max = 0.0,
EventDataType rate = 0.0, bool second_field = false);
2011-12-18 13:20:01 +00:00
//! \brief Returns this sessions DeviceID
Machine *machine() { return s_machine; }
2014-05-19 03:46:02 +00:00
//! \brief Returns true if session only contains summary data
inline bool summaryOnly() { return s_summaryOnly; }
//! \brief Returns true if there are no settings for this session
inline bool noSettings() { return s_noSettings; }
//! \brief Mark this session as containing summary data only or not (true, false)
inline void setSummaryOnly(bool b) { s_summaryOnly = b; }
//! \brief Mark this ssession as having settings data or not (true, false)
inline void setNoSettings(bool b) { s_noSettings = b; }
2014-05-19 03:46:02 +00:00
//! \brief Mark whether this session has loaded data (true, false)
void setOpened(bool b = true) {
s_events_loaded = b;
s_summary_loaded = b;
}
2014-05-19 03:46:02 +00:00
//! \brief Completely purges Session from memory and disk.
bool Destroy();
2014-09-04 14:59:54 +00:00
QString eventFile() const;
//! \brief Returns DeviceType for this session
MachineType type() { return s_machtype; }
2023-05-26 21:29:56 +00:00
#ifdef SESSION_DEBUG
QStringList session_files;
#endif
protected:
2011-06-26 08:30:44 +00:00
SessionID s_session;
Machine *s_machine;
//! \brief Time session begins (in ms since epoch)
qint64 s_first;
//! \brief Time session ends (in ms since epoch)
qint64 s_last;
2011-06-26 08:30:44 +00:00
bool s_changed;
bool s_lonesession;
2012-01-05 11:35:23 +00:00
bool s_evchecksum_checked;
2011-06-26 08:30:44 +00:00
bool _first_session;
bool s_summaryOnly;
//! \brief True if there are no settings for this session
bool s_noSettings;
2011-06-26 08:30:44 +00:00
bool s_summary_loaded;
2011-06-26 08:30:44 +00:00
bool s_events_loaded;
2023-05-26 21:29:56 +00:00
quint8 s_enabled;
// for debugging
bool destroyed;
MachineType s_machtype;
2011-06-26 08:30:44 +00:00
};
2014-09-04 14:59:54 +00:00
QDataStream & operator<<(QDataStream & out, const Session & session);
QDataStream & operator>>(QDataStream & in, Session & session);
2011-06-26 08:30:44 +00:00
#endif // SESSION_H