/* SleepLib Session Header This stuff contains the base calculation smarts Copyright (c)2011 Mark Watkins License: GPL */ #ifndef SESSION_H #define SESSION_H #include #include #include #include "SleepLib/machine.h" #include "SleepLib/schema.h" #include "SleepLib/event.h" //class EventList; class Machine; /*! \class Session \brief Contains a single Sessions worth of machine event/waveform information. This class also contains all the primary database logic for SleepLib */ class Session { public: /*! \fn Session(Machine *,SessionID); \brief Create a session object belonging to Machine, with supplied SessionID If sessionID is 0, the next in sequence will be picked */ Session(Machine *,SessionID); virtual ~Session(); //! \brief Stores the session in the directory supplied by path bool Store(QString path); //! \brief Writes the Sessions Summary Indexes to filename, in SleepLibs custom data format. bool StoreSummary(QString filename); //! \brief Writes the Sessions EventLists to filename, in SleepLibs custom data format. bool StoreEvents(QString filename); //bool Load(QString path); //! \brief Loads the Sessions Summary Indexes from filename, from SleepLibs custom data format. bool LoadSummary(QString filename); //! \brief Loads the Sessions EventLists from filename, from SleepLibs custom data format. bool LoadEvents(QString filename); //! \brief Loads the events for this session when requested (only the summaries are loaded at startup) bool OpenEvents(); //! \brief Put the events away until needed again, freeing memory void TrashEvents(); //! \brief Search for Event code happening within dist milliseconds of supplied time (ms since epoch) bool SearchEvent(ChannelID code, qint64 time, qint64 dist=15000); //! \brief Return the sessionID const SessionID & session() { return s_session; } //! \brief Returns whether or not session is being used. bool enabled(); //! \brief Sets whether or not session is being used. void setEnabled(bool b); //! \brief Return the start of this sessions time range (in milliseconds since epoch) qint64 first() { return s_first; } //! \brief Return the end of this sessions time range (in milliseconds since epoch) qint64 last() { return s_last; } //! \brief Return the millisecond length of this session qint64 length() { return s_last-s_first; } //! \brief Set this Sessions ID (Not does not update indexes) void SetSessionID(SessionID s) { s_session=s; } //! \brief Moves all of this session data by offset d milliseconds void offsetSession(qint64 d); //! \brief Just set the start of the timerange without comparing void really_set_first(qint64 d) { s_first=d; } //! \brief Just set the end of the timerange without comparing void really_set_last(qint64 d) { s_last=d; } void set_first(qint64 d) { if (!s_first) s_first=d; else if (d > eventlist; //! \brief Sessions Settings List, contianing single settings for this session. QHash settings; // Session caches QHash m_cnt; QHash m_sum; QHash m_avg; QHash m_wavg; QHash m_min; QHash m_max; QHash m_cph; // Counts per hour (eg AHI) QHash m_sph; // % indice (eg % night in CSR) QHash m_firstchan; QHash m_lastchan; QHash > m_valuesummary; QHash > m_timesummary; QHash m_gain; //! \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,int 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 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; } int count(ChannelID id); //! \brief Returns the Count of all events of type id between time range int rangeCount(ChannelID id, qint64 first,qint64 last); //! \brief Returns the Sum of all events of type id between time range double rangeSum(ChannelID id, qint64 first,qint64 last); //! \brief Returns the minimum of events of type id between time range EventDataType rangeMin(ChannelID id, qint64 first,qint64 last); //! \brief Returns the maximum of events of type id between time range EventDataType rangeMax(ChannelID id, qint64 first,qint64 last); //! \brief Returns (and caches) the Sum of all events of type id double sum(ChannelID id); //! \brief Returns (and caches) the Average of all events of type id EventDataType avg(ChannelID id); //! \brief Returns (and caches) the Time Weighted Average of all events of type id EventDataType wavg(ChannelID i); //! \brief Returns (and caches) the Minimum of all events of type id EventDataType Min(ChannelID id); //! \brief Returns (and caches) the Maximum of all events of type id EventDataType Max(ChannelID id); //! \brief Returns (and caches) the 90th Percentile of all events of type id EventDataType p90(ChannelID id); //! \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); //! \brief Returns (and caches) the Count-Per-Hour of all events of type id EventDataType cph(ChannelID id); //! \brief Returns (and caches) the Sum-Per-Hour of all events of type id EventDataType sph(ChannelID id); //! \brief Returns (without caching) the requested Percentile of all events of type id EventDataType percentile(ChannelID id,EventDataType percentile); //! \brief Returns true if the channel has events loaded, or a record of a count for when they are not bool channelExists(ChannelID name); //! \brief Returns true if the channel has event data available (must be loaded first) bool channelDataExists(ChannelID id); bool IsLoneSession() { return s_lonesession; } void SetLoneSession(bool b) { s_lonesession=b; } bool eventsLoaded() { return s_events_loaded; } //! \brief Sets the event file linked to the summary (during load, for ondemand loading) void SetEventFile(QString & filename) { s_eventfile=filename; } //! \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; } //! \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