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 > )
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 . */
//********************************************************************************************
// IMPORTANT!!!
//********************************************************************************************
// Please INCREMENT the viatom_data_version in viatom_loader.h when making changes to this loader
// that change loader behaviour or modify channels.
//********************************************************************************************
# 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-24 00:25:06 +00:00
// I don't know under what circumstances this is called...
qDebug ( ) < < " ViatomLoader::Detect( " < < path < < " ) " ;
return true ;
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-01-26 23:13:09 +00:00
int imported = 0 ;
int found = 0 ;
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 ( ) ) {
imported + = OpenFile ( fi . canonicalFilePath ( ) ) ;
found + + ;
}
}
else {
// This filename has already been filtered by QFileDialog.
imported = OpenFile ( dirpath ) ;
found + + ;
}
if ( ! found ) {
return - 1 ;
}
return imported ;
2020-01-23 17:51:58 +00:00
}
int
2020-01-24 00:25:06 +00:00
ViatomLoader : : OpenFile ( const QString & filename )
2020-01-23 17:51:58 +00:00
{
2020-01-26 23:13:09 +00:00
int imported = 0 ;
2020-01-25 01:10:41 +00:00
s_unexpectedMessages . clear ( ) ;
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:13:09 +00:00
imported = 1 ;
2020-01-25 01:10:41 +00:00
2020-01-26 23:13:09 +00:00
// TODO: Move this to Open()
2020-01-25 01:10:41 +00:00
if ( 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 - sess - > machine ( ) - > 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 " +
QObject : : tr ( " The imported data may not be entirely accurate, so the developers would like a copy of this file to make sure OSCAR is handling the data correctly. " )
, QMessageBox : : Ok ) ;
sess - > machine ( ) - > previouslySeenUnexpectedData ( ) + = newMessages ;
}
}
2020-01-24 00:11:05 +00:00
}
2020-01-25 01:10:41 +00:00
2020-01-26 23:13:09 +00:00
return imported ;
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-25 01:10:41 +00:00
// TODO: Figure out what to do about machine ID. Right now OSCAR generates a random ID since we don't specify one.
// That means you won't be able to import multiple Viatom devices in a single session/profile.
2020-01-24 00:25:06 +00:00
MachineInfo info = newInfo ( ) ;
Machine * mach = p_profile - > CreateMachine ( info ) ;
2020-01-25 21:55:32 +00:00
qint64 time_ms = v . timestamp ( ) ;
2020-01-24 20:20:00 +00:00
QDateTime data_timestamp = QDateTime : : fromMSecsSinceEpoch ( time_ms ) ;
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
qDebug ( ) < < filename < < " session already exists, skipping " < < v . sessionid ( ) ;
return nullptr ;
2020-01-24 00:25:06 +00:00
}
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 ) ;
}
AddEvent ( POS_Motion , time_ms , rec . motion ) ;
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 ) ;
EndEventList ( POS_Motion , time_ms ) ;
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 ( ) ;
2020-01-24 00:25:06 +00:00
sess - > SetChanged ( true ) ;
mach - > AddSession ( sess ) ;
mach - > Save ( ) ;
2020-01-25 22:35:58 +00:00
mach - > SaveSummaryCache ( ) ;
p_profile - > StoreMachines ( ) ;
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]);
// 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
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 ] ;
if ( sig ! = 0x0003 ) {
qDebug ( ) < < m_file . fileName ( ) < < " invalid signature for Viatom data file " < < sig ;
return false ;
}
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
//int spo2_4pct = header[20]; // number of events
CHECK_VALUE ( header [ 21 ] , 0 ) ;
//int time_under_90pct = header[22]; // in seconds
CHECK_VALUE ( header [ 23 ] , 0 ) ;
//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 ) ;
CHECK_VALUE ( datasize % RECORD_SIZE , 0 ) ;
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 ) ;
QList < ViatomFile : : Record > records ;
// Read all Pulse, SPO2 and Motion data
do {
ViatomFile : : Record rec ;
2020-01-25 21:55:32 +00:00
in > > rec . spo2 > > rec . hr > > rec . oximetry_invalid > > rec . motion > > rec . _unk ;
CHECK_VALUES ( rec . oximetry_invalid , 0 , 0xFF ) ;
CHECK_VALUE ( rec . _unk , 0 ) ;
if ( rec . oximetry_invalid = = 0xFF ) {
CHECK_VALUE ( rec . spo2 , 0xFF ) ;
CHECK_VALUE ( rec . hr , 0xFF ) ;
}
2020-01-24 20:20:00 +00:00
records . append ( rec ) ;
} while ( ! in . atEnd ( ) ) ;
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-01-25 21:55:32 +00:00
| | a . _unk ! = b . _unk ) {
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
}
}
2020-01-25 19:50:45 +00:00
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