From 58a6cd3c02f0912cf8fdea80ca5dd5f766bbaf50 Mon Sep 17 00:00:00 2001 From: Philoul Date: Sat, 16 Oct 2021 12:35:47 +0200 Subject: [PATCH 1/2] Avoid Crash in csv Export Fix Translator if null objects sent Add again @SerializedName removed in Temporaytarget and TherapyEvent --- .../nightscout/androidaps/utils/Translator.kt | 8 ++-- .../database/entities/TemporaryTarget.kt | 7 +++ .../database/entities/TherapyEvent.kt | 48 +++++++++++++++++++ 3 files changed, 59 insertions(+), 4 deletions(-) diff --git a/core/src/main/java/info/nightscout/androidaps/utils/Translator.kt b/core/src/main/java/info/nightscout/androidaps/utils/Translator.kt index fa71db66b4..b62acc246d 100644 --- a/core/src/main/java/info/nightscout/androidaps/utils/Translator.kt +++ b/core/src/main/java/info/nightscout/androidaps/utils/Translator.kt @@ -114,7 +114,7 @@ class Translator @Inject internal constructor( else -> "" } - fun translate(meterType: TherapyEvent.MeterType): String = when (meterType) { + fun translate(meterType: TherapyEvent.MeterType?): String = when (meterType) { TherapyEvent.MeterType.FINGER -> resourceHelper.gs(R.string.glucosetype_finger) TherapyEvent.MeterType.SENSOR -> resourceHelper.gs(R.string.glucosetype_sensor) TherapyEvent.MeterType.MANUAL -> resourceHelper.gs(R.string.manual) @@ -122,7 +122,7 @@ class Translator @Inject internal constructor( else -> resourceHelper.gs(R.string.unknown) } - fun translate(type: TherapyEvent.Type): String = when (type) { + fun translate(type: TherapyEvent.Type?): String = when (type) { TherapyEvent.Type.FINGER_STICK_BG_VALUE -> resourceHelper.gs(R.string.careportal_bgcheck) TherapyEvent.Type.SNACK_BOLUS -> resourceHelper.gs(R.string.careportal_snackbolus) TherapyEvent.Type.MEAL_BOLUS -> resourceHelper.gs(R.string.careportal_mealbolus) @@ -173,7 +173,7 @@ class Translator @Inject internal constructor( else -> resourceHelper.gs(R.string.unknown) } - fun translate(reason: TemporaryTarget.Reason): String = when (reason) { + fun translate(reason: TemporaryTarget.Reason?): String = when (reason) { TemporaryTarget.Reason.CUSTOM -> resourceHelper.gs(R.string.custom) TemporaryTarget.Reason.HYPOGLYCEMIA -> resourceHelper.gs(R.string.hypo) TemporaryTarget.Reason.EATING_SOON -> resourceHelper.gs(R.string.eatingsoon) @@ -184,7 +184,7 @@ class Translator @Inject internal constructor( else -> resourceHelper.gs(R.string.unknown) } - fun translate(reason: OfflineEvent.Reason): String = when (reason) { + fun translate(reason: OfflineEvent.Reason?): String = when (reason) { OfflineEvent.Reason.SUSPEND -> resourceHelper.gs(R.string.uel_suspend) OfflineEvent.Reason.DISABLE_LOOP -> resourceHelper.gs(R.string.disableloop) OfflineEvent.Reason.DISCONNECT_PUMP -> resourceHelper.gs(R.string.uel_disconnect) diff --git a/database/src/main/java/info/nightscout/androidaps/database/entities/TemporaryTarget.kt b/database/src/main/java/info/nightscout/androidaps/database/entities/TemporaryTarget.kt index bec59a6da9..a818a78904 100644 --- a/database/src/main/java/info/nightscout/androidaps/database/entities/TemporaryTarget.kt +++ b/database/src/main/java/info/nightscout/androidaps/database/entities/TemporaryTarget.kt @@ -5,6 +5,7 @@ import androidx.room.Entity import androidx.room.ForeignKey import androidx.room.Index import androidx.room.PrimaryKey +import com.google.gson.annotations.SerializedName import info.nightscout.androidaps.database.TABLE_TEMPORARY_TARGETS import info.nightscout.androidaps.database.embedments.InterfaceIDs import info.nightscout.androidaps.database.interfaces.DBEntryWithTimeAndDuration @@ -62,11 +63,17 @@ data class TemporaryTarget( interfaceIDs.nightscoutId != null enum class Reason(val text: String) { + @SerializedName("Custom") CUSTOM("Custom"), + @SerializedName("Hypo") HYPOGLYCEMIA("Hypo"), + @SerializedName("Activity") ACTIVITY("Activity"), + @SerializedName("Eating Soon") EATING_SOON("Eating Soon"), + @SerializedName("Automation") AUTOMATION("Automation"), + @SerializedName("Wear") WEAR("Wear") ; diff --git a/database/src/main/java/info/nightscout/androidaps/database/entities/TherapyEvent.kt b/database/src/main/java/info/nightscout/androidaps/database/entities/TherapyEvent.kt index 3884bcb47a..f2421908ad 100644 --- a/database/src/main/java/info/nightscout/androidaps/database/entities/TherapyEvent.kt +++ b/database/src/main/java/info/nightscout/androidaps/database/entities/TherapyEvent.kt @@ -5,6 +5,7 @@ import androidx.room.Entity import androidx.room.ForeignKey import androidx.room.Index import androidx.room.PrimaryKey +import com.google.gson.annotations.SerializedName import info.nightscout.androidaps.database.TABLE_THERAPY_EVENTS import info.nightscout.androidaps.database.embedments.InterfaceIDs import info.nightscout.androidaps.database.interfaces.DBEntryWithTimeAndDuration @@ -73,8 +74,11 @@ data class TherapyEvent( } enum class MeterType(val text: String) { + @SerializedName("Finger") FINGER("Finger"), + @SerializedName("Sensor") SENSOR("Sensor"), + @SerializedName("Manual") MANUAL("Manual") ; @@ -87,55 +91,99 @@ data class TherapyEvent( @Suppress("unused") enum class Type(val text: String, val nsNative: Boolean = false) { + @SerializedName("Site Change") CANNULA_CHANGE("Site Change", nsNative = true), + @SerializedName("Insulin Change") INSULIN_CHANGE("Insulin Change", nsNative = true), + @SerializedName("Pump Battery Change") PUMP_BATTERY_CHANGE("Pump Battery Change", nsNative = true), + @SerializedName("Sensor Change") SENSOR_CHANGE("Sensor Change", nsNative = true), + @SerializedName("Sensor Start") SENSOR_STARTED("Sensor Start", nsNative = true), + @SerializedName("Sensor Stop") SENSOR_STOPPED("Sensor Stop", nsNative = true), + @SerializedName("BG Check") FINGER_STICK_BG_VALUE("BG Check", nsNative = true), + @SerializedName("Exercise") EXERCISE("Exercise", nsNative = true), + @SerializedName("Announcement") ANNOUNCEMENT("Announcement", nsNative = true), + @SerializedName("Question") QUESTION("Question", nsNative = true), + @SerializedName("Note") NOTE("Note", nsNative = true), + @SerializedName("OpenAPS Offline") APS_OFFLINE("OpenAPS Offline", nsNative = true), + @SerializedName("D.A.D. Alert") DAD_ALERT("D.A.D. Alert", nsNative = true), + @SerializedName("Mbg") NS_MBG("Mbg", nsNative = true), // Used but not as a Therapy Event (use constants only) + @SerializedName("Carb Correction") CARBS_CORRECTION("Carb Correction", nsNative = true), + @SerializedName("Bolus Wizard") BOLUS_WIZARD("Bolus Wizard", nsNative = true), + @SerializedName("Correction Bolus") CORRECTION_BOLUS("Correction Bolus", nsNative = true), + @SerializedName("Meal Bolus") MEAL_BOLUS("Meal Bolus", nsNative = true), + @SerializedName("Combo Bolus") COMBO_BOLUS("Combo Bolus", nsNative = true), + @SerializedName("Temporary Target") TEMPORARY_TARGET("Temporary Target", nsNative = true), + @SerializedName("Temporary Target Cancel") TEMPORARY_TARGET_CANCEL("Temporary Target Cancel", nsNative = true), + @SerializedName("Profile Switch") PROFILE_SWITCH("Profile Switch", nsNative = true), + @SerializedName("Snack Bolus") SNACK_BOLUS("Snack Bolus", nsNative = true), + @SerializedName("Temp Basal") TEMPORARY_BASAL("Temp Basal", nsNative = true), + @SerializedName("Temp Basal Start") TEMPORARY_BASAL_START("Temp Basal Start", nsNative = true), + @SerializedName("Temp Basal End") TEMPORARY_BASAL_END("Temp Basal End", nsNative = true), // Not supported by NS + @SerializedName("Tube Change") TUBE_CHANGE("Tube Change"), + @SerializedName("Falling Asleep") FALLING_ASLEEP("Falling Asleep"), + @SerializedName("Battery Empty") BATTERY_EMPTY("Battery Empty"), + @SerializedName("Reservoir Empty") RESERVOIR_EMPTY("Reservoir Empty"), + @SerializedName("Occlusion") OCCLUSION("Occlusion"), + @SerializedName("Pump Stopped") PUMP_STOPPED("Pump Stopped"), + @SerializedName("Pump Started") PUMP_STARTED("Pump Started"), + @SerializedName("Pump Paused") PUMP_PAUSED("Pump Paused"), + @SerializedName("Waking Up") WAKING_UP("Waking Up"), + @SerializedName("Sickness") SICKNESS("Sickness"), + @SerializedName("Stress") STRESS("Stress"), + @SerializedName("Pre Period") PRE_PERIOD("Pre Period"), + @SerializedName("Alcohol") ALCOHOL("Alcohol"), + @SerializedName("Cortisone") CORTISONE("Cortisone"), + @SerializedName("Feeling Low") FEELING_LOW("Feeling Low"), + @SerializedName("Feeling High") FEELING_HIGH("Feeling High"), + @SerializedName("Leaking Infusion Set") LEAKING_INFUSION_SET("Leaking Infusion Set"), // Default + @SerializedName("") NONE("") ; From 6e318cfaa0acde9aba24e5e4af790c0b3bfa83e4 Mon Sep 17 00:00:00 2001 From: Philoul Date: Sat, 16 Oct 2021 18:06:18 +0200 Subject: [PATCH 2/2] Limit csv export to 90 days --- .../fragments/TreatmentsUserEntryFragment.kt | 2 +- .../general/maintenance/ImportExportPrefsImpl.kt | 10 ++++++---- .../plugins/general/maintenance/MaintenanceFragment.kt | 2 +- .../androidaps/interfaces/ImportExportPrefs.kt | 2 +- 4 files changed, 9 insertions(+), 7 deletions(-) diff --git a/app/src/main/java/info/nightscout/androidaps/activities/fragments/TreatmentsUserEntryFragment.kt b/app/src/main/java/info/nightscout/androidaps/activities/fragments/TreatmentsUserEntryFragment.kt index a0b5d1b169..af2ff771cf 100644 --- a/app/src/main/java/info/nightscout/androidaps/activities/fragments/TreatmentsUserEntryFragment.kt +++ b/app/src/main/java/info/nightscout/androidaps/activities/fragments/TreatmentsUserEntryFragment.kt @@ -68,7 +68,7 @@ class TreatmentsUserEntryFragment : DaggerFragment() { activity?.let { activity -> OKDialog.showConfirmation(activity, resourceHelper.gs(R.string.ue_export_to_csv) + "?") { uel.log(Action.EXPORT_CSV, Sources.Treatments) - importExportPrefs.exportUserEntriesCsv(activity, repository.getAllUserEntries()) + importExportPrefs.exportUserEntriesCsv(activity) } } } diff --git a/app/src/main/java/info/nightscout/androidaps/plugins/general/maintenance/ImportExportPrefsImpl.kt b/app/src/main/java/info/nightscout/androidaps/plugins/general/maintenance/ImportExportPrefsImpl.kt index 908d801137..44c26e0c57 100644 --- a/app/src/main/java/info/nightscout/androidaps/plugins/general/maintenance/ImportExportPrefsImpl.kt +++ b/app/src/main/java/info/nightscout/androidaps/plugins/general/maintenance/ImportExportPrefsImpl.kt @@ -15,7 +15,7 @@ import info.nightscout.androidaps.BuildConfig import info.nightscout.androidaps.R import info.nightscout.androidaps.activities.DaggerAppCompatActivityWithResult import info.nightscout.androidaps.activities.PreferencesActivity -import info.nightscout.androidaps.database.entities.UserEntry +import info.nightscout.androidaps.database.AppRepository import info.nightscout.androidaps.database.entities.UserEntry.Action import info.nightscout.androidaps.database.entities.UserEntry.Sources import info.nightscout.androidaps.events.EventAppExit @@ -28,6 +28,8 @@ import info.nightscout.androidaps.plugins.bus.RxBusWrapper import info.nightscout.androidaps.plugins.general.maintenance.formats.* import info.nightscout.androidaps.utils.AndroidPermission import info.nightscout.androidaps.utils.DateUtil +import info.nightscout.androidaps.utils.MidnightTime +import info.nightscout.androidaps.utils.T import info.nightscout.androidaps.utils.ToastUtils import info.nightscout.androidaps.utils.alertDialogs.OKDialog import info.nightscout.androidaps.utils.alertDialogs.PrefImportSummaryDialog @@ -37,7 +39,6 @@ import info.nightscout.androidaps.utils.buildHelper.BuildHelper import info.nightscout.androidaps.utils.protection.PasswordCheck import info.nightscout.androidaps.utils.resources.ResourceHelper import info.nightscout.androidaps.utils.sharedPreferences.SP -import io.reactivex.Single import java.io.File import java.io.FileNotFoundException import java.io.IOException @@ -63,6 +64,7 @@ class ImportExportPrefsImpl @Inject constructor( private val encryptedPrefsFormat: EncryptedPrefsFormat, private val prefFileList: PrefFileListProvider, private val uel: UserEntryLogger, + private val repository: AppRepository, private val dateUtil: DateUtil ) : ImportExportPrefs { @@ -360,8 +362,8 @@ class ImportExportPrefsImpl @Inject constructor( } } - override fun exportUserEntriesCsv(activity: FragmentActivity, singleEntries: Single>) { - val entries = singleEntries.blockingGet() + override fun exportUserEntriesCsv(activity: FragmentActivity) { + val entries = repository.getUserEntryFilteredDataFromTime(MidnightTime.calc() - T.days(90).msecs()).blockingGet() prefFileList.ensureExportDirExists() val newFile = prefFileList.newExportCsvFile() diff --git a/app/src/main/java/info/nightscout/androidaps/plugins/general/maintenance/MaintenanceFragment.kt b/app/src/main/java/info/nightscout/androidaps/plugins/general/maintenance/MaintenanceFragment.kt index a78575151c..750df25015 100644 --- a/app/src/main/java/info/nightscout/androidaps/plugins/general/maintenance/MaintenanceFragment.kt +++ b/app/src/main/java/info/nightscout/androidaps/plugins/general/maintenance/MaintenanceFragment.kt @@ -118,7 +118,7 @@ class MaintenanceFragment : DaggerFragment() { activity?.let { activity -> OKDialog.showConfirmation(activity, resourceHelper.gs(R.string.ue_export_to_csv) + "?") { uel.log(Action.EXPORT_CSV, Sources.Maintenance) - importExportPrefs.exportUserEntriesCsv(activity, repository.getAllUserEntries()) + importExportPrefs.exportUserEntriesCsv(activity) } } } diff --git a/core/src/main/java/info/nightscout/androidaps/interfaces/ImportExportPrefs.kt b/core/src/main/java/info/nightscout/androidaps/interfaces/ImportExportPrefs.kt index 81782fbb88..95dbb3467f 100644 --- a/core/src/main/java/info/nightscout/androidaps/interfaces/ImportExportPrefs.kt +++ b/core/src/main/java/info/nightscout/androidaps/interfaces/ImportExportPrefs.kt @@ -14,5 +14,5 @@ interface ImportExportPrefs { fun prefsFileExists(): Boolean fun verifyStoragePermissions(fragment: Fragment, onGranted: Runnable) fun exportSharedPreferences(f: Fragment) - fun exportUserEntriesCsv(activity: FragmentActivity, singleEntries: Single>) + fun exportUserEntriesCsv(activity: FragmentActivity) } \ No newline at end of file