2020-01-23 17:51:58 +00:00
/* SleepLib Viatom Loader Implementation
*
2020-01-24 00:25:06 +00:00
* Copyright ( c ) 2019 - 2020 The OSCAR Team
* ( Initial importer written by dave madden < dhm @ mersenne . com > )
2021-02-22 20:30:04 +00:00
* Modified 02 / 21 / 2021 to allow for CheckMe device data files by Crimson Nape < CrimsonNape @ gmail . com >
2020-01-23 17:51:58 +00:00
*
* 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 . */
//********************************************************************************************
2021-03-22 22:43:14 +00:00
// Please only INCREMENT the viatom_data_version in viatom_loader.h when making changes
// that change loader behaviour or modify channels in a manner that fixes old data imports.
// Note that changing the data version will require a reimport of existing data for which OSCAR
// does not keep a backup - so it should be avoided if possible.
// i.e. there is no need to change the version when adding support for new devices
2020-01-23 17:51:58 +00:00
//********************************************************************************************
# include <QDir>
# include <QTextStream>
2020-01-25 01:10:41 +00:00
# include <QApplication>
# include <QMessageBox>
2020-01-23 17:51:58 +00:00
# include "viatom_loader.h"
# include "SleepLib/machine.h"
2020-01-25 01:10:41 +00:00
static QSet < QString > s_unexpectedMessages ;
2020-01-23 17:51:58 +00:00
bool
ViatomLoader : : Detect ( const QString & path )
{
2020-01-26 23:28:10 +00:00
// This is only used for CPAP machines, when detecting CPAP cards.
2020-01-24 00:25:06 +00:00
qDebug ( ) < < " ViatomLoader::Detect( " < < path < < " ) " ;
2020-01-26 23:28:10 +00:00
return false ;
2020-01-23 17:51:58 +00:00
}
int
2020-01-24 00:25:06 +00:00
ViatomLoader : : Open ( const QString & dirpath )
2020-01-23 17:51:58 +00:00
{
2020-01-24 00:25:06 +00:00
qDebug ( ) < < " ViatomLoader::Open( " < < dirpath < < " ) " ;
2020-04-26 18:06:48 +00:00
m_mach = nullptr ;
2020-01-26 23:13:09 +00:00
int imported = 0 ;
int found = 0 ;
2020-01-26 23:28:10 +00:00
s_unexpectedMessages . clear ( ) ;
2021-02-22 20:30:04 +00:00
2020-01-26 23:13:09 +00:00
if ( QFileInfo ( dirpath ) . isDir ( ) ) {
QDir dir ( dirpath ) ;
dir . setFilter ( QDir : : NoDotAndDotDot | QDir : : Files | QDir : : Hidden ) ;
dir . setNameFilters ( getNameFilter ( ) ) ;
dir . setSorting ( QDir : : Name ) ;
for ( auto & fi : dir . entryInfoList ( ) ) {
2020-04-26 18:06:48 +00:00
if ( OpenFile ( fi . canonicalFilePath ( ) ) ) {
imported + + ;
}
2020-01-26 23:13:09 +00:00
found + + ;
}
}
else {
// This filename has already been filtered by QFileDialog.
2020-04-26 18:06:48 +00:00
if ( OpenFile ( dirpath ) ) {
imported + + ;
}
2020-01-26 23:13:09 +00:00
found + + ;
}
if ( ! found ) {
return - 1 ;
}
2020-04-26 18:06:48 +00:00
Machine * mach = m_mach ;
if ( imported & & mach = = nullptr ) qWarning ( ) < < " No machine record created? " ;
2020-01-27 00:50:18 +00:00
if ( mach ) {
qDebug ( ) < < " Imported " < < imported < < " sessions " ;
mach - > Save ( ) ;
mach - > SaveSummaryCache ( ) ;
p_profile - > StoreMachines ( ) ;
}
2020-01-26 23:28:10 +00:00
if ( mach & & s_unexpectedMessages . count ( ) > 0 & & p_profile - > session - > warnOnUnexpectedData ( ) ) {
// Compare this to the list of messages previously seen for this machine
// and only alert if there are new ones.
QSet < QString > newMessages = s_unexpectedMessages - mach - > previouslySeenUnexpectedData ( ) ;
if ( newMessages . count ( ) > 0 ) {
// TODO: Rework the importer call structure so that this can become an
// emit statement to the appropriate import job.
QMessageBox : : information ( QApplication : : activeWindow ( ) ,
QObject : : tr ( " Untested Data " ) ,
QObject : : tr ( " Your Viatom device generated data that OSCAR has never seen before. " ) + " \n \n " +
2020-01-27 14:15:15 +00:00
QObject : : tr ( " The imported data may not be entirely accurate, so the developers would like a copy of your Viatom files to make sure OSCAR is handling the data correctly. " )
2020-01-26 23:28:10 +00:00
, QMessageBox : : Ok ) ;
mach - > previouslySeenUnexpectedData ( ) + = newMessages ;
}
}
2020-01-26 23:13:09 +00:00
return imported ;
2020-01-23 17:51:58 +00:00
}
2020-04-26 18:06:48 +00:00
bool ViatomLoader : : OpenFile ( const QString & filename )
2020-01-23 17:51:58 +00:00
{
2020-01-26 23:28:10 +00:00
Machine * mach = nullptr ;
2021-02-22 20:30:04 +00:00
2020-01-24 00:11:05 +00:00
Session * sess = ParseFile ( filename ) ;
2020-01-26 23:13:09 +00:00
if ( sess ) {
2020-01-24 00:11:05 +00:00
SaveSessionToDatabase ( sess ) ;
2020-01-26 23:28:10 +00:00
mach = sess - > machine ( ) ;
2020-04-26 18:06:48 +00:00
m_mach = mach ;
2020-01-24 00:11:05 +00:00
}
2020-01-25 01:10:41 +00:00
2020-04-26 18:06:48 +00:00
return mach ! = nullptr ;
2020-01-24 00:11:05 +00:00
}
Session * ViatomLoader : : ParseFile ( const QString & filename )
{
2020-01-24 00:25:06 +00:00
QFile file ( filename ) ;
if ( ! file . open ( QFile : : ReadOnly ) ) {
qDebug ( ) < < " Couldn't open Viatom data file " < < filename ;
return nullptr ;
2020-01-23 17:51:58 +00:00
}
2020-01-24 20:20:00 +00:00
ViatomFile v ( file ) ;
if ( v . ParseHeader ( ) = = false ) {
2020-01-24 00:25:06 +00:00
return nullptr ;
}
2020-01-27 17:26:35 +00:00
MachineInfo info = newInfo ( ) ;
// Check whether the enclosing folder looks like a Viatom serial number, and if so, use it.
QString foldername = QFileInfo ( filename ) . dir ( ) . dirName ( ) ;
if ( foldername . length ( ) > = 9 ) {
bool numeric ;
2021-02-22 20:30:04 +00:00
foldername . rightRef ( 4 ) . toInt ( & numeric ) ;
2020-01-27 17:26:35 +00:00
if ( numeric ) {
info . serial = foldername ;
}
}
Machine * mach = p_profile - > CreateMachine ( info ) ;
2020-01-24 00:25:06 +00:00
2020-01-26 22:20:58 +00:00
if ( mach - > SessionExists ( v . sessionid ( ) ) ) {
// Skip already imported session
2020-01-27 00:50:18 +00:00
//qDebug() << filename << "session already exists, skipping" << v.sessionid();
2020-01-26 22:20:58 +00:00
return nullptr ;
2020-01-24 00:25:06 +00:00
}
2020-01-26 22:20:58 +00:00
2020-01-27 17:26:35 +00:00
qint64 time_ms = v . timestamp ( ) ;
2020-01-26 22:20:58 +00:00
m_session = new Session ( mach , v . sessionid ( ) ) ;
m_session - > set_first ( time_ms ) ;
2020-01-24 00:25:06 +00:00
2020-01-25 01:10:41 +00:00
QList < ViatomFile : : Record > records = v . ReadData ( ) ;
2020-01-25 21:55:32 +00:00
m_step = v . duration ( ) / records . size ( ) * 1000L ;
2020-01-23 17:51:58 +00:00
2020-01-24 20:20:00 +00:00
// Import data
for ( auto & rec : records ) {
2020-01-25 21:55:32 +00:00
if ( rec . oximetry_invalid ) {
EndEventList ( OXI_Pulse , time_ms ) ;
EndEventList ( OXI_SPO2 , time_ms ) ;
} else {
AddEvent ( OXI_Pulse , time_ms , rec . hr ) ;
AddEvent ( OXI_SPO2 , time_ms , rec . spo2 ) ;
}
2020-01-27 00:50:18 +00:00
AddEvent ( POS_Movement , time_ms , rec . motion ) ;
2020-01-25 21:55:32 +00:00
time_ms + = m_step ;
2020-01-24 20:20:00 +00:00
}
2020-01-25 21:55:32 +00:00
EndEventList ( OXI_Pulse , time_ms ) ;
EndEventList ( OXI_SPO2 , time_ms ) ;
2020-01-27 00:50:18 +00:00
EndEventList ( POS_Movement , time_ms ) ;
2020-01-25 21:55:32 +00:00
m_session - > set_last ( time_ms ) ;
2020-01-23 17:51:58 +00:00
2020-01-25 21:55:32 +00:00
return m_session ;
2020-01-24 00:11:05 +00:00
}
void ViatomLoader : : SaveSessionToDatabase ( Session * sess )
{
Machine * mach = sess - > machine ( ) ;
2021-02-22 20:30:04 +00:00
2020-01-24 00:25:06 +00:00
sess - > SetChanged ( true ) ;
mach - > AddSession ( sess ) ;
2020-01-23 17:51:58 +00:00
}
2020-01-25 21:55:32 +00:00
void ViatomLoader : : AddEvent ( ChannelID channel , qint64 t , EventDataType value )
{
EventList * C = m_importChannels [ channel ] ;
if ( C = = nullptr ) {
C = m_session - > AddEventList ( channel , EVL_Waveform , 1.0 , 0.0 , 0.0 , 0.0 , m_step ) ;
Q_ASSERT ( C ) ; // Once upon a time AddEventList could return nullptr, but not any more.
m_importChannels [ channel ] = C ;
}
// Add the event
C - > AddEvent ( t , value ) ;
m_importLastValue [ channel ] = value ;
}
void ViatomLoader : : EndEventList ( ChannelID channel , qint64 /*t*/ )
{
EventList * C = m_importChannels [ channel ] ;
if ( C ! = nullptr ) {
// The below would be needed for square charts if the first sample represents
// the 4 seconds following the starting timestamp:
//C->AddEvent(t, m_importLastValue[channel]);
2021-02-22 20:30:04 +00:00
2020-01-25 21:55:32 +00:00
// Mark this channel's event list as ended.
m_importChannels [ channel ] = nullptr ;
}
}
2020-01-26 22:51:56 +00:00
QStringList ViatomLoader : : getNameFilter ( )
{
return QStringList ( " 20[0-5][0-9][01][0-9][0-3][0-9][012][0-9][0-5][0-9][0-5][0-9] " ) ;
}
2020-01-23 17:51:58 +00:00
static bool viatom_initialized = false ;
void
2020-01-24 00:25:06 +00:00
ViatomLoader : : Register ( )
2020-01-23 17:51:58 +00:00
{
if ( ! viatom_initialized ) {
2020-01-24 00:25:06 +00:00
qDebug ( " Registering ViatomLoader " ) ;
RegisterLoader ( new ViatomLoader ( ) ) ;
//InitModelMap();
viatom_initialized = true ;
}
2020-01-23 17:51:58 +00:00
}
2020-01-24 20:20:00 +00:00
// ===============================================================================================
2020-01-26 22:20:58 +00:00
/*
2020-01-25 01:10:41 +00:00
static QString ts ( qint64 msecs )
2020-01-24 20:20:00 +00:00
{
2020-01-25 01:10:41 +00:00
// TODO: make this UTC so that tests don't vary by where they're run
return QDateTime : : fromMSecsSinceEpoch ( msecs ) . toString ( Qt : : ISODate ) ;
2020-01-24 20:20:00 +00:00
}
2020-01-25 01:10:41 +00:00
static QString dur ( qint64 msecs )
2020-01-24 20:20:00 +00:00
{
2020-01-25 01:10:41 +00:00
qint64 s = msecs / 1000L ;
int h = s / 3600 ; s - = h * 3600 ;
int m = s / 60 ; s - = m * 60 ;
return QString ( " %1:%2:%3 " )
. arg ( h , 2 , 10 , QChar ( ' 0 ' ) )
. arg ( m , 2 , 10 , QChar ( ' 0 ' ) )
. arg ( s , 2 , 10 , QChar ( ' 0 ' ) ) ;
}
2020-01-26 22:20:58 +00:00
*/
2020-01-24 20:20:00 +00:00
2020-01-25 01:10:41 +00:00
// TODO: Merge this with PRS1 macros and generalize for all loaders.
# define UNEXPECTED_VALUE(SRC, VALS) { \
QString message = QString ( " %1:%2: %3 = %4 != %5 " ) . arg ( __func__ ) . arg ( __LINE__ ) . arg ( # SRC ) . arg ( SRC ) . arg ( VALS ) ; \
qWarning ( ) < < this - > m_sessionid < < message ; \
s_unexpectedMessages + = message ; \
}
# define CHECK_VALUE(SRC, VAL) if ((SRC) != (VAL)) UNEXPECTED_VALUE(SRC, VAL)
# define CHECK_VALUES(SRC, VAL1, VAL2) if ((SRC) != (VAL1) && (SRC) != (VAL2)) UNEXPECTED_VALUE(SRC, #VAL1 " or " #VAL2)
// for more than 2 values, just write the test manually and use UNEXPECTED_VALUE if it fails
2020-01-24 20:20:00 +00:00
2021-02-22 20:30:04 +00:00
2020-01-25 01:10:41 +00:00
ViatomFile : : ViatomFile ( QFile & file ) : m_file ( file )
{
}
2020-01-24 20:20:00 +00:00
2020-01-25 01:10:41 +00:00
bool ViatomFile : : ParseHeader ( )
{
static const int HEADER_SIZE = 40 ;
QByteArray data = m_file . read ( HEADER_SIZE ) ;
if ( data . size ( ) < HEADER_SIZE ) {
qDebug ( ) < < m_file . fileName ( ) < < " too short for a Viatom data file " ;
return false ;
}
2020-01-24 20:20:00 +00:00
2020-01-25 01:10:41 +00:00
const unsigned char * header = ( const unsigned char * ) data . constData ( ) ;
int sig = header [ 0 ] | ( header [ 1 ] < < 8 ) ;
int year = header [ 2 ] | ( header [ 3 ] < < 8 ) ;
int month = header [ 4 ] ;
int day = header [ 5 ] ;
int hour = header [ 6 ] ;
int min = header [ 7 ] ;
int sec = header [ 8 ] ;
2021-02-22 20:30:04 +00:00
/* CN - Changed the if statement to a switch to accomdate additional Viatom/Wellue signatures in the future
2020-01-25 01:10:41 +00:00
if ( sig ! = 0x0003 ) {
qDebug ( ) < < m_file . fileName ( ) < < " invalid signature for Viatom data file " < < sig ;
return false ;
}
2021-02-22 20:30:04 +00:00
CN */
switch ( sig ) {
case 0x0003 :
case 0x0005 :
break ;
default :
qDebug ( ) < < m_file . fileName ( ) < < " invalid signature for Viatom data file " < < sig ;
return false ;
break ;
}
2020-01-25 01:10:41 +00:00
if ( ( year < 2015 | | year > 2059 ) | | ( month < 1 | | month > 12 ) | | ( day < 1 | | day > 31 ) | |
( hour > 23 ) | | ( min > 59 ) | | ( sec > 59 ) ) {
qDebug ( ) < < m_file . fileName ( ) < < " invalid timestamp in Viatom data file " ;
2020-01-24 20:20:00 +00:00
return false ;
}
2020-01-25 21:55:32 +00:00
// It's unclear what the starting timestamp represents: is it the time at which
// the device starts measuring data, and the first sample is 4s after that? Or
// is the starting timestamp the time at which the first 4s average is reported
// (and the first 4 seconds being average precede the starting timestamp)?
//
// If the former, then the chart draws the first sample too early (right at the
// starting timestamp). Technically these should probably be square charts, but
// the code currently forces them to be non-square.
2020-01-25 01:10:41 +00:00
QDateTime data_timestamp = QDateTime ( QDate ( year , month , day ) , QTime ( hour , min , sec ) ) ;
2020-01-24 20:20:00 +00:00
m_timestamp = data_timestamp . toMSecsSinceEpoch ( ) ;
2020-01-25 01:10:41 +00:00
m_sessionid = m_timestamp / 1000L ;
int filesize = header [ 9 ] | ( header [ 10 ] < < 8 ) ; // possibly 32-bit
CHECK_VALUE ( header [ 11 ] , 0 ) ;
CHECK_VALUE ( header [ 12 ] , 0 ) ;
m_duration = header [ 13 ] | ( header [ 14 ] < < 8 ) ; // possibly 32-bit
CHECK_VALUE ( header [ 15 ] , 0 ) ;
CHECK_VALUE ( header [ 16 ] , 0 ) ;
//int spo2_avg = header[17];
//int spo2_min = header[18];
//int spo2_3pct = header[19]; // number of events
2020-05-06 15:42:03 +00:00
//int spo2_4pct = header[20]; // number of events
//CHECK_VALUE(header[21], 0); // ??? sometimes nonzero; maybe pulse spike, not a threshold of SpO2 or pulse, not always smaller than spo2_4pct
2020-02-07 18:54:28 +00:00
//int time_under_90pct = header[22] | (header[23] << 8); // in seconds
2020-01-25 01:10:41 +00:00
//int events_under_90pct = header[24]; // number of distinct events
//float o2_score = header[25] * 0.1;
CHECK_VALUE ( header [ 26 ] , 0 ) ;
CHECK_VALUE ( header [ 27 ] , 0 ) ;
CHECK_VALUE ( header [ 28 ] , 0 ) ;
CHECK_VALUE ( header [ 29 ] , 0 ) ;
CHECK_VALUE ( header [ 30 ] , 0 ) ;
CHECK_VALUE ( header [ 31 ] , 0 ) ;
CHECK_VALUE ( header [ 32 ] , 0 ) ;
CHECK_VALUE ( header [ 33 ] , 0 ) ;
CHECK_VALUE ( header [ 34 ] , 0 ) ;
CHECK_VALUE ( header [ 35 ] , 0 ) ;
CHECK_VALUE ( header [ 36 ] , 0 ) ;
CHECK_VALUE ( header [ 37 ] , 0 ) ;
CHECK_VALUE ( header [ 38 ] , 0 ) ;
CHECK_VALUE ( header [ 39 ] , 0 ) ;
// Calculate timing resolution (in ms) of the data
qint64 datasize = m_file . size ( ) - HEADER_SIZE ;
m_record_count = datasize / RECORD_SIZE ;
m_resolution = m_duration / m_record_count * 1000L ;
if ( m_resolution = = 2000 ) {
// Interestingly the file size in the header corresponds the number of
// distinct samples. These files actually double-report each sample!
// So this resolution isn't really the real one. The importer should
// calculate resolution from duration / record count after reading the
// records, which will be deduplicated.
CHECK_VALUE ( filesize , ( ( m_file . size ( ) - HEADER_SIZE ) / 2 ) + HEADER_SIZE ) ;
} else {
CHECK_VALUE ( filesize , m_file . size ( ) ) ;
}
CHECK_VALUES ( m_resolution , 2000 , 4000 ) ;
2021-02-22 20:30:04 +00:00
// CHECK_VALUE(datasize % RECORD_SIZE, 0); CN - Commented out these 2 lines because CheckMe can record odd number of entries
// CHECK_VALUE(m_duration % m_record_count, 0);
2020-01-24 20:20:00 +00:00
2020-01-26 22:20:58 +00:00
//qDebug().noquote() << m_file.fileName() << ts(m_timestamp) << dur(m_duration * 1000L) << ":" << m_record_count << "records @" << m_resolution << "ms";
2020-01-24 20:20:00 +00:00
return true ;
}
QList < ViatomFile : : Record > ViatomFile : : ReadData ( )
{
QByteArray data = m_file . readAll ( ) ;
QDataStream in ( data ) ;
in . setByteOrder ( QDataStream : : LittleEndian ) ;
2021-02-22 20:30:04 +00:00
int iCheckMeAdj ; // Allows for an odd number in the CheckMe duration/# of records return
2020-01-24 20:20:00 +00:00
QList < ViatomFile : : Record > records ;
// Read all Pulse, SPO2 and Motion data
do {
ViatomFile : : Record rec ;
2020-02-03 20:37:37 +00:00
in > > rec . spo2 > > rec . hr > > rec . oximetry_invalid > > rec . motion > > rec . vibration ;
2021-02-22 20:30:04 +00:00
CHECK_VALUES ( rec . oximetry_invalid , 0 , 0xFF ) ; //If it doesn't have one of these 2 values, it's bad
if ( rec . vibration = = 0x40 ) rec . vibration = 0x80 ; //0x040 (64) or 0x80 (128) when vibration is triggered
CHECK_VALUES ( rec . vibration , 0 , 0x80 ) ; // 0x80 (128) when vibration is triggered
2020-01-25 21:55:32 +00:00
if ( rec . oximetry_invalid = = 0xFF ) {
CHECK_VALUE ( rec . spo2 , 0xFF ) ;
2021-02-22 20:30:04 +00:00
CHECK_VALUE ( rec . hr , 0xFF ) ; // if all 3 have 0xFF, then end of data
2020-01-25 21:55:32 +00:00
}
2020-01-24 20:20:00 +00:00
records . append ( rec ) ;
2021-02-22 20:30:04 +00:00
} while ( records . size ( ) < m_record_count ) ; // CN Changed to allow for an incomlpete record values
// CN } while (!in.atEnd());
2020-01-24 20:20:00 +00:00
2020-01-25 19:50:45 +00:00
// It turns out 2s files are actually just double-reported samples!
2020-01-25 01:10:41 +00:00
if ( m_resolution = = 2000 ) {
2020-01-25 19:50:45 +00:00
QList < ViatomFile : : Record > dedup ;
bool all_are_duplicated = true ;
2020-01-25 01:10:41 +00:00
CHECK_VALUE ( records . size ( ) % 2 , 0 ) ;
for ( int i = 0 ; i < records . size ( ) ; i + = 2 ) {
auto & a = records . at ( i ) ;
auto & b = records . at ( i + 1 ) ;
2020-01-25 19:50:45 +00:00
if ( a . spo2 ! = b . spo2
| | a . hr ! = b . hr
2020-01-25 21:55:32 +00:00
| | a . oximetry_invalid ! = b . oximetry_invalid
2020-01-25 19:50:45 +00:00
| | a . motion ! = b . motion
2020-02-03 20:37:37 +00:00
| | a . vibration ! = b . vibration ) {
2020-01-25 19:50:45 +00:00
all_are_duplicated = false ;
break ;
}
dedup . append ( a ) ;
}
CHECK_VALUE ( all_are_duplicated , true ) ;
if ( all_are_duplicated ) {
// Return the deduplicated list.
records = dedup ;
2020-01-25 01:10:41 +00:00
}
}
2021-02-22 20:30:04 +00:00
iCheckMeAdj = duration ( ) / records . size ( ) ;
if ( iCheckMeAdj = = 3 ) iCheckMeAdj = 4 ; // CN - Sanity check for CheckMe devices since their files do not always terminate on an even number.
CHECK_VALUE ( iCheckMeAdj , 4 ) ; // Crimson Nape - Changed to accomadate the CheckMe data files.
// CHECK_VALUE(duration() / records.size(), 4); // We've only seen 4s true resolution so far.
2020-01-25 01:10:41 +00:00
2020-01-24 20:20:00 +00:00
return records ;
}
2020-01-25 19:50:45 +00:00