2017-11-28 21:31:27 +02:00
// SPDX-License-Identifier: GPL-2.0
# ifdef __clang__
// Clang has a bug on zero-initialization of C structs.
# pragma clang diagnostic ignored "-Wmissing-field-initializers"
# endif
2018-05-22 09:07:42 +02:00
# include "ssrf.h"
2017-11-28 21:31:27 +02:00
# include "dive.h"
2018-05-11 08:25:41 -07:00
# include "subsurface-string.h"
2017-11-28 21:31:27 +02:00
# include "parse.h"
# include "divelist.h"
# include "device.h"
# include "membuffer.h"
# include "gettext.h"
2018-10-17 18:45:22 +02:00
static int shearwater_cylinders ( void * param , int columns , char * * data , char * * column )
2017-11-28 21:31:27 +02:00
{
2018-05-22 09:07:42 +02:00
UNUSED ( columns ) ;
UNUSED ( column ) ;
2018-10-17 18:45:22 +02:00
struct parser_state * state = ( struct parser_state * ) param ;
2017-11-28 21:31:27 +02:00
int o2 = lrint ( strtod_flags ( data [ 0 ] , NULL , 0 ) * 1000 ) ;
int he = lrint ( strtod_flags ( data [ 1 ] , NULL , 0 ) * 1000 ) ;
/* Shearwater allows entering only 99%, not 100%
* so assume 99 % to be pure oxygen */
if ( o2 = = 990 & & he = = 0 )
o2 = 1000 ;
2018-10-17 18:45:22 +02:00
cylinder_start ( state ) ;
state - > cur_dive - > cylinder [ state - > cur_cylinder_index ] . gasmix . o2 . permille = o2 ;
state - > cur_dive - > cylinder [ state - > cur_cylinder_index ] . gasmix . he . permille = he ;
cylinder_end ( state ) ;
2017-11-28 21:31:27 +02:00
return 0 ;
}
2018-10-17 18:45:22 +02:00
static int shearwater_changes ( void * param , int columns , char * * data , char * * column )
2017-11-28 21:31:27 +02:00
{
2018-05-22 09:07:42 +02:00
UNUSED ( columns ) ;
UNUSED ( column ) ;
2018-10-17 18:45:22 +02:00
struct parser_state * state = ( struct parser_state * ) param ;
2017-11-28 21:31:27 +02:00
if ( columns ! = 3 ) {
return 1 ;
}
if ( ! data [ 0 ] | | ! data [ 1 ] | | ! data [ 2 ] ) {
return 2 ;
}
int o2 = lrint ( strtod_flags ( data [ 1 ] , NULL , 0 ) * 1000 ) ;
int he = lrint ( strtod_flags ( data [ 2 ] , NULL , 0 ) * 1000 ) ;
/* Shearwater allows entering only 99%, not 100%
* so assume 99 % to be pure oxygen */
if ( o2 = = 990 & & he = = 0 )
o2 = 1000 ;
// Find the cylinder index
int i ;
bool found = false ;
2018-10-17 18:45:22 +02:00
for ( i = 0 ; i < state - > cur_cylinder_index ; + + i ) {
if ( state - > cur_dive - > cylinder [ i ] . gasmix . o2 . permille = = o2 & & state - > cur_dive - > cylinder [ i ] . gasmix . he . permille = = he ) {
2017-11-28 21:31:27 +02:00
found = true ;
break ;
}
}
if ( ! found ) {
// Cylinder not found, creating a new one
2018-10-17 18:45:22 +02:00
cylinder_start ( state ) ;
state - > cur_dive - > cylinder [ state - > cur_cylinder_index ] . gasmix . o2 . permille = o2 ;
state - > cur_dive - > cylinder [ state - > cur_cylinder_index ] . gasmix . he . permille = he ;
cylinder_end ( state ) ;
i = state - > cur_cylinder_index ;
2017-11-28 21:31:27 +02:00
}
2019-01-03 06:56:48 +02:00
add_gas_switch_event ( state - > cur_dive , get_dc ( state ) , state - > sample_rate ? atoi ( data [ 0 ] ) / state - > sample_rate * 10 : atoi ( data [ 0 ] ) , i ) ;
2017-11-28 21:31:27 +02:00
return 0 ;
}
2018-10-17 18:45:22 +02:00
static int shearwater_profile_sample ( void * param , int columns , char * * data , char * * column )
2017-11-28 21:31:27 +02:00
{
2018-05-22 09:07:42 +02:00
UNUSED ( columns ) ;
UNUSED ( column ) ;
2018-10-17 18:45:22 +02:00
struct parser_state * state = ( struct parser_state * ) param ;
2018-05-27 16:56:09 +03:00
int d6 , d7 ;
2017-11-28 21:31:27 +02:00
2018-10-17 18:45:22 +02:00
sample_start ( state ) ;
2017-11-28 21:31:27 +02:00
if ( data [ 0 ] )
2019-01-03 06:56:48 +02:00
state - > cur_sample - > time . seconds = state - > sample_rate ? atoi ( data [ 0 ] ) / state - > sample_rate * 10 : atoi ( data [ 0 ] ) ;
2017-11-28 21:31:27 +02:00
if ( data [ 1 ] )
2018-10-17 18:45:22 +02:00
state - > cur_sample - > depth . mm = state - > metric ? lrint ( strtod_flags ( data [ 1 ] , NULL , 0 ) * 1000 ) : feet_to_mm ( strtod_flags ( data [ 1 ] , NULL , 0 ) ) ;
2017-11-28 21:31:27 +02:00
if ( data [ 2 ] )
2018-10-17 18:45:22 +02:00
state - > cur_sample - > temperature . mkelvin = state - > metric ? C_to_mkelvin ( strtod_flags ( data [ 2 ] , NULL , 0 ) ) : F_to_mkelvin ( strtod_flags ( data [ 2 ] , NULL , 0 ) ) ;
2017-11-28 21:31:27 +02:00
if ( data [ 3 ] ) {
2018-10-17 18:45:22 +02:00
state - > cur_sample - > setpoint . mbar = lrint ( strtod_flags ( data [ 3 ] , NULL , 0 ) * 1000 ) ;
2017-11-28 21:31:27 +02:00
}
if ( data [ 4 ] )
2018-10-17 18:45:22 +02:00
state - > cur_sample - > ndl . seconds = atoi ( data [ 4 ] ) * 60 ;
2017-11-28 21:31:27 +02:00
if ( data [ 5 ] )
2018-10-17 18:45:22 +02:00
state - > cur_sample - > cns = atoi ( data [ 5 ] ) ;
2018-05-27 09:06:09 +03:00
if ( data [ 6 ] ) {
2018-05-27 16:56:09 +03:00
d6 = atoi ( data [ 6 ] ) ;
if ( d6 > 0 ) {
2018-10-17 18:45:22 +02:00
state - > cur_sample - > stopdepth . mm = state - > metric ? d6 * 1000 : feet_to_mm ( d6 ) ;
state - > cur_sample - > in_deco = 1 ;
2018-05-27 09:06:09 +03:00
} else if ( data [ 7 ] ) {
2018-05-27 16:56:09 +03:00
d7 = atoi ( data [ 7 ] ) ;
if ( d7 > 0 ) {
2018-10-17 18:45:22 +02:00
state - > cur_sample - > stopdepth . mm = state - > metric ? d7 * 1000 : feet_to_mm ( d7 ) ;
2018-05-27 17:17:15 +03:00
if ( data [ 8 ] )
2018-10-17 18:45:22 +02:00
state - > cur_sample - > stoptime . seconds = atoi ( data [ 8 ] ) * 60 ;
state - > cur_sample - > in_deco = 1 ;
2018-05-27 09:06:09 +03:00
} else {
2018-10-17 18:45:22 +02:00
state - > cur_sample - > in_deco = 0 ;
2018-05-27 09:06:09 +03:00
}
} else {
2018-10-17 18:45:22 +02:00
state - > cur_sample - > in_deco = 0 ;
2018-05-27 09:06:09 +03:00
}
}
2017-11-28 21:31:27 +02:00
/* We don't actually have data[3], but it should appear in the
* SQL query at some point .
if ( data [ 3 ] )
2018-10-17 18:45:22 +02:00
state - > cur_sample - > pressure [ 0 ] . mbar = state - > metric ? atoi ( data [ 3 ] ) * 1000 : psi_to_mbar ( atoi ( data [ 3 ] ) ) ;
2017-11-28 21:31:27 +02:00
*/
2018-10-17 18:45:22 +02:00
sample_end ( state ) ;
2017-11-28 21:31:27 +02:00
return 0 ;
}
2018-10-17 18:45:22 +02:00
static int shearwater_ai_profile_sample ( void * param , int columns , char * * data , char * * column )
2017-11-28 21:31:27 +02:00
{
2018-05-22 09:07:42 +02:00
UNUSED ( columns ) ;
UNUSED ( column ) ;
2018-10-17 18:45:22 +02:00
struct parser_state * state = ( struct parser_state * ) param ;
2018-05-27 16:56:09 +03:00
int d6 , d9 ;
2017-11-28 21:31:27 +02:00
2018-10-17 18:45:22 +02:00
sample_start ( state ) ;
2017-11-28 21:31:27 +02:00
if ( data [ 0 ] )
2019-01-03 06:56:48 +02:00
state - > cur_sample - > time . seconds = state - > sample_rate ? atoi ( data [ 0 ] ) / state - > sample_rate * 10 : atoi ( data [ 0 ] ) ;
2017-11-28 21:31:27 +02:00
if ( data [ 1 ] )
2018-10-17 18:45:22 +02:00
state - > cur_sample - > depth . mm = state - > metric ? lrint ( strtod_flags ( data [ 1 ] , NULL , 0 ) * 1000 ) : feet_to_mm ( strtod_flags ( data [ 1 ] , NULL , 0 ) ) ;
2017-11-28 21:31:27 +02:00
if ( data [ 2 ] )
2018-10-17 18:45:22 +02:00
state - > cur_sample - > temperature . mkelvin = state - > metric ? C_to_mkelvin ( strtod_flags ( data [ 2 ] , NULL , 0 ) ) : F_to_mkelvin ( strtod_flags ( data [ 2 ] , NULL , 0 ) ) ;
2017-11-28 21:31:27 +02:00
if ( data [ 3 ] ) {
2018-10-17 18:45:22 +02:00
state - > cur_sample - > setpoint . mbar = lrint ( strtod_flags ( data [ 3 ] , NULL , 0 ) * 1000 ) ;
2017-11-28 21:31:27 +02:00
}
if ( data [ 4 ] )
2018-10-17 18:45:22 +02:00
state - > cur_sample - > ndl . seconds = atoi ( data [ 4 ] ) * 60 ;
2017-11-28 21:31:27 +02:00
if ( data [ 5 ] )
2018-10-17 18:45:22 +02:00
state - > cur_sample - > cns = atoi ( data [ 5 ] ) ;
2018-05-27 09:06:09 +03:00
if ( data [ 6 ] ) {
2018-05-27 16:56:09 +03:00
d6 = atoi ( data [ 6 ] ) ;
if ( d6 > 0 ) {
2018-10-17 18:45:22 +02:00
state - > cur_sample - > stopdepth . mm = state - > metric ? d6 * 1000 : feet_to_mm ( d6 ) ;
state - > cur_sample - > in_deco = 1 ;
2018-05-27 09:06:09 +03:00
} else if ( data [ 9 ] ) {
2018-05-27 16:56:09 +03:00
d9 = atoi ( data [ 9 ] ) ;
if ( d9 > 0 ) {
2018-10-17 18:45:22 +02:00
state - > cur_sample - > stopdepth . mm = state - > metric ? d9 * 1000 : feet_to_mm ( d9 ) ;
2018-05-27 17:17:15 +03:00
if ( data [ 10 ] )
2018-10-17 18:45:22 +02:00
state - > cur_sample - > stoptime . seconds = atoi ( data [ 10 ] ) * 60 ;
state - > cur_sample - > in_deco = 1 ;
2018-05-27 09:06:09 +03:00
} else {
2018-10-17 18:45:22 +02:00
state - > cur_sample - > in_deco = 0 ;
2018-05-27 09:06:09 +03:00
}
} else {
2018-10-17 18:45:22 +02:00
state - > cur_sample - > in_deco = 0 ;
2018-05-27 09:06:09 +03:00
}
}
2017-11-28 21:31:27 +02:00
/* Weird unit conversion but seems to produce correct results.
* Also missing values seems to be reported as a 4092 ( 564 bar ) */
if ( data [ 7 ] & & atoi ( data [ 7 ] ) ! = 4092 ) {
2018-10-17 18:45:22 +02:00
state - > cur_sample - > pressure [ 0 ] . mbar = psi_to_mbar ( atoi ( data [ 7 ] ) ) * 2 ;
2017-11-28 21:31:27 +02:00
}
if ( data [ 8 ] & & atoi ( data [ 8 ] ) ! = 4092 )
2018-10-17 18:45:22 +02:00
state - > cur_sample - > pressure [ 1 ] . mbar = psi_to_mbar ( atoi ( data [ 8 ] ) ) * 2 ;
sample_end ( state ) ;
2017-11-28 21:31:27 +02:00
return 0 ;
}
2018-10-17 18:45:22 +02:00
static int shearwater_mode ( void * param , int columns , char * * data , char * * column )
2017-11-28 21:31:27 +02:00
{
2018-05-22 09:07:42 +02:00
UNUSED ( columns ) ;
UNUSED ( column ) ;
2018-10-17 18:45:22 +02:00
struct parser_state * state = ( struct parser_state * ) param ;
2017-11-28 21:31:27 +02:00
if ( data [ 0 ] )
2018-10-17 18:45:22 +02:00
state - > cur_dive - > dc . divemode = atoi ( data [ 0 ] ) = = 0 ? CCR : OC ;
2017-11-28 21:31:27 +02:00
return 0 ;
}
2018-10-17 18:45:22 +02:00
static int shearwater_dive ( void * param , int columns , char * * data , char * * column )
2017-11-28 21:31:27 +02:00
{
2018-05-22 09:07:42 +02:00
UNUSED ( columns ) ;
UNUSED ( column ) ;
2017-11-28 21:31:27 +02:00
int retval = 0 ;
2018-10-17 18:45:22 +02:00
struct parser_state * state = ( struct parser_state * ) param ;
sqlite3 * handle = state - > sql_handle ;
2017-11-28 21:31:27 +02:00
char * err = NULL ;
2018-12-29 21:31:12 +02:00
char get_profile_template [ ] = " select currentTime,currentDepth,waterTemp,averagePPO2,currentNdl,CNSPercent,decoCeiling,firstStopDepth,firstStopTime from dive_log_records where diveLogId=%ld " ;
char get_profile_template_ai [ ] = " select currentTime,currentDepth,waterTemp,averagePPO2,currentNdl,CNSPercent,decoCeiling,aiSensor0_PressurePSI,aiSensor1_PressurePSI,firstStopDepth,firstStopTime from dive_log_records where diveLogId = %ld " ;
char get_cylinder_template [ ] = " select fractionO2,fractionHe from dive_log_records where diveLogId = %ld group by fractionO2,fractionHe " ;
char get_changes_template [ ] = " select a.currentTime,a.fractionO2,a.fractionHe from dive_log_records as a,dive_log_records as b where (a.id - 1) = b.id and (a.fractionO2 != b.fractionO2 or a.fractionHe != b.fractionHe) and a.diveLogId=b.divelogId and a.diveLogId = %ld " ;
char get_mode_template [ ] = " select distinct currentCircuitSetting from dive_log_records where diveLogId = %ld " ;
2017-11-28 21:31:27 +02:00
char get_buffer [ 1024 ] ;
2018-10-17 18:45:22 +02:00
dive_start ( state ) ;
state - > cur_dive - > number = atoi ( data [ 0 ] ) ;
2017-11-28 21:31:27 +02:00
2018-10-17 18:45:22 +02:00
state - > cur_dive - > when = ( time_t ) ( atol ( data [ 1 ] ) ) ;
2017-11-28 21:31:27 +02:00
2018-12-29 21:31:12 +02:00
long int dive_id = atol ( data [ 11 ] ) ;
2017-11-28 21:31:27 +02:00
if ( data [ 2 ] )
2018-10-17 18:45:22 +02:00
add_dive_site ( data [ 2 ] , state - > cur_dive , state ) ;
2017-11-28 21:31:27 +02:00
if ( data [ 3 ] )
2018-10-17 18:45:22 +02:00
utf8_string ( data [ 3 ] , & state - > cur_dive - > buddy ) ;
2017-11-28 21:31:27 +02:00
if ( data [ 4 ] )
2018-10-17 18:45:22 +02:00
utf8_string ( data [ 4 ] , & state - > cur_dive - > notes ) ;
2017-11-28 21:31:27 +02:00
2018-10-17 18:45:22 +02:00
state - > metric = atoi ( data [ 5 ] ) = = 1 ? 0 : 1 ;
2017-11-28 21:31:27 +02:00
/* TODO: verify that metric calculation is correct */
if ( data [ 6 ] )
2018-10-17 18:45:22 +02:00
state - > cur_dive - > dc . maxdepth . mm = state - > metric ? lrint ( strtod_flags ( data [ 6 ] , NULL , 0 ) * 1000 ) : feet_to_mm ( strtod_flags ( data [ 6 ] , NULL , 0 ) ) ;
2017-11-28 21:31:27 +02:00
if ( data [ 7 ] )
2018-10-17 18:45:22 +02:00
state - > cur_dive - > dc . duration . seconds = atoi ( data [ 7 ] ) * 60 ;
2017-11-28 21:31:27 +02:00
if ( data [ 8 ] )
2018-10-17 18:45:22 +02:00
state - > cur_dive - > dc . surface_pressure . mbar = atoi ( data [ 8 ] ) ;
2017-11-28 21:31:27 +02:00
/*
* TODO : the deviceid hash should be calculated here .
*/
2018-10-17 18:45:22 +02:00
settings_start ( state ) ;
dc_settings_start ( state ) ;
2017-11-28 21:31:27 +02:00
if ( data [ 9 ] )
2018-10-17 18:45:22 +02:00
utf8_string ( data [ 9 ] , & state - > cur_settings . dc . serial_nr ) ;
2017-11-28 21:31:27 +02:00
if ( data [ 10 ] ) {
switch ( atoi ( data [ 10 ] ) ) {
case 2 :
2018-10-17 18:45:22 +02:00
state - > cur_settings . dc . model = strdup ( " Shearwater Petrel/Perdix " ) ;
2017-11-28 21:31:27 +02:00
break ;
case 4 :
2018-10-17 18:45:22 +02:00
state - > cur_settings . dc . model = strdup ( " Shearwater Predator " ) ;
2017-11-28 21:31:27 +02:00
break ;
default :
2018-10-17 18:45:22 +02:00
state - > cur_settings . dc . model = strdup ( " Shearwater import " ) ;
2017-11-28 21:31:27 +02:00
break ;
}
}
2018-10-17 18:45:22 +02:00
state - > cur_settings . dc . deviceid = atoi ( data [ 9 ] ) ;
2017-11-28 21:31:27 +02:00
2018-10-17 18:45:22 +02:00
dc_settings_end ( state ) ;
settings_end ( state ) ;
2017-11-28 21:31:27 +02:00
if ( data [ 10 ] ) {
switch ( atoi ( data [ 10 ] ) ) {
case 2 :
2018-10-17 18:45:22 +02:00
state - > cur_dive - > dc . model = strdup ( " Shearwater Petrel/Perdix " ) ;
2017-11-28 21:31:27 +02:00
break ;
case 4 :
2018-10-17 18:45:22 +02:00
state - > cur_dive - > dc . model = strdup ( " Shearwater Predator " ) ;
2017-11-28 21:31:27 +02:00
break ;
default :
2018-10-17 18:45:22 +02:00
state - > cur_dive - > dc . model = strdup ( " Shearwater import " ) ;
2017-11-28 21:31:27 +02:00
break ;
}
}
if ( data [ 11 ] ) {
snprintf ( get_buffer , sizeof ( get_buffer ) - 1 , get_mode_template , dive_id ) ;
2018-10-17 18:45:22 +02:00
retval = sqlite3_exec ( handle , get_buffer , & shearwater_mode , state , & err ) ;
2017-11-28 21:31:27 +02:00
if ( retval ! = SQLITE_OK ) {
fprintf ( stderr , " %s " , " Database query shearwater_mode failed. \n " ) ;
return 1 ;
}
}
snprintf ( get_buffer , sizeof ( get_buffer ) - 1 , get_cylinder_template , dive_id ) ;
2018-10-17 18:45:22 +02:00
retval = sqlite3_exec ( handle , get_buffer , & shearwater_cylinders , state , & err ) ;
2017-11-28 21:31:27 +02:00
if ( retval ! = SQLITE_OK ) {
fprintf ( stderr , " %s " , " Database query shearwater_cylinders failed. \n " ) ;
return 1 ;
}
snprintf ( get_buffer , sizeof ( get_buffer ) - 1 , get_changes_template , dive_id ) ;
2018-10-17 18:45:22 +02:00
retval = sqlite3_exec ( handle , get_buffer , & shearwater_changes , state , & err ) ;
2017-11-28 21:31:27 +02:00
if ( retval ! = SQLITE_OK ) {
fprintf ( stderr , " %s " , " Database query shearwater_changes failed. \n " ) ;
return 1 ;
}
snprintf ( get_buffer , sizeof ( get_buffer ) - 1 , get_profile_template_ai , dive_id ) ;
2018-10-17 18:45:22 +02:00
retval = sqlite3_exec ( handle , get_buffer , & shearwater_ai_profile_sample , state , & err ) ;
2017-11-28 21:31:27 +02:00
if ( retval ! = SQLITE_OK ) {
snprintf ( get_buffer , sizeof ( get_buffer ) - 1 , get_profile_template , dive_id ) ;
2018-10-17 18:45:22 +02:00
retval = sqlite3_exec ( handle , get_buffer , & shearwater_profile_sample , state , & err ) ;
2017-11-28 21:31:27 +02:00
if ( retval ! = SQLITE_OK ) {
fprintf ( stderr , " %s " , " Database query shearwater_profile_sample failed. \n " ) ;
return 1 ;
}
}
2018-10-17 18:45:22 +02:00
dive_end ( state ) ;
2017-11-28 21:31:27 +02:00
return SQLITE_OK ;
}
2018-12-29 21:32:55 +02:00
static int shearwater_cloud_dive ( void * param , int columns , char * * data , char * * column )
{
UNUSED ( columns ) ;
UNUSED ( column ) ;
int retval = 0 ;
struct parser_state * state = ( struct parser_state * ) param ;
sqlite3 * handle = state - > sql_handle ;
char * err = NULL ;
2019-01-01 15:32:27 +02:00
char get_profile_template [ ] = " select currentTime,currentDepth,waterTemp,averagePPO2,currentNdl,CNSPercent,decoCeiling,firstStopDepth,firstStopTime from dive_log_records where diveLogId=%ld " ;
char get_profile_template_ai [ ] = " select currentTime,currentDepth,waterTemp,averagePPO2,currentNdl,CNSPercent,decoCeiling,aiSensor0_PressurePSI,aiSensor1_PressurePSI,firstStopDepth,firstStopTime from dive_log_records where diveLogId = %ld " ;
2018-12-29 21:32:55 +02:00
char get_cylinder_template [ ] = " select fractionO2 / 100,fractionHe / 100 from dive_log_records where diveLogId = %ld group by fractionO2,fractionHe " ;
2018-12-29 22:31:52 +02:00
char get_changes_template [ ] = " select a.currentTime,a.fractionO2 / 100,a.fractionHe /100 from dive_log_records as a,dive_log_records as b where (a.id - 1) = b.id and (a.fractionO2 != b.fractionO2 or a.fractionHe != b.fractionHe) and a.diveLogId=b.divelogId and a.diveLogId = %ld " ;
2018-12-29 21:32:55 +02:00
char get_mode_template [ ] = " select distinct currentCircuitSetting from dive_log_records where diveLogId = %ld " ;
char get_buffer [ 1024 ] ;
dive_start ( state ) ;
state - > cur_dive - > number = atoi ( data [ 0 ] ) ;
state - > cur_dive - > when = ( time_t ) ( atol ( data [ 1 ] ) ) ;
long int dive_id = atol ( data [ 11 ] ) ;
2018-12-29 22:31:52 +02:00
if ( data [ 12 ] )
2019-01-03 06:56:48 +02:00
state - > sample_rate = atoi ( data [ 12 ] ) ;
2018-12-29 22:31:52 +02:00
else
2019-01-03 06:56:48 +02:00
state - > sample_rate = 0 ;
2018-12-29 21:32:55 +02:00
if ( data [ 2 ] )
add_dive_site ( data [ 2 ] , state - > cur_dive , state ) ;
if ( data [ 3 ] )
utf8_string ( data [ 3 ] , & state - > cur_dive - > buddy ) ;
if ( data [ 4 ] )
utf8_string ( data [ 4 ] , & state - > cur_dive - > notes ) ;
state - > metric = atoi ( data [ 5 ] ) = = 1 ? 0 : 1 ;
/* TODO: verify that metric calculation is correct */
if ( data [ 6 ] )
state - > cur_dive - > dc . maxdepth . mm = state - > metric ? lrint ( strtod_flags ( data [ 6 ] , NULL , 0 ) * 1000 ) : feet_to_mm ( strtod_flags ( data [ 6 ] , NULL , 0 ) ) ;
if ( data [ 7 ] )
state - > cur_dive - > dc . duration . seconds = atoi ( data [ 7 ] ) * 60 ;
if ( data [ 8 ] )
state - > cur_dive - > dc . surface_pressure . mbar = atoi ( data [ 8 ] ) ;
/*
* TODO : the deviceid hash should be calculated here .
*/
settings_start ( state ) ;
dc_settings_start ( state ) ;
if ( data [ 9 ] )
utf8_string ( data [ 9 ] , & state - > cur_settings . dc . serial_nr ) ;
if ( data [ 10 ] ) {
switch ( atoi ( data [ 10 ] ) ) {
case 2 :
state - > cur_settings . dc . model = strdup ( " Shearwater Petrel/Perdix " ) ;
break ;
case 4 :
state - > cur_settings . dc . model = strdup ( " Shearwater Predator " ) ;
break ;
default :
state - > cur_settings . dc . model = strdup ( " Shearwater import " ) ;
break ;
}
}
state - > cur_settings . dc . deviceid = atoi ( data [ 9 ] ) ;
dc_settings_end ( state ) ;
settings_end ( state ) ;
if ( data [ 10 ] ) {
switch ( atoi ( data [ 10 ] ) ) {
case 2 :
state - > cur_dive - > dc . model = strdup ( " Shearwater Petrel/Perdix " ) ;
break ;
case 4 :
state - > cur_dive - > dc . model = strdup ( " Shearwater Predator " ) ;
break ;
default :
state - > cur_dive - > dc . model = strdup ( " Shearwater import " ) ;
break ;
}
}
if ( data [ 11 ] ) {
snprintf ( get_buffer , sizeof ( get_buffer ) - 1 , get_mode_template , dive_id ) ;
retval = sqlite3_exec ( handle , get_buffer , & shearwater_mode , state , & err ) ;
if ( retval ! = SQLITE_OK ) {
fprintf ( stderr , " %s " , " Database query shearwater_mode failed. \n " ) ;
return 1 ;
}
}
snprintf ( get_buffer , sizeof ( get_buffer ) - 1 , get_cylinder_template , dive_id ) ;
retval = sqlite3_exec ( handle , get_buffer , & shearwater_cylinders , state , & err ) ;
if ( retval ! = SQLITE_OK ) {
fprintf ( stderr , " %s " , " Database query shearwater_cylinders failed. \n " ) ;
return 1 ;
}
snprintf ( get_buffer , sizeof ( get_buffer ) - 1 , get_changes_template , dive_id ) ;
retval = sqlite3_exec ( handle , get_buffer , & shearwater_changes , state , & err ) ;
if ( retval ! = SQLITE_OK ) {
fprintf ( stderr , " %s " , " Database query shearwater_changes failed. \n " ) ;
return 1 ;
}
snprintf ( get_buffer , sizeof ( get_buffer ) - 1 , get_profile_template_ai , dive_id ) ;
retval = sqlite3_exec ( handle , get_buffer , & shearwater_ai_profile_sample , state , & err ) ;
if ( retval ! = SQLITE_OK ) {
snprintf ( get_buffer , sizeof ( get_buffer ) - 1 , get_profile_template , dive_id ) ;
retval = sqlite3_exec ( handle , get_buffer , & shearwater_profile_sample , state , & err ) ;
if ( retval ! = SQLITE_OK ) {
fprintf ( stderr , " %s " , " Database query shearwater_profile_sample failed. \n " ) ;
return 1 ;
}
}
dive_end ( state ) ;
return SQLITE_OK ;
}
2017-11-28 21:31:27 +02:00
int parse_shearwater_buffer ( sqlite3 * handle , const char * url , const char * buffer , int size ,
2018-11-27 00:25:15 +01:00
struct dive_table * table , struct trip_table * trips )
2017-11-28 21:31:27 +02:00
{
2018-05-22 09:07:42 +02:00
UNUSED ( buffer ) ;
UNUSED ( size ) ;
2017-11-28 21:31:27 +02:00
int retval ;
char * err = NULL ;
2018-10-17 18:45:22 +02:00
struct parser_state state ;
init_parser_state ( & state ) ;
state . target_table = table ;
2018-11-27 00:25:15 +01:00
state . trips = trips ;
2018-10-17 18:45:22 +02:00
state . sql_handle = handle ;
2017-11-28 21:31:27 +02:00
2018-12-29 22:31:52 +02:00
// So far have not seen any sample rate in Shearwater Desktop
2019-01-03 06:56:48 +02:00
state . sample_rate = 0 ;
2018-12-29 22:31:52 +02:00
2017-11-28 21:31:27 +02:00
char get_dives [ ] = " select l.number,timestamp,location||' / '||site,buddy,notes,imperialUnits,maxDepth,maxTime,startSurfacePressure,computerSerial,computerModel,i.diveId FROM dive_info AS i JOIN dive_logs AS l ON i.diveId=l.diveId " ;
2018-10-17 18:45:22 +02:00
retval = sqlite3_exec ( handle , get_dives , & shearwater_dive , & state , & err ) ;
free_parser_state ( & state ) ;
2017-11-28 21:31:27 +02:00
if ( retval ! = SQLITE_OK ) {
fprintf ( stderr , " Database query failed '%s'. \n " , url ) ;
return 1 ;
}
return 0 ;
}
2018-12-29 21:32:55 +02:00
int parse_shearwater_cloud_buffer ( sqlite3 * handle , const char * url , const char * buffer , int size ,
2018-11-27 00:25:15 +01:00
struct dive_table * table , struct trip_table * trips )
2018-12-29 21:32:55 +02:00
{
UNUSED ( buffer ) ;
UNUSED ( size ) ;
int retval ;
char * err = NULL ;
struct parser_state state ;
init_parser_state ( & state ) ;
state . target_table = table ;
2018-11-27 00:25:15 +01:00
state . trips = trips ;
2018-12-29 21:32:55 +02:00
state . sql_handle = handle ;
2018-12-30 16:31:51 +02:00
char get_dives [ ] = " select l.number,strftime('%s', DiveDate),location||' / '||site,buddy,notes,imperialUnits,maxDepth,maxTime,startSurfacePressure,computerSerial,computerModel,d.diveId,l.sampleRateMs FROM dive_details AS d JOIN dive_logs AS l ON d.diveId=l.diveId " ;
2018-12-29 21:32:55 +02:00
retval = sqlite3_exec ( handle , get_dives , & shearwater_cloud_dive , & state , & err ) ;
free_parser_state ( & state ) ;
if ( retval ! = SQLITE_OK ) {
fprintf ( stderr , " Database query failed '%s'. \n " , url ) ;
return 1 ;
}
return 0 ;
}