AndroidAPS/app/src/main/java/info/nightscout/androidaps/data/Profile.java

567 lines
20 KiB
Java
Raw Normal View History

2017-06-02 10:25:49 +02:00
package info.nightscout.androidaps.data;
2017-06-13 15:06:41 +02:00
import android.support.v4.util.LongSparseArray;
2017-06-02 10:25:49 +02:00
import com.crashlytics.android.Crashlytics;
import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.text.DecimalFormat;
import java.util.Calendar;
import java.util.TimeZone;
import info.nightscout.androidaps.Constants;
import info.nightscout.androidaps.MainApp;
import info.nightscout.androidaps.R;
import info.nightscout.androidaps.interfaces.PumpDescription;
import info.nightscout.androidaps.interfaces.PumpInterface;
import info.nightscout.androidaps.plugins.ConfigBuilder.ConfigBuilderPlugin;
2017-06-02 10:25:49 +02:00
import info.nightscout.androidaps.plugins.Overview.events.EventDismissNotification;
import info.nightscout.androidaps.plugins.Overview.events.EventNewNotification;
2018-01-21 13:37:38 +01:00
import info.nightscout.androidaps.plugins.Overview.notifications.Notification;
2017-12-30 19:56:37 +01:00
import info.nightscout.utils.DateUtil;
2017-06-02 10:25:49 +02:00
import info.nightscout.utils.DecimalFormatter;
import info.nightscout.utils.ToastUtils;
public class Profile {
private static Logger log = LoggerFactory.getLogger(Profile.class);
private JSONObject json;
private String units = null;
2017-11-15 11:23:20 +01:00
private double dia = Constants.defaultDIA;
private TimeZone timeZone = TimeZone.getDefault();
private JSONArray isf;
2017-06-13 15:06:41 +02:00
private LongSparseArray<Double> isf_v = null; // oldest at index 0
2017-11-15 11:23:20 +01:00
private JSONArray ic;
2017-06-13 15:06:41 +02:00
private LongSparseArray<Double> ic_v = null; // oldest at index 0
2017-11-15 11:23:20 +01:00
private JSONArray basal;
2017-06-13 15:06:41 +02:00
private LongSparseArray<Double> basal_v = null; // oldest at index 0
2017-11-15 11:23:20 +01:00
private JSONArray targetLow;
2017-10-05 00:54:35 +02:00
private LongSparseArray<Double> targetLow_v = null; // oldest at index 0
2017-11-15 11:23:20 +01:00
private JSONArray targetHigh;
2017-10-05 00:54:35 +02:00
private LongSparseArray<Double> targetHigh_v = null; // oldest at index 0
2017-06-02 10:25:49 +02:00
2017-11-15 11:23:20 +01:00
private int percentage = 100;
private int timeshift = 0;
2017-09-23 23:12:08 +02:00
2018-01-21 13:37:38 +01:00
private boolean isValid = true;
private boolean isValidated = false;
2017-06-02 10:25:49 +02:00
public Profile(JSONObject json, String units) {
2017-10-05 00:54:35 +02:00
this(json, 100, 0);
2017-06-02 10:25:49 +02:00
if (this.units == null) {
if (units != null)
this.units = units;
else {
Crashlytics.log("Profile failover failed too");
this.units = Constants.MGDL;
}
}
}
2017-09-23 23:12:08 +02:00
public Profile(JSONObject json, int percentage, int timeshift) {
this.percentage = percentage;
this.timeshift = timeshift;
2017-06-02 10:25:49 +02:00
this.json = json;
try {
if (json.has("units"))
units = json.getString("units").toLowerCase();
if (json.has("dia"))
dia = json.getDouble("dia");
if (json.has("dia"))
dia = json.getDouble("dia");
if (json.has("timezone"))
timeZone = TimeZone.getTimeZone(json.getString("timezone"));
isf = json.getJSONArray("sens");
if (getIsf(0) == null) {
int defaultISF = units.equals(Constants.MGDL) ? 400 : 20;
isf = new JSONArray("[{\"time\":\"00:00\",\"value\":\"" + defaultISF + "\",\"timeAsSeconds\":\"0\"}]");
Notification noisf = new Notification(Notification.ISF_MISSING, MainApp.sResources.getString(R.string.isfmissing), Notification.URGENT);
MainApp.bus().post(new EventNewNotification(noisf));
} else {
MainApp.bus().post(new EventDismissNotification(Notification.ISF_MISSING));
}
ic = json.getJSONArray("carbratio");
if (getIc(0) == null) {
int defaultIC = 25;
ic = new JSONArray("[{\"time\":\"00:00\",\"value\":\"" + defaultIC + "\",\"timeAsSeconds\":\"0\"}]");
2017-06-02 10:25:49 +02:00
Notification noic = new Notification(Notification.IC_MISSING, MainApp.sResources.getString(R.string.icmissing), Notification.URGENT);
MainApp.bus().post(new EventNewNotification(noic));
} else {
MainApp.bus().post(new EventDismissNotification(Notification.IC_MISSING));
}
basal = json.getJSONArray("basal");
if (getBasal(0) == null) {
double defaultBasal = 0.1d;
basal = new JSONArray("[{\"time\":\"00:00\",\"value\":\"" + defaultBasal + "\",\"timeAsSeconds\":\"0\"}]");
2017-06-02 10:25:49 +02:00
Notification nobasal = new Notification(Notification.BASAL_MISSING, MainApp.sResources.getString(R.string.basalmissing), Notification.URGENT);
MainApp.bus().post(new EventNewNotification(nobasal));
} else {
MainApp.bus().post(new EventDismissNotification(Notification.BASAL_MISSING));
}
targetLow = json.getJSONArray("target_low");
if (getTargetLow(0) == null) {
double defaultLow = units.equals(Constants.MGDL) ? 120 : 6;
targetLow = new JSONArray("[{\"time\":\"00:00\",\"value\":\"" + defaultLow + "\",\"timeAsSeconds\":\"0\"}]");
2017-06-02 10:25:49 +02:00
Notification notarget = new Notification(Notification.TARGET_MISSING, MainApp.sResources.getString(R.string.targetmissing), Notification.URGENT);
MainApp.bus().post(new EventNewNotification(notarget));
} else {
MainApp.bus().post(new EventDismissNotification(Notification.TARGET_MISSING));
}
targetHigh = json.getJSONArray("target_high");
if (getTargetHigh(0) == null) {
double defaultHigh = units.equals(Constants.MGDL) ? 160 : 8;
targetHigh = new JSONArray("[{\"time\":\"00:00\",\"value\":\"" + defaultHigh + "\",\"timeAsSeconds\":\"0\"}]");
2017-06-02 10:25:49 +02:00
Notification notarget = new Notification(Notification.TARGET_MISSING, MainApp.sResources.getString(R.string.targetmissing), Notification.URGENT);
MainApp.bus().post(new EventNewNotification(notarget));
} else {
MainApp.bus().post(new EventDismissNotification(Notification.TARGET_MISSING));
}
} catch (JSONException e) {
log.error("Unhandled exception", e);
2018-01-14 21:42:36 +01:00
ToastUtils.showToastInUiThread(MainApp.instance().getApplicationContext(), MainApp.gs(R.string.invalidprofile));
2018-01-21 13:37:38 +01:00
isValid = false;
isValidated = true;
2017-06-02 10:25:49 +02:00
}
}
public String log() {
String ret = "\n";
for (Integer hour = 0; hour < 24; hour++) {
2017-06-11 14:29:02 +02:00
double value = getBasal((Integer) (hour * 60 * 60));
2017-06-02 10:25:49 +02:00
ret += "NS basal value for " + hour + ":00 is " + value + "\n";
}
ret += "NS units: " + getUnits();
return ret;
}
public JSONObject getData() {
2017-07-03 11:14:30 +02:00
if (!json.has("units"))
try {
json.put("units", units);
} catch (JSONException e) {
log.error("Unhandled exception", e);
2017-07-03 11:14:30 +02:00
}
2017-06-02 10:25:49 +02:00
return json;
}
2017-06-16 09:14:57 +02:00
public double getDia() {
2017-06-02 10:25:49 +02:00
return dia;
}
// mmol or mg/dl
public String getUnits() {
return units;
}
public TimeZone getTimeZone() {
return timeZone;
}
2017-06-13 15:06:41 +02:00
private LongSparseArray<Double> convertToSparseArray(JSONArray array) {
2017-10-05 00:54:35 +02:00
double multiplier = getMultiplier(array);
2017-06-13 15:06:41 +02:00
LongSparseArray<Double> sparse = new LongSparseArray<>();
for (Integer index = 0; index < array.length(); index++) {
try {
2017-12-14 21:49:11 +01:00
final JSONObject o = array.getJSONObject(index);
2017-12-30 19:56:37 +01:00
long tas = 0;
try {
tas = getShitfTimeSecs((int) o.getLong("timeAsSeconds"));
} catch (JSONException e) {
String time = o.getString("time");
tas = getShitfTimeSecs(DateUtil.toSeconds(time));
//log.debug(">>>>>>>>>>>> Used recalculated timeAsSecons: " + time + " " + tas);
}
2018-01-04 11:06:57 +01:00
double value = o.getDouble("value") * multiplier;
2017-06-13 15:06:41 +02:00
sparse.put(tas, value);
} catch (JSONException e) {
log.error("Unhandled exception", e);
2017-12-30 19:56:37 +01:00
log.error(json.toString());
2017-06-13 15:06:41 +02:00
}
}
2017-10-05 00:54:35 +02:00
// check if start is at 0 (midnight)
// and add last value before midnight if not
if (sparse.keyAt(0) != 0) {
sparse.put(0, sparse.valueAt(sparse.size() - 1));
}
2017-06-13 15:06:41 +02:00
return sparse;
}
2018-01-21 15:56:07 +01:00
public synchronized boolean isValid(String from) {
2018-01-21 13:37:38 +01:00
if (!isValid)
return false;
if (!isValidated) {
if (basal_v == null)
basal_v = convertToSparseArray(basal);
validate(basal_v);
if (isf_v == null)
isf_v = convertToSparseArray(isf);
validate(isf_v);
if (ic_v == null)
ic_v = convertToSparseArray(ic);
validate(ic_v);
if (targetLow_v == null)
targetLow_v = convertToSparseArray(targetLow);
validate(targetLow_v);
if (targetHigh_v == null)
targetHigh_v = convertToSparseArray(targetHigh);
validate(targetHigh_v);
isValidated = true;
}
if (isValid) {
// Check for hours alignment
for (int index = 0; index < basal_v.size(); index++) {
long secondsFromMidnight = basal_v.keyAt(index);
if (secondsFromMidnight % 3600 != 0) {
Notification notification = new Notification(Notification.BASAL_PROFILE_NOT_ALIGNED_TO_HOURS, String.format(MainApp.gs(R.string.basalprofilenotaligned), from), Notification.NORMAL);
MainApp.bus().post(new EventNewNotification(notification));
}
}
// Check for minimal basal value
PumpInterface pump = ConfigBuilderPlugin.getActivePump();
if (pump != null) {
PumpDescription description = pump.getPumpDescription();
for (int i = 0; i < basal_v.size(); i++) {
if (basal_v.valueAt(i) < description.basalMinimumRate) {
basal_v.setValueAt(i, description.basalMinimumRate);
MainApp.bus().post(new EventNewNotification(new Notification(Notification.MINIMAL_BASAL_VALUE_REPLACED, String.format(MainApp.gs(R.string.minimalbasalvaluereplaced), from), Notification.NORMAL)));
}
}
} else {
// if pump not available (at start)
// do not store converted array
basal_v = null;
}
}
return isValid;
}
private void validate(LongSparseArray array) {
if (array.size() == 0) {
isValid = false;
return;
}
for (int index = 0; index < array.size(); index++) {
if (array.valueAt(index).equals(0d)) {
isValid = false;
return;
}
}
}
2017-06-02 10:25:49 +02:00
private Double getValueToTime(JSONArray array, Integer timeAsSeconds) {
Double lastValue = null;
for (Integer index = 0; index < array.length(); index++) {
try {
JSONObject o = array.getJSONObject(index);
Integer tas = o.getInt("timeAsSeconds");
Double value = o.getDouble("value");
if (lastValue == null) lastValue = value;
if (timeAsSeconds < tas) {
break;
}
lastValue = value;
} catch (JSONException e) {
log.error("Unhandled exception", e);
2017-06-02 10:25:49 +02:00
}
}
return lastValue;
}
2017-09-23 23:12:08 +02:00
Integer getShitfTimeSecs(Integer originalTime) {
2017-10-05 00:54:35 +02:00
Integer shiftedTime = originalTime + timeshift * 60 * 60;
2017-10-07 20:42:56 +02:00
shiftedTime = (shiftedTime + 24 * 60 * 60) % (24 * 60 * 60);
2017-10-05 00:54:35 +02:00
if (timeshift != 0)
log.debug("(Sec) Original time: " + originalTime + " ShiftedTime: " + shiftedTime);
2017-09-23 23:12:08 +02:00
return shiftedTime;
}
2017-11-15 11:23:20 +01:00
private double getMultiplier(LongSparseArray<Double> array) {
2017-09-23 23:12:08 +02:00
double multiplier = 1d;
if (array == isf_v)
multiplier = 100d / percentage;
else if (array == ic_v)
multiplier = 100d / percentage;
else if (array == basal_v)
multiplier = percentage / 100d;
else
log.error("Unknown array type");
return multiplier;
}
2017-11-15 11:23:20 +01:00
private double getMultiplier(JSONArray array) {
2017-09-23 23:12:08 +02:00
double multiplier = 1d;
if (array == isf)
multiplier = 100d / percentage;
else if (array == ic)
multiplier = 100d / percentage;
else if (array == basal)
multiplier = percentage / 100d;
else if (array == targetLow)
multiplier = 1d;
else if (array == targetHigh)
multiplier = 1d;
else
log.error("Unknown array type");
return multiplier;
}
private Double getValueToTime(LongSparseArray<Double> array, Integer timeAsSeconds) {
2017-06-13 15:06:41 +02:00
Double lastValue = null;
for (Integer index = 0; index < array.size(); index++) {
long tas = array.keyAt(index);
double value = array.valueAt(index);
if (lastValue == null) lastValue = value;
2017-10-05 00:54:35 +02:00
if (timeAsSeconds < tas) {
2017-06-13 15:06:41 +02:00
break;
}
lastValue = value;
}
2017-10-05 00:54:35 +02:00
return lastValue;
2017-09-23 23:12:08 +02:00
}
private String format_HH_MM(Integer timeAsSeconds) {
String time;
int hour = timeAsSeconds / 60 / 60;
2017-10-20 11:34:33 +02:00
int minutes = (timeAsSeconds - hour * 60 * 60) / 60;
2017-09-23 23:12:08 +02:00
DecimalFormat df = new DecimalFormat("00");
time = df.format(hour) + ":" + df.format(minutes);
return time;
2017-06-13 15:06:41 +02:00
}
2017-10-05 00:54:35 +02:00
private String getValuesList(LongSparseArray<Double> array, LongSparseArray<Double> array2, DecimalFormat format, String units) {
2017-06-02 10:25:49 +02:00
String retValue = "";
2017-10-05 00:54:35 +02:00
for (Integer index = 0; index < array.size(); index++) {
retValue += format_HH_MM((int) array.keyAt(index));
retValue += " ";
retValue += format.format(array.valueAt(index));
if (array2 != null) {
retValue += " - ";
retValue += format.format(array2.valueAt(index));
2017-06-02 10:25:49 +02:00
}
2017-10-05 00:54:35 +02:00
retValue += " " + units;
if (index + 1 < array.size())
retValue += "\n";
2017-06-02 10:25:49 +02:00
}
return retValue;
}
public Double getIsf() {
2017-06-15 23:12:12 +02:00
return getIsf(secondsFromMidnight(System.currentTimeMillis()));
2017-06-02 10:25:49 +02:00
}
public Double getIsf(long time) {
return getIsf(secondsFromMidnight(time));
}
public Double getIsf(Integer timeAsSeconds) {
2017-06-13 15:06:41 +02:00
if (isf_v == null)
isf_v = convertToSparseArray(isf);
return getValueToTime(isf_v, timeAsSeconds);
2017-06-02 10:25:49 +02:00
}
public String getIsfList() {
2017-10-05 00:54:35 +02:00
return getValuesList(isf_v, null, new DecimalFormat("0.0"), getUnits() + "/U");
2017-06-02 10:25:49 +02:00
}
public Double getIc() {
2017-06-15 23:12:12 +02:00
return getIc(secondsFromMidnight(System.currentTimeMillis()));
2017-06-02 10:25:49 +02:00
}
public Double getIc(long time) {
return getIc(secondsFromMidnight(time));
}
public Double getIc(Integer timeAsSeconds) {
2017-06-13 15:06:41 +02:00
if (ic_v == null)
ic_v = convertToSparseArray(ic);
return getValueToTime(ic_v, timeAsSeconds);
2017-06-02 10:25:49 +02:00
}
public String getIcList() {
2017-10-05 00:54:35 +02:00
return getValuesList(ic_v, null, new DecimalFormat("0.0"), " g/U");
2017-06-02 10:25:49 +02:00
}
public Double getBasal() {
2017-06-15 23:12:12 +02:00
return getBasal(secondsFromMidnight(System.currentTimeMillis()));
2017-06-02 10:25:49 +02:00
}
public Double getBasal(long time) {
return getBasal(secondsFromMidnight(time));
}
2018-01-21 15:56:07 +01:00
public synchronized Double getBasal(Integer timeAsSeconds) {
if (basal_v == null) {
2017-06-13 15:06:41 +02:00
basal_v = convertToSparseArray(basal);
}
2017-06-13 15:06:41 +02:00
return getValueToTime(basal_v, timeAsSeconds);
2017-06-02 10:25:49 +02:00
}
public String getBasalList() {
2017-10-05 00:54:35 +02:00
return getValuesList(basal_v, null, new DecimalFormat("0.00"), "U");
2017-06-02 10:25:49 +02:00
}
public class BasalValue {
public BasalValue(Integer timeAsSeconds, Double value) {
this.timeAsSeconds = timeAsSeconds;
this.value = value;
}
public Integer timeAsSeconds;
public Double value;
}
2018-01-21 15:56:07 +01:00
public synchronized BasalValue[] getBasalValues() {
2018-01-14 21:42:36 +01:00
if (basal_v == null)
basal_v = convertToSparseArray(basal);
2017-10-05 00:54:35 +02:00
BasalValue[] ret = new BasalValue[basal_v.size()];
2017-06-02 10:25:49 +02:00
2017-10-05 00:54:35 +02:00
for (Integer index = 0; index < basal_v.size(); index++) {
Integer tas = (int) basal_v.keyAt(index);
Double value = basal_v.valueAt(index);
ret[index] = new BasalValue(tas, value);
2017-06-02 10:25:49 +02:00
}
2017-10-05 00:54:35 +02:00
return ret;
2017-06-02 10:25:49 +02:00
}
public Double getTargetLow() {
2017-06-15 23:12:12 +02:00
return getTargetLow(secondsFromMidnight(System.currentTimeMillis()));
2017-06-02 10:25:49 +02:00
}
public Double getTargetLow(long time) {
return getTargetLow(secondsFromMidnight(time));
}
public Double getTargetLow(Integer timeAsSeconds) {
2017-10-05 00:54:35 +02:00
if (targetLow_v == null)
targetLow_v = convertToSparseArray(targetLow);
return getValueToTime(targetLow_v, timeAsSeconds);
2017-06-02 10:25:49 +02:00
}
public Double getTargetHigh() {
2017-06-15 23:12:12 +02:00
return getTargetHigh(secondsFromMidnight(System.currentTimeMillis()));
2017-06-02 10:25:49 +02:00
}
public Double getTargetHigh(long time) {
return getTargetHigh(secondsFromMidnight(time));
}
public Double getTargetHigh(Integer timeAsSeconds) {
2017-10-05 00:54:35 +02:00
if (targetHigh_v == null)
targetHigh_v = convertToSparseArray(targetHigh);
return getValueToTime(targetHigh_v, timeAsSeconds);
2017-06-02 10:25:49 +02:00
}
public String getTargetList() {
2017-10-05 00:54:35 +02:00
return getValuesList(targetLow_v, targetHigh_v, new DecimalFormat("0.0"), getUnits());
2017-06-02 10:25:49 +02:00
}
public double getMaxDailyBasal() {
Double max = 0d;
for (Integer hour = 0; hour < 24; hour++) {
2017-08-20 19:42:41 +02:00
double value = getBasal((Integer) (hour * 60 * 60));
2017-06-02 10:25:49 +02:00
if (value > max) max = value;
}
return max;
}
2017-06-02 12:27:21 +02:00
public static Integer secondsFromMidnight() {
2017-06-02 10:25:49 +02:00
Calendar c = Calendar.getInstance();
long now = c.getTimeInMillis();
c.set(Calendar.HOUR_OF_DAY, 0);
c.set(Calendar.MINUTE, 0);
c.set(Calendar.SECOND, 0);
c.set(Calendar.MILLISECOND, 0);
long passed = now - c.getTimeInMillis();
return (int) (passed / 1000);
}
2017-06-02 12:27:21 +02:00
public static Integer secondsFromMidnight(long date) {
2017-06-02 10:25:49 +02:00
Calendar c = Calendar.getInstance();
c.setTimeInMillis(date);
c.set(Calendar.HOUR_OF_DAY, 0);
c.set(Calendar.MINUTE, 0);
c.set(Calendar.SECOND, 0);
c.set(Calendar.MILLISECOND, 0);
long passed = date - c.getTimeInMillis();
return (int) (passed / 1000);
}
public static Double toMgdl(Double value, String units) {
if (units.equals(Constants.MGDL)) return value;
else return value * Constants.MMOLL_TO_MGDL;
}
2017-08-20 19:42:41 +02:00
public static Double toMmol(Double value, String units) {
if (units.equals(Constants.MGDL)) return value * Constants.MGDL_TO_MMOLL;
else return value;
}
2017-06-02 10:25:49 +02:00
public static Double fromMgdlToUnits(Double value, String units) {
if (units.equals(Constants.MGDL)) return value;
else return value * Constants.MGDL_TO_MMOLL;
}
public static Double toUnits(Double valueInMgdl, Double valueInMmol, String units) {
if (units.equals(Constants.MGDL)) return valueInMgdl;
else return valueInMmol;
}
public static String toUnitsString(Double valueInMgdl, Double valueInMmol, String units) {
if (units.equals(Constants.MGDL)) return DecimalFormatter.to0Decimal(valueInMgdl);
else return DecimalFormatter.to1Decimal(valueInMmol);
}
2017-08-20 19:42:41 +02:00
// targets are stored in mg/dl but profile vary
public static String toTargetRangeString(double low, double high, String sourceUnits, String units) {
double lowMgdl = toMgdl(low, sourceUnits);
double highMgdl = toMgdl(high, sourceUnits);
double lowMmol = toMmol(low, sourceUnits);
double highMmol = toMmol(high, sourceUnits);
if (low == high)
return toUnitsString(lowMgdl, lowMmol, units);
else
return toUnitsString(lowMgdl, lowMmol, units) + " - " + toUnitsString(highMgdl, highMmol, units);
}
2017-10-08 14:41:51 +02:00
public double percentageBasalSum() {
double result = 0d;
for (int i = 0; i < 24; i++) {
result += getBasal((Integer) (i * 60 * 60));
}
return result;
}
public double baseBasalSum() {
double result = 0d;
for (int i = 0; i < 24; i++) {
result += getBasal((Integer) (i * 60 * 60)) / getMultiplier(basal_v);
}
return result;
}
public int getPercentage() {
return percentage;
}
public int getTimeshift() {
return timeshift;
}
2017-06-02 10:25:49 +02:00
}